AnthoLume/api/api.go

238 lines
8.4 KiB
Go
Raw Normal View History

2023-09-18 23:57:18 +00:00
package api
import (
"crypto/rand"
2023-11-29 03:01:49 +00:00
"embed"
"fmt"
2023-09-18 23:57:18 +00:00
"html/template"
2023-11-29 03:01:49 +00:00
"io/fs"
2023-09-18 23:57:18 +00:00
"net/http"
2023-11-29 01:05:50 +00:00
"path/filepath"
"strings"
2023-09-18 23:57:18 +00:00
"github.com/gin-contrib/multitemplate"
"github.com/gin-contrib/sessions"
"github.com/gin-contrib/sessions/cookie"
"github.com/gin-gonic/gin"
"github.com/microcosm-cc/bluemonday"
log "github.com/sirupsen/logrus"
2023-09-18 23:57:18 +00:00
"reichard.io/bbank/config"
"reichard.io/bbank/database"
)
type API struct {
Router *gin.Engine
Config *config.Config
DB *database.DBManager
HTMLPolicy *bluemonday.Policy
2023-11-29 03:01:49 +00:00
Assets *embed.FS
2023-12-01 12:35:51 +00:00
Templates map[string]*template.Template
2023-09-18 23:57:18 +00:00
}
2023-12-01 12:35:51 +00:00
func NewApi(db *database.DBManager, c *config.Config, assets *embed.FS) *API {
2023-09-18 23:57:18 +00:00
api := &API{
HTMLPolicy: bluemonday.StrictPolicy(),
Router: gin.Default(),
Config: c,
DB: db,
2023-12-01 12:35:51 +00:00
Assets: assets,
2023-09-18 23:57:18 +00:00
}
// Assets & Web App Templates
2023-11-29 03:01:49 +00:00
assetsDir, _ := fs.Sub(assets, "assets")
api.Router.StaticFS("/assets", http.FS(assetsDir))
2023-09-18 23:57:18 +00:00
// Generate Secure Token
var newToken []byte
var err error
if c.CookieSessionKey != "" {
log.Info("[NewApi] Utilizing Environment Cookie Session Key")
newToken = []byte(c.CookieSessionKey)
} else {
log.Info("[NewApi] Generating Cookie Session Key")
newToken, err = generateToken(64)
if err != nil {
panic("Unable to generate secure token")
}
2023-09-18 23:57:18 +00:00
}
// Configure Cookie Session Store
store := cookie.NewStore(newToken)
store.Options(sessions.Options{
MaxAge: 60 * 60 * 24 * 7,
Secure: c.CookieSecure,
HttpOnly: c.CookieHTTPOnly,
2023-09-18 23:57:18 +00:00
SameSite: http.SameSiteStrictMode,
})
api.Router.Use(sessions.Sessions("token", store))
// Register Web App Route
api.registerWebAppRoutes()
// Register API Routes
apiGroup := api.Router.Group("/api")
api.registerKOAPIRoutes(apiGroup)
api.registerOPDSRoutes(apiGroup)
2023-09-18 23:57:18 +00:00
return api
}
func (api *API) registerWebAppRoutes() {
2023-11-29 01:05:50 +00:00
// Generate Templates
api.Router.HTMLRender = *api.generateTemplates()
2023-09-18 23:57:18 +00:00
2023-10-31 10:28:22 +00:00
// Static Assets (Required @ Root)
api.Router.GET("/manifest.json", api.webManifest)
2023-11-26 00:21:18 +00:00
api.Router.GET("/favicon.ico", api.faviconIcon)
2023-10-29 00:07:24 +00:00
api.Router.GET("/sw.js", api.serviceWorker)
2023-10-31 10:28:22 +00:00
// Local / Offline Static Pages (No Template, No Auth)
api.Router.GET("/local", api.localDocuments)
// Reader (Reader Page, Document Progress, Devices)
2023-10-29 00:07:24 +00:00
api.Router.GET("/reader", api.documentReader)
api.Router.GET("/reader/devices", api.authWebAppMiddleware, api.getDevices)
api.Router.GET("/reader/progress/:document", api.authWebAppMiddleware, api.getDocumentProgress)
2023-10-29 00:07:24 +00:00
2023-10-31 10:28:22 +00:00
// Web App
2023-09-18 23:57:18 +00:00
api.Router.GET("/", api.authWebAppMiddleware, api.createAppResourcesRoute("home"))
api.Router.GET("/activity", api.authWebAppMiddleware, api.createAppResourcesRoute("activity"))
api.Router.GET("/documents", api.authWebAppMiddleware, api.createAppResourcesRoute("documents"))
api.Router.GET("/documents/:document", api.authWebAppMiddleware, api.createAppResourcesRoute("document"))
2023-09-18 23:57:18 +00:00
api.Router.GET("/documents/:document/cover", api.authWebAppMiddleware, api.getDocumentCover)
2023-10-31 10:28:22 +00:00
api.Router.GET("/documents/:document/file", api.authWebAppMiddleware, api.downloadDocument)
api.Router.GET("/login", api.createAppResourcesRoute("login"))
api.Router.GET("/logout", api.authWebAppMiddleware, api.authLogout)
api.Router.GET("/register", api.createAppResourcesRoute("login", gin.H{"Register": true}))
api.Router.GET("/settings", api.authWebAppMiddleware, api.createAppResourcesRoute("settings"))
api.Router.POST("/login", api.authFormLogin)
api.Router.POST("/register", api.authFormRegister)
// Demo Mode Enabled Configuration
if api.Config.DemoMode {
api.Router.POST("/documents", api.authWebAppMiddleware, api.demoModeAppError)
api.Router.POST("/documents/:document/delete", api.authWebAppMiddleware, api.demoModeAppError)
api.Router.POST("/documents/:document/edit", api.authWebAppMiddleware, api.demoModeAppError)
api.Router.POST("/documents/:document/identify", api.authWebAppMiddleware, api.demoModeAppError)
api.Router.POST("/settings", api.authWebAppMiddleware, api.demoModeAppError)
} else {
api.Router.POST("/documents", api.authWebAppMiddleware, api.uploadNewDocument)
api.Router.POST("/documents/:document/delete", api.authWebAppMiddleware, api.deleteDocument)
api.Router.POST("/documents/:document/edit", api.authWebAppMiddleware, api.editDocument)
api.Router.POST("/documents/:document/identify", api.authWebAppMiddleware, api.identifyDocument)
api.Router.POST("/settings", api.authWebAppMiddleware, api.editSettings)
}
2023-10-07 01:25:56 +00:00
2023-10-31 10:28:22 +00:00
// Search Enabled Configuration
2023-10-07 01:25:56 +00:00
if api.Config.SearchEnabled {
api.Router.GET("/search", api.authWebAppMiddleware, api.createAppResourcesRoute("search"))
api.Router.POST("/search", api.authWebAppMiddleware, api.saveNewDocument)
}
2023-09-18 23:57:18 +00:00
}
func (api *API) registerKOAPIRoutes(apiGroup *gin.RouterGroup) {
koGroup := apiGroup.Group("/ko")
2023-10-31 10:28:22 +00:00
// KO Sync Routes (WebApp Uses - Progress & Activity)
koGroup.GET("/documents/:document/file", api.authKOMiddleware, api.downloadDocument)
2023-10-31 10:28:22 +00:00
koGroup.GET("/syncs/progress/:document", api.authKOMiddleware, api.getProgress)
koGroup.GET("/users/auth", api.authKOMiddleware, api.authorizeUser)
koGroup.POST("/activity", api.authKOMiddleware, api.addActivities)
koGroup.POST("/syncs/activity", api.authKOMiddleware, api.checkActivitySync)
2023-10-31 10:28:22 +00:00
koGroup.POST("/users/create", api.createUser)
koGroup.PUT("/syncs/progress", api.authKOMiddleware, api.setProgress)
// Demo Mode Enabled Configuration
if api.Config.DemoMode {
koGroup.POST("/documents", api.authKOMiddleware, api.demoModeJSONError)
koGroup.POST("/syncs/documents", api.authKOMiddleware, api.demoModeJSONError)
koGroup.PUT("/documents/:document/file", api.authKOMiddleware, api.demoModeJSONError)
} else {
koGroup.POST("/documents", api.authKOMiddleware, api.addDocuments)
koGroup.POST("/syncs/documents", api.authKOMiddleware, api.checkDocumentsSync)
koGroup.PUT("/documents/:document/file", api.authKOMiddleware, api.uploadExistingDocument)
}
}
func (api *API) registerOPDSRoutes(apiGroup *gin.RouterGroup) {
opdsGroup := apiGroup.Group("/opds")
2023-10-31 10:28:22 +00:00
// OPDS Routes
opdsGroup.GET("", api.authOPDSMiddleware, api.opdsEntry)
opdsGroup.GET("/", api.authOPDSMiddleware, api.opdsEntry)
opdsGroup.GET("/search.xml", api.authOPDSMiddleware, api.opdsSearchDescription)
opdsGroup.GET("/documents", api.authOPDSMiddleware, api.opdsDocuments)
opdsGroup.GET("/documents/:document/cover", api.authOPDSMiddleware, api.getDocumentCover)
2023-10-31 10:28:22 +00:00
opdsGroup.GET("/documents/:document/file", api.authOPDSMiddleware, api.downloadDocument)
2023-09-18 23:57:18 +00:00
}
2023-11-29 01:05:50 +00:00
func (api *API) generateTemplates() *multitemplate.Renderer {
// Define Templates & Helper Functions
2023-12-01 12:35:51 +00:00
templates := make(map[string]*template.Template)
2023-11-29 01:05:50 +00:00
render := multitemplate.NewRenderer()
helperFuncs := template.FuncMap{
"GetSVGGraphData": getSVGGraphData,
"GetUTCOffsets": getUTCOffsets,
"NiceSeconds": niceSeconds,
"dict": dict,
}
// Load Base
2023-11-29 03:01:49 +00:00
b, _ := api.Assets.ReadFile("templates/base.html")
2023-11-29 01:05:50 +00:00
baseTemplate := template.Must(template.New("base").Funcs(helperFuncs).Parse(string(b)))
// Load SVGs
2023-11-29 03:01:49 +00:00
svgs, _ := api.Assets.ReadDir("templates/svgs")
for _, item := range svgs {
basename := item.Name()
path := fmt.Sprintf("templates/svgs/%s", basename)
2023-11-29 01:05:50 +00:00
name := strings.TrimSuffix(basename, filepath.Ext(basename))
2023-11-29 03:01:49 +00:00
b, _ := api.Assets.ReadFile(path)
2023-11-29 01:05:50 +00:00
baseTemplate = template.Must(baseTemplate.New("svg/" + name).Parse(string(b)))
2023-12-01 12:35:51 +00:00
templates["svg/"+name] = baseTemplate
2023-11-29 01:05:50 +00:00
}
// Load Components
2023-11-29 03:01:49 +00:00
components, _ := api.Assets.ReadDir("templates/components")
for _, item := range components {
basename := item.Name()
path := fmt.Sprintf("templates/components/%s", basename)
2023-11-29 01:05:50 +00:00
name := strings.TrimSuffix(basename, filepath.Ext(basename))
2023-12-01 12:35:51 +00:00
// Clone Base Template
2023-11-29 03:01:49 +00:00
b, _ := api.Assets.ReadFile(path)
2023-11-29 01:05:50 +00:00
baseTemplate = template.Must(baseTemplate.New("component/" + name).Parse(string(b)))
2023-12-01 12:35:51 +00:00
render.Add("component/"+name, baseTemplate)
templates["component/"+name] = baseTemplate
2023-11-29 01:05:50 +00:00
}
// Load Pages
2023-11-29 03:01:49 +00:00
pages, _ := api.Assets.ReadDir("templates/pages")
for _, item := range pages {
basename := item.Name()
path := fmt.Sprintf("templates/pages/%s", basename)
2023-11-29 01:05:50 +00:00
name := strings.TrimSuffix(basename, filepath.Ext(basename))
// Clone Base Template
2023-11-29 03:01:49 +00:00
b, _ := api.Assets.ReadFile(path)
2023-11-29 01:05:50 +00:00
pageTemplate, _ := template.Must(baseTemplate.Clone()).New("page/" + name).Parse(string(b))
render.Add("page/"+name, pageTemplate)
2023-12-01 12:35:51 +00:00
templates["page/"+name] = pageTemplate
2023-11-29 01:05:50 +00:00
}
2023-12-01 12:35:51 +00:00
api.Templates = templates
2023-11-29 01:05:50 +00:00
return &render
}
2023-09-18 23:57:18 +00:00
func generateToken(n int) ([]byte, error) {
b := make([]byte, n)
_, err := rand.Read(b)
if err != nil {
return nil, err
}
return b, nil
}