Split project into separate files

This commit is contained in:
2023-12-07 20:52:52 -06:00
parent 587f528685
commit aac44e42d1
4 changed files with 204 additions and 185 deletions

112
api.go Normal file
View File

@@ -0,0 +1,112 @@
package banner
import (
"encoding/json"
"log"
"net/http"
"strconv"
"strings"
"time"
)
func getCourseMeetingTime(term int, crn int) MeetingTimeResponse {
url := buildURL("searchResults/getFacultyMeetingTimes", map[string]string{
"term": strconv.Itoa(term),
"courseReferenceNumber": strconv.Itoa(crn),
})
// Build the request
log.Printf("GET %s", url)
req, err := http.NewRequest("GET", url, nil)
if err != nil {
log.Fatal(err)
}
AddUserAgent(req)
resp, err := client.Do(req)
if err != nil {
log.Fatal(err)
}
if !ContainsContentType(resp.Header.Get("Content-Type"), "application/json") {
log.Fatalf("Response was not JSON: %s", resp.Header.Get("Content-Type"))
}
defer resp.Body.Close()
var body map[string][]map[string]interface{}
err = json.NewDecoder(resp.Body).Decode(&body)
if err != nil {
log.Fatal(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))
timeStart, _ := strconv.ParseUint(meetingTimeMap["beginTime"].(string), 10, 0)
timeEnd, _ := strconv.ParseUint(meetingTimeMap["endTime"].(string), 10, 0)
// 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,
}
}

40
helpers.go Normal file
View File

@@ -0,0 +1,40 @@
package banner
import (
"net/http"
"strings"
)
func buildURL(path string, params map[string]string) string {
// Builds a URL for the given path and parameters
url := baseURL + path
if params != nil {
takenFirst := false
for key, value := range params {
paramChar := "&"
if !takenFirst {
paramChar = "?"
takenFirst = true
}
url += paramChar + key + "=" + value
}
}
return url
}
func AddUserAgent(req *http.Request) {
req.Header.Add("User-Agent", "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/113.0.0.0 Safari/537.36")
}
func ContainsContentType(header string, search string) bool {
// Split on commas, check if any of the types match
for _, content_type := range strings.Split(header, ";") {
if content_type == search {
return true
}
}
return false
}

187
main.go
View File

@@ -1,14 +1,10 @@
package main package banner
import ( import (
"encoding/json"
"log" "log"
"net/http" "net/http"
"net/http/cookiejar" "net/http/cookiejar"
"net/url"
"os" "os"
"strconv"
"strings"
"time" "time"
"github.com/joho/godotenv" "github.com/joho/godotenv"
@@ -48,185 +44,6 @@ type MeetingTimeResponse struct {
meetingTypeDescription string meetingTypeDescription string
} }
func setupSession() {
// Makes the initial requests that sets up the session cookies for the rest of the application
log.Println("Setting up session...")
request_queue := []string{
"/registration/registration",
"/selfServiceMenu/data",
}
for _, path := range request_queue {
req, _ := http.NewRequest("GET", buildURL(path, nil), nil)
AddUserAgent(req)
res, _ := client.Do(req)
log.Println(res)
}
// Validate that cookies were set
baseURL_parsed, _ := url.Parse(baseURL)
current_cookies := cookies.Cookies(baseURL_parsed)
required_cookies := map[string]bool{
"JSESSIONID": false,
"SSB_COOKIE": false,
}
for _, cookie := range current_cookies {
_, present := required_cookies[cookie.Name]
// Check if this cookie is required
if present {
required_cookies[cookie.Name] = true
}
}
// Check if all required cookies were set
for cookie_name, cookie_set := range required_cookies {
if !cookie_set {
log.Fatalf("Required cookie %s was not set", cookie_name)
}
}
log.Println("All cookies acquired. Session setup complete.")
// Validate that the session allows access to termSelection
}
func buildURL(path string, params map[string]string) string {
// Builds a URL for the given path and parameters
url := baseURL + path
if params != nil {
takenFirst := false
for key, value := range params {
paramChar := "&"
if !takenFirst {
paramChar = "?"
takenFirst = true
}
url += paramChar + key + "=" + value
}
}
return url
}
func AddUserAgent(req *http.Request) {
req.Header.Add("User-Agent", "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/113.0.0.0 Safari/537.36")
}
func ContainsContentType(header string, search string) bool {
// Split on commas, check if any of the types match
for _, content_type := range strings.Split(header, ";") {
if content_type == search {
return true
}
}
return false
}
func getCourseMeetingTime(term int, crn int) MeetingTimeResponse {
url := buildURL("searchResults/getFacultyMeetingTimes", map[string]string{
"term": strconv.Itoa(term),
"courseReferenceNumber": strconv.Itoa(crn),
})
// Build the request
log.Printf("GET %s", url)
req, err := http.NewRequest("GET", url, nil)
if err != nil {
log.Fatal(err)
}
AddUserAgent(req)
resp, err := client.Do(req)
if err != nil {
log.Fatal(err)
}
if !ContainsContentType(resp.Header.Get("Content-Type"), "application/json") {
log.Fatalf("Response was not JSON: %s", resp.Header.Get("Content-Type"))
}
defer resp.Body.Close()
var body map[string][]map[string]interface{}
err = json.NewDecoder(resp.Body).Decode(&body)
if err != nil {
log.Fatal(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))
timeStart, _ := strconv.ParseUint(meetingTimeMap["beginTime"].(string), 10, 0)
timeEnd, _ := strconv.ParseUint(meetingTimeMap["endTime"].(string), 10, 0)
// 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,
}
}
func main() { func main() {
err := godotenv.Load() err := godotenv.Load()
if err != nil { if err != nil {
@@ -243,7 +60,7 @@ func main() {
Jar: cookies, Jar: cookies,
} }
setupSession() setup()
meetingTime := getCourseMeetingTime(202420, 44142) meetingTime := getCourseMeetingTime(202420, 44142)
log.Println(meetingTime) log.Println(meetingTime)

50
session.go Normal file
View File

@@ -0,0 +1,50 @@
package banner
import (
"log"
"net/http"
"net/url"
)
func setup() {
// Makes the initial requests that sets up the session cookies for the rest of the application
log.Println("Setting up session...")
request_queue := []string{
"/registration/registration",
"/selfServiceMenu/data",
}
for _, path := range request_queue {
req, _ := http.NewRequest("GET", buildURL(path, nil), nil)
AddUserAgent(req)
res, _ := client.Do(req)
log.Println(res)
}
// Validate that cookies were set
baseURL_parsed, _ := url.Parse(baseURL)
current_cookies := cookies.Cookies(baseURL_parsed)
required_cookies := map[string]bool{
"JSESSIONID": false,
"SSB_COOKIE": false,
}
for _, cookie := range current_cookies {
_, present := required_cookies[cookie.Name]
// Check if this cookie is required
if present {
required_cookies[cookie.Name] = true
}
}
// Check if all required cookies were set
for cookie_name, cookie_set := range required_cookies {
if !cookie_set {
log.Fatalf("Required cookie %s was not set", cookie_name)
}
}
log.Println("All cookies acquired. Session setup complete.")
// Validate that the session allows access to termSelection
}