velocious 1.0.209 → 1.0.210
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 +5 -0
- package/build/src/configuration-types.d.ts.map +1 -1
- package/build/src/configuration-types.js +2 -1
- package/build/src/environment-handlers/node/cli/commands/generate/frontend-models.d.ts +12 -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 +38 -9
- package/package.json +1 -1
|
@@ -143,6 +143,7 @@
|
|
|
143
143
|
/**
|
|
144
144
|
* @typedef {object} BackendProjectConfiguration
|
|
145
145
|
* @property {string} path - Path to the backend project.
|
|
146
|
+
* @property {string} [frontendModelsOutputPath] - Optional output project path where `src/frontend-models` should be generated.
|
|
146
147
|
* @property {Record<string, FrontendModelResourceConfiguration>} [frontendModels] - Frontend model definitions keyed by model class name.
|
|
147
148
|
* @property {Record<string, FrontendModelResourceConfiguration>} [resources] - Alias for `frontendModels`.
|
|
148
149
|
*/
|
|
@@ -565,6 +566,10 @@ export type BackendProjectConfiguration = {
|
|
|
565
566
|
* - Path to the backend project.
|
|
566
567
|
*/
|
|
567
568
|
path: string;
|
|
569
|
+
/**
|
|
570
|
+
* - Optional output project path where `src/frontend-models` should be generated.
|
|
571
|
+
*/
|
|
572
|
+
frontendModelsOutputPath?: string;
|
|
568
573
|
/**
|
|
569
574
|
* - Frontend model definitions keyed by model class name.
|
|
570
575
|
*/
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"configuration-types.d.ts","sourceRoot":"","sources":["../../src/configuration-types.js"],"names":[],"mappings":"AAEA;;GAEG;AAEH;;GAEG;AAEH;;GAEG;AAEH;;;;;;GAMG;AAEH;;GAEG;AAEH;;;;;;;;GAQG;AAEH;;;;;;;;;;;;;;;GAeG;AAEH;;;;;;;;;;;;;;;;;;;;GAoBG;AAEH;;GAEG;AAEH;;;;;;GAMG;AAEH;;;;GAIG;AAEH;;;;GAIG;AAEH;;GAEG;AAEH;;;;;;;;;;GAUG;AAEH;;;GAGG;AAEH;;;;;GAKG;AAEH;;;GAGG;AAGH;;GAEG;AAEH;;;;;;;;GAQG;AAEH;;;;;;;GAOG;AAEH;;;;;;;;GAQG;AAEH
|
|
1
|
+
{"version":3,"file":"configuration-types.d.ts","sourceRoot":"","sources":["../../src/configuration-types.js"],"names":[],"mappings":"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"}
|
|
@@ -144,6 +144,7 @@
|
|
|
144
144
|
/**
|
|
145
145
|
* @typedef {object} BackendProjectConfiguration
|
|
146
146
|
* @property {string} path - Path to the backend project.
|
|
147
|
+
* @property {string} [frontendModelsOutputPath] - Optional output project path where `src/frontend-models` should be generated.
|
|
147
148
|
* @property {Record<string, FrontendModelResourceConfiguration>} [frontendModels] - Frontend model definitions keyed by model class name.
|
|
148
149
|
* @property {Record<string, FrontendModelResourceConfiguration>} [resources] - Alias for `frontendModels`.
|
|
149
150
|
*/
|
|
@@ -181,4 +182,4 @@
|
|
|
181
182
|
* @property {WebsocketMessageHandlerResolverType} [websocketMessageHandlerResolver] - Resolve a raw websocket message handler for each connection.
|
|
182
183
|
*/
|
|
183
184
|
export const nothing = {};
|
|
184
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"configuration-types.js","sourceRoot":"","sources":["../../src/configuration-types.js"],"names":[],"mappings":"AAAA,YAAY;AAEZ;;GAEG;AAEH;;GAEG;AAEH;;GAEG;AAEH;;;;;;GAMG;AAEH;;GAEG;AAEH;;;;;;;;GAQG;AAEH;;;;;;;;;;;;;;;GAeG;AAEH;;;;;;;;;;;;;;;;;;;;GAoBG;AAEH;;GAEG;AAEH;;;;;;GAMG;AAEH;;;;GAIG;AAEH;;;;GAIG;AAEH;;GAEG;AAEH;;;;;;;;;;GAUG;AAEH;;;GAGG;AAEH;;;;;GAKG;AAEH;;;GAGG;AAGH;;GAEG;AAEH;;;;;;;;GAQG;AAEH;;;;;;;GAOG;AAEH;;;;;;;;GAQG;AAEH;;;;;GAKG;AAEH;;GAEG;AAEH;;GAEG;AAEH;;;;;;;;;;;;;;;;;;;;;;;;;;GA0BG;AAEH,MAAM,CAAC,MAAM,OAAO,GAAG,EAAE,CAAA","sourcesContent":["// @ts-check\n\n/**\n * @module types\n */\n\n/**\n * @typedef {function({request: import(\"./http-server/client/request.js\").default | import(\"./http-server/client/websocket-request.js\").default, response: import(\"./http-server/client/response.js\").default}): Promise<void>} CorsType\n */\n\n/**\n * @typedef {function({request: import(\"./http-server/client/request.js\").default | import(\"./http-server/client/websocket-request.js\").default | undefined, subscription?: {channel: string, params?: Record<string, unknown>}, client: import(\"./http-server/client/index.js\").default, websocketSession: import(\"./http-server/client/websocket-session.js\").default, configuration: import(\"./configuration.js\").default}): typeof import(\"./http-server/websocket-channel.js\").default | import(\"./http-server/websocket-channel.js\").default | void | Promise<typeof import(\"./http-server/websocket-channel.js\").default | import(\"./http-server/websocket-channel.js\").default | void>} WebsocketChannelResolverType\n */\n\n/**\n * @typedef {object} WebsocketMessageHandler\n * @property {function({message: any, session: import(\"./http-server/client/websocket-session.js\").default}) : Promise<void> | void} [onMessage] - Handler for incoming websocket messages.\n * @property {function({session: import(\"./http-server/client/websocket-session.js\").default}) : Promise<void> | void} [onOpen] - Handler when the websocket session opens.\n * @property {function({session: import(\"./http-server/client/websocket-session.js\").default}) : Promise<void> | void} [onClose] - Handler when the websocket session closes.\n * @property {function({error: Error, session: import(\"./http-server/client/websocket-session.js\").default}) : Promise<void> | void} [onError] - Handler when a websocket message errors.\n */\n\n/**\n * @typedef {function({request: import(\"./http-server/client/request.js\").default | import(\"./http-server/client/websocket-request.js\").default | undefined, client: import(\"./http-server/client/index.js\").default, configuration: import(\"./configuration.js\").default}): WebsocketMessageHandler | void | Promise<WebsocketMessageHandler | void>} WebsocketMessageHandlerResolverType\n */\n\n/**\n * @typedef {(id: string) => {default: typeof import(\"./initializer.js\").default}} InitializersRequireContextType\n * @typedef {InitializersRequireContextType & {\n *   keys: () => string[],\n *   id: string\n * }} WebpackRequireContext\n * @typedef {{requireContext: WebpackRequireContext}} InitializersExportType\n * @typedef {function({configuration: import(\"./configuration.js\").default}) : Promise<InitializersExportType>} InitializersType\n */\n\n/**\n * @typedef {object} SqlConfig\n * @property {string} [database] - Database name for the SQL driver.\n * @property {object} [options] - Driver-specific connection options.\n * @property {boolean} [options.encrypt] - Whether to encrypt the connection (MSSQL).\n * @property {string} [options.schema] - Default schema for unqualified table lookups (MSSQL).\n * @property {string} [options.serverName] - TLS SNI server name override for MSSQL (empty string disables SNI).\n * @property {boolean} [options.trustServerCertificate] - Whether to trust the server certificate (MSSQL).\n * @property {string} [password] - Password for the SQL user.\n * @property {object} [pool] - Connection pool configuration.\n * @property {number} [pool.max] - Maximum number of connections.\n * @property {number} [pool.min] - Minimum number of connections.\n * @property {number} [pool.idleTimeoutMillis] - Idle timeout before releasing a connection.\n * @property {string} [server] - SQL server hostname.\n * @property {string} [user] - SQL username.\n */\n\n/**\n * @typedef {object} DatabaseConfigurationType\n * @property {string} [database] - Database name for this connection.\n * @property {typeof import(\"./database/drivers/base.js\").default} [driver] - Driver class to use for this database.\n * @property {typeof import(\"./database/pool/base.js\").default} [poolType] - Pool class to use for this database.\n * @property {function() : unknown} [getConnection] - Custom connection factory override.\n * @property {string} [host] - Database host.\n * @property {boolean} [migrations] - Whether migrations are enabled for this database.\n * @property {string} [password] - Password for the database user.\n * @property {number} [port] - Database port.\n * @property {string} [name] - Friendly name for the configuration.\n * @property {boolean} [readOnly] - Whether writes should be blocked for this database.\n * @property {string} [schema] - Default schema for unqualified table lookups (MSSQL).\n * @property {object} [record] - Record-level configuration.\n * @property {boolean} [record.transactions] - Whether record operations should use transactions.\n * @property {boolean} [reset] - Whether to reset the database on startup.\n * @property {SqlConfig} [sqlConfig] - Driver-specific SQL config.\n * @property {\"mssql\" | \"mysql\" | \"pgsql\" | \"sqlite\"} [type] - Database type identifier.\n * @property {string} [useDatabase] - Database to switch to after connecting.\n * @property {string} [username] - Username for database authentication.\n */\n\n/**\n * @typedef {\"debug-low-level\" | \"debug\" | \"info\" | \"warn\" | \"error\"} LogLevel\n */\n\n/**\n * @typedef {object} LoggingOutputPayload\n * @property {LogLevel} level - Log level.\n * @property {string} message - Formatted message.\n * @property {string} subject - Log subject.\n * @property {Date} timestamp - Timestamp.\n */\n\n/**\n * @typedef {object} LoggingOutput\n * @property {function(LoggingOutputPayload): Promise<void> | void} write - Write a log entry.\n * @property {LogLevel[]} [levels] - Default levels for this output.\n */\n\n/**\n * @typedef {object} LoggingOutputConfig\n * @property {LoggingOutput} output - Output instance.\n * @property {Array<LogLevel>} [levels] - Levels enabled for this output.\n */\n\n/**\n * @typedef {LoggingOutputConfig | LoggingOutput | import(\"./logger/base-logger.js\").default} LoggerConfig\n */\n\n/**\n * @typedef {object} LoggingConfiguration\n * @property {boolean} [console] - Enable/disable console logging for request logging. Defaults to true outside of \"test\" and for HTTP server logs.\n * @property {boolean} [file] - Enable/disable writing logs to a file. Defaults to true.\n * @property {string} [directory] - Directory where log files are stored. Defaults to \"<project>/log\".\n * @property {string} [filePath] - Explicit path for the log file. Defaults to \"<directory>/<environment>.log\".\n * @property {Array<\"debug-low-level\" | \"debug\" | \"info\" | \"warn\" | \"error\">} [levels] - Override which log levels are emitted.\n * @property {boolean} [debugLowLevel] - Convenience flag to include very low-level debug logs.\n * @property {LoggerConfig[]} [loggers] - Logger instances (converted to outputs when configured).\n * @property {LoggingOutputConfig[]} [outputs] - Explicit logger outputs (overrides console/file defaults when provided).\n */\n\n/**\n * @typedef {object} StructureSqlConfiguration\n * @property {string[]} [disabledEnvironments] - Environments that should skip writing structure sql files.\n */\n\n/**\n * @typedef {object} BackgroundJobsConfiguration\n * @property {string} [host] - Hostname for the background jobs main process.\n * @property {number} [port] - Port for the background jobs main process.\n * @property {string} [databaseIdentifier] - Database identifier used to store background jobs.\n */\n\n/**\n * @typedef {object} MailerBackend\n * @property {function({payload: import(\"./mailer.js\").MailerDeliveryPayload, configuration: import(\"./configuration.js\").default}) : Promise<unknown> | unknown} deliver - Deliver a mailer payload.\n */\n\n\n/**\n * @typedef {Record<string, string[]>} LocaleFallbacksType\n */\n\n/**\n * @typedef {object} FrontendModelResourceConfiguration\n * @property {string[] | Record<string, any>} attributes - Attributes to expose on the frontend model.\n * @property {FrontendModelResourceAbilitiesConfiguration} abilities - Ability actions keyed by frontend command (`index`, `find`, `create`, `update`, `destroy`).\n * @property {Record<string, string>} [commands] - Command names keyed by action (`index`, `find`, `update`, `destroy`).\n * @property {string} [path] - HTTP path prefix used by frontend model commands.\n * @property {string} [primaryKey] - Primary key attribute name.\n * @property {FrontendModelResourceServerConfiguration} [server] - Optional backend behavior overrides for built-in frontend actions.\n */\n\n/**\n * @typedef {object} FrontendModelResourceAbilitiesConfiguration\n * @property {string} [index] - Ability action for frontend index.\n * @property {string} [find] - Ability action for frontend find.\n * @property {string} [create] - Ability action for frontend create.\n * @property {string} [update] - Ability action for frontend update.\n * @property {string} [destroy] - Ability action for frontend destroy.\n */\n\n/**\n * @typedef {object} FrontendModelResourceServerConfiguration\n * @property {function({action: \"index\" | \"find\" | \"update\" | \"destroy\", controller: import(\"./controller.js\").default, params: Record<string, any>, modelClass: typeof import(\"./database/record/index.js\").default}) : (boolean | void | Promise<boolean | void>)} [beforeAction] - Optional callback run before built-in frontend actions.\n * @property {function({action: \"index\", controller: import(\"./controller.js\").default, params: Record<string, any>, modelClass: typeof import(\"./database/record/index.js\").default}) : Promise<import(\"./database/record/index.js\").default[]>} [records] - Records loader for frontendIndex.\n * @property {function({action: \"index\" | \"find\" | \"update\", controller: import(\"./controller.js\").default, params: Record<string, any>, modelClass: typeof import(\"./database/record/index.js\").default, model: import(\"./database/record/index.js\").default}) : Record<string, any> | Promise<Record<string, any>>} [serialize] - Record serializer for response payloads.\n * @property {function({action: \"find\" | \"update\" | \"destroy\", controller: import(\"./controller.js\").default, params: Record<string, any>, modelClass: typeof import(\"./database/record/index.js\").default, id: string | number}) : Promise<import(\"./database/record/index.js\").default | null>} [find] - Record loader for find/update/destroy actions.\n * @property {function({action: \"update\", controller: import(\"./controller.js\").default, params: Record<string, any>, modelClass: typeof import(\"./database/record/index.js\").default, model: import(\"./database/record/index.js\").default, attributes: Record<string, any>}) : Promise<import(\"./database/record/index.js\").default | void>} [update] - Custom update callback.\n * @property {function({action: \"destroy\", controller: import(\"./controller.js\").default, params: Record<string, any>, modelClass: typeof import(\"./database/record/index.js\").default, model: import(\"./database/record/index.js\").default}) : Promise<void>} [destroy] - Custom destroy callback.\n */\n\n/**\n * @typedef {object} BackendProjectConfiguration\n * @property {string} path - Path to the backend project.\n * @property {Record<string, FrontendModelResourceConfiguration>} [frontendModels] - Frontend model definitions keyed by model class name.\n * @property {Record<string, FrontendModelResourceConfiguration>} [resources] - Alias for `frontendModels`.\n */\n\n/**\n * @typedef {typeof import(\"./authorization/base-resource.js\").default} AbilityResourceClassType\n */\n\n/**\n * @typedef {function({configuration: import(\"./configuration.js\").default, params: Record<string, any>, request: import(\"./http-server/client/request.js\").default | import(\"./http-server/client/websocket-request.js\").default, response: import(\"./http-server/client/response.js\").default}) : import(\"./authorization/ability.js\").default | void | Promise<import(\"./authorization/ability.js\").default | void>} AbilityResolverType\n */\n\n/**\n * @typedef {object} ConfigurationArgsType\n * @property {CorsType} [cors] - CORS configuration for the HTTP server.\n * @property {string} [cookieSecret] - Secret for encrypting cookies.\n * @property {AbilityResourceClassType[]} [abilityResources] - Resource classes used to define abilities per model.\n * @property {AbilityResolverType} [abilityResolver] - Resolver for creating request-scoped ability instances.\n * @property {BackendProjectConfiguration[]} [backendProjects] - Backend project definitions used for frontend model generation.\n * @property {{[key: string]: {[key: string]: DatabaseConfigurationType}}} database - Database configurations keyed by environment and identifier.\n * @property {boolean} [debug] - Enable debug logging.\n * @property {string} [directory] - Base directory for the project.\n * @property {string} [environment] - Current environment name.\n * @property {import(\"./environment-handlers/base.js\").default} environmentHandler - Environment handler instance.\n * @property {LoggingConfiguration} [logging] - Logging configuration.\n * @property {BackgroundJobsConfiguration} [backgroundJobs] - Background jobs configuration.\n * @property {MailerBackend} [mailerBackend] - Mail delivery backend.\n * @property {function({configuration: import(\"./configuration.js\").default, type: string}) : void} initializeModels - Hook to register models for a given initialization type.\n * @property {InitializersType} [initializers] - Initializer loader for environment bootstrapping.\n * @property {string | function() : string} locale - Default locale or locale resolver.\n * @property {string[]} locales - Supported locales.\n * @property {LocaleFallbacksType} localeFallbacks - Locale fallback map.\n * @property {StructureSqlConfiguration} [structureSql] - Structure SQL generation configuration.\n * @property {string} [testing] - Path to the testing configuration file.\n * @property {number | (() => number)} [timezoneOffsetMinutes] - Default timezone offset in minutes.\n * @property {number | (() => number)} [requestTimeoutMs] - Timeout in seconds for completing a HTTP request.\n * @property {WebsocketChannelResolverType} [websocketChannelResolver] - Resolve a websocket channel class/instance for each connection.\n * @property {WebsocketMessageHandlerResolverType} [websocketMessageHandlerResolver] - Resolve a raw websocket message handler for each connection.\n */\n\nexport const nothing = {}\n"]}
|
|
185
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"configuration-types.js","sourceRoot":"","sources":["../../src/configuration-types.js"],"names":[],"mappings":"AAAA,YAAY;AAEZ;;GAEG;AAEH;;GAEG;AAEH;;GAEG;AAEH;;;;;;GAMG;AAEH;;GAEG;AAEH;;;;;;;;GAQG;AAEH;;;;;;;;;;;;;;;GAeG;AAEH;;;;;;;;;;;;;;;;;;;;GAoBG;AAEH;;GAEG;AAEH;;;;;;GAMG;AAEH;;;;GAIG;AAEH;;;;GAIG;AAEH;;GAEG;AAEH;;;;;;;;;;GAUG;AAEH;;;GAGG;AAEH;;;;;GAKG;AAEH;;;GAGG;AAGH;;GAEG;AAEH;;;;;;;;GAQG;AAEH;;;;;;;GAOG;AAEH;;;;;;;;GAQG;AAEH;;;;;;GAMG;AAEH;;GAEG;AAEH;;GAEG;AAEH;;;;;;;;;;;;;;;;;;;;;;;;;;GA0BG;AAEH,MAAM,CAAC,MAAM,OAAO,GAAG,EAAE,CAAA","sourcesContent":["// @ts-check\n\n/**\n * @module types\n */\n\n/**\n * @typedef {function({request: import(\"./http-server/client/request.js\").default | import(\"./http-server/client/websocket-request.js\").default, response: import(\"./http-server/client/response.js\").default}): Promise<void>} CorsType\n */\n\n/**\n * @typedef {function({request: import(\"./http-server/client/request.js\").default | import(\"./http-server/client/websocket-request.js\").default | undefined, subscription?: {channel: string, params?: Record<string, unknown>}, client: import(\"./http-server/client/index.js\").default, websocketSession: import(\"./http-server/client/websocket-session.js\").default, configuration: import(\"./configuration.js\").default}): typeof import(\"./http-server/websocket-channel.js\").default | import(\"./http-server/websocket-channel.js\").default | void | Promise<typeof import(\"./http-server/websocket-channel.js\").default | import(\"./http-server/websocket-channel.js\").default | void>} WebsocketChannelResolverType\n */\n\n/**\n * @typedef {object} WebsocketMessageHandler\n * @property {function({message: any, session: import(\"./http-server/client/websocket-session.js\").default}) : Promise<void> | void} [onMessage] - Handler for incoming websocket messages.\n * @property {function({session: import(\"./http-server/client/websocket-session.js\").default}) : Promise<void> | void} [onOpen] - Handler when the websocket session opens.\n * @property {function({session: import(\"./http-server/client/websocket-session.js\").default}) : Promise<void> | void} [onClose] - Handler when the websocket session closes.\n * @property {function({error: Error, session: import(\"./http-server/client/websocket-session.js\").default}) : Promise<void> | void} [onError] - Handler when a websocket message errors.\n */\n\n/**\n * @typedef {function({request: import(\"./http-server/client/request.js\").default | import(\"./http-server/client/websocket-request.js\").default | undefined, client: import(\"./http-server/client/index.js\").default, configuration: import(\"./configuration.js\").default}): WebsocketMessageHandler | void | Promise<WebsocketMessageHandler | void>} WebsocketMessageHandlerResolverType\n */\n\n/**\n * @typedef {(id: string) => {default: typeof import(\"./initializer.js\").default}} InitializersRequireContextType\n * @typedef {InitializersRequireContextType & {\n *   keys: () => string[],\n *   id: string\n * }} WebpackRequireContext\n * @typedef {{requireContext: WebpackRequireContext}} InitializersExportType\n * @typedef {function({configuration: import(\"./configuration.js\").default}) : Promise<InitializersExportType>} InitializersType\n */\n\n/**\n * @typedef {object} SqlConfig\n * @property {string} [database] - Database name for the SQL driver.\n * @property {object} [options] - Driver-specific connection options.\n * @property {boolean} [options.encrypt] - Whether to encrypt the connection (MSSQL).\n * @property {string} [options.schema] - Default schema for unqualified table lookups (MSSQL).\n * @property {string} [options.serverName] - TLS SNI server name override for MSSQL (empty string disables SNI).\n * @property {boolean} [options.trustServerCertificate] - Whether to trust the server certificate (MSSQL).\n * @property {string} [password] - Password for the SQL user.\n * @property {object} [pool] - Connection pool configuration.\n * @property {number} [pool.max] - Maximum number of connections.\n * @property {number} [pool.min] - Minimum number of connections.\n * @property {number} [pool.idleTimeoutMillis] - Idle timeout before releasing a connection.\n * @property {string} [server] - SQL server hostname.\n * @property {string} [user] - SQL username.\n */\n\n/**\n * @typedef {object} DatabaseConfigurationType\n * @property {string} [database] - Database name for this connection.\n * @property {typeof import(\"./database/drivers/base.js\").default} [driver] - Driver class to use for this database.\n * @property {typeof import(\"./database/pool/base.js\").default} [poolType] - Pool class to use for this database.\n * @property {function() : unknown} [getConnection] - Custom connection factory override.\n * @property {string} [host] - Database host.\n * @property {boolean} [migrations] - Whether migrations are enabled for this database.\n * @property {string} [password] - Password for the database user.\n * @property {number} [port] - Database port.\n * @property {string} [name] - Friendly name for the configuration.\n * @property {boolean} [readOnly] - Whether writes should be blocked for this database.\n * @property {string} [schema] - Default schema for unqualified table lookups (MSSQL).\n * @property {object} [record] - Record-level configuration.\n * @property {boolean} [record.transactions] - Whether record operations should use transactions.\n * @property {boolean} [reset] - Whether to reset the database on startup.\n * @property {SqlConfig} [sqlConfig] - Driver-specific SQL config.\n * @property {\"mssql\" | \"mysql\" | \"pgsql\" | \"sqlite\"} [type] - Database type identifier.\n * @property {string} [useDatabase] - Database to switch to after connecting.\n * @property {string} [username] - Username for database authentication.\n */\n\n/**\n * @typedef {\"debug-low-level\" | \"debug\" | \"info\" | \"warn\" | \"error\"} LogLevel\n */\n\n/**\n * @typedef {object} LoggingOutputPayload\n * @property {LogLevel} level - Log level.\n * @property {string} message - Formatted message.\n * @property {string} subject - Log subject.\n * @property {Date} timestamp - Timestamp.\n */\n\n/**\n * @typedef {object} LoggingOutput\n * @property {function(LoggingOutputPayload): Promise<void> | void} write - Write a log entry.\n * @property {LogLevel[]} [levels] - Default levels for this output.\n */\n\n/**\n * @typedef {object} LoggingOutputConfig\n * @property {LoggingOutput} output - Output instance.\n * @property {Array<LogLevel>} [levels] - Levels enabled for this output.\n */\n\n/**\n * @typedef {LoggingOutputConfig | LoggingOutput | import(\"./logger/base-logger.js\").default} LoggerConfig\n */\n\n/**\n * @typedef {object} LoggingConfiguration\n * @property {boolean} [console] - Enable/disable console logging for request logging. Defaults to true outside of \"test\" and for HTTP server logs.\n * @property {boolean} [file] - Enable/disable writing logs to a file. Defaults to true.\n * @property {string} [directory] - Directory where log files are stored. Defaults to \"<project>/log\".\n * @property {string} [filePath] - Explicit path for the log file. Defaults to \"<directory>/<environment>.log\".\n * @property {Array<\"debug-low-level\" | \"debug\" | \"info\" | \"warn\" | \"error\">} [levels] - Override which log levels are emitted.\n * @property {boolean} [debugLowLevel] - Convenience flag to include very low-level debug logs.\n * @property {LoggerConfig[]} [loggers] - Logger instances (converted to outputs when configured).\n * @property {LoggingOutputConfig[]} [outputs] - Explicit logger outputs (overrides console/file defaults when provided).\n */\n\n/**\n * @typedef {object} StructureSqlConfiguration\n * @property {string[]} [disabledEnvironments] - Environments that should skip writing structure sql files.\n */\n\n/**\n * @typedef {object} BackgroundJobsConfiguration\n * @property {string} [host] - Hostname for the background jobs main process.\n * @property {number} [port] - Port for the background jobs main process.\n * @property {string} [databaseIdentifier] - Database identifier used to store background jobs.\n */\n\n/**\n * @typedef {object} MailerBackend\n * @property {function({payload: import(\"./mailer.js\").MailerDeliveryPayload, configuration: import(\"./configuration.js\").default}) : Promise<unknown> | unknown} deliver - Deliver a mailer payload.\n */\n\n\n/**\n * @typedef {Record<string, string[]>} LocaleFallbacksType\n */\n\n/**\n * @typedef {object} FrontendModelResourceConfiguration\n * @property {string[] | Record<string, any>} attributes - Attributes to expose on the frontend model.\n * @property {FrontendModelResourceAbilitiesConfiguration} abilities - Ability actions keyed by frontend command (`index`, `find`, `create`, `update`, `destroy`).\n * @property {Record<string, string>} [commands] - Command names keyed by action (`index`, `find`, `update`, `destroy`).\n * @property {string} [path] - HTTP path prefix used by frontend model commands.\n * @property {string} [primaryKey] - Primary key attribute name.\n * @property {FrontendModelResourceServerConfiguration} [server] - Optional backend behavior overrides for built-in frontend actions.\n */\n\n/**\n * @typedef {object} FrontendModelResourceAbilitiesConfiguration\n * @property {string} [index] - Ability action for frontend index.\n * @property {string} [find] - Ability action for frontend find.\n * @property {string} [create] - Ability action for frontend create.\n * @property {string} [update] - Ability action for frontend update.\n * @property {string} [destroy] - Ability action for frontend destroy.\n */\n\n/**\n * @typedef {object} FrontendModelResourceServerConfiguration\n * @property {function({action: \"index\" | \"find\" | \"update\" | \"destroy\", controller: import(\"./controller.js\").default, params: Record<string, any>, modelClass: typeof import(\"./database/record/index.js\").default}) : (boolean | void | Promise<boolean | void>)} [beforeAction] - Optional callback run before built-in frontend actions.\n * @property {function({action: \"index\", controller: import(\"./controller.js\").default, params: Record<string, any>, modelClass: typeof import(\"./database/record/index.js\").default}) : Promise<import(\"./database/record/index.js\").default[]>} [records] - Records loader for frontendIndex.\n * @property {function({action: \"index\" | \"find\" | \"update\", controller: import(\"./controller.js\").default, params: Record<string, any>, modelClass: typeof import(\"./database/record/index.js\").default, model: import(\"./database/record/index.js\").default}) : Record<string, any> | Promise<Record<string, any>>} [serialize] - Record serializer for response payloads.\n * @property {function({action: \"find\" | \"update\" | \"destroy\", controller: import(\"./controller.js\").default, params: Record<string, any>, modelClass: typeof import(\"./database/record/index.js\").default, id: string | number}) : Promise<import(\"./database/record/index.js\").default | null>} [find] - Record loader for find/update/destroy actions.\n * @property {function({action: \"update\", controller: import(\"./controller.js\").default, params: Record<string, any>, modelClass: typeof import(\"./database/record/index.js\").default, model: import(\"./database/record/index.js\").default, attributes: Record<string, any>}) : Promise<import(\"./database/record/index.js\").default | void>} [update] - Custom update callback.\n * @property {function({action: \"destroy\", controller: import(\"./controller.js\").default, params: Record<string, any>, modelClass: typeof import(\"./database/record/index.js\").default, model: import(\"./database/record/index.js\").default}) : Promise<void>} [destroy] - Custom destroy callback.\n */\n\n/**\n * @typedef {object} BackendProjectConfiguration\n * @property {string} path - Path to the backend project.\n * @property {string} [frontendModelsOutputPath] - Optional output project path where `src/frontend-models` should be generated.\n * @property {Record<string, FrontendModelResourceConfiguration>} [frontendModels] - Frontend model definitions keyed by model class name.\n * @property {Record<string, FrontendModelResourceConfiguration>} [resources] - Alias for `frontendModels`.\n */\n\n/**\n * @typedef {typeof import(\"./authorization/base-resource.js\").default} AbilityResourceClassType\n */\n\n/**\n * @typedef {function({configuration: import(\"./configuration.js\").default, params: Record<string, any>, request: import(\"./http-server/client/request.js\").default | import(\"./http-server/client/websocket-request.js\").default, response: import(\"./http-server/client/response.js\").default}) : import(\"./authorization/ability.js\").default | void | Promise<import(\"./authorization/ability.js\").default | void>} AbilityResolverType\n */\n\n/**\n * @typedef {object} ConfigurationArgsType\n * @property {CorsType} [cors] - CORS configuration for the HTTP server.\n * @property {string} [cookieSecret] - Secret for encrypting cookies.\n * @property {AbilityResourceClassType[]} [abilityResources] - Resource classes used to define abilities per model.\n * @property {AbilityResolverType} [abilityResolver] - Resolver for creating request-scoped ability instances.\n * @property {BackendProjectConfiguration[]} [backendProjects] - Backend project definitions used for frontend model generation.\n * @property {{[key: string]: {[key: string]: DatabaseConfigurationType}}} database - Database configurations keyed by environment and identifier.\n * @property {boolean} [debug] - Enable debug logging.\n * @property {string} [directory] - Base directory for the project.\n * @property {string} [environment] - Current environment name.\n * @property {import(\"./environment-handlers/base.js\").default} environmentHandler - Environment handler instance.\n * @property {LoggingConfiguration} [logging] - Logging configuration.\n * @property {BackgroundJobsConfiguration} [backgroundJobs] - Background jobs configuration.\n * @property {MailerBackend} [mailerBackend] - Mail delivery backend.\n * @property {function({configuration: import(\"./configuration.js\").default, type: string}) : void} initializeModels - Hook to register models for a given initialization type.\n * @property {InitializersType} [initializers] - Initializer loader for environment bootstrapping.\n * @property {string | function() : string} locale - Default locale or locale resolver.\n * @property {string[]} locales - Supported locales.\n * @property {LocaleFallbacksType} localeFallbacks - Locale fallback map.\n * @property {StructureSqlConfiguration} [structureSql] - Structure SQL generation configuration.\n * @property {string} [testing] - Path to the testing configuration file.\n * @property {number | (() => number)} [timezoneOffsetMinutes] - Default timezone offset in minutes.\n * @property {number | (() => number)} [requestTimeoutMs] - Timeout in seconds for completing a HTTP request.\n * @property {WebsocketChannelResolverType} [websocketChannelResolver] - Resolve a websocket channel class/instance for each connection.\n * @property {WebsocketMessageHandlerResolverType} [websocketMessageHandlerResolver] - Resolve a raw websocket message handler for each connection.\n */\n\nexport const nothing = {}\n"]}
|
|
@@ -20,6 +20,18 @@ export default class DbGenerateFrontendModels extends BaseCommand {
|
|
|
20
20
|
frontendModels?: Record<string, any>;
|
|
21
21
|
resources?: Record<string, any>;
|
|
22
22
|
}): Record<string, any>;
|
|
23
|
+
/**
|
|
24
|
+
* @param {{frontendModelsOutputPath?: string}} backendProject - Backend project config.
|
|
25
|
+
* @returns {string} - Absolute frontend models output directory.
|
|
26
|
+
*/
|
|
27
|
+
frontendModelsDirectoryForBackendProject(backendProject: {
|
|
28
|
+
frontendModelsOutputPath?: string;
|
|
29
|
+
}): string;
|
|
30
|
+
/**
|
|
31
|
+
* @param {string} frontendModelsDir - Frontend models output directory.
|
|
32
|
+
* @returns {string} - Base class import path.
|
|
33
|
+
*/
|
|
34
|
+
importPathForFrontendModelsDirectory(frontendModelsDir: string): string;
|
|
23
35
|
/**
|
|
24
36
|
* @param {object} args - Method args.
|
|
25
37
|
* @param {string} args.className - Model class name.
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"frontend-models.d.ts","sourceRoot":"","sources":["../../../../../../../src/environment-handlers/node/cli/commands/generate/frontend-models.js"],"names":[],"mappings":"AAIA,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":"AAIA,mGAAmG;AACnG;IACE,oEAAoE;IACpE,WADc,OAAO,CAAC,IAAI,CAAC,CAkD1B;IAED;;;;;OAKG;IACH,gDAJG;QAAqB,SAAS,EAAtB,MAAM;QACoB,WAAW,EAArC,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC;KAC3B,GAAU,IAAI,CAkBhB;IAED;;;OAGG;IACH,2CAHW;QAAC,cAAc,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC;QAAC,SAAS,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAAA;KAAC,GACrE,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAU/B;IAED;;;OAGG;IACH,yDAHW;QAAC,wBAAwB,CAAC,EAAE,MAAM,CAAA;KAAC,GACjC,MAAM,CAMlB;IAED;;;OAGG;IACH,wDAHW,MAAM,GACJ,MAAM,CAUlB;IAED;;;;;;OAMG;IACH,8DALG;QAAqB,SAAS,EAAtB,MAAM;QACO,UAAU,EAAvB,MAAM;QACoB,WAAW,EAArC,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC;KAC3B,GAAU,MAAM,CA4DlB;IAED;;;OAGG;IACH,oCAHW,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,GACjB,MAAM,EAAE,CAcpB;CACF;wBA5MuB,oCAAoC"}
|
|
@@ -10,14 +10,17 @@ export default class DbGenerateFrontendModels extends BaseCommand {
|
|
|
10
10
|
if (!Array.isArray(backendProjects) || backendProjects.length === 0) {
|
|
11
11
|
throw new Error("No backend projects configured. Configure 'backendProjects' in your configuration first");
|
|
12
12
|
}
|
|
13
|
-
const rootDirectory = this.directory();
|
|
14
|
-
const frontendModelsDir = `${rootDirectory}/src/frontend-models`;
|
|
15
|
-
const devMode = frontendModelsDir.includes("/spec/dummy/src/frontend-models");
|
|
16
|
-
const importPath = devMode ? "../../../../src/frontend-models/base.js" : "velocious/build/src/frontend-models/base.js";
|
|
17
|
-
await fs.mkdir(frontendModelsDir, { recursive: true });
|
|
18
13
|
/** @type {Set<string>} */
|
|
19
14
|
const generatedModelNames = new Set();
|
|
15
|
+
/** @type {Set<string>} */
|
|
16
|
+
const ensuredDirectories = new Set();
|
|
20
17
|
for (const backendProject of backendProjects) {
|
|
18
|
+
const frontendModelsDir = this.frontendModelsDirectoryForBackendProject(backendProject);
|
|
19
|
+
const importPath = this.importPathForFrontendModelsDirectory(frontendModelsDir);
|
|
20
|
+
if (!ensuredDirectories.has(frontendModelsDir)) {
|
|
21
|
+
await fs.mkdir(frontendModelsDir, { recursive: true });
|
|
22
|
+
ensuredDirectories.add(frontendModelsDir);
|
|
23
|
+
}
|
|
21
24
|
const resources = this.resourcesForBackendProject(backendProject);
|
|
22
25
|
for (const modelClassName in resources) {
|
|
23
26
|
const modelConfig = resources[modelClassName];
|
|
@@ -69,6 +72,25 @@ export default class DbGenerateFrontendModels extends BaseCommand {
|
|
|
69
72
|
}
|
|
70
73
|
return resources;
|
|
71
74
|
}
|
|
75
|
+
/**
|
|
76
|
+
* @param {{frontendModelsOutputPath?: string}} backendProject - Backend project config.
|
|
77
|
+
* @returns {string} - Absolute frontend models output directory.
|
|
78
|
+
*/
|
|
79
|
+
frontendModelsDirectoryForBackendProject(backendProject) {
|
|
80
|
+
const outputPath = backendProject.frontendModelsOutputPath || this.directory();
|
|
81
|
+
return `${outputPath}/src/frontend-models`;
|
|
82
|
+
}
|
|
83
|
+
/**
|
|
84
|
+
* @param {string} frontendModelsDir - Frontend models output directory.
|
|
85
|
+
* @returns {string} - Base class import path.
|
|
86
|
+
*/
|
|
87
|
+
importPathForFrontendModelsDirectory(frontendModelsDir) {
|
|
88
|
+
const devMode = frontendModelsDir.includes("/spec/dummy/src/frontend-models");
|
|
89
|
+
if (devMode) {
|
|
90
|
+
return "../../../../src/frontend-models/base.js";
|
|
91
|
+
}
|
|
92
|
+
return "velocious/build/src/frontend-models/base.js";
|
|
93
|
+
}
|
|
72
94
|
/**
|
|
73
95
|
* @param {object} args - Method args.
|
|
74
96
|
* @param {string} args.className - Model class name.
|
|
@@ -78,6 +100,7 @@ export default class DbGenerateFrontendModels extends BaseCommand {
|
|
|
78
100
|
*/
|
|
79
101
|
buildModelFileContent({ className, importPath, modelConfig }) {
|
|
80
102
|
const attributes = this.attributeNamesForModel(modelConfig);
|
|
103
|
+
const attributesTypeName = `${className}Attributes`;
|
|
81
104
|
const commands = {
|
|
82
105
|
destroy: modelConfig.commands?.destroy || "destroy",
|
|
83
106
|
find: modelConfig.commands?.find || "find",
|
|
@@ -89,6 +112,12 @@ export default class DbGenerateFrontendModels extends BaseCommand {
|
|
|
89
112
|
}
|
|
90
113
|
let fileContent = "";
|
|
91
114
|
fileContent += `import FrontendModelBase from "${importPath}"\n\n`;
|
|
115
|
+
fileContent += "/**\n";
|
|
116
|
+
fileContent += ` * @typedef {object} ${attributesTypeName}\n`;
|
|
117
|
+
for (const attributeName of attributes) {
|
|
118
|
+
fileContent += ` * @property {any} ${attributeName} - Attribute value.\n`;
|
|
119
|
+
}
|
|
120
|
+
fileContent += " */\n";
|
|
92
121
|
fileContent += `/** Frontend model for ${className}. */\n`;
|
|
93
122
|
fileContent += `export default class ${className} extends FrontendModelBase {\n`;
|
|
94
123
|
fileContent += " /**\n";
|
|
@@ -107,13 +136,13 @@ export default class DbGenerateFrontendModels extends BaseCommand {
|
|
|
107
136
|
const camelizedAttributeUpper = inflection.camelize(attributeName);
|
|
108
137
|
fileContent += "\n";
|
|
109
138
|
fileContent += " /**\n";
|
|
110
|
-
fileContent +=
|
|
139
|
+
fileContent += ` * @returns {${attributesTypeName}[${JSON.stringify(attributeName)}]} - Attribute value.\n`;
|
|
111
140
|
fileContent += " */\n";
|
|
112
141
|
fileContent += ` ${camelizedAttribute}() { return this.readAttribute(${JSON.stringify(attributeName)}) }\n`;
|
|
113
142
|
fileContent += "\n";
|
|
114
143
|
fileContent += " /**\n";
|
|
115
|
-
fileContent +=
|
|
116
|
-
fileContent +=
|
|
144
|
+
fileContent += ` * @param {${attributesTypeName}[${JSON.stringify(attributeName)}]} newValue - New attribute value.\n`;
|
|
145
|
+
fileContent += ` * @returns {${attributesTypeName}[${JSON.stringify(attributeName)}]} - Assigned value.\n`;
|
|
117
146
|
fileContent += " */\n";
|
|
118
147
|
fileContent += ` set${camelizedAttributeUpper}(newValue) { return this.setAttribute(${JSON.stringify(attributeName)}, newValue) }\n`;
|
|
119
148
|
}
|
|
@@ -135,4 +164,4 @@ export default class DbGenerateFrontendModels extends BaseCommand {
|
|
|
135
164
|
return Object.keys(attributes);
|
|
136
165
|
}
|
|
137
166
|
}
|
|
138
|
-
//# 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\"\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    if (!Array.isArray(backendProjects) || backendProjects.length === 0) {\n      throw new Error(\"No backend projects configured. Configure 'backendProjects' in your configuration first\")\n    }\n\n    const rootDirectory = this.directory()\n    const frontendModelsDir = `${rootDirectory}/src/frontend-models`\n    const devMode = frontendModelsDir.includes(\"/spec/dummy/src/frontend-models\")\n    const importPath = devMode ? \"../../../../src/frontend-models/base.js\" : \"velocious/build/src/frontend-models/base.js\"\n\n    await fs.mkdir(frontendModelsDir, {recursive: true})\n\n    /** @type {Set<string>} */\n    const generatedModelNames = new Set()\n\n    for (const backendProject of backendProjects) {\n      const resources = this.resourcesForBackendProject(backendProject)\n\n      for (const modelClassName in resources) {\n        const modelConfig = 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        this.validateModelConfig({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          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 {string} args.className - Model class name.\n   * @param {Record<string, any>} args.modelConfig - Model configuration.\n   * @returns {void} - No return value.\n   */\n  validateModelConfig({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\n  /**\n   * @param {{frontendModels?: Record<string, any>, resources?: Record<string, any>}} backendProject - Backend project config.\n   * @returns {Record<string, any>} - Resource definitions keyed by model class name.\n   */\n  resourcesForBackendProject(backendProject) {\n    const resources = backendProject.frontendModels || backendProject.resources || {}\n\n    if (!resources || typeof resources !== \"object\") {\n      throw new Error(`Expected backend project resources object but got: ${resources}`)\n    }\n\n    return resources\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 {Record<string, any>} args.modelConfig - Model configuration.\n   * @returns {string} - Generated file content.\n   */\n  buildModelFileContent({className, importPath, modelConfig}) {\n    const attributes = this.attributeNamesForModel(modelConfig)\n    const commands = {\n      destroy: modelConfig.commands?.destroy || \"destroy\",\n      find: modelConfig.commands?.find || \"find\",\n      index: modelConfig.commands?.index || \"index\",\n      update: modelConfig.commands?.update || \"update\"\n    }\n\n    if (!modelConfig.path) {\n      throw new Error(`Model '${className}' is missing required 'path' config`)\n    }\n\n    let fileContent = \"\"\n\n    fileContent += `import FrontendModelBase from \"${importPath}\"\\n\\n`\n    fileContent += `/** Frontend model for ${className}. */\\n`\n    fileContent += `export default class ${className} extends FrontendModelBase {\\n`\n    fileContent += \"  /**\\n\"\n    fileContent += \"   * @returns {{attributes: string[], commands: {destroy: string, find: string, index: string, update: string}, path: string, primaryKey: string}} - Resource config.\\n\"\n    fileContent += \"   */\\n\"\n    fileContent += \"  static resourceConfig() {\\n\"\n    fileContent += \"    return {\\n\"\n    fileContent += `      attributes: ${JSON.stringify(attributes)},\\n`\n    fileContent += `      commands: ${JSON.stringify(commands)},\\n`\n    fileContent += `      path: ${JSON.stringify(modelConfig.path)},\\n`\n    fileContent += `      primaryKey: ${JSON.stringify(modelConfig.primaryKey || \"id\")}\\n`\n    fileContent += \"    }\\n\"\n    fileContent += \"  }\\n\"\n\n    for (const attributeName of attributes) {\n      const camelizedAttribute = inflection.camelize(attributeName, true)\n      const camelizedAttributeUpper = inflection.camelize(attributeName)\n\n      fileContent += \"\\n\"\n      fileContent += \"  /**\\n\"\n      fileContent += \"   * @returns {any} - Attribute value.\\n\"\n      fileContent += \"   */\\n\"\n      fileContent += `  ${camelizedAttribute}() { return this.readAttribute(${JSON.stringify(attributeName)}) }\\n`\n\n      fileContent += \"\\n\"\n      fileContent += \"  /**\\n\"\n      fileContent += \"   * @param {any} newValue - New attribute value.\\n\"\n      fileContent += \"   * @returns {any} - Assigned value.\\n\"\n      fileContent += \"   */\\n\"\n      fileContent += `  set${camelizedAttributeUpper}(newValue) { return this.setAttribute(${JSON.stringify(attributeName)}, newValue) }\\n`\n    }\n\n    fileContent += \"}\\n\"\n\n    return fileContent\n  }\n\n  /**\n   * @param {Record<string, any>} modelConfig - Model configuration.\n   * @returns {string[]} - Attribute names.\n   */\n  attributeNamesForModel(modelConfig) {\n    const attributes = modelConfig.attributes\n\n    if (Array.isArray(attributes)) {\n      return attributes\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)\n  }\n}\n"]}
|
|
167
|
+
//# 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\"\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    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\n      for (const modelClassName in resources) {\n        const modelConfig = 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        this.validateModelConfig({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          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 {string} args.className - Model class name.\n   * @param {Record<string, any>} args.modelConfig - Model configuration.\n   * @returns {void} - No return value.\n   */\n  validateModelConfig({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\n  /**\n   * @param {{frontendModels?: Record<string, any>, resources?: Record<string, any>}} backendProject - Backend project config.\n   * @returns {Record<string, any>} - Resource definitions keyed by model class name.\n   */\n  resourcesForBackendProject(backendProject) {\n    const resources = backendProject.frontendModels || backendProject.resources || {}\n\n    if (!resources || typeof resources !== \"object\") {\n      throw new Error(`Expected backend project resources object but got: ${resources}`)\n    }\n\n    return resources\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 {Record<string, any>} args.modelConfig - Model configuration.\n   * @returns {string} - Generated file content.\n   */\n  buildModelFileContent({className, importPath, modelConfig}) {\n    const attributes = this.attributeNamesForModel(modelConfig)\n    const attributesTypeName = `${className}Attributes`\n    const commands = {\n      destroy: modelConfig.commands?.destroy || \"destroy\",\n      find: modelConfig.commands?.find || \"find\",\n      index: modelConfig.commands?.index || \"index\",\n      update: modelConfig.commands?.update || \"update\"\n    }\n\n    if (!modelConfig.path) {\n      throw new Error(`Model '${className}' is missing required 'path' config`)\n    }\n\n    let fileContent = \"\"\n\n    fileContent += `import FrontendModelBase from \"${importPath}\"\\n\\n`\n    fileContent += \"/**\\n\"\n    fileContent += ` * @typedef {object} ${attributesTypeName}\\n`\n    for (const attributeName of attributes) {\n      fileContent += ` * @property {any} ${attributeName} - 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 += \"  /**\\n\"\n    fileContent += \"   * @returns {{attributes: string[], commands: {destroy: string, find: string, index: string, update: string}, path: string, primaryKey: string}} - Resource config.\\n\"\n    fileContent += \"   */\\n\"\n    fileContent += \"  static resourceConfig() {\\n\"\n    fileContent += \"    return {\\n\"\n    fileContent += `      attributes: ${JSON.stringify(attributes)},\\n`\n    fileContent += `      commands: ${JSON.stringify(commands)},\\n`\n    fileContent += `      path: ${JSON.stringify(modelConfig.path)},\\n`\n    fileContent += `      primaryKey: ${JSON.stringify(modelConfig.primaryKey || \"id\")}\\n`\n    fileContent += \"    }\\n\"\n    fileContent += \"  }\\n\"\n\n    for (const attributeName of attributes) {\n      const camelizedAttribute = inflection.camelize(attributeName, true)\n      const camelizedAttributeUpper = inflection.camelize(attributeName)\n\n      fileContent += \"\\n\"\n      fileContent += \"  /**\\n\"\n      fileContent += `   * @returns {${attributesTypeName}[${JSON.stringify(attributeName)}]} - Attribute value.\\n`\n      fileContent += \"   */\\n\"\n      fileContent += `  ${camelizedAttribute}() { return this.readAttribute(${JSON.stringify(attributeName)}) }\\n`\n\n      fileContent += \"\\n\"\n      fileContent += \"  /**\\n\"\n      fileContent += `   * @param {${attributesTypeName}[${JSON.stringify(attributeName)}]} newValue - New attribute value.\\n`\n      fileContent += `   * @returns {${attributesTypeName}[${JSON.stringify(attributeName)}]} - Assigned value.\\n`\n      fileContent += \"   */\\n\"\n      fileContent += `  set${camelizedAttributeUpper}(newValue) { return this.setAttribute(${JSON.stringify(attributeName)}, newValue) }\\n`\n    }\n\n    fileContent += \"}\\n\"\n\n    return fileContent\n  }\n\n  /**\n   * @param {Record<string, any>} modelConfig - Model configuration.\n   * @returns {string[]} - Attribute names.\n   */\n  attributeNamesForModel(modelConfig) {\n    const attributes = modelConfig.attributes\n\n    if (Array.isArray(attributes)) {\n      return attributes\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)\n  }\n}\n"]}
|