@xrmforge/cli 0.4.1 → 0.4.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.js CHANGED
@@ -336,7 +336,7 @@ function formatSize(bytes) {
336
336
  import { resolve, basename } from "path";
337
337
  import { scaffoldProject } from "@xrmforge/devkit";
338
338
  function registerInitCommand(program2) {
339
- program2.command("init [dir]").description("Scaffold a new D365 form scripting project").option("--name <name>", "Project name for package.json (default: directory name)").option("--prefix <prefix>", "Publisher prefix for D365 WebResources", "contoso").option("--namespace <ns>", "Base namespace for form scripts (default: PascalCase of prefix)").option("--skip-install", "Skip running npm install after scaffolding", false).action(async (dir, opts) => {
339
+ program2.command("init [dir]").description("Scaffold a new D365 form scripting project").option("--name <name>", "Project name for package.json (default: directory name)").option("--prefix <prefix>", "Publisher prefix for D365 WebResources", "contoso").option("--namespace <ns>", "Base namespace for form scripts (default: PascalCase of prefix)").option("--skip-install", "Skip running npm install after scaffolding", false).option("--force", "Allow scaffolding in non-empty directories (skip existing files)", false).action(async (dir, opts) => {
340
340
  try {
341
341
  await runInit(dir, opts);
342
342
  } catch (error) {
@@ -368,7 +368,8 @@ XrmForge Project Scaffolding`);
368
368
  targetDir,
369
369
  projectName,
370
370
  prefix,
371
- namespace
371
+ namespace,
372
+ force: opts.force
372
373
  });
373
374
  console.log(`Created ${result.filesCreated.length} files:`);
374
375
  for (const file of result.filesCreated) {
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/index.ts","../src/config.ts","../src/commands/generate.ts","../src/commands/build.ts","../src/commands/init.ts"],"sourcesContent":["/**\r\n * @xrmforge/cli - Command-line interface for XrmForge\r\n *\r\n * Usage:\r\n * xrmforge generate --url https://myorg.crm4.dynamics.com \\\r\n * --auth client-credentials \\\r\n * --tenant <tenant-id> --client-id <app-id> --client-secret <secret> \\\r\n * --entities account,contact \\\r\n * --output ./typings\r\n */\r\n\r\nimport { readFileSync } from 'node:fs';\r\nimport { dirname, join } from 'node:path';\r\nimport { fileURLToPath } from 'node:url';\r\nimport { Command } from 'commander';\r\nimport { registerGenerateCommand } from './commands/generate.js';\r\nimport { registerBuildCommand } from './commands/build.js';\r\nimport { registerInitCommand } from './commands/init.js';\r\n\r\n// Read version from package.json (single source of truth)\r\nconst __dirname = dirname(fileURLToPath(import.meta.url));\r\nconst pkg = JSON.parse(readFileSync(join(__dirname, '..', 'package.json'), 'utf-8'));\r\n\r\nconst program = new Command();\r\n\r\nprogram\r\n .name('xrmforge')\r\n .description('TypeScript type generator for Dynamics 365 / Dataverse')\r\n .version(pkg.version);\r\n\r\nregisterGenerateCommand(program);\r\nregisterBuildCommand(program);\r\nregisterInitCommand(program);\r\n\r\nprogram.parse();\r\n","/**\r\n * @xrmforge/cli - Configuration File Support\r\n *\r\n * Reads xrmforge.config.json from the current working directory.\r\n * CLI flags override config file values.\r\n *\r\n * Example xrmforge.config.json:\r\n * ```json\r\n * {\r\n * \"url\": \"https://myorg.crm4.dynamics.com\",\r\n * \"auth\": \"interactive\",\r\n * \"tenantId\": \"your-tenant-id\",\r\n * \"solutions\": [\"MySolution\", \"MyOtherSolution\"],\r\n * \"entities\": [\"systemuser\", \"task\"],\r\n * \"output\": \"./typings\",\r\n * \"labelLanguage\": 1033,\r\n * \"secondaryLanguage\": 1031\r\n * }\r\n * ```\r\n */\r\n\r\nimport { readFileSync, existsSync } from 'node:fs';\r\nimport { join } from 'node:path';\r\nimport type { BuildConfig } from '@xrmforge/devkit';\r\n\r\n/** Shape of xrmforge.config.json */\r\nexport interface XrmForgeConfig {\r\n /** Dataverse environment URL */\r\n url?: string;\r\n /** Authentication method */\r\n auth?: string;\r\n /** Azure AD tenant ID */\r\n tenantId?: string;\r\n /** Azure AD application (client) ID */\r\n clientId?: string;\r\n /** Client secret (NOT recommended in config file, use env vars) */\r\n clientSecret?: string;\r\n /** Entity logical names */\r\n entities?: string[];\r\n /** Solution unique names (array or comma-separated string) */\r\n solutions?: string[] | string;\r\n /** Output directory */\r\n output?: string;\r\n /** Primary label language LCID */\r\n labelLanguage?: number;\r\n /** Secondary label language LCID */\r\n secondaryLanguage?: number;\r\n /** Generate form interfaces */\r\n forms?: boolean;\r\n /** Generate OptionSet enums */\r\n optionsets?: boolean;\r\n /** Enable metadata cache for incremental generation */\r\n cache?: boolean;\r\n /** Directory for metadata cache files */\r\n cacheDir?: string;\r\n /** Build configuration for WebResource bundling */\r\n build?: BuildConfig;\r\n}\r\n\r\nconst CONFIG_FILENAME = 'xrmforge.config.json';\r\n\r\n/**\r\n * Load config from xrmforge.config.json in the current working directory.\r\n * Returns empty object if file doesn't exist.\r\n * Throws with clear message if file exists but is invalid JSON.\r\n */\r\nexport function loadConfig(cwd: string = process.cwd()): XrmForgeConfig {\r\n const configPath = join(cwd, CONFIG_FILENAME);\r\n\r\n if (!existsSync(configPath)) {\r\n return {};\r\n }\r\n\r\n try {\r\n const raw = readFileSync(configPath, 'utf-8');\r\n const config = JSON.parse(raw) as XrmForgeConfig;\r\n\r\n // Warn about secrets in config file\r\n if (config.clientSecret) {\r\n console.warn(`WARNING: clientSecret found in ${CONFIG_FILENAME}. This is a security risk.`);\r\n console.warn(' Use XRMFORGE_CLIENT_SECRET environment variable instead.\\n');\r\n }\r\n\r\n return config;\r\n } catch (error) {\r\n if (error instanceof SyntaxError) {\r\n throw new Error(`Invalid JSON in ${configPath}: ${error.message}`);\r\n }\r\n throw error;\r\n }\r\n}\r\n\r\n/**\r\n * Merge config file values with CLI options.\r\n * CLI flags take precedence over config file.\r\n */\r\nexport function mergeWithCliOptions(\r\n config: XrmForgeConfig,\r\n cliOpts: Record<string, unknown>,\r\n): Record<string, unknown> {\r\n const merged: Record<string, unknown> = { ...cliOpts };\r\n\r\n // Only fill in values from config that weren't set via CLI\r\n if (!merged['url'] && config.url) merged['url'] = config.url;\r\n if (!merged['auth'] && config.auth) merged['auth'] = config.auth;\r\n if (!merged['tenantId'] && config.tenantId) merged['tenantId'] = config.tenantId;\r\n if (!merged['clientId'] && config.clientId) merged['clientId'] = config.clientId;\r\n if (!merged['clientSecret'] && config.clientSecret) merged['clientSecret'] = config.clientSecret;\r\n // Solutions: CLI comma-separated string vs config array\r\n if (!merged['solutions'] && config.solutions) {\r\n merged['solutions'] = Array.isArray(config.solutions)\r\n ? config.solutions.join(',')\r\n : config.solutions;\r\n }\r\n if (!merged['output'] && config.output) merged['output'] = config.output;\r\n\r\n // Entities: CLI comma-separated string vs config array\r\n if (!merged['entities'] && config.entities) {\r\n merged['entities'] = config.entities.join(',');\r\n }\r\n\r\n // Label languages: config uses numbers, CLI uses strings\r\n if (!merged['labelLanguage'] && config.labelLanguage) {\r\n merged['labelLanguage'] = String(config.labelLanguage);\r\n }\r\n if (!merged['secondaryLanguage'] && config.secondaryLanguage) {\r\n merged['secondaryLanguage'] = String(config.secondaryLanguage);\r\n }\r\n\r\n // Booleans: only override if explicitly set in config\r\n if (merged['forms'] === undefined && config.forms !== undefined) {\r\n merged['forms'] = config.forms;\r\n }\r\n if (merged['optionsets'] === undefined && config.optionsets !== undefined) {\r\n merged['optionsets'] = config.optionsets;\r\n }\r\n\r\n // Cache options\r\n if (merged['cache'] === undefined && config.cache !== undefined) {\r\n merged['cache'] = config.cache;\r\n }\r\n if (!merged['cacheDir'] && config.cacheDir) {\r\n merged['cacheDir'] = config.cacheDir;\r\n }\r\n\r\n return merged;\r\n}\r\n","/**\r\n * @xrmforge/cli - Generate Command\r\n *\r\n * Orchestrates type generation from a Dataverse environment.\r\n *\r\n * Usage:\r\n * xrmforge generate --url https://myorg.crm4.dynamics.com \\\r\n * --auth client-credentials \\\r\n * --tenant <tenant-id> --client-id <app-id> --client-secret <secret> \\\r\n * --entities account,contact \\\r\n * --output ./typings\r\n *\r\n * xrmforge generate --url https://myorg.crm4.dynamics.com \\\r\n * --auth interactive \\\r\n * --tenant <tenant-id> --client-id <app-id> \\\r\n * --entities account,contact,opportunity \\\r\n * --output ./typings \\\r\n * --label-language 1033 --secondary-language 1031\r\n */\r\n\r\nimport type { Command } from 'commander';\r\nimport { loadConfig, mergeWithCliOptions } from '../config.js';\r\nimport {\r\n TypeGenerationOrchestrator,\r\n createCredential,\r\n configureLogging,\r\n ConsoleLogSink,\r\n LogLevel,\r\n} from '@xrmforge/typegen';\r\nimport type { AuthConfig } from '@xrmforge/typegen';\r\n\r\n/** CLI options for the generate command */\r\ninterface GenerateOptions {\r\n url: string;\r\n auth: string;\r\n tenantId?: string;\r\n clientId?: string;\r\n clientSecret?: string;\r\n token?: string;\r\n entities?: string;\r\n solutions?: string;\r\n output: string;\r\n labelLanguage: string;\r\n secondaryLanguage?: string;\r\n forms: boolean;\r\n optionsets: boolean;\r\n actions: boolean;\r\n actionsFilter?: string;\r\n cache: boolean;\r\n cacheDir: string;\r\n verbose: boolean;\r\n}\r\n\r\n/**\r\n * Register the 'generate' subcommand on the CLI program.\r\n */\r\nexport function registerGenerateCommand(program: Command): void {\r\n program\r\n .command('generate')\r\n .description('Generate TypeScript declarations from a Dataverse environment')\r\n\r\n // Connection (can come from xrmforge.config.json)\r\n .option('--url <url>', 'Dataverse environment URL (e.g. https://myorg.crm4.dynamics.com)')\r\n .option('--auth <method>', 'Authentication method: client-credentials, interactive, device-code, token')\r\n\r\n // Auth credentials\r\n .option('--tenant-id <id>', 'Azure AD tenant ID')\r\n .option('--client-id <id>', 'Azure AD application (client) ID')\r\n .option('--client-secret <secret>', 'Client secret (for client-credentials auth)')\r\n .option('--token <token>', 'Pre-acquired Bearer token (for --auth token). Prefer XRMFORGE_TOKEN env var for security.')\r\n\r\n // Scope\r\n .option('--entities <list>', 'Comma-separated list of entity logical names (e.g. account,contact)')\r\n .option('--solutions <list>', 'Comma-separated solution unique names to discover entities')\r\n\r\n // Output\r\n .option('--output <dir>', 'Output directory for generated .d.ts files', './typings')\r\n\r\n // Labels\r\n .option('--label-language <code>', 'Primary label language code', '1033')\r\n .option('--secondary-language <code>', 'Secondary label language code (for dual-language JSDoc)')\r\n\r\n // Feature toggles\r\n .option('--no-forms', 'Skip form interface generation')\r\n .option('--no-optionsets', 'Skip OptionSet enum generation')\r\n .option('--actions', 'Generate Custom API Action/Function executors', false)\r\n .option('--actions-filter <prefix>', 'Only generate Custom APIs whose uniquename starts with this prefix (e.g. \"markant_\")')\r\n\r\n // Cache\r\n .option('--cache', 'Enable metadata cache for incremental generation', false)\r\n .option('--no-cache', 'Force full metadata refresh (disables cache)')\r\n .option('--cache-dir <dir>', 'Directory for metadata cache files', '.xrmforge/cache')\r\n\r\n // Verbosity\r\n .option('-v, --verbose', 'Enable verbose logging', false)\r\n\r\n .action(async (opts: GenerateOptions) => {\r\n try {\r\n await runGenerate(opts);\r\n } catch (error) {\r\n if (error instanceof Error) {\r\n console.error(`\\nError: ${error.message}\\n`);\r\n if (opts.verbose && error.stack) {\r\n console.error(error.stack);\r\n }\r\n } else {\r\n console.error('\\nAn unexpected error occurred.\\n');\r\n }\r\n process.exitCode = 1;\r\n }\r\n });\r\n}\r\n\r\n/**\r\n * Execute the generate command.\r\n */\r\nasync function runGenerate(cliOpts: GenerateOptions): Promise<void> {\r\n // Load config file and merge with CLI options (CLI takes precedence)\r\n const fileConfig = loadConfig();\r\n const merged = mergeWithCliOptions(fileConfig, cliOpts as unknown as Record<string, unknown>);\r\n const opts = merged as unknown as GenerateOptions;\r\n\r\n // Configure logging\r\n configureLogging({\r\n sink: new ConsoleLogSink(),\r\n minLevel: opts.verbose ? LogLevel.DEBUG : LogLevel.INFO,\r\n });\r\n\r\n // Validate required options (may come from config file)\r\n if (!opts.url) {\r\n throw new Error('--url is required. Set it via CLI flag or in xrmforge.config.json.');\r\n }\r\n if (!opts.auth) {\r\n throw new Error('--auth is required. Set it via CLI flag or in xrmforge.config.json.');\r\n }\r\n if (!opts.entities && !opts.solutions) {\r\n throw new Error('Either --entities or --solutions must be specified (CLI or xrmforge.config.json).');\r\n }\r\n\r\n // Build auth config\r\n const authConfig = buildAuthConfig(opts);\r\n const credential = createCredential(authConfig);\r\n\r\n // Parse entity list\r\n const entities = opts.entities\r\n ? opts.entities.split(',').map((e) => e.trim().toLowerCase())\r\n : [];\r\n\r\n // Parse solutions list\r\n const solutionNames = opts.solutions\r\n ? opts.solutions.split(',').map((s) => s.trim())\r\n : [];\r\n\r\n if (entities.length === 0 && solutionNames.length === 0) {\r\n throw new Error('No entities specified. Use --entities or --solutions.');\r\n }\r\n\r\n // Build label config (R8-05: validate LCID)\r\n const primaryLanguage = parseInt(opts.labelLanguage, 10);\r\n if (isNaN(primaryLanguage)) {\r\n throw new Error(`Invalid --label-language: \"${opts.labelLanguage}\". Must be a numeric LCID (e.g. 1033, 1031).`);\r\n }\r\n let secondaryLanguage: number | undefined;\r\n if (opts.secondaryLanguage) {\r\n secondaryLanguage = parseInt(opts.secondaryLanguage, 10);\r\n if (isNaN(secondaryLanguage)) {\r\n throw new Error(`Invalid --secondary-language: \"${opts.secondaryLanguage}\". Must be a numeric LCID (e.g. 1033, 1031).`);\r\n }\r\n }\r\n\r\n console.log(`\\nXrmForge Type Generator`);\r\n console.log(`Environment: ${opts.url}`);\r\n console.log(`Auth method: ${opts.auth}`);\r\n console.log(`Entities: ${entities.length > 0 ? entities.join(', ') : '(none specified directly)'}`)\r\n if (solutionNames.length > 0) {\r\n console.log(`Solutions: ${solutionNames.join(', ')}`);\r\n }\r\n console.log(`Output: ${opts.output}`);\r\n console.log(`Languages: ${primaryLanguage}${secondaryLanguage ? ` + ${secondaryLanguage}` : ''}`);\r\n if (opts.cache) {\r\n console.log(`Cache: enabled (${opts.cacheDir})`);\r\n }\r\n console.log('');\r\n\r\n // Create orchestrator and run\r\n const orchestrator = new TypeGenerationOrchestrator(credential, {\r\n environmentUrl: opts.url,\r\n entities,\r\n solutionNames: solutionNames.length > 0 ? solutionNames : undefined,\r\n outputDir: opts.output,\r\n labelConfig: { primaryLanguage, secondaryLanguage },\r\n generateForms: opts.forms,\r\n generateOptionSets: opts.optionsets,\r\n generateActions: opts.actions,\r\n actionsFilter: opts.actionsFilter,\r\n useCache: opts.cache,\r\n cacheDir: opts.cacheDir,\r\n });\r\n\r\n // Support Ctrl+C and SIGTERM (R8-07: Docker/K8s sends SIGTERM)\r\n const controller = new AbortController();\r\n const onSignal = () => {\r\n console.log('\\nAborting generation...');\r\n controller.abort();\r\n };\r\n process.once('SIGINT', onSignal);\r\n process.once('SIGTERM', onSignal);\r\n\r\n const result = await orchestrator.generate({ signal: controller.signal });\r\n\r\n // Summary\r\n console.log('');\r\n console.log('Generation complete:');\r\n console.log(` Entities: ${result.entities.length}`);\r\n console.log(` Files: ${result.totalFiles}`);\r\n console.log(` Warnings: ${result.totalWarnings}`);\r\n console.log(` Duration: ${result.durationMs}ms`);\r\n if (result.cacheStats) {\r\n const cs = result.cacheStats;\r\n if (cs.fullRefresh) {\r\n console.log(` Cache: full refresh (${cs.entitiesFetched} entities fetched)`);\r\n } else {\r\n console.log(` Cache: ${cs.entitiesFromCache} from cache, ${cs.entitiesFetched} fetched, ${cs.entitiesDeleted} deleted`);\r\n }\r\n }\r\n\r\n // Show warnings\r\n if (result.totalWarnings > 0) {\r\n console.log('\\nWarnings:');\r\n for (const entity of result.entities) {\r\n for (const warning of entity.warnings) {\r\n console.log(` [${entity.entityLogicalName}] ${warning}`);\r\n }\r\n }\r\n }\r\n\r\n // Show failures\r\n const failures = result.entities.filter((e) => e.files.length === 0 && e.warnings.length > 0);\r\n if (failures.length > 0) {\r\n console.log(`\\n${failures.length} entity/entities failed. See warnings above.`);\r\n process.exitCode = 1;\r\n return;\r\n }\r\n\r\n console.log(`\\nTypes written to: ${opts.output}/`);\r\n}\r\n\r\n/**\r\n * Build AuthConfig from CLI options.\r\n */\r\nfunction buildAuthConfig(opts: GenerateOptions): AuthConfig {\r\n const method = opts.auth as AuthConfig['method'];\r\n\r\n switch (method) {\r\n case 'client-credentials':\r\n if (!opts.tenantId) throw new Error('--tenant-id is required for client-credentials auth.');\r\n if (!opts.clientId) throw new Error('--client-id is required for client-credentials auth.');\r\n if (!opts.clientSecret) throw new Error('--client-secret is required for client-credentials auth.');\r\n return {\r\n method: 'client-credentials',\r\n tenantId: opts.tenantId,\r\n clientId: opts.clientId,\r\n clientSecret: opts.clientSecret,\r\n };\r\n\r\n case 'interactive':\r\n if (!opts.tenantId) throw new Error('--tenant-id is required for interactive auth.');\r\n if (!opts.clientId) throw new Error('--client-id is required for interactive auth.');\r\n return {\r\n method: 'interactive',\r\n tenantId: opts.tenantId,\r\n clientId: opts.clientId,\r\n };\r\n\r\n case 'device-code':\r\n if (!opts.tenantId) throw new Error('--tenant-id is required for device-code auth.');\r\n if (!opts.clientId) throw new Error('--client-id is required for device-code auth.');\r\n return {\r\n method: 'device-code',\r\n tenantId: opts.tenantId,\r\n clientId: opts.clientId,\r\n };\r\n\r\n case 'token': {\r\n // Token from --token flag or XRMFORGE_TOKEN environment variable\r\n const token = opts.token || process.env['XRMFORGE_TOKEN'];\r\n if (!token) {\r\n throw new Error(\r\n 'Token authentication requires a token. ' +\r\n 'Set XRMFORGE_TOKEN environment variable or use --token flag.',\r\n );\r\n }\r\n if (opts.token) {\r\n console.warn('WARNING: Using --token on the command line exposes the token in process list and shell history.');\r\n console.warn(' Prefer setting XRMFORGE_TOKEN environment variable instead.\\n');\r\n }\r\n return { method: 'token', token };\r\n }\r\n\r\n default:\r\n throw new Error(\r\n `Unknown auth method: \"${opts.auth}\". ` +\r\n `Supported: client-credentials, interactive, device-code, token`,\r\n );\r\n }\r\n}\r\n","/**\n * @xrmforge/cli - Build Command\n *\n * Builds WebResources as IIFE bundles for Dynamics 365.\n *\n * Usage:\n * xrmforge build # Build all entries from xrmforge.config.json\n * xrmforge build --watch # Watch mode with incremental rebuilds\n * xrmforge build --minify # Minify output bundles\n * xrmforge build --no-sourcemap # Disable source maps\n */\n\nimport type { Command } from 'commander';\nimport { loadConfig } from '../config.js';\nimport {\n validateBuildConfig,\n build,\n watch,\n} from '@xrmforge/devkit';\n\n/** CLI options for the build command */\ninterface BuildOptions {\n watch: boolean;\n minify?: boolean;\n sourcemap: boolean;\n outDir?: string;\n verbose: boolean;\n}\n\n/**\n * Register the 'build' subcommand on the CLI program.\n */\nexport function registerBuildCommand(program: Command): void {\n program\n .command('build')\n .description('Build WebResources as IIFE bundles for Dynamics 365')\n\n .option('--watch', 'Watch mode with incremental rebuilds', false)\n .option('--minify', 'Minify output bundles')\n .option('--no-sourcemap', 'Disable source maps')\n .option('--out-dir <dir>', 'Override output directory')\n .option('-v, --verbose', 'Verbose logging', false)\n\n .action(async (opts: BuildOptions) => {\n try {\n await runBuild(opts);\n } catch (error) {\n if (error instanceof Error) {\n console.error(`\\nError: ${error.message}\\n`);\n if (opts.verbose && error.stack) {\n console.error(error.stack);\n }\n } else {\n console.error('\\nAn unexpected error occurred.\\n');\n }\n process.exitCode = 1;\n }\n });\n}\n\n/**\n * Execute the build command.\n */\nasync function runBuild(opts: BuildOptions): Promise<void> {\n const fileConfig = loadConfig();\n\n if (!fileConfig.build) {\n throw new Error(\n 'No \"build\" section found in xrmforge.config.json.\\n' +\n 'Add a build configuration with entries to get started:\\n\\n' +\n ' {\\n' +\n ' \"build\": {\\n' +\n ' \"entries\": {\\n' +\n ' \"my_script\": {\\n' +\n ' \"input\": \"./src/my-script.ts\",\\n' +\n ' \"namespace\": \"Contoso.MyScript\"\\n' +\n ' }\\n' +\n ' }\\n' +\n ' }\\n' +\n ' }\\n',\n );\n }\n\n // Apply CLI overrides\n const buildConfig = { ...fileConfig.build };\n if (opts.outDir) buildConfig.outDir = opts.outDir;\n if (opts.minify !== undefined) buildConfig.minify = opts.minify;\n if (!opts.sourcemap) buildConfig.sourcemap = false;\n\n // Validate\n const validated = validateBuildConfig(buildConfig);\n\n const entryCount = Object.keys(validated.entries).length;\n console.log(`\\nXrmForge Build (esbuild)`);\n console.log(`Entries: ${entryCount}`);\n console.log('');\n\n if (opts.watch) {\n // Watch mode\n console.log('Watching for changes...\\n');\n\n const watcher = await watch(validated, {\n onRebuild: (result) => {\n if (result.errors.length > 0) {\n for (const err of result.errors) {\n console.error(` ${err}`);\n }\n } else {\n console.log(` Rebuilt ${result.entries.length} entries in ${result.totalDurationMs}ms`);\n }\n },\n });\n\n // Support Ctrl+C and SIGTERM\n const onSignal = () => {\n console.log('\\nStopping watch mode...');\n watcher.dispose().then(() => process.exit(0));\n };\n process.once('SIGINT', onSignal);\n process.once('SIGTERM', onSignal);\n\n // Keep process alive\n await new Promise(() => {});\n }\n\n // Single build\n const result = await build(validated);\n\n // Print results table\n for (const entry of result.entries) {\n const size = formatSize(entry.sizeBytes);\n const duration = `${entry.durationMs}ms`;\n console.log(` ${entry.name.padEnd(30)} ${size.padStart(10)} ${duration.padStart(8)}`);\n }\n\n console.log('');\n\n if (result.errors.length > 0) {\n console.log('Errors:');\n for (const err of result.errors) {\n console.error(` ${err}`);\n }\n console.log('');\n process.exitCode = 1;\n return;\n }\n\n console.log(`${result.entries.length} entries built in ${result.totalDurationMs}ms\\n`);\n}\n\n/** Format bytes to human-readable string */\nfunction formatSize(bytes: number): string {\n if (bytes < 1024) return `${bytes} B`;\n const kb = bytes / 1024;\n return `${kb.toFixed(1)} kB`;\n}\n","/**\n * @xrmforge/cli - Init Command\n *\n * Scaffolds a new D365 form scripting project.\n *\n * Usage:\n * xrmforge init # Scaffold in current directory\n * xrmforge init my-project # Scaffold in ./my-project\n * xrmforge init --prefix markant # Use \"markant\" as publisher prefix\n * xrmforge init --namespace Markant # Use \"Markant\" as base namespace\n * xrmforge init --skip-install # Don't run npm install\n */\n\nimport type { Command } from 'commander';\nimport { resolve, basename } from 'node:path';\nimport { scaffoldProject } from '@xrmforge/devkit';\n\n/** CLI options for the init command */\ninterface InitOptions {\n name?: string;\n prefix: string;\n namespace?: string;\n skipInstall: boolean;\n}\n\n/**\n * Register the 'init' subcommand on the CLI program.\n */\nexport function registerInitCommand(program: Command): void {\n program\n .command('init [dir]')\n .description('Scaffold a new D365 form scripting project')\n\n .option('--name <name>', 'Project name for package.json (default: directory name)')\n .option('--prefix <prefix>', 'Publisher prefix for D365 WebResources', 'contoso')\n .option('--namespace <ns>', 'Base namespace for form scripts (default: PascalCase of prefix)')\n .option('--skip-install', 'Skip running npm install after scaffolding', false)\n\n .action(async (dir: string | undefined, opts: InitOptions) => {\n try {\n await runInit(dir, opts);\n } catch (error) {\n if (error instanceof Error) {\n console.error(`\\nError: ${error.message}\\n`);\n } else {\n console.error('\\nAn unexpected error occurred.\\n');\n }\n process.exitCode = 1;\n }\n });\n}\n\n/**\n * Execute the init command.\n */\nasync function runInit(dir: string | undefined, opts: InitOptions): Promise<void> {\n const targetDir = resolve(dir ?? '.');\n const dirName = basename(targetDir);\n\n const projectName = opts.name ?? dirName;\n const prefix = opts.prefix;\n const namespace = opts.namespace ?? toPascalCase(prefix);\n\n console.log(`\\nXrmForge Project Scaffolding`);\n console.log(`Directory: ${targetDir}`);\n console.log(`Name: ${projectName}`);\n console.log(`Prefix: ${prefix}`);\n console.log(`Namespace: ${namespace}`);\n console.log('');\n\n const result = await scaffoldProject({\n targetDir,\n projectName,\n prefix,\n namespace,\n });\n\n console.log(`Created ${result.filesCreated.length} files:`);\n for (const file of result.filesCreated) {\n console.log(` ${file}`);\n }\n\n if (result.warnings.length > 0) {\n console.log('\\nWarnings:');\n for (const w of result.warnings) {\n console.log(` ${w}`);\n }\n }\n\n console.log('\\nNext steps:');\n if (dir) {\n console.log(` cd ${dir}`);\n }\n if (!opts.skipInstall) {\n console.log(' npm install');\n }\n console.log(' xrmforge generate --url https://YOUR-ORG.crm4.dynamics.com --auth interactive --entities account --output ./typings');\n console.log(' xrmforge build');\n console.log('');\n}\n\n/** Convert a string to PascalCase (e.g. \"my-prefix\" -> \"MyPrefix\") */\nfunction toPascalCase(str: string): string {\n return str\n .split(/[-_\\s]+/)\n .map((part) => part.charAt(0).toUpperCase() + part.slice(1).toLowerCase())\n .join('');\n}\n"],"mappings":"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","names":["readFileSync","join","program","program","result","program","readFileSync","join"]}
1
+ {"version":3,"sources":["../src/index.ts","../src/config.ts","../src/commands/generate.ts","../src/commands/build.ts","../src/commands/init.ts"],"sourcesContent":["/**\r\n * @xrmforge/cli - Command-line interface for XrmForge\r\n *\r\n * Usage:\r\n * xrmforge generate --url https://myorg.crm4.dynamics.com \\\r\n * --auth client-credentials \\\r\n * --tenant <tenant-id> --client-id <app-id> --client-secret <secret> \\\r\n * --entities account,contact \\\r\n * --output ./typings\r\n */\r\n\r\nimport { readFileSync } from 'node:fs';\r\nimport { dirname, join } from 'node:path';\r\nimport { fileURLToPath } from 'node:url';\r\nimport { Command } from 'commander';\r\nimport { registerGenerateCommand } from './commands/generate.js';\r\nimport { registerBuildCommand } from './commands/build.js';\r\nimport { registerInitCommand } from './commands/init.js';\r\n\r\n// Read version from package.json (single source of truth)\r\nconst __dirname = dirname(fileURLToPath(import.meta.url));\r\nconst pkg = JSON.parse(readFileSync(join(__dirname, '..', 'package.json'), 'utf-8'));\r\n\r\nconst program = new Command();\r\n\r\nprogram\r\n .name('xrmforge')\r\n .description('TypeScript type generator for Dynamics 365 / Dataverse')\r\n .version(pkg.version);\r\n\r\nregisterGenerateCommand(program);\r\nregisterBuildCommand(program);\r\nregisterInitCommand(program);\r\n\r\nprogram.parse();\r\n","/**\r\n * @xrmforge/cli - Configuration File Support\r\n *\r\n * Reads xrmforge.config.json from the current working directory.\r\n * CLI flags override config file values.\r\n *\r\n * Example xrmforge.config.json:\r\n * ```json\r\n * {\r\n * \"url\": \"https://myorg.crm4.dynamics.com\",\r\n * \"auth\": \"interactive\",\r\n * \"tenantId\": \"your-tenant-id\",\r\n * \"solutions\": [\"MySolution\", \"MyOtherSolution\"],\r\n * \"entities\": [\"systemuser\", \"task\"],\r\n * \"output\": \"./typings\",\r\n * \"labelLanguage\": 1033,\r\n * \"secondaryLanguage\": 1031\r\n * }\r\n * ```\r\n */\r\n\r\nimport { readFileSync, existsSync } from 'node:fs';\r\nimport { join } from 'node:path';\r\nimport type { BuildConfig } from '@xrmforge/devkit';\r\n\r\n/** Shape of xrmforge.config.json */\r\nexport interface XrmForgeConfig {\r\n /** Dataverse environment URL */\r\n url?: string;\r\n /** Authentication method */\r\n auth?: string;\r\n /** Azure AD tenant ID */\r\n tenantId?: string;\r\n /** Azure AD application (client) ID */\r\n clientId?: string;\r\n /** Client secret (NOT recommended in config file, use env vars) */\r\n clientSecret?: string;\r\n /** Entity logical names */\r\n entities?: string[];\r\n /** Solution unique names (array or comma-separated string) */\r\n solutions?: string[] | string;\r\n /** Output directory */\r\n output?: string;\r\n /** Primary label language LCID */\r\n labelLanguage?: number;\r\n /** Secondary label language LCID */\r\n secondaryLanguage?: number;\r\n /** Generate form interfaces */\r\n forms?: boolean;\r\n /** Generate OptionSet enums */\r\n optionsets?: boolean;\r\n /** Enable metadata cache for incremental generation */\r\n cache?: boolean;\r\n /** Directory for metadata cache files */\r\n cacheDir?: string;\r\n /** Build configuration for WebResource bundling */\r\n build?: BuildConfig;\r\n}\r\n\r\nconst CONFIG_FILENAME = 'xrmforge.config.json';\r\n\r\n/**\r\n * Load config from xrmforge.config.json in the current working directory.\r\n * Returns empty object if file doesn't exist.\r\n * Throws with clear message if file exists but is invalid JSON.\r\n */\r\nexport function loadConfig(cwd: string = process.cwd()): XrmForgeConfig {\r\n const configPath = join(cwd, CONFIG_FILENAME);\r\n\r\n if (!existsSync(configPath)) {\r\n return {};\r\n }\r\n\r\n try {\r\n const raw = readFileSync(configPath, 'utf-8');\r\n const config = JSON.parse(raw) as XrmForgeConfig;\r\n\r\n // Warn about secrets in config file\r\n if (config.clientSecret) {\r\n console.warn(`WARNING: clientSecret found in ${CONFIG_FILENAME}. This is a security risk.`);\r\n console.warn(' Use XRMFORGE_CLIENT_SECRET environment variable instead.\\n');\r\n }\r\n\r\n return config;\r\n } catch (error) {\r\n if (error instanceof SyntaxError) {\r\n throw new Error(`Invalid JSON in ${configPath}: ${error.message}`);\r\n }\r\n throw error;\r\n }\r\n}\r\n\r\n/**\r\n * Merge config file values with CLI options.\r\n * CLI flags take precedence over config file.\r\n */\r\nexport function mergeWithCliOptions(\r\n config: XrmForgeConfig,\r\n cliOpts: Record<string, unknown>,\r\n): Record<string, unknown> {\r\n const merged: Record<string, unknown> = { ...cliOpts };\r\n\r\n // Only fill in values from config that weren't set via CLI\r\n if (!merged['url'] && config.url) merged['url'] = config.url;\r\n if (!merged['auth'] && config.auth) merged['auth'] = config.auth;\r\n if (!merged['tenantId'] && config.tenantId) merged['tenantId'] = config.tenantId;\r\n if (!merged['clientId'] && config.clientId) merged['clientId'] = config.clientId;\r\n if (!merged['clientSecret'] && config.clientSecret) merged['clientSecret'] = config.clientSecret;\r\n // Solutions: CLI comma-separated string vs config array\r\n if (!merged['solutions'] && config.solutions) {\r\n merged['solutions'] = Array.isArray(config.solutions)\r\n ? config.solutions.join(',')\r\n : config.solutions;\r\n }\r\n if (!merged['output'] && config.output) merged['output'] = config.output;\r\n\r\n // Entities: CLI comma-separated string vs config array\r\n if (!merged['entities'] && config.entities) {\r\n merged['entities'] = config.entities.join(',');\r\n }\r\n\r\n // Label languages: config uses numbers, CLI uses strings\r\n if (!merged['labelLanguage'] && config.labelLanguage) {\r\n merged['labelLanguage'] = String(config.labelLanguage);\r\n }\r\n if (!merged['secondaryLanguage'] && config.secondaryLanguage) {\r\n merged['secondaryLanguage'] = String(config.secondaryLanguage);\r\n }\r\n\r\n // Booleans: only override if explicitly set in config\r\n if (merged['forms'] === undefined && config.forms !== undefined) {\r\n merged['forms'] = config.forms;\r\n }\r\n if (merged['optionsets'] === undefined && config.optionsets !== undefined) {\r\n merged['optionsets'] = config.optionsets;\r\n }\r\n\r\n // Cache options\r\n if (merged['cache'] === undefined && config.cache !== undefined) {\r\n merged['cache'] = config.cache;\r\n }\r\n if (!merged['cacheDir'] && config.cacheDir) {\r\n merged['cacheDir'] = config.cacheDir;\r\n }\r\n\r\n return merged;\r\n}\r\n","/**\r\n * @xrmforge/cli - Generate Command\r\n *\r\n * Orchestrates type generation from a Dataverse environment.\r\n *\r\n * Usage:\r\n * xrmforge generate --url https://myorg.crm4.dynamics.com \\\r\n * --auth client-credentials \\\r\n * --tenant <tenant-id> --client-id <app-id> --client-secret <secret> \\\r\n * --entities account,contact \\\r\n * --output ./typings\r\n *\r\n * xrmforge generate --url https://myorg.crm4.dynamics.com \\\r\n * --auth interactive \\\r\n * --tenant <tenant-id> --client-id <app-id> \\\r\n * --entities account,contact,opportunity \\\r\n * --output ./typings \\\r\n * --label-language 1033 --secondary-language 1031\r\n */\r\n\r\nimport type { Command } from 'commander';\r\nimport { loadConfig, mergeWithCliOptions } from '../config.js';\r\nimport {\r\n TypeGenerationOrchestrator,\r\n createCredential,\r\n configureLogging,\r\n ConsoleLogSink,\r\n LogLevel,\r\n} from '@xrmforge/typegen';\r\nimport type { AuthConfig } from '@xrmforge/typegen';\r\n\r\n/** CLI options for the generate command */\r\ninterface GenerateOptions {\r\n url: string;\r\n auth: string;\r\n tenantId?: string;\r\n clientId?: string;\r\n clientSecret?: string;\r\n token?: string;\r\n entities?: string;\r\n solutions?: string;\r\n output: string;\r\n labelLanguage: string;\r\n secondaryLanguage?: string;\r\n forms: boolean;\r\n optionsets: boolean;\r\n actions: boolean;\r\n actionsFilter?: string;\r\n cache: boolean;\r\n cacheDir: string;\r\n verbose: boolean;\r\n}\r\n\r\n/**\r\n * Register the 'generate' subcommand on the CLI program.\r\n */\r\nexport function registerGenerateCommand(program: Command): void {\r\n program\r\n .command('generate')\r\n .description('Generate TypeScript declarations from a Dataverse environment')\r\n\r\n // Connection (can come from xrmforge.config.json)\r\n .option('--url <url>', 'Dataverse environment URL (e.g. https://myorg.crm4.dynamics.com)')\r\n .option('--auth <method>', 'Authentication method: client-credentials, interactive, device-code, token')\r\n\r\n // Auth credentials\r\n .option('--tenant-id <id>', 'Azure AD tenant ID')\r\n .option('--client-id <id>', 'Azure AD application (client) ID')\r\n .option('--client-secret <secret>', 'Client secret (for client-credentials auth)')\r\n .option('--token <token>', 'Pre-acquired Bearer token (for --auth token). Prefer XRMFORGE_TOKEN env var for security.')\r\n\r\n // Scope\r\n .option('--entities <list>', 'Comma-separated list of entity logical names (e.g. account,contact)')\r\n .option('--solutions <list>', 'Comma-separated solution unique names to discover entities')\r\n\r\n // Output\r\n .option('--output <dir>', 'Output directory for generated .d.ts files', './typings')\r\n\r\n // Labels\r\n .option('--label-language <code>', 'Primary label language code', '1033')\r\n .option('--secondary-language <code>', 'Secondary label language code (for dual-language JSDoc)')\r\n\r\n // Feature toggles\r\n .option('--no-forms', 'Skip form interface generation')\r\n .option('--no-optionsets', 'Skip OptionSet enum generation')\r\n .option('--actions', 'Generate Custom API Action/Function executors', false)\r\n .option('--actions-filter <prefix>', 'Only generate Custom APIs whose uniquename starts with this prefix (e.g. \"markant_\")')\r\n\r\n // Cache\r\n .option('--cache', 'Enable metadata cache for incremental generation', false)\r\n .option('--no-cache', 'Force full metadata refresh (disables cache)')\r\n .option('--cache-dir <dir>', 'Directory for metadata cache files', '.xrmforge/cache')\r\n\r\n // Verbosity\r\n .option('-v, --verbose', 'Enable verbose logging', false)\r\n\r\n .action(async (opts: GenerateOptions) => {\r\n try {\r\n await runGenerate(opts);\r\n } catch (error) {\r\n if (error instanceof Error) {\r\n console.error(`\\nError: ${error.message}\\n`);\r\n if (opts.verbose && error.stack) {\r\n console.error(error.stack);\r\n }\r\n } else {\r\n console.error('\\nAn unexpected error occurred.\\n');\r\n }\r\n process.exitCode = 1;\r\n }\r\n });\r\n}\r\n\r\n/**\r\n * Execute the generate command.\r\n */\r\nasync function runGenerate(cliOpts: GenerateOptions): Promise<void> {\r\n // Load config file and merge with CLI options (CLI takes precedence)\r\n const fileConfig = loadConfig();\r\n const merged = mergeWithCliOptions(fileConfig, cliOpts as unknown as Record<string, unknown>);\r\n const opts = merged as unknown as GenerateOptions;\r\n\r\n // Configure logging\r\n configureLogging({\r\n sink: new ConsoleLogSink(),\r\n minLevel: opts.verbose ? LogLevel.DEBUG : LogLevel.INFO,\r\n });\r\n\r\n // Validate required options (may come from config file)\r\n if (!opts.url) {\r\n throw new Error('--url is required. Set it via CLI flag or in xrmforge.config.json.');\r\n }\r\n if (!opts.auth) {\r\n throw new Error('--auth is required. Set it via CLI flag or in xrmforge.config.json.');\r\n }\r\n if (!opts.entities && !opts.solutions) {\r\n throw new Error('Either --entities or --solutions must be specified (CLI or xrmforge.config.json).');\r\n }\r\n\r\n // Build auth config\r\n const authConfig = buildAuthConfig(opts);\r\n const credential = createCredential(authConfig);\r\n\r\n // Parse entity list\r\n const entities = opts.entities\r\n ? opts.entities.split(',').map((e) => e.trim().toLowerCase())\r\n : [];\r\n\r\n // Parse solutions list\r\n const solutionNames = opts.solutions\r\n ? opts.solutions.split(',').map((s) => s.trim())\r\n : [];\r\n\r\n if (entities.length === 0 && solutionNames.length === 0) {\r\n throw new Error('No entities specified. Use --entities or --solutions.');\r\n }\r\n\r\n // Build label config (R8-05: validate LCID)\r\n const primaryLanguage = parseInt(opts.labelLanguage, 10);\r\n if (isNaN(primaryLanguage)) {\r\n throw new Error(`Invalid --label-language: \"${opts.labelLanguage}\". Must be a numeric LCID (e.g. 1033, 1031).`);\r\n }\r\n let secondaryLanguage: number | undefined;\r\n if (opts.secondaryLanguage) {\r\n secondaryLanguage = parseInt(opts.secondaryLanguage, 10);\r\n if (isNaN(secondaryLanguage)) {\r\n throw new Error(`Invalid --secondary-language: \"${opts.secondaryLanguage}\". Must be a numeric LCID (e.g. 1033, 1031).`);\r\n }\r\n }\r\n\r\n console.log(`\\nXrmForge Type Generator`);\r\n console.log(`Environment: ${opts.url}`);\r\n console.log(`Auth method: ${opts.auth}`);\r\n console.log(`Entities: ${entities.length > 0 ? entities.join(', ') : '(none specified directly)'}`)\r\n if (solutionNames.length > 0) {\r\n console.log(`Solutions: ${solutionNames.join(', ')}`);\r\n }\r\n console.log(`Output: ${opts.output}`);\r\n console.log(`Languages: ${primaryLanguage}${secondaryLanguage ? ` + ${secondaryLanguage}` : ''}`);\r\n if (opts.cache) {\r\n console.log(`Cache: enabled (${opts.cacheDir})`);\r\n }\r\n console.log('');\r\n\r\n // Create orchestrator and run\r\n const orchestrator = new TypeGenerationOrchestrator(credential, {\r\n environmentUrl: opts.url,\r\n entities,\r\n solutionNames: solutionNames.length > 0 ? solutionNames : undefined,\r\n outputDir: opts.output,\r\n labelConfig: { primaryLanguage, secondaryLanguage },\r\n generateForms: opts.forms,\r\n generateOptionSets: opts.optionsets,\r\n generateActions: opts.actions,\r\n actionsFilter: opts.actionsFilter,\r\n useCache: opts.cache,\r\n cacheDir: opts.cacheDir,\r\n });\r\n\r\n // Support Ctrl+C and SIGTERM (R8-07: Docker/K8s sends SIGTERM)\r\n const controller = new AbortController();\r\n const onSignal = () => {\r\n console.log('\\nAborting generation...');\r\n controller.abort();\r\n };\r\n process.once('SIGINT', onSignal);\r\n process.once('SIGTERM', onSignal);\r\n\r\n const result = await orchestrator.generate({ signal: controller.signal });\r\n\r\n // Summary\r\n console.log('');\r\n console.log('Generation complete:');\r\n console.log(` Entities: ${result.entities.length}`);\r\n console.log(` Files: ${result.totalFiles}`);\r\n console.log(` Warnings: ${result.totalWarnings}`);\r\n console.log(` Duration: ${result.durationMs}ms`);\r\n if (result.cacheStats) {\r\n const cs = result.cacheStats;\r\n if (cs.fullRefresh) {\r\n console.log(` Cache: full refresh (${cs.entitiesFetched} entities fetched)`);\r\n } else {\r\n console.log(` Cache: ${cs.entitiesFromCache} from cache, ${cs.entitiesFetched} fetched, ${cs.entitiesDeleted} deleted`);\r\n }\r\n }\r\n\r\n // Show warnings\r\n if (result.totalWarnings > 0) {\r\n console.log('\\nWarnings:');\r\n for (const entity of result.entities) {\r\n for (const warning of entity.warnings) {\r\n console.log(` [${entity.entityLogicalName}] ${warning}`);\r\n }\r\n }\r\n }\r\n\r\n // Show failures\r\n const failures = result.entities.filter((e) => e.files.length === 0 && e.warnings.length > 0);\r\n if (failures.length > 0) {\r\n console.log(`\\n${failures.length} entity/entities failed. See warnings above.`);\r\n process.exitCode = 1;\r\n return;\r\n }\r\n\r\n console.log(`\\nTypes written to: ${opts.output}/`);\r\n}\r\n\r\n/**\r\n * Build AuthConfig from CLI options.\r\n */\r\nfunction buildAuthConfig(opts: GenerateOptions): AuthConfig {\r\n const method = opts.auth as AuthConfig['method'];\r\n\r\n switch (method) {\r\n case 'client-credentials':\r\n if (!opts.tenantId) throw new Error('--tenant-id is required for client-credentials auth.');\r\n if (!opts.clientId) throw new Error('--client-id is required for client-credentials auth.');\r\n if (!opts.clientSecret) throw new Error('--client-secret is required for client-credentials auth.');\r\n return {\r\n method: 'client-credentials',\r\n tenantId: opts.tenantId,\r\n clientId: opts.clientId,\r\n clientSecret: opts.clientSecret,\r\n };\r\n\r\n case 'interactive':\r\n if (!opts.tenantId) throw new Error('--tenant-id is required for interactive auth.');\r\n if (!opts.clientId) throw new Error('--client-id is required for interactive auth.');\r\n return {\r\n method: 'interactive',\r\n tenantId: opts.tenantId,\r\n clientId: opts.clientId,\r\n };\r\n\r\n case 'device-code':\r\n if (!opts.tenantId) throw new Error('--tenant-id is required for device-code auth.');\r\n if (!opts.clientId) throw new Error('--client-id is required for device-code auth.');\r\n return {\r\n method: 'device-code',\r\n tenantId: opts.tenantId,\r\n clientId: opts.clientId,\r\n };\r\n\r\n case 'token': {\r\n // Token from --token flag or XRMFORGE_TOKEN environment variable\r\n const token = opts.token || process.env['XRMFORGE_TOKEN'];\r\n if (!token) {\r\n throw new Error(\r\n 'Token authentication requires a token. ' +\r\n 'Set XRMFORGE_TOKEN environment variable or use --token flag.',\r\n );\r\n }\r\n if (opts.token) {\r\n console.warn('WARNING: Using --token on the command line exposes the token in process list and shell history.');\r\n console.warn(' Prefer setting XRMFORGE_TOKEN environment variable instead.\\n');\r\n }\r\n return { method: 'token', token };\r\n }\r\n\r\n default:\r\n throw new Error(\r\n `Unknown auth method: \"${opts.auth}\". ` +\r\n `Supported: client-credentials, interactive, device-code, token`,\r\n );\r\n }\r\n}\r\n","/**\n * @xrmforge/cli - Build Command\n *\n * Builds WebResources as IIFE bundles for Dynamics 365.\n *\n * Usage:\n * xrmforge build # Build all entries from xrmforge.config.json\n * xrmforge build --watch # Watch mode with incremental rebuilds\n * xrmforge build --minify # Minify output bundles\n * xrmforge build --no-sourcemap # Disable source maps\n */\n\nimport type { Command } from 'commander';\nimport { loadConfig } from '../config.js';\nimport {\n validateBuildConfig,\n build,\n watch,\n} from '@xrmforge/devkit';\n\n/** CLI options for the build command */\ninterface BuildOptions {\n watch: boolean;\n minify?: boolean;\n sourcemap: boolean;\n outDir?: string;\n verbose: boolean;\n}\n\n/**\n * Register the 'build' subcommand on the CLI program.\n */\nexport function registerBuildCommand(program: Command): void {\n program\n .command('build')\n .description('Build WebResources as IIFE bundles for Dynamics 365')\n\n .option('--watch', 'Watch mode with incremental rebuilds', false)\n .option('--minify', 'Minify output bundles')\n .option('--no-sourcemap', 'Disable source maps')\n .option('--out-dir <dir>', 'Override output directory')\n .option('-v, --verbose', 'Verbose logging', false)\n\n .action(async (opts: BuildOptions) => {\n try {\n await runBuild(opts);\n } catch (error) {\n if (error instanceof Error) {\n console.error(`\\nError: ${error.message}\\n`);\n if (opts.verbose && error.stack) {\n console.error(error.stack);\n }\n } else {\n console.error('\\nAn unexpected error occurred.\\n');\n }\n process.exitCode = 1;\n }\n });\n}\n\n/**\n * Execute the build command.\n */\nasync function runBuild(opts: BuildOptions): Promise<void> {\n const fileConfig = loadConfig();\n\n if (!fileConfig.build) {\n throw new Error(\n 'No \"build\" section found in xrmforge.config.json.\\n' +\n 'Add a build configuration with entries to get started:\\n\\n' +\n ' {\\n' +\n ' \"build\": {\\n' +\n ' \"entries\": {\\n' +\n ' \"my_script\": {\\n' +\n ' \"input\": \"./src/my-script.ts\",\\n' +\n ' \"namespace\": \"Contoso.MyScript\"\\n' +\n ' }\\n' +\n ' }\\n' +\n ' }\\n' +\n ' }\\n',\n );\n }\n\n // Apply CLI overrides\n const buildConfig = { ...fileConfig.build };\n if (opts.outDir) buildConfig.outDir = opts.outDir;\n if (opts.minify !== undefined) buildConfig.minify = opts.minify;\n if (!opts.sourcemap) buildConfig.sourcemap = false;\n\n // Validate\n const validated = validateBuildConfig(buildConfig);\n\n const entryCount = Object.keys(validated.entries).length;\n console.log(`\\nXrmForge Build (esbuild)`);\n console.log(`Entries: ${entryCount}`);\n console.log('');\n\n if (opts.watch) {\n // Watch mode\n console.log('Watching for changes...\\n');\n\n const watcher = await watch(validated, {\n onRebuild: (result) => {\n if (result.errors.length > 0) {\n for (const err of result.errors) {\n console.error(` ${err}`);\n }\n } else {\n console.log(` Rebuilt ${result.entries.length} entries in ${result.totalDurationMs}ms`);\n }\n },\n });\n\n // Support Ctrl+C and SIGTERM\n const onSignal = () => {\n console.log('\\nStopping watch mode...');\n watcher.dispose().then(() => process.exit(0));\n };\n process.once('SIGINT', onSignal);\n process.once('SIGTERM', onSignal);\n\n // Keep process alive\n await new Promise(() => {});\n }\n\n // Single build\n const result = await build(validated);\n\n // Print results table\n for (const entry of result.entries) {\n const size = formatSize(entry.sizeBytes);\n const duration = `${entry.durationMs}ms`;\n console.log(` ${entry.name.padEnd(30)} ${size.padStart(10)} ${duration.padStart(8)}`);\n }\n\n console.log('');\n\n if (result.errors.length > 0) {\n console.log('Errors:');\n for (const err of result.errors) {\n console.error(` ${err}`);\n }\n console.log('');\n process.exitCode = 1;\n return;\n }\n\n console.log(`${result.entries.length} entries built in ${result.totalDurationMs}ms\\n`);\n}\n\n/** Format bytes to human-readable string */\nfunction formatSize(bytes: number): string {\n if (bytes < 1024) return `${bytes} B`;\n const kb = bytes / 1024;\n return `${kb.toFixed(1)} kB`;\n}\n","/**\n * @xrmforge/cli - Init Command\n *\n * Scaffolds a new D365 form scripting project.\n *\n * Usage:\n * xrmforge init # Scaffold in current directory\n * xrmforge init my-project # Scaffold in ./my-project\n * xrmforge init --prefix markant # Use \"markant\" as publisher prefix\n * xrmforge init --namespace Markant # Use \"Markant\" as base namespace\n * xrmforge init --skip-install # Don't run npm install\n */\n\nimport type { Command } from 'commander';\nimport { resolve, basename } from 'node:path';\nimport { scaffoldProject } from '@xrmforge/devkit';\n\n/** CLI options for the init command */\ninterface InitOptions {\n name?: string;\n prefix: string;\n namespace?: string;\n skipInstall: boolean;\n force: boolean;\n}\n\n/**\n * Register the 'init' subcommand on the CLI program.\n */\nexport function registerInitCommand(program: Command): void {\n program\n .command('init [dir]')\n .description('Scaffold a new D365 form scripting project')\n\n .option('--name <name>', 'Project name for package.json (default: directory name)')\n .option('--prefix <prefix>', 'Publisher prefix for D365 WebResources', 'contoso')\n .option('--namespace <ns>', 'Base namespace for form scripts (default: PascalCase of prefix)')\n .option('--skip-install', 'Skip running npm install after scaffolding', false)\n .option('--force', 'Allow scaffolding in non-empty directories (skip existing files)', false)\n\n .action(async (dir: string | undefined, opts: InitOptions) => {\n try {\n await runInit(dir, opts);\n } catch (error) {\n if (error instanceof Error) {\n console.error(`\\nError: ${error.message}\\n`);\n } else {\n console.error('\\nAn unexpected error occurred.\\n');\n }\n process.exitCode = 1;\n }\n });\n}\n\n/**\n * Execute the init command.\n */\nasync function runInit(dir: string | undefined, opts: InitOptions): Promise<void> {\n const targetDir = resolve(dir ?? '.');\n const dirName = basename(targetDir);\n\n const projectName = opts.name ?? dirName;\n const prefix = opts.prefix;\n const namespace = opts.namespace ?? toPascalCase(prefix);\n\n console.log(`\\nXrmForge Project Scaffolding`);\n console.log(`Directory: ${targetDir}`);\n console.log(`Name: ${projectName}`);\n console.log(`Prefix: ${prefix}`);\n console.log(`Namespace: ${namespace}`);\n console.log('');\n\n const result = await scaffoldProject({\n targetDir,\n projectName,\n prefix,\n namespace,\n force: opts.force,\n });\n\n console.log(`Created ${result.filesCreated.length} files:`);\n for (const file of result.filesCreated) {\n console.log(` ${file}`);\n }\n\n if (result.warnings.length > 0) {\n console.log('\\nWarnings:');\n for (const w of result.warnings) {\n console.log(` ${w}`);\n }\n }\n\n console.log('\\nNext steps:');\n if (dir) {\n console.log(` cd ${dir}`);\n }\n if (!opts.skipInstall) {\n console.log(' npm install');\n }\n console.log(' xrmforge generate --url https://YOUR-ORG.crm4.dynamics.com --auth interactive --entities account --output ./typings');\n console.log(' xrmforge build');\n console.log('');\n}\n\n/** Convert a string to PascalCase (e.g. \"my-prefix\" -> \"MyPrefix\") */\nfunction toPascalCase(str: string): string {\n return str\n .split(/[-_\\s]+/)\n .map((part) => part.charAt(0).toUpperCase() + part.slice(1).toLowerCase())\n .join('');\n}\n"],"mappings":"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","names":["readFileSync","join","program","program","result","program","readFileSync","join"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@xrmforge/cli",
3
- "version": "0.4.1",
3
+ "version": "0.4.2",
4
4
  "description": "CLI for XrmForge - TypeScript type generator for Dynamics 365",
5
5
  "keywords": [
6
6
  "dynamics-365",
@@ -30,8 +30,8 @@
30
30
  ],
31
31
  "dependencies": {
32
32
  "commander": "^13.0.0",
33
- "@xrmforge/devkit": "0.2.2",
34
- "@xrmforge/typegen": "0.5.1"
33
+ "@xrmforge/devkit": "0.3.0",
34
+ "@xrmforge/typegen": "0.6.0"
35
35
  },
36
36
  "devDependencies": {
37
37
  "@types/node": "^22.0.0",