sonamu 0.9.6 → 0.9.7
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/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 +2 -2
- package/src/api/decorators.ts +2 -2
- 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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"}
|
|
@@ -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
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "sonamu",
|
|
3
|
-
"version": "0.9.
|
|
3
|
+
"version": "0.9.7",
|
|
4
4
|
"description": "Sonamu — TypeScript Fullstack API Framework",
|
|
5
5
|
"keywords": [
|
|
6
6
|
"framework",
|
|
@@ -130,8 +130,8 @@
|
|
|
130
130
|
"vite": "8.0.5",
|
|
131
131
|
"vitest": "^4.1.2",
|
|
132
132
|
"@sonamu-kit/hmr-hook": "^0.5.1",
|
|
133
|
-
"@sonamu-kit/hmr-runner": "^0.2.0",
|
|
134
133
|
"@sonamu-kit/tasks": "^0.3.0",
|
|
134
|
+
"@sonamu-kit/hmr-runner": "^0.2.0",
|
|
135
135
|
"@sonamu-kit/ts-loader": "^2.2.0"
|
|
136
136
|
},
|
|
137
137
|
"devDependencies": {
|
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
|
|
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>,
|