dev_store.go 9.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324
  1. package postgres
  2. import (
  3. "context"
  4. "fmt"
  5. )
  6. type DemoBootstrapSummary struct {
  7. TenantCode string `json:"tenantCode"`
  8. ChannelCode string `json:"channelCode"`
  9. EventID string `json:"eventId"`
  10. ReleaseID string `json:"releaseId"`
  11. SourceID string `json:"sourceId"`
  12. BuildID string `json:"buildId"`
  13. CardID string `json:"cardId"`
  14. }
  15. func (s *Store) EnsureDemoData(ctx context.Context) (*DemoBootstrapSummary, error) {
  16. tx, err := s.Begin(ctx)
  17. if err != nil {
  18. return nil, err
  19. }
  20. defer tx.Rollback(ctx)
  21. var tenantID string
  22. if err := tx.QueryRow(ctx, `
  23. INSERT INTO tenants (tenant_code, name, status)
  24. VALUES ('tenant_demo', 'Demo Tenant', 'active')
  25. ON CONFLICT (tenant_code) DO UPDATE SET
  26. name = EXCLUDED.name,
  27. status = EXCLUDED.status
  28. RETURNING id
  29. `).Scan(&tenantID); err != nil {
  30. return nil, fmt.Errorf("ensure demo tenant: %w", err)
  31. }
  32. var channelID string
  33. if err := tx.QueryRow(ctx, `
  34. INSERT INTO entry_channels (
  35. tenant_id, channel_code, channel_type, platform_app_id, display_name, status, is_default
  36. )
  37. VALUES ($1, 'mini-demo', 'wechat_mini', 'wx-demo-appid', 'Demo Mini Channel', 'active', true)
  38. ON CONFLICT (tenant_id, channel_code) DO UPDATE SET
  39. channel_type = EXCLUDED.channel_type,
  40. platform_app_id = EXCLUDED.platform_app_id,
  41. display_name = EXCLUDED.display_name,
  42. status = EXCLUDED.status,
  43. is_default = EXCLUDED.is_default
  44. RETURNING id
  45. `, tenantID).Scan(&channelID); err != nil {
  46. return nil, fmt.Errorf("ensure demo entry channel: %w", err)
  47. }
  48. var eventID string
  49. if err := tx.QueryRow(ctx, `
  50. INSERT INTO events (
  51. tenant_id, event_public_id, slug, display_name, summary, status
  52. )
  53. VALUES ($1, 'evt_demo_001', 'demo-city-run', 'Demo City Run', 'Launch flow demo event', 'active')
  54. ON CONFLICT (event_public_id) DO UPDATE SET
  55. tenant_id = EXCLUDED.tenant_id,
  56. slug = EXCLUDED.slug,
  57. display_name = EXCLUDED.display_name,
  58. summary = EXCLUDED.summary,
  59. status = EXCLUDED.status
  60. RETURNING id
  61. `, tenantID).Scan(&eventID); err != nil {
  62. return nil, fmt.Errorf("ensure demo event: %w", err)
  63. }
  64. var releaseRow struct {
  65. ID string
  66. PublicID string
  67. }
  68. if err := tx.QueryRow(ctx, `
  69. INSERT INTO event_releases (
  70. release_public_id,
  71. event_id,
  72. release_no,
  73. config_label,
  74. manifest_url,
  75. manifest_checksum_sha256,
  76. route_code,
  77. status
  78. )
  79. VALUES (
  80. 'rel_demo_001',
  81. $1,
  82. 1,
  83. 'Demo Config v1',
  84. 'https://oss-mbh5.colormaprun.com/gotomars/event/classic-sequential.json',
  85. 'demo-checksum-001',
  86. 'route-demo-001',
  87. 'published'
  88. )
  89. ON CONFLICT (release_public_id) DO UPDATE SET
  90. event_id = EXCLUDED.event_id,
  91. config_label = EXCLUDED.config_label,
  92. manifest_url = EXCLUDED.manifest_url,
  93. manifest_checksum_sha256 = EXCLUDED.manifest_checksum_sha256,
  94. route_code = EXCLUDED.route_code,
  95. status = EXCLUDED.status
  96. RETURNING id, release_public_id
  97. `, eventID).Scan(&releaseRow.ID, &releaseRow.PublicID); err != nil {
  98. return nil, fmt.Errorf("ensure demo release: %w", err)
  99. }
  100. if _, err := tx.Exec(ctx, `
  101. UPDATE events
  102. SET current_release_id = $2
  103. WHERE id = $1
  104. `, eventID, releaseRow.ID); err != nil {
  105. return nil, fmt.Errorf("attach demo release: %w", err)
  106. }
  107. sourceNotes := "demo source config imported from local event sample"
  108. source, err := s.UpsertEventConfigSource(ctx, tx, UpsertEventConfigSourceParams{
  109. EventID: eventID,
  110. SourceVersionNo: 1,
  111. SourceKind: "event_bundle",
  112. SchemaID: "event-source",
  113. SchemaVersion: "1",
  114. Status: "active",
  115. Notes: &sourceNotes,
  116. Source: map[string]any{
  117. "app": map[string]any{
  118. "id": "sample-classic-001",
  119. "title": "顺序赛示例",
  120. },
  121. "branding": map[string]any{
  122. "tenantCode": "tenant_demo",
  123. "entryChannel": "mini-demo",
  124. },
  125. "map": map[string]any{
  126. "tiles": "../map/lxcb-001/tiles/",
  127. "mapmeta": "../map/lxcb-001/tiles/meta.json",
  128. },
  129. "playfield": map[string]any{
  130. "kind": "course",
  131. "source": map[string]any{
  132. "type": "kml",
  133. "url": "../kml/lxcb-001/10/c01.kml",
  134. },
  135. },
  136. "game": map[string]any{
  137. "mode": "classic-sequential",
  138. },
  139. "content": map[string]any{
  140. "h5Template": "content-h5-test-template.html",
  141. },
  142. },
  143. })
  144. if err != nil {
  145. return nil, fmt.Errorf("ensure demo event config source: %w", err)
  146. }
  147. buildLog := "demo build generated from sample classic-sequential.json"
  148. build, err := s.UpsertEventConfigBuild(ctx, tx, UpsertEventConfigBuildParams{
  149. EventID: eventID,
  150. SourceID: source.ID,
  151. BuildNo: 1,
  152. BuildStatus: "success",
  153. BuildLog: &buildLog,
  154. Manifest: map[string]any{
  155. "schemaVersion": "1",
  156. "releaseId": "rel_demo_001",
  157. "version": "2026.04.01",
  158. "app": map[string]any{
  159. "id": "sample-classic-001",
  160. "title": "顺序赛示例",
  161. },
  162. "map": map[string]any{
  163. "tiles": "https://oss-mbh5.colormaprun.com/gotomars/map/lxcb-001/tiles/",
  164. "mapmeta": "https://oss-mbh5.colormaprun.com/gotomars/map/lxcb-001/tiles/meta.json",
  165. },
  166. "playfield": map[string]any{
  167. "kind": "course",
  168. "source": map[string]any{
  169. "type": "kml",
  170. "url": "https://oss-mbh5.colormaprun.com/gotomars/kml/lxcb-001/10/c01.kml",
  171. },
  172. },
  173. "game": map[string]any{
  174. "mode": "classic-sequential",
  175. },
  176. "assets": map[string]any{
  177. "contentHtml": "https://oss-mbh5.colormaprun.com/gotomars/event/content-h5-test-template.html",
  178. },
  179. },
  180. AssetIndex: []map[string]any{
  181. {
  182. "assetType": "manifest",
  183. "assetKey": "manifest",
  184. },
  185. {
  186. "assetType": "mapmeta",
  187. "assetKey": "mapmeta",
  188. },
  189. {
  190. "assetType": "playfield",
  191. "assetKey": "playfield-kml",
  192. },
  193. {
  194. "assetType": "content_html",
  195. "assetKey": "content-html",
  196. },
  197. },
  198. })
  199. if err != nil {
  200. return nil, fmt.Errorf("ensure demo event config build: %w", err)
  201. }
  202. if err := s.AttachBuildToRelease(ctx, tx, releaseRow.ID, build.ID); err != nil {
  203. return nil, fmt.Errorf("attach demo build to release: %w", err)
  204. }
  205. tilesPath := "map/lxcb-001/tiles/"
  206. mapmetaPath := "map/lxcb-001/tiles/meta.json"
  207. playfieldPath := "kml/lxcb-001/10/c01.kml"
  208. contentPath := "event/content-h5-test-template.html"
  209. manifestChecksum := "demo-checksum-001"
  210. if err := s.ReplaceEventReleaseAssets(ctx, tx, releaseRow.ID, []UpsertEventReleaseAssetParams{
  211. {
  212. EventReleaseID: releaseRow.ID,
  213. AssetType: "manifest",
  214. AssetKey: "manifest",
  215. AssetURL: "https://oss-mbh5.colormaprun.com/gotomars/event/classic-sequential.json",
  216. Checksum: &manifestChecksum,
  217. Meta: map[string]any{"source": "release-manifest"},
  218. },
  219. {
  220. EventReleaseID: releaseRow.ID,
  221. AssetType: "tiles",
  222. AssetKey: "tiles-root",
  223. AssetPath: &tilesPath,
  224. AssetURL: "https://oss-mbh5.colormaprun.com/gotomars/map/lxcb-001/tiles/",
  225. Meta: map[string]any{"kind": "directory"},
  226. },
  227. {
  228. EventReleaseID: releaseRow.ID,
  229. AssetType: "mapmeta",
  230. AssetKey: "mapmeta",
  231. AssetPath: &mapmetaPath,
  232. AssetURL: "https://oss-mbh5.colormaprun.com/gotomars/map/lxcb-001/tiles/meta.json",
  233. Meta: map[string]any{"format": "json"},
  234. },
  235. {
  236. EventReleaseID: releaseRow.ID,
  237. AssetType: "playfield",
  238. AssetKey: "course-kml",
  239. AssetPath: &playfieldPath,
  240. AssetURL: "https://oss-mbh5.colormaprun.com/gotomars/kml/lxcb-001/10/c01.kml",
  241. Meta: map[string]any{"format": "kml"},
  242. },
  243. {
  244. EventReleaseID: releaseRow.ID,
  245. AssetType: "content_html",
  246. AssetKey: "content-html",
  247. AssetPath: &contentPath,
  248. AssetURL: "https://oss-mbh5.colormaprun.com/gotomars/event/content-h5-test-template.html",
  249. Meta: map[string]any{"kind": "content-page"},
  250. },
  251. }); err != nil {
  252. return nil, fmt.Errorf("ensure demo event release assets: %w", err)
  253. }
  254. var cardPublicID string
  255. if err := tx.QueryRow(ctx, `
  256. INSERT INTO cards (
  257. card_public_id,
  258. tenant_id,
  259. entry_channel_id,
  260. card_type,
  261. title,
  262. subtitle,
  263. cover_url,
  264. event_id,
  265. display_slot,
  266. display_priority,
  267. status
  268. )
  269. VALUES (
  270. 'card_demo_001',
  271. $1,
  272. $2,
  273. 'event',
  274. 'Demo City Run',
  275. '今日推荐路线',
  276. 'https://oss-mbh5.colormaprun.com/gotomars/assets/demo-cover.jpg',
  277. $3,
  278. 'home_primary',
  279. 100,
  280. 'active'
  281. )
  282. ON CONFLICT (card_public_id) DO UPDATE SET
  283. tenant_id = EXCLUDED.tenant_id,
  284. entry_channel_id = EXCLUDED.entry_channel_id,
  285. card_type = EXCLUDED.card_type,
  286. title = EXCLUDED.title,
  287. subtitle = EXCLUDED.subtitle,
  288. cover_url = EXCLUDED.cover_url,
  289. event_id = EXCLUDED.event_id,
  290. display_slot = EXCLUDED.display_slot,
  291. display_priority = EXCLUDED.display_priority,
  292. status = EXCLUDED.status
  293. RETURNING card_public_id
  294. `, tenantID, channelID, eventID).Scan(&cardPublicID); err != nil {
  295. return nil, fmt.Errorf("ensure demo card: %w", err)
  296. }
  297. if err := tx.Commit(ctx); err != nil {
  298. return nil, err
  299. }
  300. return &DemoBootstrapSummary{
  301. TenantCode: "tenant_demo",
  302. ChannelCode: "mini-demo",
  303. EventID: "evt_demo_001",
  304. ReleaseID: releaseRow.PublicID,
  305. SourceID: source.ID,
  306. BuildID: build.ID,
  307. CardID: cardPublicID,
  308. }, nil
  309. }