Files
scriptforge/backend/cmd/server/main.go
zhilv e6e4357a28 feat: 脚本创作+发布+市场体系
- 数据模型新增: title(必填), description(可选), status(draft/published)
- 新增 API: POST /scripts/:id/publish, GET /api/market (搜索+分页+runtime过滤)
- 前端首页重构: 选语言 → CodeMirror 编辑器(8种语言语法高亮) → 标题/描述 → 草稿/发布
- 新增 /market 页面: 浏览已发布脚本, 搜索+过滤+分页
- 详情页新增: 发布按钮(草稿→市场), title/description 展示
- Shell 类运行时显示 source 命令(继承环境变量)
- backend GetSourceCommand 支持 bash/zsh/sh/fish 四种 shell 格式

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-29 14:04:15 +08:00

132 lines
2.8 KiB
Go

package main
import (
"io/fs"
"log"
"net/http"
"os"
"os/signal"
"strings"
"syscall"
"time"
"github.com/gin-gonic/gin"
"gorm.io/driver/sqlite"
"gorm.io/gorm"
"gitea.kmux.cn/zhilv/scriptforge/internal/assets"
"gitea.kmux.cn/zhilv/scriptforge/internal/handler"
"gitea.kmux.cn/zhilv/scriptforge/internal/middleware"
"gitea.kmux.cn/zhilv/scriptforge/internal/model"
"gitea.kmux.cn/zhilv/scriptforge/internal/service"
)
func main() {
gin.SetMode(gin.ReleaseMode)
db := initDB()
syncDB(db)
svc := service.NewScriptService(db)
go cleanupLoop(svc)
r := gin.New()
r.Use(gin.Recovery(), middleware.Logger())
api := r.Group("/api")
{
api.Use(middleware.RateLimit(60))
api.POST("/scripts", middleware.RateLimit(10), handler.CreateScript(svc))
api.GET("/scripts/:id", handler.GetScript(svc))
api.POST("/scripts/:id/publish", handler.PublishScript(svc))
api.DELETE("/scripts/:id", handler.DeleteScript(svc))
api.GET("/market", handler.ListMarket(svc))
}
r.GET("/raw/:id", handler.GetRawScript(svc))
r.GET("/health", func(c *gin.Context) {
c.JSON(http.StatusOK, gin.H{"status": "ok"})
})
serveStaticOrSPA(r)
port := getPort()
log.Printf("Listening on :%s", port)
go func() {
if err := r.Run(":" + port); err != nil {
log.Fatalf("server error: %v", err)
}
}()
quit := make(chan os.Signal, 1)
signal.Notify(quit, syscall.SIGINT, syscall.SIGTERM)
<-quit
log.Println("Shutting down...")
}
func initDB() *gorm.DB {
dbPath := os.Getenv("DB_PATH")
if dbPath == "" {
dbPath = "scriptforge.db"
}
db, err := gorm.Open(sqlite.Open(dbPath), &gorm.Config{})
if err != nil {
log.Fatalf("failed to connect database: %v", err)
}
return db
}
func syncDB(db *gorm.DB) {
if err := db.AutoMigrate(&model.Script{}); err != nil {
log.Fatalf("failed to migrate database: %v", err)
}
}
func cleanupLoop(svc *service.ScriptService) {
for {
time.Sleep(1 * time.Hour)
if n, err := svc.CleanupExpired(); err != nil {
log.Printf("cleanup error: %v", err)
} else if n > 0 {
log.Printf("cleaned up %d expired scripts", n)
}
}
}
func getPort() string {
if p := os.Getenv("PORT"); p != "" {
return p
}
return "8080"
}
func serveStaticOrSPA(r *gin.Engine) {
staticFS, err := fs.Sub(assets.FS, "dist")
if err != nil {
log.Printf("no embedded static files, API-only mode: %v", err)
return
}
r.NoRoute(func(c *gin.Context) {
path := c.Request.URL.Path
if strings.HasPrefix(path, "/api") || strings.HasPrefix(path, "/raw") {
c.JSON(http.StatusNotFound, gin.H{"error": "not found"})
return
}
file, err := staticFS.Open(path)
if err != nil {
c.FileFromFS("index.html", http.FS(staticFS))
return
}
file.Close()
c.FileFromFS(path, http.FS(staticFS))
})
r.GET("/", func(c *gin.Context) {
c.FileFromFS("index.html", http.FS(staticFS))
})
}