Вы не можете выбрать более 25 тем Темы должны начинаться с буквы или цифры, могут содержать дефисы(-) и должны содержать не более 35 символов.

160 строки
5.1KB

  1. package app
  2. import (
  3. "context"
  4. "errors"
  5. "fmt"
  6. "net/http"
  7. "strings"
  8. "time"
  9. "github.com/go-chi/chi/v5"
  10. "qctextbuilder/internal/buildsvc"
  11. "qctextbuilder/internal/config"
  12. "qctextbuilder/internal/domain"
  13. "qctextbuilder/internal/draftsvc"
  14. "qctextbuilder/internal/httpserver"
  15. "qctextbuilder/internal/httpserver/handlers"
  16. "qctextbuilder/internal/httpserver/views"
  17. "qctextbuilder/internal/logging"
  18. "qctextbuilder/internal/mapping"
  19. "qctextbuilder/internal/onboarding"
  20. "qctextbuilder/internal/polling"
  21. "qctextbuilder/internal/qcclient"
  22. "qctextbuilder/internal/store"
  23. "qctextbuilder/internal/store/memory"
  24. "qctextbuilder/internal/store/sqlite"
  25. "qctextbuilder/internal/templatesvc"
  26. )
  27. type App struct {
  28. server *httpserver.Server
  29. pollingSvc *polling.Service
  30. }
  31. func New(cfg config.Config) (*App, error) {
  32. logger := logging.New()
  33. var (
  34. templateStore store.TemplateStore
  35. manifestStore store.ManifestStore
  36. buildStore store.BuildStore
  37. draftStore store.DraftStore
  38. settingsStore store.SettingsStore
  39. )
  40. driver := strings.ToLower(strings.TrimSpace(cfg.DBDriver))
  41. switch driver {
  42. case "", "sqlite":
  43. sqliteStore, err := sqlite.New(cfg.DBURL)
  44. if err != nil {
  45. return nil, fmt.Errorf("init sqlite store: %w", err)
  46. }
  47. templateStore = sqliteStore
  48. manifestStore = sqliteStore
  49. buildStore = sqliteStore
  50. draftStore = sqliteStore
  51. settingsStore = sqliteStore
  52. default:
  53. memStore := memory.New()
  54. templateStore = memStore
  55. manifestStore = memStore
  56. buildStore = memStore
  57. draftStore = memStore
  58. settingsStore = memStore
  59. }
  60. qc := qcclient.New(cfg.QCBaseURL, cfg.QCToken, 15*time.Second, logger)
  61. templateSvc := templatesvc.New(qc, templateStore, manifestStore)
  62. onboardSvc := onboarding.New(qc, templateStore, manifestStore)
  63. draftSvc := draftsvc.New(draftStore, templateStore, manifestStore)
  64. mappingSvc := mapping.New()
  65. buildSvc := buildsvc.New(qc, templateStore, manifestStore, buildStore, mappingSvc, time.Duration(cfg.PollTimeoutSeconds)*time.Second)
  66. pollingSvc := polling.New(buildSvc, buildStore, time.Duration(cfg.PollIntervalSeconds)*time.Second, cfg.PollMaxConcurrent, logger)
  67. api := handlers.NewAPI(templateSvc, onboardSvc, draftSvc, buildSvc)
  68. baseSettings := domain.AppSettings{
  69. QCBaseURL: cfg.QCBaseURL,
  70. QCBearerTokenEncrypted: cfg.QCToken,
  71. LanguageOutputMode: "EN",
  72. JobPollIntervalSeconds: cfg.PollIntervalSeconds,
  73. JobPollTimeoutSeconds: cfg.PollTimeoutSeconds,
  74. MasterPrompt: domain.SeedMasterPrompt,
  75. PromptBlocks: domain.DefaultPromptBlocks(),
  76. }
  77. if existing, err := settingsStore.GetSettings(context.Background()); err == nil && existing != nil {
  78. baseSettings.MasterPrompt = existing.MasterPrompt
  79. baseSettings.PromptBlocks = existing.PromptBlocks
  80. }
  81. _ = settingsStore.UpsertSettings(context.Background(), baseSettings)
  82. renderer, err := views.NewRenderer("web/templates/*.gohtml")
  83. if err != nil {
  84. return nil, fmt.Errorf("init renderer: %w", err)
  85. }
  86. ui := handlers.NewUI(templateSvc, onboardSvc, draftSvc, buildSvc, settingsStore, cfg, renderer)
  87. server := httpserver.New(cfg.HTTPAddr, logger, func(r chi.Router) {
  88. r.Get("/", ui.Home)
  89. r.Get("/settings", ui.Settings)
  90. r.Post("/settings/prompt", ui.SavePromptSettings)
  91. r.Get("/templates", ui.Templates)
  92. r.Post("/templates/sync", ui.SyncTemplates)
  93. r.Get("/templates/{id}", ui.TemplateDetail)
  94. r.Post("/templates/{id}/onboard", ui.OnboardTemplate)
  95. r.Post("/templates/{id}/fields", ui.UpdateTemplateFields)
  96. r.Get("/builds/new", ui.BuildNew)
  97. r.Post("/builds/drafts", ui.SaveDraft)
  98. r.Post("/builds/drafts/autofill", ui.AutofillDraft)
  99. r.Post("/builds", ui.CreateBuild)
  100. r.Get("/builds/{id}", ui.BuildDetail)
  101. r.Post("/builds/{id}/poll", ui.PollBuild)
  102. r.Post("/builds/{id}/fetch-editor-url", ui.FetchEditorURL)
  103. r.Get("/healthz", api.Health)
  104. r.Route("/api", func(r chi.Router) {
  105. r.Post("/templates/sync", api.SyncTemplates)
  106. r.Get("/templates", api.ListTemplates)
  107. r.Get("/templates/{id}", api.GetTemplateDetail)
  108. r.Post("/templates/{id}/onboard", api.OnboardTemplate)
  109. r.Put("/templates/{id}/fields", api.UpdateTemplateFields)
  110. r.Post("/drafts/intake", api.IntakeDraft)
  111. r.Get("/drafts", api.ListDrafts)
  112. r.Get("/drafts/{id}", api.GetDraft)
  113. r.Put("/drafts/{id}", api.UpdateDraft)
  114. r.Post("/site-builds", api.StartBuild)
  115. r.Get("/site-builds/{id}", api.GetBuild)
  116. r.Post("/site-builds/{id}/poll", api.PollBuildOnce)
  117. r.Post("/site-builds/{id}/fetch-editor-url", api.FetchBuildEditorURL)
  118. })
  119. })
  120. return &App{server: server, pollingSvc: pollingSvc}, nil
  121. }
  122. func (a *App) Run(ctx context.Context) error {
  123. go func() {
  124. if err := a.pollingSvc.Start(ctx); err != nil {
  125. // polling is best-effort in milestone-2; request flow works without supervisor
  126. }
  127. }()
  128. errCh := make(chan error, 1)
  129. go func() {
  130. if err := a.server.Run(); err != nil && !errors.Is(err, http.ErrServerClosed) {
  131. errCh <- fmt.Errorf("http run: %w", err)
  132. }
  133. close(errCh)
  134. }()
  135. select {
  136. case <-ctx.Done():
  137. shutdownCtx, cancel := context.WithTimeout(context.Background(), 10*time.Second)
  138. defer cancel()
  139. return a.server.Shutdown(shutdownCtx)
  140. case err := <-errCh:
  141. return err
  142. }
  143. }