@spfn/cms 0.1.0-alpha.6 → 0.1.0-alpha.60
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/dist/actions.d.ts +140 -6
- package/dist/actions.js +97 -10
- package/dist/actions.js.map +1 -1
- package/dist/client.d.ts +43 -9
- package/dist/client.js +406 -56
- package/dist/client.js.map +1 -1
- package/dist/contracts/labels.d.ts +149 -0
- package/dist/contracts/labels.js +166 -0
- package/dist/contracts/labels.js.map +1 -0
- package/dist/contracts/published-cache.d.ts +25 -0
- package/dist/contracts/published-cache.js +32 -0
- package/dist/contracts/published-cache.js.map +1 -0
- package/dist/contracts/values.d.ts +69 -0
- package/dist/contracts/values.js +100 -0
- package/dist/contracts/values.js.map +1 -0
- package/dist/entities/cms-audit-logs.d.ts +15 -70
- package/dist/entities/cms-audit-logs.js +75 -100
- package/dist/entities/cms-audit-logs.js.map +1 -1
- package/dist/entities/cms-draft-cache.d.ts +13 -73
- package/dist/entities/cms-draft-cache.js +35 -109
- package/dist/entities/cms-draft-cache.js.map +1 -1
- package/dist/entities/cms-label-values.d.ts +14 -65
- package/dist/entities/cms-label-values.js +78 -102
- package/dist/entities/cms-label-values.js.map +1 -1
- package/dist/entities/cms-label-versions.d.ts +16 -49
- package/dist/entities/cms-label-versions.js +73 -77
- package/dist/entities/cms-label-versions.js.map +1 -1
- package/dist/entities/cms-labels.d.ts +17 -14
- package/dist/entities/cms-labels.js +39 -45
- package/dist/entities/cms-labels.js.map +1 -1
- package/dist/entities/cms-published-cache.d.ts +14 -69
- package/dist/entities/cms-published-cache.js +33 -100
- package/dist/entities/cms-published-cache.js.map +1 -1
- package/dist/entities/index.d.ts +7 -10
- package/dist/entities/index.js +217 -9
- package/dist/entities/index.js.map +1 -1
- package/dist/generators/index.d.ts +8 -7
- package/dist/generators/index.js +657 -17
- package/dist/generators/index.js.map +1 -1
- package/dist/index.d.ts +134 -20
- package/dist/index.js +1115 -23
- package/dist/index.js.map +1 -1
- package/dist/{generators/label-sync-generator.d.ts → label-sync-generator-lQrcVfja.d.ts} +9 -6
- package/dist/labels/index.d.ts +31 -4
- package/dist/labels/index.js +31 -6
- package/dist/labels/index.js.map +1 -1
- package/dist/repositories/index.d.ts +205 -6
- package/dist/repositories/index.js +435 -8
- package/dist/repositories/index.js.map +1 -1
- package/dist/routes/labels/[id]/index.js +499 -89
- package/dist/routes/labels/[id]/index.js.map +1 -1
- package/dist/routes/labels/{[id] → _id_}/index.d.ts +5 -3
- package/dist/routes/labels/by-key/[key]/index.js +453 -29
- package/dist/routes/labels/by-key/[key]/index.js.map +1 -1
- package/dist/routes/labels/by-key/_key_/index.d.ts +10 -0
- package/dist/routes/labels/index.d.ts +5 -3
- package/dist/routes/labels/index.js +488 -68
- package/dist/routes/labels/index.js.map +1 -1
- package/dist/routes/published-cache/index.d.ts +5 -3
- package/dist/routes/published-cache/index.js +325 -30
- package/dist/routes/published-cache/index.js.map +1 -1
- package/dist/routes/values/[labelId]/[version]/index.js +467 -41
- package/dist/routes/values/[labelId]/[version]/index.js.map +1 -1
- package/dist/routes/values/[labelId]/index.js +463 -39
- package/dist/routes/values/[labelId]/index.js.map +1 -1
- package/dist/routes/values/_labelId_/_version_/index.d.ts +10 -0
- package/dist/routes/values/_labelId_/index.d.ts +10 -0
- package/dist/server.d.ts +17 -7
- package/dist/server.js +263 -252
- package/dist/server.js.map +1 -1
- package/dist/store.d.ts +8 -14
- package/dist/store.js +396 -198
- package/dist/store.js.map +1 -1
- package/dist/types.d.ts +14 -7
- package/dist/types.js +0 -6
- package/dist/types.js.map +1 -1
- package/migrations/0000_condemned_centennial.sql +89 -0
- package/migrations/meta/0000_snapshot.json +687 -0
- package/migrations/meta/_journal.json +13 -0
- package/package.json +33 -16
- 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-draft-cache.d.ts.map +0 -1
- package/dist/entities/cms-label-values.d.ts.map +0 -1
- package/dist/entities/cms-label-versions.d.ts.map +0 -1
- package/dist/entities/cms-labels.d.ts.map +0 -1
- package/dist/entities/cms-published-cache.d.ts.map +0 -1
- package/dist/entities/index.d.ts.map +0 -1
- package/dist/generators/index.d.ts.map +0 -1
- package/dist/generators/label-sync-generator.d.ts.map +0 -1
- package/dist/generators/label-sync-generator.js +0 -87
- package/dist/generators/label-sync-generator.js.map +0 -1
- package/dist/helpers/locale.actions.d.ts +0 -132
- 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.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.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/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/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.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.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]/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/server.d.ts.map +0 -1
- package/dist/store.d.ts.map +0 -1
- package/dist/types.d.ts.map +0 -1
package/dist/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,aAAa,CAAC;AAErB;;;;;;;GAOG;AAEH,oBAAoB;AACpB,OAAO,EAAE,YAAY,EAAE,YAAY,EAAE,cAAc,EAAE,MAAM,iBAAiB,CAAC;AAG7E,kDAAkD;AAClD,cAAc,UAAU,CAAC;AAGzB,oCAAoC;AACpC,cAAc,gBAAgB,CAAC;AAE/B,iCAAiC;AACjC,cAAc,YAAY,CAAC;AAE3B,wDAAwD;AACxD,OAAO,EAAE,WAAW,EAAE,OAAO,EAAE,aAAa,EAAE,kBAAkB,EAAE,MAAM,gBAAgB,CAAC;AAEzF,sDAAsD;AACtD,cAAc,UAAU,CAAC;AAEzB,gDAAgD;AAChD,OAAO,EAAE,wBAAwB,EAAE,kBAAkB,EAAE,MAAM,mCAAmC,CAAC"}
|
|
1
|
+
{"version":3,"sources":["../src/cms.config.ts","../src/server.ts","../src/contracts/published-cache.ts","../src/helpers/locale.actions.ts","../src/helpers/locale.constants.ts","../src/repositories/cms-labels.repository.ts","../src/entities/cms-labels.ts","../src/entities/cms-label-values.ts","../src/entities/cms-label-versions.ts","../src/entities/cms-draft-cache.ts","../src/entities/cms-published-cache.ts","../src/entities/cms-audit-logs.ts","../src/repositories/cms-label-values.repository.ts","../src/repositories/cms-draft-cache.repository.ts","../src/repositories/cms-published-cache.repository.ts","../src/helpers/sync.ts","../src/labels/helpers.ts","../src/generators/label-sync-generator.ts"],"sourcesContent":["/**\n * CMS Configuration Module\n *\n * 환경변수 기반 CMS 설정 관리\n * - SPFN_CMS_DEFAULT_LOCALE: 기본 언어 (기본값: 'ko')\n * - SPFN_CMS_SUPPORTED_LOCALES: 지원 언어 목록, 쉼표로 구분 (기본값: 'ko,en')\n * - SPFN_CMS_DETECT_BROWSER_LANGUAGE: 브라우저 언어 자동 감지 (기본값: 'false')\n */\n\n/**\n * CMS 설정 타입\n */\nexport interface CmsConfig\n{\n /**\n * 기본 언어 코드\n * @example 'ko', 'en', 'ja'\n */\n defaultLocale: string;\n\n /**\n * 지원하는 언어 목록\n * @example ['ko', 'en', 'ja']\n */\n supportedLocales: string[];\n\n /**\n * 브라우저 언어 자동 감지 여부\n * @default false\n */\n detectBrowserLanguage: boolean;\n}\n\n/**\n * 환경변수 읽기 헬퍼\n */\nfunction getEnvVar(key: string, defaultValue: string): string\n{\n return process.env[key] || defaultValue;\n}\n\n/**\n * 환경변수에서 boolean 읽기\n */\nfunction getEnvBoolean(key: string, defaultValue: boolean): boolean\n{\n const value = process.env[key];\n if (value === undefined) return defaultValue;\n return value === 'true' || value === '1';\n}\n\n/**\n * 환경변수에서 설정 로드\n */\nfunction loadConfigFromEnv(): CmsConfig\n{\n const defaultLocale = getEnvVar('SPFN_CMS_DEFAULT_LOCALE', 'en');\n const supportedLocalesStr = getEnvVar('SPFN_CMS_SUPPORTED_LOCALES', 'en,ko');\n const detectBrowserLanguage = getEnvBoolean('SPFN_CMS_DETECT_BROWSER_LANGUAGE', true);\n\n const supportedLocales = supportedLocalesStr\n .split(',')\n .map(locale => locale.trim())\n .filter(locale => locale.length > 0);\n\n // 기본 언어가 지원 목록에 없으면 추가\n if (!supportedLocales.includes(defaultLocale))\n {\n supportedLocales.unshift(defaultLocale);\n }\n\n return {\n defaultLocale,\n supportedLocales,\n detectBrowserLanguage,\n };\n}\n\n/**\n * 현재 설정 (환경변수에서 초기화)\n */\nlet currentConfig: CmsConfig = loadConfigFromEnv();\n\n/**\n * CMS 설정 조회\n *\n * @returns 현재 CMS 설정\n *\n * @example\n * ```tsx\n * import { getCmsConfig } from '@spfn/cms';\n *\n * const config = getCmsConfig();\n * console.log(config.defaultLocale); // 'ko'\n * console.log(config.supportedLocales); // ['ko', 'en']\n * ```\n */\nexport function getCmsConfig(): Readonly<CmsConfig>\n{\n return currentConfig;\n}\n\n/**\n * CMS 설정 변경 (런타임 오버라이드)\n *\n * 환경변수 설정을 런타임에 오버라이드합니다.\n * 주로 테스트나 특수한 경우에 사용됩니다.\n *\n * @param config - 변경할 설정 (부분 업데이트 가능)\n *\n * @example\n * ```tsx\n * import { configureCms } from '@spfn/cms';\n *\n * // 앱 초기화 시 (선택적)\n * configureCms({\n * defaultLocale: 'en',\n * supportedLocales: ['en', 'ko', 'ja'],\n * detectBrowserLanguage: true,\n * });\n * ```\n */\nexport function configureCms(config: Partial<CmsConfig>): void\n{\n currentConfig = {\n ...currentConfig,\n ...config,\n };\n\n // 기본 언어가 지원 목록에 있는지 확인\n if (config.defaultLocale && !currentConfig.supportedLocales.includes(config.defaultLocale))\n {\n console.warn(\n `[CMS Config] Default locale '${config.defaultLocale}' not in supported locales, adding automatically.`,\n `Supported locales: [${currentConfig.supportedLocales.join(', ')}]`\n );\n\n currentConfig.supportedLocales.unshift(config.defaultLocale);\n }\n}\n\n/**\n * 설정 초기화 (환경변수에서 재로드)\n *\n * @example\n * ```tsx\n * import { resetCmsConfig } from '@spfn/cms';\n *\n * // 환경변수 설정으로 되돌리기\n * resetCmsConfig();\n * ```\n */\nexport function resetCmsConfig(): void\n{\n currentConfig = loadConfigFromEnv();\n}","/**\n * CMS Server Module\n *\n * Next.js 서버 컴포넌트용 CMS 유틸리티\n * - React cache를 사용한 데이터 중복 제거\n * - SPFN API를 통한 contract-based 호출\n * - 변수 치환 지원\n * - 쿠키 기반 locale 자동 관리\n *\n * @note This module should only be imported in server-side code\n */\n\nimport { cache } from 'react';\nimport { client } from '@spfn/core/client';\nimport { getPublishedCacheContract } from './contracts/published-cache';\nimport { getLocale } from './helpers/locale.actions';\n\n/**\n * Section Data Type\n */\nexport type SectionData = {\n section: string;\n locale: string;\n content: Record<string, any>;\n version: number;\n publishedAt: string | null;\n};\n\n/**\n * Translation Function Type (runtime version)\n */\ntype ServerTranslationFunction = (\n key: string,\n defaultValue?: any,\n replace?: Record<string, string | number>\n) => any;\n\n/**\n * Section API Return Type\n */\nexport type SectionAPI = {\n /**\n * 라벨 값 가져오기 (변수 치환 지원)\n *\n * @param key - 라벨 키 (섹션 제외, 예: 'hero.title')\n * @param defaultValue - 기본값\n * @param replace - 변수 치환 맵 (예: { name: 'John' })\n * @returns 라벨 값 (문자열인 경우 변수 치환됨)\n */\n t: ServerTranslationFunction;\n\n /**\n * 섹션 데이터\n */\n data: SectionData;\n};\n\n/**\n * 변수 치환 헬퍼\n *\n * @param text - 치환할 텍스트 (예: 'Hello {name}!')\n * @param replace - 치환 맵 (예: { name: 'World' })\n * @returns 치환된 텍스트 (예: 'Hello World!')\n */\nfunction replaceVariables(text: string, replace: Record<string, string | number>): string\n{\n return text.replace(/\\{(\\w+)}/g, (match, key) =>\n {\n const value = replace[key];\n return value !== undefined ? String(value) : match;\n });\n}\n\n/**\n * 섹션 데이터 로드 (React cache 적용)\n *\n * 동일한 요청 내에서 같은 섹션을 여러 번 요청해도 한 번만 API 호출\n *\n * @param section - 섹션 이름 (예: 'home', 'why-futureplay')\n * @param locale - 언어 코드 (선택, 미지정시 쿠키에서 자동 조회)\n * @returns Section API ({ t, data })\n *\n * @example\n * ```tsx\n * // Server Component\n * import { getSection } from '@spfn/cms/server';\n *\n * export default async function HomePage()\n * {\n * // locale을 지정하지 않으면 쿠키에서 자동으로 가져옴\n * const { t } = await getSection('home');\n *\n * // 또는 명시적으로 locale 지정\n * const { t: tEn } = await getSection('home', 'en');\n *\n * return (\n * <div>\n * <h1>{t('hero.title')}</h1>\n * <p>{t('hero.subtitle', 'Default Subtitle')}</p>\n * <p>{t('hero.greeting', 'Hello {name}!', { name: 'World' })}</p>\n * </div>\n * );\n * }\n * ```\n */\nexport const getSection = cache(async (\n section: string,\n locale?: string\n): Promise<SectionAPI> =>\n{\n // locale이 지정되지 않으면 쿠키에서 가져옴\n const actualLocale: string = locale ?? await getLocale();\n\n try\n {\n // Call SPFN API via contract (uses singleton client)\n const response = await client.call(\n getPublishedCacheContract,\n {\n query: { sections: section, locale: actualLocale },\n }\n );\n\n // Check if response has error\n if ('error' in response)\n {\n console.warn(`[CMS] ${response.error}`);\n // Return empty section data\n const sectionData: SectionData = {\n section,\n locale: actualLocale,\n content: {} as Record<string, any>,\n version: 0,\n publishedAt: null,\n };\n\n const t: ServerTranslationFunction = (_key, defaultValue) => defaultValue ?? '';\n return { t, data: sectionData };\n }\n\n // Response is an array, get first element\n const found = response[0];\n\n if (!found)\n {\n // Section not found, return empty\n const sectionData: SectionData = {\n section,\n locale: actualLocale,\n content: {} as Record<string, any>,\n version: 0,\n publishedAt: null,\n };\n\n const t: ServerTranslationFunction = (_key, defaultValue) => defaultValue ?? '';\n return { t, data: sectionData };\n }\n\n // Success response\n const sectionData: SectionData = {\n section: found.section,\n locale: found.locale,\n content: found.content || {},\n version: found.version,\n publishedAt: found.publishedAt,\n };\n\n // Translation function\n const t: ServerTranslationFunction = (key, defaultValue, replace) =>\n {\n const fullKey = `${section}.${key}`;\n let value = sectionData.content[fullKey];\n\n if (value === undefined || value === null)\n {\n value = defaultValue ?? '';\n }\n\n // 문자열이 아니면 빈 문자열 반환\n if (typeof value !== 'string')\n {\n return '';\n }\n\n // 문자열이고 치환 맵이 있으면 변수 치환\n if (replace)\n {\n value = replaceVariables(value, replace);\n }\n\n return value;\n };\n\n return {\n t,\n data: sectionData,\n };\n }\n catch (error)\n {\n console.error(`[CMS] Failed to fetch section \"${section}\":`, error);\n\n // Return empty section data on error\n const sectionData: SectionData = {\n section,\n locale: actualLocale,\n content: {} as Record<string, any>,\n version: 0,\n publishedAt: null,\n };\n\n const t: ServerTranslationFunction = (_key, defaultValue) => defaultValue ?? '';\n return { t, data: sectionData };\n }\n});\n\n/**\n * 여러 섹션 한번에 로드 (React cache 적용)\n * 단일 API 호출로 여러 섹션을 효율적으로 가져옵니다\n *\n * @param sections - 섹션 이름 배열\n * @param locale - 언어 코드 (선택, 미지정시 쿠키에서 자동 조회)\n * @returns Section API 맵 ({ home: { t, data }, ... })\n *\n * @example\n * ```tsx\n * // Server Component\n * import { getSections } from '@spfn/cms/server';\n *\n * export default async function Page()\n * {\n * // locale을 지정하지 않으면 쿠키에서 자동으로 가져옴\n * const sections = await getSections(['home', 'why-futureplay']);\n *\n * // 또는 명시적으로 locale 지정\n * const sectionsEn = await getSections(['home', 'why-futureplay'], 'en');\n *\n * return (\n * <div>\n * <h1>{sections.home.t('hero.title')}</h1>\n * <p>{sections['why-futureplay'].t('intro.text')}</p>\n * </div>\n * );\n * }\n * ```\n */\nexport const getSections = cache(async (\n sections: string[],\n locale?: string\n): Promise<Record<string, SectionAPI>> =>\n{\n // locale이 지정되지 않으면 쿠키에서 가져옴\n const actualLocale: string = locale ?? await getLocale();\n\n try\n {\n // Call SPFN API with array of sections (single HTTP request)\n const response = await client.call(\n getPublishedCacheContract,\n {\n query: { sections, locale: actualLocale },\n }\n );\n\n // Check if response has error\n if ('error' in response)\n {\n console.warn(`[CMS] ${response.error}`);\n // Return empty sections\n const sectionsMap: Record<string, SectionAPI> = {};\n sections.forEach(section =>\n {\n sectionsMap[section] = {\n t: (_key, defaultValue) => defaultValue ?? '',\n data: {\n section,\n locale: actualLocale,\n content: {},\n version: 0,\n publishedAt: null,\n }\n };\n });\n return sectionsMap;\n }\n\n // Build sections map from response\n const sectionsMap: Record<string, SectionAPI> = {};\n\n // First, create empty entries for all requested sections\n sections.forEach(section =>\n {\n sectionsMap[section] = {\n t: (_key, defaultValue) => defaultValue ?? '',\n data: {\n section,\n locale: actualLocale,\n content: {},\n version: 0,\n publishedAt: null,\n }\n };\n });\n\n // Then, fill in data for found sections\n response.forEach(sectionData =>\n {\n const createTranslationFn = (section: string, content: Record<string, any>): ServerTranslationFunction =>\n {\n return (key, defaultValue, replace) =>\n {\n const fullKey = `${section}.${key}`;\n let value = content[fullKey];\n\n if (value === undefined || value === null)\n {\n value = defaultValue ?? '';\n }\n\n // 문자열이 아니면 빈 문자열 반환\n if (typeof value !== 'string')\n {\n return '';\n }\n\n // 문자열이고 치환 맵이 있으면 변수 치환\n if (replace)\n {\n value = replaceVariables(value, replace);\n }\n\n return value;\n };\n };\n\n sectionsMap[sectionData.section] = {\n t: createTranslationFn(sectionData.section, sectionData.content),\n data: {\n section: sectionData.section,\n locale: sectionData.locale,\n content: sectionData.content,\n version: sectionData.version,\n publishedAt: sectionData.publishedAt,\n }\n };\n });\n\n return sectionsMap;\n }\n catch (error)\n {\n console.error(`[CMS] Failed to fetch sections:`, error);\n\n // Return empty sections on error\n const sectionsMap: Record<string, SectionAPI> = {};\n sections.forEach(section =>\n {\n sectionsMap[section] = {\n t: (_key, defaultValue) => defaultValue ?? '',\n data: {\n section,\n locale: actualLocale,\n content: {},\n version: 0,\n publishedAt: null,\n }\n };\n });\n return sectionsMap;\n }\n});","import { Type } from '@sinclair/typebox';\nimport type { RouteContract } from '@spfn/core/route';\n\nconst SectionData = Type.Object({\n section: Type.String(),\n locale: Type.String(),\n content: Type.Record(Type.String(), Type.Any()),\n version: Type.Number(),\n publishedAt: Type.Union([Type.String(), Type.Null()]),\n});\n\n/**\n * GET /cms/published-cache\n * 발행된 콘텐츠 캐시 조회 (단일 또는 여러 섹션)\n */\nexport const getPublishedCacheContract = {\n method: 'GET' as const,\n path: '/cms/published-cache',\n query: Type.Object({\n sections: Type.Union([\n Type.String({ description: '단일 섹션 이름 (예: home)' }),\n Type.Array(Type.String(), { description: '여러 섹션 이름 (예: [\"home\", \"footer\"])' })\n ]),\n locale: Type.Optional(Type.String({ default: 'ko', description: '언어 코드' })),\n }),\n response: Type.Union([\n // 성공: 항상 배열로 반환\n Type.Array(SectionData),\n // 에러\n Type.Object({\n error: Type.String()\n })\n ])\n} as const satisfies RouteContract;","\"use server\";\n\n/**\n * Locale Management Server Actions\n *\n * Server Actions으로 구현된 locale 관리 함수\n * - 서버 컴포넌트: 일반 함수 호출로 동작\n * - 클라이언트 컴포넌트: Server Action으로 자동 처리\n */\n\nimport { cookies, headers } from 'next/headers.js';\nimport { getCmsConfig } from '../cms.config.js';\nimport { LOCALE_COOKIE_KEY } from './locale.constants.js';\n\n/**\n * 브라우저 언어 감지\n *\n * Accept-Language 헤더에서 지원하는 언어를 찾습니다.\n *\n * @returns 감지된 언어 코드 또는 null\n */\nasync function detectBrowserLanguage(): Promise<string | null>\n{\n try\n {\n const headersList = await headers();\n const acceptLanguage = headersList.get('accept-language');\n\n if (!acceptLanguage)\n {\n return null;\n }\n\n // \"ko-KR,ko;q=0.9,en-US;q=0.8,en;q=0.7\" 형식 파싱\n const languages = acceptLanguage\n .split(',')\n .map(lang =>\n {\n const [code] = lang.split(';');\n return code.split('-')[0].trim();\n });\n\n const config = getCmsConfig();\n\n // 지원하는 언어 중 첫 번째 매칭되는 언어 반환\n for (const lang of languages)\n {\n if (config.supportedLocales.includes(lang))\n {\n return lang;\n }\n }\n\n return null;\n }\n catch (error)\n {\n // 헤더 접근 실패 시\n return null;\n }\n}\n\n/**\n * 현재 locale 가져오기 (Server Action)\n *\n * 서버/클라이언트 컴포넌트 모두에서 사용 가능\n *\n * 우선순위:\n * 1. 쿠키 (사용자가 명시적으로 선택한 언어)\n * 2. 브라우저 언어 감지 (설정에서 활성화된 경우)\n * 3. 시스템 기본 언어 (CMS 설정)\n *\n * @returns 현재 locale (예: 'ko', 'en')\n *\n * @example\n * ```tsx\n * // Server Component\n * import { getLocale } from '@spfn/cms';\n *\n * export default async function Page()\n * {\n * const locale = await getLocale();\n * return <div>Current locale: {locale}</div>;\n * }\n * ```\n *\n * @example\n * ```tsx\n * // Client Component\n * 'use client';\n * import { getLocale } from '@spfn/cms/client';\n *\n * export default function LanguageSwitcher()\n * {\n * const [locale, setLocale] = useState('');\n *\n * useEffect(() => {\n * getLocale().then(setLocale);\n * }, []);\n *\n * return <div>Current locale: {locale}</div>;\n * }\n * ```\n */\nexport async function getLocale(): Promise<string>\n{\n const config = getCmsConfig();\n\n // 1순위: 쿠키 (사용자가 명시적으로 선택한 언어)\n const cookieStore = await cookies();\n const cookieLocale = cookieStore.get(LOCALE_COOKIE_KEY)?.value;\n\n if (cookieLocale && config.supportedLocales.includes(cookieLocale))\n {\n return cookieLocale;\n }\n\n // 2순위: 브라우저 언어 감지 (설정에서 활성화된 경우)\n if (config.detectBrowserLanguage)\n {\n const browserLang = await detectBrowserLanguage();\n if (browserLang)\n {\n return browserLang;\n }\n }\n\n // 3순위: 시스템 기본 언어\n return config.defaultLocale;\n}\n\n/**\n * Locale 설정하기 (Server Action)\n *\n * 서버/클라이언트 컴포넌트 모두에서 사용 가능\n * 쿠키에 locale을 저장합니다.\n *\n * @param locale - 설정할 locale (예: 'ko', 'en')\n * @throws {Error} 지원하지 않는 locale인 경우\n *\n * @example\n * ```tsx\n * // Server Component (Server Action)\n * import { setLocale } from '@spfn/cms';\n *\n * export default async function Page()\n * {\n * await setLocale('en');\n * return <div>Locale changed</div>;\n * }\n * ```\n *\n * @example\n * ```tsx\n * // Client Component (Server Action)\n * 'use client';\n * import { setLocale } from '@spfn/cms/client';\n *\n * export default function LanguageSwitcher()\n * {\n * const handleChange = async (newLocale: string) =>\n * {\n * await setLocale(newLocale);\n * window.location.reload(); // 페이지 새로고침\n * };\n *\n * return (\n * <button onClick={() => handleChange('en')}>\n * Switch to English\n * </button>\n * );\n * }\n * ```\n */\nexport async function setLocale(locale: string): Promise<void>\n{\n const config = getCmsConfig();\n\n // 유효성 검사\n if (!config.supportedLocales.includes(locale))\n {\n throw new Error(\n `Unsupported locale: ${locale}. Supported locales: ${config.supportedLocales.join(', ')}`\n );\n }\n\n const cookieStore = await cookies();\n\n cookieStore.set(LOCALE_COOKIE_KEY, locale, {\n path: '/',\n maxAge: 60 * 60 * 24 * 365, // 1년\n sameSite: 'lax',\n });\n}\n\n/**\n * 지원하는 locale 목록 가져오기 (Server Action)\n *\n * 서버/클라이언트 컴포넌트 모두에서 사용 가능\n *\n * @returns 지원하는 locale 배열 (예: ['ko', 'en', 'ja'])\n *\n * @example\n * ```tsx\n * // Server Component\n * import { getLocales } from '@spfn/cms';\n *\n * export default async function Page()\n * {\n * const locales = await getLocales();\n * return <div>Supported: {locales.join(', ')}</div>;\n * }\n * ```\n *\n * @example\n * ```tsx\n * // Client Component\n * 'use client';\n * import { getLocales } from '@spfn/cms/client';\n *\n * export default function LanguageSwitcher()\n * {\n * const [locales, setLocales] = useState<string[]>([]);\n *\n * useEffect(() => {\n * getLocales().then(setLocales);\n * }, []);\n *\n * return (\n * <div>\n * {locales.map(locale => (\n * <button key={locale}>{locale}</button>\n * ))}\n * </div>\n * );\n * }\n * ```\n */\nexport async function getLocales(): Promise<string[]>\n{\n const config = getCmsConfig();\n return config.supportedLocales;\n}","/**\n * Locale Constants\n *\n * Server/Client 양쪽에서 사용 가능한 locale 관련 상수\n */\n\n/**\n * Locale 쿠키 키\n */\nexport const LOCALE_COOKIE_KEY = 'spfn-locale';","/**\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 { desc } from 'drizzle-orm';\nimport { cmsLabels, type CmsLabel, type NewCmsLabel } from '../entities/index.js';\n\n/**\n * 라벨 목록 조회 (페이지네이션)\n */\nexport async function findMany(options?: {\n section?: string;\n limit?: number;\n offset?: number;\n}): Promise<CmsLabel[]>\n{\n const { section, limit = 20, offset = 0 } = options || {};\n\n return findManyHelper(cmsLabels, {\n where: section ? { section } : undefined,\n orderBy: desc(cmsLabels.updatedAt),\n limit,\n offset\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: desc(cmsLabels.updatedAt),\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 './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 // 버전 번호\n version: integer('version').notNull().default(1),\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 Label Versions Entity\n *\n * 라벨의 버전 메타데이터를 관리합니다.\n * - 버전별 상태 (draft/published/archived)\n * - 발행 정보 (publishedAt, publishedBy)\n * - 버전 노트\n */\n\nimport { serial, integer, text, timestamp, index, unique } from 'drizzle-orm/pg-core';\nimport { createFunctionSchema } from '@spfn/core/db';\nimport { cmsLabels } from '@/entities/cms-labels';\n\n// Create isolated schema for @spfn/cms\nconst schema = createFunctionSchema('@spfn/cms');\n\nexport const cmsLabelVersions = schema.table('label_versions', {\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 // 버전 번호\n version: integer('version').notNull(),\n\n // 버전 상태\n status: text('status').notNull(),\n // \"draft\" | \"published\" | \"archived\"\n\n // 발행 정보\n publishedAt: timestamp('published_at', { withTimezone: true }),\n publishedBy: text('published_by'),\n\n // 버전 노트 (변경사항 설명)\n notes: text('notes'),\n\n // 버전 생성자\n createdBy: text('created_by'),\n\n // 생성 시각\n createdAt: timestamp('created_at', { withTimezone: true }).notNull().defaultNow(),\n}, (table) => [\n // UNIQUE 제약: 각 라벨의 버전 번호는 고유\n unique('cms_label_versions_label_version_unique')\n .on(table.labelId, table.version),\n\n // 인덱스: labelId로 버전 목록 조회 최적화\n index('cms_label_versions_label_id_idx').on(table.labelId),\n\n // 인덱스: status 필터링 최적화\n index('cms_label_versions_status_idx').on(table.status),\n]);\n\n// 타입 추론\nexport type CmsLabelVersion = typeof cmsLabelVersions.$inferSelect;\nexport type NewCmsLabelVersion = typeof cmsLabelVersions.$inferInsert;\n\n/**\n * 사용 예시:\n *\n * // 새 버전 생성 (draft)\n * await db.insert(cmsLabelVersions).values({\n * labelId: 1,\n * version: 2,\n * status: 'draft',\n * notes: '문구 개선',\n * createdBy: 'editor@futureplay.com'\n * });\n *\n * // 버전 발행\n * await db.update(cmsLabelVersions)\n * .set({\n * status: 'published',\n * publishedAt: new Date(),\n * publishedBy: 'admin@futureplay.com'\n * })\n * .where(and(\n * eq(cmsLabelVersions.labelId, 1),\n * eq(cmsLabelVersions.version, 2)\n * ));\n *\n * // 라벨의 모든 버전 조회\n * const versions = await db.select()\n * .from(cmsLabelVersions)\n * .where(eq(cmsLabelVersions.labelId, 1))\n * .orderBy(desc(cmsLabelVersions.version));\n *\n * // 발행된 버전만 조회\n * const published = await db.select()\n * .from(cmsLabelVersions)\n * .where(eq(cmsLabelVersions.status, 'published'))\n * .orderBy(desc(cmsLabelVersions.publishedAt));\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 './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 */","/**\n * CMS Label Values Repository\n *\n * 라벨 값 관리를 위한 Repository\n */\n\nimport { findOne, findMany, create, updateOne, deleteMany } from '@spfn/core/db';\nimport { eq, and, SQL, isNull } from 'drizzle-orm';\nimport { cmsLabelValues, type CmsLabelValue, type NewCmsLabelValue } from '../entities/index.js';\n\n/**\n * 특정 라벨의 특정 버전 값들 조회\n */\nexport async function findByLabelIdAndVersion(\n labelId: number,\n version: number,\n options?: {\n locale?: string;\n breakpoint?: string | null;\n }\n): Promise<CmsLabelValue[]>\n{\n const { locale, breakpoint } = options || {};\n\n const conditions: SQL[] = [\n eq(cmsLabelValues.labelId, labelId),\n eq(cmsLabelValues.version, version)\n ];\n\n if (locale)\n {\n conditions.push(eq(cmsLabelValues.locale, locale));\n }\n\n if (breakpoint !== undefined)\n {\n conditions.push(\n breakpoint === null\n ? isNull(cmsLabelValues.breakpoint)\n : eq(cmsLabelValues.breakpoint, breakpoint)\n );\n }\n\n return findMany(cmsLabelValues, {\n where: and(...conditions)\n });\n}\n\n/**\n * 값 저장 (upsert)\n */\nexport async function upsert(data: NewCmsLabelValue): Promise<CmsLabelValue>\n{\n // 기존 값이 있는지 확인\n const existing = await findOne(\n cmsLabelValues,\n and(\n eq(cmsLabelValues.labelId, data.labelId),\n eq(cmsLabelValues.version, data.version ?? 1),\n eq(cmsLabelValues.locale, data.locale || 'ko'),\n data.breakpoint\n ? eq(cmsLabelValues.breakpoint, data.breakpoint)\n : isNull(cmsLabelValues.breakpoint)\n )\n );\n\n if (existing)\n {\n // UPDATE\n const updated = await updateOne(\n cmsLabelValues,\n { id: existing.id },\n { value: data.value }\n );\n return updated!;\n }\n else\n {\n // INSERT\n return create(cmsLabelValues, data);\n }\n}\n\n/**\n * 여러 값 일괄 저장\n */\nexport async function upsertMany(values: NewCmsLabelValue[]): Promise<CmsLabelValue[]>\n{\n const results = [];\n for (const value of values)\n {\n const result = await upsert(value);\n results.push(result);\n }\n return results;\n}\n\n/**\n * 특정 버전의 모든 값 삭제\n */\nexport async function deleteByVersion(labelId: number, version: number): Promise<CmsLabelValue[]>\n{\n return deleteMany(\n cmsLabelValues,\n and(\n eq(cmsLabelValues.labelId, labelId),\n eq(cmsLabelValues.version, version)\n )\n );\n}\n\n// Legacy export for backward compatibility\nexport const cmsLabelValuesRepository = {\n findByLabelIdAndVersion,\n upsert,\n upsertMany,\n deleteByVersion\n};","/**\n * CMS Draft Cache Repository\n *\n * 관리자별 초안 캐시 관리 (동시 편집 지원)\n */\n\nimport { findOne, findMany, deleteOne, deleteMany, upsert as upsertHelper } from '@spfn/core/db';\nimport { eq, and, lt } from 'drizzle-orm';\nimport { cmsDraftCache, type NewCmsDraftCache } from '../entities/index.js';\n\n/**\n * 섹션 + 언어 + 사용자로 초안 캐시 조회\n */\nexport async function findByUser(section: string, locale: string, userId: string)\n{\n return findOne(\n cmsDraftCache,\n and(\n eq(cmsDraftCache.section, section),\n eq(cmsDraftCache.locale, locale),\n eq(cmsDraftCache.userId, userId)\n )\n );\n}\n\n/**\n * 초안 캐시 생성 또는 업데이트 (UPSERT)\n */\nexport async function upsert(data: NewCmsDraftCache)\n{\n return upsertHelper(cmsDraftCache, data, {\n target: [cmsDraftCache.section, cmsDraftCache.locale, cmsDraftCache.userId],\n set: {\n content: data.content,\n updatedAt: new Date(),\n }\n });\n}\n\n/**\n * 특정 사용자의 모든 초안 조회\n */\nexport async function findAllByUser(userId: string)\n{\n return findMany(cmsDraftCache, {\n where: eq(cmsDraftCache.userId, userId)\n });\n}\n\n/**\n * 초안 삭제\n */\nexport async function deleteByUser(section: string, locale: string, userId: string)\n{\n await deleteOne(\n cmsDraftCache,\n and(\n eq(cmsDraftCache.section, section),\n eq(cmsDraftCache.locale, locale),\n eq(cmsDraftCache.userId, userId)\n )\n );\n}\n\n/**\n * 오래된 초안 정리 (30일 이상 미사용)\n */\nexport async function cleanupOldDrafts(daysOld: number = 30)\n{\n const cutoffDate = new Date();\n cutoffDate.setDate(cutoffDate.getDate() - daysOld);\n\n return deleteMany(\n cmsDraftCache,\n lt(cmsDraftCache.updatedAt, cutoffDate)\n );\n}\n\nexport const cmsDraftCacheRepository = {\n findByUser,\n upsert,\n findAllByUser,\n deleteByUser,\n cleanupOldDrafts,\n};","/**\n * CMS Published Cache Repository\n *\n * 발행된 콘텐츠 캐시 관리 (초고속 조회)\n */\n\nimport { findOne, findMany, deleteOne, deleteMany, upsert as upsertHelper } from '@spfn/core/db';\nimport { eq, and, sql } from 'drizzle-orm';\nimport { cmsPublishedCache, type NewCmsPublishedCache } from '../entities/index.js';\n\n/**\n * 섹션 + 언어로 발행된 캐시 조회\n */\nexport async function findBySection(section: string, locale: string = 'ko')\n{\n return findOne(\n cmsPublishedCache,\n and(\n eq(cmsPublishedCache.section, section),\n eq(cmsPublishedCache.locale, locale)\n )\n );\n}\n\n/**\n * 캐시 생성 또는 업데이트 (UPSERT)\n */\nexport async function upsert(data: NewCmsPublishedCache)\n{\n return upsertHelper(cmsPublishedCache, data, {\n target: [cmsPublishedCache.section, cmsPublishedCache.locale],\n set: {\n content: data.content,\n publishedAt: data.publishedAt,\n publishedBy: data.publishedBy,\n version: sql`${cmsPublishedCache.version} + 1`, // 버전 증가로 클라이언트 캐시 무효화\n }\n });\n}\n\n/**\n * 섹션별 모든 언어 캐시 조회\n */\nexport async function findAllLanguages(section: string)\n{\n return findMany(cmsPublishedCache, {\n where: eq(cmsPublishedCache.section, section)\n });\n}\n\n/**\n * 캐시 삭제\n */\nexport async function deleteBySection(section: string, locale?: string)\n{\n if (locale)\n {\n await deleteOne(\n cmsPublishedCache,\n and(\n eq(cmsPublishedCache.section, section),\n eq(cmsPublishedCache.locale, locale)\n )\n );\n }\n else\n {\n await deleteMany(\n cmsPublishedCache,\n eq(cmsPublishedCache.section, section)\n );\n }\n}\n\nexport const cmsPublishedCacheRepository = {\n findBySection,\n upsert,\n findAllLanguages,\n deleteBySection,\n};","/**\n * CMS Sync Utilities\n *\n * JSON 파일 기반 라벨 동기화\n */\n\nimport { existsSync, readdirSync, readFileSync, statSync } from 'fs';\nimport { basename, extname, join } from 'path';\nimport { extractLabels } from '../labels/index.js';\nimport { cmsLabelsRepository, cmsPublishedCacheRepository } from '../repositories/index.js';\nimport type { NestedLabels, SectionDefinition, SyncOptions, SyncResult } from '../types.js';\n\n/**\n * 여러 섹션 동기화\n */\nexport async function syncAll(\n sections: SectionDefinition[],\n options: SyncOptions = {}\n): Promise<SyncResult[]>\n{\n const results: SyncResult[] = [];\n\n for (const definition of sections)\n {\n const result = await syncSection(definition, options);\n results.push(result);\n }\n\n return results;\n}\n\n/**\n * JSON 파일에서 라벨 로드\n */\nexport function loadLabelsFromJson(labelsDir: string): SectionDefinition[]\n{\n const sections: SectionDefinition[] = [];\n\n if (!existsSync(labelsDir))\n {\n console.warn(`[CMS] Labels directory not found: ${labelsDir}`);\n console.warn(`[CMS] Expected directory structure:`);\n console.warn(`[CMS] ${labelsDir}/`);\n console.warn(`[CMS] ├── common/ # Section directory`);\n console.warn(`[CMS] │ ├── messages.json`);\n console.warn(`[CMS] │ └── errors.json`);\n console.warn(`[CMS] └── home/ # Section directory`);\n console.warn(`[CMS] └── hero.json`);\n return sections;\n }\n\n try\n {\n const entries = readdirSync(labelsDir);\n\n if (entries.length === 0)\n {\n console.warn(`[CMS] Labels directory is empty: ${labelsDir}`);\n console.warn(`[CMS] Create section directories with JSON files inside`);\n return sections;\n }\n\n const jsonFiles = entries.filter(e => extname(e) === '.json');\n if (jsonFiles.length > 0)\n {\n console.warn(`[CMS] Found JSON files directly in ${labelsDir}:`);\n jsonFiles.forEach(f => console.warn(`[CMS] - ${f} (will be ignored)`));\n console.warn(`[CMS] JSON files should be inside section directories`);\n console.warn(`[CMS] Example: Move ${jsonFiles[0]} to ${labelsDir}/${basename(jsonFiles[0], '.json')}/${jsonFiles[0]}`);\n }\n\n for (const entry of entries)\n {\n const sectionPath = join(labelsDir, entry);\n const stat = statSync(sectionPath);\n\n if (stat.isDirectory())\n {\n const sectionName = entry;\n const labels = loadSectionLabels(sectionPath);\n\n if (Object.keys(labels).length > 0)\n {\n sections.push({ section: sectionName, labels });\n }\n else\n {\n console.warn(`[CMS] Section directory \"${sectionName}\" has no valid JSON files`);\n }\n }\n }\n\n if (sections.length === 0)\n {\n console.warn(`[CMS] No valid section directories found in ${labelsDir}`);\n }\n }\n catch (error)\n {\n console.warn(`[CMS] Could not scan labels directory: ${labelsDir}`);\n console.error(`[CMS] Error:`, error);\n }\n\n return sections;\n}\n\nfunction loadSectionLabels(sectionPath: string): NestedLabels\n{\n const labels: NestedLabels = {};\n\n try\n {\n const files = readdirSync(sectionPath);\n\n for (const file of files)\n {\n if (extname(file) === '.json')\n {\n const filePath = join(sectionPath, file);\n const categoryName = basename(file, '.json');\n\n try\n {\n const content = readFileSync(filePath, 'utf-8');\n labels[categoryName] = JSON.parse(content);\n }\n catch (error)\n {\n console.warn(`[CMS] Failed to parse ${filePath}`);\n }\n }\n }\n }\n catch (error)\n {\n console.warn(`[CMS] Could not read section directory: ${sectionPath}`);\n }\n\n return labels;\n}\n\n/**\n * 섹션 라벨 동기화\n */\nexport async function syncSection(\n definition: SectionDefinition,\n options: SyncOptions = {}\n): Promise<SyncResult>\n{\n const {\n dryRun = false,\n updateExisting = false,\n removeUnused = false,\n verbose = false,\n } = options;\n\n const { section } = definition;\n const result: SyncResult = {\n section,\n created: 0,\n updated: 0,\n deleted: 0,\n unchanged: 0,\n errors: [],\n };\n\n try\n {\n const definedLabels = extractLabels(definition);\n const definedKeys = new Set(definedLabels.map((l) => l.key));\n const existingLabels = await cmsLabelsRepository.findBySection(section);\n const existingMap = new Map(existingLabels.map((l) => [l.key, l]));\n\n if (verbose)\n {\n console.log(`\\n[${section}] Found ${definedLabels.length} labels in definition`);\n console.log(`[${section}] Found ${existingLabels.length} labels in DB`);\n }\n\n // 생성 및 업데이트\n for (const label of definedLabels)\n {\n const existing = existingMap.get(label.key);\n\n if (!existing)\n {\n if (verbose) console.log(` [CREATE] ${label.key}`);\n\n if (!dryRun)\n {\n try\n {\n const defaultValue = typeof label.defaultValue === 'object'\n ? JSON.stringify(label.defaultValue)\n : label.defaultValue;\n\n await cmsLabelsRepository.create({\n section,\n key: label.key,\n type: label.type || 'text', // 라벨 타입 (기본값: 'text')\n defaultValue,\n description: label.description,\n });\n }\n catch (error)\n {\n result.errors.push({\n key: label.key,\n error: error instanceof Error ? error.message : String(error),\n });\n continue;\n }\n }\n\n result.created++;\n }\n else if (updateExisting)\n {\n const newDefaultValue = typeof label.defaultValue === 'object'\n ? JSON.stringify(label.defaultValue)\n : label.defaultValue;\n\n const hasChanged = existing.defaultValue !== newDefaultValue;\n\n if (hasChanged)\n {\n if (verbose)\n {\n console.log(` [UPDATE] ${label.key}`);\n console.log(` Old: \"${existing.defaultValue}\"`);\n console.log(` New: \"${newDefaultValue}\"`);\n }\n\n if (!dryRun)\n {\n try\n {\n await cmsLabelsRepository.updateById(existing.id, {\n defaultValue: newDefaultValue,\n description: label.description,\n });\n }\n catch (error)\n {\n result.errors.push({\n key: label.key,\n error: error instanceof Error ? error.message : String(error),\n });\n continue;\n }\n }\n\n result.updated++;\n }\n else\n {\n result.unchanged++;\n }\n }\n else\n {\n result.unchanged++;\n }\n }\n\n // 사용되지 않는 라벨 삭제\n if (removeUnused)\n {\n for (const existing of existingLabels)\n {\n if (!definedKeys.has(existing.key))\n {\n if (verbose) console.log(` [DELETE] ${existing.key}`);\n\n if (!dryRun)\n {\n try\n {\n await cmsLabelsRepository.deleteById(existing.id);\n }\n catch (error)\n {\n result.errors.push({\n key: existing.key,\n error: error instanceof Error ? error.message : String(error),\n });\n continue;\n }\n }\n\n result.deleted++;\n }\n }\n }\n\n // Published cache 업데이트\n if (!dryRun && (result.created > 0 || result.updated > 0 || result.deleted > 0))\n {\n if (verbose) console.log(` [CACHE] Updating published cache for section: ${section}`);\n await updatePublishedCache(section);\n }\n }\n catch (error)\n {\n result.errors.push({\n key: '__section__',\n error: error instanceof Error ? error.message : String(error),\n });\n }\n\n return result;\n}\n\n/**\n * Published Cache 업데이트\n */\nasync function updatePublishedCache(section: string): Promise<void>\n{\n const labels = await cmsLabelsRepository.findBySection(section);\n const localesSet = new Set<string>();\n const labelsByLocale: Record<string, Record<string, any>> = {};\n\n labels.forEach((label) =>\n {\n try\n {\n const parsed = JSON.parse(label.defaultValue || '{}');\n\n if (typeof parsed === 'object' && !Array.isArray(parsed))\n {\n // Multilingual\n Object.keys(parsed).forEach((locale) => localesSet.add(locale));\n Object.entries(parsed).forEach(([locale, value]) =>\n {\n if (!labelsByLocale[locale]) labelsByLocale[locale] = {};\n labelsByLocale[locale][label.key] = value;\n });\n }\n else\n {\n // Single value\n if (!labelsByLocale.ko) labelsByLocale.ko = {};\n labelsByLocale.ko[label.key] = label.defaultValue;\n localesSet.add('ko');\n }\n }\n catch\n {\n // Plain string\n if (!labelsByLocale.ko) labelsByLocale.ko = {};\n labelsByLocale.ko[label.key] = label.defaultValue;\n localesSet.add('ko');\n }\n });\n\n const timestamp = new Date();\n for (const locale of localesSet)\n {\n await cmsPublishedCacheRepository.upsert({\n section,\n locale,\n content: labelsByLocale[locale] || {},\n publishedAt: timestamp,\n publishedBy: 'system',\n });\n }\n}\n\n/**\n * Initialize label sync for server startup\n *\n * Call this in your server.config.ts beforeRoutes hook\n *\n * @param options - Sync options\n * @param options.labelsDir - Path to labels directory (default: 'src/cms/labels')\n *\n * @example\n * ```typescript\n * import { initLabelSync } from '@spfn/cms';\n *\n * export default {\n * beforeRoutes: async (app) => {\n * await initLabelSync({ verbose: true });\n * },\n * } satisfies ServerConfig;\n * ```\n */\nexport async function initLabelSync(options: SyncOptions & { labelsDir?: string } = {}): Promise<void>\n{\n const isDevelopment = process.env.NODE_ENV === 'development';\n const verbose = options.verbose ?? isDevelopment;\n const labelsDir = options.labelsDir ?? 'src/cms/labels';\n\n if (verbose)\n {\n console.log('\\n🔄 Initializing label sync...\\n');\n }\n\n // Load labels from JSON files\n const sections = loadLabelsFromJson(labelsDir);\n\n if (sections.length === 0)\n {\n if (verbose)\n {\n console.log('⚠️ No labels found in', labelsDir);\n console.log('');\n }\n return;\n }\n\n const results = await syncAll(sections, {\n updateExisting: true, // 🔄 항상 업데이트 (프로덕션 포함)\n ...options,\n verbose,\n });\n\n const totalCreated = results.reduce((sum, r) => sum + r.created, 0);\n const totalUpdated = results.reduce((sum, r) => sum + r.updated, 0);\n const totalUnchanged = results.reduce((sum, r) => sum + r.unchanged, 0);\n const totalErrors = results.reduce((sum, r) => sum + r.errors.length, 0);\n\n if (verbose)\n {\n console.log('✅ Label sync completed\\n');\n console.log(` Sections: ${results.length}`);\n console.log(` Created: ${totalCreated}`);\n console.log(` Updated: ${totalUpdated}`);\n console.log(` Unchanged: ${totalUnchanged}`);\n\n if (totalErrors > 0)\n {\n console.log(` Errors: ${totalErrors}\\n`);\n }\n else\n {\n console.log('');\n }\n }\n\n // Log errors\n if (totalErrors > 0)\n {\n results.forEach((result) =>\n {\n result.errors.forEach((error) =>\n {\n console.error(`[${result.section}] ${error.key}: ${error.error}`);\n });\n });\n }\n}","/**\n * CMS Label Helpers\n *\n * Utilities for processing label definitions from JSON files\n */\n\nimport type { SectionDefinition, NestedLabels, FlatLabel, LabelDefinition } from '../types.js';\n\n/**\n * 중첩된 라벨을 플랫화\n *\n * @param labels - 중첩된 라벨 객체\n * @returns 플랫화된 라벨 배열\n *\n * @example\n * ```ts\n * flattenLabels({\n * nav: {\n * home: { key: 'layout.nav.home', defaultValue: 'Home' },\n * },\n * });\n * // => [{ key: 'layout.nav.home', defaultValue: 'Home' }]\n * ```\n */\nexport function flattenLabels(labels: NestedLabels): FlatLabel[]\n{\n const result: FlatLabel[] = [];\n\n function isLabelDefinition(obj: NestedLabels | LabelDefinition): obj is LabelDefinition\n {\n return (\n 'key' in obj &&\n 'defaultValue' in obj &&\n typeof obj.key === 'string' &&\n (typeof obj.defaultValue === 'string' || typeof obj.defaultValue === 'object')\n );\n }\n\n function traverse(obj: NestedLabels | LabelDefinition)\n {\n if (isLabelDefinition(obj))\n {\n // LabelDefinition인 경우\n result.push({\n key: obj.key,\n defaultValue: obj.defaultValue,\n description: obj.description,\n });\n }\n else\n {\n // NestedLabels인 경우\n Object.values(obj).forEach((value) =>\n {\n if (typeof value === 'object' && value !== null)\n {\n traverse(value);\n }\n });\n }\n }\n\n traverse(labels);\n return result;\n}\n\n/**\n * 섹션 정의에서 모든 라벨 추출\n *\n * @param definition - 섹션 정의\n * @returns 플랫화된 라벨 배열\n */\nexport function extractLabels(definition: SectionDefinition): FlatLabel[]\n{\n return flattenLabels(definition.labels);\n}\n","/**\n * Label Sync Generator\n *\n * File-based label sync with JSON definitions\n *\n * Structure:\n * cms/labels/\n * layout/ # Section name\n * nav.json # Label definitions\n * footer.json\n * homepage/\n * hero.json\n */\n\nimport { logger } from '@spfn/core';\nimport type { Generator, GeneratorOptions } from '@spfn/core/codegen';\nimport { join } from 'path';\n\nimport { syncAll, loadLabelsFromJson } from '../helpers/sync.js';\n\nconst syncLogger = logger.child('label-sync');\n\nexport interface LabelSyncGeneratorConfig\n{\n labelsDir?: string;\n}\n\nexport class LabelSyncGenerator implements Generator\n{\n name = 'label-sync';\n private readonly labelsDir: string;\n\n constructor(config: LabelSyncGeneratorConfig = {})\n {\n this.labelsDir = config.labelsDir ?? 'src/cms/labels';\n }\n\n /**\n * Watch patterns for label definition files\n */\n get watchPatterns(): string[]\n {\n return [\n `${this.labelsDir}/**/*.json`,\n ];\n }\n\n async generate(options: GeneratorOptions): Promise<void>\n {\n if (options.debug)\n {\n syncLogger.info('Starting label sync...');\n }\n\n try\n {\n const labelsPath = join(options.cwd, this.labelsDir);\n\n // Load labels from JSON files\n const sections = loadLabelsFromJson(labelsPath);\n\n if (sections.length === 0)\n {\n syncLogger.warn(`No labels found in ${labelsPath}`);\n return;\n }\n\n syncLogger.info(`Found ${sections.length} sections`);\n\n // Sync all sections\n const results = await syncAll(sections, {\n verbose: options.debug ?? false,\n updateExisting: true, // 항상 업데이트 (프로덕션 포함)\n });\n\n const totalCreated = results.reduce((sum, r) => sum + r.created, 0);\n const totalUpdated = results.reduce((sum, r) => sum + r.updated, 0);\n const totalErrors = results.reduce((sum, r) => sum + r.errors.length, 0);\n\n if (options.debug || totalCreated > 0 || totalUpdated > 0)\n {\n syncLogger.info('Label sync completed', {\n sections: results.length,\n created: totalCreated,\n updated: totalUpdated,\n errors: totalErrors,\n });\n }\n\n // Log errors if any\n if (totalErrors > 0)\n {\n results.forEach((result) =>\n {\n result.errors.forEach((error) =>\n {\n syncLogger.error(`[${result.section}] ${error.key}: ${error.error}`);\n });\n });\n }\n }\n catch (error)\n {\n syncLogger.error(\n 'Label sync failed',\n error instanceof Error ? error : new Error(String(error))\n );\n }\n }\n\n async onFileChange(filePath: string, event: 'add' | 'change' | 'unlink'): Promise<void>\n {\n syncLogger.info(`Label file ${event}`, { file: filePath });\n\n // Re-sync all labels when any label file changes\n await this.generate({ cwd: process.cwd(), debug: true });\n }\n}\n\n/**\n * Create label sync generator instance\n */\nexport function createLabelSyncGenerator(config?: LabelSyncGeneratorConfig): Generator\n{\n return new LabelSyncGenerator(config);\n}"],"mappings":"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","names":["detectBrowserLanguage","text","sectionData","t","sectionsMap","serial","integer","text","timestamp","index","createFunctionSchema","schema","createFunctionSchema","serial","integer","text","timestamp","index","serial","integer","text","timestamp","index","unique","createFunctionSchema","schema","createFunctionSchema","serial","integer","text","timestamp","unique","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","findOne","findMany","create","updateOne","findMany","findOne","updateOne","create","findOne","findMany","deleteOne","deleteMany","eq","and","findOne","and","eq","upsert","findMany","deleteOne","deleteMany","findOne","findMany","deleteOne","deleteMany","upsertHelper","eq","and","findBySection","findOne","and","eq","upsert","upsertHelper","findMany","deleteOne","deleteMany","timestamp","join","join"]}
|
|
@@ -1,3 +1,5 @@
|
|
|
1
|
+
import { Generator, GeneratorOptions } from '@spfn/core/codegen';
|
|
2
|
+
|
|
1
3
|
/**
|
|
2
4
|
* Label Sync Generator
|
|
3
5
|
*
|
|
@@ -11,13 +13,13 @@
|
|
|
11
13
|
* homepage/
|
|
12
14
|
* hero.json
|
|
13
15
|
*/
|
|
14
|
-
|
|
15
|
-
|
|
16
|
+
|
|
17
|
+
interface LabelSyncGeneratorConfig {
|
|
16
18
|
labelsDir?: string;
|
|
17
19
|
}
|
|
18
|
-
|
|
20
|
+
declare class LabelSyncGenerator implements Generator {
|
|
19
21
|
name: string;
|
|
20
|
-
private labelsDir;
|
|
22
|
+
private readonly labelsDir;
|
|
21
23
|
constructor(config?: LabelSyncGeneratorConfig);
|
|
22
24
|
/**
|
|
23
25
|
* Watch patterns for label definition files
|
|
@@ -29,5 +31,6 @@ export declare class LabelSyncGenerator implements Generator {
|
|
|
29
31
|
/**
|
|
30
32
|
* Create label sync generator instance
|
|
31
33
|
*/
|
|
32
|
-
|
|
33
|
-
|
|
34
|
+
declare function createLabelSyncGenerator(config?: LabelSyncGeneratorConfig): Generator;
|
|
35
|
+
|
|
36
|
+
export { LabelSyncGenerator as L, createLabelSyncGenerator as c };
|
package/dist/labels/index.d.ts
CHANGED
|
@@ -1,7 +1,34 @@
|
|
|
1
|
+
import { SectionDefinition, FlatLabel, NestedLabels } from '../types.js';
|
|
2
|
+
|
|
1
3
|
/**
|
|
2
|
-
* CMS
|
|
4
|
+
* CMS Label Helpers
|
|
3
5
|
*
|
|
4
|
-
*
|
|
6
|
+
* Utilities for processing label definitions from JSON files
|
|
5
7
|
*/
|
|
6
|
-
|
|
7
|
-
|
|
8
|
+
|
|
9
|
+
/**
|
|
10
|
+
* 중첩된 라벨을 플랫화
|
|
11
|
+
*
|
|
12
|
+
* @param labels - 중첩된 라벨 객체
|
|
13
|
+
* @returns 플랫화된 라벨 배열
|
|
14
|
+
*
|
|
15
|
+
* @example
|
|
16
|
+
* ```ts
|
|
17
|
+
* flattenLabels({
|
|
18
|
+
* nav: {
|
|
19
|
+
* home: { key: 'layout.nav.home', defaultValue: 'Home' },
|
|
20
|
+
* },
|
|
21
|
+
* });
|
|
22
|
+
* // => [{ key: 'layout.nav.home', defaultValue: 'Home' }]
|
|
23
|
+
* ```
|
|
24
|
+
*/
|
|
25
|
+
declare function flattenLabels(labels: NestedLabels): FlatLabel[];
|
|
26
|
+
/**
|
|
27
|
+
* 섹션 정의에서 모든 라벨 추출
|
|
28
|
+
*
|
|
29
|
+
* @param definition - 섹션 정의
|
|
30
|
+
* @returns 플랫화된 라벨 배열
|
|
31
|
+
*/
|
|
32
|
+
declare function extractLabels(definition: SectionDefinition): FlatLabel[];
|
|
33
|
+
|
|
34
|
+
export { extractLabels, flattenLabels };
|
package/dist/labels/index.js
CHANGED
|
@@ -1,7 +1,32 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
1
|
+
// src/labels/helpers.ts
|
|
2
|
+
function flattenLabels(labels) {
|
|
3
|
+
const result = [];
|
|
4
|
+
function isLabelDefinition(obj) {
|
|
5
|
+
return "key" in obj && "defaultValue" in obj && typeof obj.key === "string" && (typeof obj.defaultValue === "string" || typeof obj.defaultValue === "object");
|
|
6
|
+
}
|
|
7
|
+
function traverse(obj) {
|
|
8
|
+
if (isLabelDefinition(obj)) {
|
|
9
|
+
result.push({
|
|
10
|
+
key: obj.key,
|
|
11
|
+
defaultValue: obj.defaultValue,
|
|
12
|
+
description: obj.description
|
|
13
|
+
});
|
|
14
|
+
} else {
|
|
15
|
+
Object.values(obj).forEach((value) => {
|
|
16
|
+
if (typeof value === "object" && value !== null) {
|
|
17
|
+
traverse(value);
|
|
18
|
+
}
|
|
19
|
+
});
|
|
20
|
+
}
|
|
21
|
+
}
|
|
22
|
+
traverse(labels);
|
|
23
|
+
return result;
|
|
24
|
+
}
|
|
25
|
+
function extractLabels(definition) {
|
|
26
|
+
return flattenLabels(definition.labels);
|
|
27
|
+
}
|
|
28
|
+
export {
|
|
29
|
+
extractLabels,
|
|
30
|
+
flattenLabels
|
|
31
|
+
};
|
|
7
32
|
//# sourceMappingURL=index.js.map
|
package/dist/labels/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"
|
|
1
|
+
{"version":3,"sources":["../../src/labels/helpers.ts"],"sourcesContent":["/**\n * CMS Label Helpers\n *\n * Utilities for processing label definitions from JSON files\n */\n\nimport type { SectionDefinition, NestedLabels, FlatLabel, LabelDefinition } from '../types.js';\n\n/**\n * 중첩된 라벨을 플랫화\n *\n * @param labels - 중첩된 라벨 객체\n * @returns 플랫화된 라벨 배열\n *\n * @example\n * ```ts\n * flattenLabels({\n * nav: {\n * home: { key: 'layout.nav.home', defaultValue: 'Home' },\n * },\n * });\n * // => [{ key: 'layout.nav.home', defaultValue: 'Home' }]\n * ```\n */\nexport function flattenLabels(labels: NestedLabels): FlatLabel[]\n{\n const result: FlatLabel[] = [];\n\n function isLabelDefinition(obj: NestedLabels | LabelDefinition): obj is LabelDefinition\n {\n return (\n 'key' in obj &&\n 'defaultValue' in obj &&\n typeof obj.key === 'string' &&\n (typeof obj.defaultValue === 'string' || typeof obj.defaultValue === 'object')\n );\n }\n\n function traverse(obj: NestedLabels | LabelDefinition)\n {\n if (isLabelDefinition(obj))\n {\n // LabelDefinition인 경우\n result.push({\n key: obj.key,\n defaultValue: obj.defaultValue,\n description: obj.description,\n });\n }\n else\n {\n // NestedLabels인 경우\n Object.values(obj).forEach((value) =>\n {\n if (typeof value === 'object' && value !== null)\n {\n traverse(value);\n }\n });\n }\n }\n\n traverse(labels);\n return result;\n}\n\n/**\n * 섹션 정의에서 모든 라벨 추출\n *\n * @param definition - 섹션 정의\n * @returns 플랫화된 라벨 배열\n */\nexport function extractLabels(definition: SectionDefinition): FlatLabel[]\n{\n return flattenLabels(definition.labels);\n}\n"],"mappings":";AAwBO,SAAS,cAAc,QAC9B;AACI,QAAM,SAAsB,CAAC;AAE7B,WAAS,kBAAkB,KAC3B;AACI,WACI,SAAS,OACT,kBAAkB,OAClB,OAAO,IAAI,QAAQ,aAClB,OAAO,IAAI,iBAAiB,YAAY,OAAO,IAAI,iBAAiB;AAAA,EAE7E;AAEA,WAAS,SAAS,KAClB;AACI,QAAI,kBAAkB,GAAG,GACzB;AAEI,aAAO,KAAK;AAAA,QACR,KAAK,IAAI;AAAA,QACT,cAAc,IAAI;AAAA,QAClB,aAAa,IAAI;AAAA,MACrB,CAAC;AAAA,IACL,OAEA;AAEI,aAAO,OAAO,GAAG,EAAE,QAAQ,CAAC,UAC5B;AACI,YAAI,OAAO,UAAU,YAAY,UAAU,MAC3C;AACI,mBAAS,KAAK;AAAA,QAClB;AAAA,MACJ,CAAC;AAAA,IACL;AAAA,EACJ;AAEA,WAAS,MAAM;AACf,SAAO;AACX;AAQO,SAAS,cAAc,YAC9B;AACI,SAAO,cAAc,WAAW,MAAM;AAC1C;","names":[]}
|
|
@@ -1,8 +1,207 @@
|
|
|
1
|
+
import { CmsLabel, NewCmsLabel } from '../entities/cms-labels.js';
|
|
2
|
+
import { CmsLabelValue, NewCmsLabelValue } from '../entities/cms-label-values.js';
|
|
3
|
+
import { NewCmsDraftCache } from '../entities/cms-draft-cache.js';
|
|
4
|
+
import { NewCmsPublishedCache } from '../entities/cms-published-cache.js';
|
|
5
|
+
import 'drizzle-orm/pg-core';
|
|
6
|
+
|
|
1
7
|
/**
|
|
2
|
-
* CMS
|
|
8
|
+
* CMS Labels Repository
|
|
9
|
+
*
|
|
10
|
+
* 라벨 메타데이터 관리를 위한 Repository
|
|
3
11
|
*/
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
12
|
+
|
|
13
|
+
/**
|
|
14
|
+
* 라벨 목록 조회 (페이지네이션)
|
|
15
|
+
*/
|
|
16
|
+
declare function findMany(options?: {
|
|
17
|
+
section?: string;
|
|
18
|
+
limit?: number;
|
|
19
|
+
offset?: number;
|
|
20
|
+
}): Promise<CmsLabel[]>;
|
|
21
|
+
/**
|
|
22
|
+
* 전체 라벨 수 조회
|
|
23
|
+
*/
|
|
24
|
+
declare function count(section?: string): Promise<number>;
|
|
25
|
+
/**
|
|
26
|
+
* ID로 라벨 조회
|
|
27
|
+
*/
|
|
28
|
+
declare function findById(id: number): Promise<CmsLabel | null>;
|
|
29
|
+
/**
|
|
30
|
+
* Key로 라벨 조회
|
|
31
|
+
*/
|
|
32
|
+
declare function findByKey(key: string): Promise<CmsLabel | null>;
|
|
33
|
+
/**
|
|
34
|
+
* 섹션으로 모든 라벨 조회
|
|
35
|
+
*/
|
|
36
|
+
declare function findBySection$1(section: string): Promise<CmsLabel[]>;
|
|
37
|
+
/**
|
|
38
|
+
* 라벨 생성
|
|
39
|
+
*/
|
|
40
|
+
declare function create(data: NewCmsLabel): Promise<CmsLabel>;
|
|
41
|
+
/**
|
|
42
|
+
* 라벨 수정
|
|
43
|
+
*/
|
|
44
|
+
declare function updateById(id: number, data: Partial<NewCmsLabel>): Promise<CmsLabel | null>;
|
|
45
|
+
/**
|
|
46
|
+
* 라벨 삭제
|
|
47
|
+
*/
|
|
48
|
+
declare function deleteById(id: number): Promise<CmsLabel | null>;
|
|
49
|
+
declare const cmsLabelsRepository: {
|
|
50
|
+
findMany: typeof findMany;
|
|
51
|
+
count: typeof count;
|
|
52
|
+
findById: typeof findById;
|
|
53
|
+
findByKey: typeof findByKey;
|
|
54
|
+
findBySection: typeof findBySection$1;
|
|
55
|
+
create: typeof create;
|
|
56
|
+
updateById: typeof updateById;
|
|
57
|
+
deleteById: typeof deleteById;
|
|
58
|
+
};
|
|
59
|
+
|
|
60
|
+
/**
|
|
61
|
+
* CMS Label Values Repository
|
|
62
|
+
*
|
|
63
|
+
* 라벨 값 관리를 위한 Repository
|
|
64
|
+
*/
|
|
65
|
+
|
|
66
|
+
/**
|
|
67
|
+
* 특정 라벨의 특정 버전 값들 조회
|
|
68
|
+
*/
|
|
69
|
+
declare function findByLabelIdAndVersion(labelId: number, version: number, options?: {
|
|
70
|
+
locale?: string;
|
|
71
|
+
breakpoint?: string | null;
|
|
72
|
+
}): Promise<CmsLabelValue[]>;
|
|
73
|
+
/**
|
|
74
|
+
* 값 저장 (upsert)
|
|
75
|
+
*/
|
|
76
|
+
declare function upsert$2(data: NewCmsLabelValue): Promise<CmsLabelValue>;
|
|
77
|
+
/**
|
|
78
|
+
* 여러 값 일괄 저장
|
|
79
|
+
*/
|
|
80
|
+
declare function upsertMany(values: NewCmsLabelValue[]): Promise<CmsLabelValue[]>;
|
|
81
|
+
/**
|
|
82
|
+
* 특정 버전의 모든 값 삭제
|
|
83
|
+
*/
|
|
84
|
+
declare function deleteByVersion(labelId: number, version: number): Promise<CmsLabelValue[]>;
|
|
85
|
+
declare const cmsLabelValuesRepository: {
|
|
86
|
+
findByLabelIdAndVersion: typeof findByLabelIdAndVersion;
|
|
87
|
+
upsert: typeof upsert$2;
|
|
88
|
+
upsertMany: typeof upsertMany;
|
|
89
|
+
deleteByVersion: typeof deleteByVersion;
|
|
90
|
+
};
|
|
91
|
+
|
|
92
|
+
/**
|
|
93
|
+
* CMS Draft Cache Repository
|
|
94
|
+
*
|
|
95
|
+
* 관리자별 초안 캐시 관리 (동시 편집 지원)
|
|
96
|
+
*/
|
|
97
|
+
|
|
98
|
+
/**
|
|
99
|
+
* 섹션 + 언어 + 사용자로 초안 캐시 조회
|
|
100
|
+
*/
|
|
101
|
+
declare function findByUser(section: string, locale: string, userId: string): Promise<{
|
|
102
|
+
section: string;
|
|
103
|
+
locale: string;
|
|
104
|
+
content: unknown;
|
|
105
|
+
id: number;
|
|
106
|
+
updatedAt: Date;
|
|
107
|
+
userId: string;
|
|
108
|
+
} | null>;
|
|
109
|
+
/**
|
|
110
|
+
* 초안 캐시 생성 또는 업데이트 (UPSERT)
|
|
111
|
+
*/
|
|
112
|
+
declare function upsert$1(data: NewCmsDraftCache): Promise<{
|
|
113
|
+
section: string;
|
|
114
|
+
locale: string;
|
|
115
|
+
content: unknown;
|
|
116
|
+
id: number;
|
|
117
|
+
updatedAt: Date;
|
|
118
|
+
userId: string;
|
|
119
|
+
}>;
|
|
120
|
+
/**
|
|
121
|
+
* 특정 사용자의 모든 초안 조회
|
|
122
|
+
*/
|
|
123
|
+
declare function findAllByUser(userId: string): Promise<{
|
|
124
|
+
section: string;
|
|
125
|
+
locale: string;
|
|
126
|
+
content: unknown;
|
|
127
|
+
id: number;
|
|
128
|
+
updatedAt: Date;
|
|
129
|
+
userId: string;
|
|
130
|
+
}[]>;
|
|
131
|
+
/**
|
|
132
|
+
* 초안 삭제
|
|
133
|
+
*/
|
|
134
|
+
declare function deleteByUser(section: string, locale: string, userId: string): Promise<void>;
|
|
135
|
+
/**
|
|
136
|
+
* 오래된 초안 정리 (30일 이상 미사용)
|
|
137
|
+
*/
|
|
138
|
+
declare function cleanupOldDrafts(daysOld?: number): Promise<{
|
|
139
|
+
section: string;
|
|
140
|
+
locale: string;
|
|
141
|
+
content: unknown;
|
|
142
|
+
id: number;
|
|
143
|
+
updatedAt: Date;
|
|
144
|
+
userId: string;
|
|
145
|
+
}[]>;
|
|
146
|
+
declare const cmsDraftCacheRepository: {
|
|
147
|
+
findByUser: typeof findByUser;
|
|
148
|
+
upsert: typeof upsert$1;
|
|
149
|
+
findAllByUser: typeof findAllByUser;
|
|
150
|
+
deleteByUser: typeof deleteByUser;
|
|
151
|
+
cleanupOldDrafts: typeof cleanupOldDrafts;
|
|
152
|
+
};
|
|
153
|
+
|
|
154
|
+
/**
|
|
155
|
+
* CMS Published Cache Repository
|
|
156
|
+
*
|
|
157
|
+
* 발행된 콘텐츠 캐시 관리 (초고속 조회)
|
|
158
|
+
*/
|
|
159
|
+
|
|
160
|
+
/**
|
|
161
|
+
* 섹션 + 언어로 발행된 캐시 조회
|
|
162
|
+
*/
|
|
163
|
+
declare function findBySection(section: string, locale?: string): Promise<{
|
|
164
|
+
section: string;
|
|
165
|
+
locale: string;
|
|
166
|
+
content: unknown;
|
|
167
|
+
version: number;
|
|
168
|
+
publishedAt: Date;
|
|
169
|
+
id: number;
|
|
170
|
+
publishedBy: string | null;
|
|
171
|
+
} | null>;
|
|
172
|
+
/**
|
|
173
|
+
* 캐시 생성 또는 업데이트 (UPSERT)
|
|
174
|
+
*/
|
|
175
|
+
declare function upsert(data: NewCmsPublishedCache): Promise<{
|
|
176
|
+
section: string;
|
|
177
|
+
locale: string;
|
|
178
|
+
content: unknown;
|
|
179
|
+
version: number;
|
|
180
|
+
publishedAt: Date;
|
|
181
|
+
id: number;
|
|
182
|
+
publishedBy: string | null;
|
|
183
|
+
}>;
|
|
184
|
+
/**
|
|
185
|
+
* 섹션별 모든 언어 캐시 조회
|
|
186
|
+
*/
|
|
187
|
+
declare function findAllLanguages(section: string): Promise<{
|
|
188
|
+
section: string;
|
|
189
|
+
locale: string;
|
|
190
|
+
content: unknown;
|
|
191
|
+
version: number;
|
|
192
|
+
publishedAt: Date;
|
|
193
|
+
id: number;
|
|
194
|
+
publishedBy: string | null;
|
|
195
|
+
}[]>;
|
|
196
|
+
/**
|
|
197
|
+
* 캐시 삭제
|
|
198
|
+
*/
|
|
199
|
+
declare function deleteBySection(section: string, locale?: string): Promise<void>;
|
|
200
|
+
declare const cmsPublishedCacheRepository: {
|
|
201
|
+
findBySection: typeof findBySection;
|
|
202
|
+
upsert: typeof upsert;
|
|
203
|
+
findAllLanguages: typeof findAllLanguages;
|
|
204
|
+
deleteBySection: typeof deleteBySection;
|
|
205
|
+
};
|
|
206
|
+
|
|
207
|
+
export { cmsDraftCacheRepository, cmsLabelValuesRepository, cmsLabelsRepository, cmsPublishedCacheRepository };
|