@xrmforge/typegen 0.7.0 → 0.7.1

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.
Files changed (39) hide show
  1. package/docs/architecture/00-README.md +26 -0
  2. package/docs/architecture/01-executive-summary.md +11 -0
  3. package/docs/architecture/02-packages.md +110 -0
  4. package/docs/architecture/03-generated-types.md +176 -0
  5. package/docs/architecture/04-cli.md +58 -0
  6. package/docs/architecture/05-build.md +50 -0
  7. package/docs/architecture/06-incremental.md +42 -0
  8. package/docs/architecture/07-http-client.md +59 -0
  9. package/docs/architecture/08-authentication.md +18 -0
  10. package/docs/architecture/09-testing.md +55 -0
  11. package/docs/architecture/10-eslint-plugin.md +82 -0
  12. package/docs/architecture/11-agent-md.md +38 -0
  13. package/docs/architecture/12-xrm-pitfalls.md +14 -0
  14. package/docs/architecture/13-helpers.md +50 -0
  15. package/docs/architecture/14-showcases.md +21 -0
  16. package/docs/architecture/15-ci-cd.md +49 -0
  17. package/docs/architecture/16-technical-debt.md +17 -0
  18. package/docs/architecture/17-roadmap.md +25 -0
  19. package/docs/architecture/18-design-principles.md +22 -0
  20. package/docs/architektur/00-README.md +26 -0
  21. package/docs/architektur/01-zusammenfassung.md +11 -0
  22. package/docs/architektur/02-packages.md +110 -0
  23. package/docs/architektur/03-generierte-typen.md +176 -0
  24. package/docs/architektur/04-cli.md +58 -0
  25. package/docs/architektur/05-build.md +50 -0
  26. package/docs/architektur/06-inkrementell.md +42 -0
  27. package/docs/architektur/07-http-client.md +59 -0
  28. package/docs/architektur/08-authentifizierung.md +18 -0
  29. package/docs/architektur/09-testing.md +55 -0
  30. package/docs/architektur/10-eslint-plugin.md +82 -0
  31. package/docs/architektur/11-agent-md.md +38 -0
  32. package/docs/architektur/12-xrm-fallstricke.md +14 -0
  33. package/docs/architektur/13-helpers.md +50 -0
  34. package/docs/architektur/14-showcases.md +21 -0
  35. package/docs/architektur/15-ci-cd.md +49 -0
  36. package/docs/architektur/16-technische-schulden.md +17 -0
  37. package/docs/architektur/17-roadmap.md +25 -0
  38. package/docs/architektur/18-designprinzipien.md +22 -0
  39. package/package.json +4 -3
@@ -0,0 +1,26 @@
1
+ # XrmForge Architecture
2
+
3
+ > **Status:** Living document describing the current implementation state.
4
+ > **Last updated:** 2026-04-04 (Session 10)
5
+ > **Version:** 7 packages, 666+ tests across all packages.
6
+
7
+ ## Chapters
8
+
9
+ 1. [Executive Summary](01-executive-summary.md)
10
+ 2. [Package Architecture](02-packages.md)
11
+ 3. [Generated Types](03-generated-types.md)
12
+ 4. [CLI Commands](04-cli.md)
13
+ 5. [Build Architecture](05-build.md)
14
+ 6. [Incremental Generation](06-incremental.md)
15
+ 7. [HTTP Client](07-http-client.md)
16
+ 8. [Authentication](08-authentication.md)
17
+ 9. [Testing Framework](09-testing.md)
18
+ 10. [ESLint Plugin](10-eslint-plugin.md)
19
+ 11. [AGENT.md System](11-agent-md.md)
20
+ 12. [@types/xrm Pitfalls](12-xrm-pitfalls.md)
21
+ 13. [@xrmforge/helpers Package](13-helpers.md)
22
+ 14. [Showcases](14-showcases.md)
23
+ 15. [CI/CD](15-ci-cd.md)
24
+ 16. [Technical Debt](16-technical-debt.md)
25
+ 17. [Roadmap](17-roadmap.md)
26
+ 18. [Design Principles](18-design-principles.md)
@@ -0,0 +1,11 @@
1
+ # Executive Summary
2
+
3
+ XrmForge is an open-source TypeScript toolkit for type-safe Dynamics 365 / Dataverse WebResource development. It generates TypeScript declarations from live Dataverse metadata, turning runtime string errors into compile-time type errors.
4
+
5
+ **Core value proposition:** Every field name, OptionSet value, tab name, entity name, and subgrid name becomes a typed constant with IDE autocomplete and compile-time validation.
6
+
7
+ **Target audience:** D365 developers who write form scripts (WebResources) in JavaScript/TypeScript and want compile-time safety, zero magic strings, and modern tooling (esbuild, vitest, ESLint).
8
+
9
+ **Tech stack:** TypeScript, pnpm monorepo with Turborepo, esbuild for IIFE bundles, vitest for testing, @azure/identity for authentication, fast-xml-parser for FormXml parsing.
10
+
11
+ **npm organization:** [@xrmforge](https://www.npmjs.com/org/xrmforge)
@@ -0,0 +1,110 @@
1
+ # Package Architecture
2
+
3
+ ## Package Overview
4
+
5
+ | Package | Version | Tests | Description |
6
+ |---------|---------|-------|-------------|
7
+ | @xrmforge/typegen | 0.6.0 | 444 | Core: type generation engine, metadata client, HTTP client, helpers |
8
+ | @xrmforge/cli | 0.4.2 | 10 | CLI: generate, build, init commands |
9
+ | @xrmforge/testing | 0.2.0 | 76 | Test utilities: createFormMock, fireOnChange, setupXrmMock |
10
+ | @xrmforge/helpers | 0.1.0 | 59 | Browser-safe runtime: select(), parseLookup(), typedForm(), Xrm constants, Action executors |
11
+ | @xrmforge/webapi | 0.1.0 | 45 | Type-safe Xrm.WebApi client with QueryBuilder |
12
+ | @xrmforge/devkit | 0.4.0 | 42 | Build orchestration, scaffolding, AGENT.md generation |
13
+ | @xrmforge/eslint-plugin | 0.2.0 | 32 | 5 D365-specific ESLint rules |
14
+
15
+ **Total:** 708 tests across 7 packages.
16
+
17
+ ## Dependency Graph
18
+
19
+ ```
20
+ @xrmforge/cli
21
+ |-- @xrmforge/typegen (generate command)
22
+ |-- @xrmforge/devkit (build + init commands)
23
+ '-- commander (CLI framework)
24
+
25
+ @xrmforge/typegen
26
+ |-- @azure/identity (authentication)
27
+ '-- fast-xml-parser (FormXml parsing)
28
+
29
+ @xrmforge/devkit
30
+ '-- esbuild (IIFE bundling)
31
+
32
+ @xrmforge/testing (no runtime deps)
33
+ @xrmforge/helpers (no runtime deps)
34
+ @xrmforge/webapi (no runtime deps)
35
+ @xrmforge/eslint-plugin (ESLint peer dep)
36
+ ```
37
+
38
+ ## Package Details
39
+
40
+ ### @xrmforge/typegen
41
+
42
+ The core package. Contains:
43
+
44
+ - **TypeGenerationOrchestrator** - Coordinates the entire generation pipeline
45
+ - **MetadataClient** - Queries Dataverse metadata (entities, forms, OptionSets, Custom APIs)
46
+ - **DataverseHttpClient** - Resilient REST client with retry, rate limiting, concurrency control
47
+ - **ChangeDetector** - Incremental generation via RetrieveMetadataChanges
48
+ - **MetadataCache** - Filesystem-based caching with version stamps
49
+ - **Generators** - Entity interfaces, form interfaces, OptionSet enums, Fields enums, EntityNames, Navigation Properties, Action/Function executors
50
+ - **Helpers** - select(), parseLookup(), parseFormattedValue() (moved to @xrmforge/helpers)
51
+ - **Xrm Constants** - DisplayState, FormNotificationLevel, RequiredLevel, SubmitMode, SaveMode, ClientType, ClientState (moved to @xrmforge/helpers)
52
+ - **Authentication** - createCredential() factory for 4 auth methods
53
+ - **Logging** - Scoped loggers with pluggable sinks (Console, JSON, Silent)
54
+ - **Errors** - Structured error hierarchy with ErrorCode enum (AUTH_1xxx, API_2xxx, META_3xxx, GEN_4xxx, CONFIG_5xxx)
55
+
56
+ ### @xrmforge/cli
57
+
58
+ Command-line interface built with commander.js. Three commands:
59
+ - `xrmforge generate` - Orchestrates TypeGenerationOrchestrator
60
+ - `xrmforge build` - Delegates to devkit build()
61
+ - `xrmforge init` - Delegates to devkit scaffoldProject()
62
+
63
+ ### @xrmforge/testing
64
+
65
+ FormContext mocking for unit tests:
66
+ - `createFormMock<TForm>(values)` - Creates a complete mock from simple key-value pairs
67
+ - `MockAttribute` - getValue/setValue, dirty tracking, onChange handlers, required level, submit mode
68
+ - `MockControl` - visible, disabled, label, notifications
69
+ - `MockUi` - Form notifications, tab/section stubs
70
+ - `MockEntity` - Entity ID, name, primary attribute
71
+ - `fireOnChange(fieldName)` - Triggers registered onChange handlers
72
+ - `setupXrmMock(options)` / `teardownXrmMock()` - Global Xrm mock with WebApi/Navigation stubs
73
+
74
+ ### @xrmforge/helpers
75
+
76
+ Consolidates all browser-safe runtime code. Zero Node.js dependencies. Contains:
77
+ - **Web API helpers** - select(), parseLookup(), parseFormattedValue()
78
+ - **Xrm constants** - DisplayState, SubmitMode, RequiredLevel, SaveMode, ClientType, ClientState, FormNotificationLevel, OperationType
79
+ - **Action/Function executors** - createBoundAction(), executeRequest(), withProgress()
80
+ - **typedForm() proxy** - Proxy-based FormContext wrapper where `form.name` delegates to `getAttribute('name')`
81
+
82
+ ### @xrmforge/webapi
83
+
84
+ Type-safe wrapper around Xrm.WebApi:
85
+ - `retrieve<T>(entityName, id, query)` - Single record
86
+ - `retrieveMultiple<T>(entityName, query, options)` - With pagination (maxPages)
87
+ - `create(entityName, data)` - Returns record ID
88
+ - `update(entityName, id, data)` - Void
89
+ - `remove(entityName, id)` - Void
90
+ - `QueryBuilder` - Fluent API: `.select().filter().orderBy().top().expand().build()`
91
+ - `WebApiError` - Structured errors with statusCode, errorCode, innerMessage
92
+
93
+ ### @xrmforge/devkit
94
+
95
+ Build orchestration and project scaffolding:
96
+ - `build(config)` - Parallel esbuild IIFE builds via Promise.allSettled
97
+ - `watch(config)` - esbuild watch mode with rebuild callbacks
98
+ - `scaffoldProject(config)` - Generates 11 project files from templates
99
+ - `validateBuildConfig(config)` / `resolveBuildConfig(config)` - Config validation
100
+ - `BuildError` with codes: CONFIG_INVALID, ENTRY_NOT_FOUND, BUILD_FAILED, WATCH_ERROR
101
+ - Template system: 7 text templates in `src/scaffold/templates/`, loaded via `template-loader.ts`
102
+
103
+ ### @xrmforge/eslint-plugin
104
+
105
+ 5 rules for D365 form scripts (ESLint v9 flat config):
106
+ - `no-xrm-page` (error) - Forbids deprecated Xrm.Page API
107
+ - `no-magic-optionset` (warn) - Forbids magic numbers in OptionSet comparisons
108
+ - `no-sync-webapi` (error) - Forbids synchronous XMLHttpRequest
109
+ - `require-error-handling` (warn) - Requires try/catch in async on* event handlers
110
+ - `require-namespace` (warn) - Forbids window/globalThis assignments
@@ -0,0 +1,176 @@
1
+ # Generated Types
2
+
3
+ Running `xrmforge generate` produces the following TypeScript declarations:
4
+
5
+ ### 3.1 Entity Interfaces (`entities/{entity}.d.ts`)
6
+
7
+ ```typescript
8
+ declare namespace XrmForge.Entities {
9
+ /** Account | Konto */
10
+ interface Account {
11
+ /** Account Name | Kontoname */
12
+ name: string | null;
13
+ accountid: string | null;
14
+ revenue: number | null;
15
+ _parentaccountid_value: string | null; // Lookup GUID
16
+ // ...
17
+ }
18
+ }
19
+ ```
20
+
21
+ **Type mapping:** String/Memo/EntityName to `string`, Integer/BigInt/Decimal/Double/Money to `number`, Boolean to `boolean`, DateTime/Uniqueidentifier/Lookup to `string`, Picklist/State/Status to `number`.
22
+
23
+ ### 3.2 Entity Fields Enums (`entities/{entity}.d.ts`)
24
+
25
+ ```typescript
26
+ declare namespace XrmForge.Entities {
27
+ const enum AccountFields {
28
+ /** Account Name | Kontoname */
29
+ Name = 'name',
30
+ Revenue = 'revenue',
31
+ // all readable attributes
32
+ }
33
+ }
34
+ ```
35
+
36
+ Used for Web API `$select`: `select(AccountFields.Name, AccountFields.Revenue)`.
37
+
38
+ ### 3.3 Navigation Properties (`entities/{entity}.d.ts`)
39
+
40
+ ```typescript
41
+ declare namespace XrmForge.Entities {
42
+ const enum AccountNavigation {
43
+ PrimaryContactId = 'primarycontactid',
44
+ ContactCustomerAccounts = 'contact_customer_accounts',
45
+ // OneToMany + ManyToMany relationships
46
+ }
47
+ }
48
+ ```
49
+
50
+ ### 3.4 Form Interfaces (`forms/{entity}.d.ts`)
51
+
52
+ ```typescript
53
+ declare namespace XrmForge.Forms.Account {
54
+ // Union type restricting valid field names
55
+ type AccountMainFormFields = 'name' | 'telephone1' | 'revenue';
56
+
57
+ // Mapped type: field name to Xrm attribute type
58
+ type AccountMainFormAttributeMap = {
59
+ name: Xrm.Attributes.StringAttribute;
60
+ telephone1: Xrm.Attributes.StringAttribute;
61
+ revenue: Xrm.Attributes.NumberAttribute;
62
+ };
63
+
64
+ // Mapped type: field name to Xrm control type
65
+ type AccountMainFormControlMap = {
66
+ name: Xrm.Controls.StringControl;
67
+ telephone1: Xrm.Controls.StringControl;
68
+ revenue: Xrm.Controls.NumberControl;
69
+ };
70
+
71
+ // Fields enum for autocomplete
72
+ const enum AccountMainFormFieldsEnum {
73
+ /** Account Name | Kontoname */
74
+ AccountName = 'name',
75
+ Telephone1 = 'telephone1',
76
+ Revenue = 'revenue',
77
+ }
78
+
79
+ // Type-safe FormContext with overloaded getAttribute/getControl
80
+ interface AccountMainForm extends Omit<Xrm.FormContext, 'getAttribute' | 'getControl'> {
81
+ getAttribute<K extends AccountMainFormFields>(name: K): AccountMainFormAttributeMap[K];
82
+ getAttribute(index: number): Xrm.Attributes.Attribute;
83
+ getAttribute(): Xrm.Attributes.Attribute[];
84
+
85
+ getControl<K extends AccountMainFormFields>(name: K): AccountMainFormControlMap[K];
86
+ getControl(index: number): Xrm.Controls.Control;
87
+ getControl(): Xrm.Controls.Control[];
88
+ }
89
+ }
90
+ ```
91
+
92
+ **Special controls** are typed based on their FormXml ClassID:
93
+ - Subgrid: `Xrm.Controls.GridControl`
94
+ - Editable Grid: `Xrm.Controls.GridControl`
95
+ - Quick View: `Xrm.Controls.QuickFormControl`
96
+ - Web Resource / iFrame: `Xrm.Controls.IframeControl`
97
+
98
+ ### 3.5 Tabs/Sections/Subgrids/QuickViews Enums
99
+
100
+ ```typescript
101
+ const enum AccountMainFormTabs { Summary = 'SUMMARY_TAB', Details = 'DETAILS_TAB' }
102
+ const enum AccountMainFormSections { General = 'GENERAL', Address = 'ADDRESS' }
103
+ const enum AccountMainFormSubgrids { Contacts = 'Contacts_Subgrid' }
104
+ const enum AccountMainFormQuickViews { ContactPreview = 'ContactQuickView' }
105
+ ```
106
+
107
+ ### 3.6 OptionSet Enums (`optionsets/{entity}.d.ts`)
108
+
109
+ ```typescript
110
+ declare namespace XrmForge.OptionSets.Account {
111
+ /** Account Category Code | Kontokategoriecode */
112
+ const enum AccountCategoryCode {
113
+ /** Preferred Customer | Bevorzugter Kunde */
114
+ PreferredCustomer = 1,
115
+ Standard = 2,
116
+ }
117
+ }
118
+ ```
119
+
120
+ Includes Picklist, Status, State, and MultiSelectPicklist attributes. Duplicate labels are disambiguated with `_{Value}` suffix.
121
+
122
+ ### 3.7 EntityNames Enum (`entity-names.d.ts`)
123
+
124
+ ```typescript
125
+ declare namespace XrmForge {
126
+ const enum EntityNames {
127
+ Account = 'account',
128
+ Contact = 'contact',
129
+ // all entities in scope
130
+ }
131
+ }
132
+ ```
133
+
134
+ ### 3.8 MockValues Types (in form interfaces)
135
+
136
+ ```typescript
137
+ type AccountMainFormMockValues = {
138
+ name?: string | null;
139
+ telephone1?: string | null;
140
+ revenue?: number | null;
141
+ };
142
+ ```
143
+
144
+ Used with `createFormMock<AccountMainForm, AccountMainFormMockValues>({ name: 'Test' })`.
145
+
146
+ ### 3.9 Action/Function Executors (`actions/{entity|global}.d.ts` + `.ts`)
147
+
148
+ **Declaration (.d.ts):**
149
+ ```typescript
150
+ declare namespace XrmForge.Actions {
151
+ interface NormalizePhoneParams { Input: string; AllowSuspicious?: boolean; }
152
+ interface NormalizePhoneResult { Normalized: string; Status: number; }
153
+ }
154
+ ```
155
+
156
+ **Runtime module (.ts):**
157
+ ```typescript
158
+ import { createUnboundAction } from '@xrmforge/typegen';
159
+ export const NormalizePhone = createUnboundAction<NormalizePhoneParams, NormalizePhoneResult>(
160
+ 'markant_NormalizePhone',
161
+ { Input: { typeName: 'String', structuralProperty: 1 } }
162
+ );
163
+ // Usage: const result = await NormalizePhone.execute({ Input: '123' });
164
+ ```
165
+
166
+ Factory functions: `createBoundAction`, `createUnboundAction`, `createBoundFunction`, `createUnboundFunction`. Batch execution via `executeMultiple()`, progress UI via `withProgress()`.
167
+
168
+ ### 3.10 Dual-Language Labels
169
+
170
+ All generated JSDoc comments support dual-language labels:
171
+ ```typescript
172
+ /** Account Name | Kontoname */
173
+ Name = 'name',
174
+ ```
175
+
176
+ German umlauts are transliterated in identifiers: ae, oe, ue, ss (e.g. "Ubergeordnet" becomes `Uebergeordnet`).
@@ -0,0 +1,58 @@
1
+ # CLI Commands
2
+
3
+ ### 4.1 `xrmforge generate`
4
+
5
+ Generates TypeScript declarations from a Dataverse environment.
6
+
7
+ | Flag | Type | Default | Description |
8
+ |------|------|---------|-------------|
9
+ | `--url <url>` | string | required | Dataverse environment URL |
10
+ | `--auth <method>` | string | required | Auth method: client-credentials, interactive, device-code, token |
11
+ | `--tenant-id <id>` | string | varies | Azure AD tenant ID |
12
+ | `--client-id <id>` | string | varies | Azure AD application ID |
13
+ | `--client-secret <s>` | string | varies | Client secret (client-credentials only) |
14
+ | `--token <token>` | string | varies | Pre-acquired bearer token (token auth only) |
15
+ | `--entities <list>` | string | - | Comma-separated entity logical names |
16
+ | `--solutions <list>` | string | - | Comma-separated solution unique names |
17
+ | `--output <dir>` | string | ./typings | Output directory |
18
+ | `--label-language <n>` | string | 1033 | Primary label language (LCID) |
19
+ | `--secondary-language <n>` | string | - | Secondary label language for JSDoc |
20
+ | `--no-forms` | flag | - | Skip form interface generation |
21
+ | `--no-optionsets` | flag | - | Skip OptionSet enum generation |
22
+ | `--actions` | flag | false | Generate Custom API executors |
23
+ | `--actions-filter <prefix>` | string | - | Filter Custom APIs by uniquename prefix |
24
+ | `--cache` | flag | false | Enable metadata caching for incremental generation |
25
+ | `--no-cache` | flag | - | Force full metadata refresh |
26
+ | `--cache-dir <dir>` | string | .xrmforge/cache | Cache directory |
27
+ | `-v, --verbose` | flag | false | Debug logging |
28
+
29
+ At least one of `--entities` or `--solutions` is required.
30
+
31
+ ### 4.2 `xrmforge build`
32
+
33
+ Builds WebResources as IIFE bundles using esbuild (via @xrmforge/devkit).
34
+
35
+ | Flag | Type | Default | Description |
36
+ |------|------|---------|-------------|
37
+ | `--watch` | flag | false | Watch mode with incremental rebuilds |
38
+ | `--minify` | flag | from config | Override minification setting |
39
+ | `--no-sourcemap` | flag | - | Disable source maps |
40
+ | `--out-dir <dir>` | string | from config | Override output directory |
41
+ | `-v, --verbose` | flag | false | Show error stacks |
42
+
43
+ Reads configuration from `xrmforge.config.json`.
44
+
45
+ ### 4.3 `xrmforge init`
46
+
47
+ Scaffolds a new D365 form scripting project.
48
+
49
+ | Flag | Type | Default | Description |
50
+ |------|------|---------|-------------|
51
+ | `[dir]` | positional | . | Target directory |
52
+ | `--name <name>` | string | dir name | Project name for package.json |
53
+ | `--prefix <prefix>` | string | contoso | Publisher prefix |
54
+ | `--namespace <ns>` | string | PascalCase(prefix) | Base namespace for scripts |
55
+ | `--skip-install` | flag | false | Skip npm install |
56
+ | `--force` | flag | false | Allow non-empty directories |
57
+
58
+ Generates 11 files: package.json, tsconfig.json, xrmforge.config.json, vitest.config.ts, .gitignore, AGENT.md, example-form.ts, example-form.test.ts, typings/.gitkeep, GitHub Actions CI, Azure DevOps Pipeline.
@@ -0,0 +1,50 @@
1
+ # Build Architecture
2
+
3
+ ### 5.1 esbuild IIFE Bundles
4
+
5
+ XrmForge uses esbuild to create IIFE (Immediately Invoked Function Expression) bundles for Dynamics 365. D365 requires scripts to be registered as namespace.function (e.g. `Contoso.Account.onLoad`).
6
+
7
+ **esbuild configuration per entry:**
8
+ ```
9
+ format: 'iife'
10
+ bundle: true
11
+ globalName: entry.namespace // e.g. 'Contoso.Account'
12
+ target: config.target // default: 'es2020'
13
+ minify: config.minify
14
+ sourcemap: config.sourcemap
15
+ external: config.external // e.g. ['fs', 'path'] for Node.js deps
16
+ ```
17
+
18
+ All entries are built in parallel using `Promise.allSettled()`, allowing partial success.
19
+
20
+ ### 5.2 xrmforge.config.json Schema
21
+
22
+ ```json
23
+ {
24
+ "build": {
25
+ "outDir": "./dist/prefix_/JS",
26
+ "target": "es2020",
27
+ "sourcemap": true,
28
+ "minify": true,
29
+ "external": [],
30
+ "entries": {
31
+ "entry_name": {
32
+ "input": "./src/forms/account-form.ts",
33
+ "namespace": "Contoso.Account",
34
+ "out": "Account/OnLoad.js"
35
+ }
36
+ }
37
+ }
38
+ }
39
+ ```
40
+
41
+ ### 5.3 globalName Handling
42
+
43
+ esbuild automatically creates nested globals from dotted namespaces:
44
+ ```javascript
45
+ // namespace: "Contoso.Account" produces:
46
+ var Contoso = Contoso || {};
47
+ Contoso.Account = (() => { return { onLoad, onSave }; })();
48
+ ```
49
+
50
+ D365 event registration: `Contoso.Account.onLoad`.
@@ -0,0 +1,42 @@
1
+ # Incremental Generation
2
+
3
+ ### 6.1 Overview
4
+
5
+ Incremental generation uses the Dataverse `RetrieveMetadataChanges` function to detect which entities have changed since the last generation. This reduces generation time from seconds to milliseconds (measured: 4720ms to 473ms, 10x improvement).
6
+
7
+ ### 6.2 Components
8
+
9
+ **ChangeDetector** (`src/metadata/change-detector.ts`):
10
+ - `getInitialVersionStamp()` - First run: fetches the initial ServerVersionStamp
11
+ - `detectChanges(clientVersionStamp)` - Subsequent runs: returns changedEntityNames, deletedEntityNames, newVersionStamp
12
+
13
+ **MetadataCache** (`src/metadata/cache.ts`):
14
+ - Filesystem-based: `.xrmforge/cache/metadata.json`
15
+ - Stores: manifest (version, environment URL, ServerVersionStamp, last refreshed, entity list) + entityTypeInfos per entity
16
+ - Validation: checks cache version, environment URL match, file existence
17
+
18
+ ### 6.3 Flow
19
+
20
+ ```
21
+ First run (no cache):
22
+ 1. Fetch all entity metadata
23
+ 2. getInitialVersionStamp()
24
+ 3. Save cache with ServerVersionStamp
25
+
26
+ Subsequent run (cache exists):
27
+ 1. Load cache, validate environment URL
28
+ 2. detectChanges(cachedVersionStamp)
29
+ 3. Fetch only changed entities
30
+ 4. Remove deleted entities from cache
31
+ 5. Save cache with new ServerVersionStamp
32
+
33
+ Expired stamp (>90 days):
34
+ Error code 0x80044352 detected
35
+ Fall back to full refresh
36
+ ```
37
+
38
+ ### 6.4 RetrieveMetadataChanges API
39
+
40
+ - **Type:** OData Function (GET, not POST)
41
+ - **URL:** `/RetrieveMetadataChanges(Query=@q,ClientVersionStamp=@s)?@q={...}&@s='...'`
42
+ - **Response:** EntityMetadata[] with HasChanged flag, ServerVersionStamp, DeletedMetadata
@@ -0,0 +1,59 @@
1
+ # HTTP Client
2
+
3
+ ### 7.1 DataverseHttpClient
4
+
5
+ The core HTTP client (`src/http/client.ts`) provides resilient communication with the Dataverse Web API.
6
+
7
+ **Key methods:**
8
+ - `get<T>(path, signal?)` - Single GET request with retry
9
+ - `getAll<T>(path, signal?)` - GET with automatic @odata.nextLink paging (max 100 pages)
10
+
11
+ ### 7.2 Read-Only Default
12
+
13
+ The client defaults to `readOnly: true`, blocking POST/PATCH/PUT/DELETE requests. This prevents accidental data mutations during type generation. Write access requires explicit `readOnly: false`.
14
+
15
+ ### 7.3 Retry with Exponential Backoff
16
+
17
+ - **Base delay:** 1000ms (configurable)
18
+ - **Max backoff:** 60 seconds
19
+ - **Jitter:** Random delay up to base delay
20
+ - **Formula:** `min(baseDelay * 2^(attempt-1) + jitter, 60000)`
21
+ - **Max retries:** configurable (default: 3)
22
+
23
+ ### 7.4 Rate Limiting (HTTP 429)
24
+
25
+ - **Separate counter** from standard retries (not mixed)
26
+ - **Retry-After header** respected (seconds converted to milliseconds)
27
+ - **Max 10 consecutive 429 retries** (DEFAULT_MAX_RATE_LIMIT_RETRIES)
28
+ - 429 responses do NOT increment the standard retry counter
29
+
30
+ ### 7.5 Concurrency Control
31
+
32
+ Non-recursive semaphore pattern:
33
+ - **maxConcurrency:** 5 (default)
34
+ - Wait queue with FIFO ordering
35
+ - All retries happen inside a single slot (prevents slot exhaustion)
36
+
37
+ ### 7.6 Token Caching
38
+
39
+ - In-memory only (never persisted to disk)
40
+ - 5-minute buffer before expiry (TOKEN_BUFFER_MS = 300000)
41
+ - Pending token refresh promise prevents concurrent token requests
42
+
43
+ ### 7.7 Input Sanitization
44
+
45
+ OData injection prevention:
46
+ - `sanitizeIdentifier()` - Regex `[a-zA-Z_][a-zA-Z0-9_]*`
47
+ - `sanitizeGuid()` - GUID format validation
48
+ - `escapeODataString()` - Single quote doubling
49
+
50
+ ### 7.8 Error Handling
51
+
52
+ | HTTP Status | Behavior | Retried |
53
+ |-------------|----------|---------|
54
+ | 2xx | Success | No |
55
+ | 401 | Clear token cache, retry once | Yes (1x) |
56
+ | 429 | Respect Retry-After, separate counter | Yes (up to 10x) |
57
+ | 5xx | Exponential backoff | Yes (up to maxRetries) |
58
+ | 404, 403 | Non-retryable | No |
59
+ | Network error | Exponential backoff | Yes |
@@ -0,0 +1,18 @@
1
+ # Authentication
2
+
3
+ ### 8.1 Credential Factory
4
+
5
+ `createCredential(config: AuthConfig)` returns a `TokenCredential` (from @azure/identity) based on the auth method:
6
+
7
+ ### 8.2 Four Auth Flows
8
+
9
+ | Method | Config | @azure/identity Class | Use Case |
10
+ |--------|--------|-----------------------|----------|
11
+ | client-credentials | tenantId, clientId, clientSecret | ClientSecretCredential | CI/CD, automated pipelines |
12
+ | interactive | tenantId, clientId? | InteractiveBrowserCredential | Developer workstation |
13
+ | device-code | tenantId, clientId? | DeviceCodeCredential | Headless CLI environments |
14
+ | token | token (string) | StaticTokenCredential | Pre-acquired tokens (e.g. from TokenVault) |
15
+
16
+ ### 8.3 Token Scope
17
+
18
+ All auth flows request the scope: `{environmentUrl}/.default`
@@ -0,0 +1,55 @@
1
+ # Testing Framework
2
+
3
+ ### 9.1 createFormMock
4
+
5
+ ```typescript
6
+ import { createFormMock } from '@xrmforge/testing';
7
+ import type { AccountMainForm, AccountMainFormMockValues } from '../typings/forms/account';
8
+
9
+ const mock = createFormMock<AccountMainForm, AccountMainFormMockValues>({
10
+ name: 'Contoso Ltd',
11
+ statuscode: 0,
12
+ revenue: 1000000,
13
+ });
14
+
15
+ // Use in tests:
16
+ onLoad(mock.executionContext);
17
+ expect(mock.formContext.getControl('revenue').getVisible()).toBe(true);
18
+ ```
19
+
20
+ **What it mocks:**
21
+ - Attributes: MockAttribute instances with getValue/setValue, dirty tracking, onChange handlers, required level, submit mode
22
+ - Controls: MockControl instances with visible/disabled/label/notification state
23
+ - UI: Form notifications, tab/section stubs
24
+ - Entity: ID, entity name, primary attribute
25
+ - Data: refresh(), save() stubs returning Promise-like
26
+ - Navigation: openForm/openAlertDialog stubs
27
+
28
+ **Lazy initialization:** Attributes accessed via `getAttribute()` that were not in the initial values are created on-the-fly with null value.
29
+
30
+ ### 9.2 fireOnChange
31
+
32
+ ```typescript
33
+ mock.fireOnChange('statuscode');
34
+ // Triggers all handlers registered via getAttribute('statuscode').addOnChange(handler)
35
+ ```
36
+
37
+ Creates a MockEventContext with the attribute as event source.
38
+
39
+ ### 9.3 setupXrmMock / teardownXrmMock
40
+
41
+ ```typescript
42
+ import { setupXrmMock, teardownXrmMock } from '@xrmforge/testing';
43
+
44
+ beforeEach(() => setupXrmMock());
45
+ afterEach(() => teardownXrmMock());
46
+
47
+ // With WebApi overrides:
48
+ setupXrmMock({
49
+ webApiOverrides: {
50
+ retrieveMultipleRecords: async () => ({ entities: [{ name: 'Test' }] }),
51
+ },
52
+ });
53
+ ```
54
+
55
+ Sets up a global `Xrm` object on `globalThis` with minimal WebApi, Navigation, and Utility stubs.