sonamu 0.9.6 → 0.9.8
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/api/decorators.js +2 -2
- package/dist/syncer/file-patterns.js +2 -2
- package/dist/syncer/syncer-actions.d.ts +8 -2
- package/dist/syncer/syncer-actions.d.ts.map +1 -1
- package/dist/syncer/syncer-actions.js +11 -4
- package/dist/syncer/syncer.d.ts.map +1 -1
- package/dist/syncer/syncer.js +4 -3
- package/dist/utils/formatter.d.ts.map +1 -1
- package/dist/utils/formatter.js +4 -5
- package/dist/utils/process-utils.d.ts +5 -5
- package/dist/utils/process-utils.js +6 -6
- package/package.json +1 -1
- package/src/api/decorators.ts +2 -2
- package/src/syncer/file-patterns.ts +1 -1
- package/src/syncer/syncer-actions.ts +15 -3
- package/src/syncer/syncer.ts +12 -5
- package/src/utils/formatter.ts +4 -6
- package/src/utils/process-utils.ts +5 -5
package/dist/api/decorators.js
CHANGED
|
@@ -175,7 +175,7 @@ function transactional(options = {}) {
|
|
|
175
175
|
return (target, propertyKey, descriptor) => {
|
|
176
176
|
const originalMethod = descriptor.value;
|
|
177
177
|
const modelName = target.constructor.name.match(/(.+)Class$/)?.[1];
|
|
178
|
-
assert(modelName, `modelName is required on @
|
|
178
|
+
assert(modelName, `modelName is required on @transactional decorator on ${target.constructor.name}.${propertyKey}`);
|
|
179
179
|
const methodName = propertyKey;
|
|
180
180
|
descriptor.value = async function(...args) {
|
|
181
181
|
this.logger.debug("transactional: {model}.{method}", {
|
|
@@ -308,4 +308,4 @@ var init_decorators = __esmMin((() => {
|
|
|
308
308
|
//#endregion
|
|
309
309
|
init_decorators();
|
|
310
310
|
export { api, init_decorators, registeredApis, stream, transactional, upload, websocket };
|
|
311
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"decorators.js","names":["api","registeredApis: {\n  /**\n   * modelName은 모델 클래스 이름입니다. (ex. \"UserModel\")\n   */\n  modelName: string;\n  methodName: string;\n  path: string;\n  options: ApiDecoratorOptions;\n  streamOptions?: StreamDecoratorOptions;\n  websocketOptions?: ResolvedWebSocketDecoratorOptions;\n  uploadOptions?: UploadDecoratorOptions;\n}[]"],"sources":["../../src/api/decorators.ts"],"sourcesContent":["import assert from \"assert\";\n\nimport { type FastifyMultipartBaseOptions } from \"@fastify/multipart\";\nimport { getLogger } from \"@logtape/logtape\";\nimport { type HTTPMethods } from \"fastify\";\nimport inflection from \"inflection\";\nimport { isEqual } from \"radashi\";\nimport { type z } from \"zod\";\n\nimport { type CacheControlConfig } from \"../cache-control/types\";\nimport { type CompressConfig } from \"../compress/types\";\nimport { BaseModelClass } from \"../database/base-model\";\nimport { DB } from \"../database/db\";\nimport { PuriTransactionWrapper } from \"../database/puri-wrapper\";\nimport { type TransactionalOptions } from \"../database/puri-wrapper\";\nimport { UpsertBuilder } from \"../database/upsert-builder\";\nimport { convertDomainToCategory } from \"../logger/category\";\nimport { type DriverKey } from \"../storage/drivers\";\nimport { type KeyGenerator } from \"../storage/types\";\nimport { type ApiParam, type ApiParamType } from \"../types/types\";\nimport { BaseFrameClass } from \"./base-frame\";\n\nexport interface GuardKeys {\n  query: true;\n  admin: true;\n  user: true;\n}\nexport type GuardKey = keyof GuardKeys;\nexport type ServiceClient =\n  | \"axios\"\n  | \"axios-multipart\"\n  | \"tanstack-query\"\n  | \"tanstack-mutation\"\n  | \"tanstack-mutation-multipart\"\n  | \"window-fetch\";\nexport type ApiDecoratorOptions = {\n  httpMethod?: HTTPMethods;\n  contentType?:\n    | \"text/plain\"\n    | \"text/html\"\n    | \"text/xml\"\n    | \"application/json\"\n    | \"application/octet-stream\";\n  clients?: ServiceClient[];\n  path?: string;\n  resourceName?: string;\n  guards?: GuardKey[];\n  description?: string;\n  timeout?: number;\n  /** API 응답의 Cache-Control 헤더 설정. 설정하지 않으면 cacheControlHandler 또는 기본값이 적용됩니다. */\n  cacheControl?: CacheControlConfig;\n  /** API 응답의 압축 설정. false로 설정하면 압축을 비활성화합니다. */\n  compress?: CompressConfig;\n};\nexport type StreamDecoratorOptions = {\n  type: \"sse\"; // | 'ws\n  // oxlint-disable-next-line @typescript-eslint/no-explicit-any -- 이벤트 키별로 넘겨주는 값이므로 어떤 타입이든 상관없음\n  events: z.ZodObject<any>;\n  path?: string;\n  resourceName?: string;\n  guards?: GuardKey[];\n  description?: string;\n};\nexport type WebSocketDecoratorOptions = {\n  // oxlint-disable-next-line @typescript-eslint/no-explicit-any -- 이벤트 키별로 넘겨주는 값이므로 어떤 타입이든 상관없음\n  outEvents: z.ZodObject<any>;\n  // oxlint-disable-next-line @typescript-eslint/no-explicit-any -- 이벤트 키별로 넘겨주는 값이므로 어떤 타입이든 상관없음\n  inEvents: z.ZodObject<any>;\n  path?: string;\n  resourceName?: string;\n  guards?: GuardKey[];\n  description?: string;\n  heartbeat?: number;\n  maxPayload?: number;\n  namespace?: string;\n};\nexport type ResolvedWebSocketDecoratorOptions = WebSocketDecoratorOptions & {\n  // codegen이 타입 이름을 재사용할 수 있도록 syncer가 AST에서 보강하는 메타데이터\n  outEventsTypeRef?: ApiParamType.Ref;\n  inEventsTypeRef?: ApiParamType.Ref;\n};\n\ntype BufferUploadOptions = {\n  consume?: \"buffer\";\n};\ntype StreamUploadOptions = {\n  consume: \"stream\";\n  destination: DriverKey;\n  keyGenerator?: KeyGenerator;\n};\nexport type UploadDecoratorOptions = {\n  guards?: GuardKey[];\n  description?: string;\n  limits?: FastifyMultipartBaseOptions[\"limits\"];\n} & (BufferUploadOptions | StreamUploadOptions);\n\nexport const registeredApis: {\n  /**\n   * modelName은 모델 클래스 이름입니다. (ex. \"UserModel\")\n   */\n  modelName: string;\n  methodName: string;\n  path: string;\n  options: ApiDecoratorOptions;\n  streamOptions?: StreamDecoratorOptions;\n  websocketOptions?: ResolvedWebSocketDecoratorOptions;\n  uploadOptions?: UploadDecoratorOptions;\n}[] = [];\nexport type ExtendedApi = {\n  modelName: string;\n  methodName: string;\n  path: string;\n  options: ApiDecoratorOptions;\n  streamOptions?: StreamDecoratorOptions;\n  websocketOptions?: ResolvedWebSocketDecoratorOptions;\n  uploadOptions?: UploadDecoratorOptions;\n  typeParameters: ApiParamType.TypeParam[];\n  parameters: ApiParam[];\n  returnType: ApiParamType;\n};\ntype DecoratorTarget = { constructor: { name: string } };\n\nconst DECORATOR_TYPES = {\n  API: Symbol(\"api\"),\n  STREAM: Symbol(\"stream\"),\n  WEBSOCKET: Symbol(\"websocket\"),\n  UPLOAD: Symbol(\"upload\"),\n} as const;\n\nfunction checkSingleDecorator(target: DecoratorTarget, propertyKey: string, decoratorType: symbol) {\n  const method = target[propertyKey as keyof typeof target] as { __decoratorType?: symbol };\n  if (method?.__decoratorType && method?.__decoratorType !== decoratorType) {\n    throw new Error(\n      `@${decoratorType.description ?? String(decoratorType)} decorator can only be used once on ${target.constructor.name}.${propertyKey}. You can use only one of @api, @stream, @websocket, or @upload decorator on the same method.`,\n    );\n  } else {\n    method.__decoratorType = decoratorType;\n  }\n}\n\nexport function api(options: ApiDecoratorOptions = {}) {\n  options = {\n    httpMethod: \"GET\",\n    contentType: \"application/json\",\n    clients: [\"axios\"],\n    ...options,\n  };\n\n  return (target: DecoratorTarget, propertyKey: string, descriptor: PropertyDescriptor) => {\n    const modelName = target.constructor.name.match(/(.+)Class$/)?.[1];\n    assert(\n      modelName,\n      `modelName is required on @api decorator on ${target.constructor.name}.${propertyKey}`,\n    );\n    const methodName = propertyKey;\n\n    // 메서드에 걸린 데코레이터 중복 체크\n    checkSingleDecorator(target, propertyKey, DECORATOR_TYPES.API);\n\n    const defaultPath = `/${inflection.camelize(\n      modelName.replace(/Model$/, \"\").replace(/Frame$/, \"\"),\n      true,\n    )}/${inflection.camelize(propertyKey, true)}`;\n    const path = options.path ?? defaultPath;\n\n    // 기존 동일한 메서드가 있는지 확인 후 있는 경우 override\n    const existingApi = registeredApis.find(\n      (api) => api.modelName === modelName && api.methodName === methodName,\n    );\n    if (existingApi) {\n      // 기존의 path와 새로운 path가 다르다면(=빈 스트링이 아니었는데 다른 스트링으로 바뀌게 된다면) 에러를 터뜨려줍니다.\n      assertNoConflictingPath(\"api\", modelName, methodName, existingApi.path, path);\n      existingApi.path = path;\n\n      // 기존의 옵션과 새로운 옵션이 겹치는 부분이 있다면 에러를 터뜨려줍니다.\n      assertNoConflictingOptions(\"api\", modelName, methodName, existingApi.options, options);\n      existingApi.options = {\n        ...existingApi.options, // 기존의 옵션을 존중하되\n        ...options, // @api 데코레이터의 옵션을 추가합니다.\n      };\n    } else {\n      registeredApis.push({\n        modelName,\n        methodName,\n        path,\n        options,\n      });\n    }\n\n    const originalMethod = descriptor.value;\n    descriptor.value = async function (this: BaseModelClass | BaseFrameClass, ...args: unknown[]) {\n      if (this instanceof BaseModelClass) {\n        getLogger(convertDomainToCategory(this.modelName, \"model\")).debug(\n          \"api: {httpMethod} {model}.{method}\",\n          {\n            httpMethod: options.httpMethod,\n            model: modelName,\n            method: methodName,\n          },\n        );\n      }\n\n      if (this instanceof BaseFrameClass) {\n        getLogger(convertDomainToCategory(this.frameName, \"frame\")).debug(\n          \"api: {httpMethod} {model}.{method}\",\n          {\n            httpMethod: options.httpMethod,\n            model: modelName,\n            method: methodName,\n          },\n        );\n      }\n\n      return originalMethod.apply(this, args);\n    };\n  };\n}\n\nexport function stream(options: StreamDecoratorOptions) {\n  return (target: DecoratorTarget, propertyKey: string, descriptor: PropertyDescriptor) => {\n    const modelName = target.constructor.name.match(/(.+)Class$/)?.[1];\n    assert(\n      modelName,\n      `modelName is required on @stream decorator on ${target.constructor.name}.${propertyKey}`,\n    );\n    const methodName = propertyKey;\n\n    // 메서드에 걸린 데코레이터 중복 체크\n    checkSingleDecorator(target, propertyKey, DECORATOR_TYPES.STREAM);\n\n    const defaultPath = `/${inflection.camelize(\n      modelName.replace(/Model$/, \"\").replace(/Frame$/, \"\"),\n      true,\n    )}/${inflection.camelize(propertyKey, true)}`;\n    const path = options.path ?? defaultPath;\n    // stream 전용 필드(events, type)는 ApiDecoratorOptions에 속하지 않으므로 제외\n    const { events: _, type: _type, ...apiOptions } = options;\n    const optionsWithDefaults = {\n      ...apiOptions,\n      httpMethod: \"GET\" as HTTPMethods,\n    };\n\n    const existingApi = registeredApis.find(\n      (api) => api.modelName === modelName && api.methodName === methodName,\n    );\n    if (existingApi) {\n      // 기존의 path와 새로운 path가 다르다면(=빈 스트링이 아니었는데 다른 스트링으로 바뀌게 된다면) 에러를 터뜨려줍니다.\n      assertNoConflictingPath(\"stream\", modelName, methodName, existingApi.path, path);\n      existingApi.path = path;\n\n      // 기존의 옵션과 새로운 옵션이 겹치는 부분이 있다면 에러를 터뜨려줍니다.\n      assertNoConflictingOptions(\n        \"stream\",\n        modelName,\n        methodName,\n        existingApi.options,\n        optionsWithDefaults,\n      );\n      existingApi.options = {\n        ...existingApi.options, // 기존의 옵션을 존중하되\n        ...optionsWithDefaults, // @stream 데코레이터의 옵션을 추가합니다.\n      };\n\n      existingApi.streamOptions = options;\n    } else {\n      registeredApis.push({\n        modelName,\n        methodName,\n        path,\n        options: optionsWithDefaults,\n        streamOptions: options,\n      });\n    }\n\n    const originalMethod = descriptor.value;\n    descriptor.value = async function (this: BaseModelClass | BaseFrameClass, ...args: unknown[]) {\n      if (this instanceof BaseModelClass) {\n        getLogger(convertDomainToCategory(this.modelName, \"model\")).debug(\n          \"stream: {model}.{method}\",\n          {\n            model: modelName,\n            method: methodName,\n          },\n        );\n      }\n\n      if (this instanceof BaseFrameClass) {\n        getLogger(convertDomainToCategory(this.frameName, \"frame\")).debug(\n          \"stream: {model}.{method}\",\n          {\n            model: modelName,\n            method: methodName,\n          },\n        );\n      }\n\n      return originalMethod.apply(this, args);\n    };\n  };\n}\n\nexport function websocket(options: WebSocketDecoratorOptions) {\n  return (target: DecoratorTarget, propertyKey: string, descriptor: PropertyDescriptor) => {\n    const modelName = target.constructor.name.match(/(.+)Class$/)?.[1];\n    assert(\n      modelName,\n      `modelName is required on @websocket decorator on ${target.constructor.name}.${propertyKey}`,\n    );\n    const methodName = propertyKey;\n\n    checkSingleDecorator(target, propertyKey, DECORATOR_TYPES.WEBSOCKET);\n\n    const defaultPath = `/${inflection.camelize(\n      modelName.replace(/Model$/, \"\").replace(/Frame$/, \"\"),\n      true,\n    )}/${inflection.camelize(propertyKey, true)}`;\n    const path = options.path ?? defaultPath;\n    const { outEvents: _outEvents, inEvents: _inEvents, ...apiOptions } = options;\n    const optionsWithDefaults = {\n      ...apiOptions,\n      httpMethod: \"GET\" as HTTPMethods,\n    };\n\n    const existingApi = registeredApis.find(\n      (api) => api.modelName === modelName && api.methodName === methodName,\n    );\n    if (existingApi) {\n      assertNoConflictingPath(\"websocket\", modelName, methodName, existingApi.path, path);\n      existingApi.path = path;\n\n      assertNoConflictingOptions(\n        \"websocket\",\n        modelName,\n        methodName,\n        existingApi.options,\n        optionsWithDefaults,\n      );\n      existingApi.options = {\n        ...existingApi.options,\n        ...optionsWithDefaults,\n      };\n\n      existingApi.websocketOptions = options;\n    } else {\n      registeredApis.push({\n        modelName,\n        methodName,\n        path,\n        options: optionsWithDefaults,\n        websocketOptions: options,\n      });\n    }\n\n    const originalMethod = descriptor.value;\n    descriptor.value = async function (this: BaseModelClass | BaseFrameClass, ...args: unknown[]) {\n      if (this instanceof BaseModelClass) {\n        getLogger(convertDomainToCategory(this.modelName, \"model\")).debug(\n          \"websocket: {model}.{method}\",\n          {\n            model: modelName,\n            method: methodName,\n          },\n        );\n      }\n\n      if (this instanceof BaseFrameClass) {\n        getLogger(convertDomainToCategory(this.frameName, \"frame\")).debug(\n          \"websocket: {model}.{method}\",\n          {\n            model: modelName,\n            method: methodName,\n          },\n        );\n      }\n\n      return originalMethod.apply(this, args);\n    };\n  };\n}\n\nexport function transactional(options: TransactionalOptions = {}) {\n  const { isolation, readOnly, dbPreset = \"w\" } = options;\n\n  return (target: DecoratorTarget, propertyKey: string, descriptor: PropertyDescriptor) => {\n    const originalMethod = descriptor.value;\n    const modelName = target.constructor.name.match(/(.+)Class$/)?.[1];\n    assert(\n      modelName,\n      `modelName is required on @stream decorator on ${target.constructor.name}.${propertyKey}`,\n    );\n    const methodName = propertyKey;\n\n    descriptor.value = async function (this: BaseModelClass, ...args: unknown[]) {\n      this.logger.debug(\"transactional: {model}.{method}\", {\n        model: modelName,\n        method: methodName,\n      });\n\n      const existingContext = DB.transactionStorage.getStore();\n\n      // AsyncLocalStorage 컨텍스트 없거나 해당 preset의 트랜잭션이 없으면 새로 시작\n      const startTransaction = async () => {\n        const puri = this.getPuri(dbPreset);\n\n        return puri.knex.transaction(\n          async (trx) => {\n            this.logger.debug(\"new transaction context: {dbPreset}\", { dbPreset });\n            const trxWrapper = new PuriTransactionWrapper(trx, new UpsertBuilder());\n            // TransactionContext에 트랜잭션 저장\n            DB.getTransactionContext().setTransaction(dbPreset, trxWrapper);\n\n            try {\n              return await originalMethod.apply(this, args);\n            } finally {\n              // 트랜잭션 제거\n              this.logger.debug(\"delete transaction context: {dbPreset}\", { dbPreset });\n              DB.getTransactionContext().deleteTransaction(dbPreset);\n            }\n          },\n          { isolationLevel: isolation, readOnly },\n        );\n      };\n\n      // AsyncLocalStorage 컨텍스트가 없으면 새로 생성\n      if (!existingContext) {\n        return DB.runWithTransaction(startTransaction);\n      }\n\n      // 이미 AsyncLocalStorage 컨텍스트 안에 있는지 확인 후 해당 preset의 트랜잭션이 이미 있으면 재사용\n      if (existingContext?.getTransaction(dbPreset)) {\n        this.logger.debug(\"reuse transaction context: {dbPreset}\", { dbPreset });\n        return originalMethod.apply(this, args);\n      }\n\n      // 컨텍스트는 있지만 이 preset의 트랜잭션은 없는 경우 (같은 컨텍스트 내에서 실행)\n      return startTransaction();\n    };\n\n    return descriptor;\n  };\n}\n\n/**\n * 파일 업로드 API를 생성해줍니다. (@api 데코레이터 없이 독립적으로 사용)\n * @param options\n * @returns\n */\nexport function upload(options: UploadDecoratorOptions = { consume: \"buffer\" }) {\n  return (target: DecoratorTarget, propertyKey: string, descriptor: PropertyDescriptor) => {\n    const originalMethod = descriptor.value;\n    const modelName = target.constructor.name.match(/(.+)Class$/)?.[1];\n    assert(\n      modelName,\n      `modelName is required on @upload decorator on ${target.constructor.name}.${propertyKey}`,\n    );\n    const methodName = propertyKey;\n\n    // 메서드에 걸린 데코레이터 중복 체크\n    checkSingleDecorator(target, propertyKey, DECORATOR_TYPES.UPLOAD);\n\n    const defaultPath = `/${inflection.camelize(\n      modelName.replace(/Model$/, \"\").replace(/Frame$/, \"\"),\n      true,\n    )}/${inflection.camelize(methodName, true)}`;\n\n    // registeredApis에서 해당 API 찾아서 uploadOptions 추가\n    const existingApi = registeredApis.find(\n      (api) => api.modelName === modelName && api.methodName === methodName,\n    );\n\n    if (existingApi) {\n      // 재등록 시 업로드 옵션만 갱신\n      existingApi.uploadOptions = options;\n    } else {\n      // 새로 등록\n      registeredApis.push({\n        modelName,\n        methodName,\n        path: defaultPath,\n        options: {\n          httpMethod: \"POST\",\n          clients: [\"axios-multipart\", \"tanstack-mutation-multipart\"],\n          guards: options.guards,\n          description: options.description,\n        },\n        uploadOptions: options,\n      });\n    }\n\n    descriptor.value = async function (this: BaseModelClass | BaseFrameClass, ...args: unknown[]) {\n      if (this instanceof BaseModelClass) {\n        getLogger(convertDomainToCategory(this.modelName, \"model\")).debug(\n          \"upload: {model}.{method}\",\n          {\n            model: modelName,\n            method: methodName,\n          },\n        );\n      }\n\n      if (this instanceof BaseFrameClass) {\n        getLogger(convertDomainToCategory(this.frameName, \"frame\")).debug(\n          \"upload: {model}.{method}\",\n          {\n            model: modelName,\n            method: methodName,\n          },\n        );\n      }\n\n      return originalMethod.apply(this, args);\n    };\n\n    return descriptor;\n  };\n}\n\n/**\n * 기존의 path와 새로운 path가 다르다면(=값이 있던 스트링이 다른 값이 있는 스트링으로 바뀌게 된다면) 에러를 터뜨려줍니다.\n * @param decoratorName 데코레이터 이름\n * @param modelName 모델 이름\n * @param methodName 메서드 이름\n * @param existingPath 기존의 path\n * @param newPath 새로운 path\n */\nfunction assertNoConflictingPath(\n  decoratorName: string,\n  modelName: string,\n  methodName: string,\n  existingPath: string,\n  newPath: string,\n) {\n  if (existingPath !== \"\" && newPath !== \"\" && existingPath !== newPath) {\n    // 이것이 무슨 상황이냐면요, api.path가 덮어씌워지는 상황입니다.\n    // 가령 @api({ path: \"/api/v1/users\" }) 데코레이터가 붙어있는 메서드에\n    // @stream({ path: \"/api/v1/users/stream\" }) 같은 것이 붙어 있는 상황입니다.\n    // 이렇게 되면 두 데코레이터가 같은 api의 path 필드를 건드리게 되므로, 에러를 터뜨려줍니다.\n    throw new Error(\n      `@${decoratorName} decorator on ${modelName}.${methodName} has conflicting path: ${newPath}. The decorator is trying to override the existing path(${existingPath}) with the new path(${newPath}).`,\n    );\n  }\n}\n\n/**\n * 기존의 옵션과 새로운 옵션이 겹치는 부분이 있다면 에러를 터뜨려줍니다.\n * @param decoratorName 데코레이터 이름\n * @param modelName 모델 이름\n * @param methodName 메서드 이름\n * @param existingOptions 기존의 옵션\n * @param newOptions 새로운 옵션\n */\nfunction assertNoConflictingOptions(\n  decoratorName: string,\n  modelName: string,\n  methodName: string,\n  // oxlint-disable-next-line @typescript-eslint/no-explicit-any -- <아 쉽게쉽게 좀 갑시다>\n  existingOptions: Record<string, any>,\n  // oxlint-disable-next-line @typescript-eslint/no-explicit-any -- <이럴 때 아니면 any 언제 씁니까>\n  newOptions: Record<string, any>,\n) {\n  Object.keys(newOptions).forEach((key) => {\n    if (existingOptions[key] && !isEqual(existingOptions[key], newOptions[key])) {\n      // 이것이 무슨 상황이냐면요, api.options가 덮어씌워지는 상황입니다.\n      // 가령 @api({ resourceName: \"Users\" }) 데코레이터가 붙어있는 메서드에\n      // @stream({ resourceName: \"Posts\" }) 같은 것이 붙어 있는 상황입니다.\n      // 이렇게 되면 두 데코레이터가 같은 api의 options 속 같은 필드를 건드리게 되므로, 에러를 터뜨려줍니다.\n      throw new Error(\n        `@${decoratorName} decorator on ${modelName}.${methodName} has conflicting options: ${key}. The decorator is trying to override the existing option(${JSON.stringify(existingOptions[key])}) with the new option(${JSON.stringify(newOptions[key])}).`,\n      );\n    }\n  });\n}\n"],"mappings":"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"}
|
|
311
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"decorators.js","names":["api","registeredApis: {\n  /**\n   * modelName은 모델 클래스 이름입니다. (ex. \"UserModel\")\n   */\n  modelName: string;\n  methodName: string;\n  path: string;\n  options: ApiDecoratorOptions;\n  streamOptions?: StreamDecoratorOptions;\n  websocketOptions?: ResolvedWebSocketDecoratorOptions;\n  uploadOptions?: UploadDecoratorOptions;\n}[]"],"sources":["../../src/api/decorators.ts"],"sourcesContent":["import assert from \"assert\";\n\nimport { type FastifyMultipartBaseOptions } from \"@fastify/multipart\";\nimport { getLogger } from \"@logtape/logtape\";\nimport { type HTTPMethods } from \"fastify\";\nimport inflection from \"inflection\";\nimport { isEqual } from \"radashi\";\nimport { type z } from \"zod\";\n\nimport { type CacheControlConfig } from \"../cache-control/types\";\nimport { type CompressConfig } from \"../compress/types\";\nimport { BaseModelClass } from \"../database/base-model\";\nimport { DB } from \"../database/db\";\nimport { PuriTransactionWrapper } from \"../database/puri-wrapper\";\nimport { type TransactionalOptions } from \"../database/puri-wrapper\";\nimport { UpsertBuilder } from \"../database/upsert-builder\";\nimport { convertDomainToCategory } from \"../logger/category\";\nimport { type DriverKey } from \"../storage/drivers\";\nimport { type KeyGenerator } from \"../storage/types\";\nimport { type ApiParam, type ApiParamType } from \"../types/types\";\nimport { BaseFrameClass } from \"./base-frame\";\n\nexport interface GuardKeys {\n  query: true;\n  admin: true;\n  user: true;\n}\nexport type GuardKey = keyof GuardKeys;\nexport type ServiceClient =\n  | \"axios\"\n  | \"axios-multipart\"\n  | \"tanstack-query\"\n  | \"tanstack-mutation\"\n  | \"tanstack-mutation-multipart\"\n  | \"window-fetch\";\nexport type ApiDecoratorOptions = {\n  httpMethod?: HTTPMethods;\n  contentType?:\n    | \"text/plain\"\n    | \"text/html\"\n    | \"text/xml\"\n    | \"application/json\"\n    | \"application/octet-stream\";\n  clients?: ServiceClient[];\n  path?: string;\n  resourceName?: string;\n  guards?: GuardKey[];\n  description?: string;\n  timeout?: number;\n  /** API 응답의 Cache-Control 헤더 설정. 설정하지 않으면 cacheControlHandler 또는 기본값이 적용됩니다. */\n  cacheControl?: CacheControlConfig;\n  /** API 응답의 압축 설정. false로 설정하면 압축을 비활성화합니다. */\n  compress?: CompressConfig;\n};\nexport type StreamDecoratorOptions = {\n  type: \"sse\";\n  // oxlint-disable-next-line @typescript-eslint/no-explicit-any -- 이벤트 키별로 넘겨주는 값이므로 어떤 타입이든 상관없음\n  events: z.ZodObject<any>;\n  path?: string;\n  resourceName?: string;\n  guards?: GuardKey[];\n  description?: string;\n};\nexport type WebSocketDecoratorOptions = {\n  // oxlint-disable-next-line @typescript-eslint/no-explicit-any -- 이벤트 키별로 넘겨주는 값이므로 어떤 타입이든 상관없음\n  outEvents: z.ZodObject<any>;\n  // oxlint-disable-next-line @typescript-eslint/no-explicit-any -- 이벤트 키별로 넘겨주는 값이므로 어떤 타입이든 상관없음\n  inEvents: z.ZodObject<any>;\n  path?: string;\n  resourceName?: string;\n  guards?: GuardKey[];\n  description?: string;\n  heartbeat?: number;\n  maxPayload?: number;\n  namespace?: string;\n};\nexport type ResolvedWebSocketDecoratorOptions = WebSocketDecoratorOptions & {\n  // codegen이 타입 이름을 재사용할 수 있도록 syncer가 AST에서 보강하는 메타데이터\n  outEventsTypeRef?: ApiParamType.Ref;\n  inEventsTypeRef?: ApiParamType.Ref;\n};\n\ntype BufferUploadOptions = {\n  consume?: \"buffer\";\n};\ntype StreamUploadOptions = {\n  consume: \"stream\";\n  destination: DriverKey;\n  keyGenerator?: KeyGenerator;\n};\nexport type UploadDecoratorOptions = {\n  guards?: GuardKey[];\n  description?: string;\n  limits?: FastifyMultipartBaseOptions[\"limits\"];\n} & (BufferUploadOptions | StreamUploadOptions);\n\nexport const registeredApis: {\n  /**\n   * modelName은 모델 클래스 이름입니다. (ex. \"UserModel\")\n   */\n  modelName: string;\n  methodName: string;\n  path: string;\n  options: ApiDecoratorOptions;\n  streamOptions?: StreamDecoratorOptions;\n  websocketOptions?: ResolvedWebSocketDecoratorOptions;\n  uploadOptions?: UploadDecoratorOptions;\n}[] = [];\nexport type ExtendedApi = {\n  modelName: string;\n  methodName: string;\n  path: string;\n  options: ApiDecoratorOptions;\n  streamOptions?: StreamDecoratorOptions;\n  websocketOptions?: ResolvedWebSocketDecoratorOptions;\n  uploadOptions?: UploadDecoratorOptions;\n  typeParameters: ApiParamType.TypeParam[];\n  parameters: ApiParam[];\n  returnType: ApiParamType;\n};\ntype DecoratorTarget = { constructor: { name: string } };\n\nconst DECORATOR_TYPES = {\n  API: Symbol(\"api\"),\n  STREAM: Symbol(\"stream\"),\n  WEBSOCKET: Symbol(\"websocket\"),\n  UPLOAD: Symbol(\"upload\"),\n} as const;\n\nfunction checkSingleDecorator(target: DecoratorTarget, propertyKey: string, decoratorType: symbol) {\n  const method = target[propertyKey as keyof typeof target] as { __decoratorType?: symbol };\n  if (method?.__decoratorType && method?.__decoratorType !== decoratorType) {\n    throw new Error(\n      `@${decoratorType.description ?? String(decoratorType)} decorator can only be used once on ${target.constructor.name}.${propertyKey}. You can use only one of @api, @stream, @websocket, or @upload decorator on the same method.`,\n    );\n  } else {\n    method.__decoratorType = decoratorType;\n  }\n}\n\nexport function api(options: ApiDecoratorOptions = {}) {\n  options = {\n    httpMethod: \"GET\",\n    contentType: \"application/json\",\n    clients: [\"axios\"],\n    ...options,\n  };\n\n  return (target: DecoratorTarget, propertyKey: string, descriptor: PropertyDescriptor) => {\n    const modelName = target.constructor.name.match(/(.+)Class$/)?.[1];\n    assert(\n      modelName,\n      `modelName is required on @api decorator on ${target.constructor.name}.${propertyKey}`,\n    );\n    const methodName = propertyKey;\n\n    // 메서드에 걸린 데코레이터 중복 체크\n    checkSingleDecorator(target, propertyKey, DECORATOR_TYPES.API);\n\n    const defaultPath = `/${inflection.camelize(\n      modelName.replace(/Model$/, \"\").replace(/Frame$/, \"\"),\n      true,\n    )}/${inflection.camelize(propertyKey, true)}`;\n    const path = options.path ?? defaultPath;\n\n    // 기존 동일한 메서드가 있는지 확인 후 있는 경우 override\n    const existingApi = registeredApis.find(\n      (api) => api.modelName === modelName && api.methodName === methodName,\n    );\n    if (existingApi) {\n      // 기존의 path와 새로운 path가 다르다면(=빈 스트링이 아니었는데 다른 스트링으로 바뀌게 된다면) 에러를 터뜨려줍니다.\n      assertNoConflictingPath(\"api\", modelName, methodName, existingApi.path, path);\n      existingApi.path = path;\n\n      // 기존의 옵션과 새로운 옵션이 겹치는 부분이 있다면 에러를 터뜨려줍니다.\n      assertNoConflictingOptions(\"api\", modelName, methodName, existingApi.options, options);\n      existingApi.options = {\n        ...existingApi.options, // 기존의 옵션을 존중하되\n        ...options, // @api 데코레이터의 옵션을 추가합니다.\n      };\n    } else {\n      registeredApis.push({\n        modelName,\n        methodName,\n        path,\n        options,\n      });\n    }\n\n    const originalMethod = descriptor.value;\n    descriptor.value = async function (this: BaseModelClass | BaseFrameClass, ...args: unknown[]) {\n      if (this instanceof BaseModelClass) {\n        getLogger(convertDomainToCategory(this.modelName, \"model\")).debug(\n          \"api: {httpMethod} {model}.{method}\",\n          {\n            httpMethod: options.httpMethod,\n            model: modelName,\n            method: methodName,\n          },\n        );\n      }\n\n      if (this instanceof BaseFrameClass) {\n        getLogger(convertDomainToCategory(this.frameName, \"frame\")).debug(\n          \"api: {httpMethod} {model}.{method}\",\n          {\n            httpMethod: options.httpMethod,\n            model: modelName,\n            method: methodName,\n          },\n        );\n      }\n\n      return originalMethod.apply(this, args);\n    };\n  };\n}\n\nexport function stream(options: StreamDecoratorOptions) {\n  return (target: DecoratorTarget, propertyKey: string, descriptor: PropertyDescriptor) => {\n    const modelName = target.constructor.name.match(/(.+)Class$/)?.[1];\n    assert(\n      modelName,\n      `modelName is required on @stream decorator on ${target.constructor.name}.${propertyKey}`,\n    );\n    const methodName = propertyKey;\n\n    // 메서드에 걸린 데코레이터 중복 체크\n    checkSingleDecorator(target, propertyKey, DECORATOR_TYPES.STREAM);\n\n    const defaultPath = `/${inflection.camelize(\n      modelName.replace(/Model$/, \"\").replace(/Frame$/, \"\"),\n      true,\n    )}/${inflection.camelize(propertyKey, true)}`;\n    const path = options.path ?? defaultPath;\n    // stream 전용 필드(events, type)는 ApiDecoratorOptions에 속하지 않으므로 제외\n    const { events: _, type: _type, ...apiOptions } = options;\n    const optionsWithDefaults = {\n      ...apiOptions,\n      httpMethod: \"GET\" as HTTPMethods,\n    };\n\n    const existingApi = registeredApis.find(\n      (api) => api.modelName === modelName && api.methodName === methodName,\n    );\n    if (existingApi) {\n      // 기존의 path와 새로운 path가 다르다면(=빈 스트링이 아니었는데 다른 스트링으로 바뀌게 된다면) 에러를 터뜨려줍니다.\n      assertNoConflictingPath(\"stream\", modelName, methodName, existingApi.path, path);\n      existingApi.path = path;\n\n      // 기존의 옵션과 새로운 옵션이 겹치는 부분이 있다면 에러를 터뜨려줍니다.\n      assertNoConflictingOptions(\n        \"stream\",\n        modelName,\n        methodName,\n        existingApi.options,\n        optionsWithDefaults,\n      );\n      existingApi.options = {\n        ...existingApi.options, // 기존의 옵션을 존중하되\n        ...optionsWithDefaults, // @stream 데코레이터의 옵션을 추가합니다.\n      };\n\n      existingApi.streamOptions = options;\n    } else {\n      registeredApis.push({\n        modelName,\n        methodName,\n        path,\n        options: optionsWithDefaults,\n        streamOptions: options,\n      });\n    }\n\n    const originalMethod = descriptor.value;\n    descriptor.value = async function (this: BaseModelClass | BaseFrameClass, ...args: unknown[]) {\n      if (this instanceof BaseModelClass) {\n        getLogger(convertDomainToCategory(this.modelName, \"model\")).debug(\n          \"stream: {model}.{method}\",\n          {\n            model: modelName,\n            method: methodName,\n          },\n        );\n      }\n\n      if (this instanceof BaseFrameClass) {\n        getLogger(convertDomainToCategory(this.frameName, \"frame\")).debug(\n          \"stream: {model}.{method}\",\n          {\n            model: modelName,\n            method: methodName,\n          },\n        );\n      }\n\n      return originalMethod.apply(this, args);\n    };\n  };\n}\n\nexport function websocket(options: WebSocketDecoratorOptions) {\n  return (target: DecoratorTarget, propertyKey: string, descriptor: PropertyDescriptor) => {\n    const modelName = target.constructor.name.match(/(.+)Class$/)?.[1];\n    assert(\n      modelName,\n      `modelName is required on @websocket decorator on ${target.constructor.name}.${propertyKey}`,\n    );\n    const methodName = propertyKey;\n\n    checkSingleDecorator(target, propertyKey, DECORATOR_TYPES.WEBSOCKET);\n\n    const defaultPath = `/${inflection.camelize(\n      modelName.replace(/Model$/, \"\").replace(/Frame$/, \"\"),\n      true,\n    )}/${inflection.camelize(propertyKey, true)}`;\n    const path = options.path ?? defaultPath;\n    const { outEvents: _outEvents, inEvents: _inEvents, ...apiOptions } = options;\n    const optionsWithDefaults = {\n      ...apiOptions,\n      httpMethod: \"GET\" as HTTPMethods,\n    };\n\n    const existingApi = registeredApis.find(\n      (api) => api.modelName === modelName && api.methodName === methodName,\n    );\n    if (existingApi) {\n      assertNoConflictingPath(\"websocket\", modelName, methodName, existingApi.path, path);\n      existingApi.path = path;\n\n      assertNoConflictingOptions(\n        \"websocket\",\n        modelName,\n        methodName,\n        existingApi.options,\n        optionsWithDefaults,\n      );\n      existingApi.options = {\n        ...existingApi.options,\n        ...optionsWithDefaults,\n      };\n\n      existingApi.websocketOptions = options;\n    } else {\n      registeredApis.push({\n        modelName,\n        methodName,\n        path,\n        options: optionsWithDefaults,\n        websocketOptions: options,\n      });\n    }\n\n    const originalMethod = descriptor.value;\n    descriptor.value = async function (this: BaseModelClass | BaseFrameClass, ...args: unknown[]) {\n      if (this instanceof BaseModelClass) {\n        getLogger(convertDomainToCategory(this.modelName, \"model\")).debug(\n          \"websocket: {model}.{method}\",\n          {\n            model: modelName,\n            method: methodName,\n          },\n        );\n      }\n\n      if (this instanceof BaseFrameClass) {\n        getLogger(convertDomainToCategory(this.frameName, \"frame\")).debug(\n          \"websocket: {model}.{method}\",\n          {\n            model: modelName,\n            method: methodName,\n          },\n        );\n      }\n\n      return originalMethod.apply(this, args);\n    };\n  };\n}\n\nexport function transactional(options: TransactionalOptions = {}) {\n  const { isolation, readOnly, dbPreset = \"w\" } = options;\n\n  return (target: DecoratorTarget, propertyKey: string, descriptor: PropertyDescriptor) => {\n    const originalMethod = descriptor.value;\n    const modelName = target.constructor.name.match(/(.+)Class$/)?.[1];\n    assert(\n      modelName,\n      `modelName is required on @transactional decorator on ${target.constructor.name}.${propertyKey}`,\n    );\n    const methodName = propertyKey;\n\n    descriptor.value = async function (this: BaseModelClass, ...args: unknown[]) {\n      this.logger.debug(\"transactional: {model}.{method}\", {\n        model: modelName,\n        method: methodName,\n      });\n\n      const existingContext = DB.transactionStorage.getStore();\n\n      // AsyncLocalStorage 컨텍스트 없거나 해당 preset의 트랜잭션이 없으면 새로 시작\n      const startTransaction = async () => {\n        const puri = this.getPuri(dbPreset);\n\n        return puri.knex.transaction(\n          async (trx) => {\n            this.logger.debug(\"new transaction context: {dbPreset}\", { dbPreset });\n            const trxWrapper = new PuriTransactionWrapper(trx, new UpsertBuilder());\n            // TransactionContext에 트랜잭션 저장\n            DB.getTransactionContext().setTransaction(dbPreset, trxWrapper);\n\n            try {\n              return await originalMethod.apply(this, args);\n            } finally {\n              // 트랜잭션 제거\n              this.logger.debug(\"delete transaction context: {dbPreset}\", { dbPreset });\n              DB.getTransactionContext().deleteTransaction(dbPreset);\n            }\n          },\n          { isolationLevel: isolation, readOnly },\n        );\n      };\n\n      // AsyncLocalStorage 컨텍스트가 없으면 새로 생성\n      if (!existingContext) {\n        return DB.runWithTransaction(startTransaction);\n      }\n\n      // 이미 AsyncLocalStorage 컨텍스트 안에 있는지 확인 후 해당 preset의 트랜잭션이 이미 있으면 재사용\n      if (existingContext?.getTransaction(dbPreset)) {\n        this.logger.debug(\"reuse transaction context: {dbPreset}\", { dbPreset });\n        return originalMethod.apply(this, args);\n      }\n\n      // 컨텍스트는 있지만 이 preset의 트랜잭션은 없는 경우 (같은 컨텍스트 내에서 실행)\n      return startTransaction();\n    };\n\n    return descriptor;\n  };\n}\n\n/**\n * 파일 업로드 API를 생성해줍니다. (@api 데코레이터 없이 독립적으로 사용)\n * @param options\n * @returns\n */\nexport function upload(options: UploadDecoratorOptions = { consume: \"buffer\" }) {\n  return (target: DecoratorTarget, propertyKey: string, descriptor: PropertyDescriptor) => {\n    const originalMethod = descriptor.value;\n    const modelName = target.constructor.name.match(/(.+)Class$/)?.[1];\n    assert(\n      modelName,\n      `modelName is required on @upload decorator on ${target.constructor.name}.${propertyKey}`,\n    );\n    const methodName = propertyKey;\n\n    // 메서드에 걸린 데코레이터 중복 체크\n    checkSingleDecorator(target, propertyKey, DECORATOR_TYPES.UPLOAD);\n\n    const defaultPath = `/${inflection.camelize(\n      modelName.replace(/Model$/, \"\").replace(/Frame$/, \"\"),\n      true,\n    )}/${inflection.camelize(methodName, true)}`;\n\n    // registeredApis에서 해당 API 찾아서 uploadOptions 추가\n    const existingApi = registeredApis.find(\n      (api) => api.modelName === modelName && api.methodName === methodName,\n    );\n\n    if (existingApi) {\n      // 재등록 시 업로드 옵션만 갱신\n      existingApi.uploadOptions = options;\n    } else {\n      // 새로 등록\n      registeredApis.push({\n        modelName,\n        methodName,\n        path: defaultPath,\n        options: {\n          httpMethod: \"POST\",\n          clients: [\"axios-multipart\", \"tanstack-mutation-multipart\"],\n          guards: options.guards,\n          description: options.description,\n        },\n        uploadOptions: options,\n      });\n    }\n\n    descriptor.value = async function (this: BaseModelClass | BaseFrameClass, ...args: unknown[]) {\n      if (this instanceof BaseModelClass) {\n        getLogger(convertDomainToCategory(this.modelName, \"model\")).debug(\n          \"upload: {model}.{method}\",\n          {\n            model: modelName,\n            method: methodName,\n          },\n        );\n      }\n\n      if (this instanceof BaseFrameClass) {\n        getLogger(convertDomainToCategory(this.frameName, \"frame\")).debug(\n          \"upload: {model}.{method}\",\n          {\n            model: modelName,\n            method: methodName,\n          },\n        );\n      }\n\n      return originalMethod.apply(this, args);\n    };\n\n    return descriptor;\n  };\n}\n\n/**\n * 기존의 path와 새로운 path가 다르다면(=값이 있던 스트링이 다른 값이 있는 스트링으로 바뀌게 된다면) 에러를 터뜨려줍니다.\n * @param decoratorName 데코레이터 이름\n * @param modelName 모델 이름\n * @param methodName 메서드 이름\n * @param existingPath 기존의 path\n * @param newPath 새로운 path\n */\nfunction assertNoConflictingPath(\n  decoratorName: string,\n  modelName: string,\n  methodName: string,\n  existingPath: string,\n  newPath: string,\n) {\n  if (existingPath !== \"\" && newPath !== \"\" && existingPath !== newPath) {\n    // 이것이 무슨 상황이냐면요, api.path가 덮어씌워지는 상황입니다.\n    // 가령 @api({ path: \"/api/v1/users\" }) 데코레이터가 붙어있는 메서드에\n    // @stream({ path: \"/api/v1/users/stream\" }) 같은 것이 붙어 있는 상황입니다.\n    // 이렇게 되면 두 데코레이터가 같은 api의 path 필드를 건드리게 되므로, 에러를 터뜨려줍니다.\n    throw new Error(\n      `@${decoratorName} decorator on ${modelName}.${methodName} has conflicting path: ${newPath}. The decorator is trying to override the existing path(${existingPath}) with the new path(${newPath}).`,\n    );\n  }\n}\n\n/**\n * 기존의 옵션과 새로운 옵션이 겹치는 부분이 있다면 에러를 터뜨려줍니다.\n * @param decoratorName 데코레이터 이름\n * @param modelName 모델 이름\n * @param methodName 메서드 이름\n * @param existingOptions 기존의 옵션\n * @param newOptions 새로운 옵션\n */\nfunction assertNoConflictingOptions(\n  decoratorName: string,\n  modelName: string,\n  methodName: string,\n  // oxlint-disable-next-line @typescript-eslint/no-explicit-any -- <아 쉽게쉽게 좀 갑시다>\n  existingOptions: Record<string, any>,\n  // oxlint-disable-next-line @typescript-eslint/no-explicit-any -- <이럴 때 아니면 any 언제 씁니까>\n  newOptions: Record<string, any>,\n) {\n  Object.keys(newOptions).forEach((key) => {\n    if (existingOptions[key] && !isEqual(existingOptions[key], newOptions[key])) {\n      // 이것이 무슨 상황이냐면요, api.options가 덮어씌워지는 상황입니다.\n      // 가령 @api({ resourceName: \"Users\" }) 데코레이터가 붙어있는 메서드에\n      // @stream({ resourceName: \"Posts\" }) 같은 것이 붙어 있는 상황입니다.\n      // 이렇게 되면 두 데코레이터가 같은 api의 options 속 같은 필드를 건드리게 되므로, 에러를 터뜨려줍니다.\n      throw new Error(\n        `@${decoratorName} decorator on ${modelName}.${methodName} has conflicting options: ${key}. The decorator is trying to override the existing option(${JSON.stringify(existingOptions[key])}) with the new option(${JSON.stringify(newOptions[key])}).`,\n      );\n    }\n  });\n}\n"],"mappings":"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"}
|
|
@@ -39,7 +39,7 @@ function getChecksumPatternGroup() {
|
|
|
39
39
|
workflow: api("src/application/**/*.workflow.ts"),
|
|
40
40
|
i18n: api("src/i18n/**/!(sd.generated).ts"),
|
|
41
41
|
generated: api("src/application/**/*.generated.{ts,tsx,sso.ts}"),
|
|
42
|
-
generatedCopied: targets("src/services/**/{sonamu,queries}.generated.{ts,tsx
|
|
42
|
+
generatedCopied: targets("src/services/**/{sonamu,queries}.generated.{ts,tsx}"),
|
|
43
43
|
httpGenerated: api("src/application/**/*.generated.http"),
|
|
44
44
|
servicesGenerated: targets("src/services/services.generated.ts"),
|
|
45
45
|
sdGenerated: anywhere("src/i18n/**/sd.generated.ts"),
|
|
@@ -87,4 +87,4 @@ var init_file_patterns = __esmMin((() => {
|
|
|
87
87
|
//#endregion
|
|
88
88
|
init_file_patterns();
|
|
89
89
|
export { GLOB_EXCLUDE, getChecksumPatternGroup, getChecksumPatternGroupInAbsolutePath, init_file_patterns };
|
|
90
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"file-patterns.js","names":[],"sources":["../../src/syncer/file-patterns.ts"],"sourcesContent":["import path from \"path\";\n\nimport { Sonamu } from \"../api/sonamu\";\nimport { type AbsolutePath, type AppRelativePath } from \"../utils/path-utils\";\n\n/**\n * Syncer가 관심 가지고 지켜보는 파일들입니다.\n * 이 파일들에 변경이 생기면 추가적인 작업(이하 \"싱크\" 또는 \"싱크 액션\")을 수행합니다.\n * 이 작업이라 함은 파일 복사 또는 템플릿 렌더링을 통한 code generation을 의미합니다.\n *\n * 경로 형식: appRoot 기준 상대 경로 (target 디렉토리로 시작, 예: \"api/src/...\", \"web/src/...\")\n * 사용: getChecksumPatternGroupInAbsolutePath()로 절대 경로 변환 후 glob 사용\n *\n * 두 가지 의미적 영역:\n * - 입력 (사용자 작성): api 디렉토리에만 위치. 사용자가 직접 편집.\n * - 출력 (sonamu 생성/복사): api 또는 target 디렉토리에 sonamu가 만들어내는 파일.\n *\n * 추적 밖 자산 (부트스트랩 phase에서 매번 보장):\n * - sonamu.shared.ts: 사용자가 커스터마이즈하는 자산. IfNotExists로 1회 생성 후 손대지 않음.\n * - entry-server.generated.tsx: 입력 의존 없는 정적 코드. 매번 overwrite generate.\n *\n * 위 둘은 sync()의 부트스트랩 phase에서 변경 검출 사이클과 무관하게 보장됩니다.\n * 추적 사이클 안에서 할 액션이 없는 자산이라 패턴 그룹에 포함되지 않습니다.\n *\n * 위치 카테고리는 api/targets/anywhere 헬퍼로 명시적으로 표현합니다.\n *\n * FileType은 이 함수의 반환 타입에서 자동 추론됩니다. 키 추가 시 별도 enum/배열을\n * 동기화할 필요 없이 여기 한 군데만 수정하면 됩니다.\n */\nexport function getChecksumPatternGroup() {\n  // 헬퍼 함수들 만들어서 가져옵니다.\n  const { api, targets, anywhere } = globBuilders();\n\n  return {\n    // Sonamu 입장에서 soruce가 되는 파일들입니다.\n    // 이 친구들이 변경되면 이들로부터 액션을 수행하고 파일을 생성합니다.\n    // 모노리포에서 이 source들은 api 프로젝트에 한정되므로, api에 있는 친구들만 리스팅합니다.\n    config: api(\"src/sonamu.config.ts\"),\n    entity: api(\"src/application/**/*.entity.json\"),\n    frame: api(\"src/application/**/*.frame.ts\"),\n    functions: api(\"src/application/**/*.functions.ts\"),\n    model: api(\"src/application/**/*.model.ts\"),\n    types: api(\"src/application/**/*.types.ts\"),\n    workflow: api(\"src/application/**/*.workflow.ts\"),\n    i18n: api(\"src/i18n/**/!(sd.generated).ts\"),\n\n    // Sonamu가 출력하는 생성 파일들입니다.\n    // 이 친구들도 정합성 검증 차원에서 sonamu.lock에 기록해야 하고,\n    // 또한 변경시 그 사실을 syncer가 알기는 해야 합니다(비록 별다른 처리가 없는 경우도 있지만).\n    //\n    // 자산 본성에 따라 위치 카테고리가 다르기 때문에, 본성별로 분리해서 표기합니다.\n    // - 양쪽-필요 자산: api에 정본이 만들어진 뒤 target에 복사됨 (sonamu.generated.*, queries.generated.ts).\n    // - api 전용 자산: api에만 만들어짐 (sonamu.generated.http).\n    // - target 전용 자산: target에만 만들어짐 (services.generated.ts는 services.template의 :target 분배).\n    //\n    // 여기에는 Sonamu의 모든 sync 산출물이 있는 것은 아닙니다.\n    // sonamu.shared.ts와 entry-server.generated.tsx와 같은\n    // sync 초반 1회성 부트스트랩 파일들은 관리 안 하기 때문에 여기에 리스팅도 안 합니다.\n    generated: api(\"src/application/**/*.generated.{ts,tsx,sso.ts}\"),\n    generatedCopied: targets(\"src/services/**/{sonamu,queries}.generated.{ts,tsx,sso.ts}\"),\n    httpGenerated: api(\"src/application/**/*.generated.http\"),\n    servicesGenerated: targets(\"src/services/services.generated.ts\"),\n    sdGenerated: anywhere(\"src/i18n/**/sd.generated.ts\"),\n    typesCopied: targets(\"src/services/**/*.types.ts\"),\n    functionsCopied: targets(\"src/services/**/*.functions.ts\"),\n    i18nCopied: targets(\"src/i18n/**/!(sd.generated).ts\"),\n  } satisfies Record<string, AppRelativePath>;\n}\n\n/**\n * 위치 카테고리별 글롭 빌더를 만들어 반환합니다.\n * - api(rest): api 디렉토리에 한정\n * - targets(rest): target 디렉토리들에 한정 (web, app 등)\n * - anywhere(rest): api와 target 모두\n */\nfunction globBuilders() {\n  const apiDir = Sonamu.config.api.dir;\n  const targetDirs = Sonamu.config.sync.targets;\n\n  // Node 내장 fs.glob의 brace expansion은 단일 멤버 {x}를 풀지 않으므로, 멤버가 1개일 때는 alternation 없이 직접 결합합니다.\n  const braceJoin = (dirs: readonly string[]) =>\n    dirs.length === 1 ? dirs[0] : `{${dirs.join(\",\")}}`;\n\n  return {\n    api: (pathFromApi: string) => `${apiDir}/${pathFromApi}` as AppRelativePath,\n    targets: (pathFromTarget: string) =>\n      `${braceJoin(targetDirs)}/${pathFromTarget}` as AppRelativePath,\n    anywhere: (pathFromAnywhere: string) =>\n      `${braceJoin([apiDir, ...targetDirs])}/${pathFromAnywhere}` as AppRelativePath,\n  };\n}\n\n/**\n * FileType은 getChecksumPatternGroup의 반환 객체 키에서 자동 추론됩니다.\n * 별도 배열/enum 동기화 불필요 — 패턴 그룹 함수가 진실의 단일 원천.\n */\nexport type FileType = keyof ReturnType<typeof getChecksumPatternGroup>;\nexport type GlobPattern<T extends AppRelativePath | AbsolutePath> = Record<FileType, T>;\n\n/**\n * 빌드 산출물 디렉토리는 alternation 글롭이 의도치 않게 휘말릴 수 있으므로 안전망으로 제외.\n * Node 내장 fs.glob의 exclude 옵션과 함께 사용합니다.\n */\nexport const GLOB_EXCLUDE = [\"**/node_modules/**\", \"**/dist/**\", \"**/build/**\", \"**/.turbo/**\"];\n\n/**\n * appRoot 기준 상대 경로 패턴을 절대 경로 패턴으로 변환합니다.\n *\n * @returns 절대 경로 기반 Glob 패턴 맵\n */\nexport function getChecksumPatternGroupInAbsolutePath(): GlobPattern<AbsolutePath> {\n  const group = getChecksumPatternGroup();\n  return Object.fromEntries(\n    Object.entries(group).map(([key, value]) => [\n      key,\n      path.join(Sonamu.appRootPath, value), // appRoot 상대 경로 → 절대 경로\n    ]),\n  ) as GlobPattern<AbsolutePath>;\n}\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;AA6BA,SAAgB,0BAA0B;CAExC,MAAM,EAAE,KAAK,SAAS,aAAa,cAAc;AAEjD,QAAO;EAIL,QAAQ,IAAI,uBAAuB;EACnC,QAAQ,IAAI,mCAAmC;EAC/C,OAAO,IAAI,gCAAgC;EAC3C,WAAW,IAAI,oCAAoC;EACnD,OAAO,IAAI,gCAAgC;EAC3C,OAAO,IAAI,gCAAgC;EAC3C,UAAU,IAAI,mCAAmC;EACjD,MAAM,IAAI,iCAAiC;EAc3C,WAAW,IAAI,iDAAiD;EAChE,iBAAiB,QAAQ,6DAA6D;EACtF,eAAe,IAAI,sCAAsC;EACzD,mBAAmB,QAAQ,qCAAqC;EAChE,aAAa,SAAS,8BAA8B;EACpD,aAAa,QAAQ,6BAA6B;EAClD,iBAAiB,QAAQ,iCAAiC;EAC1D,YAAY,QAAQ,iCAAiC;EACtD;;;;;;;;AASH,SAAS,eAAe;CACtB,MAAM,SAAS,OAAO,OAAO,IAAI;CACjC,MAAM,aAAa,OAAO,OAAO,KAAK;CAGtC,MAAM,aAAa,SACjB,KAAK,WAAW,IAAI,KAAK,KAAK,IAAI,KAAK,KAAK,IAAI,CAAC;AAEnD,QAAO;EACL,MAAM,gBAAwB,GAAG,OAAO,GAAG;EAC3C,UAAU,mBACR,GAAG,UAAU,WAAW,CAAC,GAAG;EAC9B,WAAW,qBACT,GAAG,UAAU,CAAC,QAAQ,GAAG,WAAW,CAAC,CAAC,GAAG;EAC5C;;;;;;;AAqBH,SAAgB,wCAAmE;CACjF,MAAM,QAAQ,yBAAyB;AACvC,QAAO,OAAO,YACZ,OAAO,QAAQ,MAAM,CAAC,KAAK,CAAC,KAAK,WAAW,CAC1C,KACA,KAAK,KAAK,OAAO,aAAa,MAAM,CACrC,CAAC,CACH;;;;cAnHoC;CAqG1B,eAAe;EAAC;EAAsB;EAAc;EAAe;EAAe"}
|
|
90
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"file-patterns.js","names":[],"sources":["../../src/syncer/file-patterns.ts"],"sourcesContent":["import path from \"path\";\n\nimport { Sonamu } from \"../api/sonamu\";\nimport { type AbsolutePath, type AppRelativePath } from \"../utils/path-utils\";\n\n/**\n * Syncer가 관심 가지고 지켜보는 파일들입니다.\n * 이 파일들에 변경이 생기면 추가적인 작업(이하 \"싱크\" 또는 \"싱크 액션\")을 수행합니다.\n * 이 작업이라 함은 파일 복사 또는 템플릿 렌더링을 통한 code generation을 의미합니다.\n *\n * 경로 형식: appRoot 기준 상대 경로 (target 디렉토리로 시작, 예: \"api/src/...\", \"web/src/...\")\n * 사용: getChecksumPatternGroupInAbsolutePath()로 절대 경로 변환 후 glob 사용\n *\n * 두 가지 의미적 영역:\n * - 입력 (사용자 작성): api 디렉토리에만 위치. 사용자가 직접 편집.\n * - 출력 (sonamu 생성/복사): api 또는 target 디렉토리에 sonamu가 만들어내는 파일.\n *\n * 추적 밖 자산 (부트스트랩 phase에서 매번 보장):\n * - sonamu.shared.ts: 사용자가 커스터마이즈하는 자산. IfNotExists로 1회 생성 후 손대지 않음.\n * - entry-server.generated.tsx: 입력 의존 없는 정적 코드. 매번 overwrite generate.\n *\n * 위 둘은 sync()의 부트스트랩 phase에서 변경 검출 사이클과 무관하게 보장됩니다.\n * 추적 사이클 안에서 할 액션이 없는 자산이라 패턴 그룹에 포함되지 않습니다.\n *\n * 위치 카테고리는 api/targets/anywhere 헬퍼로 명시적으로 표현합니다.\n *\n * FileType은 이 함수의 반환 타입에서 자동 추론됩니다. 키 추가 시 별도 enum/배열을\n * 동기화할 필요 없이 여기 한 군데만 수정하면 됩니다.\n */\nexport function getChecksumPatternGroup() {\n  // 헬퍼 함수들 만들어서 가져옵니다.\n  const { api, targets, anywhere } = globBuilders();\n\n  return {\n    // Sonamu 입장에서 soruce가 되는 파일들입니다.\n    // 이 친구들이 변경되면 이들로부터 액션을 수행하고 파일을 생성합니다.\n    // 모노리포에서 이 source들은 api 프로젝트에 한정되므로, api에 있는 친구들만 리스팅합니다.\n    config: api(\"src/sonamu.config.ts\"),\n    entity: api(\"src/application/**/*.entity.json\"),\n    frame: api(\"src/application/**/*.frame.ts\"),\n    functions: api(\"src/application/**/*.functions.ts\"),\n    model: api(\"src/application/**/*.model.ts\"),\n    types: api(\"src/application/**/*.types.ts\"),\n    workflow: api(\"src/application/**/*.workflow.ts\"),\n    i18n: api(\"src/i18n/**/!(sd.generated).ts\"),\n\n    // Sonamu가 출력하는 생성 파일들입니다.\n    // 이 친구들도 정합성 검증 차원에서 sonamu.lock에 기록해야 하고,\n    // 또한 변경시 그 사실을 syncer가 알기는 해야 합니다(비록 별다른 처리가 없는 경우도 있지만).\n    //\n    // 자산 본성에 따라 위치 카테고리가 다르기 때문에, 본성별로 분리해서 표기합니다.\n    // - 양쪽-필요 자산: api에 정본이 만들어진 뒤 target에 복사됨 (sonamu.generated.*, queries.generated.ts).\n    // - api 전용 자산: api에만 만들어짐 (sonamu.generated.http).\n    // - target 전용 자산: target에만 만들어짐 (services.generated.ts는 services.template의 :target 분배).\n    //\n    // 여기에는 Sonamu의 모든 sync 산출물이 있는 것은 아닙니다.\n    // sonamu.shared.ts와 entry-server.generated.tsx와 같은\n    // sync 초반 1회성 부트스트랩 파일들은 관리 안 하기 때문에 여기에 리스팅도 안 합니다.\n    generated: api(\"src/application/**/*.generated.{ts,tsx,sso.ts}\"),\n    generatedCopied: targets(\"src/services/**/{sonamu,queries}.generated.{ts,tsx}\"),\n    httpGenerated: api(\"src/application/**/*.generated.http\"),\n    servicesGenerated: targets(\"src/services/services.generated.ts\"),\n    sdGenerated: anywhere(\"src/i18n/**/sd.generated.ts\"),\n    typesCopied: targets(\"src/services/**/*.types.ts\"),\n    functionsCopied: targets(\"src/services/**/*.functions.ts\"),\n    i18nCopied: targets(\"src/i18n/**/!(sd.generated).ts\"),\n  } satisfies Record<string, AppRelativePath>;\n}\n\n/**\n * 위치 카테고리별 글롭 빌더를 만들어 반환합니다.\n * - api(rest): api 디렉토리에 한정\n * - targets(rest): target 디렉토리들에 한정 (web, app 등)\n * - anywhere(rest): api와 target 모두\n */\nfunction globBuilders() {\n  const apiDir = Sonamu.config.api.dir;\n  const targetDirs = Sonamu.config.sync.targets;\n\n  // Node 내장 fs.glob의 brace expansion은 단일 멤버 {x}를 풀지 않으므로, 멤버가 1개일 때는 alternation 없이 직접 결합합니다.\n  const braceJoin = (dirs: readonly string[]) =>\n    dirs.length === 1 ? dirs[0] : `{${dirs.join(\",\")}}`;\n\n  return {\n    api: (pathFromApi: string) => `${apiDir}/${pathFromApi}` as AppRelativePath,\n    targets: (pathFromTarget: string) =>\n      `${braceJoin(targetDirs)}/${pathFromTarget}` as AppRelativePath,\n    anywhere: (pathFromAnywhere: string) =>\n      `${braceJoin([apiDir, ...targetDirs])}/${pathFromAnywhere}` as AppRelativePath,\n  };\n}\n\n/**\n * FileType은 getChecksumPatternGroup의 반환 객체 키에서 자동 추론됩니다.\n * 별도 배열/enum 동기화 불필요 — 패턴 그룹 함수가 진실의 단일 원천.\n */\nexport type FileType = keyof ReturnType<typeof getChecksumPatternGroup>;\nexport type GlobPattern<T extends AppRelativePath | AbsolutePath> = Record<FileType, T>;\n\n/**\n * 빌드 산출물 디렉토리는 alternation 글롭이 의도치 않게 휘말릴 수 있으므로 안전망으로 제외.\n * Node 내장 fs.glob의 exclude 옵션과 함께 사용합니다.\n */\nexport const GLOB_EXCLUDE = [\"**/node_modules/**\", \"**/dist/**\", \"**/build/**\", \"**/.turbo/**\"];\n\n/**\n * appRoot 기준 상대 경로 패턴을 절대 경로 패턴으로 변환합니다.\n *\n * @returns 절대 경로 기반 Glob 패턴 맵\n */\nexport function getChecksumPatternGroupInAbsolutePath(): GlobPattern<AbsolutePath> {\n  const group = getChecksumPatternGroup();\n  return Object.fromEntries(\n    Object.entries(group).map(([key, value]) => [\n      key,\n      path.join(Sonamu.appRootPath, value), // appRoot 상대 경로 → 절대 경로\n    ]),\n  ) as GlobPattern<AbsolutePath>;\n}\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;AA6BA,SAAgB,0BAA0B;CAExC,MAAM,EAAE,KAAK,SAAS,aAAa,cAAc;AAEjD,QAAO;EAIL,QAAQ,IAAI,uBAAuB;EACnC,QAAQ,IAAI,mCAAmC;EAC/C,OAAO,IAAI,gCAAgC;EAC3C,WAAW,IAAI,oCAAoC;EACnD,OAAO,IAAI,gCAAgC;EAC3C,OAAO,IAAI,gCAAgC;EAC3C,UAAU,IAAI,mCAAmC;EACjD,MAAM,IAAI,iCAAiC;EAc3C,WAAW,IAAI,iDAAiD;EAChE,iBAAiB,QAAQ,sDAAsD;EAC/E,eAAe,IAAI,sCAAsC;EACzD,mBAAmB,QAAQ,qCAAqC;EAChE,aAAa,SAAS,8BAA8B;EACpD,aAAa,QAAQ,6BAA6B;EAClD,iBAAiB,QAAQ,iCAAiC;EAC1D,YAAY,QAAQ,iCAAiC;EACtD;;;;;;;;AASH,SAAS,eAAe;CACtB,MAAM,SAAS,OAAO,OAAO,IAAI;CACjC,MAAM,aAAa,OAAO,OAAO,KAAK;CAGtC,MAAM,aAAa,SACjB,KAAK,WAAW,IAAI,KAAK,KAAK,IAAI,KAAK,KAAK,IAAI,CAAC;AAEnD,QAAO;EACL,MAAM,gBAAwB,GAAG,OAAO,GAAG;EAC3C,UAAU,mBACR,GAAG,UAAU,WAAW,CAAC,GAAG;EAC9B,WAAW,qBACT,GAAG,UAAU,CAAC,QAAQ,GAAG,WAAW,CAAC,CAAC,GAAG;EAC5C;;;;;;;AAqBH,SAAgB,wCAAmE;CACjF,MAAM,QAAQ,yBAAyB;AACvC,QAAO,OAAO,YACZ,OAAO,QAAQ,MAAM,CAAC,KAAK,CAAC,KAAK,WAAW,CAC1C,KACA,KAAK,KAAK,OAAO,aAAa,MAAM,CACrC,CAAC,CACH;;;;cAnHoC;CAqG1B,eAAe;EAAC;EAAsB;EAAc;EAAe;EAAe"}
|
|
@@ -38,14 +38,20 @@ export declare function actionGenerateSsrQueries(): Promise<AbsolutePath[]>;
|
|
|
38
38
|
*/
|
|
39
39
|
export declare function actionGenerateSsrEntryServerIfNotExists(): Promise<AbsolutePath[]>;
|
|
40
40
|
/**
|
|
41
|
-
* 주어진 .ts 파일들(api에 있다고 가정)을
|
|
41
|
+
* 주어진 .ts 파일들(api에 있다고 가정)을 타겟 디렉토리의 services에 갖다 둡니다.
|
|
42
42
|
* 이때 내부의 sonamu import는 sonamu.shared.ts import로 치환되고,
|
|
43
43
|
* 경로의 /application/은 /services/로 치환됩니다.
|
|
44
44
|
*
|
|
45
|
+
* 기본은 Sonamu.config.sync.targets 전체를 대상으로 하며,
|
|
46
|
+
* onlyTargetsStartingWith 화이트리스트로 일부 target에만 분배할 수 있습니다.
|
|
47
|
+
* 가량 SSR-only인 queries.generated.ts같은 친구들은 "web"으로 시작하는 타겟들에만 보낼 수 있어요.
|
|
48
|
+
* ["web"]으로 넘기면 web, web-admin, webapp 등에 모두 매치됩니다.
|
|
49
|
+
*
|
|
45
50
|
* @param tsPaths 복사할 파일들의 절대 경로
|
|
51
|
+
* @param onlyTargetsStartingWith 분배할 target 접두어들의 화이트리스트. 미지정 시 모든 target.
|
|
46
52
|
* @returns 각 타겟에 복사된 파일들의 절대 경로 배열 (flat).
|
|
47
53
|
*/
|
|
48
|
-
export declare function actionSyncFilesToTargets(tsPaths: AbsolutePath[]): Promise<string[]>;
|
|
54
|
+
export declare function actionSyncFilesToTargets(tsPaths: AbsolutePath[], onlyTargetsStartingWith?: string[]): Promise<string[]>;
|
|
49
55
|
/**
|
|
50
56
|
* shared 템플릿으로부터 sonamu.shared.ts 파일을 만들어서 모든 타겟 디렉토리에 갖다 둡니다.
|
|
51
57
|
* 파일을 만드는 과정에서 여러 치환 가공이 일어납니다.
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"syncer-actions.d.ts","sourceRoot":"","sources":["../../src/syncer/syncer-actions.ts"],"names":[],"mappings":"AAOA,OAAO,EAAE,KAAK,iBAAiB,EAAE,MAAM,0BAA0B,CAAC;AAMlE,OAAO,EAAE,KAAK,YAAY,EAAE,MAAM,qBAAqB,CAAC;AAKxD,wBAAsB,gBAAgB,kBAWrC;AAED;;;;GAIG;AACH,wBAAsB,sBAAsB,CAC1C,WAAW,EAAE;IACX,WAAW,EAAE,iBAAiB,CAAC;CAChC,EAAE,GACF,OAAO,CAAC,YAAY,EAAE,CAAC,CAazB;AAED;;;GAGG;AACH,wBAAsB,0BAA0B,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,YAAY,EAAE,CAAC,CAE1F;AAED;;;GAGG;AACH,wBAAsB,qBAAqB,IAAI,OAAO,CAAC,YAAY,EAAE,CAAC,CAOrE;AAED;;;GAGG;AACH,wBAAsB,mBAAmB,IAAI,OAAO,CAAC,YAAY,CAAC,CAQjE;AAED;;;GAGG;AACH,wBAAsB,wBAAwB,IAAI,OAAO,CAAC,YAAY,EAAE,CAAC,CAExE;AAED;;;;;;GAMG;AACH,wBAAsB,uCAAuC,IAAI,OAAO,CAAC,YAAY,EAAE,CAAC,CAWvF;AAED
|
|
1
|
+
{"version":3,"file":"syncer-actions.d.ts","sourceRoot":"","sources":["../../src/syncer/syncer-actions.ts"],"names":[],"mappings":"AAOA,OAAO,EAAE,KAAK,iBAAiB,EAAE,MAAM,0BAA0B,CAAC;AAMlE,OAAO,EAAE,KAAK,YAAY,EAAE,MAAM,qBAAqB,CAAC;AAKxD,wBAAsB,gBAAgB,kBAWrC;AAED;;;;GAIG;AACH,wBAAsB,sBAAsB,CAC1C,WAAW,EAAE;IACX,WAAW,EAAE,iBAAiB,CAAC;CAChC,EAAE,GACF,OAAO,CAAC,YAAY,EAAE,CAAC,CAazB;AAED;;;GAGG;AACH,wBAAsB,0BAA0B,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,YAAY,EAAE,CAAC,CAE1F;AAED;;;GAGG;AACH,wBAAsB,qBAAqB,IAAI,OAAO,CAAC,YAAY,EAAE,CAAC,CAOrE;AAED;;;GAGG;AACH,wBAAsB,mBAAmB,IAAI,OAAO,CAAC,YAAY,CAAC,CAQjE;AAED;;;GAGG;AACH,wBAAsB,wBAAwB,IAAI,OAAO,CAAC,YAAY,EAAE,CAAC,CAExE;AAED;;;;;;GAMG;AACH,wBAAsB,uCAAuC,IAAI,OAAO,CAAC,YAAY,EAAE,CAAC,CAWvF;AAED;;;;;;;;;;;;;GAaG;AACH,wBAAsB,wBAAwB,CAC5C,OAAO,EAAE,YAAY,EAAE,EACvB,uBAAuB,CAAC,EAAE,MAAM,EAAE,GACjC,OAAO,CAAC,MAAM,EAAE,CAAC,CAqCnB;AAED;;;;;GAKG;AACH,wBAAsB,oCAAoC,IAAI,OAAO,CAAC,IAAI,CAAC,CA+D1E;AAED;;;;;;;;GAQG;AACH,wBAAsB,0BAA0B,IAAI,OAAO,CAAC,IAAI,CAAC,CAoBhE"}
|
|
@@ -79,15 +79,22 @@ async function actionGenerateSsrEntryServerIfNotExists() {
|
|
|
79
79
|
}
|
|
80
80
|
}
|
|
81
81
|
/**
|
|
82
|
-
* 주어진 .ts 파일들(api에 있다고 가정)을
|
|
82
|
+
* 주어진 .ts 파일들(api에 있다고 가정)을 타겟 디렉토리의 services에 갖다 둡니다.
|
|
83
83
|
* 이때 내부의 sonamu import는 sonamu.shared.ts import로 치환되고,
|
|
84
84
|
* 경로의 /application/은 /services/로 치환됩니다.
|
|
85
85
|
*
|
|
86
|
+
* 기본은 Sonamu.config.sync.targets 전체를 대상으로 하며,
|
|
87
|
+
* onlyTargetsStartingWith 화이트리스트로 일부 target에만 분배할 수 있습니다.
|
|
88
|
+
* 가량 SSR-only인 queries.generated.ts같은 친구들은 "web"으로 시작하는 타겟들에만 보낼 수 있어요.
|
|
89
|
+
* ["web"]으로 넘기면 web, web-admin, webapp 등에 모두 매치됩니다.
|
|
90
|
+
*
|
|
86
91
|
* @param tsPaths 복사할 파일들의 절대 경로
|
|
92
|
+
* @param onlyTargetsStartingWith 분배할 target 접두어들의 화이트리스트. 미지정 시 모든 target.
|
|
87
93
|
* @returns 각 타겟에 복사된 파일들의 절대 경로 배열 (flat).
|
|
88
94
|
*/
|
|
89
|
-
async function actionSyncFilesToTargets(tsPaths) {
|
|
90
|
-
const
|
|
95
|
+
async function actionSyncFilesToTargets(tsPaths, onlyTargetsStartingWith) {
|
|
96
|
+
const allTargets = Sonamu.config.sync.targets;
|
|
97
|
+
const targets = onlyTargetsStartingWith ? allTargets.filter((t) => onlyTargetsStartingWith.some((prefix) => t.startsWith(prefix))) : allTargets;
|
|
91
98
|
const { dir: apiDir } = Sonamu.config.api;
|
|
92
99
|
return (await Promise.all(targets.map(async (target) => Promise.all(tsPaths.map(async (realSrc) => {
|
|
93
100
|
const dst = realSrc.replace(`/${apiDir}/`, `/${target}/`).replace("/application/", "/services/");
|
|
@@ -202,4 +209,4 @@ var init_syncer_actions = __esmMin((() => {
|
|
|
202
209
|
//#endregion
|
|
203
210
|
init_syncer_actions();
|
|
204
211
|
export { actionCopySharedToTargetsIfNotExists, actionGenerateHttps, actionGenerateInitialTypes, actionGenerateSchemas, actionGenerateServices, actionGenerateSsrEntryServerIfNotExists, actionGenerateSsrQueries, actionSyncConfig, actionSyncFilesToTargets, actionSyncSonamuDictionary, init_syncer_actions };
|
|
205
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"syncer-actions.js","names":[],"sources":["../../src/syncer/syncer-actions.ts"],"sourcesContent":["import assert from \"assert\";\nimport { mkdir, readFile, writeFile } from \"fs/promises\";\nimport path, { dirname } from \"path\";\n\nimport chalk from \"chalk\";\n\nimport { Sonamu } from \"../api/sonamu\";\nimport { type EntityNamesRecord } from \"../entity/entity-manager\";\nimport { AlreadyProcessedException } from \"../exceptions/so-exceptions\";\nimport { Naite } from \"../naite/naite\";\nimport { isTest } from \"../utils/controller\";\nimport { formatCode } from \"../utils/formatter\";\nimport { copyFileWithReplaceCoreToShared, exists } from \"../utils/fs-utils\";\nimport { type AbsolutePath } from \"../utils/path-utils\";\nimport { generateTemplate } from \"./code-generator\";\nimport { trackWritten } from \"./file-tracking\";\n\n// web/.sonamu.env 에 현재 설정값 저장\nexport async function actionSyncConfig() {\n  const { host, port } = Sonamu.config.server.listen ?? {};\n  const content = `API_HOST=${host ?? \"localhost\"}\\nAPI_PORT=${port ?? 3000}`;\n\n  Naite.t(\"actionSyncConfig\", { content });\n  await Promise.all([\n    ...Sonamu.config.sync.targets.map(async (target) => {\n      await writeFile(path.join(Sonamu.appRootPath, target, \".sonamu.env\"), content);\n    }),\n    generateTemplate(\"generated_sso\", {}, { overwrite: true }),\n  ]);\n}\n\n/**\n * services.generated.ts를 생성합니다.\n * @param paramsArray\n * @returns 생성된 파일 경로 배열.\n */\nexport async function actionGenerateServices(\n  paramsArray: {\n    namesRecord: EntityNamesRecord;\n  }[],\n): Promise<AbsolutePath[]> {\n  Naite.t(\"actionGenerateServices\", paramsArray);\n\n  // services.generated.ts 통합 파일 생성\n  const servicesFile = await generateTemplate(\n    \"services\",\n    {},\n    {\n      overwrite: true,\n    },\n  );\n\n  return [...servicesFile];\n}\n\n/**\n * Entity에 딸린 초기 types.ts를 만들어줍니다.\n * @param entityId\n */\nexport async function actionGenerateInitialTypes(entityId: string): Promise<AbsolutePath[]> {\n  return generateTemplate(\"init_types\", { entityId });\n}\n\n/**\n * sonamu.generated.ts와 sonamu.generated.sso.ts를 생성합니다.\n * @returns 생성된 파일 경로 배열.\n */\nexport async function actionGenerateSchemas(): Promise<AbsolutePath[]> {\n  return (\n    await Promise.all([\n      generateTemplate(\"generated_sso\", {}, { overwrite: true }),\n      generateTemplate(\"generated\", {}, { overwrite: true }),\n    ])\n  ).flat();\n}\n\n/**\n * sonamu.generated.http를 생성합니다.\n * @returns 생성된 파일 경로.\n */\nexport async function actionGenerateHttps(): Promise<AbsolutePath> {\n  const [res] = await generateTemplate(\n    \"generated_http\",\n    { entityId: \"dummy\" },\n    { overwrite: true },\n  );\n  assert(res);\n  return res;\n}\n\n/**\n * queries.generated.ts 재생성합니다.\n * @returns 생성된 파일 경로 배열.\n */\nexport async function actionGenerateSsrQueries(): Promise<AbsolutePath[]> {\n  return generateTemplate(\"queries\", {}, { overwrite: true });\n}\n\n/**\n * entry-server.generated.tsx를 생성합니다.\n * 다른 액션들과 달리, 이미 파일이 있으면 그냥 놔둡니다. 그래서 함수 이름 끝에 써놨어요 ㅎ\n * 입력 의존 없는 정적 코드라 매번 overwrite는 mtime만 갱신하는 의미 없는 동작.\n * 템플릿 자체가 변경된 경우(Sonamu 업그레이드)에는 사용자가 파일을 삭제한 뒤 sync로 재생성.\n * @returns 생성된 파일 경로 배열 (이미 있으면 빈 배열).\n */\nexport async function actionGenerateSsrEntryServerIfNotExists(): Promise<AbsolutePath[]> {\n  try {\n    return await generateTemplate(\"entry_server\", {}, { overwrite: false });\n  } catch (e) {\n    // generateTemplate은 overwrite: false에서 파일이 이미 있으면 예외를 던집니다.\n    // IfNotExists 의미상 \"그냥 놔둔다\"가 정상이므로 빈 배열로 변환합니다.\n    if (e instanceof AlreadyProcessedException) {\n      return [];\n    }\n    throw e;\n  }\n}\n\n/**\n * 주어진 .ts 파일들(api에 있다고 가정)을 모든 타겟 디렉토리의 services에 갖다 둡니다.\n * 이때 내부의 sonamu import는 sonamu.shared.ts import로 치환되고,\n * 경로의 /application/은 /services/로 치환됩니다.\n *\n * @param tsPaths 복사할 파일들의 절대 경로\n * @returns 각 타겟에 복사된 파일들의 절대 경로 배열 (flat).\n */\nexport async function actionSyncFilesToTargets(tsPaths: AbsolutePath[]): Promise<string[]> {\n  const { targets } = Sonamu.config.sync;\n  const { dir: apiDir } = Sonamu.config.api;\n\n  return (\n    await Promise.all(\n      targets.map(async (target) =>\n        Promise.all(\n          tsPaths.map(async (realSrc) => {\n            const dst = realSrc\n              .replace(`/${apiDir}/`, `/${target}/`)\n              .replace(\"/application/\", \"/services/\");\n            const dir = dirname(dst);\n            if (!(await exists(dir))) {\n              await mkdir(dir, { recursive: true });\n            }\n            const syncHeader = [\n              \"/**\",\n              \" * @generated\",\n              \" * API에서 동기화된 파일입니다. 직접 수정하지 마세요.\",\n              \" */\",\n            ].join(\"\\n\");\n            await copyFileWithReplaceCoreToShared(realSrc, dst, syncHeader);\n            await trackWritten(dst as AbsolutePath);\n            !isTest() &&\n              console.log(\n                chalk.bold(\"Copied: \") + chalk.blue(dst.replace(`${Sonamu.appRootPath}/`, \"\")),\n              );\n            return dst;\n          }),\n        ),\n      ),\n    )\n  ).flat();\n}\n\n/**\n * shared 템플릿으로부터 sonamu.shared.ts 파일을 만들어서 모든 타겟 디렉토리에 갖다 둡니다.\n * 파일을 만드는 과정에서 여러 치환 가공이 일어납니다.\n *\n * 다른 액션들과 달리, 이미 파일이 있으면 그냥 놔둡니다. 그래서 함수 이름 끝에 써놨어요 ㅎ\n */\nexport async function actionCopySharedToTargetsIfNotExists(): Promise<void> {\n  const { targets } = Sonamu.config.sync;\n\n  // plural.ts 내용을 읽어서 shared 파일에 삽입합니다.\n  const dictUtilsPath = path.join(\n    import.meta.dirname.replace(\"/dist/\", \"/src/\"),\n    \"../dict/utils.ts\",\n  );\n  const dictUtilsCode = (await exists(dictUtilsPath)) ? await readFile(dictUtilsPath, \"utf-8\") : \"\";\n\n  // 특정 변수 치환을 위해서 사용합니다.\n  const convertMap = {\n    baseUrl:\n      Sonamu.config.server.baseUrl ??\n      `http://${Sonamu.config.server.listen?.host ?? \"localhost\"}:${Sonamu.config.server.listen?.port ?? 3000}`,\n    dictUtils: dictUtilsCode,\n  };\n\n  for (const target of targets) {\n    // 지금 가져가려는 이 파일은 Sonamu 코드베이스의 일부입니다.\n    // 그런데 dist 속 빌드된 소스 코드 파일이 필요한 것이 아니고, src에만 있는 텍스트 파일이 필요합니다.\n    // 따라서 /src/에서 찾습니다.\n    const srcPath = path.join(\n      import.meta.dirname.replace(\"/dist/\", \"/src/\"),\n      `../shared/${target}.shared.ts.txt`,\n    );\n    if (!(await exists(srcPath))) {\n      continue;\n    }\n    if (!(await exists(path.join(Sonamu.appRootPath, target)))) {\n      throw new Error(\n        `Tried to copy sonamu.shared.ts to target '${target}' but the target directory does not exist. Please check your project directory structure.`,\n      );\n    }\n\n    const fullText = await readFile(srcPath, \"utf-8\");\n    const convertedText = Object.entries(convertMap).reduce(\n      (acc, [key, value]) => acc.replace(`$[[${key}]]`, value),\n      fullText,\n    );\n\n    // 이건 프로젝트에 .ts 소스 코드 파일을 생성하는 것이므로 src의 .ts 경로로 갑니다.\n    const destPath = path.join(Sonamu.appRootPath, target, \"src/services/sonamu.shared.ts\");\n\n    // 정말 혹시나지만 target 디렉토리는 있어도 src/services 디렉토리는 없을 수 있으므로 미리 생성해줍니다.\n    if (!(await exists(path.dirname(destPath)))) {\n      await mkdir(path.dirname(destPath), { recursive: true });\n      console.warn(`Created directory '${path.dirname(destPath)}' because it did not exist.`);\n    }\n\n    // 파일이 이미 존재하면 건너뜁니다.\n    // sonamu.shared.ts는 프로젝트에서 자유롭게 커스터마이징할 수 있어야 하므로,\n    // 최초 1회만 생성하고 이후에는 덮어쓰지 않습니다.\n    // 템플릿 내용($[[dictUtils]] 등)이 변경되었을 때 반영이 필요하면,\n    // 해당 파일을 삭제한 뒤 `pnpm sonamu sync`로 재생성하면 됩니다.\n    if (await exists(destPath)) {\n      continue;\n    }\n\n    await writeFile(destPath, await formatCode(convertedText, destPath));\n    !isTest() &&\n      console.log(chalk.bold(\"Copied: \") + chalk.blue(path.relative(Sonamu.appRootPath, destPath)));\n  }\n}\n\n/**\n * Sonamu Dictionary(SD)를 모든 타겟(api + web/app 등)에 동기화합니다.\n *\n * 각 타겟에 대해:\n * - target이 api가 아니면 사용자 작성 locale 파일(ko.ts/en.ts/...)을 api → target으로 복사\n * - sd 템플릿을 렌더링해서 sd.generated.ts 생성 (overwrite)\n *\n * 한 타겟에서 실패해도 다른 타겟은 계속 진행합니다.\n */\nexport async function actionSyncSonamuDictionary(): Promise<void> {\n  const { targets } = Sonamu.config.sync;\n  const i18nConfig = Sonamu.config.i18n;\n\n  const targetList = [\"api\", ...targets] as (\"api\" | \"web\" | \"app\")[];\n\n  const apiI18nDir = path.join(Sonamu.appRootPath, Sonamu.config.api.dir, \"src/i18n\");\n\n  for (const target of targetList) {\n    try {\n      // web/app의 경우 locale 파일들을 api에서 복사\n      if (target !== \"api\") {\n        await syncLocaleFiles(target, apiI18nDir, i18nConfig.supportedLocales);\n      }\n\n      await generateTemplate(\"sd\", { target }, { overwrite: true });\n    } catch (e) {\n      console.error(`Failed to generate SD template for ${target}:`, e);\n    }\n  }\n}\n\n/**\n * api의 locale 파일을 web/app으로 복사합니다.\n */\nasync function syncLocaleFiles(\n  target: string,\n  apiI18nDir: string,\n  locales: string[],\n): Promise<void> {\n  const targetI18nDir = path.join(Sonamu.appRootPath, target, \"src/i18n\");\n\n  // 디렉토리가 없으면 생성\n  await mkdir(targetI18nDir, { recursive: true });\n\n  for (const locale of locales) {\n    const sourceFile = path.join(apiI18nDir, `${locale}.ts`);\n    const targetFile = path.join(targetI18nDir, `${locale}.ts`);\n\n    const syncHeader = [\n      \"/**\",\n      \" * @generated\",\n      \" * API에서 동기화된 파일입니다. 직접 수정하지 마세요.\",\n      \" */\",\n    ].join(\"\\n\");\n    await copyFileWithReplaceCoreToShared(sourceFile, targetFile, syncHeader);\n    await trackWritten(targetFile as AbsolutePath);\n    !isTest() &&\n      console.log(chalk.bold(\"Copied: \") + chalk.cyan(`${target}/src/i18n/${locale}.ts`));\n  }\n}\n"],"mappings":"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"}
|
|
212
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"syncer-actions.js","names":[],"sources":["../../src/syncer/syncer-actions.ts"],"sourcesContent":["import assert from \"assert\";\nimport { mkdir, readFile, writeFile } from \"fs/promises\";\nimport path, { dirname } from \"path\";\n\nimport chalk from \"chalk\";\n\nimport { Sonamu } from \"../api/sonamu\";\nimport { type EntityNamesRecord } from \"../entity/entity-manager\";\nimport { AlreadyProcessedException } from \"../exceptions/so-exceptions\";\nimport { Naite } from \"../naite/naite\";\nimport { isTest } from \"../utils/controller\";\nimport { formatCode } from \"../utils/formatter\";\nimport { copyFileWithReplaceCoreToShared, exists } from \"../utils/fs-utils\";\nimport { type AbsolutePath } from \"../utils/path-utils\";\nimport { generateTemplate } from \"./code-generator\";\nimport { trackWritten } from \"./file-tracking\";\n\n// web/.sonamu.env 에 현재 설정값 저장\nexport async function actionSyncConfig() {\n  const { host, port } = Sonamu.config.server.listen ?? {};\n  const content = `API_HOST=${host ?? \"localhost\"}\\nAPI_PORT=${port ?? 3000}`;\n\n  Naite.t(\"actionSyncConfig\", { content });\n  await Promise.all([\n    ...Sonamu.config.sync.targets.map(async (target) => {\n      await writeFile(path.join(Sonamu.appRootPath, target, \".sonamu.env\"), content);\n    }),\n    generateTemplate(\"generated_sso\", {}, { overwrite: true }),\n  ]);\n}\n\n/**\n * services.generated.ts를 생성합니다.\n * @param paramsArray\n * @returns 생성된 파일 경로 배열.\n */\nexport async function actionGenerateServices(\n  paramsArray: {\n    namesRecord: EntityNamesRecord;\n  }[],\n): Promise<AbsolutePath[]> {\n  Naite.t(\"actionGenerateServices\", paramsArray);\n\n  // services.generated.ts 통합 파일 생성\n  const servicesFile = await generateTemplate(\n    \"services\",\n    {},\n    {\n      overwrite: true,\n    },\n  );\n\n  return [...servicesFile];\n}\n\n/**\n * Entity에 딸린 초기 types.ts를 만들어줍니다.\n * @param entityId\n */\nexport async function actionGenerateInitialTypes(entityId: string): Promise<AbsolutePath[]> {\n  return generateTemplate(\"init_types\", { entityId });\n}\n\n/**\n * sonamu.generated.ts와 sonamu.generated.sso.ts를 생성합니다.\n * @returns 생성된 파일 경로 배열.\n */\nexport async function actionGenerateSchemas(): Promise<AbsolutePath[]> {\n  return (\n    await Promise.all([\n      generateTemplate(\"generated_sso\", {}, { overwrite: true }),\n      generateTemplate(\"generated\", {}, { overwrite: true }),\n    ])\n  ).flat();\n}\n\n/**\n * sonamu.generated.http를 생성합니다.\n * @returns 생성된 파일 경로.\n */\nexport async function actionGenerateHttps(): Promise<AbsolutePath> {\n  const [res] = await generateTemplate(\n    \"generated_http\",\n    { entityId: \"dummy\" },\n    { overwrite: true },\n  );\n  assert(res);\n  return res;\n}\n\n/**\n * queries.generated.ts 재생성합니다.\n * @returns 생성된 파일 경로 배열.\n */\nexport async function actionGenerateSsrQueries(): Promise<AbsolutePath[]> {\n  return generateTemplate(\"queries\", {}, { overwrite: true });\n}\n\n/**\n * entry-server.generated.tsx를 생성합니다.\n * 다른 액션들과 달리, 이미 파일이 있으면 그냥 놔둡니다. 그래서 함수 이름 끝에 써놨어요 ㅎ\n * 입력 의존 없는 정적 코드라 매번 overwrite는 mtime만 갱신하는 의미 없는 동작.\n * 템플릿 자체가 변경된 경우(Sonamu 업그레이드)에는 사용자가 파일을 삭제한 뒤 sync로 재생성.\n * @returns 생성된 파일 경로 배열 (이미 있으면 빈 배열).\n */\nexport async function actionGenerateSsrEntryServerIfNotExists(): Promise<AbsolutePath[]> {\n  try {\n    return await generateTemplate(\"entry_server\", {}, { overwrite: false });\n  } catch (e) {\n    // generateTemplate은 overwrite: false에서 파일이 이미 있으면 예외를 던집니다.\n    // IfNotExists 의미상 \"그냥 놔둔다\"가 정상이므로 빈 배열로 변환합니다.\n    if (e instanceof AlreadyProcessedException) {\n      return [];\n    }\n    throw e;\n  }\n}\n\n/**\n * 주어진 .ts 파일들(api에 있다고 가정)을 타겟 디렉토리의 services에 갖다 둡니다.\n * 이때 내부의 sonamu import는 sonamu.shared.ts import로 치환되고,\n * 경로의 /application/은 /services/로 치환됩니다.\n *\n * 기본은 Sonamu.config.sync.targets 전체를 대상으로 하며,\n * onlyTargetsStartingWith 화이트리스트로 일부 target에만 분배할 수 있습니다.\n * 가량 SSR-only인 queries.generated.ts같은 친구들은 \"web\"으로 시작하는 타겟들에만 보낼 수 있어요.\n * [\"web\"]으로 넘기면 web, web-admin, webapp 등에 모두 매치됩니다.\n *\n * @param tsPaths 복사할 파일들의 절대 경로\n * @param onlyTargetsStartingWith 분배할 target 접두어들의 화이트리스트. 미지정 시 모든 target.\n * @returns 각 타겟에 복사된 파일들의 절대 경로 배열 (flat).\n */\nexport async function actionSyncFilesToTargets(\n  tsPaths: AbsolutePath[],\n  onlyTargetsStartingWith?: string[],\n): Promise<string[]> {\n  const allTargets = Sonamu.config.sync.targets;\n  const targets = onlyTargetsStartingWith\n    ? allTargets.filter((t) => onlyTargetsStartingWith.some((prefix) => t.startsWith(prefix)))\n    : allTargets;\n  const { dir: apiDir } = Sonamu.config.api;\n\n  return (\n    await Promise.all(\n      targets.map(async (target) =>\n        Promise.all(\n          tsPaths.map(async (realSrc) => {\n            const dst = realSrc\n              .replace(`/${apiDir}/`, `/${target}/`)\n              .replace(\"/application/\", \"/services/\");\n            const dir = dirname(dst);\n            if (!(await exists(dir))) {\n              await mkdir(dir, { recursive: true });\n            }\n            const syncHeader = [\n              \"/**\",\n              \" * @generated\",\n              \" * API에서 동기화된 파일입니다. 직접 수정하지 마세요.\",\n              \" */\",\n            ].join(\"\\n\");\n            await copyFileWithReplaceCoreToShared(realSrc, dst, syncHeader);\n            await trackWritten(dst as AbsolutePath);\n            !isTest() &&\n              console.log(\n                chalk.bold(\"Copied: \") + chalk.blue(dst.replace(`${Sonamu.appRootPath}/`, \"\")),\n              );\n            return dst;\n          }),\n        ),\n      ),\n    )\n  ).flat();\n}\n\n/**\n * shared 템플릿으로부터 sonamu.shared.ts 파일을 만들어서 모든 타겟 디렉토리에 갖다 둡니다.\n * 파일을 만드는 과정에서 여러 치환 가공이 일어납니다.\n *\n * 다른 액션들과 달리, 이미 파일이 있으면 그냥 놔둡니다. 그래서 함수 이름 끝에 써놨어요 ㅎ\n */\nexport async function actionCopySharedToTargetsIfNotExists(): Promise<void> {\n  const { targets } = Sonamu.config.sync;\n\n  // plural.ts 내용을 읽어서 shared 파일에 삽입합니다.\n  const dictUtilsPath = path.join(\n    import.meta.dirname.replace(\"/dist/\", \"/src/\"),\n    \"../dict/utils.ts\",\n  );\n  const dictUtilsCode = (await exists(dictUtilsPath)) ? await readFile(dictUtilsPath, \"utf-8\") : \"\";\n\n  // 특정 변수 치환을 위해서 사용합니다.\n  const convertMap = {\n    baseUrl:\n      Sonamu.config.server.baseUrl ??\n      `http://${Sonamu.config.server.listen?.host ?? \"localhost\"}:${Sonamu.config.server.listen?.port ?? 3000}`,\n    dictUtils: dictUtilsCode,\n  };\n\n  for (const target of targets) {\n    // 지금 가져가려는 이 파일은 Sonamu 코드베이스의 일부입니다.\n    // 그런데 dist 속 빌드된 소스 코드 파일이 필요한 것이 아니고, src에만 있는 텍스트 파일이 필요합니다.\n    // 따라서 /src/에서 찾습니다.\n    const srcPath = path.join(\n      import.meta.dirname.replace(\"/dist/\", \"/src/\"),\n      `../shared/${target}.shared.ts.txt`,\n    );\n    if (!(await exists(srcPath))) {\n      continue;\n    }\n    if (!(await exists(path.join(Sonamu.appRootPath, target)))) {\n      throw new Error(\n        `Tried to copy sonamu.shared.ts to target '${target}' but the target directory does not exist. Please check your project directory structure.`,\n      );\n    }\n\n    const fullText = await readFile(srcPath, \"utf-8\");\n    const convertedText = Object.entries(convertMap).reduce(\n      (acc, [key, value]) => acc.replace(`$[[${key}]]`, value),\n      fullText,\n    );\n\n    // 이건 프로젝트에 .ts 소스 코드 파일을 생성하는 것이므로 src의 .ts 경로로 갑니다.\n    const destPath = path.join(Sonamu.appRootPath, target, \"src/services/sonamu.shared.ts\");\n\n    // 정말 혹시나지만 target 디렉토리는 있어도 src/services 디렉토리는 없을 수 있으므로 미리 생성해줍니다.\n    if (!(await exists(path.dirname(destPath)))) {\n      await mkdir(path.dirname(destPath), { recursive: true });\n      console.warn(`Created directory '${path.dirname(destPath)}' because it did not exist.`);\n    }\n\n    // 파일이 이미 존재하면 건너뜁니다.\n    // sonamu.shared.ts는 프로젝트에서 자유롭게 커스터마이징할 수 있어야 하므로,\n    // 최초 1회만 생성하고 이후에는 덮어쓰지 않습니다.\n    // 템플릿 내용($[[dictUtils]] 등)이 변경되었을 때 반영이 필요하면,\n    // 해당 파일을 삭제한 뒤 `pnpm sonamu sync`로 재생성하면 됩니다.\n    if (await exists(destPath)) {\n      continue;\n    }\n\n    await writeFile(destPath, await formatCode(convertedText, destPath));\n    !isTest() &&\n      console.log(chalk.bold(\"Copied: \") + chalk.blue(path.relative(Sonamu.appRootPath, destPath)));\n  }\n}\n\n/**\n * Sonamu Dictionary(SD)를 모든 타겟(api + web/app 등)에 동기화합니다.\n *\n * 각 타겟에 대해:\n * - target이 api가 아니면 사용자 작성 locale 파일(ko.ts/en.ts/...)을 api → target으로 복사\n * - sd 템플릿을 렌더링해서 sd.generated.ts 생성 (overwrite)\n *\n * 한 타겟에서 실패해도 다른 타겟은 계속 진행합니다.\n */\nexport async function actionSyncSonamuDictionary(): Promise<void> {\n  const { targets } = Sonamu.config.sync;\n  const i18nConfig = Sonamu.config.i18n;\n\n  const targetList = [\"api\", ...targets] as (\"api\" | \"web\" | \"app\")[];\n\n  const apiI18nDir = path.join(Sonamu.appRootPath, Sonamu.config.api.dir, \"src/i18n\");\n\n  for (const target of targetList) {\n    try {\n      // web/app의 경우 locale 파일들을 api에서 복사\n      if (target !== \"api\") {\n        await syncLocaleFiles(target, apiI18nDir, i18nConfig.supportedLocales);\n      }\n\n      await generateTemplate(\"sd\", { target }, { overwrite: true });\n    } catch (e) {\n      console.error(`Failed to generate SD template for ${target}:`, e);\n    }\n  }\n}\n\n/**\n * api의 locale 파일을 web/app으로 복사합니다.\n */\nasync function syncLocaleFiles(\n  target: string,\n  apiI18nDir: string,\n  locales: string[],\n): Promise<void> {\n  const targetI18nDir = path.join(Sonamu.appRootPath, target, \"src/i18n\");\n\n  // 디렉토리가 없으면 생성\n  await mkdir(targetI18nDir, { recursive: true });\n\n  for (const locale of locales) {\n    const sourceFile = path.join(apiI18nDir, `${locale}.ts`);\n    const targetFile = path.join(targetI18nDir, `${locale}.ts`);\n\n    const syncHeader = [\n      \"/**\",\n      \" * @generated\",\n      \" * API에서 동기화된 파일입니다. 직접 수정하지 마세요.\",\n      \" */\",\n    ].join(\"\\n\");\n    await copyFileWithReplaceCoreToShared(sourceFile, targetFile, syncHeader);\n    await trackWritten(targetFile as AbsolutePath);\n    !isTest() &&\n      console.log(chalk.bold(\"Copied: \") + chalk.cyan(`${target}/src/i18n/${locale}.ts`));\n  }\n}\n"],"mappings":"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"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"syncer.d.ts","sourceRoot":"","sources":["../../src/syncer/syncer.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,YAAY,EAAE,MAAM,QAAQ,CAAC;AAQtC,OAAO,EAAE,KAAK,CAAC,EAAE,MAAM,KAAK,CAAC;AAE7B,OAAO,EAAE,cAAc,EAAE,MAAM,mBAAmB,CAAC;AAKnD,OAAO,EAAE,KAAK,gBAAgB,EAAE,MAAM,oBAAoB,CAAC;AAE3D,OAAO,EAAE,KAAK,eAAe,EAAE,KAAK,WAAW,EAAE,MAAM,gBAAgB,CAAC;AACxE,OAAO,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAC7C,OAAO,EAAE,KAAK,eAAe,EAAE,MAAM,gBAAgB,CAAC;AAKtD,OAAO,EAAE,KAAK,YAAY,EAAE,MAAM,qBAAqB,CAAC;AAMxD,OAAO,EAAE,KAAK,QAAQ,EAAE,MAAM,iBAAiB,CAAC;AAEhD,OAAO,EAAE,KAAK,UAAU,EAAE,KAAK,YAAY,EAAE,KAAK,WAAW,EAAE,MAAM,iBAAiB,CAAC;AAGvF,KAAK,UAAU,GAAG;KACf,GAAG,IAAI,QAAQ,GAAG,YAAY,EAAE;CAClC,CAAC;AAEF,qBAAa,MAAM;IACjB,IAAI,EAAE,UAAU,CAAM;IACtB,KAAK,EAAE,WAAW,CAAM;IACxB,MAAM,EAAE,YAAY,CAAM;IAC1B,SAAS,EAAE,GAAG,CAAC,MAAM,EAAE,gBAAgB,EAAE,CAAC,CAAa;IACvD,YAAY,EAAE,YAAY,CAAsB;IAEhD;;;;OAIG;IACG,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC;IA6B3B;;;;;;OAMG;IACG,SAAS,IAAI,OAAO,CAAC,IAAI,CAAC;IAQhC;;;;;;;;;;;;OAYG;IACG,UAAU,CAAC,UAAU,EAAE,GAAG,CAAC,YAAY,EAAE,QAAQ,GAAG,KAAK,CAAC,GAAG,OAAO,CAAC,IAAI,CAAC;IAkChF,OAAO,CAAC,kCAAkC;YAc5B,mCAAmC;YAkBnC,wCAAwC;IAkDtD,+BAA+B,CAC7B,eAAe,EAAE,YAAY,GAC5B,CAAC,OAAO,cAAc,CAAC,CAAC,MAAM,CAAC,EAAE;IAiB9B,aAAa;IAIb,cAAc;IAId,YAAY;IAIZ,iBAAiB;IAKjB,iBAAiB,IAAI,OAAO,CAAC,IAAI,CAAC;IA8BxC;;;;;;OAMG;IACG,aAAa,CAAC,aAAa,EAAE,YAAY,EAAE,GAAG,OAAO,CAAC;QAAE,SAAS,EAAE,QAAQ,EAAE,CAAA;KAAE,CAAC;IA2CtF,mBAAmB,CAAC,SAAS,EAAE,YAAY,EAAE,GAAG,UAAU;IAkB1D,OAAO,CAAC,aAAa;IA8Bf,wBAAwB,CAAC,UAAU,EAAE,UAAU,GAAG,OAAO,CAAC,IAAI,CAAC;
|
|
1
|
+
{"version":3,"file":"syncer.d.ts","sourceRoot":"","sources":["../../src/syncer/syncer.ts"],"names":[],"mappings":"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"}
|
package/dist/syncer/syncer.js
CHANGED
|
@@ -296,7 +296,8 @@ var init_syncer = __esmMin((() => {
|
|
|
296
296
|
}
|
|
297
297
|
}
|
|
298
298
|
const generated = await actionGenerateSchemas();
|
|
299
|
-
|
|
299
|
+
const distributable = generated.filter((p) => !p.endsWith(".sso.ts"));
|
|
300
|
+
await actionSyncFilesToTargets(distributable);
|
|
300
301
|
}
|
|
301
302
|
async handleImplementationChanges(diffGroups) {
|
|
302
303
|
Naite.t("handleImplementationChanges", { diffGroups });
|
|
@@ -315,7 +316,7 @@ var init_syncer = __esmMin((() => {
|
|
|
315
316
|
await actionGenerateServices(params);
|
|
316
317
|
await actionGenerateHttps();
|
|
317
318
|
const queries = await actionGenerateSsrQueries();
|
|
318
|
-
await actionSyncFilesToTargets(queries);
|
|
319
|
+
await actionSyncFilesToTargets(queries, ["web"]);
|
|
319
320
|
}
|
|
320
321
|
async handleAuxiliarySymbolChanges(diffGroups) {
|
|
321
322
|
Naite.t("handleAuxiliarySymbolChanges", { diffGroups });
|
|
@@ -421,4 +422,4 @@ var init_syncer = __esmMin((() => {
|
|
|
421
422
|
//#endregion
|
|
422
423
|
init_syncer();
|
|
423
424
|
export { Syncer, init_syncer };
|
|
424
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"syncer.js","names":["SyncerActions.actionCopySharedToTargetsIfNotExists","SyncerActions.actionGenerateSsrEntryServerIfNotExists","syncTriggeringPaths: AbsolutePath[]","SyncerActions.actionGenerateInitialTypes","SyncerActions.actionSyncFilesToTargets","SyncerActions.actionGenerateSchemas","params: {\n      namesRecord: EntityNamesRecord;\n    }[]","SyncerActions.actionGenerateServices","SyncerActions.actionGenerateHttps","SyncerActions.actionGenerateSsrQueries","SyncerActions.actionSyncConfig","SyncerActions.actionSyncSonamuDictionary","keys: TemplateKey[]","p","target"],"sources":["../../src/syncer/syncer.ts"],"sourcesContent":["import assert from \"assert\";\nimport { EventEmitter } from \"events\";\nimport { unlink } from \"fs/promises\";\nimport path from \"path\";\n\nimport { hot } from \"@sonamu-kit/hmr-hook\";\nimport chalk from \"chalk\";\nimport { minimatch } from \"minimatch\";\nimport { group, unique } from \"radashi\";\nimport { type z } from \"zod\";\n\nimport { registeredApis } from \"../api/decorators\";\nimport { Sonamu } from \"../api/sonamu\";\nimport { EntityManager } from \"../entity/entity-manager\";\nimport { type EntityNamesRecord } from \"../entity/entity-manager\";\nimport { Naite } from \"../naite/naite\";\nimport { type WorkflowMetadata } from \"../tasks/decorator\";\nimport { TemplateManager } from \"../template/template-manager\";\nimport { type GenerateOptions, type PathAndCode } from \"../types/types\";\nimport { TemplateKey } from \"../types/types\";\nimport { type TemplateOptions } from \"../types/types\";\nimport { mapAsync, reduceAsync } from \"../utils/async-utils\";\nimport { centerText } from \"../utils/console-util\";\nimport { isTest } from \"../utils/controller\";\nimport { exists } from \"../utils/fs-utils\";\nimport { type AbsolutePath } from \"../utils/path-utils\";\nimport { runWithGracefulShutdown } from \"../utils/process-utils\";\nimport { findChangedFilesUsingChecksums, renewChecksums } from \"./checksum\";\nimport { generateTemplate, renderTemplate } from \"./code-generator\";\nimport { createEntity, delEntity } from \"./entity-operations\";\nimport { getChecksumPatternGroup, getChecksumPatternGroupInAbsolutePath } from \"./file-patterns\";\nimport { type FileType } from \"./file-patterns\";\nimport { loadApis, loadModels, loadTypes, loadWorkflows } from \"./module-loader\";\nimport { type LoadedApis, type LoadedModels, type LoadedTypes } from \"./module-loader\";\nimport * as SyncerActions from \"./syncer-actions\";\n\ntype DiffGroups = {\n  [key in FileType]: AbsolutePath[];\n};\n\nexport class Syncer {\n  apis: LoadedApis = [];\n  types: LoadedTypes = {};\n  models: LoadedModels = {};\n  workflows: Map<string, WorkflowMetadata[]> = new Map();\n  eventEmitter: EventEmitter = new EventEmitter();\n\n  /**\n   * 체크섬이 변경된 부분에 대해 싱크를 진행합니다.\n   * dev 서버가 처음 떴을 때, sonamu sync 할 때 실행됩니다. 이후에는 syncFromWatcher 경로를 타요.\n   * @returns\n   */\n  async sync(): Promise<void> {\n    // 초기 부트스트랩! 얘네들은 idempotent하고 가볍기 때문에 무지성 실행해도 됩니다.\n    // 얘네들은 sonamu.lock에 들어가지도 않고 따라서 HMR 경로를 타지도 않는 친구들입니다.\n    // 그래서 아무 때나 그냥 돌려주면 되는데, syncFromWatcher에서 매번 하는 것은 낭비이니 여기서 한 번만 합니다.\n    await SyncerActions.actionCopySharedToTargetsIfNotExists();\n    await SyncerActions.actionGenerateSsrEntryServerIfNotExists();\n\n    // 바뀐 것이 없으면 그냥 넘어가요.\n    const changedFiles = await findChangedFilesUsingChecksums();\n    if (changedFiles.length === 0) {\n      console.log(chalk.black.bgGreen(centerText(\"All files are synced!\")));\n      return;\n    }\n\n    // 여기서 실제 싱크 동작을 수행합니다.\n    // 다만 싱크 중에 프로세스가 죽으면 꼬여버리기 때문에,\n    // 시그널에도 잠시 버틸 수 있는 환경 속에서 싱크를 실행합니다.\n    await runWithGracefulShutdown(\n      async () => {\n        // 얘가 싱크 작업 수행하는 본체입니다.\n        await this.doSyncActions(changedFiles);\n\n        // 싱크 액션이 끝나면 항상 체크섬을 다시 갱신합니다.\n        await renewChecksums();\n      },\n      { whenThisHappens: \"SIGUSR2\", waitForUpTo: 20000 },\n    );\n  }\n\n  /**\n   * 강제 풀-싱크: lock을 무시하고 처음부터 다시 싱크합니다.\n   *\n   * **사용처**: git post-merge hook, CI, dev 서버의 `f` 핫키.\n   * **실패 안전성**: 도중에 프로세스가 죽어 lock 없는 상태로 남아도 무해 — 다음 sync에서\n   * lock 없으면 자연스럽게 풀-싱크가 트리거되어 새 lock이 작성됨.\n   */\n  async forceSync(): Promise<void> {\n    const lockPath = path.join(Sonamu.apiRootPath, \"sonamu.lock\");\n    if (await exists(lockPath)) {\n      await unlink(lockPath);\n    }\n    await this.sync();\n  }\n\n  /**\n   * Watcher가 batch로 모은 변경 파일들에 대해 한 번의 HMR/sync 사이클을 돕니다.\n   *\n   * HMR은 api/src 안에서 일어나는 모든 파일들에 대해서 수행합니다.\n   * checksumPatternGroup 매칭 여부와 무관하게 api/src 전체 대상입니다.\n   * 가령 api/src/utils/subset-loaders.ts 같은 파일도 변경되면 HMR은 해줍니다.\n   *\n   * Sync는 checksumPatternGroup으로 매칭되는 파일들에 대해서만 수행합니다.\n   * 여기에는 web/src나 app/src 같은 다른 target의 파일이 포함될 수 있습니다.\n   * 이런 non-api 경로의 파일들은 HMR과는 아무 상관이 없으므로, invalidate을 하지 않습니다.\n   *\n   * @param fileEvents - path → event 맵. event는 \"change\" | \"add\".\n   */\n  async hmrAndSync(fileEvents: Map<AbsolutePath, \"change\" | \"add\">): Promise<void> {\n    const hmrActionRequiredEvents = this.extractHmrActionRequiredFileEvents(fileEvents);\n    const syncTriggeringPaths = await this.extractSyncTriggeringFileEventPaths(fileEvents);\n\n    // HMR 영역: 파일 이벤트 중 api의 모듈 그래프에 있는 파일들에 대한 변동은 hmrActionRequiredEvents로 잡힙니다.\n    // 이 친구들은 invalidate 처리해줍니다.\n    // 이 호출은 아래 sync보다 무조건 먼저 일어나야 합니다!\n    // 왜냐하면 sync에서는 변경된 model 코드를 새로 import해서 처리해야 하는 경우도 있기 때문입니다.\n    await this.invalidateDependentsAffectedByFileEvents(hmrActionRequiredEvents);\n\n    // Sync 영역: checksumPatternGroup에 명시된 파일들에 대한 변동은 syncTriggeringPaths로 잡힙니다.\n    // 이 친구들은 적절한 sync 작업으로 대응합니다.\n    if (syncTriggeringPaths.length > 0) {\n      await this.doSyncActions(syncTriggeringPaths);\n    }\n\n    // 싱크 작업이 끝났으면 무지성 로드를 수행합니다.\n    // 싱크를 안 한 경우에도 로드는 해야 해요. 위에서 관련있는 친구들은 다 invalidate 되었거든요!\n    //\n    // 변경된 파일들에 대해서 새롭게 당겨오는(load) 행위는 doSyncActions에서 하지 않습니다.\n    // doSyncActions에서는 파일을 읽고 만드는 싱크 행위만 합니다.\n    // 거기서 딱 변경된 부분에 영향받는 autoload만 선별해서 수행하려면 너무 지저분하고 복잡해집니다.\n    //\n    // 퍼포먼스 영향은 무시해도 좋습니다.\n    // 어차피 hmr-hook에 의해 invalidate된 부분들이 아니라면 캐시 그대로 유지합니다.\n    await this.autoloadTypes();\n    await this.autoloadModels();\n    await this.autoloadApis();\n    await this.autoloadWorkflows();\n    await this.autoloadSsrRoutes();\n\n    this.eventEmitter.emit(\"onHMRCompleted\");\n  }\n\n  private extractHmrActionRequiredFileEvents(\n    fileEvents: Map<AbsolutePath, \"change\" | \"add\">,\n  ): Map<AbsolutePath, \"change\" | \"add\"> {\n    const apiSrc = path.join(Sonamu.apiRootPath, \"src\");\n    const result = new Map<AbsolutePath, \"change\" | \"add\">();\n    for (const [filePath, event] of fileEvents) {\n      if (!filePath.startsWith(apiSrc)) {\n        continue;\n      }\n      result.set(filePath, event);\n    }\n    return result;\n  }\n\n  private async extractSyncTriggeringFileEventPaths(\n    fileEvents: Map<AbsolutePath, \"change\" | \"add\">,\n  ): Promise<AbsolutePath[]> {\n    const checkPatternGroup = getChecksumPatternGroupInAbsolutePath();\n    const syncTriggeringPaths: AbsolutePath[] = [];\n    for (const [diffFilePath] of fileEvents) {\n      const isInCheckPatternGroup = Object.values(checkPatternGroup).some((pattern) =>\n        minimatch(diffFilePath, pattern),\n      );\n      if (!isInCheckPatternGroup) {\n        continue;\n      }\n      syncTriggeringPaths.push(diffFilePath);\n    }\n\n    return syncTriggeringPaths;\n  }\n\n  private async invalidateDependentsAffectedByFileEvents(\n    fileEvents: Map<AbsolutePath, \"change\" | \"add\">,\n  ) {\n    for (const [diffFilePath, event] of fileEvents) {\n      // 변경된 파일과 dependent 파일들을 invalidate 합니다.\n      // 한 번 이상 import된 친구들에 대해서만 실제 작업이 일어납니다.\n      // 그러니 안심하고 invalidate 해도 됩니다.\n      // 테스트 환경에서는 hot.invalidateFile시 초기 에러가 발생하기 때문에 invalidate 하지 않습니다.\n      if (!isTest()) {\n        const invalidatedPaths = (await hot.invalidateFile(diffFilePath, event)) as AbsolutePath[];\n\n        if (invalidatedPaths.length > 0) {\n          console.log(chalk.bold(`🔄 Invalidated:`));\n\n          for (const invalidatedPath of invalidatedPaths) {\n            try {\n              // 만약 model.ts 파일이 변경(invalidate)되었다? 그러면 registeredApis 중에서 이 모델에 해당하는 api들은 지워줘요.\n              // registeredApis는 통으로 다 날려버릴 수 없습니다. registeredApis에 올라오는 친구들은 초기 로드시 또는 HMR시에만 등록되기 때문입니다.\n              // 따라서 model.ts 파일의 변경으로 다음번 새로운 eval이 예상되는 이 시점에서만, 이 모델에서 나온 registeredApis들을 지워줄 수 있습니다.\n              const removedApis = this.removeInvalidatedRegisteredApis(invalidatedPath);\n              if (removedApis.length > 0) {\n                console.log(\n                  chalk.blue(`- ${path.relative(Sonamu.apiRootPath, invalidatedPath)}`),\n                  chalk.gray(`(with ${removedApis.length} APIs)`),\n                );\n              } else {\n                console.log(chalk.blue(`- ${path.relative(Sonamu.apiRootPath, invalidatedPath)}`));\n              }\n            } catch (e) {\n              console.error(e);\n              console.error(\n                chalk.red(`Failed to remove invalidated registered APIs for ${invalidatedPath}`),\n              );\n            }\n          }\n        }\n      }\n\n      // devRunner 활성화 시, 변경된 소스 파일을 Vitest 모듈 그래프에서도 무효화합니다.\n      // Vite의 moduleGraph.invalidateModule()이 importer 방향으로 재귀적 cascade하므로,\n      // 소스 파일 하나만 무효화하면 이를 import하는 테스트 파일도 자동으로 무효화됩니다.\n      if (!isTest() && Sonamu.config.test?.devRunner?.enabled && Sonamu.devVitestManager) {\n        Sonamu.devVitestManager.invalidateFiles([diffFilePath]);\n        console.log(\n          chalk.dim(`Test invalidated: ${path.relative(Sonamu.apiRootPath, diffFilePath)}`),\n        );\n      }\n    }\n  }\n\n  removeInvalidatedRegisteredApis(\n    invalidatedPath: AbsolutePath,\n  ): (typeof registeredApis)[number][] {\n    if (!invalidatedPath.endsWith(\".model.ts\" /*소스 코드를 다루는 상황이니 .ts 경로로 봅니다.*/)) {\n      return [];\n    }\n\n    const entityId = EntityManager.getEntityIdFromPath(invalidatedPath);\n    const toRemove = registeredApis.filter((api) => api.modelName === `${entityId}Model`);\n    for (const api of toRemove) {\n      const idx = registeredApis.indexOf(api);\n      if (idx !== -1) {\n        registeredApis.splice(idx, 1);\n      }\n    }\n\n    return toRemove;\n  }\n\n  async autoloadTypes() {\n    this.types = await loadTypes();\n  }\n\n  async autoloadModels() {\n    this.models = await loadModels();\n  }\n\n  async autoloadApis() {\n    this.apis = await loadApis();\n  }\n\n  async autoloadWorkflows() {\n    this.workflows = await loadWorkflows();\n    await Sonamu.workflows.synchronize(this.workflows);\n  }\n\n  async autoloadSsrRoutes(): Promise<void> {\n    const ssrConfigPath = path.join(Sonamu.apiRootPath, \"src/ssr\");\n\n    // 기존 routes 초기화\n    const { clearSSRRoutes } = await import(\"../ssr\");\n    clearSSRRoutes();\n\n    // ssr 폴더 없으면 스킵\n    if (!(await exists(ssrConfigPath))) {\n      return;\n    }\n\n    // ssr 폴더 안의 모든 .ts 파일 로드\n    const { globAsync } = await import(\"../utils/async-utils\");\n    const { importMembers } = await import(\"../utils/esm-utils\");\n    const { runtimePath } = await import(\"../utils/path-utils\");\n\n    // runtimePath를 사용하여 개발/프로덕션 환경에 맞는 확장자 처리\n    const files = await globAsync(path.join(ssrConfigPath, runtimePath(\"**/*.ts\")));\n\n    for (const file of files) {\n      try {\n        // importMembers를 사용하면 파일의 side effect(registerSSR 호출)가 실행됨\n        await importMembers(file);\n      } catch (e) {\n        console.error(`Failed to load SSR route: ${file}`, e);\n      }\n    }\n  }\n\n  /**\n   * 실제 싱크를 수행하는 본체입니다.\n   * 변경된 파일들을 타입별로 분류하고 각 타입에 맞는 액션을 실행합니다.\n   *\n   * @param diffFilePaths - 변경된 파일들의 절대 경로 목록\n   * @returns diffTypes - 변경된 파일의 타입 목록 (entity, types, model 등)\n   */\n  async doSyncActions(diffFilePaths: AbsolutePath[]): Promise<{ diffTypes: FileType[] }> {\n    const diffGroups = this.calculateDiffGroups(diffFilePaths);\n    const diffTypes = Object.keys(diffGroups) as FileType[];\n\n    // 여기는 별로 중요한 파트는 아닙니다.\n    // 아래의 if 전개를 깔끔하게 하려고 만든 DSL 같은 거라서, 무시하셔도 됩니다.\n    const { changeMatches, nothingMatches, unhandledPaths } = this.changeMatcher(\n      diffTypes,\n      diffGroups,\n    );\n\n    if (changeMatches(\"entity\", \"types\")) {\n      await this.handleTruthSourceChanges(diffGroups);\n    }\n\n    if (changeMatches(\"model\", \"frame\")) {\n      await this.handleImplementationChanges(diffGroups);\n    }\n\n    if (changeMatches(\"types\", \"functions\")) {\n      await this.handleAuxiliarySymbolChanges(diffGroups);\n    }\n\n    if (changeMatches(\"config\")) {\n      await this.handleConfigChanges(diffGroups);\n    }\n\n    if (changeMatches(\"i18n\", \"entity\" /*레이블*/, \"config\" /*defaultLocale등*/)) {\n      await this.handleSonamuDictionaryRelatedChanges(diffGroups);\n    }\n\n    if (nothingMatches()) {\n      // 파일 변경은 감지되었으나 저 위 어느 changeMatches에도 걸리지 않은 파일들이 drifts입니다.\n      // syncer는 소스의 변경에는 반응하지만 산출물의 변경(drift)에는 직접적으로 반응하지 않습니다.\n      // 대신 이 drift에 대해 경고 정도만 출력해줍니다.\n      await this.handleDrifts(unhandledPaths());\n    }\n\n    return {\n      diffTypes,\n    };\n  }\n\n  calculateDiffGroups(diffFiles: AbsolutePath[]): DiffGroups {\n    const patternGroup = getChecksumPatternGroup();\n    const fileTypes = Object.keys(patternGroup) as FileType[];\n\n    return group(diffFiles, (filePath) => {\n      // 절대 경로 → appRoot 기준 상대 경로 (예: \"api/src/...\", \"web/src/...\")\n      const relativePath = path.relative(Sonamu.appRootPath, filePath);\n      if (relativePath.startsWith(\"..\")) return \"unknown\";\n\n      for (const fileType of fileTypes) {\n        if (minimatch(relativePath, patternGroup[fileType])) {\n          return fileType;\n        }\n      }\n      return \"unknown\";\n    }) as unknown as DiffGroups;\n  }\n\n  private changeMatcher(diffTypes: FileType[], diffGroups: DiffGroups) {\n    const handled = new Set<FileType>();\n\n    /**\n     * 변경 사항이 인자로 받은 FileType들 중 하나 이상을 포함하는지 확인합니다.\n     * 가령 [\"entity\"]가 변경된 호출에서 changeMatches(\"entity\")는 trye를 반환하며,\n     * [\"types\", \"i18n\"]이 변경된 호출에서 changeMatches(\"types\", \"functions\")도 true를 반환하지만,\n     * [\"functions\"]가 변경된 호출에서 changeMatches(\"frame\")은 false를 반환합니다.\n     * @param types\n     */\n    const changeMatches = (...types: FileType[]) => {\n      const matching = types.filter((t) => diffTypes.includes(t));\n      matching.forEach((t) => handled.add(t));\n      return matching.length > 0;\n    };\n\n    /**\n     * changeMatches로 매칭된 것이 하나도 없는지 여부를 가져옵니다.\n     */\n    const nothingMatches = () => handled.size === 0;\n\n    /**\n     * 어떤 changeMatches 호출에도 걸리지 않은 FileType들의 실제 파일 경로를 모아서 반환합니다.\n     */\n    const unhandledPaths = (): AbsolutePath[] =>\n      diffTypes.filter((t) => !handled.has(t)).flatMap((t) => diffGroups[t] ?? []);\n\n    return { changeMatches, nothingMatches, unhandledPaths };\n  }\n\n  async handleTruthSourceChanges(diffGroups: DiffGroups): Promise<void> {\n    Naite.t(\"handleTruthSourceChanges\", { diffGroups });\n\n    await EntityManager.reload();\n\n    // types 생성(entity 새로 추가된 경우)\n    // parentId가 없고, types가 없는 경우에만 생성\n    const entityPath = diffGroups.entity?.at(0);\n    if (entityPath !== undefined) {\n      const entityId = EntityManager.getEntityIdFromPath(entityPath);\n      const entity = EntityManager.get(entityId);\n\n      // 프로젝트에 생성되어야 하는 .ts 파일의 경로입니다.\n      const typeFilePath = path.join(\n        Sonamu.apiRootPath,\n        `src/application/${entity.names.fs}/${entity.names.fs}.types.ts`,\n      ) as AbsolutePath;\n\n      if (entity.parentId === undefined && !(await exists(typeFilePath))) {\n        // *.types.ts가 만들어집니다.\n        const types = await SyncerActions.actionGenerateInitialTypes(entityId);\n\n        // 그걸 타겟에 갖다둬요.\n        await SyncerActions.actionSyncFilesToTargets(types);\n      }\n    }\n\n    // sonamu.generated.ts가 만들어집니다.\n    const generated = await SyncerActions.actionGenerateSchemas();\n\n    // 그걸 target들에도 보내요.\n    await SyncerActions.actionSyncFilesToTargets(generated);\n  }\n\n  async handleImplementationChanges(diffGroups: DiffGroups): Promise<void> {\n    Naite.t(\"handleImplementationChanges\", { diffGroups });\n    const mergedGroup = [...(diffGroups.model ?? []), ...(diffGroups.frame ?? [])];\n\n    // generated_http.template.ts에서 syncer.types를 씁니다.\n    // service.template.ts에서 syncer.apis를 씁니다.\n    await this.autoloadModels();\n    await this.autoloadTypes();\n    await this.autoloadApis();\n\n    const params: {\n      namesRecord: EntityNamesRecord;\n    }[] = mergedGroup.map((modelPath) => {\n      if (modelPath.endsWith(\".model.ts\") || modelPath.endsWith(\".frame.ts\")) {\n        const entityId = EntityManager.getEntityIdFromPath(modelPath);\n        assert(entityId);\n        return {\n          namesRecord: EntityManager.getNamesFromId(entityId),\n        };\n      }\n      throw new Error(\"not reachable\");\n    });\n\n    // services.generated.ts를 target들에, sonamu.generated.http를 api에 만들어줘요.\n    await SyncerActions.actionGenerateServices(params);\n    await SyncerActions.actionGenerateHttps();\n\n    // queries.generated.ts가 만들어집니다.\n    const queries = await SyncerActions.actionGenerateSsrQueries();\n\n    // 그걸 target들에도 보내요.\n    await SyncerActions.actionSyncFilesToTargets(queries);\n  }\n\n  async handleAuxiliarySymbolChanges(diffGroups: DiffGroups): Promise<void> {\n    Naite.t(\"handleAuxiliarySymbolChanges\", { diffGroups });\n    const tsPaths = unique([...(diffGroups.types ?? []), ...(diffGroups.functions ?? [])]);\n    await SyncerActions.actionSyncFilesToTargets(tsPaths);\n  }\n\n  async handleConfigChanges(_: DiffGroups): Promise<void> {\n    await SyncerActions.actionSyncConfig();\n  }\n\n  async handleSonamuDictionaryRelatedChanges(_: DiffGroups): Promise<void> {\n    await SyncerActions.actionSyncSonamuDictionary();\n  }\n\n  async handleDrifts(drifts: AbsolutePath[]): Promise<void> {\n    if (drifts.length > 0) {\n      console.warn(\n        chalk.yellow(\n          \"⚠️ Sonamu가 자동 생성한 파일에 대한 변경이 감지되었습니다. 파일이 Sonamu watcher 외부에서 변경된 것으로 추정됩니다.\",\n        ),\n      );\n      for (const p of drifts) {\n        console.warn(chalk.yellow(`  - ${path.relative(Sonamu.appRootPath, p)}`));\n      }\n      console.warn(chalk.dim(\"  → `pnpm sonamu sync --force`를 권장합니다.\"));\n    }\n  }\n\n  /**\n   * 주어진 엔티티와 템플릿 키에 대해, 생성된 코드가 존재하는지 확인합니다.\n   * @param entityId 엔티티 ID\n   * @param templateKey 템플릿 키\n   * @param enumId 열거형 ID\n   * @returns 생성된 코드가 존재하는지 여부\n   */\n  async checkExistsGenCode(\n    entityId: string,\n    templateKey: TemplateKey,\n    enumId?: string,\n  ): Promise<{ subPath: string; fullPath: string; isExists: boolean }> {\n    const { target, path: genPath } = TemplateManager.get(templateKey).getTargetAndPath(\n      EntityManager.getNamesFromId(entityId),\n      enumId,\n    );\n\n    const subPath = path.join(target, genPath);\n    const fullPath = path.join(Sonamu.appRootPath, subPath);\n    return {\n      subPath,\n      fullPath,\n      isExists: await exists(fullPath),\n    };\n  }\n\n  /**\n   * 주어진 엔티티와 열거형에 대해, 생성된 코드가 존재하는지 확인합니다.\n   * @param entityId 엔티티 ID\n   * @param enums 열거형 레이블\n   * @returns 생성된 코드가 존재하는지 여부\n   */\n  async checkExists(\n    entityId: string,\n    enums: {\n      [name: string]: z.ZodEnum;\n    },\n  ): Promise<Record<`${TemplateKey}${string}`, boolean>> {\n    const keys: TemplateKey[] = TemplateKey.options;\n    const names = EntityManager.getNamesFromId(entityId);\n    const enumsKeys = Object.keys(enums).filter((name) => name !== names.constant);\n\n    return await reduceAsync(\n      keys,\n      async (result, key) => {\n        const tpl = TemplateManager.get(key);\n        if (key.startsWith(\"view_enums\")) {\n          await mapAsync(enumsKeys, async (componentId) => {\n            const { target, path: p } = tpl.getTargetAndPath(names, componentId);\n            result[`${key}__${componentId}`] = await exists(\n              path.join(Sonamu.appRootPath, target, p),\n            );\n          });\n          return result;\n        }\n\n        const { target, path: p } = tpl.getTargetAndPath(names);\n        const { targets } = Sonamu.config.sync;\n        if (target.includes(\":target\")) {\n          await mapAsync(targets, async (t) => {\n            result[`${key}__${t}`] = await exists(\n              path.join(Sonamu.appRootPath, target.replace(\":target\", t), p),\n            );\n          });\n        } else {\n          result[key] = await exists(path.join(Sonamu.appRootPath, target, p));\n        }\n\n        return result;\n      },\n      {} as Record<`${TemplateKey}${string}`, boolean>,\n    );\n  }\n\n  /**\n   * 하위호환용 프록시 메소드입니다.\n   */\n  async createEntity(form: TemplateOptions[\"entity\"]) {\n    return await createEntity(form);\n  }\n\n  /**\n   * 하위호환용 프록시 메소드입니다.\n   */\n  async delEntity(entityId: string): Promise<{ delPaths: string[] }> {\n    return await delEntity(entityId);\n  }\n\n  /**\n   * 하위호환용 프록시 메소드입니다.\n   */\n  async generateTemplate<T extends TemplateKey>(\n    key: T,\n    templateOptions: TemplateOptions[T],\n    _generateOptions?: GenerateOptions,\n  ): Promise<AbsolutePath[]> {\n    return await generateTemplate(key, templateOptions, _generateOptions);\n  }\n\n  /**\n   * 하위호환용 프록시 메소드입니다.\n   */\n  async renderTemplate<T extends keyof TemplateOptions>(\n    key: T,\n    templateOptions: TemplateOptions[T],\n  ): Promise<PathAndCode[]> {\n    return await renderTemplate(key, templateOptions);\n  }\n\n  /**\n   * 하위호환용 프록시 메소드입니다.\n   */\n  async renewChecksums(): Promise<void> {\n    return await renewChecksums();\n  }\n}\n"],"mappings":"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"}
|
|
425
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"syncer.js","names":["SyncerActions.actionCopySharedToTargetsIfNotExists","SyncerActions.actionGenerateSsrEntryServerIfNotExists","syncTriggeringPaths: AbsolutePath[]","SyncerActions.actionGenerateInitialTypes","SyncerActions.actionSyncFilesToTargets","SyncerActions.actionGenerateSchemas","params: {\n      namesRecord: EntityNamesRecord;\n    }[]","SyncerActions.actionGenerateServices","SyncerActions.actionGenerateHttps","SyncerActions.actionGenerateSsrQueries","SyncerActions.actionSyncConfig","SyncerActions.actionSyncSonamuDictionary","keys: TemplateKey[]","p","target"],"sources":["../../src/syncer/syncer.ts"],"sourcesContent":["import assert from \"assert\";\nimport { EventEmitter } from \"events\";\nimport { unlink } from \"fs/promises\";\nimport path from \"path\";\n\nimport { hot } from \"@sonamu-kit/hmr-hook\";\nimport chalk from \"chalk\";\nimport { minimatch } from \"minimatch\";\nimport { group, unique } from \"radashi\";\nimport { type z } from \"zod\";\n\nimport { registeredApis } from \"../api/decorators\";\nimport { Sonamu } from \"../api/sonamu\";\nimport { EntityManager } from \"../entity/entity-manager\";\nimport { type EntityNamesRecord } from \"../entity/entity-manager\";\nimport { Naite } from \"../naite/naite\";\nimport { type WorkflowMetadata } from \"../tasks/decorator\";\nimport { TemplateManager } from \"../template/template-manager\";\nimport { type GenerateOptions, type PathAndCode } from \"../types/types\";\nimport { TemplateKey } from \"../types/types\";\nimport { type TemplateOptions } from \"../types/types\";\nimport { mapAsync, reduceAsync } from \"../utils/async-utils\";\nimport { centerText } from \"../utils/console-util\";\nimport { isTest } from \"../utils/controller\";\nimport { exists } from \"../utils/fs-utils\";\nimport { type AbsolutePath } from \"../utils/path-utils\";\nimport { runWithGracefulShutdown } from \"../utils/process-utils\";\nimport { findChangedFilesUsingChecksums, renewChecksums } from \"./checksum\";\nimport { generateTemplate, renderTemplate } from \"./code-generator\";\nimport { createEntity, delEntity } from \"./entity-operations\";\nimport { getChecksumPatternGroup, getChecksumPatternGroupInAbsolutePath } from \"./file-patterns\";\nimport { type FileType } from \"./file-patterns\";\nimport { loadApis, loadModels, loadTypes, loadWorkflows } from \"./module-loader\";\nimport { type LoadedApis, type LoadedModels, type LoadedTypes } from \"./module-loader\";\nimport * as SyncerActions from \"./syncer-actions\";\n\ntype DiffGroups = {\n  [key in FileType]: AbsolutePath[];\n};\n\nexport class Syncer {\n  apis: LoadedApis = [];\n  types: LoadedTypes = {};\n  models: LoadedModels = {};\n  workflows: Map<string, WorkflowMetadata[]> = new Map();\n  eventEmitter: EventEmitter = new EventEmitter();\n\n  /**\n   * 체크섬이 변경된 부분에 대해 싱크를 진행합니다.\n   * dev 서버가 처음 떴을 때, sonamu sync 할 때 실행됩니다. 이후에는 syncFromWatcher 경로를 타요.\n   * @returns\n   */\n  async sync(): Promise<void> {\n    // 초기 부트스트랩! 얘네들은 idempotent하고 가볍기 때문에 무지성 실행해도 됩니다.\n    // 얘네들은 sonamu.lock에 들어가지도 않고 따라서 HMR 경로를 타지도 않는 친구들입니다.\n    // 그래서 아무 때나 그냥 돌려주면 되는데, syncFromWatcher에서 매번 하는 것은 낭비이니 여기서 한 번만 합니다.\n    await SyncerActions.actionCopySharedToTargetsIfNotExists();\n    await SyncerActions.actionGenerateSsrEntryServerIfNotExists();\n\n    // 바뀐 것이 없으면 그냥 넘어가요.\n    const changedFiles = await findChangedFilesUsingChecksums();\n    if (changedFiles.length === 0) {\n      console.log(chalk.black.bgGreen(centerText(\"All files are synced!\")));\n      return;\n    }\n\n    // 여기서 실제 싱크 동작을 수행합니다.\n    // 다만 싱크 중에 프로세스가 죽으면 꼬여버리기 때문에,\n    // 시그널에도 잠시 버틸 수 있는 환경 속에서 싱크를 실행합니다.\n    await runWithGracefulShutdown(\n      async () => {\n        // 얘가 싱크 작업 수행하는 본체입니다.\n        await this.doSyncActions(changedFiles);\n\n        // 싱크 액션이 끝나면 항상 체크섬을 다시 갱신합니다.\n        await renewChecksums();\n      },\n      { whenThisHappens: \"SIGUSR2\", waitForUpTo: 20000 },\n    );\n  }\n\n  /**\n   * 강제 풀-싱크: lock을 무시하고 처음부터 다시 싱크합니다.\n   *\n   * **사용처**: git post-merge hook, CI, dev 서버의 `f` 핫키.\n   * **실패 안전성**: 도중에 프로세스가 죽어 lock 없는 상태로 남아도 무해 — 다음 sync에서\n   * lock 없으면 자연스럽게 풀-싱크가 트리거되어 새 lock이 작성됨.\n   */\n  async forceSync(): Promise<void> {\n    const lockPath = path.join(Sonamu.apiRootPath, \"sonamu.lock\");\n    if (await exists(lockPath)) {\n      await unlink(lockPath);\n    }\n    await this.sync();\n  }\n\n  /**\n   * Watcher가 batch로 모은 변경 파일들에 대해 한 번의 HMR/sync 사이클을 돕니다.\n   *\n   * HMR은 api/src 안에서 일어나는 모든 파일들에 대해서 수행합니다.\n   * checksumPatternGroup 매칭 여부와 무관하게 api/src 전체 대상입니다.\n   * 가령 api/src/utils/subset-loaders.ts 같은 파일도 변경되면 HMR은 해줍니다.\n   *\n   * Sync는 checksumPatternGroup으로 매칭되는 파일들에 대해서만 수행합니다.\n   * 여기에는 web/src나 app/src 같은 다른 target의 파일이 포함될 수 있습니다.\n   * 이런 non-api 경로의 파일들은 HMR과는 아무 상관이 없으므로, invalidate을 하지 않습니다.\n   *\n   * @param fileEvents - path → event 맵. event는 \"change\" | \"add\".\n   */\n  async hmrAndSync(fileEvents: Map<AbsolutePath, \"change\" | \"add\">): Promise<void> {\n    const hmrActionRequiredEvents = this.extractHmrActionRequiredFileEvents(fileEvents);\n    const syncTriggeringPaths = await this.extractSyncTriggeringFileEventPaths(fileEvents);\n\n    // HMR 영역: 파일 이벤트 중 api의 모듈 그래프에 있는 파일들에 대한 변동은 hmrActionRequiredEvents로 잡힙니다.\n    // 이 친구들은 invalidate 처리해줍니다.\n    // 이 호출은 아래 sync보다 무조건 먼저 일어나야 합니다!\n    // 왜냐하면 sync에서는 변경된 model 코드를 새로 import해서 처리해야 하는 경우도 있기 때문입니다.\n    await this.invalidateDependentsAffectedByFileEvents(hmrActionRequiredEvents);\n\n    // Sync 영역: checksumPatternGroup에 명시된 파일들에 대한 변동은 syncTriggeringPaths로 잡힙니다.\n    // 이 친구들은 적절한 sync 작업으로 대응합니다.\n    if (syncTriggeringPaths.length > 0) {\n      await this.doSyncActions(syncTriggeringPaths);\n    }\n\n    // 싱크 작업이 끝났으면 무지성 로드를 수행합니다.\n    // 싱크를 안 한 경우에도 로드는 해야 해요. 위에서 관련있는 친구들은 다 invalidate 되었거든요!\n    //\n    // 변경된 파일들에 대해서 새롭게 당겨오는(load) 행위는 doSyncActions에서 하지 않습니다.\n    // doSyncActions에서는 파일을 읽고 만드는 싱크 행위만 합니다.\n    // 거기서 딱 변경된 부분에 영향받는 autoload만 선별해서 수행하려면 너무 지저분하고 복잡해집니다.\n    //\n    // 퍼포먼스 영향은 무시해도 좋습니다.\n    // 어차피 hmr-hook에 의해 invalidate된 부분들이 아니라면 캐시 그대로 유지합니다.\n    await this.autoloadTypes();\n    await this.autoloadModels();\n    await this.autoloadApis();\n    await this.autoloadWorkflows();\n    await this.autoloadSsrRoutes();\n\n    this.eventEmitter.emit(\"onHMRCompleted\");\n  }\n\n  private extractHmrActionRequiredFileEvents(\n    fileEvents: Map<AbsolutePath, \"change\" | \"add\">,\n  ): Map<AbsolutePath, \"change\" | \"add\"> {\n    const apiSrc = path.join(Sonamu.apiRootPath, \"src\");\n    const result = new Map<AbsolutePath, \"change\" | \"add\">();\n    for (const [filePath, event] of fileEvents) {\n      if (!filePath.startsWith(apiSrc)) {\n        continue;\n      }\n      result.set(filePath, event);\n    }\n    return result;\n  }\n\n  private async extractSyncTriggeringFileEventPaths(\n    fileEvents: Map<AbsolutePath, \"change\" | \"add\">,\n  ): Promise<AbsolutePath[]> {\n    const checkPatternGroup = getChecksumPatternGroupInAbsolutePath();\n    const syncTriggeringPaths: AbsolutePath[] = [];\n    for (const [diffFilePath] of fileEvents) {\n      const isInCheckPatternGroup = Object.values(checkPatternGroup).some((pattern) =>\n        minimatch(diffFilePath, pattern),\n      );\n      if (!isInCheckPatternGroup) {\n        continue;\n      }\n      syncTriggeringPaths.push(diffFilePath);\n    }\n\n    return syncTriggeringPaths;\n  }\n\n  private async invalidateDependentsAffectedByFileEvents(\n    fileEvents: Map<AbsolutePath, \"change\" | \"add\">,\n  ) {\n    for (const [diffFilePath, event] of fileEvents) {\n      // 변경된 파일과 dependent 파일들을 invalidate 합니다.\n      // 한 번 이상 import된 친구들에 대해서만 실제 작업이 일어납니다.\n      // 그러니 안심하고 invalidate 해도 됩니다.\n      // 테스트 환경에서는 hot.invalidateFile시 초기 에러가 발생하기 때문에 invalidate 하지 않습니다.\n      if (!isTest()) {\n        const invalidatedPaths = (await hot.invalidateFile(diffFilePath, event)) as AbsolutePath[];\n\n        if (invalidatedPaths.length > 0) {\n          console.log(chalk.bold(`🔄 Invalidated:`));\n\n          for (const invalidatedPath of invalidatedPaths) {\n            try {\n              // 만약 model.ts 파일이 변경(invalidate)되었다? 그러면 registeredApis 중에서 이 모델에 해당하는 api들은 지워줘요.\n              // registeredApis는 통으로 다 날려버릴 수 없습니다. registeredApis에 올라오는 친구들은 초기 로드시 또는 HMR시에만 등록되기 때문입니다.\n              // 따라서 model.ts 파일의 변경으로 다음번 새로운 eval이 예상되는 이 시점에서만, 이 모델에서 나온 registeredApis들을 지워줄 수 있습니다.\n              const removedApis = this.removeInvalidatedRegisteredApis(invalidatedPath);\n              if (removedApis.length > 0) {\n                console.log(\n                  chalk.blue(`- ${path.relative(Sonamu.apiRootPath, invalidatedPath)}`),\n                  chalk.gray(`(with ${removedApis.length} APIs)`),\n                );\n              } else {\n                console.log(chalk.blue(`- ${path.relative(Sonamu.apiRootPath, invalidatedPath)}`));\n              }\n            } catch (e) {\n              console.error(e);\n              console.error(\n                chalk.red(`Failed to remove invalidated registered APIs for ${invalidatedPath}`),\n              );\n            }\n          }\n        }\n      }\n\n      // devRunner 활성화 시, 변경된 소스 파일을 Vitest 모듈 그래프에서도 무효화합니다.\n      // Vite의 moduleGraph.invalidateModule()이 importer 방향으로 재귀적 cascade하므로,\n      // 소스 파일 하나만 무효화하면 이를 import하는 테스트 파일도 자동으로 무효화됩니다.\n      if (!isTest() && Sonamu.config.test?.devRunner?.enabled && Sonamu.devVitestManager) {\n        Sonamu.devVitestManager.invalidateFiles([diffFilePath]);\n        console.log(\n          chalk.dim(`Test invalidated: ${path.relative(Sonamu.apiRootPath, diffFilePath)}`),\n        );\n      }\n    }\n  }\n\n  removeInvalidatedRegisteredApis(\n    invalidatedPath: AbsolutePath,\n  ): (typeof registeredApis)[number][] {\n    if (!invalidatedPath.endsWith(\".model.ts\" /*소스 코드를 다루는 상황이니 .ts 경로로 봅니다.*/)) {\n      return [];\n    }\n\n    const entityId = EntityManager.getEntityIdFromPath(invalidatedPath);\n    const toRemove = registeredApis.filter((api) => api.modelName === `${entityId}Model`);\n    for (const api of toRemove) {\n      const idx = registeredApis.indexOf(api);\n      if (idx !== -1) {\n        registeredApis.splice(idx, 1);\n      }\n    }\n\n    return toRemove;\n  }\n\n  async autoloadTypes() {\n    this.types = await loadTypes();\n  }\n\n  async autoloadModels() {\n    this.models = await loadModels();\n  }\n\n  async autoloadApis() {\n    this.apis = await loadApis();\n  }\n\n  async autoloadWorkflows() {\n    this.workflows = await loadWorkflows();\n    await Sonamu.workflows.synchronize(this.workflows);\n  }\n\n  async autoloadSsrRoutes(): Promise<void> {\n    const ssrConfigPath = path.join(Sonamu.apiRootPath, \"src/ssr\");\n\n    // 기존 routes 초기화\n    const { clearSSRRoutes } = await import(\"../ssr\");\n    clearSSRRoutes();\n\n    // ssr 폴더 없으면 스킵\n    if (!(await exists(ssrConfigPath))) {\n      return;\n    }\n\n    // ssr 폴더 안의 모든 .ts 파일 로드\n    const { globAsync } = await import(\"../utils/async-utils\");\n    const { importMembers } = await import(\"../utils/esm-utils\");\n    const { runtimePath } = await import(\"../utils/path-utils\");\n\n    // runtimePath를 사용하여 개발/프로덕션 환경에 맞는 확장자 처리\n    const files = await globAsync(path.join(ssrConfigPath, runtimePath(\"**/*.ts\")));\n\n    for (const file of files) {\n      try {\n        // importMembers를 사용하면 파일의 side effect(registerSSR 호출)가 실행됨\n        await importMembers(file);\n      } catch (e) {\n        console.error(`Failed to load SSR route: ${file}`, e);\n      }\n    }\n  }\n\n  /**\n   * 실제 싱크를 수행하는 본체입니다.\n   * 변경된 파일들을 타입별로 분류하고 각 타입에 맞는 액션을 실행합니다.\n   *\n   * @param diffFilePaths - 변경된 파일들의 절대 경로 목록\n   * @returns diffTypes - 변경된 파일의 타입 목록 (entity, types, model 등)\n   */\n  async doSyncActions(diffFilePaths: AbsolutePath[]): Promise<{ diffTypes: FileType[] }> {\n    const diffGroups = this.calculateDiffGroups(diffFilePaths);\n    const diffTypes = Object.keys(diffGroups) as FileType[];\n\n    // 여기는 별로 중요한 파트는 아닙니다.\n    // 아래의 if 전개를 깔끔하게 하려고 만든 DSL 같은 거라서, 무시하셔도 됩니다.\n    const { changeMatches, nothingMatches, unhandledPaths } = this.changeMatcher(\n      diffTypes,\n      diffGroups,\n    );\n\n    if (changeMatches(\"entity\", \"types\")) {\n      await this.handleTruthSourceChanges(diffGroups);\n    }\n\n    if (changeMatches(\"model\", \"frame\")) {\n      await this.handleImplementationChanges(diffGroups);\n    }\n\n    if (changeMatches(\"types\", \"functions\")) {\n      await this.handleAuxiliarySymbolChanges(diffGroups);\n    }\n\n    if (changeMatches(\"config\")) {\n      await this.handleConfigChanges(diffGroups);\n    }\n\n    if (changeMatches(\"i18n\", \"entity\" /*레이블*/, \"config\" /*defaultLocale등*/)) {\n      await this.handleSonamuDictionaryRelatedChanges(diffGroups);\n    }\n\n    if (nothingMatches()) {\n      // 파일 변경은 감지되었으나 저 위 어느 changeMatches에도 걸리지 않은 파일들이 drifts입니다.\n      // syncer는 소스의 변경에는 반응하지만 산출물의 변경(drift)에는 직접적으로 반응하지 않습니다.\n      // 대신 이 drift에 대해 경고 정도만 출력해줍니다.\n      await this.handleDrifts(unhandledPaths());\n    }\n\n    return {\n      diffTypes,\n    };\n  }\n\n  calculateDiffGroups(diffFiles: AbsolutePath[]): DiffGroups {\n    const patternGroup = getChecksumPatternGroup();\n    const fileTypes = Object.keys(patternGroup) as FileType[];\n\n    return group(diffFiles, (filePath) => {\n      // 절대 경로 → appRoot 기준 상대 경로 (예: \"api/src/...\", \"web/src/...\")\n      const relativePath = path.relative(Sonamu.appRootPath, filePath);\n      if (relativePath.startsWith(\"..\")) return \"unknown\";\n\n      for (const fileType of fileTypes) {\n        if (minimatch(relativePath, patternGroup[fileType])) {\n          return fileType;\n        }\n      }\n      return \"unknown\";\n    }) as unknown as DiffGroups;\n  }\n\n  private changeMatcher(diffTypes: FileType[], diffGroups: DiffGroups) {\n    const handled = new Set<FileType>();\n\n    /**\n     * 변경 사항이 인자로 받은 FileType들 중 하나 이상을 포함하는지 확인합니다.\n     * 가령 [\"entity\"]가 변경된 호출에서 changeMatches(\"entity\")는 trye를 반환하며,\n     * [\"types\", \"i18n\"]이 변경된 호출에서 changeMatches(\"types\", \"functions\")도 true를 반환하지만,\n     * [\"functions\"]가 변경된 호출에서 changeMatches(\"frame\")은 false를 반환합니다.\n     * @param types\n     */\n    const changeMatches = (...types: FileType[]) => {\n      const matching = types.filter((t) => diffTypes.includes(t));\n      matching.forEach((t) => handled.add(t));\n      return matching.length > 0;\n    };\n\n    /**\n     * changeMatches로 매칭된 것이 하나도 없는지 여부를 가져옵니다.\n     */\n    const nothingMatches = () => handled.size === 0;\n\n    /**\n     * 어떤 changeMatches 호출에도 걸리지 않은 FileType들의 실제 파일 경로를 모아서 반환합니다.\n     */\n    const unhandledPaths = (): AbsolutePath[] =>\n      diffTypes.filter((t) => !handled.has(t)).flatMap((t) => diffGroups[t] ?? []);\n\n    return { changeMatches, nothingMatches, unhandledPaths };\n  }\n\n  async handleTruthSourceChanges(diffGroups: DiffGroups): Promise<void> {\n    Naite.t(\"handleTruthSourceChanges\", { diffGroups });\n\n    await EntityManager.reload();\n\n    // types 생성(entity 새로 추가된 경우)\n    // parentId가 없고, types가 없는 경우에만 생성\n    const entityPath = diffGroups.entity?.at(0);\n    if (entityPath !== undefined) {\n      const entityId = EntityManager.getEntityIdFromPath(entityPath);\n      const entity = EntityManager.get(entityId);\n\n      // 프로젝트에 생성되어야 하는 .ts 파일의 경로입니다.\n      const typeFilePath = path.join(\n        Sonamu.apiRootPath,\n        `src/application/${entity.names.fs}/${entity.names.fs}.types.ts`,\n      ) as AbsolutePath;\n\n      if (entity.parentId === undefined && !(await exists(typeFilePath))) {\n        // *.types.ts가 만들어집니다.\n        const types = await SyncerActions.actionGenerateInitialTypes(entityId);\n\n        // 그걸 타겟에 갖다둬요.\n        await SyncerActions.actionSyncFilesToTargets(types);\n      }\n    }\n\n    // sonamu.generated.ts와 sonamu.generated.sso.ts가 만들어집니다.\n    const generated = await SyncerActions.actionGenerateSchemas();\n\n    // 모든 것들을 target에 보내지는 않습니다.\n    // sonamu.generated.sso.ts는 service-side-only니까 배제합니다.\n    // TODO(병준): 이 하드코드를 누가 해결 좀 해주세요. 일단은 감당 가능하니 놔두겠습니다...\n    const distributable = generated.filter((p) => !p.endsWith(\".sso.ts\"));\n\n    // 이제 보낼 것들만 target에 보내요.\n    await SyncerActions.actionSyncFilesToTargets(distributable);\n  }\n\n  async handleImplementationChanges(diffGroups: DiffGroups): Promise<void> {\n    Naite.t(\"handleImplementationChanges\", { diffGroups });\n    const mergedGroup = [...(diffGroups.model ?? []), ...(diffGroups.frame ?? [])];\n\n    // generated_http.template.ts에서 syncer.types를 씁니다.\n    // service.template.ts에서 syncer.apis를 씁니다.\n    await this.autoloadModels();\n    await this.autoloadTypes();\n    await this.autoloadApis();\n\n    const params: {\n      namesRecord: EntityNamesRecord;\n    }[] = mergedGroup.map((modelPath) => {\n      if (modelPath.endsWith(\".model.ts\") || modelPath.endsWith(\".frame.ts\")) {\n        const entityId = EntityManager.getEntityIdFromPath(modelPath);\n        assert(entityId);\n        return {\n          namesRecord: EntityManager.getNamesFromId(entityId),\n        };\n      }\n      throw new Error(\"not reachable\");\n    });\n\n    // services.generated.ts를 target들에, sonamu.generated.http를 api에 만들어줘요.\n    await SyncerActions.actionGenerateServices(params);\n    await SyncerActions.actionGenerateHttps();\n\n    // queries.generated.ts가 만들어집니다.\n    const queries = await SyncerActions.actionGenerateSsrQueries();\n\n    // queries는 SSR 디스크립터(sonamu/ssr 의존)이므로 web target에만 분배합니다.\n    // 다른 target(app 등)이 SSR을 안 쓰면 의미 없는 dead code일 뿐이라 보내지 않습니다.\n    // TODO(병준): web에만 가게 하기 위해서 target prefix를 지정해주었습니다. 나중에 target 시스템이 개선된다면 바꿔주세요.\n    await SyncerActions.actionSyncFilesToTargets(queries, [\"web\"]);\n  }\n\n  async handleAuxiliarySymbolChanges(diffGroups: DiffGroups): Promise<void> {\n    Naite.t(\"handleAuxiliarySymbolChanges\", { diffGroups });\n    const tsPaths = unique([...(diffGroups.types ?? []), ...(diffGroups.functions ?? [])]);\n    await SyncerActions.actionSyncFilesToTargets(tsPaths);\n  }\n\n  async handleConfigChanges(_: DiffGroups): Promise<void> {\n    await SyncerActions.actionSyncConfig();\n  }\n\n  async handleSonamuDictionaryRelatedChanges(_: DiffGroups): Promise<void> {\n    await SyncerActions.actionSyncSonamuDictionary();\n  }\n\n  async handleDrifts(drifts: AbsolutePath[]): Promise<void> {\n    if (drifts.length > 0) {\n      console.warn(\n        chalk.yellow(\n          \"⚠️ Sonamu가 자동 생성한 파일에 대한 변경이 감지되었습니다. 파일이 Sonamu watcher 외부에서 변경된 것으로 추정됩니다.\",\n        ),\n      );\n      for (const p of drifts) {\n        console.warn(chalk.yellow(`  - ${path.relative(Sonamu.appRootPath, p)}`));\n      }\n      console.warn(chalk.dim(\"  → `pnpm sonamu sync --force`를 권장합니다.\"));\n    }\n  }\n\n  /**\n   * 주어진 엔티티와 템플릿 키에 대해, 생성된 코드가 존재하는지 확인합니다.\n   * @param entityId 엔티티 ID\n   * @param templateKey 템플릿 키\n   * @param enumId 열거형 ID\n   * @returns 생성된 코드가 존재하는지 여부\n   */\n  async checkExistsGenCode(\n    entityId: string,\n    templateKey: TemplateKey,\n    enumId?: string,\n  ): Promise<{ subPath: string; fullPath: string; isExists: boolean }> {\n    const { target, path: genPath } = TemplateManager.get(templateKey).getTargetAndPath(\n      EntityManager.getNamesFromId(entityId),\n      enumId,\n    );\n\n    const subPath = path.join(target, genPath);\n    const fullPath = path.join(Sonamu.appRootPath, subPath);\n    return {\n      subPath,\n      fullPath,\n      isExists: await exists(fullPath),\n    };\n  }\n\n  /**\n   * 주어진 엔티티와 열거형에 대해, 생성된 코드가 존재하는지 확인합니다.\n   * @param entityId 엔티티 ID\n   * @param enums 열거형 레이블\n   * @returns 생성된 코드가 존재하는지 여부\n   */\n  async checkExists(\n    entityId: string,\n    enums: {\n      [name: string]: z.ZodEnum;\n    },\n  ): Promise<Record<`${TemplateKey}${string}`, boolean>> {\n    const keys: TemplateKey[] = TemplateKey.options;\n    const names = EntityManager.getNamesFromId(entityId);\n    const enumsKeys = Object.keys(enums).filter((name) => name !== names.constant);\n\n    return await reduceAsync(\n      keys,\n      async (result, key) => {\n        const tpl = TemplateManager.get(key);\n        if (key.startsWith(\"view_enums\")) {\n          await mapAsync(enumsKeys, async (componentId) => {\n            const { target, path: p } = tpl.getTargetAndPath(names, componentId);\n            result[`${key}__${componentId}`] = await exists(\n              path.join(Sonamu.appRootPath, target, p),\n            );\n          });\n          return result;\n        }\n\n        const { target, path: p } = tpl.getTargetAndPath(names);\n        const { targets } = Sonamu.config.sync;\n        if (target.includes(\":target\")) {\n          await mapAsync(targets, async (t) => {\n            result[`${key}__${t}`] = await exists(\n              path.join(Sonamu.appRootPath, target.replace(\":target\", t), p),\n            );\n          });\n        } else {\n          result[key] = await exists(path.join(Sonamu.appRootPath, target, p));\n        }\n\n        return result;\n      },\n      {} as Record<`${TemplateKey}${string}`, boolean>,\n    );\n  }\n\n  /**\n   * 하위호환용 프록시 메소드입니다.\n   */\n  async createEntity(form: TemplateOptions[\"entity\"]) {\n    return await createEntity(form);\n  }\n\n  /**\n   * 하위호환용 프록시 메소드입니다.\n   */\n  async delEntity(entityId: string): Promise<{ delPaths: string[] }> {\n    return await delEntity(entityId);\n  }\n\n  /**\n   * 하위호환용 프록시 메소드입니다.\n   */\n  async generateTemplate<T extends TemplateKey>(\n    key: T,\n    templateOptions: TemplateOptions[T],\n    _generateOptions?: GenerateOptions,\n  ): Promise<AbsolutePath[]> {\n    return await generateTemplate(key, templateOptions, _generateOptions);\n  }\n\n  /**\n   * 하위호환용 프록시 메소드입니다.\n   */\n  async renderTemplate<T extends keyof TemplateOptions>(\n    key: T,\n    templateOptions: TemplateOptions[T],\n  ): Promise<PathAndCode[]> {\n    return await renderTemplate(key, templateOptions);\n  }\n\n  /**\n   * 하위호환용 프록시 메소드입니다.\n   */\n  async renewChecksums(): Promise<void> {\n    return await renewChecksums();\n  }\n}\n"],"mappings":"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"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"formatter.d.ts","sourceRoot":"","sources":["../../src/utils/formatter.ts"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"formatter.d.ts","sourceRoot":"","sources":["../../src/utils/formatter.ts"],"names":[],"mappings":"AAeA;;;;;GAKG;AACH,eAAO,MAAM,UAAU,qDAGrB,CAAC"}
|
package/dist/utils/formatter.js
CHANGED
|
@@ -5,7 +5,6 @@ import { execute, init_process_utils } from "./process-utils.js";
|
|
|
5
5
|
import { createHash } from "crypto";
|
|
6
6
|
import { readFile, unlink, writeFile } from "fs/promises";
|
|
7
7
|
import path, { dirname, join } from "path";
|
|
8
|
-
import { tmpdir } from "os";
|
|
9
8
|
import { createRequire } from "module";
|
|
10
9
|
import { format } from "oxfmt";
|
|
11
10
|
|
|
@@ -14,7 +13,7 @@ import { format } from "oxfmt";
|
|
|
14
13
|
* 캐시 없는 포맷함수 엔트리.
|
|
15
14
|
*/
|
|
16
15
|
async function formatCodeInternal(code, filePath) {
|
|
17
|
-
if (filePath.endsWith("json")) {
|
|
16
|
+
if (filePath.endsWith(".json")) {
|
|
18
17
|
return runOxfmt(code, filePath);
|
|
19
18
|
}
|
|
20
19
|
return runOxfmt(await runOxlint(code), filePath);
|
|
@@ -79,7 +78,7 @@ async function runOxlint(code) {
|
|
|
79
78
|
if (isTest()) {
|
|
80
79
|
return code;
|
|
81
80
|
}
|
|
82
|
-
const tmpFile = join(
|
|
81
|
+
const tmpFile = join(process.cwd(), `.sonamu-fmt-${Date.now()}-${Math.random().toString(36).slice(2)}.ts`);
|
|
83
82
|
try {
|
|
84
83
|
await writeFile(tmpFile, code, "utf-8");
|
|
85
84
|
try {
|
|
@@ -115,7 +114,7 @@ var init_formatter = __esmMin((() => {
|
|
|
115
114
|
init_process_utils();
|
|
116
115
|
_require = createRequire(import.meta.url);
|
|
117
116
|
formatCode = cached(formatCodeInternal, (code, filePath) => {
|
|
118
|
-
const ext = filePath.endsWith(".tsx") ? "tsx" : filePath.endsWith("json") ? "json" : "ts";
|
|
117
|
+
const ext = filePath.endsWith(".tsx") ? "tsx" : filePath.endsWith(".json") ? "json" : "ts";
|
|
119
118
|
return `${ext}:${createHash("sha1").update(code).digest("hex")}`;
|
|
120
119
|
});
|
|
121
120
|
cachedOxfmtConfig = null;
|
|
@@ -124,4 +123,4 @@ var init_formatter = __esmMin((() => {
|
|
|
124
123
|
//#endregion
|
|
125
124
|
init_formatter();
|
|
126
125
|
export { formatCode, init_formatter };
|
|
127
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZm9ybWF0dGVyLmpzIiwibmFtZXMiOlsiY2FjaGVkT3hmbXRDb25maWc6IEZvcm1hdENvbmZpZyB8IG51bGwiXSwic291cmNlcyI6WyIuLi8uLi9zcmMvdXRpbHMvZm9ybWF0dGVyLnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGNyZWF0ZUhhc2ggfSBmcm9tIFwiY3J5cHRvXCI7XG4vLyDthYzsiqTtirgg7ZmY6rK97JeQ7ISc64qUIGZzL3Byb21pc2Vz6rCAIG1vY2vrkJjsp4Drp4wsIOyVhOuemCBydW5PeGxpbnTsnbQgaXNUZXN0IOqwgOuTnOuhnCDslYgg64+
|
|
126
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"formatter.js","names":["cachedOxfmtConfig: FormatConfig | null"],"sources":["../../src/utils/formatter.ts"],"sourcesContent":["import { createHash } from \"crypto\";\n// 테스트 환경에서는 fs/promises가 mock되지만, 아래 runOxlint이 isTest 가드로 안 도니까\n// 그냥 fs/promises 그대로 사용. (production에서만 임시파일 흐름이 돕니다.)\nimport { readFile, unlink, writeFile } from \"fs/promises\";\nimport { createRequire } from \"module\";\nimport path, { dirname, join } from \"path\";\n\nimport { format, type FormatConfig } from \"oxfmt\";\n\nimport { cached } from \"./async-utils\";\nimport { isTest } from \"./controller\";\nimport { execute } from \"./process-utils\";\n\nconst _require = createRequire(import.meta.url);\n\n/**\n * 코드를 프로젝트의 oxfmt + oxlint 설정에 맞춰 포매팅한 문자열을 반환합니다.\n *\n * 캐싱도 있어요 ㅎㅎ 똑같은 입력에 대해서 캐시 커버됩니다.\n * 수명은 프로세스 죽을때까지 ㅋ\n */\nexport const formatCode = cached(formatCodeInternal, (code, filePath) => {\n  const ext = filePath.endsWith(\".tsx\") ? \"tsx\" : filePath.endsWith(\".json\") ? \"json\" : \"ts\";\n  return `${ext}:${createHash(\"sha1\").update(code).digest(\"hex\")}`;\n});\n\n/**\n * 캐시 없는 포맷함수 엔트리.\n */\nasync function formatCodeInternal(code: string, filePath: string): Promise<string> {\n  // json은 포맷만 하면 됩니다.\n  if (filePath.endsWith(\".json\")) {\n    return runOxfmt(code, filePath);\n  }\n\n  // 린트 먼저 한 다음에 포맷으로 마무리해요.\n  return runOxfmt(await runOxlint(code), filePath);\n}\n\n/**\n * 프로젝트 설정을 찾아서 이에 맞춰서 코드를 포맷합니다.\n */\nasync function runOxfmt(code: string, filePath: string): Promise<string> {\n  const result = await format(path.basename(filePath), code, await loadOxfmtConfig());\n\n  const errors = result.errors.filter((e) => e.severity === \"Error\");\n  if (errors.length > 0) {\n    if (!isTest()) {\n      console.error(`oxfmt errors (${filePath}):`);\n      for (const err of errors) {\n        const label = err.labels[0];\n        if (label) {\n          const before = code.slice(Math.max(0, label.start - 80), label.start);\n          const at = code.slice(label.start, label.end);\n          const after = code.slice(label.end, Math.min(code.length, label.end + 80));\n          console.error(`  - ${err.message} (offset ${label.start}-${label.end})`);\n          console.error(`    around: ...${before}»${at}«${after}...`);\n        } else {\n          console.error(`  - ${err.message}`);\n        }\n      }\n    }\n    return code;\n  }\n  return result.code;\n}\n\nlet cachedOxfmtConfig: FormatConfig | null = null;\nasync function loadOxfmtConfig(): Promise<FormatConfig> {\n  if (cachedOxfmtConfig !== null) {\n    return cachedOxfmtConfig;\n  }\n\n  let dir = process.cwd();\n  while (true) {\n    const candidate = join(dir, \".oxfmtrc.json\");\n    try {\n      cachedOxfmtConfig = JSON.parse(await readFile(candidate, \"utf-8\")) as FormatConfig;\n      return cachedOxfmtConfig;\n    } catch (e) {\n      if ((e as NodeJS.ErrnoException).code !== \"ENOENT\") {\n        !isTest() && console.error(`Failed to load ${candidate}:`, e);\n        break;\n      }\n    }\n    const parent = dirname(dir);\n    if (parent === dir) break;\n    dir = parent;\n  }\n\n  cachedOxfmtConfig = {};\n  return cachedOxfmtConfig;\n}\n\n/**\n * 프로젝트 설정에 맞춰 코드를 lint합니다.\n *\n * 프로젝트 설정을 적용받는 oxlint cli를 찾아 띄워서,\n * 임시 파일에 in-place로 써서 그 결과를 빼오는 방식으로 작동합니다.\n * 왜 이렇게 하느냐? oxlint가 node api도 안 주고 cli에서 stdin 옵션도 안 주기 때문...\n */\nasync function runOxlint(code: string): Promise<string> {\n  if (isTest()) {\n    // 테스트 환경에서는 느려지기만 하고 검증할 가치도 없어서 안 합니다.\n    // GitHub Actions 환경에서 lint가 오래 걸려서 뻗기도 했어요. (https://github.com/cartanova-ai/sonamu/actions/runs/25267214027/job/74083630169)\n    return code;\n  }\n\n  const tmpFile = join(\n    // 타겟 파일이 루트 아래에 있어야 해요. 그래서 tmp 디렉토리같은거 안 씁니다!\n    process.cwd(),\n    `.sonamu-fmt-${Date.now()}-${Math.random().toString(36).slice(2)}.ts`,\n  );\n\n  try {\n    await writeFile(tmpFile, code, \"utf-8\");\n\n    try {\n      await execute(resolveOxlintBin(), [\"--fix\", \"--fix-suggestions\", \"--type-aware\", tmpFile], {\n        timeout: 10000,\n      });\n    } catch (e) {\n      // lint 위반 시 exit code != 0이지만 --fix는 적용됨. exec 자체 실패만 throw.\n      if (typeof (e as Error & { code?: number }).code !== \"number\") {\n        throw e;\n      }\n    }\n\n    return await readFile(tmpFile, \"utf-8\");\n  } finally {\n    try {\n      await unlink(tmpFile);\n    } catch {\n      // 삭제 실패해도 어차피 ignore됨.\n    }\n  }\n}\n\nfunction resolveOxlintBin(): string {\n  try {\n    return _require.resolve(\"oxlint/bin/oxlint\");\n  } catch {\n    return \"oxlint\";\n  }\n}\n"],"mappings":";;;;;;;;;;;;;;AA6BA,eAAe,mBAAmB,MAAc,UAAmC;AAEjF,KAAI,SAAS,SAAS,QAAQ,EAAE;AAC9B,SAAO,SAAS,MAAM,SAAS;;AAIjC,QAAO,SAAS,MAAM,UAAU,KAAK,EAAE,SAAS;;;;;AAMlD,eAAe,SAAS,MAAc,UAAmC;CACvE,MAAM,SAAS,MAAM,OAAO,KAAK,SAAS,SAAS,EAAE,MAAM,MAAM,iBAAiB,CAAC;CAEnF,MAAM,SAAS,OAAO,OAAO,QAAQ,MAAM,EAAE,aAAa,QAAQ;AAClE,KAAI,OAAO,SAAS,GAAG;AACrB,MAAI,CAAC,QAAQ,EAAE;AACb,WAAQ,MAAM,iBAAiB,SAAS,IAAI;AAC5C,QAAK,MAAM,OAAO,QAAQ;IACxB,MAAM,QAAQ,IAAI,OAAO;AACzB,QAAI,OAAO;KACT,MAAM,SAAS,KAAK,MAAM,KAAK,IAAI,GAAG,MAAM,QAAQ,GAAG,EAAE,MAAM,MAAM;KACrE,MAAM,KAAK,KAAK,MAAM,MAAM,OAAO,MAAM,IAAI;KAC7C,MAAM,QAAQ,KAAK,MAAM,MAAM,KAAK,KAAK,IAAI,KAAK,QAAQ,MAAM,MAAM,GAAG,CAAC;AAC1E,aAAQ,MAAM,OAAO,IAAI,QAAQ,WAAW,MAAM,MAAM,GAAG,MAAM,IAAI,GAAG;AACxE,aAAQ,MAAM,kBAAkB,OAAO,GAAG,GAAG,GAAG,MAAM,KAAK;WACtD;AACL,aAAQ,MAAM,OAAO,IAAI,UAAU;;;;AAIzC,SAAO;;AAET,QAAO,OAAO;;AAIhB,eAAe,kBAAyC;AACtD,KAAI,sBAAsB,MAAM;AAC9B,SAAO;;CAGT,IAAI,MAAM,QAAQ,KAAK;AACvB,QAAO,MAAM;EACX,MAAM,YAAY,KAAK,KAAK,gBAAgB;AAC5C,MAAI;AACF,uBAAoB,KAAK,MAAM,MAAM,SAAS,WAAW,QAAQ,CAAC;AAClE,UAAO;WACA,GAAG;AACV,OAAK,EAA4B,SAAS,UAAU;AAClD,KAAC,QAAQ,IAAI,QAAQ,MAAM,kBAAkB,UAAU,IAAI,EAAE;AAC7D;;;EAGJ,MAAM,SAAS,QAAQ,IAAI;AAC3B,MAAI,WAAW,IAAK;AACpB,QAAM;;AAGR,qBAAoB,EAAE;AACtB,QAAO;;;;;;;;;AAUT,eAAe,UAAU,MAA+B;AACtD,KAAI,QAAQ,EAAE;AAGZ,SAAO;;CAGT,MAAM,UAAU,KAEd,QAAQ,KAAK,EACb,eAAe,KAAK,KAAK,CAAC,GAAG,KAAK,QAAQ,CAAC,SAAS,GAAG,CAAC,MAAM,EAAE,CAAC,KAClE;AAED,KAAI;AACF,QAAM,UAAU,SAAS,MAAM,QAAQ;AAEvC,MAAI;AACF,SAAM,QAAQ,kBAAkB,EAAE;IAAC;IAAS;IAAqB;IAAgB;IAAQ,EAAE,EACzF,SAAS,KACV,CAAC;WACK,GAAG;AAEV,OAAI,OAAQ,EAAgC,SAAS,UAAU;AAC7D,UAAM;;;AAIV,SAAO,MAAM,SAAS,SAAS,QAAQ;WAC/B;AACR,MAAI;AACF,SAAM,OAAO,QAAQ;UACf;;;AAMZ,SAAS,mBAA2B;AAClC,KAAI;AACF,SAAO,SAAS,QAAQ,oBAAoB;SACtC;AACN,SAAO;;;;;mBArI4B;kBACD;qBACI;CAEpC,WAAW,cAAc,OAAO,KAAK,IAAI;CAQlC,aAAa,OAAO,qBAAqB,MAAM,aAAa;EACvE,MAAM,MAAM,SAAS,SAAS,OAAO,GAAG,QAAQ,SAAS,SAAS,QAAQ,GAAG,SAAS;AACtF,SAAO,GAAG,IAAI,GAAG,WAAW,OAAO,CAAC,OAAO,KAAK,CAAC,OAAO,MAAM;GAC9D;CA2CEA,oBAAyC"}
|
|
@@ -1,16 +1,16 @@
|
|
|
1
1
|
import { type ExecFileOptions } from "child_process";
|
|
2
2
|
/**
|
|
3
|
-
*
|
|
3
|
+
* child_process.execFile의 Promise 기반 wrapper입니다.
|
|
4
4
|
* exit code가 non-zero이면 reject해요.
|
|
5
5
|
*/
|
|
6
6
|
export declare function execute(bin: string, args: string[], options?: ExecFileOptions): Promise<string>;
|
|
7
7
|
/**
|
|
8
8
|
* 주어진 작업을 실행합니다.
|
|
9
|
-
* 주어진 프로세스 이벤트(=시그널)가 발생하였을 때에도 최대 한계(
|
|
9
|
+
* 주어진 프로세스 이벤트(=시그널)가 발생하였을 때에도 최대 한계(waitForUpTo) 동안 작업을 기다린 후 종료합니다.
|
|
10
10
|
* @param {() => Promise<void>} job - 실행할 작업
|
|
11
|
-
* @param {
|
|
12
|
-
* @param {NodeJS.Signals} options.
|
|
13
|
-
* @param {number} options.
|
|
11
|
+
* @param {{ whenThisHappens: NodeJS.Signals; waitForUpTo: number }} options - 옵션
|
|
12
|
+
* @param {NodeJS.Signals} options.whenThisHappens - 프로세스 이벤트
|
|
13
|
+
* @param {number} options.waitForUpTo - 최대 한계 시간
|
|
14
14
|
*/
|
|
15
15
|
export declare function runWithGracefulShutdown(job: () => Promise<void>, { whenThisHappens: event, waitForUpTo: delayBeforeShutdown, }?: {
|
|
16
16
|
whenThisHappens: NodeJS.Signals;
|
|
@@ -6,7 +6,7 @@ import { promisify } from "util";
|
|
|
6
6
|
|
|
7
7
|
//#region src/utils/process-utils.ts
|
|
8
8
|
/**
|
|
9
|
-
*
|
|
9
|
+
* child_process.execFile의 Promise 기반 wrapper입니다.
|
|
10
10
|
* exit code가 non-zero이면 reject해요.
|
|
11
11
|
*/
|
|
12
12
|
async function execute(bin, args, options) {
|
|
@@ -15,11 +15,11 @@ async function execute(bin, args, options) {
|
|
|
15
15
|
}
|
|
16
16
|
/**
|
|
17
17
|
* 주어진 작업을 실행합니다.
|
|
18
|
-
* 주어진 프로세스 이벤트(=시그널)가 발생하였을 때에도 최대 한계(
|
|
18
|
+
* 주어진 프로세스 이벤트(=시그널)가 발생하였을 때에도 최대 한계(waitForUpTo) 동안 작업을 기다린 후 종료합니다.
|
|
19
19
|
* @param {() => Promise<void>} job - 실행할 작업
|
|
20
|
-
* @param {
|
|
21
|
-
* @param {NodeJS.Signals} options.
|
|
22
|
-
* @param {number} options.
|
|
20
|
+
* @param {{ whenThisHappens: NodeJS.Signals; waitForUpTo: number }} options - 옵션
|
|
21
|
+
* @param {NodeJS.Signals} options.whenThisHappens - 프로세스 이벤트
|
|
22
|
+
* @param {number} options.waitForUpTo - 최대 한계 시간
|
|
23
23
|
*/
|
|
24
24
|
async function runWithGracefulShutdown(job, { whenThisHappens: event, waitForUpTo: delayBeforeShutdown } = {
|
|
25
25
|
whenThisHappens: "SIGUSR2",
|
|
@@ -52,4 +52,4 @@ var init_process_utils = __esmMin((() => {
|
|
|
52
52
|
//#endregion
|
|
53
53
|
init_process_utils();
|
|
54
54
|
export { execute, init_process_utils, runWithGracefulShutdown };
|
|
55
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,
|
|
55
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJvY2Vzcy11dGlscy5qcyIsIm5hbWVzIjpbInNldFRpbWVvdXRQcm9taXNlcyJdLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy91dGlscy9wcm9jZXNzLXV0aWxzLnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGV4ZWNGaWxlLCB0eXBlIEV4ZWNGaWxlT3B0aW9ucyB9IGZyb20gXCJjaGlsZF9wcm9jZXNzXCI7XG5pbXBvcnQgeyBzZXRUaW1lb3V0IGFzIHNldFRpbWVvdXRQcm9taXNlcyB9IGZyb20gXCJ0aW1lcnMvcHJvbWlzZXNcIjtcbmltcG9ydCB7IHByb21pc2lmeSB9IGZyb20gXCJ1dGlsXCI7XG5cbmltcG9ydCBjaGFsayBmcm9tIFwiY2hhbGtcIjtcblxuY29uc3QgZXhlY0ZpbGVBc3luYyA9IHByb21pc2lmeShleGVjRmlsZSk7XG5cbi8qKlxuICogY2hpbGRfcHJvY2Vzcy5leGVjRmlsZeydmCBQcm9taXNlIOq4sOuwmCB3cmFwcGVy7J6F64uI64ukLlxuICogZXhpdCBjb2Rl6rCAIG5vbi16ZXJv7J2066m0IHJlamVjdO2VtOyalC5cbiAqL1xuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIGV4ZWN1dGUoXG4gIGJpbjogc3RyaW5nLFxuICBhcmdzOiBzdHJpbmdbXSxcbiAgb3B0aW9ucz86IEV4ZWNGaWxlT3B0aW9ucyxcbik6IFByb21pc2U8c3RyaW5nPiB7XG4gIGNvbnN0IHsgc3Rkb3V0IH0gPSBhd2FpdCBleGVjRmlsZUFzeW5jKGJpbiwgYXJncywgb3B0aW9ucyk7XG4gIHJldHVybiB0eXBlb2Ygc3Rkb3V0ID09PSBcInN0cmluZ1wiID8gc3Rkb3V0IDogc3Rkb3V0LnRvU3RyaW5nKCk7XG59XG5cbi8qKlxuICog7KO87Ja07KeEIOyekeyXheydhCDsi6Ttlontlanri4jri6QuXG4gKiDso7zslrTsp4Qg7ZSE66Gc7IS47IqkIOydtOuypO2KuCg97Iuc6re464SQKeqwgCDrsJzsg53tlZjsmIDsnYQg65WM7JeQ64+EIOy1nOuMgCDtlZzqs4Qod2FpdEZvclVwVG8pIOuPmeyViCDsnpHsl4XsnYQg6riw64uk66awIO2bhCDsooXro4ztlanri4jri6QuXG4gKiBAcGFyYW0geygpID0+IFByb21pc2U8dm9pZD59IGpvYiAtIOyLpO2Wie2VoCDsnpHsl4VcbiAqIEBwYXJhbSB7eyB3aGVuVGhpc0hhcHBlbnM6IE5vZGVKUy5TaWduYWxzOyB3YWl0Rm9yVXBUbzogbnVtYmVyIH19IG9wdGlvbnMgLSDsmLXshZhcbiAqIEBwYXJhbSB7Tm9kZUpTLlNpZ25hbHN9IG9wdGlvbnMud2hlblRoaXNIYXBwZW5zIC0g7ZSE66Gc7IS47IqkIOydtOuypO2KuFxuICogQHBhcmFtIHtudW1iZXJ9IG9wdGlvbnMud2FpdEZvclVwVG8gLSDstZzrjIAg7ZWc6rOEIOyLnOqwhFxuICovXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gcnVuV2l0aEdyYWNlZnVsU2h1dGRvd24oXG4gIGpvYjogKCkgPT4gUHJvbWlzZTx2b2lkPixcbiAge1xuICAgIHdoZW5UaGlzSGFwcGVuczogZXZlbnQsXG4gICAgd2FpdEZvclVwVG86IGRlbGF5QmVmb3JlU2h1dGRvd24sXG4gIH06IHsgd2hlblRoaXNIYXBwZW5zOiBOb2RlSlMuU2lnbmFsczsgd2FpdEZvclVwVG86IG51bWJlciB9ID0ge1xuICAgIHdoZW5UaGlzSGFwcGVuczogXCJTSUdVU1IyXCIsXG4gICAgd2FpdEZvclVwVG86IDIwMDAwLFxuICB9LFxuKTogUHJvbWlzZTx2b2lkPiB7XG4gIGxldCBpc1J1bm5pbmcgPSB0cnVlIGFzIGJvb2xlYW47XG5cbiAgY29uc3QgYWJvcnRDb250cm9sbGVyID0gbmV3IEFib3J0Q29udHJvbGxlcigpO1xuICBjb25zdCBvbkV2ZW50ID0gYXN5bmMgKCkgPT4ge1xuICAgIGlmICghaXNSdW5uaW5nKSB7XG4gICAgICBwcm9jZXNzLmV4aXQoMCk7XG4gICAgfVxuICAgIGNvbnNvbGUubG9nKGNoYWxrLm1hZ2VudGFCcmlnaHQoYHdhaXQgZm9yIHN5bmNpbmcgZG9uZS4uLi5gKSk7XG5cbiAgICB0cnkge1xuICAgICAgYXdhaXQgc2V0VGltZW91dFByb21pc2VzKGRlbGF5QmVmb3JlU2h1dGRvd24sIFwid2FpdGluZy1zeW5jXCIsIHtcbiAgICAgICAgc2lnbmFsOiBhYm9ydENvbnRyb2xsZXIuc2lnbmFsLFxuICAgICAgfSk7XG4gICAgfSBjYXRjaCB7fVxuICAgIGNvbnNvbGUubG9nKGNoYWxrLm1hZ2VudGFCcmlnaHQoYFN5bmNpbmcgRE9ORSFgKSk7XG4gICAgcHJvY2Vzcy5leGl0KDApO1xuICB9O1xuICBwcm9jZXNzLm9uKGV2ZW50LCBvbkV2ZW50KTtcblxuICBhd2FpdCBqb2IoKTtcblxuICBpc1J1bm5pbmcgPSBmYWxzZTtcbiAgYWJvcnRDb250cm9sbGVyLmFib3J0KCk7XG4gIHByb2Nlc3Mub2ZmKGV2ZW50LCBvbkV2ZW50KTtcbn1cbiJdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7QUFZQSxlQUFzQixRQUNwQixLQUNBLE1BQ0EsU0FDaUI7Q0FDakIsTUFBTSxFQUFFLFdBQVcsTUFBTSxjQUFjLEtBQUssTUFBTSxRQUFRO0FBQzFELFFBQU8sT0FBTyxXQUFXLFdBQVcsU0FBUyxPQUFPLFVBQVU7Ozs7Ozs7Ozs7QUFXaEUsZUFBc0Isd0JBQ3BCLEtBQ0EsRUFDRSxpQkFBaUIsT0FDakIsYUFBYSx3QkFDK0M7Q0FDNUQsaUJBQWlCO0NBQ2pCLGFBQWE7Q0FDZCxFQUNjO0NBQ2YsSUFBSSxZQUFZO0NBRWhCLE1BQU0sa0JBQWtCLElBQUksaUJBQWlCO0NBQzdDLE1BQU0sVUFBVSxZQUFZO0FBQzFCLE1BQUksQ0FBQyxXQUFXO0FBQ2QsV0FBUSxLQUFLLEVBQUU7O0FBRWpCLFVBQVEsSUFBSSxNQUFNLGNBQWMsNEJBQTRCLENBQUM7QUFFN0QsTUFBSTtBQUNGLFNBQU1BLFdBQW1CLHFCQUFxQixnQkFBZ0IsRUFDNUQsUUFBUSxnQkFBZ0IsUUFDekIsQ0FBQztVQUNJO0FBQ1IsVUFBUSxJQUFJLE1BQU0sY0FBYyxnQkFBZ0IsQ0FBQztBQUNqRCxVQUFRLEtBQUssRUFBRTs7QUFFakIsU0FBUSxHQUFHLE9BQU8sUUFBUTtBQUUxQixPQUFNLEtBQUs7QUFFWCxhQUFZO0FBQ1osaUJBQWdCLE9BQU87QUFDdkIsU0FBUSxJQUFJLE9BQU8sUUFBUTs7OztDQXhEdkIsZ0JBQWdCLFVBQVUsU0FBUyJ9
|
package/package.json
CHANGED
package/src/api/decorators.ts
CHANGED
|
@@ -53,7 +53,7 @@ export type ApiDecoratorOptions = {
|
|
|
53
53
|
compress?: CompressConfig;
|
|
54
54
|
};
|
|
55
55
|
export type StreamDecoratorOptions = {
|
|
56
|
-
type: "sse";
|
|
56
|
+
type: "sse";
|
|
57
57
|
// oxlint-disable-next-line @typescript-eslint/no-explicit-any -- 이벤트 키별로 넘겨주는 값이므로 어떤 타입이든 상관없음
|
|
58
58
|
events: z.ZodObject<any>;
|
|
59
59
|
path?: string;
|
|
@@ -386,7 +386,7 @@ export function transactional(options: TransactionalOptions = {}) {
|
|
|
386
386
|
const modelName = target.constructor.name.match(/(.+)Class$/)?.[1];
|
|
387
387
|
assert(
|
|
388
388
|
modelName,
|
|
389
|
-
`modelName is required on @
|
|
389
|
+
`modelName is required on @transactional decorator on ${target.constructor.name}.${propertyKey}`,
|
|
390
390
|
);
|
|
391
391
|
const methodName = propertyKey;
|
|
392
392
|
|
|
@@ -57,7 +57,7 @@ export function getChecksumPatternGroup() {
|
|
|
57
57
|
// sonamu.shared.ts와 entry-server.generated.tsx와 같은
|
|
58
58
|
// sync 초반 1회성 부트스트랩 파일들은 관리 안 하기 때문에 여기에 리스팅도 안 합니다.
|
|
59
59
|
generated: api("src/application/**/*.generated.{ts,tsx,sso.ts}"),
|
|
60
|
-
generatedCopied: targets("src/services/**/{sonamu,queries}.generated.{ts,tsx
|
|
60
|
+
generatedCopied: targets("src/services/**/{sonamu,queries}.generated.{ts,tsx}"),
|
|
61
61
|
httpGenerated: api("src/application/**/*.generated.http"),
|
|
62
62
|
servicesGenerated: targets("src/services/services.generated.ts"),
|
|
63
63
|
sdGenerated: anywhere("src/i18n/**/sd.generated.ts"),
|
|
@@ -117,15 +117,27 @@ export async function actionGenerateSsrEntryServerIfNotExists(): Promise<Absolut
|
|
|
117
117
|
}
|
|
118
118
|
|
|
119
119
|
/**
|
|
120
|
-
* 주어진 .ts 파일들(api에 있다고 가정)을
|
|
120
|
+
* 주어진 .ts 파일들(api에 있다고 가정)을 타겟 디렉토리의 services에 갖다 둡니다.
|
|
121
121
|
* 이때 내부의 sonamu import는 sonamu.shared.ts import로 치환되고,
|
|
122
122
|
* 경로의 /application/은 /services/로 치환됩니다.
|
|
123
123
|
*
|
|
124
|
+
* 기본은 Sonamu.config.sync.targets 전체를 대상으로 하며,
|
|
125
|
+
* onlyTargetsStartingWith 화이트리스트로 일부 target에만 분배할 수 있습니다.
|
|
126
|
+
* 가량 SSR-only인 queries.generated.ts같은 친구들은 "web"으로 시작하는 타겟들에만 보낼 수 있어요.
|
|
127
|
+
* ["web"]으로 넘기면 web, web-admin, webapp 등에 모두 매치됩니다.
|
|
128
|
+
*
|
|
124
129
|
* @param tsPaths 복사할 파일들의 절대 경로
|
|
130
|
+
* @param onlyTargetsStartingWith 분배할 target 접두어들의 화이트리스트. 미지정 시 모든 target.
|
|
125
131
|
* @returns 각 타겟에 복사된 파일들의 절대 경로 배열 (flat).
|
|
126
132
|
*/
|
|
127
|
-
export async function actionSyncFilesToTargets(
|
|
128
|
-
|
|
133
|
+
export async function actionSyncFilesToTargets(
|
|
134
|
+
tsPaths: AbsolutePath[],
|
|
135
|
+
onlyTargetsStartingWith?: string[],
|
|
136
|
+
): Promise<string[]> {
|
|
137
|
+
const allTargets = Sonamu.config.sync.targets;
|
|
138
|
+
const targets = onlyTargetsStartingWith
|
|
139
|
+
? allTargets.filter((t) => onlyTargetsStartingWith.some((prefix) => t.startsWith(prefix)))
|
|
140
|
+
: allTargets;
|
|
129
141
|
const { dir: apiDir } = Sonamu.config.api;
|
|
130
142
|
|
|
131
143
|
return (
|
package/src/syncer/syncer.ts
CHANGED
|
@@ -414,11 +414,16 @@ export class Syncer {
|
|
|
414
414
|
}
|
|
415
415
|
}
|
|
416
416
|
|
|
417
|
-
// sonamu.generated.ts가 만들어집니다.
|
|
417
|
+
// sonamu.generated.ts와 sonamu.generated.sso.ts가 만들어집니다.
|
|
418
418
|
const generated = await SyncerActions.actionGenerateSchemas();
|
|
419
419
|
|
|
420
|
-
//
|
|
421
|
-
|
|
420
|
+
// 모든 것들을 target에 보내지는 않습니다.
|
|
421
|
+
// sonamu.generated.sso.ts는 service-side-only니까 배제합니다.
|
|
422
|
+
// TODO(병준): 이 하드코드를 누가 해결 좀 해주세요. 일단은 감당 가능하니 놔두겠습니다...
|
|
423
|
+
const distributable = generated.filter((p) => !p.endsWith(".sso.ts"));
|
|
424
|
+
|
|
425
|
+
// 이제 보낼 것들만 target에 보내요.
|
|
426
|
+
await SyncerActions.actionSyncFilesToTargets(distributable);
|
|
422
427
|
}
|
|
423
428
|
|
|
424
429
|
async handleImplementationChanges(diffGroups: DiffGroups): Promise<void> {
|
|
@@ -451,8 +456,10 @@ export class Syncer {
|
|
|
451
456
|
// queries.generated.ts가 만들어집니다.
|
|
452
457
|
const queries = await SyncerActions.actionGenerateSsrQueries();
|
|
453
458
|
|
|
454
|
-
//
|
|
455
|
-
|
|
459
|
+
// queries는 SSR 디스크립터(sonamu/ssr 의존)이므로 web target에만 분배합니다.
|
|
460
|
+
// 다른 target(app 등)이 SSR을 안 쓰면 의미 없는 dead code일 뿐이라 보내지 않습니다.
|
|
461
|
+
// TODO(병준): web에만 가게 하기 위해서 target prefix를 지정해주었습니다. 나중에 target 시스템이 개선된다면 바꿔주세요.
|
|
462
|
+
await SyncerActions.actionSyncFilesToTargets(queries, ["web"]);
|
|
456
463
|
}
|
|
457
464
|
|
|
458
465
|
async handleAuxiliarySymbolChanges(diffGroups: DiffGroups): Promise<void> {
|
package/src/utils/formatter.ts
CHANGED
|
@@ -3,7 +3,6 @@ import { createHash } from "crypto";
|
|
|
3
3
|
// 그냥 fs/promises 그대로 사용. (production에서만 임시파일 흐름이 돕니다.)
|
|
4
4
|
import { readFile, unlink, writeFile } from "fs/promises";
|
|
5
5
|
import { createRequire } from "module";
|
|
6
|
-
import { tmpdir } from "os";
|
|
7
6
|
import path, { dirname, join } from "path";
|
|
8
7
|
|
|
9
8
|
import { format, type FormatConfig } from "oxfmt";
|
|
@@ -21,7 +20,7 @@ const _require = createRequire(import.meta.url);
|
|
|
21
20
|
* 수명은 프로세스 죽을때까지 ㅋ
|
|
22
21
|
*/
|
|
23
22
|
export const formatCode = cached(formatCodeInternal, (code, filePath) => {
|
|
24
|
-
const ext = filePath.endsWith(".tsx") ? "tsx" : filePath.endsWith("json") ? "json" : "ts";
|
|
23
|
+
const ext = filePath.endsWith(".tsx") ? "tsx" : filePath.endsWith(".json") ? "json" : "ts";
|
|
25
24
|
return `${ext}:${createHash("sha1").update(code).digest("hex")}`;
|
|
26
25
|
});
|
|
27
26
|
|
|
@@ -30,7 +29,7 @@ export const formatCode = cached(formatCodeInternal, (code, filePath) => {
|
|
|
30
29
|
*/
|
|
31
30
|
async function formatCodeInternal(code: string, filePath: string): Promise<string> {
|
|
32
31
|
// json은 포맷만 하면 됩니다.
|
|
33
|
-
if (filePath.endsWith("json")) {
|
|
32
|
+
if (filePath.endsWith(".json")) {
|
|
34
33
|
return runOxfmt(code, filePath);
|
|
35
34
|
}
|
|
36
35
|
|
|
@@ -107,10 +106,9 @@ async function runOxlint(code: string): Promise<string> {
|
|
|
107
106
|
return code;
|
|
108
107
|
}
|
|
109
108
|
|
|
110
|
-
// OS tmp dir에 두는 이유: process.cwd()가 watcher 스코프(api/src 아래)일 가능성을 차단합니다.
|
|
111
|
-
// cwd가 그 위치라면 write/unlink가 짧은 순간 watcher 이벤트로 잡혀 batch에 흘러들어갈 수 있어요.
|
|
112
109
|
const tmpFile = join(
|
|
113
|
-
|
|
110
|
+
// 타겟 파일이 루트 아래에 있어야 해요. 그래서 tmp 디렉토리같은거 안 씁니다!
|
|
111
|
+
process.cwd(),
|
|
114
112
|
`.sonamu-fmt-${Date.now()}-${Math.random().toString(36).slice(2)}.ts`,
|
|
115
113
|
);
|
|
116
114
|
|
|
@@ -7,7 +7,7 @@ import chalk from "chalk";
|
|
|
7
7
|
const execFileAsync = promisify(execFile);
|
|
8
8
|
|
|
9
9
|
/**
|
|
10
|
-
*
|
|
10
|
+
* child_process.execFile의 Promise 기반 wrapper입니다.
|
|
11
11
|
* exit code가 non-zero이면 reject해요.
|
|
12
12
|
*/
|
|
13
13
|
export async function execute(
|
|
@@ -21,11 +21,11 @@ export async function execute(
|
|
|
21
21
|
|
|
22
22
|
/**
|
|
23
23
|
* 주어진 작업을 실행합니다.
|
|
24
|
-
* 주어진 프로세스 이벤트(=시그널)가 발생하였을 때에도 최대 한계(
|
|
24
|
+
* 주어진 프로세스 이벤트(=시그널)가 발생하였을 때에도 최대 한계(waitForUpTo) 동안 작업을 기다린 후 종료합니다.
|
|
25
25
|
* @param {() => Promise<void>} job - 실행할 작업
|
|
26
|
-
* @param {
|
|
27
|
-
* @param {NodeJS.Signals} options.
|
|
28
|
-
* @param {number} options.
|
|
26
|
+
* @param {{ whenThisHappens: NodeJS.Signals; waitForUpTo: number }} options - 옵션
|
|
27
|
+
* @param {NodeJS.Signals} options.whenThisHappens - 프로세스 이벤트
|
|
28
|
+
* @param {number} options.waitForUpTo - 최대 한계 시간
|
|
29
29
|
*/
|
|
30
30
|
export async function runWithGracefulShutdown(
|
|
31
31
|
job: () => Promise<void>,
|