選択できるのは25トピックまでです。 トピックは、先頭が英数字で、英数字とダッシュ('-')を使用した35文字以内のものにしてください。

189 行
6.6KB

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