2023-10-25 13:14:02 +00:00
|
|
|
package routes
|
|
|
|
|
|
|
|
import (
|
|
|
|
"aunefyren/wrapperr/files"
|
|
|
|
"aunefyren/wrapperr/middlewares"
|
|
|
|
"aunefyren/wrapperr/models"
|
2023-10-29 17:03:55 +00:00
|
|
|
"aunefyren/wrapperr/modules"
|
2023-10-25 13:14:02 +00:00
|
|
|
"aunefyren/wrapperr/utilities"
|
|
|
|
"log"
|
|
|
|
"net/http"
|
2023-10-29 17:03:55 +00:00
|
|
|
"strconv"
|
2023-10-25 13:14:02 +00:00
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/gin-gonic/gin"
|
|
|
|
)
|
|
|
|
|
|
|
|
// API route used to retrieve the Wrapperr configuration file.
|
|
|
|
func ApiGetConfig(context *gin.Context) {
|
|
|
|
config, err := files.GetConfig()
|
|
|
|
if err != nil {
|
|
|
|
log.Println("Failed to load Wrapperr configuration. Error: " + err.Error())
|
|
|
|
context.JSON(http.StatusInternalServerError, gin.H{"error": "Failed to load Wrapperr configuration."})
|
|
|
|
context.Abort()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
adminConfig, err := files.GetAdminConfig()
|
|
|
|
if err != nil {
|
|
|
|
log.Println("Failed to load Wrapperr admin configuration. Error: " + err.Error())
|
|
|
|
context.JSON(http.StatusInternalServerError, gin.H{"error": "Failed to load Wrapperr admin configuration."})
|
|
|
|
context.Abort()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
authorizationHeader := context.GetHeader("Authorization")
|
|
|
|
payload, httpStatus, err := middlewares.AuthGetPayloadFromAuthorization(authorizationHeader, config, adminConfig)
|
|
|
|
if err != nil {
|
|
|
|
log.Println("Failed to get payload from Authorization token. Error: " + err.Error())
|
|
|
|
context.JSON(httpStatus, gin.H{"error": err.Error()})
|
|
|
|
context.Abort()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
configReply := models.ConfigReply{
|
|
|
|
Data: config,
|
|
|
|
Message: "Retrieved Wrapperr config.",
|
|
|
|
Error: false,
|
|
|
|
Username: payload.Username,
|
|
|
|
}
|
|
|
|
|
|
|
|
ipString := utilities.GetOriginIPString(context)
|
|
|
|
log.Println("Retrieved Wrapperr configuration." + ipString)
|
|
|
|
|
|
|
|
context.JSON(http.StatusOK, configReply)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// API route used to update the Wrapperr configuration file.
|
|
|
|
func ApiSetConfig(context *gin.Context) {
|
|
|
|
config, err := files.GetConfig()
|
|
|
|
if err != nil {
|
|
|
|
log.Println("Failed to load Wrapperr configuration. Error: " + err.Error())
|
|
|
|
context.JSON(http.StatusInternalServerError, gin.H{"error": "Failed to load Wrapperr configuration."})
|
|
|
|
context.Abort()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
originalRoot := config.WrapperrRoot
|
|
|
|
|
|
|
|
// Read payload from Post input
|
|
|
|
var config_payload models.SetWrapperrConfig
|
|
|
|
if err := context.ShouldBindJSON(&config_payload); err != nil {
|
|
|
|
log.Println("Failed to parse request. Error: " + err.Error())
|
|
|
|
context.JSON(http.StatusBadRequest, gin.H{"error": "Failed to parse request."})
|
|
|
|
context.Abort()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Confirm username length
|
|
|
|
if config_payload.DataType == "" {
|
|
|
|
log.Println("Cannot set new config. Invalid data type received.")
|
|
|
|
context.JSON(http.StatusBadRequest, gin.H{"error": "Data type specified is invalid."})
|
|
|
|
context.Abort()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if config_payload.DataType == "tautulli_config" {
|
|
|
|
config.TautulliConfig = config_payload.TautulliConfig
|
|
|
|
|
|
|
|
err = files.SaveConfig(config)
|
|
|
|
if err != nil {
|
|
|
|
log.Println("Failed to save new Wrapperr configuration. Error: " + err.Error())
|
|
|
|
context.JSON(http.StatusInternalServerError, gin.H{"error": "Failed to save new Wrapperr configuration."})
|
|
|
|
context.Abort()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
} else if config_payload.DataType == "wrapperr_customize" {
|
|
|
|
config.WrapperrCustomize = config_payload.WrapperrCustomize
|
|
|
|
|
|
|
|
err = files.SaveConfig(config)
|
|
|
|
if err != nil {
|
|
|
|
log.Println("Failed to save new Wrapperr configuration. Error: " + err.Error())
|
|
|
|
context.JSON(http.StatusInternalServerError, gin.H{"error": "Failed to save new Wrapperr configuration."})
|
|
|
|
context.Abort()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
} else if config_payload.DataType == "wrapperr_data" {
|
|
|
|
new_time, err := time.LoadLocation(config_payload.WrapperrData.Timezone)
|
|
|
|
if err != nil {
|
|
|
|
log.Println("Failed to set the new time zone. Error: " + err.Error())
|
|
|
|
context.JSON(http.StatusBadRequest, gin.H{"error": "Given time zone is invalid."})
|
|
|
|
context.Abort()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
config.UseCache = config_payload.WrapperrData.UseCache
|
|
|
|
config.UseLogs = config_payload.WrapperrData.UseLogs
|
|
|
|
config.PlexAuth = config_payload.WrapperrData.PlexAuth
|
|
|
|
config.BasicAuth = config_payload.WrapperrData.BasicAuth
|
|
|
|
config.WrapperrRoot = config_payload.WrapperrData.WrapperrRoot
|
|
|
|
config.CreateShareLinks = config_payload.WrapperrData.CreateShareLinks
|
|
|
|
config.Timezone = config_payload.WrapperrData.Timezone
|
|
|
|
config.ApplicationName = config_payload.WrapperrData.ApplicationName
|
|
|
|
config.ApplicationURL = config_payload.WrapperrData.ApplicationURL
|
|
|
|
config.WrappedEnd = config_payload.WrapperrData.WrappedEnd
|
|
|
|
config.WrappedStart = config_payload.WrapperrData.WrappedStart
|
|
|
|
config.WinterTheme = config_payload.WrapperrData.WinterTheme
|
|
|
|
|
|
|
|
err = files.SaveConfig(config)
|
|
|
|
if err != nil {
|
|
|
|
log.Println("Failed to save new Wrapperr configuration. Error: " + err.Error())
|
|
|
|
context.JSON(http.StatusInternalServerError, gin.H{"error": "Failed to save new Wrapperr configuration."})
|
|
|
|
context.Abort()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
time.Local = new_time
|
|
|
|
} else {
|
|
|
|
log.Println("Cannot set new config. Invalid data type received. Type: " + config_payload.DataType)
|
|
|
|
context.JSON(http.StatusBadRequest, gin.H{"error": "Failed to save new Wrapperr configuration."})
|
|
|
|
context.Abort()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if config_payload.ClearCache {
|
|
|
|
log.Println("Clear cache setting set to true. Clearing cache.")
|
|
|
|
|
|
|
|
err = files.ClearCache()
|
|
|
|
if err != nil {
|
|
|
|
log.Println("Failed to clear cache. Error: " + err.Error())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Println("New Wrapperr configuration saved for type: " + config_payload.DataType + ".")
|
|
|
|
context.JSON(http.StatusOK, gin.H{"message": "Saved new Wrapperr config."})
|
|
|
|
|
|
|
|
if config.WrapperrRoot != originalRoot {
|
|
|
|
log.Println("Root changed, attempting Wrapperr restart.")
|
|
|
|
err = utilities.RestartSelf()
|
|
|
|
if err != nil {
|
|
|
|
log.Println("Failed to restart. Error: " + err.Error())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// API route used to update admin accounts details (username, password).
|
|
|
|
func ApiUpdateAdmin(context *gin.Context) {
|
|
|
|
adminConfig, err := files.GetAdminConfig()
|
|
|
|
if err != nil {
|
|
|
|
log.Println("Failed to load Wrapperr admin configuration. Error: " + err.Error())
|
|
|
|
context.JSON(http.StatusInternalServerError, gin.H{"error": "Failed to load Wrapperr admin configuration."})
|
|
|
|
context.Abort()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Read payload from Post input
|
|
|
|
var admin_payload models.AdminConfigUpdateRequest
|
|
|
|
if err := context.ShouldBindJSON(&admin_payload); err != nil {
|
|
|
|
log.Println("Failed to parse request. Error: " + err.Error())
|
|
|
|
context.JSON(http.StatusBadRequest, gin.H{"error": "Failed to parse request."})
|
|
|
|
context.Abort()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Test old password
|
|
|
|
passwordValidity := utilities.ComparePasswords(adminConfig.AdminPassword, admin_payload.AdminPasswordOriginal)
|
|
|
|
if !passwordValidity {
|
|
|
|
ipString := utilities.GetOriginIPString(context)
|
|
|
|
log.Println("Admin login update failed. Incorrect password." + ipString)
|
|
|
|
context.JSON(http.StatusUnauthorized, gin.H{"error": "Invalid credentials."})
|
|
|
|
context.Abort()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Confirm username length
|
|
|
|
if len(admin_payload.AdminUsername) < 4 {
|
|
|
|
context.JSON(http.StatusBadRequest, gin.H{"error": "Admin username is too short. Four characters or more required."})
|
|
|
|
context.Abort()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Confirm password length
|
|
|
|
if len(admin_payload.AdminPassword) < 8 {
|
|
|
|
context.JSON(http.StatusBadRequest, gin.H{"error": "Admin password is too short. Eight characters or more required."})
|
|
|
|
context.Abort()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Hash new password
|
|
|
|
hash, err := utilities.HashAndSalt(admin_payload.AdminPassword)
|
|
|
|
if err != nil {
|
|
|
|
log.Println("Failed to hash your password. Error: " + err.Error())
|
|
|
|
context.JSON(http.StatusInternalServerError, gin.H{"error": "Failed to hash your password."})
|
|
|
|
context.Abort()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
adminConfig.AdminPassword = hash
|
|
|
|
|
|
|
|
// Save new admin config
|
|
|
|
err = files.SaveAdminConfig(adminConfig)
|
|
|
|
if err != nil {
|
|
|
|
log.Println("Failed to update admin. Error: " + err.Error())
|
|
|
|
context.JSON(http.StatusInternalServerError, gin.H{"error": "Failed to update admin."})
|
|
|
|
context.Abort()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update the private key to delete old logins
|
|
|
|
_, err = files.UpdatePrivateKey()
|
|
|
|
if err != nil {
|
|
|
|
log.Println("Failed to update private key. Error: " + err.Error())
|
|
|
|
log.Println("Admin account updated, but failed to rotate private key. Old logins still function.")
|
|
|
|
context.JSON(http.StatusInternalServerError, gin.H{"error": "Admin account updated, but failed to rotate private key. Old logins still function."})
|
|
|
|
context.Abort()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Println("New admin account created. Server is now claimed.")
|
|
|
|
|
|
|
|
context.JSON(http.StatusOK, gin.H{"message": "Admin created."})
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// API route which validates an admin JWT token
|
|
|
|
func ApiValidateAdmin(context *gin.Context) {
|
|
|
|
log.Println("Admin login session JWT validated.")
|
|
|
|
context.JSON(http.StatusOK, gin.H{"message": "Admin session is valid."})
|
|
|
|
}
|
|
|
|
|
|
|
|
// API route which retrieves lines from the log file
|
|
|
|
func ApiGetLog(context *gin.Context) {
|
|
|
|
logLines, err := files.GetLogLines()
|
|
|
|
if err != nil {
|
|
|
|
log.Println("Error trying to retrieve log lines. Error: " + err.Error())
|
|
|
|
context.JSON(http.StatusInternalServerError, gin.H{"error": "Error trying to retrieve log lines."})
|
|
|
|
context.Abort()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
logLinesReturn := models.WrapperrLogLineReply{
|
|
|
|
Message: "Log lines retrieved",
|
|
|
|
Error: false,
|
|
|
|
Data: logLines,
|
|
|
|
Limit: files.GetMaxLogLinesReturned(),
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Println("Log lines retrieved for admin.")
|
|
|
|
context.JSON(http.StatusOK, logLinesReturn)
|
|
|
|
return
|
|
|
|
}
|
2023-10-28 14:43:59 +00:00
|
|
|
|
|
|
|
// API route to get all timezones
|
|
|
|
func ApiGetTimezones(context *gin.Context) {
|
|
|
|
timezones, err := files.GetTimezones()
|
|
|
|
if err != nil {
|
|
|
|
log.Println("Error trying to get timezones. Error: " + err.Error())
|
|
|
|
context.JSON(http.StatusInternalServerError, gin.H{"error": "Failed to get timezones."})
|
|
|
|
context.Abort()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
context.JSON(http.StatusOK, gin.H{"message": "Timezones received.", "error": false, "data": timezones.Timezones})
|
|
|
|
return
|
|
|
|
}
|
2023-10-29 17:03:55 +00:00
|
|
|
|
|
|
|
func ApiWrapperCacheStatistics(context *gin.Context) {
|
|
|
|
log.Println("New caching request.")
|
|
|
|
|
|
|
|
configBool, err := files.GetConfigState()
|
|
|
|
if err != nil {
|
|
|
|
log.Println("Failed to retrieve configuration state. Error: " + err.Error())
|
|
|
|
context.JSON(http.StatusInternalServerError, gin.H{"error": "Failed to retrieve configuration state."})
|
|
|
|
context.Abort()
|
|
|
|
return
|
|
|
|
} else if !configBool {
|
|
|
|
context.JSON(http.StatusBadRequest, gin.H{"error": "Wrapperr is not configured."})
|
|
|
|
context.Abort()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
config, err := files.GetConfig()
|
|
|
|
if err != nil {
|
|
|
|
log.Println("Failed to load Wrapperr configuration. Error: " + err.Error())
|
|
|
|
context.JSON(http.StatusInternalServerError, gin.H{"error": "Failed to load Wrapperr configuration."})
|
|
|
|
context.Abort()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
adminConfig, err := files.GetAdminConfig()
|
|
|
|
if err != nil {
|
|
|
|
log.Println("Failed to load Wrapperr admin configuration. Error: " + err.Error())
|
|
|
|
context.JSON(http.StatusInternalServerError, gin.H{"error": "Failed to load Wrapperr admin configuration."})
|
|
|
|
context.Abort()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check connection to every Tautulli server
|
|
|
|
for i := 0; i < len(config.TautulliConfig); i++ {
|
|
|
|
log.Println("Checking Tautulli server '" + config.TautulliConfig[i].TautulliName + "'.")
|
|
|
|
tautulli_state, err := modules.TautulliTestConnection(config.TautulliConfig[i].TautulliPort, config.TautulliConfig[i].TautulliIP, config.TautulliConfig[i].TautulliHttps, config.TautulliConfig[i].TautulliRoot, config.TautulliConfig[i].TautulliApiKey)
|
|
|
|
if err != nil {
|
|
|
|
log.Println("Failed to reach Tautulli server '" + config.TautulliConfig[i].TautulliName + "'. Error: " + err.Error())
|
|
|
|
context.JSON(http.StatusInternalServerError, gin.H{"error": "Failed to reach Tautulli server '" + config.TautulliConfig[i].TautulliName + "'."})
|
|
|
|
context.Abort()
|
|
|
|
return
|
|
|
|
} else if !tautulli_state {
|
|
|
|
log.Println("Failed to ping Tautulli server '" + config.TautulliConfig[i].TautulliName + "' before retrieving statistics.")
|
|
|
|
context.JSON(http.StatusInternalServerError, gin.H{"error": "Failed to reach Tautulli server '" + config.TautulliConfig[i].TautulliName + "'."})
|
|
|
|
context.Abort()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var userName string = "Caching mode"
|
|
|
|
var userId int = 0
|
|
|
|
var userEmail string = "N/A"
|
|
|
|
var userFriendlyName = "Caching mode"
|
|
|
|
|
|
|
|
// Read payload from Post input
|
|
|
|
var cachingRequest models.CacheWrapperrRequest
|
|
|
|
if err := context.ShouldBindJSON(&cachingRequest); err != nil {
|
|
|
|
log.Println("Failed to parse request. Error: " + err.Error())
|
|
|
|
context.JSON(http.StatusBadRequest, gin.H{"error": "Failed to parse request."})
|
|
|
|
context.Abort()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if cachingRequest.CacheLimit < 1 {
|
|
|
|
context.JSON(http.StatusBadRequest, gin.H{"error": "Chache limit too low."})
|
|
|
|
context.Abort()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
_, cachingComplete, err := modules.GetWrapperStatistics(userName, userFriendlyName, userId, userEmail, config, adminConfig, true, cachingRequest.CacheLimit)
|
|
|
|
if err != nil {
|
|
|
|
log.Println("Failed to get statistics. Error: " + err.Error())
|
|
|
|
context.JSON(http.StatusInternalServerError, gin.H{"error": "Failed to get statistics."})
|
|
|
|
context.Abort()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
booleanReply := models.BooleanReply{
|
|
|
|
Message: "Completed caching request.",
|
|
|
|
Error: false,
|
|
|
|
Data: *cachingComplete,
|
|
|
|
}
|
|
|
|
|
|
|
|
context.JSON(http.StatusBadRequest, booleanReply)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func ApiGetUsers(context *gin.Context) {
|
|
|
|
configBool, err := files.GetConfigState()
|
|
|
|
if err != nil {
|
|
|
|
log.Println("Failed to retrieve configuration state. Error: " + err.Error())
|
|
|
|
context.JSON(http.StatusInternalServerError, gin.H{"error": "Failed to retrieve configuration state."})
|
|
|
|
context.Abort()
|
|
|
|
return
|
|
|
|
} else if !configBool {
|
|
|
|
context.JSON(http.StatusBadRequest, gin.H{"error": "Wrapperr is not configured."})
|
|
|
|
context.Abort()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
users, err := files.GetUsers()
|
|
|
|
if err != nil {
|
|
|
|
log.Println("Failed to get users. Error: " + err.Error())
|
|
|
|
context.JSON(http.StatusInternalServerError, gin.H{"error": "Failed to get users."})
|
|
|
|
context.Abort()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
context.JSON(http.StatusOK, gin.H{"message": "Users recieved.", "data": users})
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func ApiGetUser(context *gin.Context) {
|
|
|
|
configBool, err := files.GetConfigState()
|
|
|
|
if err != nil {
|
|
|
|
log.Println("Failed to retrieve configuration state. Error: " + err.Error())
|
|
|
|
context.JSON(http.StatusInternalServerError, gin.H{"error": "Failed to retrieve configuration state."})
|
|
|
|
context.Abort()
|
|
|
|
return
|
|
|
|
} else if !configBool {
|
|
|
|
context.JSON(http.StatusBadRequest, gin.H{"error": "Wrapperr is not configured."})
|
|
|
|
context.Abort()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
var userId = context.Param("userId")
|
|
|
|
|
|
|
|
userIDInt, err := strconv.Atoi(userId)
|
|
|
|
if err != nil {
|
|
|
|
log.Println("Failed to parse user ID. Error: " + err.Error())
|
|
|
|
context.JSON(http.StatusBadRequest, gin.H{"error": "Failed to parse user ID."})
|
|
|
|
context.Abort()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
user, err := modules.UsersGetUser(userIDInt)
|
|
|
|
if err != nil {
|
|
|
|
log.Println("Failed to get user. Error: " + err.Error())
|
|
|
|
context.JSON(http.StatusInternalServerError, gin.H{"error": "Failed to get user."})
|
|
|
|
context.Abort()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
context.JSON(http.StatusOK, gin.H{"message": "Users recieved.", "data": user})
|
|
|
|
return
|
|
|
|
}
|