package routes import ( "aunefyren/wrapperr/files" "aunefyren/wrapperr/models" "aunefyren/wrapperr/modules" "aunefyren/wrapperr/utilities" "encoding/json" "errors" "fmt" "io/ioutil" "log" "net/http" "strings" "time" ) // API route which retrieves the Wrapperr version and some minor details (application name, Plex-Auth...). func ApiGetWrapperrVersion(w http.ResponseWriter, r *http.Request) { configured_bool, err := files.GetConfigState() if err != nil { log.Println("Failed to retrieve configuration state. Error: ") log.Println(err) utilities.RespondDefaultError(w, r, errors.New("Failed to retrieve configuration state."), 500) return } config, err := files.GetConfig() if err != nil { log.Println("Failed to load configuration file. Error: ") log.Println(err) utilities.RespondDefaultError(w, r, errors.New("Failed to retrieve configuration state."), 500) return } version_reply := models.WrapperrVersion{ WrapperrVersion: config.WrapperrVersion, ApplicationName: config.ApplicationName, PlexAuth: config.PlexAuth, WrapperrFrontPageTitle: config.WrapperrCustomize.WrapperrFrontPageTitle, WrapperrFrontPageSubtitle: config.WrapperrCustomize.WrapperrFrontPageSubtitle, ClientKey: config.ClientKey, WrapperrConfigured: configured_bool, WinterTheme: config.WinterTheme, Message: "Retrieved Wrapperr version.", Error: false, WrapperrRoot: config.WrapperrRoot, } ip_string := utilities.GetOriginIPString(w, r) log.Println("Retrieved Wrapperr version." + ip_string) utilities.RespondWithJSON(w, http.StatusOK, version_reply) return } // API route which returns if whether or not a Wrapperr admin is configured. func ApiGetAdminState(w http.ResponseWriter, r *http.Request) { admin, err := files.GetAdminState() if err != nil { log.Println(err) log.Println("Failed to load admin state.") return } boolean_reply := models.BooleanReply{ Message: "Retrieved Wrapperr version.", Error: false, Data: admin, } ip_string := utilities.GetOriginIPString(w, r) log.Println("Retrieved Wrapperr admin state." + ip_string) utilities.RespondWithJSON(w, http.StatusOK, boolean_reply) return } // API route which retrieves the Wrapperr settings needed for the front-end. func ApiGetFunctions(w http.ResponseWriter, r *http.Request) { config, err := files.GetConfig() if err != nil { log.Println(err) log.Println("Failed to load configuration file.") fmt.Println("Failed to load configuration file.") return } function_reply := models.WrapperrFunctions{ WrapperrVersion: config.WrapperrVersion, PlexAuth: config.PlexAuth, WrapperrCustomize: config.WrapperrCustomize, CreateShareLinks: config.CreateShareLinks, } ip_string := utilities.GetOriginIPString(w, r) log.Println("Retrieved Wrapperr functions." + ip_string) utilities.RespondWithJSON(w, http.StatusOK, function_reply) return } // API route used to create the admin account and claim the Wrapperr server func ApiCreateAdmin(w http.ResponseWriter, r *http.Request) { admin, err := files.GetAdminState() if err != nil { log.Println(err) log.Println("Failed to load admin state.") fmt.Println("Failed to load admin state.") return } if admin { log.Println("Admin creation failed. Admin already configured.") utilities.RespondDefaultError(w, r, errors.New("Admin already configured."), 401) return } else { // Read payload from Post input reqBody, _ := ioutil.ReadAll(r.Body) var admin_payload models.AdminConfig json.Unmarshal(reqBody, &admin_payload) // Confirm username length if len(admin_payload.AdminUsername) < 4 { log.Println("Admin creation failed. Admin username requires four or more characters.") utilities.RespondDefaultError(w, r, errors.New("Admin username is too short. Four characters or more required."), 500) return } // Confirm password length if len(admin_payload.AdminPassword) < 8 { log.Println("Admin creation failed. Admin password requires eight or more characters.") utilities.RespondDefaultError(w, r, errors.New("Admin password is too short. Eight characters or more required."), 500) return } // Hash new password hash, err := utilities.HashAndSalt(admin_payload.AdminPassword) if err != nil { errors.New("Admin creation failed. Could not hash new password. Error: ") log.Println(err) utilities.RespondDefaultError(w, r, errors.New("Failed to hash your password."), 500) return } admin_payload.AdminPassword = hash // Save new admin config err = files.SaveAdminConfig(admin_payload) if err != nil { errors.New("Admin creation failed. Could not save configuration. Error: ") log.Println(err) utilities.RespondDefaultError(w, r, errors.New("Failed to save new admin."), 500) return } log.Println("New admin account created. Server is now claimed.") fmt.Println("New admin account created. Server is now claimed.") utilities.RespondDefaultOkay(w, r, "Admin created.") return } } // API route which returns if whether or not Wrapperr is configured. func ApiWrapperrConfigured(w http.ResponseWriter, r *http.Request) { bool, err := files.GetConfigState() if err != nil { log.Panicln(err) utilities.RespondDefaultError(w, r, errors.New("Failed to retrieve configuration state."), 500) return } else { boolean_reply := models.BooleanReply{ Message: "Retrieved Wrapperr configuration state.", Error: false, Data: bool, } ip_string := utilities.GetOriginIPString(w, r) log.Println("Retrieved Wrapperr configuration state." + ip_string) utilities.RespondWithJSON(w, http.StatusOK, boolean_reply) return } } // API route which trades admin login credentials for an admin JWT session token. Valid for three days. func ApiLogInAdmin(w http.ResponseWriter, r *http.Request) { admin, err := files.GetAdminState() if err != nil { log.Println(err) log.Println("Failed to load admin state.") fmt.Println("Failed to load admin state.") return } if !admin { log.Println("Admin login failed. Admin is not configured.") utilities.RespondDefaultError(w, r, errors.New("No admin configured."), 400) return } else { admin_config, err := files.GetAdminConfig() if err != nil { log.Println(err) log.Println("Failed to load admin config.") fmt.Println("Failed to load admin config.") return } // Read payload from Post input reqBody, _ := ioutil.ReadAll(r.Body) var admin_payload models.AdminConfig json.Unmarshal(reqBody, &admin_payload) // Confirm username length if len(admin_payload.AdminUsername) < 4 { log.Println("Admin creation failed. Admin username requires four or more characters.") utilities.RespondDefaultError(w, r, errors.New("Admin username is too short. Four characters or more required."), 500) return } // Confirm password length if len(admin_payload.AdminPassword) < 8 { log.Println("Admin creation failed. Admin password requires eight or more characters.") utilities.RespondDefaultError(w, r, errors.New("Admin password is too short. Eight characters or more required."), 500) return } // Hash new password password_validity := utilities.ComparePasswords(admin_config.AdminPassword, admin_payload.AdminPassword) // Validate admin username and password if !password_validity || admin_config.AdminUsername != admin_payload.AdminUsername { ip_string := utilities.GetOriginIPString(w, r) log.Println("Admin login failed. Incorrect admin username or password." + ip_string) fmt.Println("Admin login failed. Incorrect admin username or password." + ip_string) utilities.RespondDefaultError(w, r, errors.New("Login failed. Username or password is incorrect."), 401) return } token, err := modules.CreateToken(admin_config.AdminUsername, true, "") if err != nil { log.Println(err) utilities.RespondDefaultError(w, r, errors.New("Failed to create JWT token."), 500) return } string_reply := models.StringReply{ Message: "Login cookie created", Error: false, Data: token, } ip_string := utilities.GetOriginIPString(w, r) log.Println("Created and retrieved admin login JWT Token." + ip_string) fmt.Println("Created and retrieved admin login JWT Token." + ip_string) utilities.RespondWithJSON(w, http.StatusOK, string_reply) return } } // APi route which trades admin login credentials for an admin JWT session token. Valid for three days. func ApiGetTautulliConncection(w http.ResponseWriter, r *http.Request) { // Read payload from Post input reqBody, _ := ioutil.ReadAll(r.Body) var tautulli_connection models.TautulliConfig json.Unmarshal(reqBody, &tautulli_connection) if tautulli_connection.TautulliApiKey == "" || tautulli_connection.TautulliIP == "" || tautulli_connection.TautulliPort == 0 { log.Println("Cannot test Tautulli connection. Invalid Tautulli connection details received.") utilities.RespondDefaultError(w, r, errors.New("Tautulli connection details specified are invalid."), 400) return } tautulli_state, err := modules.TautulliTestConnection(tautulli_connection.TautulliPort, tautulli_connection.TautulliIP, tautulli_connection.TautulliHttps, tautulli_connection.TautulliRoot, tautulli_connection.TautulliApiKey) if err != nil { utilities.RespondDefaultError(w, r, err, 500) return } boolean_reply := models.BooleanReply{ Message: "Tested Tautulli connection.", Error: false, Data: tautulli_state, } ip_string := utilities.GetOriginIPString(w, r) log.Println("Tested Tautulli connection." + ip_string) utilities.RespondWithJSON(w, http.StatusOK, boolean_reply) return } // Get shareable link func ApiGetShareLink(w http.ResponseWriter, r *http.Request) { config_bool, err := files.GetConfigState() if err != nil { log.Println("Failed to retrieve configuration state. Error: ") log.Println(err) utilities.RespondDefaultError(w, r, errors.New("Failed to retrieve configuration state"), 500) return } else if !config_bool { log.Println("Wrapperr is not configured.") utilities.RespondDefaultError(w, r, errors.New("Wrapperr is not configured."), 400) return } config, err := files.GetConfig() if err != nil { log.Println(err) utilities.RespondDefaultError(w, r, errors.New("Failed to load Wrapperr configuration."), 500) return } if !config.CreateShareLinks { log.Println("Shareable links are not enabled in the Wrapperr configuration.") utilities.RespondDefaultError(w, r, errors.New("Shareable links are not enabled in the Wrapperr configuration."), 400) return } // Read payload from Post input reqBody, err := ioutil.ReadAll(r.Body) if err != nil { log.Println(err) utilities.RespondDefaultError(w, r, errors.New("Failed to parse payload for request."), 500) return } var link_payload models.WrapperrShareLinkGetRequest json.Unmarshal(reqBody, &link_payload) var fileName string var hash string hash_array := strings.Split(link_payload.Hash, "-") if len(hash_array) < 2 { log.Println("Failed to split hash string while looking for user ID.") utilities.RespondDefaultError(w, r, errors.New("Failed to parse payload hash for Wrapperr link."), 500) return } if config.PlexAuth { for j := 1; j < len(hash_array); j++ { if j != 1 { hash = hash + "-" } hash = hash + hash_array[j] } fileName = hash_array[0] } else { for j := 1; j < len(hash_array); j++ { if j != 1 { hash = hash + "-" } hash = hash + hash_array[j] } fileName = hash } share_link_object, err := files.GetLink(fileName) if err != nil { log.Println(err) utilities.RespondDefaultError(w, r, err, 500) return } currentTime := time.Now() linkTime, err := time.Parse("2006-01-02", share_link_object.Date) if err != nil { log.Println(err) utilities.RespondDefaultError(w, r, errors.New("Failed to retrieve saved Wrapperr link."), 500) return } linkTime = linkTime.Add(7 * 24 * time.Hour) if !linkTime.Before(currentTime) && share_link_object.Hash == hash && !share_link_object.Expired { share_link_object.Message = "Shared link retrieved." share_link_object.Error = false ip_string := utilities.GetOriginIPString(w, r) log.Println("Retrieved Wrapperr share link made by User ID/with hash: " + fileName + "." + ip_string) utilities.RespondWithJSON(w, http.StatusOK, share_link_object) return } else { return_error := errors.New("Invalid share link.") if linkTime.Before(currentTime) { share_link_object.Expired = true err = files.SaveLink(share_link_object, config.PlexAuth) if err != nil { log.Println(err) } return_error = errors.New("This Wrapped link has expired.") } log.Println("Failed to retrieve Wrapperr share link with hash: " + link_payload.Hash + ".") utilities.RespondDefaultError(w, r, return_error, 401) return } }