quicknotes/main.go

124 lines
2.9 KiB
Go
Raw Normal View History

2025-02-17 14:33:55 +01:00
package main
import (
"embed"
"log"
"mime"
"net/http"
"path"
"path/filepath"
"strings"
2025-02-21 08:47:07 +01:00
"github.com/gin-gonic/gin"
"github.com/glebarez/sqlite"
"gorm.io/gorm"
"qn/notes"
2025-02-17 14:33:55 +01:00
)
2025-02-21 08:47:07 +01:00
func serveStaticFile(c *gin.Context, prefix string) error {
cleanPath := path.Clean(c.Request.URL.Path)
2025-02-17 14:33:55 +01:00
if cleanPath == "/" {
cleanPath = "/index.html"
}
2025-02-17 14:41:57 +01:00
// Try to read the exact file first
2025-02-17 14:33:55 +01:00
content, err := frontend.ReadFile(prefix + cleanPath)
2025-02-17 14:41:57 +01:00
ext := strings.ToLower(filepath.Ext(cleanPath))
// If file not found OR the path has no extension (likely a route path), serve index.html
if err != nil || ext == "" {
content, err = frontend.ReadFile(prefix + "/index.html")
if err != nil {
return err
}
2025-02-21 08:47:07 +01:00
c.Header("Content-Type", "text/html; charset=utf-8")
// Add security headers for HTML content
2025-02-21 08:47:07 +01:00
c.Header("X-Content-Type-Options", "nosniff")
c.Data(http.StatusOK, "text/html; charset=utf-8", content)
2025-02-17 14:41:57 +01:00
return nil
2025-02-17 14:33:55 +01:00
}
2025-02-17 14:41:57 +01:00
// For actual files, set the correct MIME type
var mimeType string
switch ext {
case ".js":
mimeType = "application/javascript; charset=utf-8"
case ".css":
mimeType = "text/css; charset=utf-8"
case ".html":
mimeType = "text/html; charset=utf-8"
case ".json":
mimeType = "application/json; charset=utf-8"
case ".png":
mimeType = "image/png"
case ".svg":
mimeType = "image/svg+xml"
default:
mimeType = mime.TypeByExtension(ext)
if mimeType == "" {
// Try to detect content type from the content itself
mimeType = http.DetectContentType(content)
}
2025-02-17 14:33:55 +01:00
}
// Set security headers for all responses
2025-02-21 08:47:07 +01:00
c.Header("X-Content-Type-Options", "nosniff")
c.Data(http.StatusOK, mimeType, content)
2025-02-17 14:33:55 +01:00
return nil
}
//go:embed frontend/build/* frontend/static/*
2025-02-17 14:33:55 +01:00
var frontend embed.FS
func main() {
// Initialize database
db, err := gorm.Open(sqlite.Open("notes.db"), &gorm.Config{})
2025-02-17 14:33:55 +01:00
if err != nil {
log.Fatal(err)
}
// Auto migrate the schema
if err := db.AutoMigrate(&notes.Note{}, &notes.NoteLink{}); err != nil {
2025-02-17 14:33:55 +01:00
log.Fatal(err)
}
// Initialize services
noteService := notes.NewService(db)
noteHandler := notes.NewHandler(noteService)
2025-02-21 08:47:07 +01:00
// Create Gin router
r := gin.Default()
// Trust only loopback addresses
if err := r.SetTrustedProxies([]string{"127.0.0.1", "::1"}); err != nil {
log.Fatal(err)
}
2025-02-17 14:33:55 +01:00
// API routes
2025-02-21 08:47:07 +01:00
api := r.Group("/api")
{
noteHandler.RegisterRoutes(api)
// TODO: Add feeds and links routes when implemented
2025-02-21 08:47:07 +01:00
}
2025-02-17 14:33:55 +01:00
// Serve frontend
2025-02-21 08:47:07 +01:00
r.NoRoute(handleFrontend)
2025-02-17 14:33:55 +01:00
log.Printf("INFO: Server starting on http://localhost:3000")
2025-02-21 08:47:07 +01:00
log.Fatal(r.Run(":3000"))
2025-02-17 14:33:55 +01:00
}
2025-02-21 08:47:07 +01:00
func handleFrontend(c *gin.Context) {
2025-02-17 14:33:55 +01:00
// Don't serve API routes
2025-02-21 08:47:07 +01:00
if path.Dir(c.Request.URL.Path) == "/api" {
c.Status(http.StatusNotFound)
2025-02-17 14:33:55 +01:00
return
}
2025-02-21 08:47:07 +01:00
err := serveStaticFile(c, "frontend/build")
2025-02-17 14:41:57 +01:00
if err != nil { // if serveStaticFile returns an error, it has already tried to serve index.html as fallback
2025-02-21 08:47:07 +01:00
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
2025-02-17 14:33:55 +01:00
}
}