package main import ( "bytes" "encoding/json" "io" "strconv" "strings" "time" log "github.com/rs/zerolog/log" ) var ( sessionID string ) func init() { sessionID = RandomString(5) + Nonce() log.Debug().Str("sessionId", sessionID).Msg("Session ID Generated") } type Term struct { } // GET /classSearch/getTerms?searchTerm=&offset=1&max=10&_=1702069154094 func GetTerms(search string, offset int, max int) ([]Term, error) { req := BuildRequest("GET", "/classSearch/getTerms", map[string]string{ "searchTerm": search, "offset": strconv.Itoa(offset), "max": strconv.Itoa(max), "_": Nonce(), }) res, err := doRequest(req) if err != nil { return nil, err } // print the response body body, _ := io.ReadAll(res.Body) log.Printf("Response Body: %s", body) // Assert that the response is JSON if !ContainsContentType(res.Header.Get("Content-Type"), "application/json") { log.Printf("ERR Response was not JSON: %s", res.Header.Get("Content-Type")) } return make([]Term, 0, 0), nil } type TermParts struct { } // GET /classSearch/get_partOfTerm?searchTerm=&term=202420&offset=1&max=10&uniqueSessionId=4bzai1701944879219&_=1702070282288 func GetPartOfTerms(search string, term int, offset int, max int) ([]TermParts, error) { req := BuildRequest("GET", "/classSearch/get_partOfTerm", map[string]string{ "searchTerm": search, "term": strconv.Itoa(term), "offset": strconv.Itoa(offset), "max": strconv.Itoa(max), "uniqueSessionId": sessionID, "_": Nonce(), }) res, err := client.Do(req) if err != nil { return nil, err } // Assert that the response is JSON if !ContainsContentType(res.Header.Get("Content-Type"), "application/json") { log.Printf("ERR Response was not JSON: %s", res.Header.Get("Content-Type")) } return make([]TermParts, 0), nil } type Instructor struct { } // GET /classSearch/get_instructor?searchTerm=&term=202420&offset=1&max=10&uniqueSessionId=4bzai1701944879219&_=1701951338584 func GetInstructor(search string, term int, offset int, max int) []Instructor { req := BuildRequest("GET", "/classSearch/get_instructor", map[string]string{ "searchTerm": search, "term": strconv.Itoa(term), "offset": strconv.Itoa(offset), "max": strconv.Itoa(max), "uniqueSessionId": sessionID, "_": Nonce(), }) res, err := client.Do(req) if err != nil { log.Printf("ERR %s", err) return nil } // Assert that the response is JSON if !ContainsContentType(res.Header.Get("Content-Type"), "application/json") { log.Printf("ERR Response was not JSON: %s", res.Header.Get("Content-Type")) } return make([]Instructor, 0) } type ClassDetails struct { } func GetClassDetails(term int, crn int) *ClassDetails { body, _ := json.Marshal(map[string]string{ "term": strconv.Itoa(term), "courseReferenceNumber": strconv.Itoa(crn), "first": "first", // TODO: What is this? }) req := BuildRequestWithBody("GET", "/searchResults/getClassDetails", nil, bytes.NewBuffer(body)) res, err := client.Do(req) if err != nil { log.Printf("ERR %s", err) return nil } // Assert that the response is JSON if !ContainsContentType(res.Header.Get("Content-Type"), "application/json") { log.Printf("ERR Response was not JSON: %s", res.Header.Get("Content-Type")) } return &ClassDetails{} } // GET /searchResults/searchResults?txt_instructor=77521&txt_term=202420&startDatepicker=&endDatepicker=&uniqueSessionId=4bzai1701944879219&pageOffset=0&pageMaxSize=10&sortColumn=subjectDescription&sortDirection=asc // GET /searchResults/searchResults?txt_subject=CS&txt_keywordlike=Application&txt_term=202420&startDatepicker=&endDatepicker=&uniqueSessionId=4bzai1701944879219&pageOffset=0&pageMaxSize=10&sortColumn=subjectDescription&sortDirection=asc func Search(subject string, keyword string, term string, startDate time.Time, endDate time.Time, offset int, max int, sort string, sortDescending bool) []string { req := BuildRequest("GET", "/classSearch/get_subject", map[string]string{ "txt_subject": subject, "txt_keywordlike": keyword, "txt_term": term, "startDatepicker": "", "endDatepicker": "", "uniqueSessionId": sessionID, "pageOffset": strconv.Itoa(offset), "pageMaxSize": strconv.Itoa(max), "sortColumn": "subjectDescription", "sortDirection": "asc", }) res, err := client.Do(req) if err != nil { log.Printf("ERR %s", err) return nil } // Assert that the response is JSON if !ContainsContentType(res.Header.Get("Content-Type"), "application/json") { log.Printf("ERR Response was not JSON: %s", res.Header.Get("Content-Type")) } return make([]string, 0) } type Subject struct { } // GET /classSearch/get_subject?searchTerm=&term=202420&offset=1&max=10&uniqueSessionId=4bzai1701944879219&_=1702069787420 func GetSubjects(search string, term int, offset int, max int) []Subject { req := BuildRequest("GET", "/classSearch/get_subject", map[string]string{ "searchTerm": search, "term": strconv.Itoa(term), "offset": strconv.Itoa(offset), "max": strconv.Itoa(max), "uniqueSessionId": sessionID, "_": Nonce(), }) res, err := client.Do(req) if err != nil { log.Printf("ERR %s", err) return nil } // Assert that the response is JSON if !ContainsContentType(res.Header.Get("Content-Type"), "application/json") { log.Printf("ERR Response was not JSON: %s", res.Header.Get("Content-Type")) } return make([]Subject, 0) } type Campus struct { } // /classSearch/get_campus?searchTerm=&term=202420&offset=1&max=10&uniqueSessionId=4bzai1701944879219&_=1702070341071 func GetCampuses(search string, term int, offset int, max int) []Campus { req := BuildRequest("GET", "/classSearch/get_campus", map[string]string{ "searchTerm": search, "term": strconv.Itoa(term), "offset": strconv.Itoa(offset), "max": strconv.Itoa(max), "uniqueSessionId": sessionID, "_": Nonce(), }) res, err := client.Do(req) if err != nil { log.Printf("ERR %s", err) return nil } // Assert that the response is JSON if !ContainsContentType(res.Header.Get("Content-Type"), "application/json") { log.Printf("ERR Response was not JSON: %s", res.Header.Get("Content-Type")) } return make([]Campus, 0) } type InstructionalMethod struct { } // / classSearch/get_instructionalMethod?searchTerm=&term=202420&offset=1&max=10&uniqueSessionId=4bzai1701944879219&_=1702070364082 func GetInstructionalMethods(search string, term int, offset int, max int) ([]InstructionalMethod, error) { req := BuildRequest("GET", "/classSearch/get_instructionalMethod", map[string]string{ "searchTerm": search, "term": strconv.Itoa(term), "offset": strconv.Itoa(offset), "max": strconv.Itoa(max), "uniqueSessionId": sessionID, "_": Nonce(), }) res, err := client.Do(req) if err != nil { log.Printf("ERR %s", err) return nil, err } // Assert that the response is JSON if !ContainsContentType(res.Header.Get("Content-Type"), "application/json") { log.Printf("ERR Response was not JSON: %s", res.Header.Get("Content-Type")) } return make([]InstructionalMethod, 0), nil } // GetCourseMeetingTime retrieves the meeting time information for a course based on the given term and course reference number (CRN). // It makes an HTTP GET request to the appropriate API endpoint and parses the response to extract the meeting time data. // The function returns a MeetingTimeResponse struct containing the extracted information. func GetCourseMeetingTime(term int, crn int) (*MeetingTimeResponse, error) { req := BuildRequest("GET", "/searchResults/getFacultyMeetingTimes", map[string]string{ "term": strconv.Itoa(term), "courseReferenceNumber": strconv.Itoa(crn), }) log.Printf("GET %s", req.URL.String()) res, err := client.Do(req) if err != nil { return nil, err } // Assert that the response is JSON if !ContainsContentType(res.Header.Get("Content-Type"), "application/json") { log.Fatal().Msgf("Response was not JSON: %s", res.Header.Get("Content-Type")) } // Read the response body into JSON defer res.Body.Close() var body map[string][]map[string]interface{} err = json.NewDecoder(res.Body).Decode(&body) if err != nil { return nil, err } if len(body["fmt"]) > 1 { log.Printf("Expected only one fmt child, found %d", len(body["fmt"])) } // Acquire & cast the meeting time data meetingTimeMap := body["fmt"][0]["meetingTime"].(map[string]interface{}) // Extract weekdays weekdays := make(map[time.Weekday]bool) for i := int(time.Sunday); i <= int(time.Saturday); i++ { day := time.Weekday(i) dayActive := meetingTimeMap[strings.ToLower(day.String())].(bool) weekdays[day] = dayActive } meetingScheduleType := meetingTimeMap["meetingScheduleType"].(string) meetingType := meetingTimeMap["meetingType"].(string) meetingTypeDescription := meetingTimeMap["meetingTypeDescription"].(string) // Extract other data campus := meetingTimeMap["campus"].(string) campusDescription := meetingTimeMap["campusDescription"].(string) building := meetingTimeMap["building"].(string) buildingDescription := meetingTimeMap["buildingDescription"].(string) room := "N/A" if meetingTimeMap["room"] != nil { room = meetingTimeMap["room"].(string) } creditHours := meetingTimeMap["creditHourSession"].(float64) hoursPerWeek := meetingTimeMap["hoursWeek"].(float64) // Parse dates & times const layout = "01/02/2006" dateStart, _ := time.Parse(layout, meetingTimeMap["startDate"].(string)) dateEnd, _ := time.Parse(layout, meetingTimeMap["endDate"].(string)) timeStartInt, _ := strconv.ParseUint(meetingTimeMap["beginTime"].(string), 10, 0) timeStart := ParseNaiveTime(timeStartInt) timeEndInt, _ := strconv.ParseUint(meetingTimeMap["endTime"].(string), 10, 0) timeEnd := ParseNaiveTime(timeEndInt) // Extract faculty data faculty := make([]MeetingTimeFaculty, 0) for _, facultyMap := range body["fmt"][0]["faculty"].([]interface{}) { facultyMap := facultyMap.(map[string]interface{}) bannerId, _ := strconv.Atoi(facultyMap["bannerId"].(string)) faculty = append(faculty, MeetingTimeFaculty{ bannerId: bannerId, category: facultyMap["category"].(string), displayName: facultyMap["displayName"].(string), email: facultyMap["emailAddress"].(string), primary: facultyMap["primaryIndicator"].(bool), }) } return &MeetingTimeResponse{ faculty: faculty, weekdays: weekdays, campus: campus, campusDescription: campusDescription, creditHours: int(creditHours), building: building, buildingDescription: buildingDescription, room: room, timeStart: timeStart, timeEnd: timeEnd, dateStart: dateStart, dateEnd: dateEnd, hoursPerWeek: float32(hoursPerWeek), meetingScheduleType: meetingScheduleType, meetingType: meetingType, meetingTypeDescription: meetingTypeDescription, }, nil }