velocious 1.0.256 → 1.0.258
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/build/src/environment-handlers/node/cli/commands/generate/frontend-models.d.ts +8 -0
- package/build/src/environment-handlers/node/cli/commands/generate/frontend-models.d.ts.map +1 -1
- package/build/src/environment-handlers/node/cli/commands/generate/frontend-models.js +28 -16
- package/build/src/frontend-model-resource/base-resource.d.ts +0 -2
- package/build/src/frontend-model-resource/base-resource.d.ts.map +1 -1
- package/build/src/frontend-model-resource/base-resource.js +8 -6
- package/build/src/frontend-models/base.d.ts +14 -2
- package/build/src/frontend-models/base.d.ts.map +1 -1
- package/build/src/frontend-models/base.js +21 -3
- package/build/src/frontend-models/model-registry.d.ts +13 -0
- package/build/src/frontend-models/model-registry.d.ts.map +1 -0
- package/build/src/frontend-models/model-registry.js +24 -0
- package/build/src/frontend-models/query.d.ts.map +1 -1
- package/build/src/frontend-models/query.js +3 -2
- package/build/src/frontend-models/resource-definition.d.ts.map +1 -1
- package/build/src/frontend-models/resource-definition.js +1 -7
- package/build/src/utils/ransack.d.ts.map +1 -1
- package/build/src/utils/ransack.js +3 -2
- package/build/tsconfig.tsbuildinfo +1 -1
- package/package.json +1 -1
|
@@ -50,6 +50,14 @@ export default class DbGenerateFrontendModels extends BaseCommand {
|
|
|
50
50
|
modelClass: typeof import("../../../../../database/record/index.js").default | undefined;
|
|
51
51
|
modelConfig: Record<string, any>;
|
|
52
52
|
}): string;
|
|
53
|
+
/**
|
|
54
|
+
* @param {Array<{className: string, fileName: string}>} generatedFiles - Generated model files.
|
|
55
|
+
* @returns {string} - Index file content that imports and re-exports all models.
|
|
56
|
+
*/
|
|
57
|
+
buildIndexFileContent(generatedFiles: Array<{
|
|
58
|
+
className: string;
|
|
59
|
+
fileName: string;
|
|
60
|
+
}>): string;
|
|
53
61
|
/**
|
|
54
62
|
* @param {object} args - Formatting args.
|
|
55
63
|
* @param {string} args.indent - Base indentation.
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"frontend-models.d.ts","sourceRoot":"","sources":["../../../../../../../src/environment-handlers/node/cli/commands/generate/frontend-models.js"],"names":[],"mappings":"AAKA,mGAAmG;AACnG;IACE,oEAAoE;IACpE,WADc,OAAO,CAAC,IAAI,CAAC,
|
|
1
|
+
{"version":3,"file":"frontend-models.d.ts","sourceRoot":"","sources":["../../../../../../../src/environment-handlers/node/cli/commands/generate/frontend-models.js"],"names":[],"mappings":"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"}
|
|
@@ -16,6 +16,8 @@ export default class DbGenerateFrontendModels extends BaseCommand {
|
|
|
16
16
|
const generatedModelNames = new Set();
|
|
17
17
|
/** @type {Set<string>} */
|
|
18
18
|
const ensuredDirectories = new Set();
|
|
19
|
+
/** @type {Map<string, Array<{className: string, fileName: string}>>} */
|
|
20
|
+
const generatedFilesByDirectory = new Map();
|
|
19
21
|
for (const backendProject of backendProjects) {
|
|
20
22
|
const frontendModelsDir = this.frontendModelsDirectoryForBackendProject(backendProject);
|
|
21
23
|
const importPath = this.importPathForFrontendModelsDirectory(frontendModelsDir);
|
|
@@ -23,6 +25,10 @@ export default class DbGenerateFrontendModels extends BaseCommand {
|
|
|
23
25
|
await fs.mkdir(frontendModelsDir, { recursive: true });
|
|
24
26
|
ensuredDirectories.add(frontendModelsDir);
|
|
25
27
|
}
|
|
28
|
+
if (!generatedFilesByDirectory.has(frontendModelsDir)) {
|
|
29
|
+
generatedFilesByDirectory.set(frontendModelsDir, []);
|
|
30
|
+
}
|
|
31
|
+
const generatedFiles = generatedFilesByDirectory.get(frontendModelsDir);
|
|
26
32
|
const resources = this.resourcesForBackendProject(backendProject);
|
|
27
33
|
const availableFrontendModelClassNames = this.availableFrontendModelClassNames(resources);
|
|
28
34
|
for (const modelClassName in resources) {
|
|
@@ -45,9 +51,15 @@ export default class DbGenerateFrontendModels extends BaseCommand {
|
|
|
45
51
|
modelConfig
|
|
46
52
|
});
|
|
47
53
|
await fs.writeFile(filePath, fileContent);
|
|
54
|
+
generatedFiles.push({ className, fileName });
|
|
48
55
|
console.log(`create src/frontend-models/${fileName}`);
|
|
49
56
|
}
|
|
50
57
|
}
|
|
58
|
+
for (const [frontendModelsDir, generatedFiles] of generatedFilesByDirectory) {
|
|
59
|
+
const indexContent = this.buildIndexFileContent(generatedFiles);
|
|
60
|
+
await fs.writeFile(`${frontendModelsDir}/index.js`, indexContent);
|
|
61
|
+
console.log("create src/frontend-models/index.js");
|
|
62
|
+
}
|
|
51
63
|
}
|
|
52
64
|
/**
|
|
53
65
|
* @param {object} args - Arguments.
|
|
@@ -155,18 +167,6 @@ export default class DbGenerateFrontendModels extends BaseCommand {
|
|
|
155
167
|
&& builtInMemberCommands.url === "url";
|
|
156
168
|
let fileContent = "";
|
|
157
169
|
fileContent += `import FrontendModelBase from "${importPath}"\n`;
|
|
158
|
-
if (relationships.length > 0) {
|
|
159
|
-
/** @type {Set<string>} */
|
|
160
|
-
const importedTargetClasses = new Set();
|
|
161
|
-
for (const relationship of relationships) {
|
|
162
|
-
if (relationship.targetClassName == className)
|
|
163
|
-
continue;
|
|
164
|
-
if (importedTargetClasses.has(relationship.targetClassName))
|
|
165
|
-
continue;
|
|
166
|
-
fileContent += `import ${relationship.targetClassName} from "./${relationship.targetFileName}.js"\n`;
|
|
167
|
-
importedTargetClasses.add(relationship.targetClassName);
|
|
168
|
-
}
|
|
169
|
-
}
|
|
170
170
|
fileContent += "\n";
|
|
171
171
|
fileContent += "/**\n";
|
|
172
172
|
fileContent += ` * @typedef {object} ${attributesTypeName}\n`;
|
|
@@ -247,12 +247,11 @@ export default class DbGenerateFrontendModels extends BaseCommand {
|
|
|
247
247
|
fileContent += " }\n";
|
|
248
248
|
fileContent += " }\n";
|
|
249
249
|
fileContent += "\n";
|
|
250
|
-
fileContent += " /** @returns {Record<string,
|
|
250
|
+
fileContent += " /** @returns {Record<string, string>} - Relationship model class names. */\n";
|
|
251
251
|
fileContent += " static relationshipModelClasses() {\n";
|
|
252
252
|
fileContent += " return {\n";
|
|
253
253
|
for (const relationship of relationships) {
|
|
254
|
-
|
|
255
|
-
fileContent += ` ${relationship.relationshipName}: ${targetClassReference},\n`;
|
|
254
|
+
fileContent += ` ${relationship.relationshipName}: ${JSON.stringify(relationship.targetClassName)},\n`;
|
|
256
255
|
}
|
|
257
256
|
fileContent += " }\n";
|
|
258
257
|
fileContent += " }\n";
|
|
@@ -334,8 +333,21 @@ export default class DbGenerateFrontendModels extends BaseCommand {
|
|
|
334
333
|
}
|
|
335
334
|
}
|
|
336
335
|
fileContent += "}\n";
|
|
336
|
+
fileContent += "\n";
|
|
337
|
+
fileContent += `FrontendModelBase.registerModel(${className})\n`;
|
|
337
338
|
return fileContent;
|
|
338
339
|
}
|
|
340
|
+
/**
|
|
341
|
+
* @param {Array<{className: string, fileName: string}>} generatedFiles - Generated model files.
|
|
342
|
+
* @returns {string} - Index file content that imports and re-exports all models.
|
|
343
|
+
*/
|
|
344
|
+
buildIndexFileContent(generatedFiles) {
|
|
345
|
+
let content = "";
|
|
346
|
+
for (const { className, fileName } of generatedFiles) {
|
|
347
|
+
content += `export {default as ${className}} from "./${fileName}"\n`;
|
|
348
|
+
}
|
|
349
|
+
return content;
|
|
350
|
+
}
|
|
339
351
|
/**
|
|
340
352
|
* @param {object} args - Formatting args.
|
|
341
353
|
* @param {string} args.indent - Base indentation.
|
|
@@ -561,4 +573,4 @@ export default class DbGenerateFrontendModels extends BaseCommand {
|
|
|
561
573
|
};
|
|
562
574
|
}
|
|
563
575
|
}
|
|
564
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"frontend-models.js","sourceRoot":"","sources":["../../../../../../../src/environment-handlers/node/cli/commands/generate/frontend-models.js"],"names":[],"mappings":"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","sourcesContent":["import BaseCommand from \"../../../../../cli/base-command.js\"\nimport fs from \"fs/promises\"\nimport * as inflection from \"inflection\"\nimport {frontendModelResourceConfigurationFromDefinition, frontendModelResourcesForBackendProject} from \"../../../../../frontend-models/resource-definition.js\"\n\n/** Node CLI command that generates frontend model classes from backend project resource config. */\nexport default class DbGenerateFrontendModels extends BaseCommand {\n  /** @returns {Promise<void>} - Resolves when files are generated. */\n  async execute() {\n    const configuration = this.getConfiguration()\n    const backendProjects = configuration.getBackendProjects()\n\n    await configuration.initializeModels()\n\n    if (!Array.isArray(backendProjects) || backendProjects.length === 0) {\n      throw new Error(\"No backend projects configured. Configure 'backendProjects' in your configuration first\")\n    }\n\n    /** @type {Set<string>} */\n    const generatedModelNames = new Set()\n    /** @type {Set<string>} */\n    const ensuredDirectories = new Set()\n\n    for (const backendProject of backendProjects) {\n      const frontendModelsDir = this.frontendModelsDirectoryForBackendProject(backendProject)\n      const importPath = this.importPathForFrontendModelsDirectory(frontendModelsDir)\n\n      if (!ensuredDirectories.has(frontendModelsDir)) {\n        await fs.mkdir(frontendModelsDir, {recursive: true})\n        ensuredDirectories.add(frontendModelsDir)\n      }\n\n      const resources = this.resourcesForBackendProject(backendProject)\n      const availableFrontendModelClassNames = this.availableFrontendModelClassNames(resources)\n\n      for (const modelClassName in resources) {\n        const modelConfig = frontendModelResourceConfigurationFromDefinition(resources[modelClassName])\n        const className = inflection.camelize(modelClassName.replaceAll(\"-\", \"_\"))\n        const fileName = `${inflection.dasherize(inflection.underscore(className))}.js`\n        const filePath = `${frontendModelsDir}/${fileName}`\n\n        if (!modelConfig) {\n          throw new Error(`Invalid frontend model resource definition for '${className}'`)\n        }\n\n        this.validateModelConfig({availableFrontendModelClassNames, className, modelConfig})\n\n        if (generatedModelNames.has(className)) {\n          throw new Error(`Duplicate frontend model definition for '${className}'`)\n        }\n\n        generatedModelNames.add(className)\n\n        const fileContent = this.buildModelFileContent({\n          className,\n          importPath,\n          modelClass: configuration.getModelClasses()[className],\n          modelConfig\n        })\n\n        await fs.writeFile(filePath, fileContent)\n\n        console.log(`create src/frontend-models/${fileName}`)\n      }\n    }\n  }\n\n  /**\n   * @param {object} args - Arguments.\n   * @param {Set<string>} args.availableFrontendModelClassNames - Available frontend model class names in backend project.\n   * @param {string} args.className - Model class name.\n   * @param {Record<string, any>} args.modelConfig - Model configuration.\n   * @returns {void} - No return value.\n   */\n  validateModelConfig({availableFrontendModelClassNames, className, modelConfig}) {\n    const abilities = modelConfig.abilities\n\n    if (!abilities || typeof abilities !== \"object\") {\n      throw new Error(`Model '${className}' is missing required 'abilities' config`)\n    }\n\n    const readActions = [\"index\", \"find\"]\n\n    for (const action of readActions) {\n      const abilityAction = abilities[action]\n\n      if (typeof abilityAction !== \"string\" || abilityAction.length < 1) {\n        throw new Error(`Model '${className}' is missing required abilities.${action} config`)\n      }\n    }\n\n    const relationships = modelConfig.relationships\n\n    if (relationships === undefined) return\n\n    const normalizedRelationships = this.relationshipsForModel({className, modelConfig})\n\n    for (const relationship of normalizedRelationships) {\n      if (!availableFrontendModelClassNames.has(relationship.targetClassName)) {\n        throw new Error(`Model '${className}' relationship '${relationship.relationshipName}' references '${relationship.targetClassName}', but no frontend model resource exists for that target in this backend project`)\n      }\n    }\n  }\n\n  /**\n   * @param {import(\"../../../../../configuration-types.js\").BackendProjectConfiguration} backendProject - Backend project config.\n   * @returns {Record<string, import(\"../../../../../configuration-types.js\").FrontendModelResourceDefinition>} - Resource definitions keyed by model class name.\n   */\n  resourcesForBackendProject(backendProject) {\n    return frontendModelResourcesForBackendProject(backendProject)\n  }\n\n  /**\n   * @param {Record<string, any>} resources - Resource configuration keyed by model name.\n   * @returns {Set<string>} - Available frontend model class names.\n   */\n  availableFrontendModelClassNames(resources) {\n    /** @type {Set<string>} */\n    const classNames = new Set()\n\n    for (const resourceModelName in resources) {\n      classNames.add(inflection.camelize(resourceModelName.replaceAll(\"-\", \"_\")))\n    }\n\n    return classNames\n  }\n\n  /**\n   * @param {{frontendModelsOutputPath?: string}} backendProject - Backend project config.\n   * @returns {string} - Absolute frontend models output directory.\n   */\n  frontendModelsDirectoryForBackendProject(backendProject) {\n    const outputPath = backendProject.frontendModelsOutputPath || this.directory()\n\n    return `${outputPath}/src/frontend-models`\n  }\n\n  /**\n   * @param {string} frontendModelsDir - Frontend models output directory.\n   * @returns {string} - Base class import path.\n   */\n  importPathForFrontendModelsDirectory(frontendModelsDir) {\n    const devMode = frontendModelsDir.includes(\"/spec/dummy/src/frontend-models\")\n\n    if (devMode) {\n      return \"../../../../src/frontend-models/base.js\"\n    }\n\n    return \"velocious/build/src/frontend-models/base.js\"\n  }\n\n  /**\n   * @param {object} args - Method args.\n   * @param {string} args.className - Model class name.\n   * @param {string} args.importPath - Base class import path.\n   * @param {typeof import(\"../../../../../database/record/index.js\").default | undefined} args.modelClass - Backend model class.\n   * @param {Record<string, any>} args.modelConfig - Model configuration.\n   * @returns {string} - Generated file content.\n   */\n  buildModelFileContent({className, importPath, modelClass, modelConfig}) {\n    const attributes = this.attributeDefinitionsForModel({modelClass, modelConfig})\n    const relationships = this.relationshipsForModel({className, modelConfig})\n    const attachments = modelConfig.attachments && typeof modelConfig.attachments === \"object\"\n      ? modelConfig.attachments\n      : {}\n    const attributesTypeName = `${className}Attributes`\n    const attributeNames = attributes.map((attribute) => attribute.name)\n    const builtInCollectionCommands = {\n      create: modelConfig.builtInCollectionCommands.create || \"create\",\n      index: modelConfig.builtInCollectionCommands.index || \"index\"\n    }\n    const builtInMemberCommands = {\n      attach: modelConfig.builtInMemberCommands.attach || \"attach\",\n      destroy: modelConfig.builtInMemberCommands.destroy || \"destroy\",\n      download: modelConfig.builtInMemberCommands.download || \"download\",\n      find: modelConfig.builtInMemberCommands.find || \"find\",\n      update: modelConfig.builtInMemberCommands.update || \"update\",\n      url: modelConfig.builtInMemberCommands.url || \"url\"\n    }\n    const collectionCommands = modelConfig.collectionCommands\n    const memberCommands = modelConfig.memberCommands\n    const builtInCollectionCommandsAreDefault = builtInCollectionCommands.create === \"create\" && builtInCollectionCommands.index === \"index\"\n    const builtInMemberCommandsAreDefault = builtInMemberCommands.attach === \"attach\"\n      && builtInMemberCommands.destroy === \"destroy\"\n      && builtInMemberCommands.download === \"download\"\n      && builtInMemberCommands.find === \"find\"\n      && builtInMemberCommands.update === \"update\"\n      && builtInMemberCommands.url === \"url\"\n\n    let fileContent = \"\"\n\n    fileContent += `import FrontendModelBase from \"${importPath}\"\\n`\n\n    if (relationships.length > 0) {\n      /** @type {Set<string>} */\n      const importedTargetClasses = new Set()\n\n      for (const relationship of relationships) {\n        if (relationship.targetClassName == className) continue\n        if (importedTargetClasses.has(relationship.targetClassName)) continue\n\n        fileContent += `import ${relationship.targetClassName} from \"./${relationship.targetFileName}.js\"\\n`\n        importedTargetClasses.add(relationship.targetClassName)\n      }\n    }\n\n    fileContent += \"\\n\"\n    fileContent += \"/**\\n\"\n    fileContent += ` * @typedef {object} ${attributesTypeName}\\n`\n    for (const attribute of attributes) {\n      fileContent += ` * @property {${attribute.jsDocType}} ${attribute.name} - Attribute value.\\n`\n    }\n    fileContent += \" */\\n\"\n    fileContent += `/** Frontend model for ${className}. */\\n`\n    fileContent += `export default class ${className} extends FrontendModelBase {\\n`\n    fileContent += \"  /** @returns {{attachments?: Record<string, {type: \\\"hasOne\\\" | \\\"hasMany\\\"}>, attributes: string[], builtInCollectionCommands?: Record<string, string>, builtInMemberCommands?: Record<string, string>, collectionCommands?: Record<string, string>, memberCommands?: Record<string, string>, primaryKey?: string}} - Resource config. */\\n\"\n    fileContent += \"  static resourceConfig() {\\n\"\n    fileContent += \"    return {\\n\"\n    if (Object.keys(attachments).length > 0) {\n      fileContent += \"      attachments: {\\n\"\n      for (const [attachmentName, attachmentConfig] of Object.entries(attachments)) {\n        const attachmentType = attachmentConfig && typeof attachmentConfig === \"object\" && attachmentConfig.type === \"hasMany\"\n          ? \"hasMany\"\n          : \"hasOne\"\n\n        fileContent += `        ${attachmentName}: {type: ${JSON.stringify(attachmentType)}},\\n`\n      }\n      fileContent += \"      },\\n\"\n    }\n    fileContent += this.formattedArrayProperty({\n      indent: \"      \",\n      propertyName: \"attributes\",\n      values: attributeNames\n    })\n    if (!builtInCollectionCommandsAreDefault) {\n      fileContent += this.formattedObjectProperty({\n        filterDefaultValues: {create: \"create\", index: \"index\"},\n        indent: \"      \",\n        propertyName: \"builtInCollectionCommands\",\n        values: builtInCollectionCommands\n      })\n    }\n    if (!builtInMemberCommandsAreDefault) {\n      fileContent += this.formattedObjectProperty({\n        filterDefaultValues: {\n          attach: \"attach\",\n          destroy: \"destroy\",\n          download: \"download\",\n          find: \"find\",\n          update: \"update\",\n          url: \"url\"\n        },\n        indent: \"      \",\n        propertyName: \"builtInMemberCommands\",\n        values: builtInMemberCommands\n      })\n    }\n    if (Object.keys(collectionCommands).length > 0) {\n      fileContent += this.formattedObjectProperty({\n        indent: \"      \",\n        propertyName: \"collectionCommands\",\n        values: collectionCommands\n      })\n    }\n    if (Object.keys(memberCommands).length > 0) {\n      fileContent += this.formattedObjectProperty({\n        indent: \"      \",\n        propertyName: \"memberCommands\",\n        values: memberCommands\n      })\n    }\n    if (modelClass && modelClass.primaryKey() !== \"id\") {\n      fileContent += `      primaryKey: ${JSON.stringify(modelClass.primaryKey())},\\n`\n    }\n    fileContent += \"    }\\n\"\n    fileContent += \"  }\\n\"\n\n    if (relationships.length > 0) {\n      fileContent += \"\\n\"\n      fileContent += \"  /** @returns {Record<string, {type: \\\"belongsTo\\\" | \\\"hasOne\\\" | \\\"hasMany\\\"}>} - Relationship definitions. */\\n\"\n      fileContent += \"  static relationshipDefinitions() {\\n\"\n      fileContent += \"    return {\\n\"\n      for (const relationship of relationships) {\n        fileContent += `      ${relationship.relationshipName}: {type: ${JSON.stringify(relationship.type)}},\\n`\n      }\n      fileContent += \"    }\\n\"\n      fileContent += \"  }\\n\"\n\n      fileContent += \"\\n\"\n      fileContent += \"  /** @returns {Record<string, typeof FrontendModelBase>} - Relationship model classes. */\\n\"\n      fileContent += \"  static relationshipModelClasses() {\\n\"\n      fileContent += \"    return {\\n\"\n      for (const relationship of relationships) {\n        const targetClassReference = relationship.targetClassName == className ? className : relationship.targetClassName\n        fileContent += `      ${relationship.relationshipName}: ${targetClassReference},\\n`\n      }\n      fileContent += \"    }\\n\"\n      fileContent += \"  }\\n\"\n    }\n\n    for (const attribute of attributes) {\n      const camelizedAttribute = inflection.camelize(attribute.name, true)\n      const camelizedAttributeUpper = inflection.camelize(attribute.name)\n\n      fileContent += \"\\n\"\n      fileContent += `  /** @returns {${attributesTypeName}[${JSON.stringify(attribute.name)}]} - Attribute value. */\\n`\n      fileContent += `  ${camelizedAttribute}() { return this.readAttribute(${JSON.stringify(attribute.name)}) }\\n`\n\n      fileContent += \"\\n\"\n      fileContent += \"  /**\\n\"\n      fileContent += `   * @param {${attributesTypeName}[${JSON.stringify(attribute.name)}]} newValue - New attribute value.\\n`\n      fileContent += `   * @returns {${attributesTypeName}[${JSON.stringify(attribute.name)}]} - Assigned value.\\n`\n      fileContent += \"   */\\n\"\n      fileContent += `  set${camelizedAttributeUpper}(newValue) { return this.setAttribute(${JSON.stringify(attribute.name)}, newValue) }\\n`\n    }\n\n    for (const methodName of Object.keys(collectionCommands)) {\n      fileContent += \"\\n\"\n      fileContent += \"  /**\\n\"\n      fileContent += \"   * @param {Record<string, any>} [payload={}] - Command payload.\\n\"\n      fileContent += \"   * @returns {Promise<Record<string, any>>} - Command response.\\n\"\n      fileContent += \"   */\\n\"\n      fileContent += `  static async ${methodName}(payload = {}) {\\n`\n      fileContent += \"    return await this.executeCustomCommand({\\n\"\n      fileContent += `      commandName: ${JSON.stringify(collectionCommands[methodName])},\\n`\n      fileContent += `      commandType: ${JSON.stringify(collectionCommands[methodName])},\\n`\n      fileContent += \"      payload,\\n\"\n      fileContent += \"      resourcePath: this.resourcePath()\\n\"\n      fileContent += \"    })\\n\"\n      fileContent += \"  }\\n\"\n    }\n\n    for (const methodName of Object.keys(memberCommands)) {\n      fileContent += \"\\n\"\n      fileContent += \"  /**\\n\"\n      fileContent += \"   * @param {Record<string, any>} [payload={}] - Command payload.\\n\"\n      fileContent += \"   * @returns {Promise<Record<string, any>>} - Command response.\\n\"\n      fileContent += \"   */\\n\"\n      fileContent += `  async ${methodName}(payload = {}) {\\n`\n      fileContent += `    return await ${className}.executeCustomCommand({\\n`\n      fileContent += `      commandName: ${JSON.stringify(memberCommands[methodName])},\\n`\n      fileContent += `      commandType: ${JSON.stringify(memberCommands[methodName])},\\n`\n      fileContent += \"      memberId: this.primaryKeyValue(),\\n\"\n      fileContent += \"      payload,\\n\"\n      fileContent += `      resourcePath: ${className}.resourcePath()\\n`\n      fileContent += \"    })\\n\"\n      fileContent += \"  }\\n\"\n    }\n\n    for (const relationship of relationships) {\n      const relationshipNameCamelized = inflection.camelize(relationship.relationshipName)\n      const targetImportPath = `./${relationship.targetFileName}.js`\n\n      if (relationship.type == \"hasMany\") {\n        fileContent += \"\\n\"\n        fileContent += `  /** @returns {import(${JSON.stringify(importPath)}).FrontendModelHasManyRelationship<typeof import(${JSON.stringify(`./${inflection.dasherize(inflection.underscore(className))}.js`)}).default, typeof import(${JSON.stringify(targetImportPath)}).default>} - Relationship helper. */\\n`\n        fileContent += `  ${relationship.relationshipName}() { return /** @type {import(${JSON.stringify(importPath)}).FrontendModelHasManyRelationship<typeof import(${JSON.stringify(`./${inflection.dasherize(inflection.underscore(className))}.js`)}).default, typeof import(${JSON.stringify(targetImportPath)}).default>} */ (this.getRelationshipByName(${JSON.stringify(relationship.relationshipName)})) }\\n`\n\n        fileContent += \"\\n\"\n        fileContent += `  /** @returns {Array<import(${JSON.stringify(targetImportPath)}).default>} - Loaded related models. */\\n`\n        fileContent += `  ${relationship.relationshipName}Loaded() { return /** @type {Array<import(${JSON.stringify(targetImportPath)}).default>} */ (this.getRelationshipByName(${JSON.stringify(relationship.relationshipName)}).loaded()) }\\n`\n\n        fileContent += \"\\n\"\n        fileContent += `  /** @returns {Promise<Array<import(${JSON.stringify(targetImportPath)}).default>>} - Loaded related models. */\\n`\n        fileContent += `  async load${relationshipNameCamelized}() { return /** @type {Promise<Array<import(${JSON.stringify(targetImportPath)}).default>>} */ (this.loadRelationship(${JSON.stringify(relationship.relationshipName)})) }\\n`\n      } else {\n        fileContent += \"\\n\"\n        fileContent += `  /** @returns {import(${JSON.stringify(targetImportPath)}).default | null} - Loaded related model. */\\n`\n        fileContent += `  ${relationship.relationshipName}() { return /** @type {import(${JSON.stringify(targetImportPath)}).default | null} */ (this.getRelationshipByName(${JSON.stringify(relationship.relationshipName)}).loaded()) }\\n`\n\n        fileContent += \"\\n\"\n        fileContent += \"  /**\\n\"\n        fileContent += `   * @param {Record<string, any>} [attributes] - Attributes for the new related model.\\n`\n        fileContent += `   * @returns {import(${JSON.stringify(targetImportPath)}).default} - Built related model.\\n`\n        fileContent += \"   */\\n\"\n        fileContent += `  build${relationshipNameCamelized}(attributes = {}) { return /** @type {import(${JSON.stringify(targetImportPath)}).default} */ (this.getRelationshipByName(${JSON.stringify(relationship.relationshipName)}).build(attributes)) }\\n`\n\n        fileContent += \"\\n\"\n        fileContent += `  /** @returns {Promise<import(${JSON.stringify(targetImportPath)}).default | null>} - Loaded related model. */\\n`\n        fileContent += `  async load${relationshipNameCamelized}() { return /** @type {Promise<import(${JSON.stringify(targetImportPath)}).default | null>} */ (this.loadRelationship(${JSON.stringify(relationship.relationshipName)})) }\\n`\n\n        fileContent += \"\\n\"\n        fileContent += `  /** @param {import(${JSON.stringify(targetImportPath)}).default | null} model - Related model. @returns {import(${JSON.stringify(targetImportPath)}).default | null} - Assigned related model. */\\n`\n        fileContent += `  set${relationshipNameCamelized}(model) { return /** @type {import(${JSON.stringify(targetImportPath)}).default | null} */ (this.setRelationship(${JSON.stringify(relationship.relationshipName)}, model)) }\\n`\n      }\n    }\n\n    fileContent += \"}\\n\"\n\n    return fileContent\n  }\n\n  /**\n   * @param {object} args - Formatting args.\n   * @param {string} args.indent - Base indentation.\n   * @param {string} args.propertyName - Object property name.\n   * @param {string[]} args.values - String values.\n   * @returns {string} - Formatted multiline array property.\n   */\n  formattedArrayProperty({indent, propertyName, values}) {\n    let output = `${indent}${propertyName}: [\\n`\n\n    for (const value of values) {\n      output += `${indent}  ${JSON.stringify(value)},\\n`\n    }\n\n    output += `${indent}],\\n`\n\n    return output\n  }\n\n  /**\n   * @param {object} args - Formatting args.\n   * @param {string} args.indent - Base indentation.\n   * @param {string} args.propertyName - Object property name.\n   * @param {Record<string, string>} args.values - Object key-values.\n   * @param {Record<string, string>} [args.filterDefaultValues] - Default values to omit from output.\n   * @returns {string} - Formatted multiline object property.\n   */\n  formattedObjectProperty({filterDefaultValues, indent, propertyName, values}) {\n    let output = `${indent}${propertyName}: {\\n`\n\n    for (const objectKey of Object.keys(values)) {\n      if (filterDefaultValues && filterDefaultValues[objectKey] === values[objectKey]) continue\n\n      output += `${indent}  ${objectKey}: ${JSON.stringify(values[objectKey])},\\n`\n    }\n\n    output += `${indent}},\\n`\n\n    return output\n  }\n\n  /**\n   * @param {object} args - Arguments.\n   * @param {typeof import(\"../../../../../database/record/index.js\").default | undefined} args.modelClass - Backend model class.\n   * @param {Record<string, any>} args.modelConfig - Model configuration.\n   * @returns {Array<{jsDocType: string, name: string}>} - Attribute definitions.\n   */\n  attributeDefinitionsForModel({modelClass, modelConfig}) {\n    const attributes = modelConfig.attributes\n\n    if (Array.isArray(attributes)) {\n      return attributes.map((attributeName) => ({\n        jsDocType: this.jsDocTypeForFrontendAttribute({\n          attributeConfig: this.frontendAttributeConfigForModelAttribute({attributeName, modelClass})\n        }),\n        name: attributeName\n      }))\n    }\n\n    if (!attributes || typeof attributes !== \"object\") {\n      throw new Error(`Expected 'attributes' as array or object but got: ${attributes}`)\n    }\n\n    return Object.keys(attributes).map((attributeName) => {\n      const attributeConfig = attributes[attributeName]\n\n      return {\n        jsDocType: this.jsDocTypeForFrontendAttribute({attributeConfig}),\n        name: attributeName\n      }\n    })\n  }\n\n  /**\n   * @param {object} args - Arguments.\n   * @param {any} args.attributeConfig - Attribute configuration value.\n   * @returns {string} - JSDoc type.\n   */\n  jsDocTypeForFrontendAttribute({attributeConfig}) {\n    const jsDocType = this.jsDocTypeForFrontendAttributeBaseType(attributeConfig)\n\n    if (!this.frontendAttributeCanBeNull(attributeConfig)) {\n      return jsDocType\n    }\n\n    return `${jsDocType} | null`\n  }\n\n  /**\n   * @param {any} attributeConfig - Attribute configuration value.\n   * @returns {string} - Non-nullable JSDoc type.\n   */\n  jsDocTypeForFrontendAttributeBaseType(attributeConfig) {\n    if (!attributeConfig || typeof attributeConfig !== \"object\") {\n      return \"any\"\n    }\n\n    const type = this.frontendAttributeTypeValue(attributeConfig)\n\n    if (type == \"boolean\") {\n      return \"boolean\"\n    } else if (type == \"json\" || type == \"jsonb\") {\n      return \"Record<string, any>\"\n    } else if ([\"blob\", \"char\", \"nvarchar\", \"varchar\", \"text\", \"longtext\", \"uuid\", \"character varying\"].includes(type)) {\n      return \"string\"\n    } else if ([\"bit\", \"bigint\", \"decimal\", \"double\", \"double precision\", \"float\", \"int\", \"integer\", \"numeric\", \"real\", \"smallint\", \"tinyint\"].includes(type)) {\n      return \"number\"\n    } else if ([\"date\", \"datetime\", \"timestamp\", \"timestamp without time zone\", \"timestamptz\"].includes(type)) {\n      return \"Date\"\n    } else {\n      return \"any\"\n    }\n  }\n\n  /**\n   * @param {any} attributeConfig - Attribute configuration value.\n   * @returns {boolean} - Whether the attribute allows null values.\n   */\n  frontendAttributeCanBeNull(attributeConfig) {\n    if (!attributeConfig || typeof attributeConfig !== \"object\") {\n      return false\n    }\n\n    if (typeof attributeConfig.getNull == \"function\") {\n      return attributeConfig.getNull() === true\n    }\n\n    return attributeConfig.null === true\n  }\n\n  /**\n   * @param {any} attributeConfig - Attribute configuration value.\n   * @returns {string | null} - Normalized column type.\n   */\n  frontendAttributeTypeValue(attributeConfig) {\n    if (!attributeConfig || typeof attributeConfig !== \"object\") {\n      return null\n    }\n\n    if (typeof attributeConfig.getType == \"function\") {\n      return String(attributeConfig.getType())\n    }\n\n    const typeValue = attributeConfig.type || attributeConfig.columnType || attributeConfig.sqlType || attributeConfig.dataType\n\n    if (typeof typeValue !== \"string\") {\n      return null\n    }\n\n    return typeValue\n  }\n\n  /**\n   * @param {object} args - Arguments.\n   * @param {string} args.attributeName - Frontend model attribute name.\n   * @param {typeof import(\"../../../../../database/record/index.js\").default | undefined} args.modelClass - Backend model class.\n   * @returns {any} - Attribute config inferred from the backend model when available.\n   */\n  frontendAttributeConfigForModelAttribute({attributeName, modelClass}) {\n    if (!modelClass) {\n      return null\n    }\n\n    const columnName = modelClass.getAttributeNameToColumnNameMap()[attributeName]\n\n    if (!columnName) {\n      return null\n    }\n\n    return modelClass.getColumnsHash()[columnName] || null\n  }\n\n  /**\n   * @param {object} args - Arguments.\n   * @param {string} args.className - Model class name.\n   * @param {Record<string, any>} args.modelConfig - Model configuration.\n   * @returns {Array<{relationshipName: string, targetClassName: string, targetFileName: string, type: \"belongsTo\" | \"hasOne\" | \"hasMany\"}>} - Relationships.\n   */\n  relationshipsForModel({className, modelConfig}) {\n    const relationships = modelConfig.relationships\n\n    if (relationships === undefined || relationships === null) {\n      return []\n    }\n\n    if (Array.isArray(relationships)) {\n      return relationships.map((relationshipName) => this.inferredRelationshipDefinition({className, relationshipName}))\n    }\n\n    if (typeof relationships !== \"object\") {\n      throw new Error(`Model '${className}' has invalid relationships config`)\n    }\n\n    /** @type {Array<{relationshipName: string, targetClassName: string, targetFileName: string, type: \"belongsTo\" | \"hasOne\" | \"hasMany\"}>} */\n    const normalized = []\n\n    for (const relationshipName in relationships) {\n      const relationship = relationships[relationshipName]\n\n      if (relationship === true) {\n        normalized.push(this.inferredRelationshipDefinition({className, relationshipName}))\n        continue\n      }\n\n      if (!relationship || typeof relationship !== \"object\" || Array.isArray(relationship)) {\n        throw new Error(`Model '${className}' relationship '${relationshipName}' must be an object or true`)\n      }\n\n      const relationshipModelName = relationship.modelClassName || relationship.className || relationship.model\n      const hasExplicitType = typeof relationship.type === \"string\" && relationship.type.length > 0\n      const hasExplicitModel = typeof relationshipModelName === \"string\" && relationshipModelName.length > 0\n      const inferredRelationship = hasExplicitType && hasExplicitModel\n        ? null\n        : this.inferredRelationshipDefinition({className, relationshipName})\n      const relationshipType = relationship.type || inferredRelationship?.type\n\n      if (relationshipType !== \"belongsTo\" && relationshipType !== \"hasOne\" && relationshipType !== \"hasMany\") {\n        throw new Error(`Model '${className}' relationship '${relationshipName}' has invalid type '${relationshipType}'`)\n      }\n\n      const targetClassName = hasExplicitModel\n        ? inflection.camelize(String(relationshipModelName).replaceAll(\"-\", \"_\"))\n        : inferredRelationship?.targetClassName\n\n      normalized.push({\n        relationshipName,\n        targetClassName,\n        targetFileName: inflection.dasherize(inflection.underscore(targetClassName)),\n        type: relationshipType\n      })\n    }\n\n    return normalized\n  }\n\n  /**\n   * @param {object} args - Arguments.\n   * @param {string} args.className - Model class name.\n   * @param {string} args.relationshipName - Relationship name.\n   * @returns {{relationshipName: string, targetClassName: string, targetFileName: string, type: \"belongsTo\" | \"hasOne\" | \"hasMany\"}} Inferred relationship definition.\n   */\n  inferredRelationshipDefinition({className, relationshipName}) {\n    const modelClass = this.getConfiguration().getModelClass(className)\n\n    if (!modelClass) {\n      throw new Error(`Could not find backend model class '${className}' for relationship '${relationshipName}'`)\n    }\n\n    const relationship = modelClass.getRelationshipByName(relationshipName)\n    const targetModelClass = relationship.getTargetModelClass()\n    const relationshipType = relationship.getType()\n\n    if (relationshipType !== \"belongsTo\" && relationshipType !== \"hasOne\" && relationshipType !== \"hasMany\") {\n      throw new Error(`Model '${className}' relationship '${relationshipName}' has unsupported type '${relationshipType}'`)\n    }\n\n    if (!targetModelClass) {\n      throw new Error(`Model '${className}' relationship '${relationshipName}' has no target model class`)\n    }\n\n    const targetClassName = targetModelClass.name\n\n    return {\n      relationshipName,\n      targetClassName,\n      targetFileName: inflection.dasherize(inflection.underscore(targetClassName)),\n      type: relationshipType\n    }\n  }\n}\n"]}
|
|
576
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"frontend-models.js","sourceRoot":"","sources":["../../../../../../../src/environment-handlers/node/cli/commands/generate/frontend-models.js"],"names":[],"mappings":"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","sourcesContent":["import BaseCommand from \"../../../../../cli/base-command.js\"\nimport fs from \"fs/promises\"\nimport * as inflection from \"inflection\"\nimport {frontendModelResourceConfigurationFromDefinition, frontendModelResourcesForBackendProject} from \"../../../../../frontend-models/resource-definition.js\"\n\n/** Node CLI command that generates frontend model classes from backend project resource config. */\nexport default class DbGenerateFrontendModels extends BaseCommand {\n  /** @returns {Promise<void>} - Resolves when files are generated. */\n  async execute() {\n    const configuration = this.getConfiguration()\n    const backendProjects = configuration.getBackendProjects()\n\n    await configuration.initializeModels()\n\n    if (!Array.isArray(backendProjects) || backendProjects.length === 0) {\n      throw new Error(\"No backend projects configured. Configure 'backendProjects' in your configuration first\")\n    }\n\n    /** @type {Set<string>} */\n    const generatedModelNames = new Set()\n    /** @type {Set<string>} */\n    const ensuredDirectories = new Set()\n    /** @type {Map<string, Array<{className: string, fileName: string}>>} */\n    const generatedFilesByDirectory = new Map()\n\n    for (const backendProject of backendProjects) {\n      const frontendModelsDir = this.frontendModelsDirectoryForBackendProject(backendProject)\n      const importPath = this.importPathForFrontendModelsDirectory(frontendModelsDir)\n\n      if (!ensuredDirectories.has(frontendModelsDir)) {\n        await fs.mkdir(frontendModelsDir, {recursive: true})\n        ensuredDirectories.add(frontendModelsDir)\n      }\n\n      if (!generatedFilesByDirectory.has(frontendModelsDir)) {\n        generatedFilesByDirectory.set(frontendModelsDir, [])\n      }\n\n      const generatedFiles = generatedFilesByDirectory.get(frontendModelsDir)\n      const resources = this.resourcesForBackendProject(backendProject)\n      const availableFrontendModelClassNames = this.availableFrontendModelClassNames(resources)\n\n      for (const modelClassName in resources) {\n        const modelConfig = frontendModelResourceConfigurationFromDefinition(resources[modelClassName])\n        const className = inflection.camelize(modelClassName.replaceAll(\"-\", \"_\"))\n        const fileName = `${inflection.dasherize(inflection.underscore(className))}.js`\n        const filePath = `${frontendModelsDir}/${fileName}`\n\n        if (!modelConfig) {\n          throw new Error(`Invalid frontend model resource definition for '${className}'`)\n        }\n\n        this.validateModelConfig({availableFrontendModelClassNames, className, modelConfig})\n\n        if (generatedModelNames.has(className)) {\n          throw new Error(`Duplicate frontend model definition for '${className}'`)\n        }\n\n        generatedModelNames.add(className)\n\n        const fileContent = this.buildModelFileContent({\n          className,\n          importPath,\n          modelClass: configuration.getModelClasses()[className],\n          modelConfig\n        })\n\n        await fs.writeFile(filePath, fileContent)\n        generatedFiles.push({className, fileName})\n\n        console.log(`create src/frontend-models/${fileName}`)\n      }\n    }\n\n    for (const [frontendModelsDir, generatedFiles] of generatedFilesByDirectory) {\n      const indexContent = this.buildIndexFileContent(generatedFiles)\n\n      await fs.writeFile(`${frontendModelsDir}/index.js`, indexContent)\n\n      console.log(\"create src/frontend-models/index.js\")\n    }\n  }\n\n  /**\n   * @param {object} args - Arguments.\n   * @param {Set<string>} args.availableFrontendModelClassNames - Available frontend model class names in backend project.\n   * @param {string} args.className - Model class name.\n   * @param {Record<string, any>} args.modelConfig - Model configuration.\n   * @returns {void} - No return value.\n   */\n  validateModelConfig({availableFrontendModelClassNames, className, modelConfig}) {\n    const abilities = modelConfig.abilities\n\n    if (!abilities || typeof abilities !== \"object\") {\n      throw new Error(`Model '${className}' is missing required 'abilities' config`)\n    }\n\n    const readActions = [\"index\", \"find\"]\n\n    for (const action of readActions) {\n      const abilityAction = abilities[action]\n\n      if (typeof abilityAction !== \"string\" || abilityAction.length < 1) {\n        throw new Error(`Model '${className}' is missing required abilities.${action} config`)\n      }\n    }\n\n    const relationships = modelConfig.relationships\n\n    if (relationships === undefined) return\n\n    const normalizedRelationships = this.relationshipsForModel({className, modelConfig})\n\n    for (const relationship of normalizedRelationships) {\n      if (!availableFrontendModelClassNames.has(relationship.targetClassName)) {\n        throw new Error(`Model '${className}' relationship '${relationship.relationshipName}' references '${relationship.targetClassName}', but no frontend model resource exists for that target in this backend project`)\n      }\n    }\n  }\n\n  /**\n   * @param {import(\"../../../../../configuration-types.js\").BackendProjectConfiguration} backendProject - Backend project config.\n   * @returns {Record<string, import(\"../../../../../configuration-types.js\").FrontendModelResourceDefinition>} - Resource definitions keyed by model class name.\n   */\n  resourcesForBackendProject(backendProject) {\n    return frontendModelResourcesForBackendProject(backendProject)\n  }\n\n  /**\n   * @param {Record<string, any>} resources - Resource configuration keyed by model name.\n   * @returns {Set<string>} - Available frontend model class names.\n   */\n  availableFrontendModelClassNames(resources) {\n    /** @type {Set<string>} */\n    const classNames = new Set()\n\n    for (const resourceModelName in resources) {\n      classNames.add(inflection.camelize(resourceModelName.replaceAll(\"-\", \"_\")))\n    }\n\n    return classNames\n  }\n\n  /**\n   * @param {{frontendModelsOutputPath?: string}} backendProject - Backend project config.\n   * @returns {string} - Absolute frontend models output directory.\n   */\n  frontendModelsDirectoryForBackendProject(backendProject) {\n    const outputPath = backendProject.frontendModelsOutputPath || this.directory()\n\n    return `${outputPath}/src/frontend-models`\n  }\n\n  /**\n   * @param {string} frontendModelsDir - Frontend models output directory.\n   * @returns {string} - Base class import path.\n   */\n  importPathForFrontendModelsDirectory(frontendModelsDir) {\n    const devMode = frontendModelsDir.includes(\"/spec/dummy/src/frontend-models\")\n\n    if (devMode) {\n      return \"../../../../src/frontend-models/base.js\"\n    }\n\n    return \"velocious/build/src/frontend-models/base.js\"\n  }\n\n  /**\n   * @param {object} args - Method args.\n   * @param {string} args.className - Model class name.\n   * @param {string} args.importPath - Base class import path.\n   * @param {typeof import(\"../../../../../database/record/index.js\").default | undefined} args.modelClass - Backend model class.\n   * @param {Record<string, any>} args.modelConfig - Model configuration.\n   * @returns {string} - Generated file content.\n   */\n  buildModelFileContent({className, importPath, modelClass, modelConfig}) {\n    const attributes = this.attributeDefinitionsForModel({modelClass, modelConfig})\n    const relationships = this.relationshipsForModel({className, modelConfig})\n    const attachments = modelConfig.attachments && typeof modelConfig.attachments === \"object\"\n      ? modelConfig.attachments\n      : {}\n    const attributesTypeName = `${className}Attributes`\n    const attributeNames = attributes.map((attribute) => attribute.name)\n    const builtInCollectionCommands = {\n      create: modelConfig.builtInCollectionCommands.create || \"create\",\n      index: modelConfig.builtInCollectionCommands.index || \"index\"\n    }\n    const builtInMemberCommands = {\n      attach: modelConfig.builtInMemberCommands.attach || \"attach\",\n      destroy: modelConfig.builtInMemberCommands.destroy || \"destroy\",\n      download: modelConfig.builtInMemberCommands.download || \"download\",\n      find: modelConfig.builtInMemberCommands.find || \"find\",\n      update: modelConfig.builtInMemberCommands.update || \"update\",\n      url: modelConfig.builtInMemberCommands.url || \"url\"\n    }\n    const collectionCommands = modelConfig.collectionCommands\n    const memberCommands = modelConfig.memberCommands\n    const builtInCollectionCommandsAreDefault = builtInCollectionCommands.create === \"create\" && builtInCollectionCommands.index === \"index\"\n    const builtInMemberCommandsAreDefault = builtInMemberCommands.attach === \"attach\"\n      && builtInMemberCommands.destroy === \"destroy\"\n      && builtInMemberCommands.download === \"download\"\n      && builtInMemberCommands.find === \"find\"\n      && builtInMemberCommands.update === \"update\"\n      && builtInMemberCommands.url === \"url\"\n\n    let fileContent = \"\"\n\n    fileContent += `import FrontendModelBase from \"${importPath}\"\\n`\n\n    fileContent += \"\\n\"\n    fileContent += \"/**\\n\"\n    fileContent += ` * @typedef {object} ${attributesTypeName}\\n`\n    for (const attribute of attributes) {\n      fileContent += ` * @property {${attribute.jsDocType}} ${attribute.name} - Attribute value.\\n`\n    }\n    fileContent += \" */\\n\"\n    fileContent += `/** Frontend model for ${className}. */\\n`\n    fileContent += `export default class ${className} extends FrontendModelBase {\\n`\n    fileContent += \"  /** @returns {{attachments?: Record<string, {type: \\\"hasOne\\\" | \\\"hasMany\\\"}>, attributes: string[], builtInCollectionCommands?: Record<string, string>, builtInMemberCommands?: Record<string, string>, collectionCommands?: Record<string, string>, memberCommands?: Record<string, string>, primaryKey?: string}} - Resource config. */\\n\"\n    fileContent += \"  static resourceConfig() {\\n\"\n    fileContent += \"    return {\\n\"\n    if (Object.keys(attachments).length > 0) {\n      fileContent += \"      attachments: {\\n\"\n      for (const [attachmentName, attachmentConfig] of Object.entries(attachments)) {\n        const attachmentType = attachmentConfig && typeof attachmentConfig === \"object\" && attachmentConfig.type === \"hasMany\"\n          ? \"hasMany\"\n          : \"hasOne\"\n\n        fileContent += `        ${attachmentName}: {type: ${JSON.stringify(attachmentType)}},\\n`\n      }\n      fileContent += \"      },\\n\"\n    }\n    fileContent += this.formattedArrayProperty({\n      indent: \"      \",\n      propertyName: \"attributes\",\n      values: attributeNames\n    })\n    if (!builtInCollectionCommandsAreDefault) {\n      fileContent += this.formattedObjectProperty({\n        filterDefaultValues: {create: \"create\", index: \"index\"},\n        indent: \"      \",\n        propertyName: \"builtInCollectionCommands\",\n        values: builtInCollectionCommands\n      })\n    }\n    if (!builtInMemberCommandsAreDefault) {\n      fileContent += this.formattedObjectProperty({\n        filterDefaultValues: {\n          attach: \"attach\",\n          destroy: \"destroy\",\n          download: \"download\",\n          find: \"find\",\n          update: \"update\",\n          url: \"url\"\n        },\n        indent: \"      \",\n        propertyName: \"builtInMemberCommands\",\n        values: builtInMemberCommands\n      })\n    }\n    if (Object.keys(collectionCommands).length > 0) {\n      fileContent += this.formattedObjectProperty({\n        indent: \"      \",\n        propertyName: \"collectionCommands\",\n        values: collectionCommands\n      })\n    }\n    if (Object.keys(memberCommands).length > 0) {\n      fileContent += this.formattedObjectProperty({\n        indent: \"      \",\n        propertyName: \"memberCommands\",\n        values: memberCommands\n      })\n    }\n    if (modelClass && modelClass.primaryKey() !== \"id\") {\n      fileContent += `      primaryKey: ${JSON.stringify(modelClass.primaryKey())},\\n`\n    }\n    fileContent += \"    }\\n\"\n    fileContent += \"  }\\n\"\n\n    if (relationships.length > 0) {\n      fileContent += \"\\n\"\n      fileContent += \"  /** @returns {Record<string, {type: \\\"belongsTo\\\" | \\\"hasOne\\\" | \\\"hasMany\\\"}>} - Relationship definitions. */\\n\"\n      fileContent += \"  static relationshipDefinitions() {\\n\"\n      fileContent += \"    return {\\n\"\n      for (const relationship of relationships) {\n        fileContent += `      ${relationship.relationshipName}: {type: ${JSON.stringify(relationship.type)}},\\n`\n      }\n      fileContent += \"    }\\n\"\n      fileContent += \"  }\\n\"\n\n      fileContent += \"\\n\"\n      fileContent += \"  /** @returns {Record<string, string>} - Relationship model class names. */\\n\"\n      fileContent += \"  static relationshipModelClasses() {\\n\"\n      fileContent += \"    return {\\n\"\n      for (const relationship of relationships) {\n        fileContent += `      ${relationship.relationshipName}: ${JSON.stringify(relationship.targetClassName)},\\n`\n      }\n      fileContent += \"    }\\n\"\n      fileContent += \"  }\\n\"\n    }\n\n    for (const attribute of attributes) {\n      const camelizedAttribute = inflection.camelize(attribute.name, true)\n      const camelizedAttributeUpper = inflection.camelize(attribute.name)\n\n      fileContent += \"\\n\"\n      fileContent += `  /** @returns {${attributesTypeName}[${JSON.stringify(attribute.name)}]} - Attribute value. */\\n`\n      fileContent += `  ${camelizedAttribute}() { return this.readAttribute(${JSON.stringify(attribute.name)}) }\\n`\n\n      fileContent += \"\\n\"\n      fileContent += \"  /**\\n\"\n      fileContent += `   * @param {${attributesTypeName}[${JSON.stringify(attribute.name)}]} newValue - New attribute value.\\n`\n      fileContent += `   * @returns {${attributesTypeName}[${JSON.stringify(attribute.name)}]} - Assigned value.\\n`\n      fileContent += \"   */\\n\"\n      fileContent += `  set${camelizedAttributeUpper}(newValue) { return this.setAttribute(${JSON.stringify(attribute.name)}, newValue) }\\n`\n    }\n\n    for (const methodName of Object.keys(collectionCommands)) {\n      fileContent += \"\\n\"\n      fileContent += \"  /**\\n\"\n      fileContent += \"   * @param {Record<string, any>} [payload={}] - Command payload.\\n\"\n      fileContent += \"   * @returns {Promise<Record<string, any>>} - Command response.\\n\"\n      fileContent += \"   */\\n\"\n      fileContent += `  static async ${methodName}(payload = {}) {\\n`\n      fileContent += \"    return await this.executeCustomCommand({\\n\"\n      fileContent += `      commandName: ${JSON.stringify(collectionCommands[methodName])},\\n`\n      fileContent += `      commandType: ${JSON.stringify(collectionCommands[methodName])},\\n`\n      fileContent += \"      payload,\\n\"\n      fileContent += \"      resourcePath: this.resourcePath()\\n\"\n      fileContent += \"    })\\n\"\n      fileContent += \"  }\\n\"\n    }\n\n    for (const methodName of Object.keys(memberCommands)) {\n      fileContent += \"\\n\"\n      fileContent += \"  /**\\n\"\n      fileContent += \"   * @param {Record<string, any>} [payload={}] - Command payload.\\n\"\n      fileContent += \"   * @returns {Promise<Record<string, any>>} - Command response.\\n\"\n      fileContent += \"   */\\n\"\n      fileContent += `  async ${methodName}(payload = {}) {\\n`\n      fileContent += `    return await ${className}.executeCustomCommand({\\n`\n      fileContent += `      commandName: ${JSON.stringify(memberCommands[methodName])},\\n`\n      fileContent += `      commandType: ${JSON.stringify(memberCommands[methodName])},\\n`\n      fileContent += \"      memberId: this.primaryKeyValue(),\\n\"\n      fileContent += \"      payload,\\n\"\n      fileContent += `      resourcePath: ${className}.resourcePath()\\n`\n      fileContent += \"    })\\n\"\n      fileContent += \"  }\\n\"\n    }\n\n    for (const relationship of relationships) {\n      const relationshipNameCamelized = inflection.camelize(relationship.relationshipName)\n      const targetImportPath = `./${relationship.targetFileName}.js`\n\n      if (relationship.type == \"hasMany\") {\n        fileContent += \"\\n\"\n        fileContent += `  /** @returns {import(${JSON.stringify(importPath)}).FrontendModelHasManyRelationship<typeof import(${JSON.stringify(`./${inflection.dasherize(inflection.underscore(className))}.js`)}).default, typeof import(${JSON.stringify(targetImportPath)}).default>} - Relationship helper. */\\n`\n        fileContent += `  ${relationship.relationshipName}() { return /** @type {import(${JSON.stringify(importPath)}).FrontendModelHasManyRelationship<typeof import(${JSON.stringify(`./${inflection.dasherize(inflection.underscore(className))}.js`)}).default, typeof import(${JSON.stringify(targetImportPath)}).default>} */ (this.getRelationshipByName(${JSON.stringify(relationship.relationshipName)})) }\\n`\n\n        fileContent += \"\\n\"\n        fileContent += `  /** @returns {Array<import(${JSON.stringify(targetImportPath)}).default>} - Loaded related models. */\\n`\n        fileContent += `  ${relationship.relationshipName}Loaded() { return /** @type {Array<import(${JSON.stringify(targetImportPath)}).default>} */ (this.getRelationshipByName(${JSON.stringify(relationship.relationshipName)}).loaded()) }\\n`\n\n        fileContent += \"\\n\"\n        fileContent += `  /** @returns {Promise<Array<import(${JSON.stringify(targetImportPath)}).default>>} - Loaded related models. */\\n`\n        fileContent += `  async load${relationshipNameCamelized}() { return /** @type {Promise<Array<import(${JSON.stringify(targetImportPath)}).default>>} */ (this.loadRelationship(${JSON.stringify(relationship.relationshipName)})) }\\n`\n      } else {\n        fileContent += \"\\n\"\n        fileContent += `  /** @returns {import(${JSON.stringify(targetImportPath)}).default | null} - Loaded related model. */\\n`\n        fileContent += `  ${relationship.relationshipName}() { return /** @type {import(${JSON.stringify(targetImportPath)}).default | null} */ (this.getRelationshipByName(${JSON.stringify(relationship.relationshipName)}).loaded()) }\\n`\n\n        fileContent += \"\\n\"\n        fileContent += \"  /**\\n\"\n        fileContent += `   * @param {Record<string, any>} [attributes] - Attributes for the new related model.\\n`\n        fileContent += `   * @returns {import(${JSON.stringify(targetImportPath)}).default} - Built related model.\\n`\n        fileContent += \"   */\\n\"\n        fileContent += `  build${relationshipNameCamelized}(attributes = {}) { return /** @type {import(${JSON.stringify(targetImportPath)}).default} */ (this.getRelationshipByName(${JSON.stringify(relationship.relationshipName)}).build(attributes)) }\\n`\n\n        fileContent += \"\\n\"\n        fileContent += `  /** @returns {Promise<import(${JSON.stringify(targetImportPath)}).default | null>} - Loaded related model. */\\n`\n        fileContent += `  async load${relationshipNameCamelized}() { return /** @type {Promise<import(${JSON.stringify(targetImportPath)}).default | null>} */ (this.loadRelationship(${JSON.stringify(relationship.relationshipName)})) }\\n`\n\n        fileContent += \"\\n\"\n        fileContent += `  /** @param {import(${JSON.stringify(targetImportPath)}).default | null} model - Related model. @returns {import(${JSON.stringify(targetImportPath)}).default | null} - Assigned related model. */\\n`\n        fileContent += `  set${relationshipNameCamelized}(model) { return /** @type {import(${JSON.stringify(targetImportPath)}).default | null} */ (this.setRelationship(${JSON.stringify(relationship.relationshipName)}, model)) }\\n`\n      }\n    }\n\n    fileContent += \"}\\n\"\n    fileContent += \"\\n\"\n    fileContent += `FrontendModelBase.registerModel(${className})\\n`\n\n    return fileContent\n  }\n\n  /**\n   * @param {Array<{className: string, fileName: string}>} generatedFiles - Generated model files.\n   * @returns {string} - Index file content that imports and re-exports all models.\n   */\n  buildIndexFileContent(generatedFiles) {\n    let content = \"\"\n\n    for (const {className, fileName} of generatedFiles) {\n      content += `export {default as ${className}} from \"./${fileName}\"\\n`\n    }\n\n    return content\n  }\n\n  /**\n   * @param {object} args - Formatting args.\n   * @param {string} args.indent - Base indentation.\n   * @param {string} args.propertyName - Object property name.\n   * @param {string[]} args.values - String values.\n   * @returns {string} - Formatted multiline array property.\n   */\n  formattedArrayProperty({indent, propertyName, values}) {\n    let output = `${indent}${propertyName}: [\\n`\n\n    for (const value of values) {\n      output += `${indent}  ${JSON.stringify(value)},\\n`\n    }\n\n    output += `${indent}],\\n`\n\n    return output\n  }\n\n  /**\n   * @param {object} args - Formatting args.\n   * @param {string} args.indent - Base indentation.\n   * @param {string} args.propertyName - Object property name.\n   * @param {Record<string, string>} args.values - Object key-values.\n   * @param {Record<string, string>} [args.filterDefaultValues] - Default values to omit from output.\n   * @returns {string} - Formatted multiline object property.\n   */\n  formattedObjectProperty({filterDefaultValues, indent, propertyName, values}) {\n    let output = `${indent}${propertyName}: {\\n`\n\n    for (const objectKey of Object.keys(values)) {\n      if (filterDefaultValues && filterDefaultValues[objectKey] === values[objectKey]) continue\n\n      output += `${indent}  ${objectKey}: ${JSON.stringify(values[objectKey])},\\n`\n    }\n\n    output += `${indent}},\\n`\n\n    return output\n  }\n\n  /**\n   * @param {object} args - Arguments.\n   * @param {typeof import(\"../../../../../database/record/index.js\").default | undefined} args.modelClass - Backend model class.\n   * @param {Record<string, any>} args.modelConfig - Model configuration.\n   * @returns {Array<{jsDocType: string, name: string}>} - Attribute definitions.\n   */\n  attributeDefinitionsForModel({modelClass, modelConfig}) {\n    const attributes = modelConfig.attributes\n\n    if (Array.isArray(attributes)) {\n      return attributes.map((attributeName) => ({\n        jsDocType: this.jsDocTypeForFrontendAttribute({\n          attributeConfig: this.frontendAttributeConfigForModelAttribute({attributeName, modelClass})\n        }),\n        name: attributeName\n      }))\n    }\n\n    if (!attributes || typeof attributes !== \"object\") {\n      throw new Error(`Expected 'attributes' as array or object but got: ${attributes}`)\n    }\n\n    return Object.keys(attributes).map((attributeName) => {\n      const attributeConfig = attributes[attributeName]\n\n      return {\n        jsDocType: this.jsDocTypeForFrontendAttribute({attributeConfig}),\n        name: attributeName\n      }\n    })\n  }\n\n  /**\n   * @param {object} args - Arguments.\n   * @param {any} args.attributeConfig - Attribute configuration value.\n   * @returns {string} - JSDoc type.\n   */\n  jsDocTypeForFrontendAttribute({attributeConfig}) {\n    const jsDocType = this.jsDocTypeForFrontendAttributeBaseType(attributeConfig)\n\n    if (!this.frontendAttributeCanBeNull(attributeConfig)) {\n      return jsDocType\n    }\n\n    return `${jsDocType} | null`\n  }\n\n  /**\n   * @param {any} attributeConfig - Attribute configuration value.\n   * @returns {string} - Non-nullable JSDoc type.\n   */\n  jsDocTypeForFrontendAttributeBaseType(attributeConfig) {\n    if (!attributeConfig || typeof attributeConfig !== \"object\") {\n      return \"any\"\n    }\n\n    const type = this.frontendAttributeTypeValue(attributeConfig)\n\n    if (type == \"boolean\") {\n      return \"boolean\"\n    } else if (type == \"json\" || type == \"jsonb\") {\n      return \"Record<string, any>\"\n    } else if ([\"blob\", \"char\", \"nvarchar\", \"varchar\", \"text\", \"longtext\", \"uuid\", \"character varying\"].includes(type)) {\n      return \"string\"\n    } else if ([\"bit\", \"bigint\", \"decimal\", \"double\", \"double precision\", \"float\", \"int\", \"integer\", \"numeric\", \"real\", \"smallint\", \"tinyint\"].includes(type)) {\n      return \"number\"\n    } else if ([\"date\", \"datetime\", \"timestamp\", \"timestamp without time zone\", \"timestamptz\"].includes(type)) {\n      return \"Date\"\n    } else {\n      return \"any\"\n    }\n  }\n\n  /**\n   * @param {any} attributeConfig - Attribute configuration value.\n   * @returns {boolean} - Whether the attribute allows null values.\n   */\n  frontendAttributeCanBeNull(attributeConfig) {\n    if (!attributeConfig || typeof attributeConfig !== \"object\") {\n      return false\n    }\n\n    if (typeof attributeConfig.getNull == \"function\") {\n      return attributeConfig.getNull() === true\n    }\n\n    return attributeConfig.null === true\n  }\n\n  /**\n   * @param {any} attributeConfig - Attribute configuration value.\n   * @returns {string | null} - Normalized column type.\n   */\n  frontendAttributeTypeValue(attributeConfig) {\n    if (!attributeConfig || typeof attributeConfig !== \"object\") {\n      return null\n    }\n\n    if (typeof attributeConfig.getType == \"function\") {\n      return String(attributeConfig.getType())\n    }\n\n    const typeValue = attributeConfig.type || attributeConfig.columnType || attributeConfig.sqlType || attributeConfig.dataType\n\n    if (typeof typeValue !== \"string\") {\n      return null\n    }\n\n    return typeValue\n  }\n\n  /**\n   * @param {object} args - Arguments.\n   * @param {string} args.attributeName - Frontend model attribute name.\n   * @param {typeof import(\"../../../../../database/record/index.js\").default | undefined} args.modelClass - Backend model class.\n   * @returns {any} - Attribute config inferred from the backend model when available.\n   */\n  frontendAttributeConfigForModelAttribute({attributeName, modelClass}) {\n    if (!modelClass) {\n      return null\n    }\n\n    const columnName = modelClass.getAttributeNameToColumnNameMap()[attributeName]\n\n    if (!columnName) {\n      return null\n    }\n\n    return modelClass.getColumnsHash()[columnName] || null\n  }\n\n  /**\n   * @param {object} args - Arguments.\n   * @param {string} args.className - Model class name.\n   * @param {Record<string, any>} args.modelConfig - Model configuration.\n   * @returns {Array<{relationshipName: string, targetClassName: string, targetFileName: string, type: \"belongsTo\" | \"hasOne\" | \"hasMany\"}>} - Relationships.\n   */\n  relationshipsForModel({className, modelConfig}) {\n    const relationships = modelConfig.relationships\n\n    if (relationships === undefined || relationships === null) {\n      return []\n    }\n\n    if (Array.isArray(relationships)) {\n      return relationships.map((relationshipName) => this.inferredRelationshipDefinition({className, relationshipName}))\n    }\n\n    if (typeof relationships !== \"object\") {\n      throw new Error(`Model '${className}' has invalid relationships config`)\n    }\n\n    /** @type {Array<{relationshipName: string, targetClassName: string, targetFileName: string, type: \"belongsTo\" | \"hasOne\" | \"hasMany\"}>} */\n    const normalized = []\n\n    for (const relationshipName in relationships) {\n      const relationship = relationships[relationshipName]\n\n      if (relationship === true) {\n        normalized.push(this.inferredRelationshipDefinition({className, relationshipName}))\n        continue\n      }\n\n      if (!relationship || typeof relationship !== \"object\" || Array.isArray(relationship)) {\n        throw new Error(`Model '${className}' relationship '${relationshipName}' must be an object or true`)\n      }\n\n      const relationshipModelName = relationship.modelClassName || relationship.className || relationship.model\n      const hasExplicitType = typeof relationship.type === \"string\" && relationship.type.length > 0\n      const hasExplicitModel = typeof relationshipModelName === \"string\" && relationshipModelName.length > 0\n      const inferredRelationship = hasExplicitType && hasExplicitModel\n        ? null\n        : this.inferredRelationshipDefinition({className, relationshipName})\n      const relationshipType = relationship.type || inferredRelationship?.type\n\n      if (relationshipType !== \"belongsTo\" && relationshipType !== \"hasOne\" && relationshipType !== \"hasMany\") {\n        throw new Error(`Model '${className}' relationship '${relationshipName}' has invalid type '${relationshipType}'`)\n      }\n\n      const targetClassName = hasExplicitModel\n        ? inflection.camelize(String(relationshipModelName).replaceAll(\"-\", \"_\"))\n        : inferredRelationship?.targetClassName\n\n      normalized.push({\n        relationshipName,\n        targetClassName,\n        targetFileName: inflection.dasherize(inflection.underscore(targetClassName)),\n        type: relationshipType\n      })\n    }\n\n    return normalized\n  }\n\n  /**\n   * @param {object} args - Arguments.\n   * @param {string} args.className - Model class name.\n   * @param {string} args.relationshipName - Relationship name.\n   * @returns {{relationshipName: string, targetClassName: string, targetFileName: string, type: \"belongsTo\" | \"hasOne\" | \"hasMany\"}} Inferred relationship definition.\n   */\n  inferredRelationshipDefinition({className, relationshipName}) {\n    const modelClass = this.getConfiguration().getModelClass(className)\n\n    if (!modelClass) {\n      throw new Error(`Could not find backend model class '${className}' for relationship '${relationshipName}'`)\n    }\n\n    const relationship = modelClass.getRelationshipByName(relationshipName)\n    const targetModelClass = relationship.getTargetModelClass()\n    const relationshipType = relationship.getType()\n\n    if (relationshipType !== \"belongsTo\" && relationshipType !== \"hasOne\" && relationshipType !== \"hasMany\") {\n      throw new Error(`Model '${className}' relationship '${relationshipName}' has unsupported type '${relationshipType}'`)\n    }\n\n    if (!targetModelClass) {\n      throw new Error(`Model '${className}' relationship '${relationshipName}' has no target model class`)\n    }\n\n    const targetClassName = targetModelClass.name\n\n    return {\n      relationshipName,\n      targetClassName,\n      targetFileName: inflection.dasherize(inflection.underscore(targetClassName)),\n      type: relationshipType\n    }\n  }\n}\n"]}
|
|
@@ -34,8 +34,6 @@ export default class FrontendModelBaseResource extends AuthorizationBaseResource
|
|
|
34
34
|
static memberCommands: Record<string, string> | string[] | undefined;
|
|
35
35
|
/** @type {Record<string, string> | string[] | undefined} */
|
|
36
36
|
static builtInMemberCommands: Record<string, string> | string[] | undefined;
|
|
37
|
-
/** @type {string | undefined} */
|
|
38
|
-
static path: string | undefined;
|
|
39
37
|
/** @type {Record<string, any> | undefined} */
|
|
40
38
|
static relationships: Record<string, any> | undefined;
|
|
41
39
|
/**
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"base-resource.d.ts","sourceRoot":"","sources":["../../../src/frontend-model-resource/base-resource.js"],"names":[],"mappings":"AAKA;;;;;;;GAOG;AAEH;;;;;;;;;GASG;AAEH;;GAEG;AACH;IACE,yDAAyD;IACzD,mBADW,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,GAAG,MAAM,EAAE,GAAG,SAAS,CACxB;IAC7B,iDAAiD;IACjD,kBADW,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,GAAG,SAAS,CACjB;IAC5B,8CAA8C;IAC9C,oBADW,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,GAAG,SAAS,CACZ;IAC9B,iDAAiD;IACjD,2BADW,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,GAAG,SAAS,CACR;IACrC,4DAA4D;IAC5D,kCADW,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,GAAG,MAAM,EAAE,GAAG,SAAS,CACZ;IAC5C,4DAA4D;IAC5D,uBADW,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,GAAG,MAAM,EAAE,GAAG,SAAS,CACvB;IACjC,4DAA4D;IAC5D,8BADW,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,GAAG,MAAM,EAAE,GAAG,SAAS,CAChB;IACxC,
|
|
1
|
+
{"version":3,"file":"base-resource.d.ts","sourceRoot":"","sources":["../../../src/frontend-model-resource/base-resource.js"],"names":[],"mappings":"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"}
|
|
@@ -37,8 +37,6 @@ export default class FrontendModelBaseResource extends AuthorizationBaseResource
|
|
|
37
37
|
static memberCommands = undefined;
|
|
38
38
|
/** @type {Record<string, string> | string[] | undefined} */
|
|
39
39
|
static builtInMemberCommands = undefined;
|
|
40
|
-
/** @type {string | undefined} */
|
|
41
|
-
static path = undefined;
|
|
42
40
|
/** @type {Record<string, any> | undefined} */
|
|
43
41
|
static relationships = undefined;
|
|
44
42
|
/**
|
|
@@ -86,8 +84,6 @@ export default class FrontendModelBaseResource extends AuthorizationBaseResource
|
|
|
86
84
|
config.collectionCommands = this.collectionCommands;
|
|
87
85
|
if (this.memberCommands)
|
|
88
86
|
config.memberCommands = this.memberCommands;
|
|
89
|
-
if (this.path)
|
|
90
|
-
config.path = this.path;
|
|
91
87
|
if (this.relationships)
|
|
92
88
|
config.relationships = this.relationships;
|
|
93
89
|
return config;
|
|
@@ -224,7 +220,13 @@ function filterWritableFrontendModelAttributes(receiver, attributes) {
|
|
|
224
220
|
*/
|
|
225
221
|
function normalizeFrontendModelResourceAbilities(abilities) {
|
|
226
222
|
if (!abilities) {
|
|
227
|
-
return {
|
|
223
|
+
return {
|
|
224
|
+
create: "create",
|
|
225
|
+
destroy: "destroy",
|
|
226
|
+
find: "read",
|
|
227
|
+
index: "read",
|
|
228
|
+
update: "update"
|
|
229
|
+
};
|
|
228
230
|
}
|
|
229
231
|
if (!Array.isArray(abilities)) {
|
|
230
232
|
return abilities;
|
|
@@ -251,4 +253,4 @@ function normalizeFrontendModelResourceAbilities(abilities) {
|
|
|
251
253
|
normalized.update = "update";
|
|
252
254
|
return normalized;
|
|
253
255
|
}
|
|
254
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"base-resource.js","sourceRoot":"","sources":["../../../src/frontend-model-resource/base-resource.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport AuthorizationBaseResource from \"../authorization/base-resource.js\"\nimport * as inflection from \"inflection\"\n\n/**\n * @typedef {object} FrontendModelResourceControllerArgs\n * @property {import(\"../controller.js\").default} controller - Frontend-model controller instance.\n * @property {typeof import(\"../database/record/index.js\").default} modelClass - Backing model class.\n * @property {string} modelName - Model name.\n * @property {Record<string, any>} params - Request params.\n * @property {import(\"../configuration-types.js\").FrontendModelResourceConfiguration} resourceConfiguration - Normalized resource configuration.\n */\n\n/**\n * @typedef {object} FrontendModelResourceAbilityArgs\n * @property {import(\"../authorization/ability.js\").default} [ability] - Ability instance when the resource is used directly for authorization.\n * @property {Record<string, any>} [context] - Ability context.\n * @property {Record<string, any>} [locals] - Ability locals.\n * @property {typeof import(\"../database/record/index.js\").default} [modelClass] - Optional backing model class override.\n * @property {string} [modelName] - Optional model name override.\n * @property {Record<string, any>} [params] - Optional params override.\n * @property {import(\"../configuration-types.js\").FrontendModelResourceConfiguration} [resourceConfiguration] - Optional normalized resource configuration.\n */\n\n/**\n * Base class for backend frontend-model resources.\n */\nexport default class FrontendModelBaseResource extends AuthorizationBaseResource {\n  /** @type {Record<string, any> | string[] | undefined} */\n  static attributes = undefined\n  /** @type {Record<string, string> | undefined} */\n  static abilities = undefined\n  /** @type {Record<string, any> | undefined} */\n  static attachments = undefined\n  /** @type {Record<string, string> | undefined} */\n  static collectionCommands = undefined\n  /** @type {Record<string, string> | string[] | undefined} */\n  static builtInCollectionCommands = undefined\n  /** @type {Record<string, string> | string[] | undefined} */\n  static memberCommands = undefined\n  /** @type {Record<string, string> | string[] | undefined} */\n  static builtInMemberCommands = undefined\n  /** @type {string | undefined} */\n  static path = undefined\n  /** @type {Record<string, any> | undefined} */\n  static relationships = undefined\n\n  /**\n   * @param {FrontendModelResourceAbilityArgs | FrontendModelResourceControllerArgs} args - Resource args.\n   */\n  constructor(args) {\n    super({\n      ability: \"ability\" in args ? args.ability : undefined,\n      context: \"context\" in args ? args.context || {} : {},\n      locals: \"locals\" in args ? args.locals || {} : {}\n    })\n\n    this.controller = \"controller\" in args ? args.controller : undefined\n    this.modelClassValue = \"modelClass\" in args ? args.modelClass : /** @type {typeof import(\"../database/record/index.js\").default | undefined} */ (/** @type {typeof FrontendModelBaseResource} */ (this.constructor).modelClass())\n    this.modelNameValue = \"modelName\" in args ? args.modelName : this.modelClassValue?.name || \"\"\n    this.paramsValue = \"params\" in args ? args.params : undefined\n    this.resourceConfigurationValue = \"resourceConfiguration\" in args ? args.resourceConfiguration : /** @type {import(\"../configuration-types.js\").FrontendModelResourceConfiguration} */ ({abilities: {}, attributes: []})\n  }\n\n  /**\n   * @returns {import(\"../controller.js\").default & {\n   *   frontendModelAuthorizedQuery: (action: \"index\" | \"find\" | \"create\" | \"update\" | \"destroy\" | \"attach\" | \"download\" | \"url\") => import(\"../database/query/model-class-query.js\").default<any>,\n   *   frontendModelIndexQuery: () => import(\"../database/query/model-class-query.js\").default<any>,\n   *   frontendModelPreload: () => import(\"../database/query/index.js\").NestedPreloadRecord | null,\n   *   serializeFrontendModel: (model: import(\"../database/record/index.js\").default) => Promise<Record<string, any>>\n   * }} - Controller instance with frontend-model helpers.\n   */\n  typedControllerInstance() {\n    return /** @type {any} */ (this.controller)\n  }\n\n  /**\n   * @returns {import(\"../configuration-types.js\").FrontendModelResourceConfiguration} - Static resource config.\n   */\n  static resourceConfig() {\n    /** @type {import(\"../configuration-types.js\").FrontendModelResourceConfiguration} */\n    const config = {\n      abilities: normalizeFrontendModelResourceAbilities(this.abilities),\n      attributes: this.attributes || []\n    }\n\n    if (this.attachments) config.attachments = this.attachments\n    if (this.builtInCollectionCommands) config.builtInCollectionCommands = this.builtInCollectionCommands\n    if (this.builtInMemberCommands) config.builtInMemberCommands = this.builtInMemberCommands\n    if (this.collectionCommands) config.collectionCommands = this.collectionCommands\n    if (this.memberCommands) config.memberCommands = this.memberCommands\n    if (this.path) config.path = this.path\n    if (this.relationships) config.relationships = this.relationships\n\n    return config\n  }\n\n  /** @returns {import(\"../controller.js\").default} - Controller instance. */\n  controllerInstance() { return this.controller }\n\n  /** @returns {typeof import(\"../database/record/index.js\").default} - Model class. */\n  modelClass() {\n    if (!this.modelClassValue) {\n      throw new Error(`${this.constructor.name} requires a model class.`)\n    }\n\n    return this.modelClassValue\n  }\n\n  /** @returns {string} - Model name. */\n  modelName() { return this.modelNameValue }\n\n  /** @returns {Record<string, any>} - Params. */\n  params() { return this.paramsValue || super.params() || {} }\n\n  /** @returns {import(\"../configuration-types.js\").FrontendModelResourceConfiguration} - Normalized resource config. */\n  resourceConfiguration() { return this.resourceConfigurationValue }\n\n  /** @returns {string} - Primary key. */\n  primaryKey() { return this.modelClass().primaryKey() }\n\n  /**\n   * @param {\"index\" | \"find\" | \"create\" | \"update\" | \"destroy\" | \"attach\" | \"download\" | \"url\"} action - Ability action.\n   * @returns {import(\"../database/query/model-class-query.js\").default<any>} - Authorized query.\n   */\n  authorizedQuery(action) {\n    return this.typedControllerInstance().frontendModelAuthorizedQuery(action)\n  }\n\n  /**\n   * @param {\"index\" | \"find\" | \"create\" | \"update\" | \"destroy\" | \"attach\" | \"download\" | \"url\"} action - Action.\n   * @returns {boolean | Promise<boolean>} - Whether pluck is supported.\n   */\n  supportsPluck(action) {\n    void action\n\n    return Object.getPrototypeOf(this).records === FrontendModelBaseResource.prototype.records\n  }\n\n  /**\n   * @param {\"index\" | \"find\" | \"create\" | \"update\" | \"destroy\" | \"attach\" | \"download\" | \"url\"} action - Action.\n   * @returns {boolean | void | Promise<boolean | void>} - Continue processing unless false.\n   */\n  beforeAction(action) {\n    void action\n\n    // No-op by default.\n  }\n\n  /**\n   * @returns {Promise<import(\"../database/record/index.js\").default[]>} - Records for index action.\n   */\n  async records() {\n    return await this.typedControllerInstance().frontendModelIndexQuery().toArray()\n  }\n\n  /**\n   * @param {\"find\" | \"update\" | \"destroy\" | \"attach\" | \"download\" | \"url\"} action - Action.\n   * @param {string | number} id - Record id.\n   * @returns {Promise<import(\"../database/record/index.js\").default | null>} - Located model.\n   */\n  async find(action, id) {\n    let query = this.authorizedQuery(action)\n    const preload = action === \"find\" ? this.typedControllerInstance().frontendModelPreload() : null\n\n    if (preload) {\n      query = query.preload(preload)\n    }\n\n    return await query.findBy({[this.primaryKey()]: id})\n  }\n\n  /**\n   * @param {Record<string, any>} attributes - Create attributes.\n   * @returns {Promise<import(\"../database/record/index.js\").default>} - Created model.\n   */\n  async create(attributes) {\n    return await this.modelClass().create(filterWritableFrontendModelAttributes(this.modelClass().prototype, attributes))\n  }\n\n  /**\n   * @param {import(\"../database/record/index.js\").default} model - Created model.\n   * @returns {Promise<void>} - Cleanup after failed authorization.\n   */\n  async handleUnauthorizedCreatedModel(model) {\n    await model.destroy()\n  }\n\n  /**\n   * @param {import(\"../database/record/index.js\").default} model - Existing model.\n   * @param {Record<string, any>} attributes - Update attributes.\n   * @returns {Promise<import(\"../database/record/index.js\").default>} - Updated model.\n   */\n  async update(model, attributes) {\n    model.assign(filterWritableFrontendModelAttributes(model, attributes))\n    await model.save()\n\n    return model\n  }\n\n  /**\n   * @param {import(\"../database/record/index.js\").default} model - Existing model.\n   * @returns {Promise<void>} - No return value.\n   */\n  async destroy(model) {\n    await model.destroy()\n  }\n\n  /**\n   * @param {import(\"../database/record/index.js\").default} model - Model to serialize.\n   * @param {\"index\" | \"find\" | \"create\" | \"update\"} [action] - Action.\n   * @returns {Promise<Record<string, any>>} - Serialized model payload.\n   */\n  async serialize(model, action) {\n    void action\n\n    return await this.typedControllerInstance().serializeFrontendModel(model)\n  }\n}\n\n/**\n * @param {Record<string, any>} receiver - Model instance or prototype.\n * @param {Record<string, any>} attributes - Incoming frontend-model attributes.\n * @returns {Record<string, any>} - Writable attributes only.\n */\nfunction filterWritableFrontendModelAttributes(receiver, attributes) {\n  // Frontend-model writes should fail fast when callers submit read-only or unknown attrs.\n  // Silent drops hide contract mistakes in generated models and app-side wrapper code.\n  /** @type {Record<string, any>} */\n  const writableAttributes = {}\n  /** @type {string[]} */\n  const invalidAttributes = []\n\n  for (const [attributeName, value] of Object.entries(attributes)) {\n    const setterName = `set${inflection.camelize(attributeName)}`\n\n    if (setterName in receiver) {\n      writableAttributes[attributeName] = value\n    } else {\n      invalidAttributes.push(attributeName)\n    }\n  }\n\n  if (invalidAttributes.length > 0) {\n    throw new Error(`Invalid frontend model write attributes: ${invalidAttributes.join(\", \")}`)\n  }\n\n  return writableAttributes\n}\n\n/**\n * @param {Record<string, string> | string[] | undefined} abilities - Resource abilities config.\n * @returns {Record<string, string>} - Normalized abilities config.\n */\nfunction normalizeFrontendModelResourceAbilities(abilities) {\n  if (!abilities) {\n    return {}\n  }\n\n  if (!Array.isArray(abilities)) {\n    return abilities\n  }\n\n  /** @type {Record<string, string>} */\n  const normalized = {}\n\n  if (abilities.includes(\"manage\")) {\n    normalized.create = \"manage\"\n    normalized.destroy = \"manage\"\n    normalized.find = \"manage\"\n    normalized.index = \"manage\"\n    normalized.update = \"manage\"\n\n    return normalized\n  }\n\n  if (abilities.includes(\"create\")) normalized.create = \"create\"\n  if (abilities.includes(\"destroy\")) normalized.destroy = \"destroy\"\n  if (abilities.includes(\"read\")) {\n    normalized.find = \"read\"\n    normalized.index = \"read\"\n  }\n  if (abilities.includes(\"update\")) normalized.update = \"update\"\n\n  return normalized\n}\n"]}
|
|
256
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"base-resource.js","sourceRoot":"","sources":["../../../src/frontend-model-resource/base-resource.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport AuthorizationBaseResource from \"../authorization/base-resource.js\"\nimport * as inflection from \"inflection\"\n\n/**\n * @typedef {object} FrontendModelResourceControllerArgs\n * @property {import(\"../controller.js\").default} controller - Frontend-model controller instance.\n * @property {typeof import(\"../database/record/index.js\").default} modelClass - Backing model class.\n * @property {string} modelName - Model name.\n * @property {Record<string, any>} params - Request params.\n * @property {import(\"../configuration-types.js\").FrontendModelResourceConfiguration} resourceConfiguration - Normalized resource configuration.\n */\n\n/**\n * @typedef {object} FrontendModelResourceAbilityArgs\n * @property {import(\"../authorization/ability.js\").default} [ability] - Ability instance when the resource is used directly for authorization.\n * @property {Record<string, any>} [context] - Ability context.\n * @property {Record<string, any>} [locals] - Ability locals.\n * @property {typeof import(\"../database/record/index.js\").default} [modelClass] - Optional backing model class override.\n * @property {string} [modelName] - Optional model name override.\n * @property {Record<string, any>} [params] - Optional params override.\n * @property {import(\"../configuration-types.js\").FrontendModelResourceConfiguration} [resourceConfiguration] - Optional normalized resource configuration.\n */\n\n/**\n * Base class for backend frontend-model resources.\n */\nexport default class FrontendModelBaseResource extends AuthorizationBaseResource {\n  /** @type {Record<string, any> | string[] | undefined} */\n  static attributes = undefined\n  /** @type {Record<string, string> | undefined} */\n  static abilities = undefined\n  /** @type {Record<string, any> | undefined} */\n  static attachments = undefined\n  /** @type {Record<string, string> | undefined} */\n  static collectionCommands = undefined\n  /** @type {Record<string, string> | string[] | undefined} */\n  static builtInCollectionCommands = undefined\n  /** @type {Record<string, string> | string[] | undefined} */\n  static memberCommands = undefined\n  /** @type {Record<string, string> | string[] | undefined} */\n  static builtInMemberCommands = undefined\n  /** @type {Record<string, any> | undefined} */\n  static relationships = undefined\n\n  /**\n   * @param {FrontendModelResourceAbilityArgs | FrontendModelResourceControllerArgs} args - Resource args.\n   */\n  constructor(args) {\n    super({\n      ability: \"ability\" in args ? args.ability : undefined,\n      context: \"context\" in args ? args.context || {} : {},\n      locals: \"locals\" in args ? args.locals || {} : {}\n    })\n\n    this.controller = \"controller\" in args ? args.controller : undefined\n    this.modelClassValue = \"modelClass\" in args ? args.modelClass : /** @type {typeof import(\"../database/record/index.js\").default | undefined} */ (/** @type {typeof FrontendModelBaseResource} */ (this.constructor).modelClass())\n    this.modelNameValue = \"modelName\" in args ? args.modelName : this.modelClassValue?.name || \"\"\n    this.paramsValue = \"params\" in args ? args.params : undefined\n    this.resourceConfigurationValue = \"resourceConfiguration\" in args ? args.resourceConfiguration : /** @type {import(\"../configuration-types.js\").FrontendModelResourceConfiguration} */ ({abilities: {}, attributes: []})\n  }\n\n  /**\n   * @returns {import(\"../controller.js\").default & {\n   *   frontendModelAuthorizedQuery: (action: \"index\" | \"find\" | \"create\" | \"update\" | \"destroy\" | \"attach\" | \"download\" | \"url\") => import(\"../database/query/model-class-query.js\").default<any>,\n   *   frontendModelIndexQuery: () => import(\"../database/query/model-class-query.js\").default<any>,\n   *   frontendModelPreload: () => import(\"../database/query/index.js\").NestedPreloadRecord | null,\n   *   serializeFrontendModel: (model: import(\"../database/record/index.js\").default) => Promise<Record<string, any>>\n   * }} - Controller instance with frontend-model helpers.\n   */\n  typedControllerInstance() {\n    return /** @type {any} */ (this.controller)\n  }\n\n  /**\n   * @returns {import(\"../configuration-types.js\").FrontendModelResourceConfiguration} - Static resource config.\n   */\n  static resourceConfig() {\n    /** @type {import(\"../configuration-types.js\").FrontendModelResourceConfiguration} */\n    const config = {\n      abilities: normalizeFrontendModelResourceAbilities(this.abilities),\n      attributes: this.attributes || []\n    }\n\n    if (this.attachments) config.attachments = this.attachments\n    if (this.builtInCollectionCommands) config.builtInCollectionCommands = this.builtInCollectionCommands\n    if (this.builtInMemberCommands) config.builtInMemberCommands = this.builtInMemberCommands\n    if (this.collectionCommands) config.collectionCommands = this.collectionCommands\n    if (this.memberCommands) config.memberCommands = this.memberCommands\n    if (this.relationships) config.relationships = this.relationships\n\n    return config\n  }\n\n  /** @returns {import(\"../controller.js\").default} - Controller instance. */\n  controllerInstance() { return this.controller }\n\n  /** @returns {typeof import(\"../database/record/index.js\").default} - Model class. */\n  modelClass() {\n    if (!this.modelClassValue) {\n      throw new Error(`${this.constructor.name} requires a model class.`)\n    }\n\n    return this.modelClassValue\n  }\n\n  /** @returns {string} - Model name. */\n  modelName() { return this.modelNameValue }\n\n  /** @returns {Record<string, any>} - Params. */\n  params() { return this.paramsValue || super.params() || {} }\n\n  /** @returns {import(\"../configuration-types.js\").FrontendModelResourceConfiguration} - Normalized resource config. */\n  resourceConfiguration() { return this.resourceConfigurationValue }\n\n  /** @returns {string} - Primary key. */\n  primaryKey() { return this.modelClass().primaryKey() }\n\n  /**\n   * @param {\"index\" | \"find\" | \"create\" | \"update\" | \"destroy\" | \"attach\" | \"download\" | \"url\"} action - Ability action.\n   * @returns {import(\"../database/query/model-class-query.js\").default<any>} - Authorized query.\n   */\n  authorizedQuery(action) {\n    return this.typedControllerInstance().frontendModelAuthorizedQuery(action)\n  }\n\n  /**\n   * @param {\"index\" | \"find\" | \"create\" | \"update\" | \"destroy\" | \"attach\" | \"download\" | \"url\"} action - Action.\n   * @returns {boolean | Promise<boolean>} - Whether pluck is supported.\n   */\n  supportsPluck(action) {\n    void action\n\n    return Object.getPrototypeOf(this).records === FrontendModelBaseResource.prototype.records\n  }\n\n  /**\n   * @param {\"index\" | \"find\" | \"create\" | \"update\" | \"destroy\" | \"attach\" | \"download\" | \"url\"} action - Action.\n   * @returns {boolean | void | Promise<boolean | void>} - Continue processing unless false.\n   */\n  beforeAction(action) {\n    void action\n\n    // No-op by default.\n  }\n\n  /**\n   * @returns {Promise<import(\"../database/record/index.js\").default[]>} - Records for index action.\n   */\n  async records() {\n    return await this.typedControllerInstance().frontendModelIndexQuery().toArray()\n  }\n\n  /**\n   * @param {\"find\" | \"update\" | \"destroy\" | \"attach\" | \"download\" | \"url\"} action - Action.\n   * @param {string | number} id - Record id.\n   * @returns {Promise<import(\"../database/record/index.js\").default | null>} - Located model.\n   */\n  async find(action, id) {\n    let query = this.authorizedQuery(action)\n    const preload = action === \"find\" ? this.typedControllerInstance().frontendModelPreload() : null\n\n    if (preload) {\n      query = query.preload(preload)\n    }\n\n    return await query.findBy({[this.primaryKey()]: id})\n  }\n\n  /**\n   * @param {Record<string, any>} attributes - Create attributes.\n   * @returns {Promise<import(\"../database/record/index.js\").default>} - Created model.\n   */\n  async create(attributes) {\n    return await this.modelClass().create(filterWritableFrontendModelAttributes(this.modelClass().prototype, attributes))\n  }\n\n  /**\n   * @param {import(\"../database/record/index.js\").default} model - Created model.\n   * @returns {Promise<void>} - Cleanup after failed authorization.\n   */\n  async handleUnauthorizedCreatedModel(model) {\n    await model.destroy()\n  }\n\n  /**\n   * @param {import(\"../database/record/index.js\").default} model - Existing model.\n   * @param {Record<string, any>} attributes - Update attributes.\n   * @returns {Promise<import(\"../database/record/index.js\").default>} - Updated model.\n   */\n  async update(model, attributes) {\n    model.assign(filterWritableFrontendModelAttributes(model, attributes))\n    await model.save()\n\n    return model\n  }\n\n  /**\n   * @param {import(\"../database/record/index.js\").default} model - Existing model.\n   * @returns {Promise<void>} - No return value.\n   */\n  async destroy(model) {\n    await model.destroy()\n  }\n\n  /**\n   * @param {import(\"../database/record/index.js\").default} model - Model to serialize.\n   * @param {\"index\" | \"find\" | \"create\" | \"update\"} [action] - Action.\n   * @returns {Promise<Record<string, any>>} - Serialized model payload.\n   */\n  async serialize(model, action) {\n    void action\n\n    return await this.typedControllerInstance().serializeFrontendModel(model)\n  }\n}\n\n/**\n * @param {Record<string, any>} receiver - Model instance or prototype.\n * @param {Record<string, any>} attributes - Incoming frontend-model attributes.\n * @returns {Record<string, any>} - Writable attributes only.\n */\nfunction filterWritableFrontendModelAttributes(receiver, attributes) {\n  // Frontend-model writes should fail fast when callers submit read-only or unknown attrs.\n  // Silent drops hide contract mistakes in generated models and app-side wrapper code.\n  /** @type {Record<string, any>} */\n  const writableAttributes = {}\n  /** @type {string[]} */\n  const invalidAttributes = []\n\n  for (const [attributeName, value] of Object.entries(attributes)) {\n    const setterName = `set${inflection.camelize(attributeName)}`\n\n    if (setterName in receiver) {\n      writableAttributes[attributeName] = value\n    } else {\n      invalidAttributes.push(attributeName)\n    }\n  }\n\n  if (invalidAttributes.length > 0) {\n    throw new Error(`Invalid frontend model write attributes: ${invalidAttributes.join(\", \")}`)\n  }\n\n  return writableAttributes\n}\n\n/**\n * @param {Record<string, string> | string[] | undefined} abilities - Resource abilities config.\n * @returns {Record<string, string>} - Normalized abilities config.\n */\nfunction normalizeFrontendModelResourceAbilities(abilities) {\n  if (!abilities) {\n    return {\n      create: \"create\",\n      destroy: \"destroy\",\n      find: \"read\",\n      index: \"read\",\n      update: \"update\"\n    }\n  }\n\n  if (!Array.isArray(abilities)) {\n    return abilities\n  }\n\n  /** @type {Record<string, string>} */\n  const normalized = {}\n\n  if (abilities.includes(\"manage\")) {\n    normalized.create = \"manage\"\n    normalized.destroy = \"manage\"\n    normalized.find = \"manage\"\n    normalized.index = \"manage\"\n    normalized.update = \"manage\"\n\n    return normalized\n  }\n\n  if (abilities.includes(\"create\")) normalized.create = \"create\"\n  if (abilities.includes(\"destroy\")) normalized.destroy = \"destroy\"\n  if (abilities.includes(\"read\")) {\n    normalized.find = \"read\"\n    normalized.index = \"read\"\n  }\n  if (abilities.includes(\"update\")) normalized.update = \"update\"\n\n  return normalized\n}\n"]}
|