Nie możesz wybrać więcej, niż 25 tematów Tematy muszą się zaczynać od litery lub cyfry, mogą zawierać myślniki ('-') i mogą mieć do 35 znaków.

180 wiersze
6.2KB

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