2024-11-12 00:31:10 +00:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
2024-11-17 18:10:35 +00:00
|
|
|
"context"
|
2024-11-29 23:51:21 +00:00
|
|
|
"embed"
|
2024-11-12 00:31:10 +00:00
|
|
|
"errors"
|
|
|
|
"flag"
|
|
|
|
"fmt"
|
|
|
|
"log"
|
|
|
|
"net/http"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
2024-12-02 13:45:49 +00:00
|
|
|
"tesseract/internal/apierror"
|
2024-11-12 00:31:10 +00:00
|
|
|
"tesseract/internal/migration"
|
|
|
|
"tesseract/internal/service"
|
|
|
|
"tesseract/internal/template"
|
|
|
|
"tesseract/internal/workspace"
|
2024-11-29 23:51:21 +00:00
|
|
|
|
|
|
|
"github.com/golang-migrate/migrate/v4"
|
|
|
|
"github.com/labstack/echo/v4"
|
|
|
|
"github.com/labstack/echo/v4/middleware"
|
2024-11-12 00:31:10 +00:00
|
|
|
)
|
|
|
|
|
2024-11-29 23:51:21 +00:00
|
|
|
//go:embed web/dist/*
|
|
|
|
var web embed.FS
|
|
|
|
|
2024-11-12 00:31:10 +00:00
|
|
|
func main() {
|
|
|
|
execPath, err := os.Executable()
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalln(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
var configPath string
|
|
|
|
flag.StringVar(&configPath, "config", filepath.Join(execPath, "config.json"), "absolute/relative path to the config file.")
|
|
|
|
|
|
|
|
flag.Parse()
|
|
|
|
|
|
|
|
configPath, err = filepath.Abs(configPath)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalln(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
f, err := os.Open(configPath)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalln(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
config, err := service.ReadConfigFrom(f)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalln(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
services, err := service.Initialize(config)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalln(err)
|
|
|
|
}
|
|
|
|
|
2024-11-28 19:17:37 +00:00
|
|
|
err = migration.Up(fmt.Sprintf("sqlite://%s", config.DatabasePath))
|
2024-11-12 00:31:10 +00:00
|
|
|
if err != nil && !errors.Is(err, migrate.ErrNoChange) {
|
|
|
|
log.Fatalln(err)
|
|
|
|
}
|
|
|
|
|
2024-11-17 18:10:35 +00:00
|
|
|
log.Println("syncing all workspaces...")
|
2024-11-20 21:05:31 +00:00
|
|
|
syncCtx, cancel := context.WithCancel(context.Background())
|
|
|
|
if err = workspace.SyncAll(syncCtx, services); err != nil {
|
2024-11-17 18:10:35 +00:00
|
|
|
log.Fatalln(err)
|
|
|
|
}
|
2024-11-20 21:05:31 +00:00
|
|
|
cancel()
|
2024-11-17 18:10:35 +00:00
|
|
|
|
2024-11-12 00:31:10 +00:00
|
|
|
apiServer := echo.New()
|
2024-12-01 19:05:13 +00:00
|
|
|
apiServer.Use(middleware.StaticWithConfig(middleware.StaticConfig{
|
|
|
|
HTML5: true,
|
|
|
|
Root: "web/dist",
|
|
|
|
Filesystem: http.FS(web),
|
|
|
|
}))
|
2024-11-29 23:51:21 +00:00
|
|
|
|
|
|
|
apiServer.Use(services.ReverseProxy.Middleware(), services.Middleware(), middleware.CORS())
|
|
|
|
|
2024-11-12 00:31:10 +00:00
|
|
|
g := apiServer.Group("/api")
|
2024-11-20 21:05:31 +00:00
|
|
|
workspace.DefineRoutes(g, services)
|
2024-11-29 23:52:19 +00:00
|
|
|
template.DefineRoutes(g, services)
|
2024-11-12 00:31:10 +00:00
|
|
|
|
|
|
|
apiServer.HTTPErrorHandler = func(err error, c echo.Context) {
|
|
|
|
var he *echo.HTTPError
|
|
|
|
if errors.As(err, &he) {
|
|
|
|
if err = c.JSON(he.Code, he.Message); err != nil {
|
|
|
|
c.Logger().Error(err)
|
|
|
|
_ = c.NoContent(http.StatusInternalServerError)
|
|
|
|
}
|
2024-12-02 13:45:49 +00:00
|
|
|
return
|
2024-11-12 00:31:10 +00:00
|
|
|
}
|
2024-12-02 13:45:49 +00:00
|
|
|
|
|
|
|
var apiErr *apierror.APIError
|
|
|
|
if errors.As(err, &apiErr) {
|
|
|
|
if err = c.JSON(apiErr.StatusCode, apiErr); err != nil {
|
|
|
|
c.Logger().Error(err)
|
|
|
|
_ = c.NoContent(http.StatusInternalServerError)
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
c.Logger().Error(err)
|
|
|
|
_ = c.NoContent(http.StatusInternalServerError)
|
2024-11-12 00:31:10 +00:00
|
|
|
}
|
|
|
|
|
2024-12-02 17:15:39 +00:00
|
|
|
apiServer.Logger.Fatal(apiServer.Start(fmt.Sprintf(":%d", config.Port)))
|
2024-11-12 00:31:10 +00:00
|
|
|
}
|