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

751 строка
22KB

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