alepha 0.20.4 → 0.20.6
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/api/audits/index.d.ts +391 -359
- package/dist/api/audits/index.d.ts.map +1 -1
- package/dist/api/audits/index.js +23 -1
- package/dist/api/audits/index.js.map +1 -1
- package/dist/api/files/index.d.ts +18 -0
- package/dist/api/files/index.d.ts.map +1 -1
- package/dist/api/files/index.js +51 -0
- package/dist/api/files/index.js.map +1 -1
- package/dist/api/jobs/index.browser.js +33 -14
- package/dist/api/jobs/index.browser.js.map +1 -1
- package/dist/api/jobs/index.d.ts +452 -155
- package/dist/api/jobs/index.d.ts.map +1 -1
- package/dist/api/jobs/index.js +474 -159
- package/dist/api/jobs/index.js.map +1 -1
- package/dist/api/keys/index.d.ts +32 -4
- package/dist/api/keys/index.d.ts.map +1 -1
- package/dist/api/keys/index.js +53 -0
- package/dist/api/keys/index.js.map +1 -1
- package/dist/api/notifications/index.d.ts +29 -1
- package/dist/api/notifications/index.d.ts.map +1 -1
- package/dist/api/notifications/index.js +55 -13
- package/dist/api/notifications/index.js.map +1 -1
- package/dist/api/organizations/index.js.map +1 -1
- package/dist/api/parameters/index.d.ts +15 -0
- package/dist/api/parameters/index.d.ts.map +1 -1
- package/dist/api/parameters/index.js +37 -0
- package/dist/api/parameters/index.js.map +1 -1
- package/dist/api/payments/index.js.map +1 -1
- package/dist/api/users/index.d.ts +150 -9
- package/dist/api/users/index.d.ts.map +1 -1
- package/dist/api/users/index.js +237 -28
- package/dist/api/users/index.js.map +1 -1
- package/dist/api/verifications/index.d.ts +3 -3
- package/dist/api/verifications/index.js.map +1 -1
- package/dist/batch/index.js.map +1 -1
- package/dist/bin/index.js +0 -0
- package/dist/bucket/index.d.ts +18 -0
- package/dist/bucket/index.d.ts.map +1 -1
- package/dist/bucket/index.js +47 -0
- package/dist/bucket/index.js.map +1 -1
- package/dist/bucket/index.workerd.js +24 -0
- package/dist/bucket/index.workerd.js.map +1 -1
- package/dist/cache/core/index.d.ts +20 -3
- package/dist/cache/core/index.d.ts.map +1 -1
- package/dist/cache/core/index.js.map +1 -1
- package/dist/cache/core/index.workerd.js.map +1 -1
- package/dist/cache/database/index.d.ts +155 -0
- package/dist/cache/database/index.d.ts.map +1 -0
- package/dist/cache/database/index.js +266 -0
- package/dist/cache/database/index.js.map +1 -0
- package/dist/cache/redis/index.js.map +1 -1
- package/dist/captcha/index.js.map +1 -1
- package/dist/cli/config/index.js.map +1 -1
- package/dist/cli/core/index.d.ts +35 -5
- package/dist/cli/core/index.d.ts.map +1 -1
- package/dist/cli/core/index.js +85 -6
- package/dist/cli/core/index.js.map +1 -1
- package/dist/cli/devtools/index.js.map +1 -1
- package/dist/cli/platform/index.js +1 -1
- package/dist/cli/platform/index.js.map +1 -1
- package/dist/cli/vendor/index.js.map +1 -1
- package/dist/command/index.js.map +1 -1
- package/dist/core/index.browser.js.map +1 -1
- package/dist/core/index.js.map +1 -1
- package/dist/core/index.native.js.map +1 -1
- package/dist/core/index.workerd.js.map +1 -1
- package/dist/crypto/index.browser.js.map +1 -1
- package/dist/crypto/index.js.map +1 -1
- package/dist/datetime/index.js.map +1 -1
- package/dist/email/brevo/index.js.map +1 -1
- package/dist/email/core/index.js.map +1 -1
- package/dist/email/core/index.workerd.js.map +1 -1
- package/dist/email/smtp/index.js.map +1 -1
- package/dist/fake/index.js.map +1 -1
- package/dist/lock/core/index.js.map +1 -1
- package/dist/lock/redis/index.js.map +1 -1
- package/dist/logger/index.js.map +1 -1
- package/dist/mcp/index.js.map +1 -1
- package/dist/orm/core/index.browser.js.map +1 -1
- package/dist/orm/core/index.bun.js.map +1 -1
- package/dist/orm/core/index.js.map +1 -1
- package/dist/orm/postgres/index.bun.js.map +1 -1
- package/dist/orm/postgres/index.js.map +1 -1
- package/dist/queue/core/index.js.map +1 -1
- package/dist/queue/core/index.workerd.js.map +1 -1
- package/dist/queue/redis/index.js.map +1 -1
- package/dist/react/auth/index.browser.js.map +1 -1
- package/dist/react/auth/index.js.map +1 -1
- package/dist/react/core/index.js.map +1 -1
- package/dist/react/form/index.js +2 -0
- package/dist/react/form/index.js.map +1 -1
- package/dist/react/head/index.browser.js.map +1 -1
- package/dist/react/head/index.js.map +1 -1
- package/dist/react/i18n/index.js.map +1 -1
- package/dist/react/intro/index.js.map +1 -1
- package/dist/react/router/index.browser.js.map +1 -1
- package/dist/react/router/index.js.map +1 -1
- package/dist/react/testing/index.js.map +1 -1
- package/dist/react/ui/index.js.map +1 -1
- package/dist/react/websocket/index.js.map +1 -1
- package/dist/redis/index.bun.js.map +1 -1
- package/dist/redis/index.js.map +1 -1
- package/dist/retry/index.js.map +1 -1
- package/dist/router/index.js.map +1 -1
- package/dist/scheduler/index.d.ts +22 -0
- package/dist/scheduler/index.d.ts.map +1 -1
- package/dist/scheduler/index.js +12 -0
- package/dist/scheduler/index.js.map +1 -1
- package/dist/scheduler/index.workerd.js +12 -0
- package/dist/scheduler/index.workerd.js.map +1 -1
- package/dist/security/index.browser.js.map +1 -1
- package/dist/security/index.js.map +1 -1
- package/dist/server/auth/index.js.map +1 -1
- package/dist/server/cookies/index.browser.js.map +1 -1
- package/dist/server/cookies/index.js.map +1 -1
- package/dist/server/core/index.browser.js.map +1 -1
- package/dist/server/core/index.js.map +1 -1
- package/dist/server/cors/index.js.map +1 -1
- package/dist/server/etag/index.js.map +1 -1
- package/dist/server/health/index.js.map +1 -1
- package/dist/server/links/index.browser.js.map +1 -1
- package/dist/server/links/index.js.map +1 -1
- package/dist/server/metrics/index.js.map +1 -1
- package/dist/server/proxy/index.js.map +1 -1
- package/dist/server/rate-limit/index.js.map +1 -1
- package/dist/server/static/index.js.map +1 -1
- package/dist/server/swagger/index.js.map +1 -1
- package/dist/sms/index.js.map +1 -1
- package/dist/system/index.browser.js.map +1 -1
- package/dist/system/index.js.map +1 -1
- package/dist/system/index.workerd.js.map +1 -1
- package/dist/topic/core/index.js.map +1 -1
- package/dist/topic/redis/index.js.map +1 -1
- package/dist/websocket/index.browser.js +4 -0
- package/dist/websocket/index.browser.js.map +1 -1
- package/dist/websocket/index.js +10 -0
- package/dist/websocket/index.js.map +1 -1
- package/package.json +282 -272
- package/src/api/audits/controllers/AdminAuditController.ts +29 -0
- package/src/api/files/controllers/FileController.ts +24 -0
- package/src/api/files/services/FileService.ts +41 -0
- package/src/api/jobs/__tests__/$job.spec.ts +427 -2
- package/src/api/jobs/entities/jobExecutionEntity.ts +3 -3
- package/src/api/jobs/index.ts +47 -10
- package/src/api/jobs/primitives/$job.ts +22 -9
- package/src/api/jobs/providers/DirectJobDispatcher.ts +71 -0
- package/src/api/jobs/providers/JobDispatcher.ts +49 -0
- package/src/api/jobs/providers/JobProvider.ts +365 -142
- package/src/api/jobs/providers/JobQueueProvider.ts +43 -18
- package/src/api/jobs/schemas/jobConfigAtom.ts +4 -3
- package/src/api/jobs/schemas/jobExecutionResourceSchema.ts +11 -0
- package/src/api/jobs/schemas/jobRegistrationSchema.ts +4 -2
- package/src/api/jobs/services/JobService.ts +21 -11
- package/src/api/keys/controllers/AdminApiKeyController.ts +23 -0
- package/src/api/keys/services/ApiKeyService.ts +42 -0
- package/src/api/notifications/__tests__/AlephaApiNotifications.spec.ts +63 -0
- package/src/api/notifications/controllers/AdminNotificationController.ts +48 -1
- package/src/api/notifications/index.ts +13 -3
- package/src/api/notifications/jobs/NotificationJobs.ts +0 -6
- package/src/api/parameters/controllers/AdminParameterController.ts +26 -0
- package/src/api/parameters/services/ParameterProvider.ts +18 -0
- package/src/api/users/__tests__/Registration-emailMode.spec.ts +203 -0
- package/src/api/users/__tests__/UsernameSlugger.spec.ts +138 -0
- package/src/api/users/atoms/realmAuthSettingsAtom.ts +41 -3
- package/src/api/users/controllers/AdminSessionController.ts +29 -0
- package/src/api/users/controllers/AdminUserController.ts +32 -0
- package/src/api/users/index.ts +3 -0
- package/src/api/users/services/CredentialService.ts +5 -0
- package/src/api/users/services/RegistrationService.ts +49 -1
- package/src/api/users/services/SessionCrudService.ts +16 -0
- package/src/api/users/services/SessionService.ts +17 -59
- package/src/api/users/services/UsernameSlugger.ts +195 -0
- package/src/bucket/primitives/$bucket.ts +21 -0
- package/src/bucket/providers/CloudflareR2Provider.ts +15 -0
- package/src/bucket/providers/FileStorageProvider.ts +9 -0
- package/src/bucket/providers/LocalFileStorageProvider.ts +14 -0
- package/src/bucket/providers/MemoryFileStorageProvider.ts +9 -0
- package/src/bucket/providers/NodeS3BucketProvider.ts +35 -0
- package/src/cache/core/primitives/$cache.ts +20 -3
- package/src/cache/database/__tests__/DatabaseCacheProvider.behavior.spec.ts +203 -0
- package/src/cache/database/__tests__/DatabaseCacheProvider.spec.ts +110 -0
- package/src/cache/database/entities/cacheEntries.ts +55 -0
- package/src/cache/database/index.ts +36 -0
- package/src/cache/database/providers/DatabaseCacheProvider.ts +348 -0
- package/src/cli/core/services/ProjectScaffolder.ts +0 -2
- package/src/cli/core/tasks/BuildCloudflareTask.ts +17 -3
- package/src/cli/core/tasks/BuildSitemapTask.ts +7 -0
- package/src/cli/core/tasks/BuildVercelTask.ts +82 -3
- package/src/cli/platform/__tests__/detectResources.spec.ts +96 -0
- package/src/cli/platform/commands/platform.ts +7 -1
- package/src/scheduler/index.ts +14 -0
- package/src/scheduler/providers/CronProvider.ts +13 -0
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.js","names":[],"sources":["../../../src/cli/vendor/atoms/vendorOptions.ts","../../../src/cli/vendor/services/VendorService.ts","../../../src/cli/vendor/commands/VendorCommand.ts","../../../src/cli/vendor/index.ts"],"sourcesContent":["import { $atom, type Static, t } from \"alepha\";\n\n/**\n * Vendor configuration atom.\n *\n * Filled from the `vendor` section of `alepha.config.ts`.\n * Read by `VendorCommand` to resolve remote, branch, and packages.\n */\nexport const vendorOptions = $atom({\n name: \"alepha.cli.vendor.options\",\n description: \"Vendor synchronization configuration\",\n schema: t.optional(\n t.object({\n /**\n * Git remote URL.\n *\n * @default \"git@github.com:feunard/alepha.git\"\n */\n remote: t.optional(t.text()),\n\n /**\n * Branch to sync from.\n *\n * @default \"main\"\n */\n branch: t.optional(t.text()),\n\n /**\n * Package directory names under `packages/` to sync.\n *\n * @example [\"alepha\", \"@alepha/payments-stripe\"]\n */\n packages: t.array(t.text()),\n }),\n ),\n});\n\n/**\n * Type for vendor options.\n */\nexport type VendorOptions = Static<typeof vendorOptions.schema>;\n","import { $inject } from \"alepha\";\nimport { $logger } from \"alepha/logger\";\nimport { FileSystemProvider, ShellProvider } from \"alepha/system\";\n\n/**\n * Options for syncing vendored packages from a remote repository.\n */\nexport interface VendorSyncOptions {\n root: string;\n remote: string;\n branch: string;\n packages: string[];\n force?: boolean;\n}\n\n/**\n * Result of a vendor sync operation.\n */\nexport interface VendorSyncResult {\n synced: string[];\n errors: string[];\n aborted?: VendorDiffResult;\n}\n\n/**\n * Options for diffing vendored packages against a remote repository.\n */\nexport interface VendorDiffOptions {\n root: string;\n remote: string;\n branch: string;\n packages: string[];\n}\n\n/**\n * A single line change within a modified file.\n */\nexport interface VendorLineDiff {\n line: number;\n type: \"added\" | \"removed\";\n text: string;\n}\n\n/**\n * A modified file with its line-level changes.\n */\nexport interface VendorFileDiff {\n file: string;\n changes: VendorLineDiff[];\n}\n\n/**\n * Diff result for a single vendored package.\n */\nexport interface VendorPackageDiff {\n name: string;\n added: string[];\n modified: VendorFileDiff[];\n removed: string[];\n}\n\n/**\n * Result of a vendor diff operation.\n */\nexport interface VendorDiffResult {\n packages: VendorPackageDiff[];\n totalChanges: number;\n}\n\n/**\n * Shape of the <root>/.alepha/vendor.json lock file.\n */\nexport interface VendorLock {\n commit: string;\n}\n\n/**\n * Handles syncing and diffing vendored packages from a remote git repository.\n */\nexport class VendorService {\n protected readonly log = $logger();\n protected readonly shell = $inject(ShellProvider);\n protected readonly fs = $inject(FileSystemProvider);\n\n /**\n * Sync vendored packages from a remote repository.\n *\n * Without `force`: checks for local modifications by comparing the local\n * copy against the last-synced commit (stored in .alepha/vendor.json).\n * If modifications are found, aborts without touching local files.\n *\n * With `force` (or first sync): replaces local copies unconditionally.\n */\n async sync(options: VendorSyncOptions): Promise<VendorSyncResult> {\n const synced: string[] = [];\n const errors: string[] = [];\n\n if (!options.force) {\n const lock = await this.readLock(options.root);\n\n if (lock) {\n let baselineDir: string | undefined;\n try {\n baselineDir = await this.cloneAtCommit(options.remote, lock.commit);\n const diffResult = await this.diffFromClone(\n options.root,\n baselineDir,\n options.packages,\n );\n\n if (diffResult.totalChanges > 0) {\n return { synced: [], errors: [], aborted: diffResult };\n }\n } finally {\n if (baselineDir) {\n await this.fs.rm(baselineDir, { recursive: true, force: true });\n }\n }\n }\n }\n\n let tmpDir: string | undefined;\n\n try {\n tmpDir = await this.cloneRemote(options.remote, options.branch);\n\n for (const pkg of options.packages) {\n const remotePkgDir = this.fs.join(tmpDir, \"packages\", pkg);\n const localPkgDir = this.fs.join(options.root, \"packages\", pkg);\n\n const remoteExists = await this.fs.exists(remotePkgDir);\n if (!remoteExists) {\n errors.push(`Package \"${pkg}\" not found in remote`);\n continue;\n }\n\n this.log.debug(`Syncing package: ${pkg}`);\n\n await this.fs.rm(localPkgDir, { recursive: true, force: true });\n await this.fs.cp(remotePkgDir, localPkgDir, { recursive: true });\n await this.removeIgnoredFiles(localPkgDir);\n\n synced.push(pkg);\n }\n\n const commit = await this.getCommitHash(tmpDir);\n await this.writeLock(options.root, { commit });\n } finally {\n if (tmpDir) {\n await this.fs.rm(tmpDir, { recursive: true, force: true });\n }\n }\n\n return { synced, errors };\n }\n\n /**\n * Diff vendored packages against the last-synced commit.\n *\n * Reads the commit hash from .alepha/vendor.json, clones at that commit,\n * and compares local files to detect modifications since last sync.\n */\n async diff(options: VendorDiffOptions): Promise<VendorDiffResult> {\n const lock = await this.readLock(options.root);\n if (!lock) {\n return { packages: [], totalChanges: 0 };\n }\n\n let tmpDir: string | undefined;\n\n try {\n tmpDir = await this.cloneAtCommit(options.remote, lock.commit);\n return await this.diffFromClone(options.root, tmpDir, options.packages);\n } finally {\n if (tmpDir) {\n await this.fs.rm(tmpDir, { recursive: true, force: true });\n }\n }\n }\n\n /**\n * Diff local packages against an already-cloned remote.\n */\n protected async diffFromClone(\n root: string,\n tmpDir: string,\n packages: string[],\n ): Promise<VendorDiffResult> {\n const results: VendorPackageDiff[] = [];\n let totalChanges = 0;\n\n for (const pkg of packages) {\n const remotePkgDir = this.fs.join(tmpDir, \"packages\", pkg);\n const localPkgDir = this.fs.join(root, \"packages\", pkg);\n\n const remoteExists = await this.fs.exists(remotePkgDir);\n const localExists = await this.fs.exists(localPkgDir);\n\n if (!remoteExists && !localExists) {\n results.push({ name: pkg, added: [], modified: [], removed: [] });\n continue;\n }\n\n if (!remoteExists) {\n // No baseline = everything local was added by user\n const localFiles = await this.fs.ls(localPkgDir, { recursive: true });\n results.push({\n name: pkg,\n added: localFiles,\n modified: [],\n removed: [],\n });\n totalChanges += localFiles.length;\n continue;\n }\n\n if (!localExists) {\n // Baseline exists but local doesn't = user deleted everything\n const remoteFiles = await this.fs.ls(remotePkgDir, { recursive: true });\n results.push({\n name: pkg,\n added: [],\n modified: [],\n removed: remoteFiles,\n });\n totalChanges += remoteFiles.length;\n continue;\n }\n\n const result = await this.diffDirectories(localPkgDir, remotePkgDir);\n const pkgChanges =\n result.added.length + result.modified.length + result.removed.length;\n totalChanges += pkgChanges;\n\n results.push({\n name: pkg,\n added: result.added,\n modified: result.modified,\n removed: result.removed,\n });\n }\n\n return { packages: results, totalChanges };\n }\n\n /**\n * Remove test files and ignored directories from a synced package.\n */\n protected async removeIgnoredFiles(pkgDir: string): Promise<void> {\n const allFiles = await this.fs.ls(pkgDir, { recursive: true });\n\n // Remove ignored files\n for (const file of allFiles) {\n if (\n file.endsWith(\".spec.ts\") ||\n file.endsWith(\".spec.tsx\") ||\n file === \"LICENSE\" ||\n file === \"tsdown.config.ts\"\n ) {\n await this.fs.rm(this.fs.join(pkgDir, file), { force: true });\n }\n }\n\n // Remove ignored directories (find all occurrences at any depth)\n for (const file of allFiles) {\n for (const ignored of this.ignoredPaths) {\n if (\n file === ignored ||\n file.startsWith(`${ignored}/`) ||\n file.includes(`/${ignored}/`) ||\n file.endsWith(`/${ignored}`)\n ) {\n // Extract the path to the ignored directory itself\n const idx = file.indexOf(ignored);\n const dirPath = this.fs.join(\n pkgDir,\n file.substring(0, idx + ignored.length),\n );\n await this.fs.rm(dirPath, { recursive: true, force: true });\n }\n }\n }\n }\n\n /**\n * Clone a remote repository into a temporary directory.\n */\n protected async cloneRemote(remote: string, branch: string): Promise<string> {\n const tmpDir = this.fs.join(\n process.env.TMPDIR || \"/tmp\",\n `.alepha-vendor-${Date.now()}`,\n );\n\n this.log.debug(`Cloning ${remote}#${branch} into ${tmpDir}`);\n\n const output = await this.shell.run(\n `git clone --depth 1 --branch ${branch} --filter=blob:none ${remote} ${tmpDir}`,\n { capture: true },\n );\n\n if (output) {\n this.log.debug(output);\n }\n\n return tmpDir;\n }\n\n /**\n * Clone a remote repository at a specific commit hash.\n */\n protected async cloneAtCommit(\n remote: string,\n commit: string,\n ): Promise<string> {\n const tmpDir = this.fs.join(\n process.env.TMPDIR || \"/tmp\",\n `.alepha-vendor-${Date.now()}`,\n );\n\n this.log.debug(`Cloning ${remote}@${commit} into ${tmpDir}`);\n\n await this.shell.run(`git init ${tmpDir}`, { capture: true });\n await this.shell.run(`git -C ${tmpDir} remote add origin ${remote}`, {\n capture: true,\n });\n await this.shell.run(`git -C ${tmpDir} fetch --depth 1 origin ${commit}`, {\n capture: true,\n });\n await this.shell.run(`git -C ${tmpDir} checkout FETCH_HEAD`, {\n capture: true,\n });\n\n return tmpDir;\n }\n\n /**\n * Get the HEAD commit hash from a cloned repository.\n */\n protected async getCommitHash(repoDir: string): Promise<string> {\n const hash = await this.shell.run(`git -C ${repoDir} rev-parse HEAD`, {\n capture: true,\n });\n return hash.trim();\n }\n\n /**\n * Read the vendor lock file.\n */\n protected async readLock(root: string): Promise<VendorLock | undefined> {\n const lockPath = this.fs.join(root, \".alepha\", \"vendor.json\");\n const exists = await this.fs.exists(lockPath);\n if (!exists) {\n return undefined;\n }\n const content = await this.fs.readFile(lockPath);\n return JSON.parse(content.toString());\n }\n\n /**\n * Write the vendor lock file.\n */\n protected async writeLock(root: string, lock: VendorLock): Promise<void> {\n const dir = this.fs.join(root, \".alepha\");\n await this.fs.mkdir(dir, { recursive: true });\n await this.fs.writeFile(\n this.fs.join(dir, \"vendor.json\"),\n JSON.stringify(lock, null, 2),\n );\n }\n\n /**\n * Directories to ignore during diff comparisons.\n */\n protected readonly ignoredPaths = [\n \"__tests__\",\n \"assets/swagger-ui\",\n \"node_modules\",\n \"dist\",\n ];\n\n /**\n * Check if a file path should be ignored during diff.\n */\n protected isIgnored(filePath: string): boolean {\n if (\n filePath.endsWith(\".spec.ts\") ||\n filePath.endsWith(\".spec.tsx\") ||\n filePath === \"LICENSE\" ||\n filePath === \"tsdown.config.ts\"\n ) {\n return true;\n }\n return this.ignoredPaths.some(\n (p) =>\n filePath === p ||\n filePath.startsWith(`${p}/`) ||\n filePath.includes(`/${p}/`) ||\n filePath.endsWith(`/${p}`),\n );\n }\n\n /**\n * Recursively compare two directories and return the differences.\n */\n protected async diffDirectories(\n localDir: string,\n remoteDir: string,\n ): Promise<{\n added: string[];\n modified: VendorFileDiff[];\n removed: string[];\n }> {\n const added: string[] = [];\n const modified: VendorFileDiff[] = [];\n const removed: string[] = [];\n\n const [localFiles, remoteFiles] = await Promise.all([\n this.fs.ls(localDir, { recursive: true }),\n this.fs.ls(remoteDir, { recursive: true }),\n ]);\n\n const filteredLocal = localFiles.filter((f) => !this.isIgnored(f));\n const filteredRemote = remoteFiles.filter((f) => !this.isIgnored(f));\n\n const localSet = new Set(filteredLocal);\n const remoteSet = new Set(filteredRemote);\n\n // Files in baseline but not local = user deleted them\n for (const file of filteredRemote) {\n if (!localSet.has(file)) {\n removed.push(file);\n continue;\n }\n\n try {\n const [localContent, remoteContent] = await Promise.all([\n this.fs.readFile(this.fs.join(localDir, file)),\n this.fs.readFile(this.fs.join(remoteDir, file)),\n ]);\n\n if (!localContent.equals(remoteContent)) {\n const changes = this.computeLineDiff(\n remoteContent.toString(),\n localContent.toString(),\n );\n modified.push({ file, changes });\n }\n } catch {\n // Skip directories and unreadable entries\n }\n }\n\n // Files in local but not baseline = user added them\n for (const file of filteredLocal) {\n if (!remoteSet.has(file)) {\n added.push(file);\n }\n }\n\n return { added, modified, removed };\n }\n\n /**\n * Compute line-level differences between two file contents.\n *\n * Uses a longest-common-subsequence algorithm to produce minimal\n * added/removed line changes with accurate line numbers.\n */\n protected computeLineDiff(baseline: string, local: string): VendorLineDiff[] {\n const baseLines = baseline.split(\"\\n\");\n const localLines = local.split(\"\\n\");\n const lcs = this.longestCommonSubsequence(baseLines, localLines);\n const changes: VendorLineDiff[] = [];\n\n let bi = 0;\n let li = 0;\n let ci = 0;\n\n while (bi < baseLines.length || li < localLines.length) {\n if (ci < lcs.length && bi < baseLines.length && li < localLines.length) {\n if (baseLines[bi] === lcs[ci] && localLines[li] === lcs[ci]) {\n // Line is unchanged\n bi++;\n li++;\n ci++;\n } else if (baseLines[bi] !== lcs[ci]) {\n changes.push({ line: bi + 1, type: \"removed\", text: baseLines[bi] });\n bi++;\n } else {\n changes.push({ line: li + 1, type: \"added\", text: localLines[li] });\n li++;\n }\n } else if (bi < baseLines.length) {\n changes.push({ line: bi + 1, type: \"removed\", text: baseLines[bi] });\n bi++;\n } else {\n changes.push({ line: li + 1, type: \"added\", text: localLines[li] });\n li++;\n }\n }\n\n return changes;\n }\n\n /**\n * Compute the longest common subsequence of two string arrays.\n */\n protected longestCommonSubsequence(a: string[], b: string[]): string[] {\n const m = a.length;\n const n = b.length;\n const dp: number[][] = Array.from({ length: m + 1 }, () =>\n Array(n + 1).fill(0),\n );\n\n for (let i = 1; i <= m; i++) {\n for (let j = 1; j <= n; j++) {\n dp[i][j] =\n a[i - 1] === b[j - 1]\n ? dp[i - 1][j - 1] + 1\n : Math.max(dp[i - 1][j], dp[i][j - 1]);\n }\n }\n\n const result: string[] = [];\n let i = m;\n let j = n;\n while (i > 0 && j > 0) {\n if (a[i - 1] === b[j - 1]) {\n result.unshift(a[i - 1]);\n i--;\n j--;\n } else if (dp[i - 1][j] > dp[i][j - 1]) {\n i--;\n } else {\n j--;\n }\n }\n\n return result;\n }\n}\n","import { $inject, $state, AlephaError, t } from \"alepha\";\nimport { PackageManagerUtils } from \"alepha/cli\";\nimport { $command } from \"alepha/command\";\nimport { $logger, ConsoleColorProvider } from \"alepha/logger\";\nimport { vendorOptions } from \"../atoms/vendorOptions.ts\";\nimport type {\n VendorDiffResult,\n VendorPackageDiff,\n VendorSyncResult,\n} from \"../services/VendorService.ts\";\nimport { VendorService } from \"../services/VendorService.ts\";\n\n/**\n * Default remote when none is configured.\n */\nconst DEFAULT_REMOTE = \"git@github.com:feunard/alepha.git\";\n\nexport class VendorCommand {\n protected readonly log = $logger();\n protected readonly options = $state(vendorOptions);\n protected readonly vendorService = $inject(VendorService);\n protected readonly color = $inject(ConsoleColorProvider);\n protected readonly pm = $inject(PackageManagerUtils);\n\n /**\n * Ensure vendor config is present and return resolved options.\n */\n protected resolveOptions() {\n if (!this.options) {\n throw new AlephaError(\n 'Missing vendor configuration. Add a \"vendor\" section to alepha.config.ts.',\n );\n }\n return {\n remote: this.options.remote ?? DEFAULT_REMOTE,\n branch: this.options.branch ?? \"main\",\n packages: this.options.packages,\n };\n }\n\n // ─────────────────────────────────────────────────────────────────────────\n // alepha vendor sync\n // ─────────────────────────────────────────────────────────────────────────\n\n protected readonly syncFlags = t.object({\n force: t.optional(\n t.boolean({\n aliases: [\"f\"],\n description: \"Skip local modification check\",\n }),\n ),\n });\n\n protected readonly sync = $command({\n name: \"sync\",\n description: \"Replace local packages with remote source\",\n flags: this.syncFlags,\n handler: async ({ flags, root, run }) => {\n const opts = this.resolveOptions();\n const c = this.color;\n\n let result: VendorSyncResult = { synced: [], errors: [] };\n\n await run({\n name: `Syncing from ${opts.branch}`,\n handler: async () => {\n result = await this.vendorService.sync({\n root,\n remote: opts.remote,\n branch: opts.branch,\n packages: opts.packages,\n force: flags.force,\n });\n },\n });\n\n if (result.aborted) {\n run.end();\n\n process.stdout.write(\n `\\nLocal modifications detected. Use ${c.set(\"CYAN\", \"--force\")} to overwrite.\\n`,\n );\n\n for (const pkg of result.aborted.packages) {\n this.printPackageDiff(pkg);\n }\n\n process.stdout.write(\"\\n\");\n return;\n }\n\n if (result.synced.length > 0) {\n const pmName = await this.pm.getPackageManager(root);\n await run(`${pmName} install`, { root });\n }\n\n run.end();\n\n if (result.errors.length > 0) {\n for (const error of result.errors) {\n process.stdout.write(`${c.set(\"RED\", \" error\")} ${error}\\n`);\n }\n }\n\n if (result.synced.length > 0) {\n process.stdout.write(\n `\\nSynced ${c.set(\"CYAN\", String(result.synced.length))} ${result.synced.length === 1 ? \"package\" : \"packages\"} from ${c.set(\"CYAN\", opts.branch)}\\n`,\n );\n for (const pkg of result.synced) {\n process.stdout.write(` ${c.set(\"GREEN\", \"\\u2713\")} ${pkg}\\n`);\n }\n }\n\n process.stdout.write(\"\\n\");\n },\n });\n\n // ─────────────────────────────────────────────────────────────────────────\n // alepha vendor diff\n // ─────────────────────────────────────────────────────────────────────────\n\n protected readonly diff = $command({\n name: \"diff\",\n description: \"Compare local packages against remote\",\n handler: async ({ root, run }) => {\n const opts = this.resolveOptions();\n\n let result: VendorDiffResult = { packages: [], totalChanges: 0 };\n\n await run({\n name: `Cloning ${opts.remote} at ${opts.branch}`,\n handler: async () => {\n result = await this.vendorService.diff({\n root,\n remote: opts.remote,\n branch: opts.branch,\n packages: opts.packages,\n });\n },\n });\n\n run.end();\n\n if (result.totalChanges === 0) {\n process.stdout.write(\"\\nNo changes\\n\\n\");\n return;\n }\n\n for (const pkg of result.packages) {\n this.printPackageDiff(pkg);\n }\n\n process.stdout.write(\"\\n\");\n },\n });\n\n // ─────────────────────────────────────────────────────────────────────────\n // Helpers\n // ─────────────────────────────────────────────────────────────────────────\n\n protected printPackageDiff(pkg: VendorPackageDiff) {\n const c = this.color;\n const count = pkg.added.length + pkg.modified.length + pkg.removed.length;\n\n if (count === 0) {\n process.stdout.write(`\\n${c.set(\"CYAN\", pkg.name)}: no changes\\n`);\n return;\n }\n\n process.stdout.write(\n `\\n${c.set(\"CYAN\", pkg.name)}: ${count} ${count === 1 ? \"file differs\" : \"files differ\"}\\n`,\n );\n\n for (const file of pkg.added) {\n process.stdout.write(` ${c.set(\"GREEN\", \"A\")} ${file}\\n`);\n }\n\n for (const fileDiff of pkg.modified) {\n process.stdout.write(` ${c.set(\"ORANGE\", \"M\")} ${fileDiff.file}\\n`);\n for (const change of fileDiff.changes) {\n const prefix = change.type === \"removed\" ? \"-\" : \"+\";\n const color = change.type === \"removed\" ? \"RED\" : \"GREEN\";\n const lineNum = `L${change.line}`;\n process.stdout.write(\n ` ${c.set(\"DIM\", lineNum.padEnd(5))} ${c.set(color, `${prefix} ${change.text}`)}\\n`,\n );\n }\n }\n\n for (const file of pkg.removed) {\n process.stdout.write(` ${c.set(\"RED\", \"D\")} ${file}\\n`);\n }\n }\n\n // ─────────────────────────────────────────────────────────────────────────\n // Parent command\n // ─────────────────────────────────────────────────────────────────────────\n\n public readonly vendor = $command({\n name: \"vendor\",\n description: \"Vendor Alepha packages into the project\",\n children: [this.sync, this.diff],\n handler: async ({ help }) => {\n help();\n },\n });\n}\n","import { $context, $module } from \"alepha\";\nimport { type VendorOptions, vendorOptions } from \"./atoms/vendorOptions.ts\";\nimport { VendorCommand } from \"./commands/VendorCommand.ts\";\nimport { VendorService } from \"./services/VendorService.ts\";\n\n// ---------------------------------------------------------------------------\n\n/**\n * CLI plugin for vendoring Alepha packages into external projects.\n *\n * Copies package source code from a git remote into the current project's\n * `packages/` directory. Useful for corporate projects that need a local\n * copy of Alepha for AI tooling, audits, documentation, or quick fixes.\n *\n * Commands:\n * - `alepha vendor sync` — replace local packages with remote source\n * - `alepha vendor diff` — compare local packages against remote HEAD\n *\n * Configuration in `alepha.config.ts`:\n *\n * ```typescript\n * import { vendor } from \"alepha/cli/vendor\";\n *\n * export default defineConfig({\n * plugins: [\n * vendor({\n * branch: \"main\",\n * packages: [\"alepha\", \"@alepha/payments-stripe\"],\n * }),\n * ],\n * });\n * ```\n */\nexport const AlephaCliVendorPlugin = $module({\n name: \"alepha.cli.plugins.vendor\",\n atoms: [vendorOptions],\n services: [VendorCommand, VendorService],\n});\n\nexport const vendor = (options: VendorOptions) => {\n return () => {\n const { alepha } = $context();\n alepha.with(AlephaCliVendorPlugin).set(vendorOptions, options);\n };\n};\n\n// ---------------------------------------------------------------------------\n\nexport * from \"./atoms/vendorOptions.ts\";\nexport * from \"./commands/VendorCommand.ts\";\nexport * from \"./services/VendorService.ts\";\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"index.js","names":[],"sources":["../../../src/cli/vendor/atoms/vendorOptions.ts","../../../src/cli/vendor/services/VendorService.ts","../../../src/cli/vendor/commands/VendorCommand.ts","../../../src/cli/vendor/index.ts"],"sourcesContent":["import { $atom, type Static, t } from \"alepha\";\n\n/**\n * Vendor configuration atom.\n *\n * Filled from the `vendor` section of `alepha.config.ts`.\n * Read by `VendorCommand` to resolve remote, branch, and packages.\n */\nexport const vendorOptions = $atom({\n name: \"alepha.cli.vendor.options\",\n description: \"Vendor synchronization configuration\",\n schema: t.optional(\n t.object({\n /**\n * Git remote URL.\n *\n * @default \"git@github.com:feunard/alepha.git\"\n */\n remote: t.optional(t.text()),\n\n /**\n * Branch to sync from.\n *\n * @default \"main\"\n */\n branch: t.optional(t.text()),\n\n /**\n * Package directory names under `packages/` to sync.\n *\n * @example [\"alepha\", \"@alepha/payments-stripe\"]\n */\n packages: t.array(t.text()),\n }),\n ),\n});\n\n/**\n * Type for vendor options.\n */\nexport type VendorOptions = Static<typeof vendorOptions.schema>;\n","import { $inject } from \"alepha\";\nimport { $logger } from \"alepha/logger\";\nimport { FileSystemProvider, ShellProvider } from \"alepha/system\";\n\n/**\n * Options for syncing vendored packages from a remote repository.\n */\nexport interface VendorSyncOptions {\n root: string;\n remote: string;\n branch: string;\n packages: string[];\n force?: boolean;\n}\n\n/**\n * Result of a vendor sync operation.\n */\nexport interface VendorSyncResult {\n synced: string[];\n errors: string[];\n aborted?: VendorDiffResult;\n}\n\n/**\n * Options for diffing vendored packages against a remote repository.\n */\nexport interface VendorDiffOptions {\n root: string;\n remote: string;\n branch: string;\n packages: string[];\n}\n\n/**\n * A single line change within a modified file.\n */\nexport interface VendorLineDiff {\n line: number;\n type: \"added\" | \"removed\";\n text: string;\n}\n\n/**\n * A modified file with its line-level changes.\n */\nexport interface VendorFileDiff {\n file: string;\n changes: VendorLineDiff[];\n}\n\n/**\n * Diff result for a single vendored package.\n */\nexport interface VendorPackageDiff {\n name: string;\n added: string[];\n modified: VendorFileDiff[];\n removed: string[];\n}\n\n/**\n * Result of a vendor diff operation.\n */\nexport interface VendorDiffResult {\n packages: VendorPackageDiff[];\n totalChanges: number;\n}\n\n/**\n * Shape of the <root>/.alepha/vendor.json lock file.\n */\nexport interface VendorLock {\n commit: string;\n}\n\n/**\n * Handles syncing and diffing vendored packages from a remote git repository.\n */\nexport class VendorService {\n protected readonly log = $logger();\n protected readonly shell = $inject(ShellProvider);\n protected readonly fs = $inject(FileSystemProvider);\n\n /**\n * Sync vendored packages from a remote repository.\n *\n * Without `force`: checks for local modifications by comparing the local\n * copy against the last-synced commit (stored in .alepha/vendor.json).\n * If modifications are found, aborts without touching local files.\n *\n * With `force` (or first sync): replaces local copies unconditionally.\n */\n async sync(options: VendorSyncOptions): Promise<VendorSyncResult> {\n const synced: string[] = [];\n const errors: string[] = [];\n\n if (!options.force) {\n const lock = await this.readLock(options.root);\n\n if (lock) {\n let baselineDir: string | undefined;\n try {\n baselineDir = await this.cloneAtCommit(options.remote, lock.commit);\n const diffResult = await this.diffFromClone(\n options.root,\n baselineDir,\n options.packages,\n );\n\n if (diffResult.totalChanges > 0) {\n return { synced: [], errors: [], aborted: diffResult };\n }\n } finally {\n if (baselineDir) {\n await this.fs.rm(baselineDir, { recursive: true, force: true });\n }\n }\n }\n }\n\n let tmpDir: string | undefined;\n\n try {\n tmpDir = await this.cloneRemote(options.remote, options.branch);\n\n for (const pkg of options.packages) {\n const remotePkgDir = this.fs.join(tmpDir, \"packages\", pkg);\n const localPkgDir = this.fs.join(options.root, \"packages\", pkg);\n\n const remoteExists = await this.fs.exists(remotePkgDir);\n if (!remoteExists) {\n errors.push(`Package \"${pkg}\" not found in remote`);\n continue;\n }\n\n this.log.debug(`Syncing package: ${pkg}`);\n\n await this.fs.rm(localPkgDir, { recursive: true, force: true });\n await this.fs.cp(remotePkgDir, localPkgDir, { recursive: true });\n await this.removeIgnoredFiles(localPkgDir);\n\n synced.push(pkg);\n }\n\n const commit = await this.getCommitHash(tmpDir);\n await this.writeLock(options.root, { commit });\n } finally {\n if (tmpDir) {\n await this.fs.rm(tmpDir, { recursive: true, force: true });\n }\n }\n\n return { synced, errors };\n }\n\n /**\n * Diff vendored packages against the last-synced commit.\n *\n * Reads the commit hash from .alepha/vendor.json, clones at that commit,\n * and compares local files to detect modifications since last sync.\n */\n async diff(options: VendorDiffOptions): Promise<VendorDiffResult> {\n const lock = await this.readLock(options.root);\n if (!lock) {\n return { packages: [], totalChanges: 0 };\n }\n\n let tmpDir: string | undefined;\n\n try {\n tmpDir = await this.cloneAtCommit(options.remote, lock.commit);\n return await this.diffFromClone(options.root, tmpDir, options.packages);\n } finally {\n if (tmpDir) {\n await this.fs.rm(tmpDir, { recursive: true, force: true });\n }\n }\n }\n\n /**\n * Diff local packages against an already-cloned remote.\n */\n protected async diffFromClone(\n root: string,\n tmpDir: string,\n packages: string[],\n ): Promise<VendorDiffResult> {\n const results: VendorPackageDiff[] = [];\n let totalChanges = 0;\n\n for (const pkg of packages) {\n const remotePkgDir = this.fs.join(tmpDir, \"packages\", pkg);\n const localPkgDir = this.fs.join(root, \"packages\", pkg);\n\n const remoteExists = await this.fs.exists(remotePkgDir);\n const localExists = await this.fs.exists(localPkgDir);\n\n if (!remoteExists && !localExists) {\n results.push({ name: pkg, added: [], modified: [], removed: [] });\n continue;\n }\n\n if (!remoteExists) {\n // No baseline = everything local was added by user\n const localFiles = await this.fs.ls(localPkgDir, { recursive: true });\n results.push({\n name: pkg,\n added: localFiles,\n modified: [],\n removed: [],\n });\n totalChanges += localFiles.length;\n continue;\n }\n\n if (!localExists) {\n // Baseline exists but local doesn't = user deleted everything\n const remoteFiles = await this.fs.ls(remotePkgDir, { recursive: true });\n results.push({\n name: pkg,\n added: [],\n modified: [],\n removed: remoteFiles,\n });\n totalChanges += remoteFiles.length;\n continue;\n }\n\n const result = await this.diffDirectories(localPkgDir, remotePkgDir);\n const pkgChanges =\n result.added.length + result.modified.length + result.removed.length;\n totalChanges += pkgChanges;\n\n results.push({\n name: pkg,\n added: result.added,\n modified: result.modified,\n removed: result.removed,\n });\n }\n\n return { packages: results, totalChanges };\n }\n\n /**\n * Remove test files and ignored directories from a synced package.\n */\n protected async removeIgnoredFiles(pkgDir: string): Promise<void> {\n const allFiles = await this.fs.ls(pkgDir, { recursive: true });\n\n // Remove ignored files\n for (const file of allFiles) {\n if (\n file.endsWith(\".spec.ts\") ||\n file.endsWith(\".spec.tsx\") ||\n file === \"LICENSE\" ||\n file === \"tsdown.config.ts\"\n ) {\n await this.fs.rm(this.fs.join(pkgDir, file), { force: true });\n }\n }\n\n // Remove ignored directories (find all occurrences at any depth)\n for (const file of allFiles) {\n for (const ignored of this.ignoredPaths) {\n if (\n file === ignored ||\n file.startsWith(`${ignored}/`) ||\n file.includes(`/${ignored}/`) ||\n file.endsWith(`/${ignored}`)\n ) {\n // Extract the path to the ignored directory itself\n const idx = file.indexOf(ignored);\n const dirPath = this.fs.join(\n pkgDir,\n file.substring(0, idx + ignored.length),\n );\n await this.fs.rm(dirPath, { recursive: true, force: true });\n }\n }\n }\n }\n\n /**\n * Clone a remote repository into a temporary directory.\n */\n protected async cloneRemote(remote: string, branch: string): Promise<string> {\n const tmpDir = this.fs.join(\n process.env.TMPDIR || \"/tmp\",\n `.alepha-vendor-${Date.now()}`,\n );\n\n this.log.debug(`Cloning ${remote}#${branch} into ${tmpDir}`);\n\n const output = await this.shell.run(\n `git clone --depth 1 --branch ${branch} --filter=blob:none ${remote} ${tmpDir}`,\n { capture: true },\n );\n\n if (output) {\n this.log.debug(output);\n }\n\n return tmpDir;\n }\n\n /**\n * Clone a remote repository at a specific commit hash.\n */\n protected async cloneAtCommit(\n remote: string,\n commit: string,\n ): Promise<string> {\n const tmpDir = this.fs.join(\n process.env.TMPDIR || \"/tmp\",\n `.alepha-vendor-${Date.now()}`,\n );\n\n this.log.debug(`Cloning ${remote}@${commit} into ${tmpDir}`);\n\n await this.shell.run(`git init ${tmpDir}`, { capture: true });\n await this.shell.run(`git -C ${tmpDir} remote add origin ${remote}`, {\n capture: true,\n });\n await this.shell.run(`git -C ${tmpDir} fetch --depth 1 origin ${commit}`, {\n capture: true,\n });\n await this.shell.run(`git -C ${tmpDir} checkout FETCH_HEAD`, {\n capture: true,\n });\n\n return tmpDir;\n }\n\n /**\n * Get the HEAD commit hash from a cloned repository.\n */\n protected async getCommitHash(repoDir: string): Promise<string> {\n const hash = await this.shell.run(`git -C ${repoDir} rev-parse HEAD`, {\n capture: true,\n });\n return hash.trim();\n }\n\n /**\n * Read the vendor lock file.\n */\n protected async readLock(root: string): Promise<VendorLock | undefined> {\n const lockPath = this.fs.join(root, \".alepha\", \"vendor.json\");\n const exists = await this.fs.exists(lockPath);\n if (!exists) {\n return undefined;\n }\n const content = await this.fs.readFile(lockPath);\n return JSON.parse(content.toString());\n }\n\n /**\n * Write the vendor lock file.\n */\n protected async writeLock(root: string, lock: VendorLock): Promise<void> {\n const dir = this.fs.join(root, \".alepha\");\n await this.fs.mkdir(dir, { recursive: true });\n await this.fs.writeFile(\n this.fs.join(dir, \"vendor.json\"),\n JSON.stringify(lock, null, 2),\n );\n }\n\n /**\n * Directories to ignore during diff comparisons.\n */\n protected readonly ignoredPaths = [\n \"__tests__\",\n \"assets/swagger-ui\",\n \"node_modules\",\n \"dist\",\n ];\n\n /**\n * Check if a file path should be ignored during diff.\n */\n protected isIgnored(filePath: string): boolean {\n if (\n filePath.endsWith(\".spec.ts\") ||\n filePath.endsWith(\".spec.tsx\") ||\n filePath === \"LICENSE\" ||\n filePath === \"tsdown.config.ts\"\n ) {\n return true;\n }\n return this.ignoredPaths.some(\n (p) =>\n filePath === p ||\n filePath.startsWith(`${p}/`) ||\n filePath.includes(`/${p}/`) ||\n filePath.endsWith(`/${p}`),\n );\n }\n\n /**\n * Recursively compare two directories and return the differences.\n */\n protected async diffDirectories(\n localDir: string,\n remoteDir: string,\n ): Promise<{\n added: string[];\n modified: VendorFileDiff[];\n removed: string[];\n }> {\n const added: string[] = [];\n const modified: VendorFileDiff[] = [];\n const removed: string[] = [];\n\n const [localFiles, remoteFiles] = await Promise.all([\n this.fs.ls(localDir, { recursive: true }),\n this.fs.ls(remoteDir, { recursive: true }),\n ]);\n\n const filteredLocal = localFiles.filter((f) => !this.isIgnored(f));\n const filteredRemote = remoteFiles.filter((f) => !this.isIgnored(f));\n\n const localSet = new Set(filteredLocal);\n const remoteSet = new Set(filteredRemote);\n\n // Files in baseline but not local = user deleted them\n for (const file of filteredRemote) {\n if (!localSet.has(file)) {\n removed.push(file);\n continue;\n }\n\n try {\n const [localContent, remoteContent] = await Promise.all([\n this.fs.readFile(this.fs.join(localDir, file)),\n this.fs.readFile(this.fs.join(remoteDir, file)),\n ]);\n\n if (!localContent.equals(remoteContent)) {\n const changes = this.computeLineDiff(\n remoteContent.toString(),\n localContent.toString(),\n );\n modified.push({ file, changes });\n }\n } catch {\n // Skip directories and unreadable entries\n }\n }\n\n // Files in local but not baseline = user added them\n for (const file of filteredLocal) {\n if (!remoteSet.has(file)) {\n added.push(file);\n }\n }\n\n return { added, modified, removed };\n }\n\n /**\n * Compute line-level differences between two file contents.\n *\n * Uses a longest-common-subsequence algorithm to produce minimal\n * added/removed line changes with accurate line numbers.\n */\n protected computeLineDiff(baseline: string, local: string): VendorLineDiff[] {\n const baseLines = baseline.split(\"\\n\");\n const localLines = local.split(\"\\n\");\n const lcs = this.longestCommonSubsequence(baseLines, localLines);\n const changes: VendorLineDiff[] = [];\n\n let bi = 0;\n let li = 0;\n let ci = 0;\n\n while (bi < baseLines.length || li < localLines.length) {\n if (ci < lcs.length && bi < baseLines.length && li < localLines.length) {\n if (baseLines[bi] === lcs[ci] && localLines[li] === lcs[ci]) {\n // Line is unchanged\n bi++;\n li++;\n ci++;\n } else if (baseLines[bi] !== lcs[ci]) {\n changes.push({ line: bi + 1, type: \"removed\", text: baseLines[bi] });\n bi++;\n } else {\n changes.push({ line: li + 1, type: \"added\", text: localLines[li] });\n li++;\n }\n } else if (bi < baseLines.length) {\n changes.push({ line: bi + 1, type: \"removed\", text: baseLines[bi] });\n bi++;\n } else {\n changes.push({ line: li + 1, type: \"added\", text: localLines[li] });\n li++;\n }\n }\n\n return changes;\n }\n\n /**\n * Compute the longest common subsequence of two string arrays.\n */\n protected longestCommonSubsequence(a: string[], b: string[]): string[] {\n const m = a.length;\n const n = b.length;\n const dp: number[][] = Array.from({ length: m + 1 }, () =>\n Array(n + 1).fill(0),\n );\n\n for (let i = 1; i <= m; i++) {\n for (let j = 1; j <= n; j++) {\n dp[i][j] =\n a[i - 1] === b[j - 1]\n ? dp[i - 1][j - 1] + 1\n : Math.max(dp[i - 1][j], dp[i][j - 1]);\n }\n }\n\n const result: string[] = [];\n let i = m;\n let j = n;\n while (i > 0 && j > 0) {\n if (a[i - 1] === b[j - 1]) {\n result.unshift(a[i - 1]);\n i--;\n j--;\n } else if (dp[i - 1][j] > dp[i][j - 1]) {\n i--;\n } else {\n j--;\n }\n }\n\n return result;\n }\n}\n","import { $inject, $state, AlephaError, t } from \"alepha\";\nimport { PackageManagerUtils } from \"alepha/cli\";\nimport { $command } from \"alepha/command\";\nimport { $logger, ConsoleColorProvider } from \"alepha/logger\";\nimport { vendorOptions } from \"../atoms/vendorOptions.ts\";\nimport type {\n VendorDiffResult,\n VendorPackageDiff,\n VendorSyncResult,\n} from \"../services/VendorService.ts\";\nimport { VendorService } from \"../services/VendorService.ts\";\n\n/**\n * Default remote when none is configured.\n */\nconst DEFAULT_REMOTE = \"git@github.com:feunard/alepha.git\";\n\nexport class VendorCommand {\n protected readonly log = $logger();\n protected readonly options = $state(vendorOptions);\n protected readonly vendorService = $inject(VendorService);\n protected readonly color = $inject(ConsoleColorProvider);\n protected readonly pm = $inject(PackageManagerUtils);\n\n /**\n * Ensure vendor config is present and return resolved options.\n */\n protected resolveOptions() {\n if (!this.options) {\n throw new AlephaError(\n 'Missing vendor configuration. Add a \"vendor\" section to alepha.config.ts.',\n );\n }\n return {\n remote: this.options.remote ?? DEFAULT_REMOTE,\n branch: this.options.branch ?? \"main\",\n packages: this.options.packages,\n };\n }\n\n // ─────────────────────────────────────────────────────────────────────────\n // alepha vendor sync\n // ─────────────────────────────────────────────────────────────────────────\n\n protected readonly syncFlags = t.object({\n force: t.optional(\n t.boolean({\n aliases: [\"f\"],\n description: \"Skip local modification check\",\n }),\n ),\n });\n\n protected readonly sync = $command({\n name: \"sync\",\n description: \"Replace local packages with remote source\",\n flags: this.syncFlags,\n handler: async ({ flags, root, run }) => {\n const opts = this.resolveOptions();\n const c = this.color;\n\n let result: VendorSyncResult = { synced: [], errors: [] };\n\n await run({\n name: `Syncing from ${opts.branch}`,\n handler: async () => {\n result = await this.vendorService.sync({\n root,\n remote: opts.remote,\n branch: opts.branch,\n packages: opts.packages,\n force: flags.force,\n });\n },\n });\n\n if (result.aborted) {\n run.end();\n\n process.stdout.write(\n `\\nLocal modifications detected. Use ${c.set(\"CYAN\", \"--force\")} to overwrite.\\n`,\n );\n\n for (const pkg of result.aborted.packages) {\n this.printPackageDiff(pkg);\n }\n\n process.stdout.write(\"\\n\");\n return;\n }\n\n if (result.synced.length > 0) {\n const pmName = await this.pm.getPackageManager(root);\n await run(`${pmName} install`, { root });\n }\n\n run.end();\n\n if (result.errors.length > 0) {\n for (const error of result.errors) {\n process.stdout.write(`${c.set(\"RED\", \" error\")} ${error}\\n`);\n }\n }\n\n if (result.synced.length > 0) {\n process.stdout.write(\n `\\nSynced ${c.set(\"CYAN\", String(result.synced.length))} ${result.synced.length === 1 ? \"package\" : \"packages\"} from ${c.set(\"CYAN\", opts.branch)}\\n`,\n );\n for (const pkg of result.synced) {\n process.stdout.write(` ${c.set(\"GREEN\", \"\\u2713\")} ${pkg}\\n`);\n }\n }\n\n process.stdout.write(\"\\n\");\n },\n });\n\n // ─────────────────────────────────────────────────────────────────────────\n // alepha vendor diff\n // ─────────────────────────────────────────────────────────────────────────\n\n protected readonly diff = $command({\n name: \"diff\",\n description: \"Compare local packages against remote\",\n handler: async ({ root, run }) => {\n const opts = this.resolveOptions();\n\n let result: VendorDiffResult = { packages: [], totalChanges: 0 };\n\n await run({\n name: `Cloning ${opts.remote} at ${opts.branch}`,\n handler: async () => {\n result = await this.vendorService.diff({\n root,\n remote: opts.remote,\n branch: opts.branch,\n packages: opts.packages,\n });\n },\n });\n\n run.end();\n\n if (result.totalChanges === 0) {\n process.stdout.write(\"\\nNo changes\\n\\n\");\n return;\n }\n\n for (const pkg of result.packages) {\n this.printPackageDiff(pkg);\n }\n\n process.stdout.write(\"\\n\");\n },\n });\n\n // ─────────────────────────────────────────────────────────────────────────\n // Helpers\n // ─────────────────────────────────────────────────────────────────────────\n\n protected printPackageDiff(pkg: VendorPackageDiff) {\n const c = this.color;\n const count = pkg.added.length + pkg.modified.length + pkg.removed.length;\n\n if (count === 0) {\n process.stdout.write(`\\n${c.set(\"CYAN\", pkg.name)}: no changes\\n`);\n return;\n }\n\n process.stdout.write(\n `\\n${c.set(\"CYAN\", pkg.name)}: ${count} ${count === 1 ? \"file differs\" : \"files differ\"}\\n`,\n );\n\n for (const file of pkg.added) {\n process.stdout.write(` ${c.set(\"GREEN\", \"A\")} ${file}\\n`);\n }\n\n for (const fileDiff of pkg.modified) {\n process.stdout.write(` ${c.set(\"ORANGE\", \"M\")} ${fileDiff.file}\\n`);\n for (const change of fileDiff.changes) {\n const prefix = change.type === \"removed\" ? \"-\" : \"+\";\n const color = change.type === \"removed\" ? \"RED\" : \"GREEN\";\n const lineNum = `L${change.line}`;\n process.stdout.write(\n ` ${c.set(\"DIM\", lineNum.padEnd(5))} ${c.set(color, `${prefix} ${change.text}`)}\\n`,\n );\n }\n }\n\n for (const file of pkg.removed) {\n process.stdout.write(` ${c.set(\"RED\", \"D\")} ${file}\\n`);\n }\n }\n\n // ─────────────────────────────────────────────────────────────────────────\n // Parent command\n // ─────────────────────────────────────────────────────────────────────────\n\n public readonly vendor = $command({\n name: \"vendor\",\n description: \"Vendor Alepha packages into the project\",\n children: [this.sync, this.diff],\n handler: async ({ help }) => {\n help();\n },\n });\n}\n","import { $context, $module } from \"alepha\";\nimport { type VendorOptions, vendorOptions } from \"./atoms/vendorOptions.ts\";\nimport { VendorCommand } from \"./commands/VendorCommand.ts\";\nimport { VendorService } from \"./services/VendorService.ts\";\n\n// ---------------------------------------------------------------------------\n\n/**\n * CLI plugin for vendoring Alepha packages into external projects.\n *\n * Copies package source code from a git remote into the current project's\n * `packages/` directory. Useful for corporate projects that need a local\n * copy of Alepha for AI tooling, audits, documentation, or quick fixes.\n *\n * Commands:\n * - `alepha vendor sync` — replace local packages with remote source\n * - `alepha vendor diff` — compare local packages against remote HEAD\n *\n * Configuration in `alepha.config.ts`:\n *\n * ```typescript\n * import { vendor } from \"alepha/cli/vendor\";\n *\n * export default defineConfig({\n * plugins: [\n * vendor({\n * branch: \"main\",\n * packages: [\"alepha\", \"@alepha/payments-stripe\"],\n * }),\n * ],\n * });\n * ```\n */\nexport const AlephaCliVendorPlugin = $module({\n name: \"alepha.cli.plugins.vendor\",\n atoms: [vendorOptions],\n services: [VendorCommand, VendorService],\n});\n\nexport const vendor = (options: VendorOptions) => {\n return () => {\n const { alepha } = $context();\n alepha.with(AlephaCliVendorPlugin).set(vendorOptions, options);\n };\n};\n\n// ---------------------------------------------------------------------------\n\nexport * from \"./atoms/vendorOptions.ts\";\nexport * from \"./commands/VendorCommand.ts\";\nexport * from \"./services/VendorService.ts\";\n"],"mappings":";;;;;;;;;;;;AAQA,MAAa,gBAAgB,MAAM;CACjC,MAAM;CACN,aAAa;CACb,QAAQ,EAAE,SACR,EAAE,OAAO;;;;;;EAMP,QAAQ,EAAE,SAAS,EAAE,MAAM,CAAC;;;;;;EAO5B,QAAQ,EAAE,SAAS,EAAE,MAAM,CAAC;;;;;;EAO5B,UAAU,EAAE,MAAM,EAAE,MAAM,CAAC;EAC5B,CAAC,CACH;CACF,CAAC;;;;;;AC4CF,IAAa,gBAAb,MAA2B;CACzB,MAAyB,SAAS;CAClC,QAA2B,QAAQ,cAAc;CACjD,KAAwB,QAAQ,mBAAmB;;;;;;;;;;CAWnD,MAAM,KAAK,SAAuD;EAChE,MAAM,SAAmB,EAAE;EAC3B,MAAM,SAAmB,EAAE;EAE3B,IAAI,CAAC,QAAQ,OAAO;GAClB,MAAM,OAAO,MAAM,KAAK,SAAS,QAAQ,KAAK;GAE9C,IAAI,MAAM;IACR,IAAI;IACJ,IAAI;KACF,cAAc,MAAM,KAAK,cAAc,QAAQ,QAAQ,KAAK,OAAO;KACnE,MAAM,aAAa,MAAM,KAAK,cAC5B,QAAQ,MACR,aACA,QAAQ,SACT;KAED,IAAI,WAAW,eAAe,GAC5B,OAAO;MAAE,QAAQ,EAAE;MAAE,QAAQ,EAAE;MAAE,SAAS;MAAY;cAEhD;KACR,IAAI,aACF,MAAM,KAAK,GAAG,GAAG,aAAa;MAAE,WAAW;MAAM,OAAO;MAAM,CAAC;;;;EAMvE,IAAI;EAEJ,IAAI;GACF,SAAS,MAAM,KAAK,YAAY,QAAQ,QAAQ,QAAQ,OAAO;GAE/D,KAAK,MAAM,OAAO,QAAQ,UAAU;IAClC,MAAM,eAAe,KAAK,GAAG,KAAK,QAAQ,YAAY,IAAI;IAC1D,MAAM,cAAc,KAAK,GAAG,KAAK,QAAQ,MAAM,YAAY,IAAI;IAG/D,IAAI,CAAC,MADsB,KAAK,GAAG,OAAO,aAAa,EACpC;KACjB,OAAO,KAAK,YAAY,IAAI,uBAAuB;KACnD;;IAGF,KAAK,IAAI,MAAM,oBAAoB,MAAM;IAEzC,MAAM,KAAK,GAAG,GAAG,aAAa;KAAE,WAAW;KAAM,OAAO;KAAM,CAAC;IAC/D,MAAM,KAAK,GAAG,GAAG,cAAc,aAAa,EAAE,WAAW,MAAM,CAAC;IAChE,MAAM,KAAK,mBAAmB,YAAY;IAE1C,OAAO,KAAK,IAAI;;GAGlB,MAAM,SAAS,MAAM,KAAK,cAAc,OAAO;GAC/C,MAAM,KAAK,UAAU,QAAQ,MAAM,EAAE,QAAQ,CAAC;YACtC;GACR,IAAI,QACF,MAAM,KAAK,GAAG,GAAG,QAAQ;IAAE,WAAW;IAAM,OAAO;IAAM,CAAC;;EAI9D,OAAO;GAAE;GAAQ;GAAQ;;;;;;;;CAS3B,MAAM,KAAK,SAAuD;EAChE,MAAM,OAAO,MAAM,KAAK,SAAS,QAAQ,KAAK;EAC9C,IAAI,CAAC,MACH,OAAO;GAAE,UAAU,EAAE;GAAE,cAAc;GAAG;EAG1C,IAAI;EAEJ,IAAI;GACF,SAAS,MAAM,KAAK,cAAc,QAAQ,QAAQ,KAAK,OAAO;GAC9D,OAAO,MAAM,KAAK,cAAc,QAAQ,MAAM,QAAQ,QAAQ,SAAS;YAC/D;GACR,IAAI,QACF,MAAM,KAAK,GAAG,GAAG,QAAQ;IAAE,WAAW;IAAM,OAAO;IAAM,CAAC;;;;;;CAQhE,MAAgB,cACd,MACA,QACA,UAC2B;EAC3B,MAAM,UAA+B,EAAE;EACvC,IAAI,eAAe;EAEnB,KAAK,MAAM,OAAO,UAAU;GAC1B,MAAM,eAAe,KAAK,GAAG,KAAK,QAAQ,YAAY,IAAI;GAC1D,MAAM,cAAc,KAAK,GAAG,KAAK,MAAM,YAAY,IAAI;GAEvD,MAAM,eAAe,MAAM,KAAK,GAAG,OAAO,aAAa;GACvD,MAAM,cAAc,MAAM,KAAK,GAAG,OAAO,YAAY;GAErD,IAAI,CAAC,gBAAgB,CAAC,aAAa;IACjC,QAAQ,KAAK;KAAE,MAAM;KAAK,OAAO,EAAE;KAAE,UAAU,EAAE;KAAE,SAAS,EAAE;KAAE,CAAC;IACjE;;GAGF,IAAI,CAAC,cAAc;IAEjB,MAAM,aAAa,MAAM,KAAK,GAAG,GAAG,aAAa,EAAE,WAAW,MAAM,CAAC;IACrE,QAAQ,KAAK;KACX,MAAM;KACN,OAAO;KACP,UAAU,EAAE;KACZ,SAAS,EAAE;KACZ,CAAC;IACF,gBAAgB,WAAW;IAC3B;;GAGF,IAAI,CAAC,aAAa;IAEhB,MAAM,cAAc,MAAM,KAAK,GAAG,GAAG,cAAc,EAAE,WAAW,MAAM,CAAC;IACvE,QAAQ,KAAK;KACX,MAAM;KACN,OAAO,EAAE;KACT,UAAU,EAAE;KACZ,SAAS;KACV,CAAC;IACF,gBAAgB,YAAY;IAC5B;;GAGF,MAAM,SAAS,MAAM,KAAK,gBAAgB,aAAa,aAAa;GACpE,MAAM,aACJ,OAAO,MAAM,SAAS,OAAO,SAAS,SAAS,OAAO,QAAQ;GAChE,gBAAgB;GAEhB,QAAQ,KAAK;IACX,MAAM;IACN,OAAO,OAAO;IACd,UAAU,OAAO;IACjB,SAAS,OAAO;IACjB,CAAC;;EAGJ,OAAO;GAAE,UAAU;GAAS;GAAc;;;;;CAM5C,MAAgB,mBAAmB,QAA+B;EAChE,MAAM,WAAW,MAAM,KAAK,GAAG,GAAG,QAAQ,EAAE,WAAW,MAAM,CAAC;EAG9D,KAAK,MAAM,QAAQ,UACjB,IACE,KAAK,SAAS,WAAW,IACzB,KAAK,SAAS,YAAY,IAC1B,SAAS,aACT,SAAS,oBAET,MAAM,KAAK,GAAG,GAAG,KAAK,GAAG,KAAK,QAAQ,KAAK,EAAE,EAAE,OAAO,MAAM,CAAC;EAKjE,KAAK,MAAM,QAAQ,UACjB,KAAK,MAAM,WAAW,KAAK,cACzB,IACE,SAAS,WACT,KAAK,WAAW,GAAG,QAAQ,GAAG,IAC9B,KAAK,SAAS,IAAI,QAAQ,GAAG,IAC7B,KAAK,SAAS,IAAI,UAAU,EAC5B;GAEA,MAAM,MAAM,KAAK,QAAQ,QAAQ;GACjC,MAAM,UAAU,KAAK,GAAG,KACtB,QACA,KAAK,UAAU,GAAG,MAAM,QAAQ,OAAO,CACxC;GACD,MAAM,KAAK,GAAG,GAAG,SAAS;IAAE,WAAW;IAAM,OAAO;IAAM,CAAC;;;;;;CASnE,MAAgB,YAAY,QAAgB,QAAiC;EAC3E,MAAM,SAAS,KAAK,GAAG,KACrB,QAAQ,IAAI,UAAU,QACtB,kBAAkB,KAAK,KAAK,GAC7B;EAED,KAAK,IAAI,MAAM,WAAW,OAAO,GAAG,OAAO,QAAQ,SAAS;EAE5D,MAAM,SAAS,MAAM,KAAK,MAAM,IAC9B,gCAAgC,OAAO,sBAAsB,OAAO,GAAG,UACvE,EAAE,SAAS,MAAM,CAClB;EAED,IAAI,QACF,KAAK,IAAI,MAAM,OAAO;EAGxB,OAAO;;;;;CAMT,MAAgB,cACd,QACA,QACiB;EACjB,MAAM,SAAS,KAAK,GAAG,KACrB,QAAQ,IAAI,UAAU,QACtB,kBAAkB,KAAK,KAAK,GAC7B;EAED,KAAK,IAAI,MAAM,WAAW,OAAO,GAAG,OAAO,QAAQ,SAAS;EAE5D,MAAM,KAAK,MAAM,IAAI,YAAY,UAAU,EAAE,SAAS,MAAM,CAAC;EAC7D,MAAM,KAAK,MAAM,IAAI,UAAU,OAAO,qBAAqB,UAAU,EACnE,SAAS,MACV,CAAC;EACF,MAAM,KAAK,MAAM,IAAI,UAAU,OAAO,0BAA0B,UAAU,EACxE,SAAS,MACV,CAAC;EACF,MAAM,KAAK,MAAM,IAAI,UAAU,OAAO,uBAAuB,EAC3D,SAAS,MACV,CAAC;EAEF,OAAO;;;;;CAMT,MAAgB,cAAc,SAAkC;EAI9D,QAAO,MAHY,KAAK,MAAM,IAAI,UAAU,QAAQ,kBAAkB,EACpE,SAAS,MACV,CAAC,EACU,MAAM;;;;;CAMpB,MAAgB,SAAS,MAA+C;EACtE,MAAM,WAAW,KAAK,GAAG,KAAK,MAAM,WAAW,cAAc;EAE7D,IAAI,CAAC,MADgB,KAAK,GAAG,OAAO,SAAS,EAE3C;EAEF,MAAM,UAAU,MAAM,KAAK,GAAG,SAAS,SAAS;EAChD,OAAO,KAAK,MAAM,QAAQ,UAAU,CAAC;;;;;CAMvC,MAAgB,UAAU,MAAc,MAAiC;EACvE,MAAM,MAAM,KAAK,GAAG,KAAK,MAAM,UAAU;EACzC,MAAM,KAAK,GAAG,MAAM,KAAK,EAAE,WAAW,MAAM,CAAC;EAC7C,MAAM,KAAK,GAAG,UACZ,KAAK,GAAG,KAAK,KAAK,cAAc,EAChC,KAAK,UAAU,MAAM,MAAM,EAAE,CAC9B;;;;;CAMH,eAAkC;EAChC;EACA;EACA;EACA;EACD;;;;CAKD,UAAoB,UAA2B;EAC7C,IACE,SAAS,SAAS,WAAW,IAC7B,SAAS,SAAS,YAAY,IAC9B,aAAa,aACb,aAAa,oBAEb,OAAO;EAET,OAAO,KAAK,aAAa,MACtB,MACC,aAAa,KACb,SAAS,WAAW,GAAG,EAAE,GAAG,IAC5B,SAAS,SAAS,IAAI,EAAE,GAAG,IAC3B,SAAS,SAAS,IAAI,IAAI,CAC7B;;;;;CAMH,MAAgB,gBACd,UACA,WAKC;EACD,MAAM,QAAkB,EAAE;EAC1B,MAAM,WAA6B,EAAE;EACrC,MAAM,UAAoB,EAAE;EAE5B,MAAM,CAAC,YAAY,eAAe,MAAM,QAAQ,IAAI,CAClD,KAAK,GAAG,GAAG,UAAU,EAAE,WAAW,MAAM,CAAC,EACzC,KAAK,GAAG,GAAG,WAAW,EAAE,WAAW,MAAM,CAAC,CAC3C,CAAC;EAEF,MAAM,gBAAgB,WAAW,QAAQ,MAAM,CAAC,KAAK,UAAU,EAAE,CAAC;EAClE,MAAM,iBAAiB,YAAY,QAAQ,MAAM,CAAC,KAAK,UAAU,EAAE,CAAC;EAEpE,MAAM,WAAW,IAAI,IAAI,cAAc;EACvC,MAAM,YAAY,IAAI,IAAI,eAAe;EAGzC,KAAK,MAAM,QAAQ,gBAAgB;GACjC,IAAI,CAAC,SAAS,IAAI,KAAK,EAAE;IACvB,QAAQ,KAAK,KAAK;IAClB;;GAGF,IAAI;IACF,MAAM,CAAC,cAAc,iBAAiB,MAAM,QAAQ,IAAI,CACtD,KAAK,GAAG,SAAS,KAAK,GAAG,KAAK,UAAU,KAAK,CAAC,EAC9C,KAAK,GAAG,SAAS,KAAK,GAAG,KAAK,WAAW,KAAK,CAAC,CAChD,CAAC;IAEF,IAAI,CAAC,aAAa,OAAO,cAAc,EAAE;KACvC,MAAM,UAAU,KAAK,gBACnB,cAAc,UAAU,EACxB,aAAa,UAAU,CACxB;KACD,SAAS,KAAK;MAAE;MAAM;MAAS,CAAC;;WAE5B;;EAMV,KAAK,MAAM,QAAQ,eACjB,IAAI,CAAC,UAAU,IAAI,KAAK,EACtB,MAAM,KAAK,KAAK;EAIpB,OAAO;GAAE;GAAO;GAAU;GAAS;;;;;;;;CASrC,gBAA0B,UAAkB,OAAiC;EAC3E,MAAM,YAAY,SAAS,MAAM,KAAK;EACtC,MAAM,aAAa,MAAM,MAAM,KAAK;EACpC,MAAM,MAAM,KAAK,yBAAyB,WAAW,WAAW;EAChE,MAAM,UAA4B,EAAE;EAEpC,IAAI,KAAK;EACT,IAAI,KAAK;EACT,IAAI,KAAK;EAET,OAAO,KAAK,UAAU,UAAU,KAAK,WAAW,QAC9C,IAAI,KAAK,IAAI,UAAU,KAAK,UAAU,UAAU,KAAK,WAAW,QAC9D,IAAI,UAAU,QAAQ,IAAI,OAAO,WAAW,QAAQ,IAAI,KAAK;GAE3D;GACA;GACA;SACK,IAAI,UAAU,QAAQ,IAAI,KAAK;GACpC,QAAQ,KAAK;IAAE,MAAM,KAAK;IAAG,MAAM;IAAW,MAAM,UAAU;IAAK,CAAC;GACpE;SACK;GACL,QAAQ,KAAK;IAAE,MAAM,KAAK;IAAG,MAAM;IAAS,MAAM,WAAW;IAAK,CAAC;GACnE;;OAEG,IAAI,KAAK,UAAU,QAAQ;GAChC,QAAQ,KAAK;IAAE,MAAM,KAAK;IAAG,MAAM;IAAW,MAAM,UAAU;IAAK,CAAC;GACpE;SACK;GACL,QAAQ,KAAK;IAAE,MAAM,KAAK;IAAG,MAAM;IAAS,MAAM,WAAW;IAAK,CAAC;GACnE;;EAIJ,OAAO;;;;;CAMT,yBAAmC,GAAa,GAAuB;EACrE,MAAM,IAAI,EAAE;EACZ,MAAM,IAAI,EAAE;EACZ,MAAM,KAAiB,MAAM,KAAK,EAAE,QAAQ,IAAI,GAAG,QACjD,MAAM,IAAI,EAAE,CAAC,KAAK,EAAE,CACrB;EAED,KAAK,IAAI,IAAI,GAAG,KAAK,GAAG,KACtB,KAAK,IAAI,IAAI,GAAG,KAAK,GAAG,KACtB,GAAG,GAAG,KACJ,EAAE,IAAI,OAAO,EAAE,IAAI,KACf,GAAG,IAAI,GAAG,IAAI,KAAK,IACnB,KAAK,IAAI,GAAG,IAAI,GAAG,IAAI,GAAG,GAAG,IAAI,GAAG;EAI9C,MAAM,SAAmB,EAAE;EAC3B,IAAI,IAAI;EACR,IAAI,IAAI;EACR,OAAO,IAAI,KAAK,IAAI,GAClB,IAAI,EAAE,IAAI,OAAO,EAAE,IAAI,IAAI;GACzB,OAAO,QAAQ,EAAE,IAAI,GAAG;GACxB;GACA;SACK,IAAI,GAAG,IAAI,GAAG,KAAK,GAAG,GAAG,IAAI,IAClC;OAEA;EAIJ,OAAO;;;;;;;;AC3gBX,MAAM,iBAAiB;AAEvB,IAAa,gBAAb,MAA2B;CACzB,MAAyB,SAAS;CAClC,UAA6B,OAAO,cAAc;CAClD,gBAAmC,QAAQ,cAAc;CACzD,QAA2B,QAAQ,qBAAqB;CACxD,KAAwB,QAAQ,oBAAoB;;;;CAKpD,iBAA2B;EACzB,IAAI,CAAC,KAAK,SACR,MAAM,IAAI,YACR,8EACD;EAEH,OAAO;GACL,QAAQ,KAAK,QAAQ,UAAU;GAC/B,QAAQ,KAAK,QAAQ,UAAU;GAC/B,UAAU,KAAK,QAAQ;GACxB;;CAOH,YAA+B,EAAE,OAAO,EACtC,OAAO,EAAE,SACP,EAAE,QAAQ;EACR,SAAS,CAAC,IAAI;EACd,aAAa;EACd,CAAC,CACH,EACF,CAAC;CAEF,OAA0B,SAAS;EACjC,MAAM;EACN,aAAa;EACb,OAAO,KAAK;EACZ,SAAS,OAAO,EAAE,OAAO,MAAM,UAAU;GACvC,MAAM,OAAO,KAAK,gBAAgB;GAClC,MAAM,IAAI,KAAK;GAEf,IAAI,SAA2B;IAAE,QAAQ,EAAE;IAAE,QAAQ,EAAE;IAAE;GAEzD,MAAM,IAAI;IACR,MAAM,gBAAgB,KAAK;IAC3B,SAAS,YAAY;KACnB,SAAS,MAAM,KAAK,cAAc,KAAK;MACrC;MACA,QAAQ,KAAK;MACb,QAAQ,KAAK;MACb,UAAU,KAAK;MACf,OAAO,MAAM;MACd,CAAC;;IAEL,CAAC;GAEF,IAAI,OAAO,SAAS;IAClB,IAAI,KAAK;IAET,QAAQ,OAAO,MACb,uCAAuC,EAAE,IAAI,QAAQ,UAAU,CAAC,kBACjE;IAED,KAAK,MAAM,OAAO,OAAO,QAAQ,UAC/B,KAAK,iBAAiB,IAAI;IAG5B,QAAQ,OAAO,MAAM,KAAK;IAC1B;;GAGF,IAAI,OAAO,OAAO,SAAS,GAEzB,MAAM,IAAI,GAAG,MADQ,KAAK,GAAG,kBAAkB,KAAK,CAChC,WAAW,EAAE,MAAM,CAAC;GAG1C,IAAI,KAAK;GAET,IAAI,OAAO,OAAO,SAAS,GACzB,KAAK,MAAM,SAAS,OAAO,QACzB,QAAQ,OAAO,MAAM,GAAG,EAAE,IAAI,OAAO,UAAU,CAAC,GAAG,MAAM,IAAI;GAIjE,IAAI,OAAO,OAAO,SAAS,GAAG;IAC5B,QAAQ,OAAO,MACb,YAAY,EAAE,IAAI,QAAQ,OAAO,OAAO,OAAO,OAAO,CAAC,CAAC,GAAG,OAAO,OAAO,WAAW,IAAI,YAAY,WAAW,QAAQ,EAAE,IAAI,QAAQ,KAAK,OAAO,CAAC,IACnJ;IACD,KAAK,MAAM,OAAO,OAAO,QACvB,QAAQ,OAAO,MAAM,KAAK,EAAE,IAAI,SAAS,IAAS,CAAC,GAAG,IAAI,IAAI;;GAIlE,QAAQ,OAAO,MAAM,KAAK;;EAE7B,CAAC;CAMF,OAA0B,SAAS;EACjC,MAAM;EACN,aAAa;EACb,SAAS,OAAO,EAAE,MAAM,UAAU;GAChC,MAAM,OAAO,KAAK,gBAAgB;GAElC,IAAI,SAA2B;IAAE,UAAU,EAAE;IAAE,cAAc;IAAG;GAEhE,MAAM,IAAI;IACR,MAAM,WAAW,KAAK,OAAO,MAAM,KAAK;IACxC,SAAS,YAAY;KACnB,SAAS,MAAM,KAAK,cAAc,KAAK;MACrC;MACA,QAAQ,KAAK;MACb,QAAQ,KAAK;MACb,UAAU,KAAK;MAChB,CAAC;;IAEL,CAAC;GAEF,IAAI,KAAK;GAET,IAAI,OAAO,iBAAiB,GAAG;IAC7B,QAAQ,OAAO,MAAM,mBAAmB;IACxC;;GAGF,KAAK,MAAM,OAAO,OAAO,UACvB,KAAK,iBAAiB,IAAI;GAG5B,QAAQ,OAAO,MAAM,KAAK;;EAE7B,CAAC;CAMF,iBAA2B,KAAwB;EACjD,MAAM,IAAI,KAAK;EACf,MAAM,QAAQ,IAAI,MAAM,SAAS,IAAI,SAAS,SAAS,IAAI,QAAQ;EAEnE,IAAI,UAAU,GAAG;GACf,QAAQ,OAAO,MAAM,KAAK,EAAE,IAAI,QAAQ,IAAI,KAAK,CAAC,gBAAgB;GAClE;;EAGF,QAAQ,OAAO,MACb,KAAK,EAAE,IAAI,QAAQ,IAAI,KAAK,CAAC,IAAI,MAAM,GAAG,UAAU,IAAI,iBAAiB,eAAe,IACzF;EAED,KAAK,MAAM,QAAQ,IAAI,OACrB,QAAQ,OAAO,MAAM,KAAK,EAAE,IAAI,SAAS,IAAI,CAAC,GAAG,KAAK,IAAI;EAG5D,KAAK,MAAM,YAAY,IAAI,UAAU;GACnC,QAAQ,OAAO,MAAM,KAAK,EAAE,IAAI,UAAU,IAAI,CAAC,GAAG,SAAS,KAAK,IAAI;GACpE,KAAK,MAAM,UAAU,SAAS,SAAS;IACrC,MAAM,SAAS,OAAO,SAAS,YAAY,MAAM;IACjD,MAAM,QAAQ,OAAO,SAAS,YAAY,QAAQ;IAClD,MAAM,UAAU,IAAI,OAAO;IAC3B,QAAQ,OAAO,MACb,SAAS,EAAE,IAAI,OAAO,QAAQ,OAAO,EAAE,CAAC,CAAC,GAAG,EAAE,IAAI,OAAO,GAAG,OAAO,GAAG,OAAO,OAAO,CAAC,IACtF;;;EAIL,KAAK,MAAM,QAAQ,IAAI,SACrB,QAAQ,OAAO,MAAM,KAAK,EAAE,IAAI,OAAO,IAAI,CAAC,GAAG,KAAK,IAAI;;CAQ5D,SAAyB,SAAS;EAChC,MAAM;EACN,aAAa;EACb,UAAU,CAAC,KAAK,MAAM,KAAK,KAAK;EAChC,SAAS,OAAO,EAAE,WAAW;GAC3B,MAAM;;EAET,CAAC;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AC5KJ,MAAa,wBAAwB,QAAQ;CAC3C,MAAM;CACN,OAAO,CAAC,cAAc;CACtB,UAAU,CAAC,eAAe,cAAc;CACzC,CAAC;AAEF,MAAa,UAAU,YAA2B;CAChD,aAAa;EACX,MAAM,EAAE,WAAW,UAAU;EAC7B,OAAO,KAAK,sBAAsB,CAAC,IAAI,eAAe,QAAQ"}
|