@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.
- package/README.md +28 -416
- package/dist/{helpers/locale.actions.d.ts → actions-BEFWwQsh.d.ts} +70 -7
- package/dist/actions.d.ts +2 -9
- package/dist/actions.js +99 -10
- package/dist/actions.js.map +1 -1
- package/dist/api.d.ts +319 -0
- package/dist/api.js +467 -0
- package/dist/api.js.map +1 -0
- package/dist/client.d.ts +135 -127
- package/dist/client.js +1318 -59
- package/dist/client.js.map +1 -1
- package/dist/{types.d.ts → index-Dh5FjWzR.d.ts} +45 -7
- package/dist/index.d.ts +112 -16
- package/dist/index.js +625 -23
- package/dist/index.js.map +1 -1
- package/dist/label-sync-generator-B0EmvtWM.d.ts +32 -0
- package/dist/lib/contracts/labels.d.ts +244 -0
- package/dist/lib/contracts/labels.js +269 -0
- package/dist/lib/contracts/labels.js.map +1 -0
- package/dist/lib/contracts/published-cache.d.ts +48 -0
- package/dist/lib/contracts/published-cache.js +49 -0
- package/dist/lib/contracts/published-cache.js.map +1 -0
- package/dist/lib/contracts/values.d.ts +71 -0
- package/dist/lib/contracts/values.js +104 -0
- package/dist/lib/contracts/values.js.map +1 -0
- package/dist/locale.constants-BNkSdNP1.d.ts +108 -0
- package/dist/{entities → server/entities}/cms-audit-logs.d.ts +15 -70
- package/dist/server/entities/cms-audit-logs.js +78 -0
- package/dist/server/entities/cms-audit-logs.js.map +1 -0
- package/dist/{entities → server/entities}/cms-draft-cache.d.ts +13 -73
- package/dist/server/entities/cms-draft-cache.js +38 -0
- package/dist/server/entities/cms-draft-cache.js.map +1 -0
- package/dist/{entities → server/entities}/cms-label-values.d.ts +16 -67
- package/dist/server/entities/cms-label-values.js +81 -0
- package/dist/server/entities/cms-label-values.js.map +1 -0
- package/dist/{entities → server/entities}/cms-labels.d.ts +17 -14
- package/dist/server/entities/cms-labels.js +42 -0
- package/dist/server/entities/cms-labels.js.map +1 -0
- package/dist/{entities → server/entities}/cms-published-cache.d.ts +14 -69
- package/dist/server/entities/cms-published-cache.js +36 -0
- package/dist/server/entities/cms-published-cache.js.map +1 -0
- package/dist/server/entities/index.d.ts +6 -0
- package/dist/server/entities/index.js +185 -0
- package/dist/server/entities/index.js.map +1 -0
- package/dist/server/generators/index.d.ts +19 -0
- package/dist/server/generators/index.js +731 -0
- package/dist/server/generators/index.js.map +1 -0
- package/dist/server/labels/index.d.ts +1 -0
- package/dist/server/labels/index.js +33 -0
- package/dist/server/labels/index.js.map +1 -0
- package/dist/server/repositories/index.d.ts +212 -0
- package/dist/server/repositories/index.js +418 -0
- package/dist/server/repositories/index.js.map +1 -0
- package/dist/server/routes/labels/[id]/admin/index.js +679 -0
- package/dist/server/routes/labels/[id]/admin/index.js.map +1 -0
- package/dist/server/routes/labels/[id]/index.js +576 -0
- package/dist/server/routes/labels/[id]/index.js.map +1 -0
- package/dist/server/routes/labels/[id]/publish/index.js +720 -0
- package/dist/server/routes/labels/[id]/publish/index.js.map +1 -0
- package/dist/server/routes/labels/[id]/versions/index.js +548 -0
- package/dist/server/routes/labels/[id]/versions/index.js.map +1 -0
- package/dist/server/routes/labels/_id_/admin/index.d.ts +11 -0
- package/dist/{routes/labels/[id] → server/routes/labels/_id_}/index.d.ts +5 -3
- package/dist/server/routes/labels/_id_/publish/index.d.ts +11 -0
- package/dist/server/routes/labels/_id_/versions/index.d.ts +11 -0
- package/dist/server/routes/labels/by-key/[key]/index.js +525 -0
- package/dist/server/routes/labels/by-key/[key]/index.js.map +1 -0
- package/dist/server/routes/labels/by-key/_key_/index.d.ts +10 -0
- package/dist/server/routes/labels/index.d.ts +12 -0
- package/dist/server/routes/labels/index.js +684 -0
- package/dist/server/routes/labels/index.js.map +1 -0
- package/dist/server/routes/published-cache/index.d.ts +11 -0
- package/dist/server/routes/published-cache/index.js +337 -0
- package/dist/server/routes/published-cache/index.js.map +1 -0
- package/dist/server/routes/values/[labelId]/[version]/index.js +457 -0
- package/dist/server/routes/values/[labelId]/[version]/index.js.map +1 -0
- package/dist/server/routes/values/[labelId]/index.js +452 -0
- package/dist/server/routes/values/[labelId]/index.js.map +1 -0
- package/dist/server/routes/values/_labelId_/_version_/index.d.ts +10 -0
- package/dist/server/routes/values/_labelId_/index.d.ts +10 -0
- package/dist/server.d.ts +77 -7
- package/dist/server.js +1747 -247
- package/dist/server.js.map +1 -1
- package/migrations/0000_init.sql +3 -0
- package/migrations/0001_far_lady_vermin.sql +86 -0
- package/migrations/0002_heavy_the_enforcers.sql +2 -0
- package/migrations/0003_rare_runaways.sql +1 -0
- package/migrations/meta/0000_snapshot.json +15 -0
- package/migrations/meta/0001_snapshot.json +687 -0
- package/migrations/meta/0002_snapshot.json +686 -0
- package/migrations/meta/0003_snapshot.json +563 -0
- package/migrations/meta/_journal.json +34 -0
- package/package.json +55 -36
- package/dist/actions.d.ts.map +0 -1
- package/dist/client.d.ts.map +0 -1
- package/dist/cms.config.d.ts +0 -77
- package/dist/cms.config.d.ts.map +0 -1
- package/dist/cms.config.js +0 -111
- package/dist/cms.config.js.map +0 -1
- package/dist/entities/cms-audit-logs.d.ts.map +0 -1
- package/dist/entities/cms-audit-logs.js +0 -103
- package/dist/entities/cms-audit-logs.js.map +0 -1
- package/dist/entities/cms-draft-cache.d.ts.map +0 -1
- package/dist/entities/cms-draft-cache.js +0 -112
- package/dist/entities/cms-draft-cache.js.map +0 -1
- package/dist/entities/cms-label-values.d.ts.map +0 -1
- package/dist/entities/cms-label-values.js +0 -105
- package/dist/entities/cms-label-values.js.map +0 -1
- package/dist/entities/cms-label-versions.d.ts +0 -207
- package/dist/entities/cms-label-versions.d.ts.map +0 -1
- package/dist/entities/cms-label-versions.js +0 -80
- package/dist/entities/cms-label-versions.js.map +0 -1
- package/dist/entities/cms-labels.d.ts.map +0 -1
- package/dist/entities/cms-labels.js +0 -48
- package/dist/entities/cms-labels.js.map +0 -1
- package/dist/entities/cms-published-cache.d.ts.map +0 -1
- package/dist/entities/cms-published-cache.js +0 -103
- package/dist/entities/cms-published-cache.js.map +0 -1
- package/dist/entities/index.d.ts +0 -10
- package/dist/entities/index.d.ts.map +0 -1
- package/dist/entities/index.js +0 -10
- package/dist/entities/index.js.map +0 -1
- package/dist/generators/index.d.ts +0 -19
- package/dist/generators/index.d.ts.map +0 -1
- package/dist/generators/index.js +0 -19
- package/dist/generators/index.js.map +0 -1
- package/dist/generators/label-sync-generator.d.ts +0 -33
- package/dist/generators/label-sync-generator.d.ts.map +0 -1
- package/dist/generators/label-sync-generator.js +0 -86
- package/dist/generators/label-sync-generator.js.map +0 -1
- package/dist/helpers/locale.actions.d.ts.map +0 -1
- package/dist/helpers/locale.actions.js +0 -210
- package/dist/helpers/locale.actions.js.map +0 -1
- package/dist/helpers/locale.constants.d.ts +0 -10
- package/dist/helpers/locale.constants.d.ts.map +0 -1
- package/dist/helpers/locale.constants.js +0 -10
- package/dist/helpers/locale.constants.js.map +0 -1
- package/dist/helpers/locale.d.ts +0 -17
- package/dist/helpers/locale.d.ts.map +0 -1
- package/dist/helpers/locale.js +0 -20
- package/dist/helpers/locale.js.map +0 -1
- package/dist/helpers/sync.d.ts +0 -41
- package/dist/helpers/sync.d.ts.map +0 -1
- package/dist/helpers/sync.js +0 -309
- package/dist/helpers/sync.js.map +0 -1
- package/dist/index.d.ts.map +0 -1
- package/dist/init.d.ts +0 -31
- package/dist/init.d.ts.map +0 -1
- package/dist/init.js +0 -36
- package/dist/init.js.map +0 -1
- package/dist/labels/helpers.d.ts +0 -31
- package/dist/labels/helpers.d.ts.map +0 -1
- package/dist/labels/helpers.js +0 -60
- package/dist/labels/helpers.js.map +0 -1
- package/dist/labels/index.d.ts +0 -7
- package/dist/labels/index.d.ts.map +0 -1
- package/dist/labels/index.js +0 -7
- package/dist/labels/index.js.map +0 -1
- package/dist/repositories/cms-draft-cache.repository.d.ts +0 -62
- package/dist/repositories/cms-draft-cache.repository.d.ts.map +0 -1
- package/dist/repositories/cms-draft-cache.repository.js +0 -56
- package/dist/repositories/cms-draft-cache.repository.js.map +0 -1
- package/dist/repositories/cms-label-values.repository.d.ts +0 -32
- package/dist/repositories/cms-label-values.repository.d.ts.map +0 -1
- package/dist/repositories/cms-label-values.repository.js +0 -72
- package/dist/repositories/cms-label-values.repository.js.map +0 -1
- package/dist/repositories/cms-labels.repository.d.ts +0 -53
- package/dist/repositories/cms-labels.repository.d.ts.map +0 -1
- package/dist/repositories/cms-labels.repository.js +0 -77
- package/dist/repositories/cms-labels.repository.js.map +0 -1
- package/dist/repositories/cms-published-cache.repository.d.ts +0 -53
- package/dist/repositories/cms-published-cache.repository.d.ts.map +0 -1
- package/dist/repositories/cms-published-cache.repository.js +0 -54
- package/dist/repositories/cms-published-cache.repository.js.map +0 -1
- package/dist/repositories/index.d.ts +0 -8
- package/dist/repositories/index.d.ts.map +0 -1
- package/dist/repositories/index.js +0 -9
- package/dist/repositories/index.js.map +0 -1
- package/dist/routes/labels/[id]/contract.d.ts +0 -68
- package/dist/routes/labels/[id]/contract.d.ts.map +0 -1
- package/dist/routes/labels/[id]/contract.js +0 -84
- package/dist/routes/labels/[id]/contract.js.map +0 -1
- package/dist/routes/labels/[id]/index.d.ts.map +0 -1
- package/dist/routes/labels/[id]/index.js +0 -96
- package/dist/routes/labels/[id]/index.js.map +0 -1
- package/dist/routes/labels/by-key/[key]/contract.d.ts +0 -24
- package/dist/routes/labels/by-key/[key]/contract.d.ts.map +0 -1
- package/dist/routes/labels/by-key/[key]/contract.js +0 -28
- package/dist/routes/labels/by-key/[key]/contract.js.map +0 -1
- package/dist/routes/labels/by-key/[key]/index.d.ts +0 -8
- package/dist/routes/labels/by-key/[key]/index.d.ts.map +0 -1
- package/dist/routes/labels/by-key/[key]/index.js +0 -32
- package/dist/routes/labels/by-key/[key]/index.js.map +0 -1
- package/dist/routes/labels/contract.d.ts +0 -59
- package/dist/routes/labels/contract.d.ts.map +0 -1
- package/dist/routes/labels/contract.js +0 -75
- package/dist/routes/labels/contract.js.map +0 -1
- package/dist/routes/labels/index.d.ts +0 -10
- package/dist/routes/labels/index.d.ts.map +0 -1
- package/dist/routes/labels/index.js +0 -73
- package/dist/routes/labels/index.js.map +0 -1
- package/dist/routes/published-cache/contract.d.ts +0 -25
- package/dist/routes/published-cache/contract.d.ts.map +0 -1
- package/dist/routes/published-cache/contract.js +0 -35
- package/dist/routes/published-cache/contract.js.map +0 -1
- package/dist/routes/published-cache/index.d.ts +0 -8
- package/dist/routes/published-cache/index.d.ts.map +0 -1
- package/dist/routes/published-cache/index.js +0 -33
- package/dist/routes/published-cache/index.js.map +0 -1
- package/dist/routes/values/[labelId]/[version]/contract.d.ts +0 -29
- package/dist/routes/values/[labelId]/[version]/contract.d.ts.map +0 -1
- package/dist/routes/values/[labelId]/[version]/contract.js +0 -33
- package/dist/routes/values/[labelId]/[version]/contract.js.map +0 -1
- package/dist/routes/values/[labelId]/[version]/index.d.ts +0 -8
- package/dist/routes/values/[labelId]/[version]/index.d.ts.map +0 -1
- package/dist/routes/values/[labelId]/[version]/index.js +0 -45
- package/dist/routes/values/[labelId]/[version]/index.js.map +0 -1
- package/dist/routes/values/[labelId]/contract.d.ts +0 -38
- package/dist/routes/values/[labelId]/contract.d.ts.map +0 -1
- package/dist/routes/values/[labelId]/contract.js +0 -59
- package/dist/routes/values/[labelId]/contract.js.map +0 -1
- package/dist/routes/values/[labelId]/index.d.ts +0 -8
- package/dist/routes/values/[labelId]/index.d.ts.map +0 -1
- package/dist/routes/values/[labelId]/index.js +0 -42
- package/dist/routes/values/[labelId]/index.js.map +0 -1
- package/dist/server.d.ts.map +0 -1
- package/dist/store.d.ts +0 -87
- package/dist/store.d.ts.map +0 -1
- package/dist/store.js +0 -205
- package/dist/store.js.map +0 -1
- package/dist/types.d.ts.map +0 -1
- package/dist/types.js +0 -7
- 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 };
|