@spfn/cms 0.1.0-alpha.8 → 0.1.0-alpha.81

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (233) hide show
  1. package/README.md +28 -416
  2. package/dist/{helpers/locale.actions.d.ts → actions-BEFWwQsh.d.ts} +70 -7
  3. package/dist/actions.d.ts +2 -9
  4. package/dist/actions.js +99 -10
  5. package/dist/actions.js.map +1 -1
  6. package/dist/api.d.ts +319 -0
  7. package/dist/api.js +467 -0
  8. package/dist/api.js.map +1 -0
  9. package/dist/client.d.ts +135 -127
  10. package/dist/client.js +1318 -59
  11. package/dist/client.js.map +1 -1
  12. package/dist/{types.d.ts → index-Dh5FjWzR.d.ts} +45 -7
  13. package/dist/index.d.ts +112 -16
  14. package/dist/index.js +625 -23
  15. package/dist/index.js.map +1 -1
  16. package/dist/label-sync-generator-B0EmvtWM.d.ts +32 -0
  17. package/dist/lib/contracts/labels.d.ts +244 -0
  18. package/dist/lib/contracts/labels.js +269 -0
  19. package/dist/lib/contracts/labels.js.map +1 -0
  20. package/dist/lib/contracts/published-cache.d.ts +48 -0
  21. package/dist/lib/contracts/published-cache.js +49 -0
  22. package/dist/lib/contracts/published-cache.js.map +1 -0
  23. package/dist/lib/contracts/values.d.ts +71 -0
  24. package/dist/lib/contracts/values.js +104 -0
  25. package/dist/lib/contracts/values.js.map +1 -0
  26. package/dist/locale.constants-BNkSdNP1.d.ts +108 -0
  27. package/dist/{entities → server/entities}/cms-audit-logs.d.ts +15 -70
  28. package/dist/server/entities/cms-audit-logs.js +78 -0
  29. package/dist/server/entities/cms-audit-logs.js.map +1 -0
  30. package/dist/{entities → server/entities}/cms-draft-cache.d.ts +13 -73
  31. package/dist/server/entities/cms-draft-cache.js +38 -0
  32. package/dist/server/entities/cms-draft-cache.js.map +1 -0
  33. package/dist/{entities → server/entities}/cms-label-values.d.ts +16 -67
  34. package/dist/server/entities/cms-label-values.js +81 -0
  35. package/dist/server/entities/cms-label-values.js.map +1 -0
  36. package/dist/{entities → server/entities}/cms-labels.d.ts +17 -14
  37. package/dist/server/entities/cms-labels.js +42 -0
  38. package/dist/server/entities/cms-labels.js.map +1 -0
  39. package/dist/{entities → server/entities}/cms-published-cache.d.ts +14 -69
  40. package/dist/server/entities/cms-published-cache.js +36 -0
  41. package/dist/server/entities/cms-published-cache.js.map +1 -0
  42. package/dist/server/entities/index.d.ts +6 -0
  43. package/dist/server/entities/index.js +185 -0
  44. package/dist/server/entities/index.js.map +1 -0
  45. package/dist/server/generators/index.d.ts +19 -0
  46. package/dist/server/generators/index.js +731 -0
  47. package/dist/server/generators/index.js.map +1 -0
  48. package/dist/server/labels/index.d.ts +1 -0
  49. package/dist/server/labels/index.js +33 -0
  50. package/dist/server/labels/index.js.map +1 -0
  51. package/dist/server/repositories/index.d.ts +212 -0
  52. package/dist/server/repositories/index.js +418 -0
  53. package/dist/server/repositories/index.js.map +1 -0
  54. package/dist/server/routes/labels/[id]/admin/index.js +679 -0
  55. package/dist/server/routes/labels/[id]/admin/index.js.map +1 -0
  56. package/dist/server/routes/labels/[id]/index.js +576 -0
  57. package/dist/server/routes/labels/[id]/index.js.map +1 -0
  58. package/dist/server/routes/labels/[id]/publish/index.js +720 -0
  59. package/dist/server/routes/labels/[id]/publish/index.js.map +1 -0
  60. package/dist/server/routes/labels/[id]/versions/index.js +548 -0
  61. package/dist/server/routes/labels/[id]/versions/index.js.map +1 -0
  62. package/dist/server/routes/labels/_id_/admin/index.d.ts +11 -0
  63. package/dist/{routes/labels/[id] → server/routes/labels/_id_}/index.d.ts +5 -3
  64. package/dist/server/routes/labels/_id_/publish/index.d.ts +11 -0
  65. package/dist/server/routes/labels/_id_/versions/index.d.ts +11 -0
  66. package/dist/server/routes/labels/by-key/[key]/index.js +525 -0
  67. package/dist/server/routes/labels/by-key/[key]/index.js.map +1 -0
  68. package/dist/server/routes/labels/by-key/_key_/index.d.ts +10 -0
  69. package/dist/server/routes/labels/index.d.ts +12 -0
  70. package/dist/server/routes/labels/index.js +684 -0
  71. package/dist/server/routes/labels/index.js.map +1 -0
  72. package/dist/server/routes/published-cache/index.d.ts +11 -0
  73. package/dist/server/routes/published-cache/index.js +337 -0
  74. package/dist/server/routes/published-cache/index.js.map +1 -0
  75. package/dist/server/routes/values/[labelId]/[version]/index.js +457 -0
  76. package/dist/server/routes/values/[labelId]/[version]/index.js.map +1 -0
  77. package/dist/server/routes/values/[labelId]/index.js +452 -0
  78. package/dist/server/routes/values/[labelId]/index.js.map +1 -0
  79. package/dist/server/routes/values/_labelId_/_version_/index.d.ts +10 -0
  80. package/dist/server/routes/values/_labelId_/index.d.ts +10 -0
  81. package/dist/server.d.ts +77 -7
  82. package/dist/server.js +1747 -247
  83. package/dist/server.js.map +1 -1
  84. package/migrations/0000_init.sql +3 -0
  85. package/migrations/0001_far_lady_vermin.sql +86 -0
  86. package/migrations/0002_heavy_the_enforcers.sql +2 -0
  87. package/migrations/0003_rare_runaways.sql +1 -0
  88. package/migrations/meta/0000_snapshot.json +15 -0
  89. package/migrations/meta/0001_snapshot.json +687 -0
  90. package/migrations/meta/0002_snapshot.json +686 -0
  91. package/migrations/meta/0003_snapshot.json +563 -0
  92. package/migrations/meta/_journal.json +34 -0
  93. package/package.json +55 -36
  94. package/dist/actions.d.ts.map +0 -1
  95. package/dist/client.d.ts.map +0 -1
  96. package/dist/cms.config.d.ts +0 -77
  97. package/dist/cms.config.d.ts.map +0 -1
  98. package/dist/cms.config.js +0 -111
  99. package/dist/cms.config.js.map +0 -1
  100. package/dist/entities/cms-audit-logs.d.ts.map +0 -1
  101. package/dist/entities/cms-audit-logs.js +0 -103
  102. package/dist/entities/cms-audit-logs.js.map +0 -1
  103. package/dist/entities/cms-draft-cache.d.ts.map +0 -1
  104. package/dist/entities/cms-draft-cache.js +0 -112
  105. package/dist/entities/cms-draft-cache.js.map +0 -1
  106. package/dist/entities/cms-label-values.d.ts.map +0 -1
  107. package/dist/entities/cms-label-values.js +0 -105
  108. package/dist/entities/cms-label-values.js.map +0 -1
  109. package/dist/entities/cms-label-versions.d.ts +0 -207
  110. package/dist/entities/cms-label-versions.d.ts.map +0 -1
  111. package/dist/entities/cms-label-versions.js +0 -80
  112. package/dist/entities/cms-label-versions.js.map +0 -1
  113. package/dist/entities/cms-labels.d.ts.map +0 -1
  114. package/dist/entities/cms-labels.js +0 -48
  115. package/dist/entities/cms-labels.js.map +0 -1
  116. package/dist/entities/cms-published-cache.d.ts.map +0 -1
  117. package/dist/entities/cms-published-cache.js +0 -103
  118. package/dist/entities/cms-published-cache.js.map +0 -1
  119. package/dist/entities/index.d.ts +0 -10
  120. package/dist/entities/index.d.ts.map +0 -1
  121. package/dist/entities/index.js +0 -10
  122. package/dist/entities/index.js.map +0 -1
  123. package/dist/generators/index.d.ts +0 -19
  124. package/dist/generators/index.d.ts.map +0 -1
  125. package/dist/generators/index.js +0 -19
  126. package/dist/generators/index.js.map +0 -1
  127. package/dist/generators/label-sync-generator.d.ts +0 -33
  128. package/dist/generators/label-sync-generator.d.ts.map +0 -1
  129. package/dist/generators/label-sync-generator.js +0 -86
  130. package/dist/generators/label-sync-generator.js.map +0 -1
  131. package/dist/helpers/locale.actions.d.ts.map +0 -1
  132. package/dist/helpers/locale.actions.js +0 -210
  133. package/dist/helpers/locale.actions.js.map +0 -1
  134. package/dist/helpers/locale.constants.d.ts +0 -10
  135. package/dist/helpers/locale.constants.d.ts.map +0 -1
  136. package/dist/helpers/locale.constants.js +0 -10
  137. package/dist/helpers/locale.constants.js.map +0 -1
  138. package/dist/helpers/locale.d.ts +0 -17
  139. package/dist/helpers/locale.d.ts.map +0 -1
  140. package/dist/helpers/locale.js +0 -20
  141. package/dist/helpers/locale.js.map +0 -1
  142. package/dist/helpers/sync.d.ts +0 -41
  143. package/dist/helpers/sync.d.ts.map +0 -1
  144. package/dist/helpers/sync.js +0 -309
  145. package/dist/helpers/sync.js.map +0 -1
  146. package/dist/index.d.ts.map +0 -1
  147. package/dist/init.d.ts +0 -31
  148. package/dist/init.d.ts.map +0 -1
  149. package/dist/init.js +0 -36
  150. package/dist/init.js.map +0 -1
  151. package/dist/labels/helpers.d.ts +0 -31
  152. package/dist/labels/helpers.d.ts.map +0 -1
  153. package/dist/labels/helpers.js +0 -60
  154. package/dist/labels/helpers.js.map +0 -1
  155. package/dist/labels/index.d.ts +0 -7
  156. package/dist/labels/index.d.ts.map +0 -1
  157. package/dist/labels/index.js +0 -7
  158. package/dist/labels/index.js.map +0 -1
  159. package/dist/repositories/cms-draft-cache.repository.d.ts +0 -62
  160. package/dist/repositories/cms-draft-cache.repository.d.ts.map +0 -1
  161. package/dist/repositories/cms-draft-cache.repository.js +0 -56
  162. package/dist/repositories/cms-draft-cache.repository.js.map +0 -1
  163. package/dist/repositories/cms-label-values.repository.d.ts +0 -32
  164. package/dist/repositories/cms-label-values.repository.d.ts.map +0 -1
  165. package/dist/repositories/cms-label-values.repository.js +0 -72
  166. package/dist/repositories/cms-label-values.repository.js.map +0 -1
  167. package/dist/repositories/cms-labels.repository.d.ts +0 -53
  168. package/dist/repositories/cms-labels.repository.d.ts.map +0 -1
  169. package/dist/repositories/cms-labels.repository.js +0 -77
  170. package/dist/repositories/cms-labels.repository.js.map +0 -1
  171. package/dist/repositories/cms-published-cache.repository.d.ts +0 -53
  172. package/dist/repositories/cms-published-cache.repository.d.ts.map +0 -1
  173. package/dist/repositories/cms-published-cache.repository.js +0 -54
  174. package/dist/repositories/cms-published-cache.repository.js.map +0 -1
  175. package/dist/repositories/index.d.ts +0 -8
  176. package/dist/repositories/index.d.ts.map +0 -1
  177. package/dist/repositories/index.js +0 -9
  178. package/dist/repositories/index.js.map +0 -1
  179. package/dist/routes/labels/[id]/contract.d.ts +0 -68
  180. package/dist/routes/labels/[id]/contract.d.ts.map +0 -1
  181. package/dist/routes/labels/[id]/contract.js +0 -84
  182. package/dist/routes/labels/[id]/contract.js.map +0 -1
  183. package/dist/routes/labels/[id]/index.d.ts.map +0 -1
  184. package/dist/routes/labels/[id]/index.js +0 -96
  185. package/dist/routes/labels/[id]/index.js.map +0 -1
  186. package/dist/routes/labels/by-key/[key]/contract.d.ts +0 -24
  187. package/dist/routes/labels/by-key/[key]/contract.d.ts.map +0 -1
  188. package/dist/routes/labels/by-key/[key]/contract.js +0 -28
  189. package/dist/routes/labels/by-key/[key]/contract.js.map +0 -1
  190. package/dist/routes/labels/by-key/[key]/index.d.ts +0 -8
  191. package/dist/routes/labels/by-key/[key]/index.d.ts.map +0 -1
  192. package/dist/routes/labels/by-key/[key]/index.js +0 -32
  193. package/dist/routes/labels/by-key/[key]/index.js.map +0 -1
  194. package/dist/routes/labels/contract.d.ts +0 -59
  195. package/dist/routes/labels/contract.d.ts.map +0 -1
  196. package/dist/routes/labels/contract.js +0 -75
  197. package/dist/routes/labels/contract.js.map +0 -1
  198. package/dist/routes/labels/index.d.ts +0 -10
  199. package/dist/routes/labels/index.d.ts.map +0 -1
  200. package/dist/routes/labels/index.js +0 -73
  201. package/dist/routes/labels/index.js.map +0 -1
  202. package/dist/routes/published-cache/contract.d.ts +0 -25
  203. package/dist/routes/published-cache/contract.d.ts.map +0 -1
  204. package/dist/routes/published-cache/contract.js +0 -35
  205. package/dist/routes/published-cache/contract.js.map +0 -1
  206. package/dist/routes/published-cache/index.d.ts +0 -8
  207. package/dist/routes/published-cache/index.d.ts.map +0 -1
  208. package/dist/routes/published-cache/index.js +0 -33
  209. package/dist/routes/published-cache/index.js.map +0 -1
  210. package/dist/routes/values/[labelId]/[version]/contract.d.ts +0 -29
  211. package/dist/routes/values/[labelId]/[version]/contract.d.ts.map +0 -1
  212. package/dist/routes/values/[labelId]/[version]/contract.js +0 -33
  213. package/dist/routes/values/[labelId]/[version]/contract.js.map +0 -1
  214. package/dist/routes/values/[labelId]/[version]/index.d.ts +0 -8
  215. package/dist/routes/values/[labelId]/[version]/index.d.ts.map +0 -1
  216. package/dist/routes/values/[labelId]/[version]/index.js +0 -45
  217. package/dist/routes/values/[labelId]/[version]/index.js.map +0 -1
  218. package/dist/routes/values/[labelId]/contract.d.ts +0 -38
  219. package/dist/routes/values/[labelId]/contract.d.ts.map +0 -1
  220. package/dist/routes/values/[labelId]/contract.js +0 -59
  221. package/dist/routes/values/[labelId]/contract.js.map +0 -1
  222. package/dist/routes/values/[labelId]/index.d.ts +0 -8
  223. package/dist/routes/values/[labelId]/index.d.ts.map +0 -1
  224. package/dist/routes/values/[labelId]/index.js +0 -42
  225. package/dist/routes/values/[labelId]/index.js.map +0 -1
  226. package/dist/server.d.ts.map +0 -1
  227. package/dist/store.d.ts +0 -87
  228. package/dist/store.d.ts.map +0 -1
  229. package/dist/store.js +0 -205
  230. package/dist/store.js.map +0 -1
  231. package/dist/types.d.ts.map +0 -1
  232. package/dist/types.js +0 -7
  233. package/dist/types.js.map +0 -1
@@ -0,0 +1,548 @@
1
+ // src/server/routes/labels/[id]/versions/index.ts
2
+ import { createApp } from "@spfn/core/route";
3
+
4
+ // src/lib/contracts/labels.ts
5
+ import { Type } from "@sinclair/typebox";
6
+ var getLabelsContract = {
7
+ method: "GET",
8
+ path: "/_cms/labels",
9
+ query: Type.Object({
10
+ section: Type.Optional(Type.String({ description: "\uC139\uC158\uC73C\uB85C \uD544\uD130\uB9C1 (\uC608: home, why-futureplay)" })),
11
+ includeDefaultValues: Type.Optional(Type.Boolean({ description: "\uAE30\uBCF8\uAC12 \uD3EC\uD568 \uC5EC\uBD80" }))
12
+ }),
13
+ response: Type.Object({
14
+ labels: Type.Array(Type.Object({
15
+ id: Type.Number(),
16
+ key: Type.String(),
17
+ section: Type.String(),
18
+ type: Type.String(),
19
+ description: Type.Union([Type.String(), Type.Null()], { description: "\uB77C\uBCA8 \uC124\uBA85" }),
20
+ publishedVersion: Type.Union([Type.Number(), Type.Null()]),
21
+ createdBy: Type.Union([Type.String(), Type.Null()]),
22
+ createdAt: Type.String(),
23
+ updatedAt: Type.String(),
24
+ defaultValue: Type.Optional(Type.Any({ description: "\uB77C\uBCA8 \uC815\uC758 \uD30C\uC77C\uC758 \uAE30\uBCF8\uAC12" }))
25
+ })),
26
+ total: Type.Number()
27
+ })
28
+ };
29
+ var createLabelContract = {
30
+ method: "POST",
31
+ path: "/_cms/labels",
32
+ body: Type.Object({
33
+ key: Type.String({
34
+ description: "\uACE0\uC720 \uD0A4 (\uC608: home.hero.title)",
35
+ pattern: "^[a-z0-9-]+\\.[a-z0-9-]+\\.[a-z0-9-]+$"
36
+ }),
37
+ section: Type.String({
38
+ description: "\uC139\uC158 \uC774\uB984 (\uC608: home, why-futureplay)",
39
+ pattern: "^[a-z0-9-]+$"
40
+ }),
41
+ type: Type.Union([
42
+ Type.Literal("text"),
43
+ Type.Literal("image"),
44
+ Type.Literal("video"),
45
+ Type.Literal("file"),
46
+ Type.Literal("object")
47
+ ], { description: "\uAC12 \uD0C0\uC785" }),
48
+ createdBy: Type.Optional(Type.String({ description: "\uC0DD\uC131\uC790 ID" }))
49
+ }),
50
+ response: Type.Union([
51
+ Type.Object({
52
+ id: Type.Number(),
53
+ key: Type.String(),
54
+ section: Type.String(),
55
+ type: Type.String(),
56
+ publishedVersion: Type.Union([Type.Number(), Type.Null()]),
57
+ createdBy: Type.Union([Type.String(), Type.Null()]),
58
+ createdAt: Type.String(),
59
+ updatedAt: Type.String()
60
+ }),
61
+ Type.Object({
62
+ error: Type.String(),
63
+ key: Type.Optional(Type.String())
64
+ })
65
+ ])
66
+ };
67
+ var getLabelContract = {
68
+ method: "GET",
69
+ path: "/_cms/labels/:id",
70
+ params: Type.Object({
71
+ id: Type.String({ description: "\uB77C\uBCA8 ID" })
72
+ }),
73
+ response: Type.Union([
74
+ Type.Object({
75
+ id: Type.Number(),
76
+ key: Type.String(),
77
+ section: Type.String(),
78
+ type: Type.String(),
79
+ description: Type.Union([Type.String(), Type.Null()]),
80
+ publishedVersion: Type.Union([Type.Number(), Type.Null()]),
81
+ createdBy: Type.Union([Type.String(), Type.Null()]),
82
+ createdAt: Type.String(),
83
+ updatedAt: Type.String()
84
+ }),
85
+ Type.Object({
86
+ error: Type.String()
87
+ })
88
+ ])
89
+ };
90
+ var updateLabelContract = {
91
+ method: "PATCH",
92
+ path: "/_cms/labels/:id",
93
+ params: Type.Object({
94
+ id: Type.String({ description: "\uB77C\uBCA8 ID" })
95
+ }),
96
+ body: Type.Object({
97
+ section: Type.Optional(Type.String({ description: "\uC139\uC158 \uBCC0\uACBD" })),
98
+ type: Type.Optional(Type.Union([
99
+ Type.Literal("text"),
100
+ Type.Literal("image"),
101
+ Type.Literal("video"),
102
+ Type.Literal("file"),
103
+ Type.Literal("object")
104
+ ]))
105
+ }),
106
+ response: Type.Union([
107
+ Type.Object({
108
+ id: Type.Number(),
109
+ key: Type.String(),
110
+ section: Type.String(),
111
+ type: Type.String(),
112
+ description: Type.Union([Type.String(), Type.Null()]),
113
+ publishedVersion: Type.Union([Type.Number(), Type.Null()]),
114
+ createdBy: Type.Union([Type.String(), Type.Null()]),
115
+ createdAt: Type.String(),
116
+ updatedAt: Type.String()
117
+ }),
118
+ Type.Object({
119
+ error: Type.String()
120
+ })
121
+ ])
122
+ };
123
+ var deleteLabelContract = {
124
+ method: "DELETE",
125
+ path: "/_cms/labels/:id",
126
+ params: Type.Object({
127
+ id: Type.String({ description: "\uB77C\uBCA8 ID" })
128
+ }),
129
+ response: Type.Union([
130
+ Type.Object({
131
+ success: Type.Boolean(),
132
+ id: Type.Number()
133
+ }),
134
+ Type.Object({
135
+ error: Type.String()
136
+ })
137
+ ])
138
+ };
139
+ var getLabelByKeyContract = {
140
+ method: "GET",
141
+ path: "/_cms/labels/by-key/:key",
142
+ params: Type.Object({
143
+ key: Type.String({ description: "\uB77C\uBCA8 Key (\uC608: home.hero.title)" })
144
+ }),
145
+ response: Type.Union([
146
+ Type.Object({
147
+ id: Type.Number(),
148
+ key: Type.String(),
149
+ section: Type.String(),
150
+ type: Type.String(),
151
+ description: Type.Union([Type.String(), Type.Null()]),
152
+ publishedVersion: Type.Union([Type.Number(), Type.Null()]),
153
+ createdBy: Type.Union([Type.String(), Type.Null()]),
154
+ createdAt: Type.String(),
155
+ updatedAt: Type.String()
156
+ }),
157
+ Type.Object({
158
+ error: Type.String(),
159
+ key: Type.Optional(Type.String())
160
+ })
161
+ ])
162
+ };
163
+ var publishLabelContract = {
164
+ method: "POST",
165
+ path: "/_cms/labels/:id/publish",
166
+ params: Type.Object({
167
+ id: Type.String({ description: "\uB77C\uBCA8 ID" })
168
+ }),
169
+ body: Type.Object({
170
+ notes: Type.Optional(Type.String({ description: "\uBC1C\uD589 \uB178\uD2B8 (\uBC84\uC804 \uC124\uBA85)" })),
171
+ publishedBy: Type.Optional(Type.String({ description: "\uBC1C\uD589\uC790 ID" }))
172
+ }),
173
+ response: Type.Union([
174
+ Type.Object({
175
+ success: Type.Boolean(),
176
+ id: Type.Number(),
177
+ version: Type.Number(),
178
+ message: Type.String()
179
+ }),
180
+ Type.Object({
181
+ error: Type.String()
182
+ })
183
+ ])
184
+ };
185
+ var getAdminLabelContract = {
186
+ method: "GET",
187
+ path: "/_cms/labels/:id/admin",
188
+ params: Type.Object({
189
+ id: Type.String({ description: "\uB77C\uBCA8 ID" })
190
+ }),
191
+ response: Type.Union([
192
+ Type.Object({
193
+ label: Type.Object({
194
+ id: Type.Number(),
195
+ key: Type.String(),
196
+ section: Type.String(),
197
+ type: Type.String(),
198
+ description: Type.Union([Type.String(), Type.Null()]),
199
+ publishedVersion: Type.Union([Type.Number(), Type.Null()]),
200
+ createdBy: Type.Union([Type.String(), Type.Null()]),
201
+ createdAt: Type.String(),
202
+ updatedAt: Type.String()
203
+ }),
204
+ draft: Type.Array(Type.Object({
205
+ id: Type.Number(),
206
+ labelId: Type.Number(),
207
+ version: Type.Null(),
208
+ locale: Type.String(),
209
+ breakpoint: Type.Union([Type.String(), Type.Null()]),
210
+ value: Type.Any(),
211
+ createdAt: Type.String()
212
+ })),
213
+ published: Type.Array(Type.Object({
214
+ id: Type.Number(),
215
+ labelId: Type.Number(),
216
+ version: Type.Number(),
217
+ locale: Type.String(),
218
+ breakpoint: Type.Union([Type.String(), Type.Null()]),
219
+ value: Type.Any(),
220
+ createdAt: Type.String()
221
+ })),
222
+ status: Type.Union([
223
+ Type.Literal("default-only"),
224
+ Type.Literal("unpublished"),
225
+ Type.Literal("published"),
226
+ Type.Literal("modified")
227
+ ])
228
+ }),
229
+ Type.Object({
230
+ error: Type.String()
231
+ })
232
+ ])
233
+ };
234
+ var getLabelVersionsContract = {
235
+ method: "GET",
236
+ path: "/_cms/labels/:id/versions",
237
+ params: Type.Object({
238
+ id: Type.String({ description: "\uB77C\uBCA8 ID" })
239
+ }),
240
+ response: Type.Union([
241
+ Type.Object({
242
+ versions: Type.Array(Type.Object({
243
+ version: Type.Number({ description: "\uBC84\uC804 \uBC88\uD638" }),
244
+ publishedAt: Type.String({ description: "\uBC1C\uD589 \uC2DC\uAC01 (ISO 8601)" }),
245
+ publishedBy: Type.Union([Type.String(), Type.Null()], { description: "\uBC1C\uD589\uC790 ID" }),
246
+ notes: Type.Union([Type.String(), Type.Null()], { description: "\uBC1C\uD589 \uB178\uD2B8" }),
247
+ values: Type.Array(Type.Object({
248
+ id: Type.Number(),
249
+ locale: Type.String(),
250
+ breakpoint: Type.Union([Type.String(), Type.Null()]),
251
+ value: Type.Any(),
252
+ createdAt: Type.String()
253
+ }))
254
+ }))
255
+ }),
256
+ Type.Object({
257
+ error: Type.String()
258
+ })
259
+ ])
260
+ };
261
+
262
+ // src/server/repositories/cms-labels.repository.ts
263
+ import { findOne, findMany as findManyHelper, create as createHelper, updateOne, deleteOne, count as countHelper } from "@spfn/core/db";
264
+ import { asc } from "drizzle-orm";
265
+
266
+ // src/server/entities/cms-labels.ts
267
+ import { index, integer, serial, text, timestamp } from "drizzle-orm/pg-core";
268
+ import { createFunctionSchema } from "@spfn/core/db";
269
+ var schema = createFunctionSchema("@spfn/cms");
270
+ var cmsLabels = schema.table("labels", {
271
+ // Primary Key
272
+ id: serial("id").primaryKey(),
273
+ // 라벨 식별자
274
+ key: text("key").notNull().unique(),
275
+ // 예: "home.hero.title", "why-futureplay.hero.subtitle"
276
+ // 구조: {section}.{component}.{property}
277
+ // 섹션 분류 (페이지 단위)
278
+ section: text("section").notNull(),
279
+ // 예: "home", "why-futureplay", "team"
280
+ // 값 타입
281
+ type: text("type").notNull(),
282
+ // "text" | "image" | "video" | "file" | "object"
283
+ // 기본값
284
+ defaultValue: text("default_value"),
285
+ // 라벨의 기본값 (sync 시 설정)
286
+ // 설명
287
+ description: text("description"),
288
+ // 라벨에 대한 설명 (optional)
289
+ // 현재 발행된 버전 번호
290
+ publishedVersion: integer("published_version"),
291
+ // null = 미발행 상태
292
+ // 1, 2, 3... = 발행된 버전 번호
293
+ // 생성자 추적
294
+ createdBy: text("created_by"),
295
+ // 타임스탬프
296
+ createdAt: timestamp("created_at", { withTimezone: true }).notNull().defaultNow(),
297
+ updatedAt: timestamp("updated_at", { withTimezone: true }).notNull().defaultNow()
298
+ }, (table) => [
299
+ // 인덱스: 섹션별 조회 최적화
300
+ index("cms_labels_section_idx").on(table.section),
301
+ // 인덱스: key로 조회 최적화 (unique 제약으로 자동 생성되지만 명시)
302
+ index("cms_labels_key_idx").on(table.key)
303
+ ]);
304
+
305
+ // src/server/entities/cms-label-values.ts
306
+ import { serial as serial2, integer as integer2, text as text2, jsonb, timestamp as timestamp2, index as index2, unique } from "drizzle-orm/pg-core";
307
+ import { createFunctionSchema as createFunctionSchema2 } from "@spfn/core/db";
308
+ var schema2 = createFunctionSchema2("@spfn/cms");
309
+ var cmsLabelValues = schema2.table("label_values", {
310
+ // Primary Key
311
+ id: serial2("id").primaryKey(),
312
+ // Foreign Key: cms_labels
313
+ labelId: integer2("label_id").notNull().references(() => cmsLabels.id, { onDelete: "cascade" }),
314
+ // 버전 번호 (null = draft, number = published version)
315
+ version: integer2("version"),
316
+ // 언어 코드
317
+ locale: text2("locale").notNull().default("ko"),
318
+ // "ko" | "en" | "ja"
319
+ // 반응형 브레이크포인트
320
+ breakpoint: text2("breakpoint"),
321
+ // null = 기본값 (모든 화면 크기)
322
+ // "sm" | "md" | "lg" | "xl" | "2xl"
323
+ // 실제 값 (JSONB)
324
+ value: jsonb("value").notNull(),
325
+ // LabelValue 타입:
326
+ // - TextValue: { type: "text", content: string }
327
+ // - ImageValue: { type: "image", url: string, alt?: string, width?: number, height?: number }
328
+ // - VideoValue: { type: "video", url: string, thumbnail?: string, duration?: number }
329
+ // - FileValue: { type: "file", url: string, filename: string, size?: number }
330
+ // - ObjectValue: { type: "object", fields: Record<string, LabelValue> }
331
+ // 생성 시각
332
+ createdAt: timestamp2("created_at", { withTimezone: true }).notNull().defaultNow()
333
+ }, (table) => [
334
+ // UNIQUE 제약: 같은 버전에서 locale + breakpoint 조합은 유일
335
+ unique("cms_label_values_locale_breakpoint_unique").on(table.labelId, table.version, table.locale, table.breakpoint),
336
+ // 인덱스: labelId + version 복합 조회 최적화
337
+ index2("cms_label_values_label_version_idx").on(table.labelId, table.version),
338
+ // 인덱스: locale 필터링 최적화
339
+ index2("cms_label_values_locale_idx").on(table.locale)
340
+ ]);
341
+
342
+ // src/server/entities/cms-draft-cache.ts
343
+ import { serial as serial3, text as text3, jsonb as jsonb2, timestamp as timestamp3, index as index3, unique as unique2 } from "drizzle-orm/pg-core";
344
+ import { createFunctionSchema as createFunctionSchema3 } from "@spfn/core/db";
345
+ var schema3 = createFunctionSchema3("@spfn/cms");
346
+ var cmsDraftCache = schema3.table("draft_cache", {
347
+ // Primary Key
348
+ id: serial3("id").primaryKey(),
349
+ // 섹션 (페이지 단위)
350
+ section: text3("section").notNull(),
351
+ // "home" | "why-futureplay" | "team" | "our-companies" | "apply"
352
+ // 언어
353
+ locale: text3("locale").notNull(),
354
+ // "ko" | "en" | "ja"
355
+ // 사용자 ID (핵심 필드!)
356
+ userId: text3("user_id").notNull(),
357
+ // 각 관리자의 독립적인 작업 공간
358
+ // Draft 콘텐츠 (JSONB)
359
+ content: jsonb2("content").notNull(),
360
+ // Record<string, LabelValue>
361
+ // {
362
+ // "home.hero.title": { type: "text", content: "수정 중..." },
363
+ // "home.hero.subtitle": { type: "text", content: "새로운 문구" },
364
+ // ...
365
+ // }
366
+ // 최종 수정 시각
367
+ updatedAt: timestamp3("updated_at", { withTimezone: true }).notNull().defaultNow()
368
+ }, (table) => [
369
+ // UNIQUE 제약: section + locale + userId 조합은 유일
370
+ unique2("cms_draft_cache_unique").on(table.section, table.locale, table.userId),
371
+ // 인덱스: section으로 조회 최적화
372
+ index3("cms_draft_cache_section_idx").on(table.section),
373
+ // 인덱스: userId로 사용자의 모든 draft 조회 최적화
374
+ index3("cms_draft_cache_user_idx").on(table.userId)
375
+ ]);
376
+
377
+ // src/server/entities/cms-published-cache.ts
378
+ import { serial as serial4, text as text4, jsonb as jsonb3, integer as integer3, timestamp as timestamp4, index as index4, unique as unique3 } from "drizzle-orm/pg-core";
379
+ import { createFunctionSchema as createFunctionSchema4 } from "@spfn/core/db";
380
+ var schema4 = createFunctionSchema4("@spfn/cms");
381
+ var cmsPublishedCache = schema4.table("published_cache", {
382
+ // Primary Key
383
+ id: serial4("id").primaryKey(),
384
+ // 섹션 (페이지 단위)
385
+ section: text4("section").notNull(),
386
+ // "home" | "why-futureplay" | "team" | "our-companies" | "apply"
387
+ // 언어
388
+ locale: text4("locale").notNull(),
389
+ // "ko" | "en" | "ja"
390
+ // 캐시된 콘텐츠 (JSONB)
391
+ content: jsonb3("content").notNull(),
392
+ // Record<string, LabelValue>
393
+ // {
394
+ // "home.hero.title": { type: "text", content: "..." },
395
+ // "home.hero.image": { type: "image", url: "...", alt: "..." },
396
+ // ...
397
+ // }
398
+ // 발행 정보
399
+ publishedAt: timestamp4("published_at", { withTimezone: true }).notNull(),
400
+ publishedBy: text4("published_by"),
401
+ // 캐시 버전 (클라이언트 캐싱용)
402
+ version: integer3("version").notNull().default(1)
403
+ }, (table) => [
404
+ // UNIQUE 제약: section + locale 조합은 유일
405
+ unique3("cms_published_cache_unique").on(table.section, table.locale),
406
+ // 인덱스: section으로 조회 최적화
407
+ index4("cms_published_cache_section_idx").on(table.section)
408
+ ]);
409
+
410
+ // src/server/entities/cms-audit-logs.ts
411
+ import { serial as serial5, integer as integer4, text as text5, jsonb as jsonb4, timestamp as timestamp5, index as index5 } from "drizzle-orm/pg-core";
412
+ import { createFunctionSchema as createFunctionSchema5 } from "@spfn/core/db";
413
+ var schema5 = createFunctionSchema5("@spfn/cms");
414
+ var cmsAuditLogs = schema5.table("audit_logs", {
415
+ // Primary Key
416
+ id: serial5("id").primaryKey(),
417
+ // Foreign Key: cms_labels (nullable - 라벨 삭제 시 로그는 유지)
418
+ labelId: integer4("label_id").references(() => cmsLabels.id, { onDelete: "set null" }),
419
+ // 작업 유형
420
+ action: text5("action").notNull(),
421
+ // "create" | "update" | "publish" | "unpublish" | "archive" | "delete" | "rollback" | "duplicate"
422
+ // 사용자 정보
423
+ userId: text5("user_id").notNull(),
424
+ userName: text5("user_name"),
425
+ // 변경 내용 (before/after)
426
+ changes: jsonb4("changes"),
427
+ // { before: {...}, after: {...} }
428
+ // 추가 메타데이터
429
+ metadata: jsonb4("metadata"),
430
+ // { version: number, ip: string, userAgent: string, ... }
431
+ // 작업 시각
432
+ createdAt: timestamp5("created_at", { withTimezone: true }).notNull().defaultNow()
433
+ }, (table) => [
434
+ // 인덱스: labelId로 이력 조회 최적화
435
+ index5("cms_audit_logs_label_id_idx").on(table.labelId),
436
+ // 인덱스: userId로 사용자 활동 조회 최적화
437
+ index5("cms_audit_logs_user_id_idx").on(table.userId),
438
+ // 인덱스: action 필터링 최적화
439
+ index5("cms_audit_logs_action_idx").on(table.action),
440
+ // 인덱스: 시간순 조회 최적화
441
+ index5("cms_audit_logs_created_at_idx").on(table.createdAt)
442
+ ]);
443
+
444
+ // src/server/repositories/cms-labels.repository.ts
445
+ async function findMany(options) {
446
+ const { section } = options || {};
447
+ return findManyHelper(cmsLabels, {
448
+ where: section ? { section } : void 0,
449
+ orderBy: asc(cmsLabels.key)
450
+ // key 오름차순 정렬 (JSON 파일의 순서 유지)
451
+ });
452
+ }
453
+ async function count(section) {
454
+ return countHelper(cmsLabels, section ? { section } : void 0);
455
+ }
456
+ async function findById(id) {
457
+ return findOne(cmsLabels, { id });
458
+ }
459
+ async function findByKey(key) {
460
+ return findOne(cmsLabels, { key });
461
+ }
462
+ async function findBySection(section) {
463
+ return findManyHelper(cmsLabels, {
464
+ where: { section },
465
+ orderBy: asc(cmsLabels.key)
466
+ // key 오름차순 정렬 (JSON 파일의 순서 유지)
467
+ });
468
+ }
469
+ async function create(data) {
470
+ return createHelper(cmsLabels, data);
471
+ }
472
+ async function updateById(id, data) {
473
+ return updateOne(cmsLabels, { id }, { ...data, updatedAt: /* @__PURE__ */ new Date() });
474
+ }
475
+ async function deleteById(id) {
476
+ return deleteOne(cmsLabels, { id });
477
+ }
478
+ var cmsLabelsRepository = {
479
+ findMany,
480
+ count,
481
+ findById,
482
+ findByKey,
483
+ findBySection,
484
+ create,
485
+ updateById,
486
+ deleteById
487
+ };
488
+
489
+ // src/server/routes/labels/[id]/versions/index.ts
490
+ import { getDatabase } from "@spfn/core/db";
491
+ import { eq, and } from "drizzle-orm";
492
+ var app = createApp();
493
+ app.bind(getLabelVersionsContract, async (c) => {
494
+ const { id } = c.params;
495
+ try {
496
+ const label = await cmsLabelsRepository.findById(parseInt(id));
497
+ if (!label) {
498
+ return c.json(
499
+ { error: "Label not found" },
500
+ 404
501
+ );
502
+ }
503
+ if (!label.publishedVersion) {
504
+ return c.json({ versions: [] });
505
+ }
506
+ const db = getDatabase("read");
507
+ const versionsWithValues = [];
508
+ for (let version = 1; version <= label.publishedVersion; version++) {
509
+ const values = await db.select().from(cmsLabelValues).where(
510
+ and(
511
+ eq(cmsLabelValues.labelId, label.id),
512
+ eq(cmsLabelValues.version, version)
513
+ )
514
+ ).orderBy(cmsLabelValues.locale);
515
+ if (values.length > 0) {
516
+ versionsWithValues.push({
517
+ version,
518
+ publishedAt: values[0].createdAt.toISOString(),
519
+ // 첫 번째 값의 생성 시각 사용
520
+ publishedBy: null,
521
+ // label_values에는 publishedBy 정보가 없음
522
+ notes: null,
523
+ // label_values에는 notes 정보가 없음
524
+ values: values.map((v) => ({
525
+ id: v.id,
526
+ locale: v.locale,
527
+ breakpoint: v.breakpoint,
528
+ value: v.value,
529
+ createdAt: v.createdAt.toISOString()
530
+ }))
531
+ });
532
+ }
533
+ }
534
+ versionsWithValues.sort((a, b) => b.version - a.version);
535
+ return c.json({ versions: versionsWithValues });
536
+ } catch (error) {
537
+ const err = error;
538
+ return c.json(
539
+ { error: err.message },
540
+ 500
541
+ );
542
+ }
543
+ });
544
+ var versions_default = app;
545
+ export {
546
+ versions_default as default
547
+ };
548
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../../../../../src/server/routes/labels/%5Bid%5D/versions/index.ts","../../../../../../src/lib/contracts/labels.ts","../../../../../../src/server/repositories/cms-labels.repository.ts","../../../../../../src/server/entities/cms-labels.ts","../../../../../../src/server/entities/cms-label-values.ts","../../../../../../src/server/entities/cms-draft-cache.ts","../../../../../../src/server/entities/cms-published-cache.ts","../../../../../../src/server/entities/cms-audit-logs.ts"],"sourcesContent":["/**\n * CMS Label Versions Route\n *\n * 라벨 버전 히스토리 조회 API\n * - GET /labels/:id/versions - 모든 버전 히스토리 조회 (final: /_cms/labels/:id/versions)\n */\n\nimport { createApp } from '@spfn/core/route';\nimport { getLabelVersionsContract } from '@/lib/contracts/labels';\nimport { cmsLabelsRepository } from '@/server/repositories/cms-labels.repository';\nimport { getDatabase } from '@spfn/core/db';\nimport { cmsLabelValues } from '@/server/entities/cms-label-values';\nimport { eq, and } from 'drizzle-orm';\n\nconst app = createApp();\n\n/**\n * GET /labels/:id/versions\n * 라벨의 모든 버전 히스토리 조회\n */\napp.bind(getLabelVersionsContract, async (c) =>\n{\n const { id } = c.params;\n\n try\n {\n // 라벨 존재 확인\n const label = await cmsLabelsRepository.findById(parseInt(id));\n if (!label)\n {\n return c.json(\n { error: 'Label not found' },\n 404\n );\n }\n\n // publishedVersion이 없으면 빈 배열 반환\n if (!label.publishedVersion)\n {\n return c.json({ versions: [] });\n }\n\n // DB 접근 (읽기 전용)\n const db = getDatabase('read')!;\n\n // label_values 테이블에서 version이 null이 아닌 레코드들을 version별로 그룹화하여 조회\n // 1부터 publishedVersion까지의 모든 버전 조회\n const versionsWithValues: any[] = [];\n\n for (let version = 1; version <= label.publishedVersion; version++)\n {\n // 해당 버전의 모든 값 조회\n const values = await db\n .select()\n .from(cmsLabelValues)\n .where(\n and(\n eq(cmsLabelValues.labelId, label.id),\n eq(cmsLabelValues.version, version)\n )\n )\n .orderBy(cmsLabelValues.locale);\n\n if (values.length > 0)\n {\n versionsWithValues.push({\n version,\n publishedAt: values[0].createdAt.toISOString(), // 첫 번째 값의 생성 시각 사용\n publishedBy: null, // label_values에는 publishedBy 정보가 없음\n notes: null, // label_values에는 notes 정보가 없음\n values: values.map(v => ({\n id: v.id,\n locale: v.locale,\n breakpoint: v.breakpoint,\n value: v.value,\n createdAt: v.createdAt.toISOString()\n }))\n });\n }\n }\n\n // 버전 내림차순 정렬 (최신 버전이 먼저)\n versionsWithValues.sort((a, b) => b.version - a.version);\n\n return c.json({ versions: versionsWithValues });\n }\n catch (error)\n {\n const err = error as Error;\n return c.json(\n { error: err.message },\n 500\n );\n }\n});\n\nexport default app;","import { Type } from '@sinclair/typebox';\nimport type { RouteContract } from '@spfn/core/route';\n\n/**\n * CMS Labels Contracts\n *\n * 라벨 메타데이터 관리 API\n */\n\n/**\n * GET /_cms/labels - 라벨 목록 조회\n */\nexport const getLabelsContract = {\n method: 'GET' as const,\n path: '/_cms/labels',\n query: Type.Object({\n section: Type.Optional(Type.String({ description: '섹션으로 필터링 (예: home, why-futureplay)' })),\n includeDefaultValues: Type.Optional(Type.Boolean({ description: '기본값 포함 여부' }))\n }),\n response: Type.Object({\n labels: Type.Array(Type.Object({\n id: Type.Number(),\n key: Type.String(),\n section: Type.String(),\n type: Type.String(),\n description: Type.Union([Type.String(), Type.Null()], { description: '라벨 설명' }),\n publishedVersion: Type.Union([Type.Number(), Type.Null()]),\n createdBy: Type.Union([Type.String(), Type.Null()]),\n createdAt: Type.String(),\n updatedAt: Type.String(),\n defaultValue: Type.Optional(Type.Any({ description: '라벨 정의 파일의 기본값' }))\n })),\n total: Type.Number()\n })\n} as const satisfies RouteContract;\n\n/**\n * POST /_cms/labels - 새 라벨 생성\n */\nexport const createLabelContract = {\n method: 'POST' as const,\n path: '/_cms/labels',\n body: Type.Object({\n key: Type.String({\n description: '고유 키 (예: home.hero.title)',\n pattern: '^[a-z0-9-]+\\\\.[a-z0-9-]+\\\\.[a-z0-9-]+$'\n }),\n section: Type.String({\n description: '섹션 이름 (예: home, why-futureplay)',\n pattern: '^[a-z0-9-]+$'\n }),\n type: Type.Union([\n Type.Literal('text'),\n Type.Literal('image'),\n Type.Literal('video'),\n Type.Literal('file'),\n Type.Literal('object')\n ], { description: '값 타입' }),\n createdBy: Type.Optional(Type.String({ description: '생성자 ID' }))\n }),\n response: Type.Union([\n Type.Object({\n id: Type.Number(),\n key: Type.String(),\n section: Type.String(),\n type: Type.String(),\n publishedVersion: Type.Union([Type.Number(), Type.Null()]),\n createdBy: Type.Union([Type.String(), Type.Null()]),\n createdAt: Type.String(),\n updatedAt: Type.String()\n }),\n Type.Object({\n error: Type.String(),\n key: Type.Optional(Type.String())\n })\n ])\n} as const satisfies RouteContract;\n\n/**\n * GET /_cms/labels/:id - 라벨 단건 조회\n */\nexport const getLabelContract = {\n method: 'GET' as const,\n path: '/_cms/labels/:id',\n params: Type.Object({\n id: Type.String({ description: '라벨 ID' })\n }),\n response: Type.Union([\n Type.Object({\n id: Type.Number(),\n key: Type.String(),\n section: Type.String(),\n type: Type.String(),\n description: Type.Union([Type.String(), Type.Null()]),\n publishedVersion: Type.Union([Type.Number(), Type.Null()]),\n createdBy: Type.Union([Type.String(), Type.Null()]),\n createdAt: Type.String(),\n updatedAt: Type.String()\n }),\n Type.Object({\n error: Type.String()\n })\n ])\n} as const satisfies RouteContract;\n\n/**\n * PATCH /_cms/labels/:id - 라벨 메타데이터 수정\n */\nexport const updateLabelContract = {\n method: 'PATCH' as const,\n path: '/_cms/labels/:id',\n params: Type.Object({\n id: Type.String({ description: '라벨 ID' })\n }),\n body: Type.Object({\n section: Type.Optional(Type.String({ description: '섹션 변경' })),\n type: Type.Optional(Type.Union([\n Type.Literal('text'),\n Type.Literal('image'),\n Type.Literal('video'),\n Type.Literal('file'),\n Type.Literal('object')\n ]))\n }),\n response: Type.Union([\n Type.Object({\n id: Type.Number(),\n key: Type.String(),\n section: Type.String(),\n type: Type.String(),\n description: Type.Union([Type.String(), Type.Null()]),\n publishedVersion: Type.Union([Type.Number(), Type.Null()]),\n createdBy: Type.Union([Type.String(), Type.Null()]),\n createdAt: Type.String(),\n updatedAt: Type.String()\n }),\n Type.Object({\n error: Type.String()\n })\n ])\n} as const satisfies RouteContract;\n\n/**\n * DELETE /_cms/labels/:id - 라벨 삭제\n */\nexport const deleteLabelContract = {\n method: 'DELETE' as const,\n path: '/_cms/labels/:id',\n params: Type.Object({\n id: Type.String({ description: '라벨 ID' })\n }),\n response: Type.Union([\n Type.Object({\n success: Type.Boolean(),\n id: Type.Number()\n }),\n Type.Object({\n error: Type.String()\n })\n ])\n} as const satisfies RouteContract;\n\n/**\n * GET /_cms/labels/by-key/:key - Key로 라벨 조회\n */\nexport const getLabelByKeyContract = {\n method: 'GET' as const,\n path: '/_cms/labels/by-key/:key',\n params: Type.Object({\n key: Type.String({ description: '라벨 Key (예: home.hero.title)' })\n }),\n response: Type.Union([\n Type.Object({\n id: Type.Number(),\n key: Type.String(),\n section: Type.String(),\n type: Type.String(),\n description: Type.Union([Type.String(), Type.Null()]),\n publishedVersion: Type.Union([Type.Number(), Type.Null()]),\n createdBy: Type.Union([Type.String(), Type.Null()]),\n createdAt: Type.String(),\n updatedAt: Type.String()\n }),\n Type.Object({\n error: Type.String(),\n key: Type.Optional(Type.String())\n })\n ])\n} as const satisfies RouteContract;\n\n/**\n * POST /_cms/labels/:id/publish - 라벨 발행 (Draft → Published)\n */\nexport const publishLabelContract = {\n method: 'POST' as const,\n path: '/_cms/labels/:id/publish',\n params: Type.Object({\n id: Type.String({ description: '라벨 ID' })\n }),\n body: Type.Object({\n notes: Type.Optional(Type.String({ description: '발행 노트 (버전 설명)' })),\n publishedBy: Type.Optional(Type.String({ description: '발행자 ID' }))\n }),\n response: Type.Union([\n Type.Object({\n success: Type.Boolean(),\n id: Type.Number(),\n version: Type.Number(),\n message: Type.String()\n }),\n Type.Object({\n error: Type.String()\n })\n ])\n} as const satisfies RouteContract;\n\n/**\n * GET /_cms/labels/:id/admin - 관리자용 라벨 조회 (Draft + Published + Status)\n */\nexport const getAdminLabelContract = {\n method: 'GET' as const,\n path: '/_cms/labels/:id/admin',\n params: Type.Object({\n id: Type.String({ description: '라벨 ID' })\n }),\n response: Type.Union([\n Type.Object({\n label: Type.Object({\n id: Type.Number(),\n key: Type.String(),\n section: Type.String(),\n type: Type.String(),\n description: Type.Union([Type.String(), Type.Null()]),\n publishedVersion: Type.Union([Type.Number(), Type.Null()]),\n createdBy: Type.Union([Type.String(), Type.Null()]),\n createdAt: Type.String(),\n updatedAt: Type.String()\n }),\n draft: Type.Array(Type.Object({\n id: Type.Number(),\n labelId: Type.Number(),\n version: Type.Null(),\n locale: Type.String(),\n breakpoint: Type.Union([Type.String(), Type.Null()]),\n value: Type.Any(),\n createdAt: Type.String()\n })),\n published: Type.Array(Type.Object({\n id: Type.Number(),\n labelId: Type.Number(),\n version: Type.Number(),\n locale: Type.String(),\n breakpoint: Type.Union([Type.String(), Type.Null()]),\n value: Type.Any(),\n createdAt: Type.String()\n })),\n status: Type.Union([\n Type.Literal('default-only'),\n Type.Literal('unpublished'),\n Type.Literal('published'),\n Type.Literal('modified')\n ])\n }),\n Type.Object({\n error: Type.String()\n })\n ])\n} as const satisfies RouteContract;\n\n/**\n * GET /_cms/labels/:id/versions - 라벨 버전 히스토리 조회\n */\nexport const getLabelVersionsContract = {\n method: 'GET' as const,\n path: '/_cms/labels/:id/versions',\n params: Type.Object({\n id: Type.String({ description: '라벨 ID' })\n }),\n response: Type.Union([\n Type.Object({\n versions: Type.Array(Type.Object({\n version: Type.Number({ description: '버전 번호' }),\n publishedAt: Type.String({ description: '발행 시각 (ISO 8601)' }),\n publishedBy: Type.Union([Type.String(), Type.Null()], { description: '발행자 ID' }),\n notes: Type.Union([Type.String(), Type.Null()], { description: '발행 노트' }),\n values: Type.Array(Type.Object({\n id: Type.Number(),\n locale: Type.String(),\n breakpoint: Type.Union([Type.String(), Type.Null()]),\n value: Type.Any(),\n createdAt: Type.String()\n }))\n }))\n }),\n Type.Object({\n error: Type.String()\n })\n ])\n} as const satisfies RouteContract;","/**\n * CMS Labels Repository\n *\n * 라벨 메타데이터 관리를 위한 Repository\n */\n\nimport { findOne, findMany as findManyHelper, create as createHelper, updateOne, deleteOne, count as countHelper } from '@spfn/core/db';\nimport { asc } from 'drizzle-orm';\nimport { cmsLabels, type CmsLabel, type NewCmsLabel } from '@/server/entities';\n\n/**\n * 라벨 목록 조회\n */\nexport async function findMany(options?: {\n section?: string;\n}): Promise<CmsLabel[]>\n{\n const { section } = options || {};\n\n return findManyHelper(cmsLabels, {\n where: section ? { section } : undefined,\n orderBy: asc(cmsLabels.key), // key 오름차순 정렬 (JSON 파일의 순서 유지)\n });\n}\n\n/**\n * 전체 라벨 수 조회\n */\nexport async function count(section?: string): Promise<number>\n{\n return countHelper(cmsLabels, section ? { section } : undefined);\n}\n\n/**\n * ID로 라벨 조회\n */\nexport async function findById(id: number): Promise<CmsLabel | null>\n{\n return findOne(cmsLabels, { id });\n}\n\n/**\n * Key로 라벨 조회\n */\nexport async function findByKey(key: string): Promise<CmsLabel | null>\n{\n return findOne(cmsLabels, { key });\n}\n\n/**\n * 섹션으로 모든 라벨 조회\n */\nexport async function findBySection(section: string): Promise<CmsLabel[]>\n{\n return findManyHelper(cmsLabels, {\n where: { section },\n orderBy: asc(cmsLabels.key), // key 오름차순 정렬 (JSON 파일의 순서 유지)\n });\n}\n\n/**\n * 라벨 생성\n */\nexport async function create(data: NewCmsLabel): Promise<CmsLabel>\n{\n return createHelper(cmsLabels, data);\n}\n\n/**\n * 라벨 수정\n */\nexport async function updateById(id: number, data: Partial<NewCmsLabel>): Promise<CmsLabel | null>\n{\n return updateOne(cmsLabels, { id }, { ...data, updatedAt: new Date() });\n}\n\n/**\n * 라벨 삭제\n */\nexport async function deleteById(id: number): Promise<CmsLabel | null>\n{\n return deleteOne(cmsLabels, { id });\n}\n\n// Legacy export for backward compatibility\nexport const cmsLabelsRepository = {\n findMany,\n count,\n findById,\n findByKey,\n findBySection,\n create,\n updateById,\n deleteById\n};","/**\n * CMS Labels Entity\n *\n * 라벨의 메타데이터와 현재 발행 상태를 관리합니다.\n * - 라벨 식별 (id, key)\n * - 섹션 분류 (section)\n * - 타입 정의 (type)\n * - 발행 상태 (publishedVersion)\n */\n\nimport { index, integer, serial, text, timestamp } from 'drizzle-orm/pg-core';\nimport { createFunctionSchema } from '@spfn/core/db';\n\n// Create isolated schema for @spfn/cms\nconst schema = createFunctionSchema('@spfn/cms');\n\nexport const cmsLabels = schema.table('labels', {\n // Primary Key\n id: serial('id').primaryKey(),\n\n // 라벨 식별자\n key: text('key').notNull().unique(),\n // 예: \"home.hero.title\", \"why-futureplay.hero.subtitle\"\n // 구조: {section}.{component}.{property}\n\n // 섹션 분류 (페이지 단위)\n section: text('section').notNull(),\n // 예: \"home\", \"why-futureplay\", \"team\"\n\n // 값 타입\n type: text('type').notNull(),\n // \"text\" | \"image\" | \"video\" | \"file\" | \"object\"\n\n // 기본값\n defaultValue: text('default_value'),\n // 라벨의 기본값 (sync 시 설정)\n\n // 설명\n description: text('description'),\n // 라벨에 대한 설명 (optional)\n\n // 현재 발행된 버전 번호\n publishedVersion: integer('published_version'),\n // null = 미발행 상태\n // 1, 2, 3... = 발행된 버전 번호\n\n // 생성자 추적\n createdBy: text('created_by'),\n\n // 타임스탬프\n createdAt: timestamp('created_at', { withTimezone: true }).notNull().defaultNow(),\n updatedAt: timestamp('updated_at', { withTimezone: true }).notNull().defaultNow(),\n}, (table) => [\n // 인덱스: 섹션별 조회 최적화\n index('cms_labels_section_idx').on(table.section),\n\n // 인덱스: key로 조회 최적화 (unique 제약으로 자동 생성되지만 명시)\n index('cms_labels_key_idx').on(table.key),\n]);\n\n// 타입 추론\nexport type CmsLabel = typeof cmsLabels.$inferSelect;\nexport type NewCmsLabel = typeof cmsLabels.$inferInsert;","/**\n * CMS Label Values Entity\n *\n * 라벨의 실제 값을 저장합니다.\n * - 다국어 지원 (locale)\n * - 반응형 지원 (breakpoint)\n * - 버전 관리 (version)\n * - JSONB로 유연한 값 저장\n */\n\nimport { serial, integer, text, jsonb, timestamp, index, unique } from 'drizzle-orm/pg-core';\nimport { createFunctionSchema } from '@spfn/core/db';\nimport { cmsLabels } from '@/server/entities/cms-labels';\n\n// Create isolated schema for @spfn/cms\nconst schema = createFunctionSchema('@spfn/cms');\n\nexport const cmsLabelValues = schema.table('label_values', {\n // Primary Key\n id: serial('id').primaryKey(),\n\n // Foreign Key: cms_labels\n labelId: integer('label_id')\n .notNull()\n .references(() => cmsLabels.id, { onDelete: 'cascade' }),\n\n // 버전 번호 (null = draft, number = published version)\n version: integer('version'),\n\n // 언어 코드\n locale: text('locale').notNull().default('ko'),\n // \"ko\" | \"en\" | \"ja\"\n\n // 반응형 브레이크포인트\n breakpoint: text('breakpoint'),\n // null = 기본값 (모든 화면 크기)\n // \"sm\" | \"md\" | \"lg\" | \"xl\" | \"2xl\"\n\n // 실제 값 (JSONB)\n value: jsonb('value').notNull(),\n // LabelValue 타입:\n // - TextValue: { type: \"text\", content: string }\n // - ImageValue: { type: \"image\", url: string, alt?: string, width?: number, height?: number }\n // - VideoValue: { type: \"video\", url: string, thumbnail?: string, duration?: number }\n // - FileValue: { type: \"file\", url: string, filename: string, size?: number }\n // - ObjectValue: { type: \"object\", fields: Record<string, LabelValue> }\n\n // 생성 시각\n createdAt: timestamp('created_at', { withTimezone: true }).notNull().defaultNow(),\n}, (table) => [\n // UNIQUE 제약: 같은 버전에서 locale + breakpoint 조합은 유일\n unique('cms_label_values_locale_breakpoint_unique')\n .on(table.labelId, table.version, table.locale, table.breakpoint),\n\n // 인덱스: labelId + version 복합 조회 최적화\n index('cms_label_values_label_version_idx')\n .on(table.labelId, table.version),\n\n // 인덱스: locale 필터링 최적화\n index('cms_label_values_locale_idx').on(table.locale),\n]);\n\n// 타입 추론\nexport type CmsLabelValue = typeof cmsLabelValues.$inferSelect;\nexport type NewCmsLabelValue = typeof cmsLabelValues.$inferInsert;\n\n/**\n * 사용 예시:\n *\n * // 텍스트 값 저장\n * await db.insert(cmsLabelValues).values({\n * labelId: 1,\n * version: 1,\n * locale: 'ko',\n * breakpoint: null,\n * value: {\n * type: 'text',\n * content: '미래를 만드는 기업'\n * }\n * });\n *\n * // 반응형 이미지 저장 (모바일용)\n * await db.insert(cmsLabelValues).values({\n * labelId: 2,\n * version: 1,\n * locale: 'ko',\n * breakpoint: 'sm',\n * value: {\n * type: 'image',\n * url: '/uploads/hero-mobile.jpg',\n * alt: 'Hero Image',\n * width: 640,\n * height: 480\n * }\n * });\n *\n * // 특정 버전의 한국어 값 조회\n * const values = await db.select()\n * .from(cmsLabelValues)\n * .where(and(\n * eq(cmsLabelValues.labelId, 1),\n * eq(cmsLabelValues.version, 2),\n * eq(cmsLabelValues.locale, 'ko')\n * ));\n *\n * // Object 타입 값 저장 (재귀 구조)\n * await db.insert(cmsLabelValues).values({\n * labelId: 3,\n * version: 1,\n * locale: 'ko',\n * value: {\n * type: 'object',\n * fields: {\n * title: { type: 'text', content: '특징 1' },\n * icon: { type: 'image', url: '/icons/feature1.svg', alt: 'Icon' },\n * description: { type: 'text', content: '상세 설명...' }\n * }\n * }\n * });\n */","/**\n * CMS Draft Cache Entity\n *\n * 관리자별 Draft 콘텐츠를 캐싱합니다.\n * - 사용자별 격리 (userId)\n * - 실시간 미리보기 지원\n * - 동시 편집 가능\n *\n * 핵심 기능:\n * - 여러 관리자가 같은 섹션을 동시에 편집\n * - 각자의 변경사항은 자신의 미리보기에만 표시\n * - 충돌 없이 안전하게 작업\n */\n\nimport { serial, text, jsonb, timestamp, index, unique } from 'drizzle-orm/pg-core';\nimport { createFunctionSchema } from '@spfn/core/db';\n\n// Create isolated schema for @spfn/cms\nconst schema = createFunctionSchema('@spfn/cms');\n\nexport const cmsDraftCache = schema.table('draft_cache', {\n // Primary Key\n id: serial('id').primaryKey(),\n\n // 섹션 (페이지 단위)\n section: text('section').notNull(),\n // \"home\" | \"why-futureplay\" | \"team\" | \"our-companies\" | \"apply\"\n\n // 언어\n locale: text('locale').notNull(),\n // \"ko\" | \"en\" | \"ja\"\n\n // 사용자 ID (핵심 필드!)\n userId: text('user_id').notNull(),\n // 각 관리자의 독립적인 작업 공간\n\n // Draft 콘텐츠 (JSONB)\n content: jsonb('content').notNull(),\n // Record<string, LabelValue>\n // {\n // \"home.hero.title\": { type: \"text\", content: \"수정 중...\" },\n // \"home.hero.subtitle\": { type: \"text\", content: \"새로운 문구\" },\n // ...\n // }\n\n // 최종 수정 시각\n updatedAt: timestamp('updated_at', { withTimezone: true }).notNull().defaultNow(),\n}, (table) => [\n // UNIQUE 제약: section + locale + userId 조합은 유일\n unique('cms_draft_cache_unique')\n .on(table.section, table.locale, table.userId),\n\n // 인덱스: section으로 조회 최적화\n index('cms_draft_cache_section_idx').on(table.section),\n\n // 인덱스: userId로 사용자의 모든 draft 조회 최적화\n index('cms_draft_cache_user_idx').on(table.userId),\n]);\n\n// 타입 추론\nexport type CmsDraftCache = typeof cmsDraftCache.$inferSelect;\nexport type NewCmsDraftCache = typeof cmsDraftCache.$inferInsert;\n\n/**\n * 사용 예시:\n *\n * // Draft 초기화 (편집 시작)\n * await db.insert(cmsDraftCache)\n * .values({\n * section: 'home',\n * locale: 'ko',\n * userId: 'user-a@futureplay.com',\n * content: publishedContent // 발행 버전 복사\n * });\n *\n * // Draft 업데이트 (값 수정 시)\n * const cache = await db.select()\n * .from(cmsDraftCache)\n * .where(and(\n * eq(cmsDraftCache.section, 'home'),\n * eq(cmsDraftCache.locale, 'ko'),\n * eq(cmsDraftCache.userId, userId)\n * ))\n * .limit(1);\n *\n * const updatedContent = {\n * ...cache[0].content,\n * 'home.hero.title': newValue // 부분 업데이트\n * };\n *\n * await db.update(cmsDraftCache)\n * .set({ content: updatedContent, updatedAt: new Date() })\n * .where(eq(cmsDraftCache.id, cache[0].id));\n *\n * // Draft 조회 (미리보기)\n * const draft = await db.select()\n * .from(cmsDraftCache)\n * .where(and(\n * eq(cmsDraftCache.section, 'home'),\n * eq(cmsDraftCache.locale, 'ko'),\n * eq(cmsDraftCache.userId, session.user.id)\n * ))\n * .limit(1);\n *\n * // 사용자의 모든 작업 중인 섹션 조회\n * const userDrafts = await db.select()\n * .from(cmsDraftCache)\n * .where(eq(cmsDraftCache.userId, userId))\n * .orderBy(desc(cmsDraftCache.updatedAt));\n *\n * // 오래된 Draft 정리 (30일 이상)\n * const stale = await db.delete(cmsDraftCache)\n * .where(lt(\n * cmsDraftCache.updatedAt,\n * new Date(Date.now() - 30 * 24 * 60 * 60 * 1000)\n * ))\n * .returning();\n *\n * // Draft 폐기 (변경사항 버리기)\n * await db.delete(cmsDraftCache)\n * .where(and(\n * eq(cmsDraftCache.section, 'home'),\n * eq(cmsDraftCache.locale, 'ko'),\n * eq(cmsDraftCache.userId, userId)\n * ));\n */","/**\n * CMS Published Cache Entity\n *\n * 발행된 콘텐츠를 섹션+언어 단위로 캐싱합니다.\n * - 초고속 읽기 성능 (5ms)\n * - 단일 쿼리로 섹션 전체 로드\n * - JSONB로 즉시 사용 가능한 데이터\n *\n * 성능 비교:\n * - 정규화 테이블 JOIN: 87ms\n * - 캐시 테이블: 5ms (17배 빠름!)\n */\n\nimport { serial, text, jsonb, integer, timestamp, index, unique } from 'drizzle-orm/pg-core';\nimport { createFunctionSchema } from '@spfn/core/db';\n\n// Create isolated schema for @spfn/cms\nconst schema = createFunctionSchema('@spfn/cms');\n\nexport const cmsPublishedCache = schema.table('published_cache', {\n // Primary Key\n id: serial('id').primaryKey(),\n\n // 섹션 (페이지 단위)\n section: text('section').notNull(),\n // \"home\" | \"why-futureplay\" | \"team\" | \"our-companies\" | \"apply\"\n\n // 언어\n locale: text('locale').notNull(),\n // \"ko\" | \"en\" | \"ja\"\n\n // 캐시된 콘텐츠 (JSONB)\n content: jsonb('content').notNull(),\n // Record<string, LabelValue>\n // {\n // \"home.hero.title\": { type: \"text\", content: \"...\" },\n // \"home.hero.image\": { type: \"image\", url: \"...\", alt: \"...\" },\n // ...\n // }\n\n // 발행 정보\n publishedAt: timestamp('published_at', { withTimezone: true }).notNull(),\n publishedBy: text('published_by'),\n\n // 캐시 버전 (클라이언트 캐싱용)\n version: integer('version').notNull().default(1),\n}, (table) => [\n // UNIQUE 제약: section + locale 조합은 유일\n unique('cms_published_cache_unique').on(table.section, table.locale),\n\n // 인덱스: section으로 조회 최적화\n index('cms_published_cache_section_idx').on(table.section),\n]);\n\n// 타입 추론\nexport type CmsPublishedCache = typeof cmsPublishedCache.$inferSelect;\nexport type NewCmsPublishedCache = typeof cmsPublishedCache.$inferInsert;\n\n/**\n * 사용 예시:\n *\n * // 캐시 생성/업데이트 (UPSERT)\n * await db.insert(cmsPublishedCache)\n * .values({\n * section: 'home',\n * locale: 'ko',\n * content: {\n * 'home.hero.title': {\n * type: 'text',\n * content: '미래를 만드는 기업'\n * },\n * 'home.hero.image': {\n * type: 'image',\n * url: '/uploads/hero.jpg',\n * alt: 'Hero',\n * width: 1920,\n * height: 1080\n * }\n * },\n * publishedAt: new Date(),\n * publishedBy: 'admin@futureplay.com'\n * })\n * .onConflictDoUpdate({\n * target: [cmsPublishedCache.section, cmsPublishedCache.locale],\n * set: {\n * content: sql`EXCLUDED.content`,\n * publishedAt: sql`EXCLUDED.published_at`,\n * publishedBy: sql`EXCLUDED.published_by`,\n * version: sql`${cmsPublishedCache.version} + 1`\n * }\n * });\n *\n * // 캐시 조회 (초고속!)\n * const cache = await db.select()\n * .from(cmsPublishedCache)\n * .where(and(\n * eq(cmsPublishedCache.section, 'home'),\n * eq(cmsPublishedCache.locale, 'ko')\n * ))\n * .limit(1);\n *\n * const labels = cache[0].content; // 즉시 사용 가능!\n *\n * // 섹션의 모든 언어 캐시 조회\n * const allLocales = await db.select()\n * .from(cmsPublishedCache)\n * .where(eq(cmsPublishedCache.section, 'home'));\n *\n * // 오래된 캐시 감지\n * const stale = await db.select()\n * .from(cmsPublishedCache)\n * .where(lt(\n * cmsPublishedCache.publishedAt,\n * new Date(Date.now() - 7 * 24 * 60 * 60 * 1000)\n * ));\n */","/**\n * CMS Audit Logs Entity\n *\n * CMS의 모든 변경사항을 추적합니다.\n * - 누가 (userId, userName)\n * - 언제 (createdAt)\n * - 무엇을 (action, changes)\n * - 왜 (metadata)\n */\n\nimport { serial, integer, text, jsonb, timestamp, index } from 'drizzle-orm/pg-core';\nimport { createFunctionSchema } from '@spfn/core/db';\nimport { cmsLabels } from '@/server/entities/cms-labels';\n\n// Create isolated schema for @spfn/cms\nconst schema = createFunctionSchema('@spfn/cms');\n\nexport const cmsAuditLogs = schema.table('audit_logs', {\n // Primary Key\n id: serial('id').primaryKey(),\n\n // Foreign Key: cms_labels (nullable - 라벨 삭제 시 로그는 유지)\n labelId: integer('label_id')\n .references(() => cmsLabels.id, { onDelete: 'set null' }),\n\n // 작업 유형\n action: text('action').notNull(),\n // \"create\" | \"update\" | \"publish\" | \"unpublish\" | \"archive\" | \"delete\" | \"rollback\" | \"duplicate\"\n\n // 사용자 정보\n userId: text('user_id').notNull(),\n userName: text('user_name'),\n\n // 변경 내용 (before/after)\n changes: jsonb('changes'),\n // { before: {...}, after: {...} }\n\n // 추가 메타데이터\n metadata: jsonb('metadata'),\n // { version: number, ip: string, userAgent: string, ... }\n\n // 작업 시각\n createdAt: timestamp('created_at', { withTimezone: true }).notNull().defaultNow(),\n}, (table) => [\n // 인덱스: labelId로 이력 조회 최적화\n index('cms_audit_logs_label_id_idx').on(table.labelId),\n\n // 인덱스: userId로 사용자 활동 조회 최적화\n index('cms_audit_logs_user_id_idx').on(table.userId),\n\n // 인덱스: action 필터링 최적화\n index('cms_audit_logs_action_idx').on(table.action),\n\n // 인덱스: 시간순 조회 최적화\n index('cms_audit_logs_created_at_idx').on(table.createdAt),\n]);\n\n// 타입 추론\nexport type CmsAuditLog = typeof cmsAuditLogs.$inferSelect;\nexport type NewCmsAuditLog = typeof cmsAuditLogs.$inferInsert;\n\n/**\n * 사용 예시:\n *\n * // 라벨 생성 로그\n * await db.insert(cmsAuditLogs).values({\n * labelId: 1,\n * action: 'create',\n * userId: 'user123',\n * userName: '김철수',\n * changes: {\n * before: null,\n * after: {\n * key: 'home.hero.title',\n * section: 'home',\n * type: 'text'\n * }\n * },\n * metadata: {\n * ip: '192.168.1.1',\n * userAgent: 'Mozilla/5.0...'\n * }\n * });\n *\n * // 발행 로그\n * await db.insert(cmsAuditLogs).values({\n * labelId: 1,\n * action: 'publish',\n * userId: 'admin123',\n * userName: '관리자',\n * changes: {\n * before: { status: 'draft', publishedVersion: null },\n * after: { status: 'published', publishedVersion: 2 }\n * },\n * metadata: {\n * version: 2,\n * notes: '신규 브랜딩 적용'\n * }\n * });\n *\n * // 라벨별 이력 조회\n * const logs = await db.select()\n * .from(cmsAuditLogs)\n * .where(eq(cmsAuditLogs.labelId, 1))\n * .orderBy(desc(cmsAuditLogs.createdAt))\n * .limit(20);\n *\n * // 사용자별 활동 조회\n * const userActivity = await db.select()\n * .from(cmsAuditLogs)\n * .where(eq(cmsAuditLogs.userId, 'user123'))\n * .orderBy(desc(cmsAuditLogs.createdAt));\n *\n * // 최근 24시간 변경 이력\n * const recent = await db.select()\n * .from(cmsAuditLogs)\n * .where(gte(cmsAuditLogs.createdAt, new Date(Date.now() - 24 * 60 * 60 * 1000)))\n * .orderBy(desc(cmsAuditLogs.createdAt));\n */"],"mappings":"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","names":["serial","integer","text","timestamp","index","createFunctionSchema","schema","createFunctionSchema","serial","integer","text","timestamp","index","serial","text","jsonb","timestamp","index","unique","createFunctionSchema","schema","serial","text","jsonb","integer","timestamp","index","unique","createFunctionSchema","schema","serial","integer","text","jsonb","timestamp","index","createFunctionSchema","schema","createFunctionSchema","serial","integer","text","jsonb","timestamp","index"]}
@@ -0,0 +1,11 @@
1
+ import * as _spfn_core_route from '@spfn/core/route';
2
+
3
+ /**
4
+ * CMS Label Admin Route
5
+ *
6
+ * 관리자용 라벨 조회 API (Draft + Published + Status)
7
+ * - GET /labels/:id/admin - 라벨 상세 정보 조회 (final: /_cms/labels/:id/admin)
8
+ */
9
+ declare const app: _spfn_core_route.SPFNApp;
10
+
11
+ export { app as default };