aboutsummaryrefslogtreecommitdiff
path: root/course_types_groups.go
diff options
context:
space:
mode:
Diffstat (limited to 'course_types_groups.go')
-rw-r--r--course_types_groups.go182
1 files changed, 182 insertions, 0 deletions
diff --git a/course_types_groups.go b/course_types_groups.go
new file mode 100644
index 0000000..3c298ca
--- /dev/null
+++ b/course_types_groups.go
@@ -0,0 +1,182 @@
+/*
+ * Course types and groups
+ *
+ * Copyright (C) 2024 Runxi Yu <https://runxiyu.org>
+ * SPDX-License-Identifier: AGPL-3.0-or-later
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Affero General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU Affero General Public License for more details.
+ *
+ * You should have received a copy of the GNU Affero General Public License
+ * along with this program. If not, see <https://www.gnu.org/licenses/>.
+ */
+
+package main
+
+import (
+ "context"
+ "fmt"
+)
+
+/* Course types, e.g. Sport */
+
+const (
+ sport string = "Sport"
+ nonSport string = "Non-sport"
+)
+
+var courseTypes = map[string]struct{}{
+ sport: {},
+ nonSport: {},
+}
+
+func checkCourseType(ct string) bool {
+ _, ok := courseTypes[ct]
+ return ok
+}
+
+type userCourseTypesT map[string]int
+
+func getCourseTypeMinimumForYearGroup(yearGroup, courseType string) (int, error) {
+ switch yearGroup {
+ case "Y9":
+ switch courseType {
+ case sport:
+ return config.Req.Y9.Sport, nil
+ case nonSport:
+ return config.Req.Y9.NonSport, nil
+ default:
+ return 0, errInvalidCourseType
+ }
+ case "Y10":
+ switch courseType {
+ case sport:
+ return config.Req.Y10.Sport, nil
+ case nonSport:
+ return config.Req.Y10.NonSport, nil
+ default:
+ return 0, errInvalidCourseType
+ }
+ case "Y11":
+ switch courseType {
+ case sport:
+ return config.Req.Y11.Sport, nil
+ case nonSport:
+ return config.Req.Y11.NonSport, nil
+ default:
+ return 0, errInvalidCourseType
+ }
+ case "Y12":
+ switch courseType {
+ case sport:
+ return config.Req.Y12.Sport, nil
+ case nonSport:
+ return config.Req.Y12.NonSport, nil
+ default:
+ return 0, errInvalidCourseType
+ }
+ default:
+ return 0, errNoSuchYearGroup
+ }
+}
+
+/* Course groups, e.g. MW1 */
+
+type userCourseGroupsT map[string]struct{}
+
+func checkCourseGroup(cg string) bool {
+ _, ok := courseGroups[cg]
+ return ok
+}
+
+const (
+ mw1 string = "MW1"
+ mw2 string = "MW2"
+ mw3 string = "MW3"
+ tt1 string = "TT1"
+ tt2 string = "TT2"
+ tt3 string = "TT3"
+)
+
+var courseGroups = map[string]string{
+ mw1: "Monday/Wednesday CCA1",
+ mw2: "Monday/Wednesday CCA2",
+ mw3: "Monday/Wednesday CCA3",
+ tt1: "Tuesday/Thursday CCA1",
+ tt2: "Tuesday/Thursday CCA2",
+ tt3: "Tuesday/Thursday CCA3",
+}
+
+/* Populate both */
+
+func populateUserCourseTypesAndGroups(
+ ctx context.Context,
+ userCourseTypes *userCourseTypesT,
+ userCourseGroups *userCourseGroupsT,
+ userID string,
+) error {
+ rows, err := db.Query(
+ ctx,
+ "SELECT courseid FROM choices WHERE userid = $1",
+ userID,
+ )
+ if err != nil {
+ return wrapError(
+ errUnexpectedDBError,
+ err,
+ )
+ }
+ for {
+ if !rows.Next() {
+ err := rows.Err()
+ if err != nil {
+ return wrapError(
+ errUnexpectedDBError,
+ err,
+ )
+ }
+ break
+ }
+ var thisCourseID int
+ err := rows.Scan(&thisCourseID)
+ if err != nil {
+ return wrapError(
+ errUnexpectedDBError,
+ err,
+ )
+ }
+ var thisGroupName, thisTypeName string
+ _course, ok := courses.Load(thisCourseID)
+ if !ok {
+ return fmt.Errorf(
+ "%w: %d",
+ errNoSuchCourse,
+ thisCourseID,
+ )
+ }
+ course, ok := _course.(*courseT)
+ if !ok {
+ panic("courses map has non-\"*courseT\" items")
+ }
+ thisGroupName = course.Group
+ thisTypeName = course.Type
+ if _, ok := (*userCourseGroups)[thisGroupName]; ok {
+ return fmt.Errorf(
+ "%w: user %v, group %v",
+ errMultipleChoicesInOneGroup,
+ userID,
+ thisGroupName,
+ )
+ }
+ (*userCourseGroups)[thisGroupName] = struct{}{}
+ (*userCourseTypes)[thisTypeName]++
+ }
+ return nil
+}