Ви не можете вибрати більше 25 тем Теми мають розпочинатися з літери або цифри, можуть містити дефіси (-) і не повинні перевищувати 35 символів.

817 рядки
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, llm_temperature, llm_max_tokens,
  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. llm_temperature = excluded.llm_temperature,
  399. llm_max_tokens = excluded.llm_max_tokens,
  400. openai_api_key_encrypted = excluded.openai_api_key_encrypted,
  401. anthropic_api_key_encrypted = excluded.anthropic_api_key_encrypted,
  402. google_api_key_encrypted = excluded.google_api_key_encrypted,
  403. xai_api_key_encrypted = excluded.xai_api_key_encrypted,
  404. ollama_api_key_encrypted = excluded.ollama_api_key_encrypted,
  405. master_prompt = excluded.master_prompt,
  406. prompt_blocks_json = excluded.prompt_blocks_json,
  407. updated_at = excluded.updated_at`,
  408. settings.QCBaseURL,
  409. settings.QCBearerTokenEncrypted,
  410. defaultString(settings.LanguageOutputMode, "EN"),
  411. settings.JobPollIntervalSeconds,
  412. settings.JobPollTimeoutSeconds,
  413. provider,
  414. model,
  415. strings.TrimSpace(settings.LLMBaseURL),
  416. domain.NormalizeLLMTemperature(settings.LLMTemperature),
  417. domain.NormalizeLLMMaxTokens(settings.LLMMaxTokens),
  418. strings.TrimSpace(settings.OpenAIAPIKeyEncrypted),
  419. strings.TrimSpace(settings.AnthropicAPIKeyEncrypted),
  420. strings.TrimSpace(settings.GoogleAPIKeyEncrypted),
  421. strings.TrimSpace(settings.XAIAPIKeyEncrypted),
  422. strings.TrimSpace(settings.OllamaAPIKeyEncrypted),
  423. domain.NormalizeMasterPrompt(settings.MasterPrompt),
  424. promptBlocksRaw,
  425. time.Now().UTC().Format(time.RFC3339Nano),
  426. )
  427. return err
  428. }
  429. func (s *Store) GetSettings(ctx context.Context) (*domain.AppSettings, error) {
  430. row := s.db.QueryRowContext(ctx, `
  431. SELECT qc_base_url, qc_bearer_token_encrypted, language_output_mode, job_poll_interval_seconds, job_poll_timeout_seconds,
  432. llm_active_provider, llm_active_model, llm_base_url, llm_temperature, llm_max_tokens,
  433. openai_api_key_encrypted, anthropic_api_key_encrypted, google_api_key_encrypted, xai_api_key_encrypted, ollama_api_key_encrypted,
  434. master_prompt, prompt_blocks_json
  435. FROM app_settings
  436. WHERE id = 1`)
  437. var settings domain.AppSettings
  438. var promptBlocksRaw []byte
  439. if err := row.Scan(
  440. &settings.QCBaseURL,
  441. &settings.QCBearerTokenEncrypted,
  442. &settings.LanguageOutputMode,
  443. &settings.JobPollIntervalSeconds,
  444. &settings.JobPollTimeoutSeconds,
  445. &settings.LLMActiveProvider,
  446. &settings.LLMActiveModel,
  447. &settings.LLMBaseURL,
  448. &settings.LLMTemperature,
  449. &settings.LLMMaxTokens,
  450. &settings.OpenAIAPIKeyEncrypted,
  451. &settings.AnthropicAPIKeyEncrypted,
  452. &settings.GoogleAPIKeyEncrypted,
  453. &settings.XAIAPIKeyEncrypted,
  454. &settings.OllamaAPIKeyEncrypted,
  455. &settings.MasterPrompt,
  456. &promptBlocksRaw,
  457. ); err != nil {
  458. if err == sql.ErrNoRows {
  459. return nil, store.ErrNotFound
  460. }
  461. return nil, err
  462. }
  463. settings.MasterPrompt = domain.NormalizeMasterPrompt(settings.MasterPrompt)
  464. if len(promptBlocksRaw) > 0 {
  465. _ = json.Unmarshal(promptBlocksRaw, &settings.PromptBlocks)
  466. }
  467. settings.LLMActiveProvider = domain.NormalizeLLMProvider(settings.LLMActiveProvider)
  468. settings.LLMActiveModel = domain.NormalizeLLMModel(settings.LLMActiveProvider, settings.LLMActiveModel)
  469. settings.LLMBaseURL = strings.TrimSpace(settings.LLMBaseURL)
  470. settings.LLMTemperature = domain.NormalizeLLMTemperature(settings.LLMTemperature)
  471. settings.LLMMaxTokens = domain.NormalizeLLMMaxTokens(settings.LLMMaxTokens)
  472. settings.OpenAIAPIKeyEncrypted = strings.TrimSpace(settings.OpenAIAPIKeyEncrypted)
  473. settings.AnthropicAPIKeyEncrypted = strings.TrimSpace(settings.AnthropicAPIKeyEncrypted)
  474. settings.GoogleAPIKeyEncrypted = strings.TrimSpace(settings.GoogleAPIKeyEncrypted)
  475. settings.XAIAPIKeyEncrypted = strings.TrimSpace(settings.XAIAPIKeyEncrypted)
  476. settings.OllamaAPIKeyEncrypted = strings.TrimSpace(settings.OllamaAPIKeyEncrypted)
  477. settings.PromptBlocks = domain.NormalizePromptBlocks(settings.PromptBlocks)
  478. return &settings, nil
  479. }
  480. func (s *Store) CreateDraft(ctx context.Context, draft domain.BuildDraft) error {
  481. _, err := s.db.ExecContext(ctx, `
  482. INSERT INTO build_drafts (
  483. id, template_id, manifest_id, source, request_name, global_data_json,
  484. field_values_json, draft_context_json, suggestion_state_json, status, notes, created_at, updated_at
  485. ) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)`,
  486. draft.ID, nullableInt64(draft.TemplateID), draft.ManifestID, draft.Source, draft.RequestName, asRaw(draft.GlobalDataJSON),
  487. 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),
  488. )
  489. return err
  490. }
  491. func (s *Store) UpdateDraft(ctx context.Context, draft domain.BuildDraft) error {
  492. res, err := s.db.ExecContext(ctx, `
  493. UPDATE build_drafts
  494. SET template_id = ?, manifest_id = ?, source = ?, request_name = ?, global_data_json = ?, field_values_json = ?, draft_context_json = ?, suggestion_state_json = ?,
  495. status = ?, notes = ?, updated_at = ?
  496. WHERE id = ?`,
  497. nullableInt64(draft.TemplateID), draft.ManifestID, draft.Source, draft.RequestName, asRaw(draft.GlobalDataJSON), asRaw(draft.FieldValuesJSON), asRaw(draft.DraftContextJSON), asRaw(draft.SuggestionStateJSON),
  498. draft.Status, draft.Notes, draft.UpdatedAt.UTC().Format(time.RFC3339Nano), draft.ID,
  499. )
  500. if err != nil {
  501. return err
  502. }
  503. n, _ := res.RowsAffected()
  504. if n == 0 {
  505. return store.ErrNotFound
  506. }
  507. return nil
  508. }
  509. func (s *Store) GetDraftByID(ctx context.Context, id string) (*domain.BuildDraft, error) {
  510. row := s.db.QueryRowContext(ctx, `
  511. 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
  512. FROM build_drafts
  513. WHERE id = ?`, id)
  514. return scanDraft(row.Scan)
  515. }
  516. func (s *Store) ListDrafts(ctx context.Context, limit int) ([]domain.BuildDraft, error) {
  517. query := `
  518. 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
  519. FROM build_drafts
  520. ORDER BY updated_at DESC`
  521. args := make([]any, 0, 1)
  522. if limit > 0 {
  523. query += " LIMIT ?"
  524. args = append(args, limit)
  525. }
  526. rows, err := s.db.QueryContext(ctx, query, args...)
  527. if err != nil {
  528. return nil, err
  529. }
  530. defer rows.Close()
  531. out := make([]domain.BuildDraft, 0)
  532. for rows.Next() {
  533. draft, err := scanDraft(rows.Scan)
  534. if err != nil {
  535. return nil, err
  536. }
  537. out = append(out, *draft)
  538. }
  539. return out, rows.Err()
  540. }
  541. func runMigrations(db *sql.DB) error {
  542. if _, err := db.Exec(`
  543. CREATE TABLE IF NOT EXISTS schema_migrations (
  544. version TEXT PRIMARY KEY,
  545. applied_at TEXT NOT NULL
  546. )`); err != nil {
  547. return err
  548. }
  549. entries, err := fs.ReadDir(migrationFS, "migrations")
  550. if err != nil {
  551. return err
  552. }
  553. files := make([]string, 0, len(entries))
  554. for _, e := range entries {
  555. if e.IsDir() || !strings.HasSuffix(e.Name(), ".sql") {
  556. continue
  557. }
  558. files = append(files, e.Name())
  559. }
  560. sort.Strings(files)
  561. for _, name := range files {
  562. var exists int
  563. if err := db.QueryRow(`SELECT COUNT(1) FROM schema_migrations WHERE version = ?`, name).Scan(&exists); err != nil {
  564. return err
  565. }
  566. if exists > 0 {
  567. continue
  568. }
  569. raw, err := migrationFS.ReadFile("migrations/" + name)
  570. if err != nil {
  571. return err
  572. }
  573. tx, err := db.Begin()
  574. if err != nil {
  575. return err
  576. }
  577. if _, err := tx.Exec(string(raw)); err != nil {
  578. _ = tx.Rollback()
  579. return fmt.Errorf("apply %s: %w", name, err)
  580. }
  581. if _, err := tx.Exec(`INSERT INTO schema_migrations(version, applied_at) VALUES(?, ?)`, name, time.Now().UTC().Format(time.RFC3339Nano)); err != nil {
  582. _ = tx.Rollback()
  583. return err
  584. }
  585. if err := tx.Commit(); err != nil {
  586. return err
  587. }
  588. }
  589. return nil
  590. }
  591. func scanTemplate(scan func(dest ...any) error) (*domain.Template, error) {
  592. var t domain.Template
  593. var paletteReady int
  594. var isAITemplate int
  595. var isOnboarded int
  596. var lastDiscovered sql.NullString
  597. var raw []byte
  598. if err := scan(
  599. &t.ID, &t.Name, &t.Description, &t.Locale, &t.ThumbnailURL, &t.TemplatePreviewURL, &t.Type,
  600. &paletteReady, &raw, &isAITemplate, &isOnboarded, &t.ManifestStatus, &lastDiscovered,
  601. ); err != nil {
  602. if err == sql.ErrNoRows {
  603. return nil, store.ErrNotFound
  604. }
  605. return nil, err
  606. }
  607. t.PaletteReady = paletteReady == 1
  608. t.IsAITemplate = isAITemplate == 1
  609. t.IsOnboarded = isOnboarded == 1
  610. t.RawJSON = cloneBytes(raw)
  611. if lastDiscovered.Valid {
  612. if ts, err := time.Parse(time.RFC3339Nano, lastDiscovered.String); err == nil {
  613. t.LastDiscoveredAt = &ts
  614. }
  615. }
  616. return &t, nil
  617. }
  618. func scanManifest(scan func(dest ...any) error) (*domain.TemplateManifest, error) {
  619. var m domain.TemplateManifest
  620. var isActive int
  621. var payloadRaw []byte
  622. var responseRaw []byte
  623. var flattenedRaw []byte
  624. var createdAtRaw string
  625. var updatedAtRaw string
  626. if err := scan(
  627. &m.ID, &m.TemplateID, &m.Version, &m.Source, &m.LanguageUsedDiscovery, &payloadRaw,
  628. &responseRaw, &flattenedRaw, &isActive, &createdAtRaw, &updatedAtRaw,
  629. ); err != nil {
  630. if err == sql.ErrNoRows {
  631. return nil, store.ErrNotFound
  632. }
  633. return nil, err
  634. }
  635. m.IsActive = isActive == 1
  636. m.DiscoveryPayloadJSON = cloneBytes(payloadRaw)
  637. m.DiscoveryResponseJSON = cloneBytes(responseRaw)
  638. m.FlattenedManifestJSON = cloneBytes(flattenedRaw)
  639. m.CreatedAt, _ = time.Parse(time.RFC3339Nano, createdAtRaw)
  640. m.UpdatedAt, _ = time.Parse(time.RFC3339Nano, updatedAtRaw)
  641. return &m, nil
  642. }
  643. func scanBuild(scan func(dest ...any) error) (*domain.SiteBuild, error) {
  644. var b domain.SiteBuild
  645. var globalRaw []byte
  646. var aiDataRaw []byte
  647. var payloadRaw []byte
  648. var resultRaw []byte
  649. var errorRaw []byte
  650. var startedAtRaw sql.NullString
  651. var finishedAtRaw sql.NullString
  652. var jobID sql.NullInt64
  653. var siteID sql.NullInt64
  654. if err := scan(
  655. &b.ID, &b.TemplateID, &b.ManifestID, &b.RequestName, &globalRaw, &aiDataRaw, &payloadRaw,
  656. &jobID, &siteID, &b.QCStatus, &b.QCPreviewURL, &b.QCEditorURL, &resultRaw, &errorRaw, &startedAtRaw, &finishedAtRaw,
  657. ); err != nil {
  658. if err == sql.ErrNoRows {
  659. return nil, store.ErrNotFound
  660. }
  661. return nil, err
  662. }
  663. b.GlobalDataJSON = cloneBytes(globalRaw)
  664. b.AIDataJSON = cloneBytes(aiDataRaw)
  665. b.FinalSitesPayload = cloneBytes(payloadRaw)
  666. b.QCResultJSON = cloneBytes(resultRaw)
  667. b.QCErrorJSON = cloneBytes(errorRaw)
  668. if jobID.Valid {
  669. id := jobID.Int64
  670. b.QCJobID = &id
  671. }
  672. if siteID.Valid {
  673. id := siteID.Int64
  674. b.QCSiteID = &id
  675. }
  676. b.StartedAt = parseTimePtr(startedAtRaw)
  677. b.FinishedAt = parseTimePtr(finishedAtRaw)
  678. return &b, nil
  679. }
  680. func scanDraft(scan func(dest ...any) error) (*domain.BuildDraft, error) {
  681. var d domain.BuildDraft
  682. var templateID sql.NullInt64
  683. var globalRaw []byte
  684. var fieldsRaw []byte
  685. var draftContextRaw []byte
  686. var suggestionStateRaw []byte
  687. var createdAtRaw string
  688. var updatedAtRaw string
  689. if err := scan(
  690. &d.ID, &templateID, &d.ManifestID, &d.Source, &d.RequestName, &globalRaw, &fieldsRaw, &draftContextRaw, &suggestionStateRaw, &d.Status, &d.Notes, &createdAtRaw, &updatedAtRaw,
  691. ); err != nil {
  692. if err == sql.ErrNoRows {
  693. return nil, store.ErrNotFound
  694. }
  695. return nil, err
  696. }
  697. if templateID.Valid {
  698. d.TemplateID = templateID.Int64
  699. }
  700. d.GlobalDataJSON = cloneBytes(globalRaw)
  701. d.FieldValuesJSON = cloneBytes(fieldsRaw)
  702. d.DraftContextJSON = cloneBytes(draftContextRaw)
  703. d.SuggestionStateJSON = cloneBytes(suggestionStateRaw)
  704. d.CreatedAt, _ = time.Parse(time.RFC3339Nano, createdAtRaw)
  705. d.UpdatedAt, _ = time.Parse(time.RFC3339Nano, updatedAtRaw)
  706. return &d, nil
  707. }
  708. func rollback(tx *sql.Tx) {
  709. _ = tx.Rollback()
  710. }
  711. func boolToInt(v bool) int {
  712. if v {
  713. return 1
  714. }
  715. return 0
  716. }
  717. func defaultString(value, fallback string) string {
  718. if strings.TrimSpace(value) == "" {
  719. return fallback
  720. }
  721. return strings.TrimSpace(value)
  722. }
  723. func asRaw(raw json.RawMessage) []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 cloneBytes(raw []byte) []byte {
  732. if len(raw) == 0 {
  733. return nil
  734. }
  735. out := make([]byte, len(raw))
  736. copy(out, raw)
  737. return out
  738. }
  739. func asRFC3339Ptr(t *time.Time) *string {
  740. if t == nil {
  741. return nil
  742. }
  743. v := t.UTC().Format(time.RFC3339Nano)
  744. return &v
  745. }
  746. func parseTimePtr(value sql.NullString) *time.Time {
  747. if !value.Valid {
  748. return nil
  749. }
  750. ts, err := time.Parse(time.RFC3339Nano, value.String)
  751. if err != nil {
  752. return nil
  753. }
  754. return &ts
  755. }
  756. func nullableInt64(value int64) any {
  757. if value <= 0 {
  758. return nil
  759. }
  760. return value
  761. }