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"
|
2025-02-21 08:33:00 +01:00
|
|
|
"github.com/glebarez/sqlite"
|
|
|
|
"gorm.io/gorm"
|
2025-02-21 09:35:37 +01:00
|
|
|
|
|
|
|
"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")
|
2025-02-17 18:08:31 +01:00
|
|
|
// 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
|
2025-02-17 18:08:31 +01:00
|
|
|
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
|
|
|
}
|
|
|
|
|
2025-02-17 18:08:31 +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
|
|
|
|
}
|
|
|
|
|
2025-02-17 18:08:31 +01:00
|
|
|
//go:embed frontend/build/* frontend/static/*
|
2025-02-17 14:33:55 +01:00
|
|
|
var frontend embed.FS
|
|
|
|
|
|
|
|
func main() {
|
2025-02-21 09:35:37 +01:00
|
|
|
// 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)
|
|
|
|
}
|
2025-02-21 08:33:00 +01:00
|
|
|
|
|
|
|
// Auto migrate the schema
|
2025-02-21 09:35:37 +01:00
|
|
|
if err := db.AutoMigrate(¬es.Note{}, ¬es.NoteLink{}); err != nil {
|
2025-02-17 14:33:55 +01:00
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2025-02-21 09:35:37 +01:00
|
|
|
// Initialize services
|
|
|
|
noteService := notes.NewService(db)
|
|
|
|
noteHandler := notes.NewHandler(noteService)
|
|
|
|
|
2025-02-21 08:47:07 +01:00
|
|
|
// Create Gin router
|
|
|
|
r := gin.Default()
|
|
|
|
|
2025-02-17 14:33:55 +01:00
|
|
|
// API routes
|
2025-02-21 08:47:07 +01:00
|
|
|
api := r.Group("/api")
|
|
|
|
{
|
2025-02-21 09:35:37 +01:00
|
|
|
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
|
|
|
}
|
|
|
|
}
|