velocious 1.0.320 → 1.0.321
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/configuration-types.d.ts +0 -9
- package/build/src/configuration-types.d.ts.map +1 -1
- package/build/src/configuration-types.js +1 -2
- package/build/src/environment-handlers/node/cli/commands/generate/frontend-models.d.ts +13 -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 +60 -1
- package/build/src/frontend-model-resource/base-resource.d.ts +50 -49
- package/build/src/frontend-model-resource/base-resource.d.ts.map +1 -1
- package/build/src/frontend-model-resource/base-resource.js +131 -90
- package/build/src/frontend-models/resource-definition.d.ts.map +1 -1
- package/build/src/frontend-models/resource-definition.js +1 -2
- package/package.json +1 -1
|
@@ -249,6 +249,14 @@ export default class DbGenerateFrontendModels extends BaseCommand {
|
|
|
249
249
|
if (modelClass && modelClass.primaryKey() !== "id") {
|
|
250
250
|
fileContent += ` primaryKey: ${JSON.stringify(modelClass.primaryKey())},\n`;
|
|
251
251
|
}
|
|
252
|
+
const nestedRelationshipNames = this.nestedRelationshipNamesForGenerator(resourceClass || null);
|
|
253
|
+
if (nestedRelationshipNames.length > 0) {
|
|
254
|
+
fileContent += " nestedAttributes: {\n";
|
|
255
|
+
for (const relationshipName of nestedRelationshipNames) {
|
|
256
|
+
fileContent += ` ${relationshipName}: {},\n`;
|
|
257
|
+
}
|
|
258
|
+
fileContent += " },\n";
|
|
259
|
+
}
|
|
252
260
|
fileContent += " }\n";
|
|
253
261
|
fileContent += " }\n";
|
|
254
262
|
if (relationships.length > 0) {
|
|
@@ -378,6 +386,57 @@ export default class DbGenerateFrontendModels extends BaseCommand {
|
|
|
378
386
|
}
|
|
379
387
|
return content;
|
|
380
388
|
}
|
|
389
|
+
/**
|
|
390
|
+
* Invokes a backend resource's `permittedParams()` instance method at
|
|
391
|
+
* generation time and extracts the relationship names that accept
|
|
392
|
+
* nested writes (`{fooAttributes: [...]}` entries). The generator
|
|
393
|
+
* emits those names into the frontend model's `resourceConfig()` so
|
|
394
|
+
* the client `save()` walker knows which relationships to ship.
|
|
395
|
+
*
|
|
396
|
+
* Constructed with no controller/ability so resource overrides must
|
|
397
|
+
* support being called without a request context.
|
|
398
|
+
* @param {typeof import("../../../../../frontend-model-resource/base-resource.js").default | null} resourceClass - Resource class.
|
|
399
|
+
* @returns {string[]} - Relationship names that accept nested writes (empty when none).
|
|
400
|
+
*/
|
|
401
|
+
nestedRelationshipNamesForGenerator(resourceClass) {
|
|
402
|
+
if (!resourceClass || typeof resourceClass !== "function")
|
|
403
|
+
return [];
|
|
404
|
+
const prototypeWithMethod = /** @type {{permittedParams?: (arg?: object) => Array<string | Record<string, any>>}} */ (resourceClass.prototype);
|
|
405
|
+
if (typeof prototypeWithMethod?.permittedParams !== "function")
|
|
406
|
+
return [];
|
|
407
|
+
let spec;
|
|
408
|
+
try {
|
|
409
|
+
const instance = new resourceClass({
|
|
410
|
+
ability: undefined,
|
|
411
|
+
context: {},
|
|
412
|
+
locals: {},
|
|
413
|
+
modelClass: resourceClass.ModelClass,
|
|
414
|
+
modelName: resourceClass.ModelClass?.getModelName?.() || resourceClass.name,
|
|
415
|
+
params: {},
|
|
416
|
+
resourceConfiguration: /** @type {import("../../../../../configuration-types.js").FrontendModelResourceConfiguration} */ ({ abilities: {}, attributes: [] })
|
|
417
|
+
});
|
|
418
|
+
spec = instance.permittedParams();
|
|
419
|
+
}
|
|
420
|
+
catch (error) {
|
|
421
|
+
throw new Error(`Failed to invoke ${resourceClass.name}.permittedParams() while generating frontend models: ${error instanceof Error ? error.message : String(error)}`, { cause: error });
|
|
422
|
+
}
|
|
423
|
+
if (!Array.isArray(spec))
|
|
424
|
+
return [];
|
|
425
|
+
/** @type {string[]} */
|
|
426
|
+
const relationshipNames = [];
|
|
427
|
+
for (const entry of spec) {
|
|
428
|
+
if (!entry || typeof entry !== "object" || Array.isArray(entry))
|
|
429
|
+
continue;
|
|
430
|
+
for (const key of Object.keys(entry)) {
|
|
431
|
+
if (!key.endsWith("Attributes"))
|
|
432
|
+
continue;
|
|
433
|
+
const name = key.slice(0, -"Attributes".length);
|
|
434
|
+
if (name)
|
|
435
|
+
relationshipNames.push(name);
|
|
436
|
+
}
|
|
437
|
+
}
|
|
438
|
+
return relationshipNames;
|
|
439
|
+
}
|
|
381
440
|
/**
|
|
382
441
|
* @param {object} args - Formatting args.
|
|
383
442
|
* @param {string} args.indent - Base indentation.
|
|
@@ -617,4 +676,4 @@ export default class DbGenerateFrontendModels extends BaseCommand {
|
|
|
617
676
|
};
|
|
618
677
|
}
|
|
619
678
|
}
|
|
620
|
-
//# 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 {frontendModelResourceClassFromDefinition, 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    const environmentHandler = configuration.getEnvironmentHandler()\n\n    if (typeof environmentHandler.autoDiscoverResources === \"function\") {\n      await environmentHandler.autoDiscoverResources(configuration)\n    }\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\n      if (!generatedFiles) throw new Error(`Generated files list missing for ${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, resourceClass: frontendModelResourceClassFromDefinition(resources[modelClassName])})\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          resourceClass: frontendModelResourceClassFromDefinition(resources[modelClassName])\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      const setupContent = this.buildSetupFileContent(generatedFiles)\n\n      await fs.writeFile(`${frontendModelsDir}/setup.js`, setupContent)\n\n      console.log(\"create src/frontend-models/setup.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   * @param {typeof import(\"../../../../../frontend-model-resource/base-resource.js\").default | null} [args.resourceClass]\n   * @returns {void} - No return value.\n   */\n  validateModelConfig({availableFrontendModelClassNames, className, modelConfig, resourceClass}) {\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, resourceClass})\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   * @param {typeof import(\"../../../../../frontend-model-resource/base-resource.js\").default | null} [args.resourceClass]\n   * @returns {string} - Generated file content.\n   */\n  buildModelFileContent({className, importPath, modelClass, modelConfig, resourceClass}) {\n    const attributes = this.attributeDefinitionsForModel({modelClass, modelConfig})\n    const relationships = this.relationshipsForModel({className, modelConfig, resourceClass})\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 += `/** @typedef {import(\"${importPath}\").FrontendModelResourceConfig} FrontendModelResourceConfig */\\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 {FrontendModelResourceConfig} - Resource config. */\\n\"\n    fileContent += \"  static resourceConfig() {\\n\"\n    fileContent += \"    return {\\n\"\n    fileContent += `      modelName: ${JSON.stringify(className)},\\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.formattedCommandsProperty({\n        indent: \"      \",\n        propertyName: \"collectionCommands\",\n        values: collectionCommands\n      })\n    }\n    if (Object.keys(memberCommands).length > 0) {\n      fileContent += this.formattedCommandsProperty({\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 {...any} commandArguments - Custom command arguments.\\n\"\n      fileContent += \"   * @returns {Promise<Record<string, any>>} - Command response.\\n\"\n      fileContent += \"   */\\n\"\n      fileContent += `  static async ${methodName}(...commandArguments) {\\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: ${className}.normalizeCustomCommandPayloadArguments(commandArguments),\\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 {...any} commandArguments - Custom command arguments.\\n\"\n      fileContent += \"   * @returns {Promise<Record<string, any>>} - Command response.\\n\"\n      fileContent += \"   */\\n\"\n      fileContent += `  async ${methodName}(...commandArguments) {\\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: ${className}.normalizeCustomCommandPayloadArguments(commandArguments),\\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 += `  /** @returns {Promise<import(${JSON.stringify(targetImportPath)}).default | null>} - Loaded related model. */\\n`\n        fileContent += `  async ${relationship.relationshipName}OrLoad() { return /** @type {Promise<import(${JSON.stringify(targetImportPath)}).default | null>} */ (this.relationshipOrLoad(${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 {Array<{className: string, fileName: string}>} generatedFiles - Generated model files.\n   * @returns {string} - Setup file content with side-effect imports for model registration.\n   */\n  buildSetupFileContent(generatedFiles) {\n    let content = \"// This file is auto-generated by Velocious. Do not edit manually.\\n\"\n\n    content += \"// Run `velocious g:frontend-models` to regenerate.\\n\"\n\n    for (const {fileName} of generatedFiles) {\n      content += `import \"./${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  /**\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 - Command key-values.\n   * @returns {string} - Formatted property (array when keys match values, object otherwise).\n   */\n  formattedCommandsProperty({indent, propertyName, values}) {\n    const allKeysMatchValues = Object.entries(values).every(([key, value]) => key === value)\n\n    if (allKeysMatchValues) {\n      return this.formattedArrayProperty({indent, propertyName, values: Object.keys(values)})\n    }\n\n    return this.formattedObjectProperty({indent, propertyName, values})\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    let attributes = modelConfig.attributes\n\n    // Auto-derive attributes from model columns when not explicitly defined\n    if ((!attributes || (Array.isArray(attributes) && attributes.length === 0)) && modelClass) {\n      try {\n        const columns = modelClass.getColumns()\n\n        if (Array.isArray(columns)) {\n          attributes = columns.map((column) => inflection.camelize(column.getName(), true))\n        }\n      } catch {\n        // Model may not be initialized yet\n      }\n    }\n\n    if (Array.isArray(attributes)) {\n      return attributes.map((entry) => {\n        const attributeName = typeof entry === \"string\" ? entry : entry.name\n\n        return {\n          jsDocType: this.jsDocTypeForFrontendAttribute({\n            attributeConfig: this.frontendAttributeConfigForModelAttribute({attributeName, modelClass})\n          }),\n          name: attributeName\n        }\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 (type && [\"blob\", \"char\", \"nvarchar\", \"varchar\", \"text\", \"longtext\", \"uuid\", \"character varying\"].includes(type)) {\n      return \"string\"\n    } else if (type && [\"bit\", \"bigint\", \"decimal\", \"double\", \"double precision\", \"float\", \"int\", \"integer\", \"numeric\", \"real\", \"smallint\", \"tinyint\"].includes(type)) {\n      return \"number\"\n    } else if (type && [\"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   * @param {typeof import(\"../../../../../frontend-model-resource/base-resource.js\").default | null} [args.resourceClass]\n   * @returns {Array<{relationshipName: string, targetClassName: string, targetFileName: string, type: \"belongsTo\" | \"hasOne\" | \"hasMany\"}>} - Relationships.\n   */\n  relationshipsForModel({className, modelConfig, resourceClass}) {\n    const relationships = modelConfig.relationships\n\n    if (relationships === undefined || relationships === null) {\n      return []\n    }\n\n    if (!Array.isArray(relationships)) {\n      throw new Error(`Model '${className}' has invalid relationships config — must be an array of relationship names, got ${typeof relationships}`)\n    }\n\n    return relationships.map((relationshipName) => this.inferredRelationshipDefinition({className, relationshipName, resourceClass}))\n  }\n\n  /**\n   * @param {object} args - Arguments.\n   * @param {string} args.className - Model class name.\n   * @param {string} args.relationshipName - Relationship name.\n   * @param {typeof import(\"../../../../../frontend-model-resource/base-resource.js\").default | null} [args.resourceClass]\n   * @returns {{relationshipName: string, targetClassName: string, targetFileName: string, type: \"belongsTo\" | \"hasOne\" | \"hasMany\"}} Inferred relationship definition.\n   */\n  inferredRelationshipDefinition({className, relationshipName, resourceClass}) {\n    const modelClass = resourceClass?.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 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    let targetClassName\n\n    try {\n      const targetModelClass = relationship.getTargetModelClass()\n\n      targetClassName = targetModelClass?.getModelName()\n    } catch {\n      // Model class not registered yet — fall back to className from relationship definition\n    }\n\n    if (!targetClassName) {\n      targetClassName = relationship.className\n\n      if (!targetClassName) {\n        throw new Error(`Model '${className}' relationship '${relationshipName}' has no target model class`)\n      }\n    }\n\n    return {\n      relationshipName,\n      targetClassName,\n      targetFileName: inflection.dasherize(inflection.underscore(targetClassName)),\n      type: relationshipType\n    }\n  }\n}\n"]}
|
|
679
|
+
//# 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 {frontendModelResourceClassFromDefinition, frontendModelResourceConfigurationFromDefinition, frontendModelResourcesForBackendProject} from \"../../../../../frontend-models/resource-definition.js\"\n\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    const environmentHandler = configuration.getEnvironmentHandler()\n\n    if (typeof environmentHandler.autoDiscoverResources === \"function\") {\n      await environmentHandler.autoDiscoverResources(configuration)\n    }\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\n      if (!generatedFiles) throw new Error(`Generated files list missing for ${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, resourceClass: frontendModelResourceClassFromDefinition(resources[modelClassName])})\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          resourceClass: frontendModelResourceClassFromDefinition(resources[modelClassName])\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      const setupContent = this.buildSetupFileContent(generatedFiles)\n\n      await fs.writeFile(`${frontendModelsDir}/setup.js`, setupContent)\n\n      console.log(\"create src/frontend-models/setup.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   * @param {typeof import(\"../../../../../frontend-model-resource/base-resource.js\").default | null} [args.resourceClass]\n   * @returns {void} - No return value.\n   */\n  validateModelConfig({availableFrontendModelClassNames, className, modelConfig, resourceClass}) {\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, resourceClass})\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   * @param {typeof import(\"../../../../../frontend-model-resource/base-resource.js\").default | null} [args.resourceClass]\n   * @returns {string} - Generated file content.\n   */\n  buildModelFileContent({className, importPath, modelClass, modelConfig, resourceClass}) {\n    const attributes = this.attributeDefinitionsForModel({modelClass, modelConfig})\n    const relationships = this.relationshipsForModel({className, modelConfig, resourceClass})\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 += `/** @typedef {import(\"${importPath}\").FrontendModelResourceConfig} FrontendModelResourceConfig */\\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 {FrontendModelResourceConfig} - Resource config. */\\n\"\n    fileContent += \"  static resourceConfig() {\\n\"\n    fileContent += \"    return {\\n\"\n    fileContent += `      modelName: ${JSON.stringify(className)},\\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.formattedCommandsProperty({\n        indent: \"      \",\n        propertyName: \"collectionCommands\",\n        values: collectionCommands\n      })\n    }\n    if (Object.keys(memberCommands).length > 0) {\n      fileContent += this.formattedCommandsProperty({\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    const nestedRelationshipNames = this.nestedRelationshipNamesForGenerator(resourceClass || null)\n    if (nestedRelationshipNames.length > 0) {\n      fileContent += \"      nestedAttributes: {\\n\"\n      for (const relationshipName of nestedRelationshipNames) {\n        fileContent += `        ${relationshipName}: {},\\n`\n      }\n      fileContent += \"      },\\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 {...any} commandArguments - Custom command arguments.\\n\"\n      fileContent += \"   * @returns {Promise<Record<string, any>>} - Command response.\\n\"\n      fileContent += \"   */\\n\"\n      fileContent += `  static async ${methodName}(...commandArguments) {\\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: ${className}.normalizeCustomCommandPayloadArguments(commandArguments),\\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 {...any} commandArguments - Custom command arguments.\\n\"\n      fileContent += \"   * @returns {Promise<Record<string, any>>} - Command response.\\n\"\n      fileContent += \"   */\\n\"\n      fileContent += `  async ${methodName}(...commandArguments) {\\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: ${className}.normalizeCustomCommandPayloadArguments(commandArguments),\\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 += `  /** @returns {Promise<import(${JSON.stringify(targetImportPath)}).default | null>} - Loaded related model. */\\n`\n        fileContent += `  async ${relationship.relationshipName}OrLoad() { return /** @type {Promise<import(${JSON.stringify(targetImportPath)}).default | null>} */ (this.relationshipOrLoad(${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 {Array<{className: string, fileName: string}>} generatedFiles - Generated model files.\n   * @returns {string} - Setup file content with side-effect imports for model registration.\n   */\n  buildSetupFileContent(generatedFiles) {\n    let content = \"// This file is auto-generated by Velocious. Do not edit manually.\\n\"\n\n    content += \"// Run `velocious g:frontend-models` to regenerate.\\n\"\n\n    for (const {fileName} of generatedFiles) {\n      content += `import \"./${fileName}\"\\n`\n    }\n\n    return content\n  }\n\n  /**\n   * Invokes a backend resource's `permittedParams()` instance method at\n   * generation time and extracts the relationship names that accept\n   * nested writes (`{fooAttributes: [...]}` entries). The generator\n   * emits those names into the frontend model's `resourceConfig()` so\n   * the client `save()` walker knows which relationships to ship.\n   *\n   * Constructed with no controller/ability so resource overrides must\n   * support being called without a request context.\n   * @param {typeof import(\"../../../../../frontend-model-resource/base-resource.js\").default | null} resourceClass - Resource class.\n   * @returns {string[]} - Relationship names that accept nested writes (empty when none).\n   */\n  nestedRelationshipNamesForGenerator(resourceClass) {\n    if (!resourceClass || typeof resourceClass !== \"function\") return []\n\n    const prototypeWithMethod = /** @type {{permittedParams?: (arg?: object) => Array<string | Record<string, any>>}} */ (resourceClass.prototype)\n\n    if (typeof prototypeWithMethod?.permittedParams !== \"function\") return []\n\n    let spec\n\n    try {\n      const instance = new resourceClass({\n        ability: undefined,\n        context: {},\n        locals: {},\n        modelClass: resourceClass.ModelClass,\n        modelName: resourceClass.ModelClass?.getModelName?.() || resourceClass.name,\n        params: {},\n        resourceConfiguration: /** @type {import(\"../../../../../configuration-types.js\").FrontendModelResourceConfiguration} */ ({abilities: {}, attributes: []})\n      })\n      spec = instance.permittedParams()\n    } catch (error) {\n      throw new Error(`Failed to invoke ${resourceClass.name}.permittedParams() while generating frontend models: ${error instanceof Error ? error.message : String(error)}`, {cause: error})\n    }\n\n    if (!Array.isArray(spec)) return []\n\n    /** @type {string[]} */\n    const relationshipNames = []\n\n    for (const entry of spec) {\n      if (!entry || typeof entry !== \"object\" || Array.isArray(entry)) continue\n\n      for (const key of Object.keys(entry)) {\n        if (!key.endsWith(\"Attributes\")) continue\n        const name = key.slice(0, -\"Attributes\".length)\n        if (name) relationshipNames.push(name)\n      }\n    }\n\n    return relationshipNames\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  /**\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 - Command key-values.\n   * @returns {string} - Formatted property (array when keys match values, object otherwise).\n   */\n  formattedCommandsProperty({indent, propertyName, values}) {\n    const allKeysMatchValues = Object.entries(values).every(([key, value]) => key === value)\n\n    if (allKeysMatchValues) {\n      return this.formattedArrayProperty({indent, propertyName, values: Object.keys(values)})\n    }\n\n    return this.formattedObjectProperty({indent, propertyName, values})\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    let attributes = modelConfig.attributes\n\n    // Auto-derive attributes from model columns when not explicitly defined\n    if ((!attributes || (Array.isArray(attributes) && attributes.length === 0)) && modelClass) {\n      try {\n        const columns = modelClass.getColumns()\n\n        if (Array.isArray(columns)) {\n          attributes = columns.map((column) => inflection.camelize(column.getName(), true))\n        }\n      } catch {\n        // Model may not be initialized yet\n      }\n    }\n\n    if (Array.isArray(attributes)) {\n      return attributes.map((entry) => {\n        const attributeName = typeof entry === \"string\" ? entry : entry.name\n\n        return {\n          jsDocType: this.jsDocTypeForFrontendAttribute({\n            attributeConfig: this.frontendAttributeConfigForModelAttribute({attributeName, modelClass})\n          }),\n          name: attributeName\n        }\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 (type && [\"blob\", \"char\", \"nvarchar\", \"varchar\", \"text\", \"longtext\", \"uuid\", \"character varying\"].includes(type)) {\n      return \"string\"\n    } else if (type && [\"bit\", \"bigint\", \"decimal\", \"double\", \"double precision\", \"float\", \"int\", \"integer\", \"numeric\", \"real\", \"smallint\", \"tinyint\"].includes(type)) {\n      return \"number\"\n    } else if (type && [\"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   * @param {typeof import(\"../../../../../frontend-model-resource/base-resource.js\").default | null} [args.resourceClass]\n   * @returns {Array<{relationshipName: string, targetClassName: string, targetFileName: string, type: \"belongsTo\" | \"hasOne\" | \"hasMany\"}>} - Relationships.\n   */\n  relationshipsForModel({className, modelConfig, resourceClass}) {\n    const relationships = modelConfig.relationships\n\n    if (relationships === undefined || relationships === null) {\n      return []\n    }\n\n    if (!Array.isArray(relationships)) {\n      throw new Error(`Model '${className}' has invalid relationships config — must be an array of relationship names, got ${typeof relationships}`)\n    }\n\n    return relationships.map((relationshipName) => this.inferredRelationshipDefinition({className, relationshipName, resourceClass}))\n  }\n\n  /**\n   * @param {object} args - Arguments.\n   * @param {string} args.className - Model class name.\n   * @param {string} args.relationshipName - Relationship name.\n   * @param {typeof import(\"../../../../../frontend-model-resource/base-resource.js\").default | null} [args.resourceClass]\n   * @returns {{relationshipName: string, targetClassName: string, targetFileName: string, type: \"belongsTo\" | \"hasOne\" | \"hasMany\"}} Inferred relationship definition.\n   */\n  inferredRelationshipDefinition({className, relationshipName, resourceClass}) {\n    const modelClass = resourceClass?.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 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    let targetClassName\n\n    try {\n      const targetModelClass = relationship.getTargetModelClass()\n\n      targetClassName = targetModelClass?.getModelName()\n    } catch {\n      // Model class not registered yet — fall back to className from relationship definition\n    }\n\n    if (!targetClassName) {\n      targetClassName = relationship.className\n\n      if (!targetClassName) {\n        throw new Error(`Model '${className}' relationship '${relationshipName}' has no target model class`)\n      }\n    }\n\n    return {\n      relationshipName,\n      targetClassName,\n      targetFileName: inflection.dasherize(inflection.underscore(targetClassName)),\n      type: relationshipType\n    }\n  }\n}\n"]}
|
|
@@ -38,27 +38,6 @@ export default class FrontendModelBaseResource extends AuthorizationBaseResource
|
|
|
38
38
|
static relationships: string[] | undefined;
|
|
39
39
|
/** @type {string[] | undefined} */
|
|
40
40
|
static translatedAttributes: string[] | undefined;
|
|
41
|
-
/**
|
|
42
|
-
* Declares relationships on this resource that accept nested writes through `save()`.
|
|
43
|
-
* Keys are relationship names (matching the model's relationship definitions).
|
|
44
|
-
* Values describe the policy for each relationship.
|
|
45
|
-
*
|
|
46
|
-
* Example:
|
|
47
|
-
* static nestedAttributes = {
|
|
48
|
-
* aiActions: {allowDestroy: true, limit: 100}
|
|
49
|
-
* }
|
|
50
|
-
*
|
|
51
|
-
* Recursion into deeper levels follows each child resource's own `nestedAttributes`
|
|
52
|
-
* declaration — parents do NOT inline the full tree, so authorization and policy
|
|
53
|
-
* stay on the child resource that actually owns the records being written.
|
|
54
|
-
*
|
|
55
|
-
* @type {Record<string, {allowDestroy?: boolean, limit?: number, rejectIf?: (attributes: Record<string, any>) => boolean}> | undefined}
|
|
56
|
-
*/
|
|
57
|
-
static nestedAttributes: Record<string, {
|
|
58
|
-
allowDestroy?: boolean;
|
|
59
|
-
limit?: number;
|
|
60
|
-
rejectIf?: (attributes: Record<string, any>) => boolean;
|
|
61
|
-
}> | undefined;
|
|
62
41
|
/**
|
|
63
42
|
* @returns {import("../configuration-types.js").FrontendModelResourceConfiguration} - Static resource config.
|
|
64
43
|
*/
|
|
@@ -97,36 +76,49 @@ export default class FrontendModelBaseResource extends AuthorizationBaseResource
|
|
|
97
76
|
/** @returns {import("../configuration-types.js").FrontendModelResourceConfiguration} - Normalized resource config. */
|
|
98
77
|
resourceConfiguration(): import("../configuration-types.js").FrontendModelResourceConfiguration;
|
|
99
78
|
/**
|
|
100
|
-
* Returns a Rails-strong-params-style permit spec declaring
|
|
101
|
-
* and nested attributes are writable for the current
|
|
102
|
-
*
|
|
79
|
+
* Returns a Rails-strong-params / api_maker-style permit spec declaring
|
|
80
|
+
* which attributes and nested attributes are writable for the current
|
|
81
|
+
* request. Submitting an attribute or nested-relationship key that is
|
|
82
|
+
* not permitted raises an error and fails the write.
|
|
103
83
|
*
|
|
104
|
-
*
|
|
105
|
-
* -
|
|
106
|
-
* -
|
|
107
|
-
*
|
|
84
|
+
* The returned value is a flat array that mixes:
|
|
85
|
+
* - `"attributeName"` strings for plain attribute writes
|
|
86
|
+
* - `{<relationshipName>Attributes: [...]}` objects where the value
|
|
87
|
+
* is itself a permit spec for the nested relationship
|
|
108
88
|
*
|
|
109
|
-
*
|
|
110
|
-
*
|
|
111
|
-
*
|
|
112
|
-
*
|
|
113
|
-
*
|
|
89
|
+
* This matches Rails strong_params (`permit(:first_name, :last_name,
|
|
90
|
+
* contact_attributes: [:email, details_attributes: [:detail]])`) and
|
|
91
|
+
* the api_maker sister project. Include `"_destroy"` inside a nested
|
|
92
|
+
* permit to allow `_destroy: true` entries for that relationship —
|
|
93
|
+
* the model must also declare `acceptsNestedAttributesFor(name,
|
|
94
|
+
* {allowDestroy: true})` for the destroy to be applied.
|
|
95
|
+
*
|
|
96
|
+
* Example:
|
|
97
|
+
*
|
|
98
|
+
* class ProjectResource extends FrontendModelBaseResource {
|
|
99
|
+
* permittedParams(arg) {
|
|
100
|
+
* return [
|
|
101
|
+
* "name",
|
|
102
|
+
* "description",
|
|
103
|
+
* {tasksAttributes: ["id", "_destroy", "name",
|
|
104
|
+
* {subtasksAttributes: ["id", "_destroy", "name"]}
|
|
105
|
+
* ]}
|
|
106
|
+
* ]
|
|
107
|
+
* }
|
|
108
|
+
* }
|
|
109
|
+
*
|
|
110
|
+
* Default implementation returns `[]` — nothing permitted. Subclasses
|
|
111
|
+
* must override to enable writes. A resource that does not declare
|
|
112
|
+
* `permittedParams` cannot accept any write.
|
|
114
113
|
* @param {{action?: "create" | "update", params?: Record<string, any>, ability?: import("../authorization/ability.js").default, locals?: Record<string, any>}} [arg] - Request context.
|
|
115
|
-
* @returns {
|
|
114
|
+
* @returns {Array<string | Record<string, any>>} - Permit spec.
|
|
116
115
|
*/
|
|
117
116
|
permittedParams(arg?: {
|
|
118
117
|
action?: "create" | "update";
|
|
119
118
|
params?: Record<string, any>;
|
|
120
119
|
ability?: import("../authorization/ability.js").default;
|
|
121
120
|
locals?: Record<string, any>;
|
|
122
|
-
}):
|
|
123
|
-
attributes: string[] | null;
|
|
124
|
-
nestedAttributes: Record<string, {
|
|
125
|
-
allowDestroy?: boolean;
|
|
126
|
-
limit?: number;
|
|
127
|
-
rejectIf?: (attrs: Record<string, any>) => boolean;
|
|
128
|
-
}>;
|
|
129
|
-
};
|
|
121
|
+
}): Array<string | Record<string, any>>;
|
|
130
122
|
/** @returns {string} - Primary key. */
|
|
131
123
|
primaryKey(): string;
|
|
132
124
|
/**
|
|
@@ -214,15 +206,19 @@ export default class FrontendModelBaseResource extends AuthorizationBaseResource
|
|
|
214
206
|
* parent's). Destroys run before updates, updates before creates, to avoid
|
|
215
207
|
* unique-constraint conflicts when replacing a child at the same natural key.
|
|
216
208
|
*
|
|
209
|
+
* Attribute filtering for nested children uses the parent resource's
|
|
210
|
+
* permit spec for that relationship — api_maker-style. Policy options
|
|
211
|
+
* (allowDestroy, limit, rejectIf) come from the MODEL's
|
|
212
|
+
* `acceptedNestedAttributesFor(name)` declaration.
|
|
217
213
|
* @param {import("../database/record/index.js").default} parent - Parent model instance.
|
|
218
214
|
* @param {Record<string, any>} nestedAttributes - Nested-attribute payload keyed by relationship name.
|
|
219
215
|
* @param {any} controller - Controller instance for resource resolution and authorization.
|
|
220
|
-
* @param {{attributes: string[]
|
|
216
|
+
* @param {{attributes: string[], nested: Record<string, any>} | null} [parentPermit] - Parsed parent permit spec.
|
|
221
217
|
* @returns {Promise<void>}
|
|
222
218
|
*/
|
|
223
|
-
_applyNestedAttributes(parent: import("../database/record/index.js").default, nestedAttributes: Record<string, any>, controller: any,
|
|
224
|
-
attributes: string[]
|
|
225
|
-
|
|
219
|
+
_applyNestedAttributes(parent: import("../database/record/index.js").default, nestedAttributes: Record<string, any>, controller: any, parentPermit?: {
|
|
220
|
+
attributes: string[];
|
|
221
|
+
nested: Record<string, any>;
|
|
226
222
|
} | null): Promise<void>;
|
|
227
223
|
/**
|
|
228
224
|
* Resolves the ability action for a child resource using the child's own
|
|
@@ -294,13 +290,18 @@ export default class FrontendModelBaseResource extends AuthorizationBaseResource
|
|
|
294
290
|
foreignKey?: string;
|
|
295
291
|
}): string;
|
|
296
292
|
/**
|
|
297
|
-
* After nested writes, preload every relationship declared in
|
|
298
|
-
* so the post-save serialize step emits them and the
|
|
293
|
+
* After nested writes, preload every relationship declared in the
|
|
294
|
+
* parent's permit so the post-save serialize step emits them and the
|
|
295
|
+
* client can reconcile ids.
|
|
299
296
|
*
|
|
300
297
|
* @param {import("../database/record/index.js").default} model - Saved parent model.
|
|
298
|
+
* @param {{attributes: string[], nested: Record<string, any>}} permit - Parsed parent permit.
|
|
301
299
|
* @returns {Promise<void>}
|
|
302
300
|
*/
|
|
303
|
-
_preloadNestedWritableRelationships(model: import("../database/record/index.js").default
|
|
301
|
+
_preloadNestedWritableRelationships(model: import("../database/record/index.js").default, permit: {
|
|
302
|
+
attributes: string[];
|
|
303
|
+
nested: Record<string, any>;
|
|
304
|
+
}): Promise<void>;
|
|
304
305
|
}
|
|
305
306
|
export type FrontendModelResourceControllerArgs = {
|
|
306
307
|
/**
|
|
@@ -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,mCAAmC;IACnC,sBADW,MAAM,EAAE,GAAG,SAAS,CACC;IAChC,mCAAmC;IACnC,6BADW,MAAM,EAAE,GAAG,SAAS,CACQ;
|
|
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,mCAAmC;IACnC,sBADW,MAAM,EAAE,GAAG,SAAS,CACC;IAChC,mCAAmC;IACnC,6BADW,MAAM,EAAE,GAAG,SAAS,CACQ;IA+BvC;;OAEG;IACH,yBAFa,OAAO,2BAA2B,EAAE,kCAAkC,CAiBlF;IA/CD;;OAEG;IACH,kBAFW,gCAAgC,GAAG,mCAAmC,EAchF;IALC,2DAAoE;IACpE,kFAAiO;IACjO,mCAA0K;IAC1K,6CAA6D;IAC7D,+GAAwN;IAG1N;;;;;;;OAOG;IACH,2BAPa,OAAO,kBAAkB,EAAE,OAAO,GAAG;QAC7C,4BAA4B,EAAE,CAAC,MAAM,EAAE,OAAO,GAAG,MAAM,GAAG,QAAQ,GAAG,QAAQ,GAAG,SAAS,GAAG,QAAQ,GAAG,UAAU,GAAG,KAAK,KAAK,OAAO,wCAAwC,EAAE,OAAO,CAAC,GAAG,CAAC,CAAC;QAC5L,uBAAuB,EAAE,MAAM,OAAO,wCAAwC,EAAE,OAAO,CAAC,GAAG,CAAC,CAAC;QAC7F,oBAAoB,EAAE,MAAM,OAAO,4BAA4B,EAAE,mBAAmB,GAAG,IAAI,CAAC;QAC5F,sBAAsB,EAAE,CAAC,KAAK,EAAE,OAAO,6BAA6B,EAAE,OAAO,KAAK,OAAO,CAAC,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC,CAAA;KAC/G,CAIH;IAsBD,2EAA2E;IAC3E,sBADc,OAAO,kBAAkB,EAAE,OAAO,CAK/C;IAED,qFAAqF;IACrF,cADc,cAAc,6BAA6B,EAAE,OAAO,CAOjE;IAED,sCAAsC;IACtC,aADc,MAAM,CAKnB;IAED,+CAA+C;IAC/C,UADc,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAC2B;IAE5D,sHAAsH;IACtH,yBADc,OAAO,2BAA2B,EAAE,kCAAkC,CAKnF;IAED;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;OAqCG;IACH,sBAHW;QAAC,MAAM,CAAC,EAAE,QAAQ,GAAG,QAAQ,CAAC;QAAC,MAAM,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC;QAAC,OAAO,CAAC,EAAE,OAAO,6BAA6B,EAAE,OAAO,CAAC;QAAC,MAAM,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAAA;KAAC,GACjJ,KAAK,CAAC,MAAM,GAAG,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC,CAM/C;IAED,uCAAuC;IACvC,cADc,MAAM,CACkC;IAEtD;;;OAGG;IACH,wBAHW,OAAO,GAAG,MAAM,GAAG,QAAQ,GAAG,QAAQ,GAAG,SAAS,GAAG,QAAQ,GAAG,UAAU,GAAG,KAAK,GAChF,OAAO,wCAAwC,EAAE,OAAO,CAAC,GAAG,CAAC,CAIzE;IAED;;;OAGG;IACH,sBAHW,OAAO,GAAG,MAAM,GAAG,QAAQ,GAAG,QAAQ,GAAG,SAAS,GAAG,QAAQ,GAAG,UAAU,GAAG,KAAK,GAChF,OAAO,GAAG,OAAO,CAAC,OAAO,CAAC,CAMtC;IAED;;;OAGG;IACH,qBAHW,OAAO,GAAG,MAAM,GAAG,QAAQ,GAAG,QAAQ,GAAG,SAAS,GAAG,QAAQ,GAAG,UAAU,GAAG,KAAK,GAChF,OAAO,GAAG,IAAI,GAAG,OAAO,CAAC,OAAO,GAAG,IAAI,CAAC,CAMpD;IAED;;OAEG;IACH,WAFa,OAAO,CAAC,OAAO,6BAA6B,EAAE,OAAO,EAAE,CAAC,CAIpE;IAED;;;;OAIG;IACH,aAJW,MAAM,GAAG,QAAQ,GAAG,SAAS,GAAG,QAAQ,GAAG,UAAU,GAAG,KAAK,MAC7D,MAAM,GAAG,MAAM,GACb,OAAO,CAAC,OAAO,6BAA6B,EAAE,OAAO,GAAG,IAAI,CAAC,CAWzE;IAED;;;;OAIG;IACH,mBAJW,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,YACnB;QAAC,UAAU,CAAC,EAAE,GAAG,CAAC;QAAC,gBAAgB,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,GAAG,IAAI,CAAA;KAAC,GAC/D,OAAO,CAAC,OAAO,6BAA6B,EAAE,OAAO,CAAC,CAoBlE;IAED;;;OAGG;IACH,sCAHW,OAAO,6BAA6B,EAAE,OAAO,GAC3C,OAAO,CAAC,IAAI,CAAC,CAIzB;IAED;;;;;OAKG;IACH,cALW,OAAO,6BAA6B,EAAE,OAAO,cAC7C,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,YACnB;QAAC,UAAU,CAAC,EAAE,GAAG,CAAC;QAAC,gBAAgB,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,GAAG,IAAI,CAAA;KAAC,GAC/D,OAAO,CAAC,OAAO,6BAA6B,EAAE,OAAO,CAAC,CAmBlE;IAED;;;;;;OAMG;IACH,iCAJW,OAAO,6BAA6B,EAAE,OAAO,cAC7C,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,GACjB,OAAO,CAAC,IAAI,CAAC,CAsBzB;IAED;;;;;;;OAOG;IACH,sCALW,OAAO,6BAA6B,EAAE,OAAO,QAC7C,MAAM,SACN,GAAG,GACD,OAAO,CAAC,IAAI,CAAC,CAoCzB;IAED;;;OAGG;IACH,eAHW,OAAO,6BAA6B,EAAE,OAAO,GAC3C,OAAO,CAAC,IAAI,CAAC,CAIzB;IAED;;;;OAIG;IACH,iBAJW,OAAO,6BAA6B,EAAE,OAAO,WAC7C,OAAO,GAAG,MAAM,GAAG,QAAQ,GAAG,QAAQ,GACpC,OAAO,CAAC,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC,CAMxC;IAED;;;;;;;;;;;;;;;;;OAiBG;IACH,+BANW,OAAO,6BAA6B,EAAE,OAAO,oBAC7C,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,cACnB,GAAG,iBACH;QAAC,UAAU,EAAE,MAAM,EAAE,CAAC;QAAC,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAAA;KAAC,GAAG,IAAI,GACxD,OAAO,CAAC,IAAI,CAAC,CA8JzB;IAED;;;;;;;;;OASG;IACH,uDAJW,OAAO,2BAA2B,EAAE,kCAAkC,UACtE,QAAQ,GAAG,QAAQ,GAAG,SAAS,GAC7B,MAAM,CAgBlB;IAED;;;;;;;;;;;;;;;;;OAiBG;IACH,8HAVG;QAAwE,OAAO,EAAvE,OAAO,6BAA6B,EAAE,OAAO,GAAG,SAAS;QAC9B,MAAM,EAAjC,QAAQ,GAAG,SAAS;QACyD,0BAA0B,EAAvG,OAAO,2BAA2B,EAAE,kCAAkC;QACzD,UAAU,EAAvB,MAAM;QACgB,EAAE,EAAxB,MAAM,GAAG,MAAM;QACqC,MAAM,EAA1D,OAAO,6BAA6B,EAAE,OAAO;QAChC,gBAAgB,EAA7B,MAAM;QACqD,gBAAgB,EAA3E,cAAc,6BAA6B,EAAE,OAAO;KAC5D,GAAU,OAAO,CAAC,OAAO,6BAA6B,EAAE,OAAO,CAAC,CAgBlE;IAED;;;;;;;;;;;;OAYG;IACH,2GAPG;QAAwE,OAAO,EAAvE,OAAO,6BAA6B,EAAE,OAAO,GAAG,SAAS;QACL,KAAK,EAAzD,OAAO,6BAA6B,EAAE,OAAO;QACgC,0BAA0B,EAAvG,OAAO,2BAA2B,EAAE,kCAAkC;QACzD,gBAAgB,EAA7B,MAAM;QACqD,gBAAgB,EAA3E,cAAc,6BAA6B,EAAE,OAAO;KAC5D,GAAU,OAAO,CAAC,IAAI,CAAC,CAezB;IAED;;;;;;OAMG;IACH,yBAJW,OAAO,6BAA6B,EAAE,OAAO,cAC7C;QAAC,UAAU,CAAC,EAAE,MAAM,CAAA;KAAC,GACnB,MAAM,CASlB;IAED;;;;;;;;OAQG;IACH,2CAJW,OAAO,6BAA6B,EAAE,OAAO,UAC7C;QAAC,UAAU,EAAE,MAAM,EAAE,CAAC;QAAC,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAAA;KAAC,GACjD,OAAO,CAAC,IAAI,CAAC,CAYzB;CACF;;;;;gBA/pBa,OAAO,kBAAkB,EAAE,OAAO;;;;gBAClC,cAAc,6BAA6B,EAAE,OAAO;;;;eACpD,MAAM;;;;YACN,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC;;;;2BACnB,OAAO,2BAA2B,EAAE,kCAAkC;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;sCAT9C,mCAAmC"}
|