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.

809 wiersze
26KB

  1. package sqlite
  2. import (
  3. "context"
  4. "database/sql"
  5. "embed"
  6. "encoding/json"
  7. "fmt"
  8. "io/fs"
  9. "os"
  10. "path/filepath"
  11. "sort"
  12. "strings"
  13. "time"
  14. _ "modernc.org/sqlite"
  15. "qctextbuilder/internal/domain"
  16. "qctextbuilder/internal/store"
  17. )
  18. //go:embed migrations/*.sql
  19. var migrationFS embed.FS
  20. type Store struct {
  21. db *sql.DB
  22. }
  23. func New(dbPath string) (*Store, error) {
  24. path := strings.TrimSpace(dbPath)
  25. if path == "" {
  26. path = "data/qctextbuilder.db"
  27. }
  28. if err := os.MkdirAll(filepath.Dir(path), 0o755); err != nil {
  29. return nil, fmt.Errorf("create db directory: %w", err)
  30. }
  31. db, err := sql.Open("sqlite", path)
  32. if err != nil {
  33. return nil, fmt.Errorf("open sqlite: %w", err)
  34. }
  35. db.SetMaxOpenConns(1)
  36. if _, err := db.Exec("PRAGMA foreign_keys = ON;"); err != nil {
  37. _ = db.Close()
  38. return nil, fmt.Errorf("enable foreign keys: %w", err)
  39. }
  40. if err := runMigrations(db); err != nil {
  41. _ = db.Close()
  42. return nil, fmt.Errorf("run migrations: %w", err)
  43. }
  44. return &Store{db: db}, nil
  45. }
  46. func (s *Store) Close() error {
  47. if s == nil || s.db == nil {
  48. return nil
  49. }
  50. return s.db.Close()
  51. }
  52. func (s *Store) UpsertTemplates(ctx context.Context, templates []domain.Template) error {
  53. tx, err := s.db.BeginTx(ctx, nil)
  54. if err != nil {
  55. return err
  56. }
  57. defer rollback(tx)
  58. stmt := `
  59. INSERT INTO qc_templates (
  60. id, name, description, locale, thumbnail_url, template_preview_url, type,
  61. palette_ready, raw_template_json, is_ai_template, is_onboarded, manifest_status, last_discovered_at, updated_at
  62. ) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, COALESCE((SELECT is_onboarded FROM qc_templates WHERE id = ?), ?),
  63. COALESCE((SELECT manifest_status FROM qc_templates WHERE id = ?), ?),
  64. COALESCE((SELECT last_discovered_at FROM qc_templates WHERE id = ?), ?),
  65. ?
  66. )
  67. ON CONFLICT(id) DO UPDATE SET
  68. name = excluded.name,
  69. description = excluded.description,
  70. locale = excluded.locale,
  71. thumbnail_url = excluded.thumbnail_url,
  72. template_preview_url = excluded.template_preview_url,
  73. type = excluded.type,
  74. palette_ready = excluded.palette_ready,
  75. raw_template_json = excluded.raw_template_json,
  76. is_ai_template = excluded.is_ai_template,
  77. updated_at = excluded.updated_at;
  78. `
  79. now := time.Now().UTC()
  80. for _, t := range templates {
  81. _, err := tx.ExecContext(ctx, stmt,
  82. t.ID, t.Name, t.Description, t.Locale, t.ThumbnailURL, t.TemplatePreviewURL, t.Type,
  83. boolToInt(t.PaletteReady), asRaw(t.RawJSON), boolToInt(t.IsAITemplate),
  84. t.ID, boolToInt(t.IsOnboarded),
  85. t.ID, defaultString(t.ManifestStatus, "missing"),
  86. t.ID, asRFC3339Ptr(t.LastDiscoveredAt),
  87. now.Format(time.RFC3339Nano),
  88. )
  89. if err != nil {
  90. return fmt.Errorf("upsert template %d: %w", t.ID, err)
  91. }
  92. }
  93. return tx.Commit()
  94. }
  95. func (s *Store) GetTemplateByID(ctx context.Context, id int64) (*domain.Template, error) {
  96. row := s.db.QueryRowContext(ctx, `
  97. SELECT id, name, description, locale, thumbnail_url, template_preview_url, type,
  98. palette_ready, raw_template_json, is_ai_template, is_onboarded, manifest_status, last_discovered_at
  99. FROM qc_templates
  100. WHERE id = ?`, id)
  101. t, err := scanTemplate(row.Scan)
  102. if err != nil {
  103. return nil, err
  104. }
  105. return t, nil
  106. }
  107. func (s *Store) ListTemplates(ctx context.Context) ([]domain.Template, error) {
  108. rows, err := s.db.QueryContext(ctx, `
  109. SELECT id, name, description, locale, thumbnail_url, template_preview_url, type,
  110. palette_ready, raw_template_json, is_ai_template, is_onboarded, manifest_status, last_discovered_at
  111. FROM qc_templates`)
  112. if err != nil {
  113. return nil, err
  114. }
  115. defer rows.Close()
  116. out := make([]domain.Template, 0)
  117. for rows.Next() {
  118. t, err := scanTemplate(rows.Scan)
  119. if err != nil {
  120. return nil, err
  121. }
  122. out = append(out, *t)
  123. }
  124. return out, rows.Err()
  125. }
  126. func (s *Store) SetTemplateManifestStatus(ctx context.Context, templateID int64, status string, onboarded bool) error {
  127. res, err := s.db.ExecContext(ctx, `
  128. UPDATE qc_templates
  129. SET manifest_status = ?, is_onboarded = ?, last_discovered_at = ?, updated_at = ?
  130. WHERE id = ?`,
  131. defaultString(status, "missing"),
  132. boolToInt(onboarded),
  133. time.Now().UTC().Format(time.RFC3339Nano),
  134. time.Now().UTC().Format(time.RFC3339Nano),
  135. templateID,
  136. )
  137. if err != nil {
  138. return err
  139. }
  140. n, _ := res.RowsAffected()
  141. if n == 0 {
  142. return store.ErrNotFound
  143. }
  144. return nil
  145. }
  146. func (s *Store) CreateManifest(ctx context.Context, manifest domain.TemplateManifest, fields []domain.TemplateField) error {
  147. tx, err := s.db.BeginTx(ctx, nil)
  148. if err != nil {
  149. return err
  150. }
  151. defer rollback(tx)
  152. if _, err := tx.ExecContext(ctx, `UPDATE qc_template_manifests SET is_active = 0 WHERE template_id = ?`, manifest.TemplateID); err != nil {
  153. return err
  154. }
  155. _, err = tx.ExecContext(ctx, `
  156. INSERT INTO qc_template_manifests (
  157. id, template_id, manifest_version, source, language_used_discovery, discovery_payload_json,
  158. discovery_response_json, flattened_manifest_json, is_active, created_at, updated_at
  159. ) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)`,
  160. manifest.ID, manifest.TemplateID, manifest.Version, manifest.Source, manifest.LanguageUsedDiscovery,
  161. asRaw(manifest.DiscoveryPayloadJSON), asRaw(manifest.DiscoveryResponseJSON), asRaw(manifest.FlattenedManifestJSON),
  162. boolToInt(manifest.IsActive), manifest.CreatedAt.UTC().Format(time.RFC3339Nano), manifest.UpdatedAt.UTC().Format(time.RFC3339Nano),
  163. )
  164. if err != nil {
  165. return err
  166. }
  167. if _, err := tx.ExecContext(ctx, `DELETE FROM qc_template_fields WHERE manifest_id = ?`, manifest.ID); err != nil {
  168. return err
  169. }
  170. for _, f := range fields {
  171. _, err := tx.ExecContext(ctx, `
  172. INSERT INTO qc_template_fields (
  173. id, template_id, manifest_id, section, website_section, key_name, path, field_kind,
  174. sample_value, is_enabled, is_required_by_us, display_label, display_order, notes
  175. ) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)`,
  176. f.ID, f.TemplateID, f.ManifestID, f.Section, domain.NormalizeWebsiteSection(f.WebsiteSection), f.KeyName, f.Path, f.FieldKind,
  177. f.SampleValue, boolToInt(f.IsEnabled), boolToInt(f.IsRequiredByUs), f.DisplayLabel, f.DisplayOrder, f.Notes,
  178. )
  179. if err != nil {
  180. return err
  181. }
  182. }
  183. return tx.Commit()
  184. }
  185. func (s *Store) GetActiveManifestByTemplateID(ctx context.Context, templateID int64) (*domain.TemplateManifest, error) {
  186. row := s.db.QueryRowContext(ctx, `
  187. SELECT id, template_id, manifest_version, source, language_used_discovery, discovery_payload_json,
  188. discovery_response_json, flattened_manifest_json, is_active, created_at, updated_at
  189. FROM qc_template_manifests
  190. WHERE template_id = ? AND is_active = 1
  191. ORDER BY created_at DESC
  192. LIMIT 1`, templateID)
  193. manifest, err := scanManifest(row.Scan)
  194. if err != nil {
  195. return nil, err
  196. }
  197. return manifest, nil
  198. }
  199. func (s *Store) ListFieldsByManifestID(ctx context.Context, manifestID string) ([]domain.TemplateField, error) {
  200. rows, err := s.db.QueryContext(ctx, `
  201. SELECT id, template_id, manifest_id, section, website_section, key_name, path, field_kind, sample_value,
  202. is_enabled, is_required_by_us, display_label, display_order, notes
  203. FROM qc_template_fields
  204. WHERE manifest_id = ?
  205. ORDER BY display_order ASC, id ASC`, manifestID)
  206. if err != nil {
  207. return nil, err
  208. }
  209. defer rows.Close()
  210. fields := make([]domain.TemplateField, 0)
  211. for rows.Next() {
  212. var f domain.TemplateField
  213. var isEnabled, isRequired int
  214. if err := rows.Scan(
  215. &f.ID, &f.TemplateID, &f.ManifestID, &f.Section, &f.WebsiteSection, &f.KeyName, &f.Path, &f.FieldKind, &f.SampleValue,
  216. &isEnabled, &isRequired, &f.DisplayLabel, &f.DisplayOrder, &f.Notes,
  217. ); err != nil {
  218. return nil, err
  219. }
  220. f.IsEnabled = isEnabled == 1
  221. f.IsRequiredByUs = isRequired == 1
  222. f.WebsiteSection = domain.NormalizeWebsiteSection(f.WebsiteSection)
  223. fields = append(fields, f)
  224. }
  225. if err := rows.Err(); err != nil {
  226. return nil, err
  227. }
  228. if len(fields) == 0 {
  229. return nil, store.ErrNotFound
  230. }
  231. return fields, nil
  232. }
  233. func (s *Store) UpdateFields(ctx context.Context, manifestID string, fields []domain.TemplateField) error {
  234. tx, err := s.db.BeginTx(ctx, nil)
  235. if err != nil {
  236. return err
  237. }
  238. defer rollback(tx)
  239. var exists int
  240. if err := tx.QueryRowContext(ctx, `SELECT COUNT(1) FROM qc_template_manifests WHERE id = ?`, manifestID).Scan(&exists); err != nil {
  241. return err
  242. }
  243. if exists == 0 {
  244. return store.ErrNotFound
  245. }
  246. if _, err := tx.ExecContext(ctx, `DELETE FROM qc_template_fields WHERE manifest_id = ?`, manifestID); err != nil {
  247. return err
  248. }
  249. for _, f := range fields {
  250. _, err := tx.ExecContext(ctx, `
  251. INSERT INTO qc_template_fields (
  252. id, template_id, manifest_id, section, website_section, key_name, path, field_kind,
  253. sample_value, is_enabled, is_required_by_us, display_label, display_order, notes
  254. ) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)`,
  255. f.ID, f.TemplateID, f.ManifestID, f.Section, domain.NormalizeWebsiteSection(f.WebsiteSection), f.KeyName, f.Path, f.FieldKind,
  256. f.SampleValue, boolToInt(f.IsEnabled), boolToInt(f.IsRequiredByUs), f.DisplayLabel, f.DisplayOrder, f.Notes,
  257. )
  258. if err != nil {
  259. return err
  260. }
  261. }
  262. return tx.Commit()
  263. }
  264. func (s *Store) CreateBuild(ctx context.Context, build domain.SiteBuild) error {
  265. _, err := s.db.ExecContext(ctx, `
  266. INSERT INTO site_builds (
  267. id, template_id, manifest_id, request_name, global_data_json, ai_data_json, final_sites_payload_json,
  268. qc_job_id, qc_site_id, qc_status, qc_preview_url, qc_editor_url, qc_result_json, qc_error_json, started_at, finished_at
  269. ) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)`,
  270. build.ID, build.TemplateID, build.ManifestID, build.RequestName, asRaw(build.GlobalDataJSON), asRaw(build.AIDataJSON), asRaw(build.FinalSitesPayload),
  271. build.QCJobID, build.QCSiteID, build.QCStatus, build.QCPreviewURL, build.QCEditorURL, asRaw(build.QCResultJSON), asRaw(build.QCErrorJSON),
  272. asRFC3339Ptr(build.StartedAt), asRFC3339Ptr(build.FinishedAt),
  273. )
  274. return err
  275. }
  276. func (s *Store) GetBuildByID(ctx context.Context, id string) (*domain.SiteBuild, error) {
  277. row := s.db.QueryRowContext(ctx, `
  278. SELECT id, template_id, manifest_id, request_name, global_data_json, ai_data_json, final_sites_payload_json,
  279. qc_job_id, qc_site_id, qc_status, qc_preview_url, qc_editor_url, qc_result_json, qc_error_json, started_at, finished_at
  280. FROM site_builds
  281. WHERE id = ?`, id)
  282. build, err := scanBuild(row.Scan)
  283. if err != nil {
  284. return nil, err
  285. }
  286. return build, nil
  287. }
  288. func (s *Store) ListBuildsByStatuses(ctx context.Context, statuses []string, limit int) ([]domain.SiteBuild, error) {
  289. base := `
  290. SELECT id, template_id, manifest_id, request_name, global_data_json, ai_data_json, final_sites_payload_json,
  291. qc_job_id, qc_site_id, qc_status, qc_preview_url, qc_editor_url, qc_result_json, qc_error_json, started_at, finished_at
  292. FROM site_builds`
  293. args := make([]any, 0)
  294. parts := make([]string, 0)
  295. if len(statuses) > 0 {
  296. placeholders := make([]string, 0, len(statuses))
  297. for _, status := range statuses {
  298. placeholders = append(placeholders, "?")
  299. args = append(args, status)
  300. }
  301. parts = append(parts, "qc_status IN ("+strings.Join(placeholders, ", ")+")")
  302. }
  303. query := base
  304. if len(parts) > 0 {
  305. query += " WHERE " + strings.Join(parts, " AND ")
  306. }
  307. query += " ORDER BY started_at ASC, id ASC"
  308. if limit > 0 {
  309. query += " LIMIT ?"
  310. args = append(args, limit)
  311. }
  312. rows, err := s.db.QueryContext(ctx, query, args...)
  313. if err != nil {
  314. return nil, err
  315. }
  316. defer rows.Close()
  317. builds := make([]domain.SiteBuild, 0)
  318. for rows.Next() {
  319. build, err := scanBuild(rows.Scan)
  320. if err != nil {
  321. return nil, err
  322. }
  323. builds = append(builds, *build)
  324. }
  325. return builds, rows.Err()
  326. }
  327. func (s *Store) MarkBuildSubmitted(ctx context.Context, buildID string, jobID int64, status string, qcResult json.RawMessage, startedAt time.Time) error {
  328. res, err := s.db.ExecContext(ctx, `
  329. UPDATE site_builds
  330. SET qc_job_id = ?, qc_status = ?, qc_result_json = ?, started_at = ?
  331. WHERE id = ?`,
  332. jobID, status, asRaw(qcResult), startedAt.UTC().Format(time.RFC3339Nano), buildID,
  333. )
  334. if err != nil {
  335. return err
  336. }
  337. n, _ := res.RowsAffected()
  338. if n == 0 {
  339. return store.ErrNotFound
  340. }
  341. return nil
  342. }
  343. func (s *Store) UpdateBuildFromJob(ctx context.Context, buildID string, status string, siteID *int64, previewURL string, qcResult json.RawMessage, qcError json.RawMessage, finishedAt *time.Time) error {
  344. res, err := s.db.ExecContext(ctx, `
  345. UPDATE site_builds
  346. SET qc_status = ?, qc_site_id = COALESCE(?, qc_site_id), qc_preview_url = ?, qc_result_json = ?, qc_error_json = ?, finished_at = ?
  347. WHERE id = ?`,
  348. status, siteID, previewURL, asRaw(qcResult), asRaw(qcError), asRFC3339Ptr(finishedAt), buildID,
  349. )
  350. if err != nil {
  351. return err
  352. }
  353. n, _ := res.RowsAffected()
  354. if n == 0 {
  355. return store.ErrNotFound
  356. }
  357. return nil
  358. }
  359. func (s *Store) UpdateBuildEditorURL(ctx context.Context, buildID string, editorURL string, qcResult json.RawMessage) error {
  360. res, err := s.db.ExecContext(ctx, `
  361. UPDATE site_builds
  362. SET qc_editor_url = ?, qc_result_json = ?
  363. WHERE id = ?`,
  364. editorURL, asRaw(qcResult), buildID,
  365. )
  366. if err != nil {
  367. return err
  368. }
  369. n, _ := res.RowsAffected()
  370. if n == 0 {
  371. return store.ErrNotFound
  372. }
  373. return nil
  374. }
  375. func (s *Store) UpsertSettings(ctx context.Context, settings domain.AppSettings) error {
  376. promptBlocksRaw, err := json.Marshal(domain.NormalizePromptBlocks(settings.PromptBlocks))
  377. if err != nil {
  378. return fmt.Errorf("marshal prompt blocks: %w", err)
  379. }
  380. provider := domain.NormalizeLLMProvider(settings.LLMActiveProvider)
  381. model := domain.NormalizeLLMModel(provider, settings.LLMActiveModel)
  382. _, err = s.db.ExecContext(ctx, `
  383. INSERT INTO app_settings (
  384. id, qc_base_url, qc_bearer_token_encrypted, language_output_mode, job_poll_interval_seconds, job_poll_timeout_seconds,
  385. llm_active_provider, llm_active_model, llm_base_url,
  386. openai_api_key_encrypted, anthropic_api_key_encrypted, google_api_key_encrypted, xai_api_key_encrypted, ollama_api_key_encrypted,
  387. master_prompt, prompt_blocks_json, updated_at
  388. ) VALUES (1, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
  389. ON CONFLICT(id) DO UPDATE SET
  390. qc_base_url = excluded.qc_base_url,
  391. qc_bearer_token_encrypted = excluded.qc_bearer_token_encrypted,
  392. language_output_mode = excluded.language_output_mode,
  393. job_poll_interval_seconds = excluded.job_poll_interval_seconds,
  394. job_poll_timeout_seconds = excluded.job_poll_timeout_seconds,
  395. llm_active_provider = excluded.llm_active_provider,
  396. llm_active_model = excluded.llm_active_model,
  397. llm_base_url = excluded.llm_base_url,
  398. openai_api_key_encrypted = excluded.openai_api_key_encrypted,
  399. anthropic_api_key_encrypted = excluded.anthropic_api_key_encrypted,
  400. google_api_key_encrypted = excluded.google_api_key_encrypted,
  401. xai_api_key_encrypted = excluded.xai_api_key_encrypted,
  402. ollama_api_key_encrypted = excluded.ollama_api_key_encrypted,
  403. master_prompt = excluded.master_prompt,
  404. prompt_blocks_json = excluded.prompt_blocks_json,
  405. updated_at = excluded.updated_at`,
  406. settings.QCBaseURL,
  407. settings.QCBearerTokenEncrypted,
  408. defaultString(settings.LanguageOutputMode, "EN"),
  409. settings.JobPollIntervalSeconds,
  410. settings.JobPollTimeoutSeconds,
  411. provider,
  412. model,
  413. strings.TrimSpace(settings.LLMBaseURL),
  414. strings.TrimSpace(settings.OpenAIAPIKeyEncrypted),
  415. strings.TrimSpace(settings.AnthropicAPIKeyEncrypted),
  416. strings.TrimSpace(settings.GoogleAPIKeyEncrypted),
  417. strings.TrimSpace(settings.XAIAPIKeyEncrypted),
  418. strings.TrimSpace(settings.OllamaAPIKeyEncrypted),
  419. domain.NormalizeMasterPrompt(settings.MasterPrompt),
  420. promptBlocksRaw,
  421. time.Now().UTC().Format(time.RFC3339Nano),
  422. )
  423. return err
  424. }
  425. func (s *Store) GetSettings(ctx context.Context) (*domain.AppSettings, error) {
  426. row := s.db.QueryRowContext(ctx, `
  427. SELECT qc_base_url, qc_bearer_token_encrypted, language_output_mode, job_poll_interval_seconds, job_poll_timeout_seconds,
  428. llm_active_provider, llm_active_model, llm_base_url,
  429. openai_api_key_encrypted, anthropic_api_key_encrypted, google_api_key_encrypted, xai_api_key_encrypted, ollama_api_key_encrypted,
  430. master_prompt, prompt_blocks_json
  431. FROM app_settings
  432. WHERE id = 1`)
  433. var settings domain.AppSettings
  434. var promptBlocksRaw []byte
  435. if err := row.Scan(
  436. &settings.QCBaseURL,
  437. &settings.QCBearerTokenEncrypted,
  438. &settings.LanguageOutputMode,
  439. &settings.JobPollIntervalSeconds,
  440. &settings.JobPollTimeoutSeconds,
  441. &settings.LLMActiveProvider,
  442. &settings.LLMActiveModel,
  443. &settings.LLMBaseURL,
  444. &settings.OpenAIAPIKeyEncrypted,
  445. &settings.AnthropicAPIKeyEncrypted,
  446. &settings.GoogleAPIKeyEncrypted,
  447. &settings.XAIAPIKeyEncrypted,
  448. &settings.OllamaAPIKeyEncrypted,
  449. &settings.MasterPrompt,
  450. &promptBlocksRaw,
  451. ); err != nil {
  452. if err == sql.ErrNoRows {
  453. return nil, store.ErrNotFound
  454. }
  455. return nil, err
  456. }
  457. settings.MasterPrompt = domain.NormalizeMasterPrompt(settings.MasterPrompt)
  458. if len(promptBlocksRaw) > 0 {
  459. _ = json.Unmarshal(promptBlocksRaw, &settings.PromptBlocks)
  460. }
  461. settings.LLMActiveProvider = domain.NormalizeLLMProvider(settings.LLMActiveProvider)
  462. settings.LLMActiveModel = domain.NormalizeLLMModel(settings.LLMActiveProvider, settings.LLMActiveModel)
  463. settings.LLMBaseURL = strings.TrimSpace(settings.LLMBaseURL)
  464. settings.OpenAIAPIKeyEncrypted = strings.TrimSpace(settings.OpenAIAPIKeyEncrypted)
  465. settings.AnthropicAPIKeyEncrypted = strings.TrimSpace(settings.AnthropicAPIKeyEncrypted)
  466. settings.GoogleAPIKeyEncrypted = strings.TrimSpace(settings.GoogleAPIKeyEncrypted)
  467. settings.XAIAPIKeyEncrypted = strings.TrimSpace(settings.XAIAPIKeyEncrypted)
  468. settings.OllamaAPIKeyEncrypted = strings.TrimSpace(settings.OllamaAPIKeyEncrypted)
  469. settings.PromptBlocks = domain.NormalizePromptBlocks(settings.PromptBlocks)
  470. return &settings, nil
  471. }
  472. func (s *Store) CreateDraft(ctx context.Context, draft domain.BuildDraft) error {
  473. _, err := s.db.ExecContext(ctx, `
  474. INSERT INTO build_drafts (
  475. id, template_id, manifest_id, source, request_name, global_data_json,
  476. field_values_json, draft_context_json, suggestion_state_json, status, notes, created_at, updated_at
  477. ) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)`,
  478. draft.ID, nullableInt64(draft.TemplateID), draft.ManifestID, draft.Source, draft.RequestName, asRaw(draft.GlobalDataJSON),
  479. asRaw(draft.FieldValuesJSON), asRaw(draft.DraftContextJSON), asRaw(draft.SuggestionStateJSON), draft.Status, draft.Notes, draft.CreatedAt.UTC().Format(time.RFC3339Nano), draft.UpdatedAt.UTC().Format(time.RFC3339Nano),
  480. )
  481. return err
  482. }
  483. func (s *Store) UpdateDraft(ctx context.Context, draft domain.BuildDraft) error {
  484. res, err := s.db.ExecContext(ctx, `
  485. UPDATE build_drafts
  486. SET template_id = ?, manifest_id = ?, source = ?, request_name = ?, global_data_json = ?, field_values_json = ?, draft_context_json = ?, suggestion_state_json = ?,
  487. status = ?, notes = ?, updated_at = ?
  488. WHERE id = ?`,
  489. nullableInt64(draft.TemplateID), draft.ManifestID, draft.Source, draft.RequestName, asRaw(draft.GlobalDataJSON), asRaw(draft.FieldValuesJSON), asRaw(draft.DraftContextJSON), asRaw(draft.SuggestionStateJSON),
  490. draft.Status, draft.Notes, draft.UpdatedAt.UTC().Format(time.RFC3339Nano), draft.ID,
  491. )
  492. if err != nil {
  493. return err
  494. }
  495. n, _ := res.RowsAffected()
  496. if n == 0 {
  497. return store.ErrNotFound
  498. }
  499. return nil
  500. }
  501. func (s *Store) GetDraftByID(ctx context.Context, id string) (*domain.BuildDraft, error) {
  502. row := s.db.QueryRowContext(ctx, `
  503. SELECT id, template_id, manifest_id, source, request_name, global_data_json, field_values_json, draft_context_json, suggestion_state_json, status, notes, created_at, updated_at
  504. FROM build_drafts
  505. WHERE id = ?`, id)
  506. return scanDraft(row.Scan)
  507. }
  508. func (s *Store) ListDrafts(ctx context.Context, limit int) ([]domain.BuildDraft, error) {
  509. query := `
  510. SELECT id, template_id, manifest_id, source, request_name, global_data_json, field_values_json, draft_context_json, suggestion_state_json, status, notes, created_at, updated_at
  511. FROM build_drafts
  512. ORDER BY updated_at DESC`
  513. args := make([]any, 0, 1)
  514. if limit > 0 {
  515. query += " LIMIT ?"
  516. args = append(args, limit)
  517. }
  518. rows, err := s.db.QueryContext(ctx, query, args...)
  519. if err != nil {
  520. return nil, err
  521. }
  522. defer rows.Close()
  523. out := make([]domain.BuildDraft, 0)
  524. for rows.Next() {
  525. draft, err := scanDraft(rows.Scan)
  526. if err != nil {
  527. return nil, err
  528. }
  529. out = append(out, *draft)
  530. }
  531. return out, rows.Err()
  532. }
  533. func runMigrations(db *sql.DB) error {
  534. if _, err := db.Exec(`
  535. CREATE TABLE IF NOT EXISTS schema_migrations (
  536. version TEXT PRIMARY KEY,
  537. applied_at TEXT NOT NULL
  538. )`); err != nil {
  539. return err
  540. }
  541. entries, err := fs.ReadDir(migrationFS, "migrations")
  542. if err != nil {
  543. return err
  544. }
  545. files := make([]string, 0, len(entries))
  546. for _, e := range entries {
  547. if e.IsDir() || !strings.HasSuffix(e.Name(), ".sql") {
  548. continue
  549. }
  550. files = append(files, e.Name())
  551. }
  552. sort.Strings(files)
  553. for _, name := range files {
  554. var exists int
  555. if err := db.QueryRow(`SELECT COUNT(1) FROM schema_migrations WHERE version = ?`, name).Scan(&exists); err != nil {
  556. return err
  557. }
  558. if exists > 0 {
  559. continue
  560. }
  561. raw, err := migrationFS.ReadFile("migrations/" + name)
  562. if err != nil {
  563. return err
  564. }
  565. tx, err := db.Begin()
  566. if err != nil {
  567. return err
  568. }
  569. if _, err := tx.Exec(string(raw)); err != nil {
  570. _ = tx.Rollback()
  571. return fmt.Errorf("apply %s: %w", name, err)
  572. }
  573. if _, err := tx.Exec(`INSERT INTO schema_migrations(version, applied_at) VALUES(?, ?)`, name, time.Now().UTC().Format(time.RFC3339Nano)); err != nil {
  574. _ = tx.Rollback()
  575. return err
  576. }
  577. if err := tx.Commit(); err != nil {
  578. return err
  579. }
  580. }
  581. return nil
  582. }
  583. func scanTemplate(scan func(dest ...any) error) (*domain.Template, error) {
  584. var t domain.Template
  585. var paletteReady int
  586. var isAITemplate int
  587. var isOnboarded int
  588. var lastDiscovered sql.NullString
  589. var raw []byte
  590. if err := scan(
  591. &t.ID, &t.Name, &t.Description, &t.Locale, &t.ThumbnailURL, &t.TemplatePreviewURL, &t.Type,
  592. &paletteReady, &raw, &isAITemplate, &isOnboarded, &t.ManifestStatus, &lastDiscovered,
  593. ); err != nil {
  594. if err == sql.ErrNoRows {
  595. return nil, store.ErrNotFound
  596. }
  597. return nil, err
  598. }
  599. t.PaletteReady = paletteReady == 1
  600. t.IsAITemplate = isAITemplate == 1
  601. t.IsOnboarded = isOnboarded == 1
  602. t.RawJSON = cloneBytes(raw)
  603. if lastDiscovered.Valid {
  604. if ts, err := time.Parse(time.RFC3339Nano, lastDiscovered.String); err == nil {
  605. t.LastDiscoveredAt = &ts
  606. }
  607. }
  608. return &t, nil
  609. }
  610. func scanManifest(scan func(dest ...any) error) (*domain.TemplateManifest, error) {
  611. var m domain.TemplateManifest
  612. var isActive int
  613. var payloadRaw []byte
  614. var responseRaw []byte
  615. var flattenedRaw []byte
  616. var createdAtRaw string
  617. var updatedAtRaw string
  618. if err := scan(
  619. &m.ID, &m.TemplateID, &m.Version, &m.Source, &m.LanguageUsedDiscovery, &payloadRaw,
  620. &responseRaw, &flattenedRaw, &isActive, &createdAtRaw, &updatedAtRaw,
  621. ); err != nil {
  622. if err == sql.ErrNoRows {
  623. return nil, store.ErrNotFound
  624. }
  625. return nil, err
  626. }
  627. m.IsActive = isActive == 1
  628. m.DiscoveryPayloadJSON = cloneBytes(payloadRaw)
  629. m.DiscoveryResponseJSON = cloneBytes(responseRaw)
  630. m.FlattenedManifestJSON = cloneBytes(flattenedRaw)
  631. m.CreatedAt, _ = time.Parse(time.RFC3339Nano, createdAtRaw)
  632. m.UpdatedAt, _ = time.Parse(time.RFC3339Nano, updatedAtRaw)
  633. return &m, nil
  634. }
  635. func scanBuild(scan func(dest ...any) error) (*domain.SiteBuild, error) {
  636. var b domain.SiteBuild
  637. var globalRaw []byte
  638. var aiDataRaw []byte
  639. var payloadRaw []byte
  640. var resultRaw []byte
  641. var errorRaw []byte
  642. var startedAtRaw sql.NullString
  643. var finishedAtRaw sql.NullString
  644. var jobID sql.NullInt64
  645. var siteID sql.NullInt64
  646. if err := scan(
  647. &b.ID, &b.TemplateID, &b.ManifestID, &b.RequestName, &globalRaw, &aiDataRaw, &payloadRaw,
  648. &jobID, &siteID, &b.QCStatus, &b.QCPreviewURL, &b.QCEditorURL, &resultRaw, &errorRaw, &startedAtRaw, &finishedAtRaw,
  649. ); err != nil {
  650. if err == sql.ErrNoRows {
  651. return nil, store.ErrNotFound
  652. }
  653. return nil, err
  654. }
  655. b.GlobalDataJSON = cloneBytes(globalRaw)
  656. b.AIDataJSON = cloneBytes(aiDataRaw)
  657. b.FinalSitesPayload = cloneBytes(payloadRaw)
  658. b.QCResultJSON = cloneBytes(resultRaw)
  659. b.QCErrorJSON = cloneBytes(errorRaw)
  660. if jobID.Valid {
  661. id := jobID.Int64
  662. b.QCJobID = &id
  663. }
  664. if siteID.Valid {
  665. id := siteID.Int64
  666. b.QCSiteID = &id
  667. }
  668. b.StartedAt = parseTimePtr(startedAtRaw)
  669. b.FinishedAt = parseTimePtr(finishedAtRaw)
  670. return &b, nil
  671. }
  672. func scanDraft(scan func(dest ...any) error) (*domain.BuildDraft, error) {
  673. var d domain.BuildDraft
  674. var templateID sql.NullInt64
  675. var globalRaw []byte
  676. var fieldsRaw []byte
  677. var draftContextRaw []byte
  678. var suggestionStateRaw []byte
  679. var createdAtRaw string
  680. var updatedAtRaw string
  681. if err := scan(
  682. &d.ID, &templateID, &d.ManifestID, &d.Source, &d.RequestName, &globalRaw, &fieldsRaw, &draftContextRaw, &suggestionStateRaw, &d.Status, &d.Notes, &createdAtRaw, &updatedAtRaw,
  683. ); err != nil {
  684. if err == sql.ErrNoRows {
  685. return nil, store.ErrNotFound
  686. }
  687. return nil, err
  688. }
  689. if templateID.Valid {
  690. d.TemplateID = templateID.Int64
  691. }
  692. d.GlobalDataJSON = cloneBytes(globalRaw)
  693. d.FieldValuesJSON = cloneBytes(fieldsRaw)
  694. d.DraftContextJSON = cloneBytes(draftContextRaw)
  695. d.SuggestionStateJSON = cloneBytes(suggestionStateRaw)
  696. d.CreatedAt, _ = time.Parse(time.RFC3339Nano, createdAtRaw)
  697. d.UpdatedAt, _ = time.Parse(time.RFC3339Nano, updatedAtRaw)
  698. return &d, nil
  699. }
  700. func rollback(tx *sql.Tx) {
  701. _ = tx.Rollback()
  702. }
  703. func boolToInt(v bool) int {
  704. if v {
  705. return 1
  706. }
  707. return 0
  708. }
  709. func defaultString(value, fallback string) string {
  710. if strings.TrimSpace(value) == "" {
  711. return fallback
  712. }
  713. return strings.TrimSpace(value)
  714. }
  715. func asRaw(raw json.RawMessage) []byte {
  716. if len(raw) == 0 {
  717. return nil
  718. }
  719. out := make([]byte, len(raw))
  720. copy(out, raw)
  721. return out
  722. }
  723. func cloneBytes(raw []byte) []byte {
  724. if len(raw) == 0 {
  725. return nil
  726. }
  727. out := make([]byte, len(raw))
  728. copy(out, raw)
  729. return out
  730. }
  731. func asRFC3339Ptr(t *time.Time) *string {
  732. if t == nil {
  733. return nil
  734. }
  735. v := t.UTC().Format(time.RFC3339Nano)
  736. return &v
  737. }
  738. func parseTimePtr(value sql.NullString) *time.Time {
  739. if !value.Valid {
  740. return nil
  741. }
  742. ts, err := time.Parse(time.RFC3339Nano, value.String)
  743. if err != nil {
  744. return nil
  745. }
  746. return &ts
  747. }
  748. func nullableInt64(value int64) any {
  749. if value <= 0 {
  750. return nil
  751. }
  752. return value
  753. }