expand-my-type 0.1.0 → 0.3.0
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/README.md +93 -7
- package/dist/cli.js +213 -0
- package/dist/index.cjs +10 -11
- package/dist/index.js +10 -11
- package/package.json +9 -3
package/README.md
CHANGED
|
@@ -23,23 +23,76 @@ type ExpandedType = Compute<SomeType>;
|
|
|
23
23
|
|
|
24
24
|
This is mostly done to improve the readability of type hints shown in editors/IDEs.
|
|
25
25
|
|
|
26
|
-
**Expand My Type** provides a programmatic way to do the same
|
|
27
|
-
source code as input, along with
|
|
28
|
-
form as a string. That can be useful
|
|
29
|
-
complex type errors.
|
|
26
|
+
**Expand My Type** provides a programmatic way to do the same
|
|
27
|
+
(see [limitations](#limitations)). It gets the source code as input, along with
|
|
28
|
+
a type expression and returns the expanded form as a string. That can be useful
|
|
29
|
+
for code-generation, testing, and debugging complex type errors.
|
|
30
30
|
|
|
31
31
|
Under the hood, it uses the [TypeScript Compiler API][ts-compiler-api] to expand
|
|
32
32
|
the type expression and optionally formats the output using
|
|
33
33
|
[Prettier][prettier].
|
|
34
34
|
|
|
35
|
-
#
|
|
35
|
+
# CLI
|
|
36
|
+
|
|
37
|
+
### Installation
|
|
38
|
+
|
|
39
|
+
To install the CLI globally, run:
|
|
40
|
+
|
|
41
|
+
```sh
|
|
42
|
+
npm install -g expand-my-type
|
|
43
|
+
```
|
|
44
|
+
|
|
45
|
+
Alternatively, the CLI can be run using `npx`:
|
|
46
|
+
|
|
47
|
+
```sh
|
|
48
|
+
npx expand-my-type
|
|
49
|
+
```
|
|
50
|
+
|
|
51
|
+
### CLI Usage
|
|
52
|
+
|
|
53
|
+
```
|
|
54
|
+
Usage:
|
|
55
|
+
expand-my-type [options] <source-file> <expression>
|
|
56
|
+
|
|
57
|
+
Options:
|
|
58
|
+
-h, --help Show this help message
|
|
59
|
+
-p, --prettify Prettify the output (default)
|
|
60
|
+
-P, --no-prettify Do not prettify the output
|
|
61
|
+
-c, --tsconfig <file> Use the specified tsconfig.json file
|
|
62
|
+
```
|
|
63
|
+
|
|
64
|
+
Given a file named `example.ts` with the following contents:
|
|
65
|
+
|
|
66
|
+
```typescript
|
|
67
|
+
interface SomeInterface {
|
|
68
|
+
a: number;
|
|
69
|
+
b?: string;
|
|
70
|
+
c: number | undefined;
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
type SomeType = {
|
|
74
|
+
d: number;
|
|
75
|
+
e: SomeInterface;
|
|
76
|
+
f: () => void;
|
|
77
|
+
};
|
|
78
|
+
```
|
|
79
|
+
|
|
80
|
+
Running the following command expands the type expression `SomeType["e"]`:
|
|
81
|
+
|
|
82
|
+
```sh
|
|
83
|
+
expand-my-type example.ts 'SomeType["e"]'
|
|
84
|
+
# Output:
|
|
85
|
+
# { a: number; b?: string; c: number }
|
|
86
|
+
```
|
|
87
|
+
|
|
88
|
+
# API
|
|
36
89
|
|
|
37
90
|
This package exports a function named `expandMyType` that can be used in one of
|
|
38
91
|
the following ways:
|
|
39
92
|
|
|
40
93
|
1. Expand the type expression in a source file:
|
|
41
94
|
|
|
42
|
-
Given a file named `example.ts` with the following
|
|
95
|
+
Given a file named `example.ts` with the following contents:
|
|
43
96
|
|
|
44
97
|
```typescript
|
|
45
98
|
interface SomeInterface {
|
|
@@ -69,7 +122,7 @@ the following ways:
|
|
|
69
122
|
/* {
|
|
70
123
|
d: number
|
|
71
124
|
e: { a: number; b?: string; c: number }
|
|
72
|
-
f:
|
|
125
|
+
f: () => void
|
|
73
126
|
} */
|
|
74
127
|
```
|
|
75
128
|
|
|
@@ -95,6 +148,39 @@ the following ways:
|
|
|
95
148
|
/* { a: string; b: number } */
|
|
96
149
|
```
|
|
97
150
|
|
|
151
|
+
# Limitations
|
|
152
|
+
|
|
153
|
+
Expand My Type uses the following utility type to expand the type expression:
|
|
154
|
+
|
|
155
|
+
```typescript
|
|
156
|
+
type Expand<T> = T extends (...args: infer A) => infer R
|
|
157
|
+
? (...args: Expand<A>) => Expand<R>
|
|
158
|
+
: { [K in keyof T]: Expand<T[K]> } & {};
|
|
159
|
+
```
|
|
160
|
+
|
|
161
|
+
As a result, the output it generates is what the TypeScript compiler would
|
|
162
|
+
infer from wrapping the type expression with the `Expand` utility type. That
|
|
163
|
+
comes with some limitations. If you know a better approach to expand types, feel
|
|
164
|
+
free to open an issue or a pull request.
|
|
165
|
+
|
|
166
|
+
## Type References to Union String Literal Types
|
|
167
|
+
|
|
168
|
+
Type references to union string literal types are not expanded. For example,
|
|
169
|
+
expanding T in the following code:
|
|
170
|
+
|
|
171
|
+
```typescript
|
|
172
|
+
type T = { s: S };
|
|
173
|
+
type S = "a" | "b";
|
|
174
|
+
```
|
|
175
|
+
|
|
176
|
+
results in `{ s: S }` instead of `{ s: "a" | "b" }`.
|
|
177
|
+
|
|
178
|
+
## Recursive Type References
|
|
179
|
+
|
|
180
|
+
Expanding recursive type references can lead to infinite loops. When an infinite
|
|
181
|
+
loop happens, the generated output gets truncated. Prettifying the truncated
|
|
182
|
+
output will throw an error.
|
|
183
|
+
|
|
98
184
|
[compute-type]: https://github.com/microsoft/TypeScript/blob/main/tests/cases/compiler/computedTypesKeyofNoIndexSignatureType.ts
|
|
99
185
|
[prettier]: https://prettier.io
|
|
100
186
|
[ts-compiler-api]: https://github.com/microsoft/TypeScript/wiki/Using-the-Compiler-API
|
package/dist/cli.js
ADDED
|
@@ -0,0 +1,213 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
|
|
3
|
+
// src/create-expander-compiler-host.ts
|
|
4
|
+
import ts from "typescript";
|
|
5
|
+
var identifierPrefix = "__TYPE_EXPANDER__";
|
|
6
|
+
var createExpanderCompilerHost = (sourceFileName2, typeExpression2, compilerOptions, getSourceFileFunction) => {
|
|
7
|
+
const customCompilerHost = ts.createCompilerHost(compilerOptions ?? {});
|
|
8
|
+
const originalGetSourceFile = customCompilerHost.getSourceFile;
|
|
9
|
+
customCompilerHost.getSourceFile = (fileName, ...args) => {
|
|
10
|
+
if (fileName === sourceFileName2) {
|
|
11
|
+
const sourceFile = (getSourceFileFunction ?? originalGetSourceFile)(
|
|
12
|
+
fileName,
|
|
13
|
+
...args
|
|
14
|
+
);
|
|
15
|
+
if (sourceFile === void 0) {
|
|
16
|
+
return void 0;
|
|
17
|
+
}
|
|
18
|
+
const sourceText = `type ${identifierPrefix}Result = ${identifierPrefix}Expand<${identifierPrefix}Expression>;
|
|
19
|
+
type ${identifierPrefix}Expression = ${typeExpression2};
|
|
20
|
+
|
|
21
|
+
type ${identifierPrefix}Expand<T> = T extends (...args: infer A) => infer R
|
|
22
|
+
? (...args: ${identifierPrefix}Expand<A>) => ${identifierPrefix}Expand<R>
|
|
23
|
+
: { [K in keyof T]: ${identifierPrefix}Expand<T[K]>; } & {};
|
|
24
|
+
|
|
25
|
+
${sourceFile.getFullText()}`;
|
|
26
|
+
return ts.createSourceFile(
|
|
27
|
+
fileName,
|
|
28
|
+
sourceText,
|
|
29
|
+
sourceFile.languageVersion,
|
|
30
|
+
true
|
|
31
|
+
);
|
|
32
|
+
}
|
|
33
|
+
return originalGetSourceFile(fileName, ...args);
|
|
34
|
+
};
|
|
35
|
+
return customCompilerHost;
|
|
36
|
+
};
|
|
37
|
+
|
|
38
|
+
// src/index.ts
|
|
39
|
+
import { format } from "prettier";
|
|
40
|
+
import ts2 from "typescript";
|
|
41
|
+
var findTypeExpanderResultNode = (node) => {
|
|
42
|
+
if (node.getChildCount() == 0) {
|
|
43
|
+
if (!ts2.isIdentifier(node)) {
|
|
44
|
+
return void 0;
|
|
45
|
+
}
|
|
46
|
+
return node;
|
|
47
|
+
}
|
|
48
|
+
return ts2.forEachChild(node, findTypeExpanderResultNode);
|
|
49
|
+
};
|
|
50
|
+
async function expandMyType(options) {
|
|
51
|
+
if (options.typeExpression.trim() === "") {
|
|
52
|
+
return "never";
|
|
53
|
+
}
|
|
54
|
+
if ("sourceText" in options) {
|
|
55
|
+
const sourceFile2 = ts2.createSourceFile(
|
|
56
|
+
"test.ts",
|
|
57
|
+
options.sourceText,
|
|
58
|
+
ts2.ScriptTarget.Latest,
|
|
59
|
+
true
|
|
60
|
+
);
|
|
61
|
+
return expandMyType({
|
|
62
|
+
sourceFileName: "test.ts",
|
|
63
|
+
typeExpression: options.typeExpression,
|
|
64
|
+
getSourceFileFunction: () => sourceFile2,
|
|
65
|
+
tsCompilerOptions: options.tsCompilerOptions
|
|
66
|
+
});
|
|
67
|
+
}
|
|
68
|
+
const tsCompilerOptions = options.tsCompilerOptions ?? {
|
|
69
|
+
noEmit: true,
|
|
70
|
+
allowSyntheticDefaultImports: true,
|
|
71
|
+
allowArbitraryExtensions: true,
|
|
72
|
+
allowImportingTsExtensions: true,
|
|
73
|
+
allowJs: true
|
|
74
|
+
};
|
|
75
|
+
const compilerHost = createExpanderCompilerHost(
|
|
76
|
+
options.sourceFileName,
|
|
77
|
+
options.typeExpression,
|
|
78
|
+
tsCompilerOptions,
|
|
79
|
+
options.getSourceFileFunction
|
|
80
|
+
);
|
|
81
|
+
const program = ts2.createProgram(
|
|
82
|
+
[options.sourceFileName],
|
|
83
|
+
tsCompilerOptions,
|
|
84
|
+
compilerHost
|
|
85
|
+
);
|
|
86
|
+
const sourceFile = program.getSourceFile(options.sourceFileName);
|
|
87
|
+
if (!sourceFile) {
|
|
88
|
+
throw new Error("Source file not found!");
|
|
89
|
+
}
|
|
90
|
+
const firstIdentifier = findTypeExpanderResultNode(sourceFile);
|
|
91
|
+
if (!firstIdentifier) {
|
|
92
|
+
throw new Error("No node found!");
|
|
93
|
+
}
|
|
94
|
+
const typeChecker = program.getTypeChecker();
|
|
95
|
+
const expandedType = typeChecker.typeToString(
|
|
96
|
+
typeChecker.getTypeAtLocation(firstIdentifier),
|
|
97
|
+
void 0,
|
|
98
|
+
ts2.TypeFormatFlags.NodeBuilderFlagsMask
|
|
99
|
+
);
|
|
100
|
+
if (options.prettify && options.prettify.enabled === false) {
|
|
101
|
+
return expandedType;
|
|
102
|
+
}
|
|
103
|
+
const dummyTypeName = "__TYPE_EXPANDER_RESULT__";
|
|
104
|
+
return (await format(
|
|
105
|
+
`type ${dummyTypeName} = ${expandedType}`,
|
|
106
|
+
options.prettify?.options ?? {
|
|
107
|
+
parser: "typescript",
|
|
108
|
+
semi: false
|
|
109
|
+
}
|
|
110
|
+
)).trim().substring(`type ${dummyTypeName} = `.length);
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
// src/cli.ts
|
|
114
|
+
import path from "node:path";
|
|
115
|
+
import { parseArgs } from "node:util";
|
|
116
|
+
import ts3 from "typescript";
|
|
117
|
+
var tryParse = (options) => {
|
|
118
|
+
let result2;
|
|
119
|
+
try {
|
|
120
|
+
result2 = parseArgs({
|
|
121
|
+
options,
|
|
122
|
+
tokens: true,
|
|
123
|
+
allowPositionals: true
|
|
124
|
+
});
|
|
125
|
+
} catch (error2) {
|
|
126
|
+
return {
|
|
127
|
+
error: error2,
|
|
128
|
+
values: {},
|
|
129
|
+
positionals: []
|
|
130
|
+
};
|
|
131
|
+
}
|
|
132
|
+
const { tokens, values: values2, positionals: positionals2 } = result2;
|
|
133
|
+
for (const token of tokens ?? []) {
|
|
134
|
+
if (token.kind === "option-terminator" || token.kind === "positional") {
|
|
135
|
+
continue;
|
|
136
|
+
}
|
|
137
|
+
if (token.name.startsWith("no-")) {
|
|
138
|
+
const positiveName = token.name.slice("no-".length);
|
|
139
|
+
values2[positiveName] = false;
|
|
140
|
+
delete values2[token.name];
|
|
141
|
+
} else {
|
|
142
|
+
values2[token.name] = token.value ?? true;
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
return {
|
|
146
|
+
error: void 0,
|
|
147
|
+
values: values2,
|
|
148
|
+
positionals: positionals2
|
|
149
|
+
};
|
|
150
|
+
};
|
|
151
|
+
var { error, values, positionals } = tryParse({
|
|
152
|
+
help: {
|
|
153
|
+
type: "boolean",
|
|
154
|
+
short: "h"
|
|
155
|
+
},
|
|
156
|
+
prettify: {
|
|
157
|
+
type: "boolean",
|
|
158
|
+
short: "p",
|
|
159
|
+
default: true
|
|
160
|
+
},
|
|
161
|
+
"no-prettify": {
|
|
162
|
+
type: "boolean",
|
|
163
|
+
short: "P"
|
|
164
|
+
},
|
|
165
|
+
tsconfig: {
|
|
166
|
+
type: "string",
|
|
167
|
+
short: "c"
|
|
168
|
+
}
|
|
169
|
+
});
|
|
170
|
+
if (error) {
|
|
171
|
+
console.error(error.message);
|
|
172
|
+
process.exit(1);
|
|
173
|
+
}
|
|
174
|
+
var usagePrompt = [
|
|
175
|
+
"Usage:",
|
|
176
|
+
" expand-my-type [options] <source-file> <expression>",
|
|
177
|
+
"",
|
|
178
|
+
"Options:",
|
|
179
|
+
" -h, --help Show this help message",
|
|
180
|
+
" -p, --prettify Prettify the output (default)",
|
|
181
|
+
" -P, --no-prettify Do not prettify the output",
|
|
182
|
+
" -c, --tsconfig <file> Use the specified tsconfig.json file"
|
|
183
|
+
].join("\n");
|
|
184
|
+
if (positionals.length !== 2) {
|
|
185
|
+
console.error(usagePrompt);
|
|
186
|
+
process.exit(1);
|
|
187
|
+
}
|
|
188
|
+
if (values.help) {
|
|
189
|
+
console.error(usagePrompt);
|
|
190
|
+
process.exit(0);
|
|
191
|
+
}
|
|
192
|
+
var [sourceFileName, typeExpression] = positionals;
|
|
193
|
+
var { prettify, tsconfig: tsConfigFileName } = values;
|
|
194
|
+
var tsParsedCommandLine;
|
|
195
|
+
if (tsConfigFileName) {
|
|
196
|
+
const configFile = ts3.readConfigFile(tsConfigFileName, ts3.sys.readFile);
|
|
197
|
+
const compilerOptions = ts3.parseJsonConfigFileContent(
|
|
198
|
+
configFile.config,
|
|
199
|
+
ts3.sys,
|
|
200
|
+
"./"
|
|
201
|
+
);
|
|
202
|
+
tsParsedCommandLine = compilerOptions;
|
|
203
|
+
}
|
|
204
|
+
var result = await expandMyType({
|
|
205
|
+
sourceFileName: path.resolve(sourceFileName),
|
|
206
|
+
typeExpression,
|
|
207
|
+
prettify: {
|
|
208
|
+
enabled: prettify
|
|
209
|
+
},
|
|
210
|
+
tsCompilerOptions: tsParsedCommandLine?.options
|
|
211
|
+
});
|
|
212
|
+
console.log(result);
|
|
213
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/create-expander-compiler-host.ts", "../src/index.ts", "../src/cli.ts"],
  "sourcesContent": ["import ts from \"typescript\";\n\nconst identifierPrefix = \"__TYPE_EXPANDER__\";\n\n/**\n * Creates a custom compiler host that augments the specified source file for expanding a type expression.\n *\n * @param sourceFileName Name of the source file to augment.\n * @param typeExpression Type expression.\n * @param compilerOptions TypeScript compiler options.\n * @param getSourceFileFunction The implementation of the `ts.CompilerHost[\"getSourceFile\"]` function.\n * @returns A custom compiler host that returns an augmented source file that can be used to expand the type expression.\n */\nexport const createExpanderCompilerHost = (\n  sourceFileName: string,\n  typeExpression: string,\n  compilerOptions?: ts.CompilerOptions,\n  getSourceFileFunction?: ts.CompilerHost[\"getSourceFile\"],\n) => {\n  const customCompilerHost = ts.createCompilerHost(compilerOptions ?? {});\n  const originalGetSourceFile = customCompilerHost.getSourceFile;\n\n  customCompilerHost.getSourceFile = (fileName, ...args) => {\n    if (fileName === sourceFileName) {\n      const sourceFile = (getSourceFileFunction ?? originalGetSourceFile)(\n        fileName,\n        ...args,\n      );\n\n      if (sourceFile === undefined) {\n        return undefined;\n      }\n\n      // https://github.com/microsoft/TypeScript/blob/main/tests/cases/compiler/computedTypesKeyofNoIndexSignatureType.ts\n      const sourceText = `type ${identifierPrefix}Result = ${identifierPrefix}Expand<${identifierPrefix}Expression>;\n        type ${identifierPrefix}Expression = ${typeExpression};\n\n        type ${identifierPrefix}Expand<T> = T extends (...args: infer A) => infer R\n          ? (...args: ${identifierPrefix}Expand<A>) => ${identifierPrefix}Expand<R>\n          : { [K in keyof T]: ${identifierPrefix}Expand<T[K]>; } & {};\n          \n        ${sourceFile.getFullText()}`;\n\n      return ts.createSourceFile(\n        fileName,\n        sourceText,\n        sourceFile.languageVersion,\n        true,\n      );\n    }\n\n    return originalGetSourceFile(fileName, ...args);\n  };\n\n  return customCompilerHost;\n};\n", "import { createExpanderCompilerHost } from \"./create-expander-compiler-host.ts\";\nimport { format, type Options as PrettierOptions } from \"prettier\";\nimport ts from \"typescript\";\n\n/**\n * Finds the node that represents the TYPE_EXPANDER_RESULT type.\n *\n * @param node Node in which the TYPE_EXPANDER_RESULT type should be searched.\n * @returns The node that represents the TYPE_EXPANDER_RESULT type.\n */\nconst findTypeExpanderResultNode = (node: ts.Node): ts.Node | undefined => {\n  if (node.getChildCount() == 0) {\n    if (!ts.isIdentifier(node)) {\n      return undefined;\n    }\n\n    // Since we put the __TYPE_EXPANDER_RESULT__ type at the beginning of the\n    // file, we can return the first identifier we find.\n    return node;\n  }\n\n  return ts.forEachChild(node, findTypeExpanderResultNode);\n};\n\nexport type ExpandTypeOptionsBase = {\n  /**\n   * The type expression to expand.\n   * @example \"ReturnType<typeof myFunction>\"\n   */\n  typeExpression: string;\n\n  /**\n   * TypeScript compiler options.\n   */\n  tsCompilerOptions?: ts.CompilerOptions;\n\n  /**\n   * Prettier options.\n   */\n  prettify?: {\n    /**\n     * Whether to prettify the output.\n     * @default true\n     */\n    enabled?: boolean;\n    /**\n     * Prettier options. Don't forget to set the parser to \"typescript\".\n     * @default { parser: \"typescript\", semi: false }\n     */\n    options?: PrettierOptions;\n  };\n};\nexport type ExpandTypeFromSourceFileOptions = ExpandTypeOptionsBase & {\n  /**\n   * Name of the source file to evaluate the type expression in.\n   */\n  sourceFileName: string;\n  /**\n   * Function that returns the source file. When not specified, the implementation of the default TypeScript compiler host is used.\n   */\n  getSourceFileFunction?: ts.CompilerHost[\"getSourceFile\"];\n};\nexport type ExpandTypeFromSourceTextOptions = ExpandTypeOptionsBase & {\n  /**\n   * TypeScript source text to evaluate the type expression in.\n   */\n  sourceText: string;\n};\nexport type ExpandMyTypeOptions =\n  | ExpandTypeFromSourceFileOptions\n  | ExpandTypeFromSourceTextOptions;\n\nexport async function expandMyType(\n  options: ExpandTypeFromSourceTextOptions,\n): Promise<string>;\nexport async function expandMyType(\n  options: ExpandTypeFromSourceFileOptions,\n): Promise<string>;\n\n/**\n * Expands a TypeScript type expression.\n *\n * @param options\n * @returns The expanded type expression.\n */\nexport async function expandMyType(options: ExpandMyTypeOptions) {\n  if (options.typeExpression.trim() === \"\") {\n    return \"never\";\n  }\n\n  if (\"sourceText\" in options) {\n    const sourceFile = ts.createSourceFile(\n      \"test.ts\",\n      options.sourceText,\n      ts.ScriptTarget.Latest,\n      true,\n    );\n\n    return expandMyType({\n      sourceFileName: \"test.ts\",\n      typeExpression: options.typeExpression,\n      getSourceFileFunction: () => sourceFile,\n      tsCompilerOptions: options.tsCompilerOptions,\n    });\n  }\n\n  const tsCompilerOptions = options.tsCompilerOptions ?? {\n    noEmit: true,\n\n    allowSyntheticDefaultImports: true,\n    allowArbitraryExtensions: true,\n    allowImportingTsExtensions: true,\n    allowJs: true,\n  };\n\n  const compilerHost = createExpanderCompilerHost(\n    options.sourceFileName,\n    options.typeExpression,\n    tsCompilerOptions,\n    options.getSourceFileFunction,\n  );\n\n  const program = ts.createProgram(\n    [options.sourceFileName],\n    tsCompilerOptions,\n    compilerHost,\n  );\n\n  const sourceFile = program.getSourceFile(options.sourceFileName);\n  if (!sourceFile) {\n    throw new Error(\"Source file not found!\");\n  }\n\n  const firstIdentifier = findTypeExpanderResultNode(sourceFile);\n  if (!firstIdentifier) {\n    throw new Error(\"No node found!\");\n  }\n\n  const typeChecker = program.getTypeChecker();\n  const expandedType = typeChecker.typeToString(\n    typeChecker.getTypeAtLocation(firstIdentifier),\n    undefined,\n    ts.TypeFormatFlags.NodeBuilderFlagsMask,\n  );\n\n  if (options.prettify && options.prettify.enabled === false) {\n    return expandedType;\n  }\n\n  const dummyTypeName = \"__TYPE_EXPANDER_RESULT__\";\n\n  return (\n    await format(\n      `type ${dummyTypeName} = ${expandedType}`,\n      options.prettify?.options ?? {\n        parser: \"typescript\",\n        semi: false,\n      },\n    )\n  )\n    .trim()\n    .substring(`type ${dummyTypeName} = `.length);\n}\n", "#!/usr/bin/env node\nimport { expandMyType } from \"./index.ts\";\nimport path from \"node:path\";\nimport { parseArgs, type ParseArgsConfig } from \"node:util\";\nimport ts from \"typescript\";\n\ntype Values = {\n  help?: boolean;\n  prettify: boolean;\n  tsconfig?: string;\n};\n\nconst tryParse = <\n  Values extends Record<\n    string,\n    string | boolean | Array<string | boolean> | undefined\n  >,\n  Options extends ParseArgsConfig[\"options\"] = ParseArgsConfig[\"options\"],\n>(\n  options: Options,\n): { error?: Error; values: Values; positionals: string[] } => {\n  let result: ReturnType<typeof parseArgs>;\n  try {\n    result = parseArgs({\n      options,\n      tokens: true,\n      allowPositionals: true,\n    });\n  } catch (error) {\n    return {\n      error,\n      values: {} as Values,\n      positionals: [],\n    };\n  }\n\n  const { tokens, values, positionals } = result;\n\n  // Reprocess the option tokens and overwrite the returned values.\n  for (const token of tokens ?? []) {\n    if (token.kind === \"option-terminator\" || token.kind === \"positional\") {\n      continue;\n    }\n\n    if (token.name.startsWith(\"no-\")) {\n      // Store foo:false for --no-foo\n      const positiveName = token.name.slice(\"no-\".length);\n      values[positiveName] = false;\n      delete values[token.name];\n    } else {\n      // Resave value so last one wins if both --foo and --no-foo.\n      values[token.name] = token.value ?? true;\n    }\n  }\n\n  return {\n    error: undefined,\n    values: values as Values,\n    positionals,\n  };\n};\n\nconst { error, values, positionals } = tryParse<Values>({\n  help: {\n    type: \"boolean\",\n    short: \"h\",\n  },\n  prettify: {\n    type: \"boolean\",\n    short: \"p\",\n    default: true,\n  },\n  \"no-prettify\": {\n    type: \"boolean\",\n    short: \"P\",\n  },\n  tsconfig: {\n    type: \"string\",\n    short: \"c\",\n  },\n});\n\nif (error) {\n  console.error(error.message);\n  process.exit(1);\n}\n\nconst usagePrompt = [\n  \"Usage:\",\n  \"  expand-my-type [options] <source-file> <expression>\",\n  \"\",\n  \"Options:\",\n  \"  -h, --help\\t\\t\\tShow this help message\",\n  \"  -p, --prettify\\t\\tPrettify the output (default)\",\n  \"  -P, --no-prettify\\t\\tDo not prettify the output\",\n  \"  -c, --tsconfig <file>\\t\\tUse the specified tsconfig.json file\",\n].join(\"\\n\");\n\nif (positionals.length !== 2) {\n  console.error(usagePrompt);\n  process.exit(1);\n}\n\nif (values.help) {\n  console.error(usagePrompt);\n  process.exit(0);\n}\n\nconst [sourceFileName, typeExpression] = positionals;\nconst { prettify, tsconfig: tsConfigFileName } = values;\n\nlet tsParsedCommandLine: ts.ParsedCommandLine | undefined;\n\nif (tsConfigFileName) {\n  const configFile = ts.readConfigFile(tsConfigFileName, ts.sys.readFile);\n  const compilerOptions = ts.parseJsonConfigFileContent(\n    configFile.config,\n    ts.sys,\n    \"./\",\n  );\n\n  tsParsedCommandLine = compilerOptions;\n}\n\nconst result = await expandMyType({\n  sourceFileName: path.resolve(sourceFileName),\n  typeExpression,\n  prettify: {\n    enabled: prettify,\n  },\n  tsCompilerOptions: tsParsedCommandLine?.options,\n});\n\nconsole.log(result);\n"],
  "mappings": "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",
  "names": ["sourceFileName", "typeExpression", "ts", "sourceFile", "ts", "result", "error", "values", "positionals"]
}

|
package/dist/index.cjs
CHANGED
|
@@ -35,6 +35,7 @@ module.exports = __toCommonJS(src_exports);
|
|
|
35
35
|
|
|
36
36
|
// src/create-expander-compiler-host.ts
|
|
37
37
|
var import_typescript = __toESM(require("typescript"), 1);
|
|
38
|
+
var identifierPrefix = "__TYPE_EXPANDER__";
|
|
38
39
|
var createExpanderCompilerHost = (sourceFileName, typeExpression, compilerOptions, getSourceFileFunction) => {
|
|
39
40
|
const customCompilerHost = import_typescript.default.createCompilerHost(compilerOptions ?? {});
|
|
40
41
|
const originalGetSourceFile = customCompilerHost.getSourceFile;
|
|
@@ -47,16 +48,14 @@ var createExpanderCompilerHost = (sourceFileName, typeExpression, compilerOption
|
|
|
47
48
|
if (sourceFile === void 0) {
|
|
48
49
|
return void 0;
|
|
49
50
|
}
|
|
50
|
-
const sourceText =
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
sourceFile.getFullText()
|
|
59
|
-
].join("\n");
|
|
51
|
+
const sourceText = `type ${identifierPrefix}Result = ${identifierPrefix}Expand<${identifierPrefix}Expression>;
|
|
52
|
+
type ${identifierPrefix}Expression = ${typeExpression};
|
|
53
|
+
|
|
54
|
+
type ${identifierPrefix}Expand<T> = T extends (...args: infer A) => infer R
|
|
55
|
+
? (...args: ${identifierPrefix}Expand<A>) => ${identifierPrefix}Expand<R>
|
|
56
|
+
: { [K in keyof T]: ${identifierPrefix}Expand<T[K]>; } & {};
|
|
57
|
+
|
|
58
|
+
${sourceFile.getFullText()}`;
|
|
60
59
|
return import_typescript.default.createSourceFile(
|
|
61
60
|
fileName,
|
|
62
61
|
sourceText,
|
|
@@ -147,4 +146,4 @@ async function expandMyType(options) {
|
|
|
147
146
|
0 && (module.exports = {
|
|
148
147
|
expandMyType
|
|
149
148
|
});
|
|
150
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/index.ts", "../src/create-expander-compiler-host.ts"],
  "sourcesContent": ["import { createExpanderCompilerHost } from \"./create-expander-compiler-host.ts\";\nimport { format, type Options as PrettierOptions } from \"prettier\";\nimport ts from \"typescript\";\n\n/**\n * Finds the node that represents the TYPE_EXPANDER_RESULT type.\n *\n * @param node Node in which the TYPE_EXPANDER_RESULT type should be searched.\n * @returns The node that represents the TYPE_EXPANDER_RESULT type.\n */\nconst findTypeExpanderResultNode = (node: ts.Node): ts.Node | undefined => {\n  if (node.getChildCount() == 0) {\n    if (!ts.isIdentifier(node)) {\n      return undefined;\n    }\n\n    // Since we put the __TYPE_EXPANDER_RESULT__ type at the beginning of the\n    // file, we can return the first identifier we find.\n    return node;\n  }\n\n  return ts.forEachChild(node, findTypeExpanderResultNode);\n};\n\nexport type ExpandTypeOptionsBase = {\n  /**\n   * The type expression to expand.\n   * @example \"ReturnType<typeof myFunction>\"\n   */\n  typeExpression: string;\n\n  /**\n   * TypeScript compiler options.\n   */\n  tsCompilerOptions?: ts.CompilerOptions;\n\n  /**\n   * Prettier options.\n   */\n  prettify?: {\n    /**\n     * Whether to prettify the output.\n     * @default true\n     */\n    enabled?: boolean;\n    /**\n     * Prettier options. Don't forget to set the parser to \"typescript\".\n     * @default { parser: \"typescript\", semi: false }\n     */\n    options?: PrettierOptions;\n  };\n};\nexport type ExpandTypeFromSourceFileOptions = ExpandTypeOptionsBase & {\n  /**\n   * Name of the source file to evaluate the type expression in.\n   */\n  sourceFileName: string;\n  /**\n   * Function that returns the source file. When not specified, the implementation of the default TypeScript compiler host is used.\n   */\n  getSourceFileFunction?: ts.CompilerHost[\"getSourceFile\"];\n};\nexport type ExpandTypeFromSourceTextOptions = ExpandTypeOptionsBase & {\n  /**\n   * TypeScript source text to evaluate the type expression in.\n   */\n  sourceText: string;\n};\nexport type ExpandMyTypeOptions =\n  | ExpandTypeFromSourceFileOptions\n  | ExpandTypeFromSourceTextOptions;\n\nexport async function expandMyType(\n  options: ExpandTypeFromSourceTextOptions,\n): Promise<string>;\nexport async function expandMyType(\n  options: ExpandTypeFromSourceFileOptions,\n): Promise<string>;\n\n/**\n * Expands a TypeScript type expression.\n *\n * @param options\n * @returns The expanded type expression.\n */\nexport async function expandMyType(options: ExpandMyTypeOptions) {\n  if (options.typeExpression.trim() === \"\") {\n    return \"never\";\n  }\n\n  if (\"sourceText\" in options) {\n    const sourceFile = ts.createSourceFile(\n      \"test.ts\",\n      options.sourceText,\n      ts.ScriptTarget.Latest,\n      true,\n    );\n\n    return expandMyType({\n      sourceFileName: \"test.ts\",\n      typeExpression: options.typeExpression,\n      getSourceFileFunction: () => sourceFile,\n      tsCompilerOptions: options.tsCompilerOptions,\n    });\n  }\n\n  const tsCompilerOptions = options.tsCompilerOptions ?? {\n    noEmit: true,\n\n    allowSyntheticDefaultImports: true,\n    allowArbitraryExtensions: true,\n    allowImportingTsExtensions: true,\n    allowJs: true,\n  };\n\n  const compilerHost = createExpanderCompilerHost(\n    options.sourceFileName,\n    options.typeExpression,\n    tsCompilerOptions,\n    options.getSourceFileFunction,\n  );\n\n  const program = ts.createProgram(\n    [options.sourceFileName],\n    tsCompilerOptions,\n    compilerHost,\n  );\n\n  const sourceFile = program.getSourceFile(options.sourceFileName);\n  if (!sourceFile) {\n    throw new Error(\"Source file not found!\");\n  }\n\n  const firstIdentifier = findTypeExpanderResultNode(sourceFile);\n  if (!firstIdentifier) {\n    throw new Error(\"No node found!\");\n  }\n\n  const typeChecker = program.getTypeChecker();\n  const expandedType = typeChecker.typeToString(\n    typeChecker.getTypeAtLocation(firstIdentifier),\n    undefined,\n    ts.TypeFormatFlags.NodeBuilderFlagsMask,\n  );\n\n  if (options.prettify && options.prettify.enabled === false) {\n    return expandedType;\n  }\n\n  const dummyTypeName = \"__TYPE_EXPANDER_RESULT__\";\n\n  return (\n    await format(\n      `type ${dummyTypeName} = ${expandedType}`,\n      options.prettify?.options ?? {\n        parser: \"typescript\",\n        semi: false,\n      },\n    )\n  )\n    .trim()\n    .substring(`type ${dummyTypeName} = `.length);\n}\n", "import ts from \"typescript\";\n\n/**\n * Creates a custom compiler host that augments the specified source file for expanding a type expression.\n *\n * @param sourceFileName Name of the source file to augment.\n * @param typeExpression Type expression.\n * @param compilerOptions TypeScript compiler options.\n * @param getSourceFileFunction The implementation of the `ts.CompilerHost[\"getSourceFile\"]` function.\n * @returns A custom compiler host that returns an augmented source file that can be used to expand the type expression.\n */\nexport const createExpanderCompilerHost = (\n  sourceFileName: string,\n  typeExpression: string,\n  compilerOptions?: ts.CompilerOptions,\n  getSourceFileFunction?: ts.CompilerHost[\"getSourceFile\"],\n) => {\n  const customCompilerHost = ts.createCompilerHost(compilerOptions ?? {});\n  const originalGetSourceFile = customCompilerHost.getSourceFile;\n\n  customCompilerHost.getSourceFile = (fileName, ...args) => {\n    if (fileName === sourceFileName) {\n      const sourceFile = (getSourceFileFunction ?? originalGetSourceFile)(\n        fileName,\n        ...args,\n      );\n\n      if (sourceFile === undefined) {\n        return undefined;\n      }\n\n      const sourceText = [\n        ...[\n          \"type __TYPE_EXPANDER_RESULT__ = __TYPE_EXPANDER_EXPAND__<__TYPE_EXPANDER_EXPRESSION__>;\",\n          `type __TYPE_EXPANDER_EXPRESSION__ = ${typeExpression};`,\n          \"type __TYPE_EXPANDER_EXPAND__<T> = {\",\n          \"  [K in keyof T]: __TYPE_EXPANDER_EXPAND__<T[K]>;\",\n          \"} & {};\",\n        ],\n        sourceFile.getFullText(),\n      ].join(\"\\n\");\n\n      return ts.createSourceFile(\n        fileName,\n        sourceText,\n        sourceFile.languageVersion,\n        true,\n      );\n    }\n\n    return originalGetSourceFile(fileName, ...args);\n  };\n\n  return customCompilerHost;\n};\n"],
  "mappings": "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",
  "names": ["ts", "import_typescript", "ts", "sourceFile"]
}

|
|
149
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/index.ts", "../src/create-expander-compiler-host.ts"],
  "sourcesContent": ["import { createExpanderCompilerHost } from \"./create-expander-compiler-host.ts\";\nimport { format, type Options as PrettierOptions } from \"prettier\";\nimport ts from \"typescript\";\n\n/**\n * Finds the node that represents the TYPE_EXPANDER_RESULT type.\n *\n * @param node Node in which the TYPE_EXPANDER_RESULT type should be searched.\n * @returns The node that represents the TYPE_EXPANDER_RESULT type.\n */\nconst findTypeExpanderResultNode = (node: ts.Node): ts.Node | undefined => {\n  if (node.getChildCount() == 0) {\n    if (!ts.isIdentifier(node)) {\n      return undefined;\n    }\n\n    // Since we put the __TYPE_EXPANDER_RESULT__ type at the beginning of the\n    // file, we can return the first identifier we find.\n    return node;\n  }\n\n  return ts.forEachChild(node, findTypeExpanderResultNode);\n};\n\nexport type ExpandTypeOptionsBase = {\n  /**\n   * The type expression to expand.\n   * @example \"ReturnType<typeof myFunction>\"\n   */\n  typeExpression: string;\n\n  /**\n   * TypeScript compiler options.\n   */\n  tsCompilerOptions?: ts.CompilerOptions;\n\n  /**\n   * Prettier options.\n   */\n  prettify?: {\n    /**\n     * Whether to prettify the output.\n     * @default true\n     */\n    enabled?: boolean;\n    /**\n     * Prettier options. Don't forget to set the parser to \"typescript\".\n     * @default { parser: \"typescript\", semi: false }\n     */\n    options?: PrettierOptions;\n  };\n};\nexport type ExpandTypeFromSourceFileOptions = ExpandTypeOptionsBase & {\n  /**\n   * Name of the source file to evaluate the type expression in.\n   */\n  sourceFileName: string;\n  /**\n   * Function that returns the source file. When not specified, the implementation of the default TypeScript compiler host is used.\n   */\n  getSourceFileFunction?: ts.CompilerHost[\"getSourceFile\"];\n};\nexport type ExpandTypeFromSourceTextOptions = ExpandTypeOptionsBase & {\n  /**\n   * TypeScript source text to evaluate the type expression in.\n   */\n  sourceText: string;\n};\nexport type ExpandMyTypeOptions =\n  | ExpandTypeFromSourceFileOptions\n  | ExpandTypeFromSourceTextOptions;\n\nexport async function expandMyType(\n  options: ExpandTypeFromSourceTextOptions,\n): Promise<string>;\nexport async function expandMyType(\n  options: ExpandTypeFromSourceFileOptions,\n): Promise<string>;\n\n/**\n * Expands a TypeScript type expression.\n *\n * @param options\n * @returns The expanded type expression.\n */\nexport async function expandMyType(options: ExpandMyTypeOptions) {\n  if (options.typeExpression.trim() === \"\") {\n    return \"never\";\n  }\n\n  if (\"sourceText\" in options) {\n    const sourceFile = ts.createSourceFile(\n      \"test.ts\",\n      options.sourceText,\n      ts.ScriptTarget.Latest,\n      true,\n    );\n\n    return expandMyType({\n      sourceFileName: \"test.ts\",\n      typeExpression: options.typeExpression,\n      getSourceFileFunction: () => sourceFile,\n      tsCompilerOptions: options.tsCompilerOptions,\n    });\n  }\n\n  const tsCompilerOptions = options.tsCompilerOptions ?? {\n    noEmit: true,\n\n    allowSyntheticDefaultImports: true,\n    allowArbitraryExtensions: true,\n    allowImportingTsExtensions: true,\n    allowJs: true,\n  };\n\n  const compilerHost = createExpanderCompilerHost(\n    options.sourceFileName,\n    options.typeExpression,\n    tsCompilerOptions,\n    options.getSourceFileFunction,\n  );\n\n  const program = ts.createProgram(\n    [options.sourceFileName],\n    tsCompilerOptions,\n    compilerHost,\n  );\n\n  const sourceFile = program.getSourceFile(options.sourceFileName);\n  if (!sourceFile) {\n    throw new Error(\"Source file not found!\");\n  }\n\n  const firstIdentifier = findTypeExpanderResultNode(sourceFile);\n  if (!firstIdentifier) {\n    throw new Error(\"No node found!\");\n  }\n\n  const typeChecker = program.getTypeChecker();\n  const expandedType = typeChecker.typeToString(\n    typeChecker.getTypeAtLocation(firstIdentifier),\n    undefined,\n    ts.TypeFormatFlags.NodeBuilderFlagsMask,\n  );\n\n  if (options.prettify && options.prettify.enabled === false) {\n    return expandedType;\n  }\n\n  const dummyTypeName = \"__TYPE_EXPANDER_RESULT__\";\n\n  return (\n    await format(\n      `type ${dummyTypeName} = ${expandedType}`,\n      options.prettify?.options ?? {\n        parser: \"typescript\",\n        semi: false,\n      },\n    )\n  )\n    .trim()\n    .substring(`type ${dummyTypeName} = `.length);\n}\n", "import ts from \"typescript\";\n\nconst identifierPrefix = \"__TYPE_EXPANDER__\";\n\n/**\n * Creates a custom compiler host that augments the specified source file for expanding a type expression.\n *\n * @param sourceFileName Name of the source file to augment.\n * @param typeExpression Type expression.\n * @param compilerOptions TypeScript compiler options.\n * @param getSourceFileFunction The implementation of the `ts.CompilerHost[\"getSourceFile\"]` function.\n * @returns A custom compiler host that returns an augmented source file that can be used to expand the type expression.\n */\nexport const createExpanderCompilerHost = (\n  sourceFileName: string,\n  typeExpression: string,\n  compilerOptions?: ts.CompilerOptions,\n  getSourceFileFunction?: ts.CompilerHost[\"getSourceFile\"],\n) => {\n  const customCompilerHost = ts.createCompilerHost(compilerOptions ?? {});\n  const originalGetSourceFile = customCompilerHost.getSourceFile;\n\n  customCompilerHost.getSourceFile = (fileName, ...args) => {\n    if (fileName === sourceFileName) {\n      const sourceFile = (getSourceFileFunction ?? originalGetSourceFile)(\n        fileName,\n        ...args,\n      );\n\n      if (sourceFile === undefined) {\n        return undefined;\n      }\n\n      // https://github.com/microsoft/TypeScript/blob/main/tests/cases/compiler/computedTypesKeyofNoIndexSignatureType.ts\n      const sourceText = `type ${identifierPrefix}Result = ${identifierPrefix}Expand<${identifierPrefix}Expression>;\n        type ${identifierPrefix}Expression = ${typeExpression};\n\n        type ${identifierPrefix}Expand<T> = T extends (...args: infer A) => infer R\n          ? (...args: ${identifierPrefix}Expand<A>) => ${identifierPrefix}Expand<R>\n          : { [K in keyof T]: ${identifierPrefix}Expand<T[K]>; } & {};\n          \n        ${sourceFile.getFullText()}`;\n\n      return ts.createSourceFile(\n        fileName,\n        sourceText,\n        sourceFile.languageVersion,\n        true,\n      );\n    }\n\n    return originalGetSourceFile(fileName, ...args);\n  };\n\n  return customCompilerHost;\n};\n"],
  "mappings": "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",
  "names": ["ts", "import_typescript", "ts", "sourceFile"]
}

|
package/dist/index.js
CHANGED
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
// src/create-expander-compiler-host.ts
|
|
2
2
|
import ts from "typescript";
|
|
3
|
+
var identifierPrefix = "__TYPE_EXPANDER__";
|
|
3
4
|
var createExpanderCompilerHost = (sourceFileName, typeExpression, compilerOptions, getSourceFileFunction) => {
|
|
4
5
|
const customCompilerHost = ts.createCompilerHost(compilerOptions ?? {});
|
|
5
6
|
const originalGetSourceFile = customCompilerHost.getSourceFile;
|
|
@@ -12,16 +13,14 @@ var createExpanderCompilerHost = (sourceFileName, typeExpression, compilerOption
|
|
|
12
13
|
if (sourceFile === void 0) {
|
|
13
14
|
return void 0;
|
|
14
15
|
}
|
|
15
|
-
const sourceText =
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
sourceFile.getFullText()
|
|
24
|
-
].join("\n");
|
|
16
|
+
const sourceText = `type ${identifierPrefix}Result = ${identifierPrefix}Expand<${identifierPrefix}Expression>;
|
|
17
|
+
type ${identifierPrefix}Expression = ${typeExpression};
|
|
18
|
+
|
|
19
|
+
type ${identifierPrefix}Expand<T> = T extends (...args: infer A) => infer R
|
|
20
|
+
? (...args: ${identifierPrefix}Expand<A>) => ${identifierPrefix}Expand<R>
|
|
21
|
+
: { [K in keyof T]: ${identifierPrefix}Expand<T[K]>; } & {};
|
|
22
|
+
|
|
23
|
+
${sourceFile.getFullText()}`;
|
|
25
24
|
return ts.createSourceFile(
|
|
26
25
|
fileName,
|
|
27
26
|
sourceText,
|
|
@@ -111,4 +110,4 @@ async function expandMyType(options) {
|
|
|
111
110
|
export {
|
|
112
111
|
expandMyType
|
|
113
112
|
};
|
|
114
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/create-expander-compiler-host.ts", "../src/index.ts"],
  "sourcesContent": ["import ts from \"typescript\";\n\n/**\n * Creates a custom compiler host that augments the specified source file for expanding a type expression.\n *\n * @param sourceFileName Name of the source file to augment.\n * @param typeExpression Type expression.\n * @param compilerOptions TypeScript compiler options.\n * @param getSourceFileFunction The implementation of the `ts.CompilerHost[\"getSourceFile\"]` function.\n * @returns A custom compiler host that returns an augmented source file that can be used to expand the type expression.\n */\nexport const createExpanderCompilerHost = (\n  sourceFileName: string,\n  typeExpression: string,\n  compilerOptions?: ts.CompilerOptions,\n  getSourceFileFunction?: ts.CompilerHost[\"getSourceFile\"],\n) => {\n  const customCompilerHost = ts.createCompilerHost(compilerOptions ?? {});\n  const originalGetSourceFile = customCompilerHost.getSourceFile;\n\n  customCompilerHost.getSourceFile = (fileName, ...args) => {\n    if (fileName === sourceFileName) {\n      const sourceFile = (getSourceFileFunction ?? originalGetSourceFile)(\n        fileName,\n        ...args,\n      );\n\n      if (sourceFile === undefined) {\n        return undefined;\n      }\n\n      const sourceText = [\n        ...[\n          \"type __TYPE_EXPANDER_RESULT__ = __TYPE_EXPANDER_EXPAND__<__TYPE_EXPANDER_EXPRESSION__>;\",\n          `type __TYPE_EXPANDER_EXPRESSION__ = ${typeExpression};`,\n          \"type __TYPE_EXPANDER_EXPAND__<T> = {\",\n          \"  [K in keyof T]: __TYPE_EXPANDER_EXPAND__<T[K]>;\",\n          \"} & {};\",\n        ],\n        sourceFile.getFullText(),\n      ].join(\"\\n\");\n\n      return ts.createSourceFile(\n        fileName,\n        sourceText,\n        sourceFile.languageVersion,\n        true,\n      );\n    }\n\n    return originalGetSourceFile(fileName, ...args);\n  };\n\n  return customCompilerHost;\n};\n", "import { createExpanderCompilerHost } from \"./create-expander-compiler-host.ts\";\nimport { format, type Options as PrettierOptions } from \"prettier\";\nimport ts from \"typescript\";\n\n/**\n * Finds the node that represents the TYPE_EXPANDER_RESULT type.\n *\n * @param node Node in which the TYPE_EXPANDER_RESULT type should be searched.\n * @returns The node that represents the TYPE_EXPANDER_RESULT type.\n */\nconst findTypeExpanderResultNode = (node: ts.Node): ts.Node | undefined => {\n  if (node.getChildCount() == 0) {\n    if (!ts.isIdentifier(node)) {\n      return undefined;\n    }\n\n    // Since we put the __TYPE_EXPANDER_RESULT__ type at the beginning of the\n    // file, we can return the first identifier we find.\n    return node;\n  }\n\n  return ts.forEachChild(node, findTypeExpanderResultNode);\n};\n\nexport type ExpandTypeOptionsBase = {\n  /**\n   * The type expression to expand.\n   * @example \"ReturnType<typeof myFunction>\"\n   */\n  typeExpression: string;\n\n  /**\n   * TypeScript compiler options.\n   */\n  tsCompilerOptions?: ts.CompilerOptions;\n\n  /**\n   * Prettier options.\n   */\n  prettify?: {\n    /**\n     * Whether to prettify the output.\n     * @default true\n     */\n    enabled?: boolean;\n    /**\n     * Prettier options. Don't forget to set the parser to \"typescript\".\n     * @default { parser: \"typescript\", semi: false }\n     */\n    options?: PrettierOptions;\n  };\n};\nexport type ExpandTypeFromSourceFileOptions = ExpandTypeOptionsBase & {\n  /**\n   * Name of the source file to evaluate the type expression in.\n   */\n  sourceFileName: string;\n  /**\n   * Function that returns the source file. When not specified, the implementation of the default TypeScript compiler host is used.\n   */\n  getSourceFileFunction?: ts.CompilerHost[\"getSourceFile\"];\n};\nexport type ExpandTypeFromSourceTextOptions = ExpandTypeOptionsBase & {\n  /**\n   * TypeScript source text to evaluate the type expression in.\n   */\n  sourceText: string;\n};\nexport type ExpandMyTypeOptions =\n  | ExpandTypeFromSourceFileOptions\n  | ExpandTypeFromSourceTextOptions;\n\nexport async function expandMyType(\n  options: ExpandTypeFromSourceTextOptions,\n): Promise<string>;\nexport async function expandMyType(\n  options: ExpandTypeFromSourceFileOptions,\n): Promise<string>;\n\n/**\n * Expands a TypeScript type expression.\n *\n * @param options\n * @returns The expanded type expression.\n */\nexport async function expandMyType(options: ExpandMyTypeOptions) {\n  if (options.typeExpression.trim() === \"\") {\n    return \"never\";\n  }\n\n  if (\"sourceText\" in options) {\n    const sourceFile = ts.createSourceFile(\n      \"test.ts\",\n      options.sourceText,\n      ts.ScriptTarget.Latest,\n      true,\n    );\n\n    return expandMyType({\n      sourceFileName: \"test.ts\",\n      typeExpression: options.typeExpression,\n      getSourceFileFunction: () => sourceFile,\n      tsCompilerOptions: options.tsCompilerOptions,\n    });\n  }\n\n  const tsCompilerOptions = options.tsCompilerOptions ?? {\n    noEmit: true,\n\n    allowSyntheticDefaultImports: true,\n    allowArbitraryExtensions: true,\n    allowImportingTsExtensions: true,\n    allowJs: true,\n  };\n\n  const compilerHost = createExpanderCompilerHost(\n    options.sourceFileName,\n    options.typeExpression,\n    tsCompilerOptions,\n    options.getSourceFileFunction,\n  );\n\n  const program = ts.createProgram(\n    [options.sourceFileName],\n    tsCompilerOptions,\n    compilerHost,\n  );\n\n  const sourceFile = program.getSourceFile(options.sourceFileName);\n  if (!sourceFile) {\n    throw new Error(\"Source file not found!\");\n  }\n\n  const firstIdentifier = findTypeExpanderResultNode(sourceFile);\n  if (!firstIdentifier) {\n    throw new Error(\"No node found!\");\n  }\n\n  const typeChecker = program.getTypeChecker();\n  const expandedType = typeChecker.typeToString(\n    typeChecker.getTypeAtLocation(firstIdentifier),\n    undefined,\n    ts.TypeFormatFlags.NodeBuilderFlagsMask,\n  );\n\n  if (options.prettify && options.prettify.enabled === false) {\n    return expandedType;\n  }\n\n  const dummyTypeName = \"__TYPE_EXPANDER_RESULT__\";\n\n  return (\n    await format(\n      `type ${dummyTypeName} = ${expandedType}`,\n      options.prettify?.options ?? {\n        parser: \"typescript\",\n        semi: false,\n      },\n    )\n  )\n    .trim()\n    .substring(`type ${dummyTypeName} = `.length);\n}\n"],
  "mappings": ";AAAA,OAAO,QAAQ;AAWR,IAAM,6BAA6B,CACxC,gBACA,gBACA,iBACA,0BACG;AACH,QAAM,qBAAqB,GAAG,mBAAmB,mBAAmB,CAAC,CAAC;AACtE,QAAM,wBAAwB,mBAAmB;AAEjD,qBAAmB,gBAAgB,CAAC,aAAa,SAAS;AACxD,QAAI,aAAa,gBAAgB;AAC/B,YAAM,cAAc,yBAAyB;AAAA,QAC3C;AAAA,QACA,GAAG;AAAA,MACL;AAEA,UAAI,eAAe,QAAW;AAC5B,eAAO;AAAA,MACT;AAEA,YAAM,aAAa;AAAA,QACjB,GAAG;AAAA,UACD;AAAA,UACA,uCAAuC,cAAc;AAAA,UACrD;AAAA,UACA;AAAA,UACA;AAAA,QACF;AAAA,QACA,WAAW,YAAY;AAAA,MACzB,EAAE,KAAK,IAAI;AAEX,aAAO,GAAG;AAAA,QACR;AAAA,QACA;AAAA,QACA,WAAW;AAAA,QACX;AAAA,MACF;AAAA,IACF;AAEA,WAAO,sBAAsB,UAAU,GAAG,IAAI;AAAA,EAChD;AAEA,SAAO;AACT;;;ACrDA,SAAS,cAA+C;AACxD,OAAOA,SAAQ;AAQf,IAAM,6BAA6B,CAAC,SAAuC;AACzE,MAAI,KAAK,cAAc,KAAK,GAAG;AAC7B,QAAI,CAACA,IAAG,aAAa,IAAI,GAAG;AAC1B,aAAO;AAAA,IACT;AAIA,WAAO;AAAA,EACT;AAEA,SAAOA,IAAG,aAAa,MAAM,0BAA0B;AACzD;AA+DA,eAAsB,aAAa,SAA8B;AAC/D,MAAI,QAAQ,eAAe,KAAK,MAAM,IAAI;AACxC,WAAO;AAAA,EACT;AAEA,MAAI,gBAAgB,SAAS;AAC3B,UAAMC,cAAaD,IAAG;AAAA,MACpB;AAAA,MACA,QAAQ;AAAA,MACRA,IAAG,aAAa;AAAA,MAChB;AAAA,IACF;AAEA,WAAO,aAAa;AAAA,MAClB,gBAAgB;AAAA,MAChB,gBAAgB,QAAQ;AAAA,MACxB,uBAAuB,MAAMC;AAAA,MAC7B,mBAAmB,QAAQ;AAAA,IAC7B,CAAC;AAAA,EACH;AAEA,QAAM,oBAAoB,QAAQ,qBAAqB;AAAA,IACrD,QAAQ;AAAA,IAER,8BAA8B;AAAA,IAC9B,0BAA0B;AAAA,IAC1B,4BAA4B;AAAA,IAC5B,SAAS;AAAA,EACX;AAEA,QAAM,eAAe;AAAA,IACnB,QAAQ;AAAA,IACR,QAAQ;AAAA,IACR;AAAA,IACA,QAAQ;AAAA,EACV;AAEA,QAAM,UAAUD,IAAG;AAAA,IACjB,CAAC,QAAQ,cAAc;AAAA,IACvB;AAAA,IACA;AAAA,EACF;AAEA,QAAM,aAAa,QAAQ,cAAc,QAAQ,cAAc;AAC/D,MAAI,CAAC,YAAY;AACf,UAAM,IAAI,MAAM,wBAAwB;AAAA,EAC1C;AAEA,QAAM,kBAAkB,2BAA2B,UAAU;AAC7D,MAAI,CAAC,iBAAiB;AACpB,UAAM,IAAI,MAAM,gBAAgB;AAAA,EAClC;AAEA,QAAM,cAAc,QAAQ,eAAe;AAC3C,QAAM,eAAe,YAAY;AAAA,IAC/B,YAAY,kBAAkB,eAAe;AAAA,IAC7C;AAAA,IACAA,IAAG,gBAAgB;AAAA,EACrB;AAEA,MAAI,QAAQ,YAAY,QAAQ,SAAS,YAAY,OAAO;AAC1D,WAAO;AAAA,EACT;AAEA,QAAM,gBAAgB;AAEtB,UACE,MAAM;AAAA,IACJ,QAAQ,aAAa,MAAM,YAAY;AAAA,IACvC,QAAQ,UAAU,WAAW;AAAA,MAC3B,QAAQ;AAAA,MACR,MAAM;AAAA,IACR;AAAA,EACF,GAEC,KAAK,EACL,UAAU,QAAQ,aAAa,MAAM,MAAM;AAChD;",
  "names": ["ts", "sourceFile"]
}

|
|
113
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/create-expander-compiler-host.ts", "../src/index.ts"],
  "sourcesContent": ["import ts from \"typescript\";\n\nconst identifierPrefix = \"__TYPE_EXPANDER__\";\n\n/**\n * Creates a custom compiler host that augments the specified source file for expanding a type expression.\n *\n * @param sourceFileName Name of the source file to augment.\n * @param typeExpression Type expression.\n * @param compilerOptions TypeScript compiler options.\n * @param getSourceFileFunction The implementation of the `ts.CompilerHost[\"getSourceFile\"]` function.\n * @returns A custom compiler host that returns an augmented source file that can be used to expand the type expression.\n */\nexport const createExpanderCompilerHost = (\n  sourceFileName: string,\n  typeExpression: string,\n  compilerOptions?: ts.CompilerOptions,\n  getSourceFileFunction?: ts.CompilerHost[\"getSourceFile\"],\n) => {\n  const customCompilerHost = ts.createCompilerHost(compilerOptions ?? {});\n  const originalGetSourceFile = customCompilerHost.getSourceFile;\n\n  customCompilerHost.getSourceFile = (fileName, ...args) => {\n    if (fileName === sourceFileName) {\n      const sourceFile = (getSourceFileFunction ?? originalGetSourceFile)(\n        fileName,\n        ...args,\n      );\n\n      if (sourceFile === undefined) {\n        return undefined;\n      }\n\n      // https://github.com/microsoft/TypeScript/blob/main/tests/cases/compiler/computedTypesKeyofNoIndexSignatureType.ts\n      const sourceText = `type ${identifierPrefix}Result = ${identifierPrefix}Expand<${identifierPrefix}Expression>;\n        type ${identifierPrefix}Expression = ${typeExpression};\n\n        type ${identifierPrefix}Expand<T> = T extends (...args: infer A) => infer R\n          ? (...args: ${identifierPrefix}Expand<A>) => ${identifierPrefix}Expand<R>\n          : { [K in keyof T]: ${identifierPrefix}Expand<T[K]>; } & {};\n          \n        ${sourceFile.getFullText()}`;\n\n      return ts.createSourceFile(\n        fileName,\n        sourceText,\n        sourceFile.languageVersion,\n        true,\n      );\n    }\n\n    return originalGetSourceFile(fileName, ...args);\n  };\n\n  return customCompilerHost;\n};\n", "import { createExpanderCompilerHost } from \"./create-expander-compiler-host.ts\";\nimport { format, type Options as PrettierOptions } from \"prettier\";\nimport ts from \"typescript\";\n\n/**\n * Finds the node that represents the TYPE_EXPANDER_RESULT type.\n *\n * @param node Node in which the TYPE_EXPANDER_RESULT type should be searched.\n * @returns The node that represents the TYPE_EXPANDER_RESULT type.\n */\nconst findTypeExpanderResultNode = (node: ts.Node): ts.Node | undefined => {\n  if (node.getChildCount() == 0) {\n    if (!ts.isIdentifier(node)) {\n      return undefined;\n    }\n\n    // Since we put the __TYPE_EXPANDER_RESULT__ type at the beginning of the\n    // file, we can return the first identifier we find.\n    return node;\n  }\n\n  return ts.forEachChild(node, findTypeExpanderResultNode);\n};\n\nexport type ExpandTypeOptionsBase = {\n  /**\n   * The type expression to expand.\n   * @example \"ReturnType<typeof myFunction>\"\n   */\n  typeExpression: string;\n\n  /**\n   * TypeScript compiler options.\n   */\n  tsCompilerOptions?: ts.CompilerOptions;\n\n  /**\n   * Prettier options.\n   */\n  prettify?: {\n    /**\n     * Whether to prettify the output.\n     * @default true\n     */\n    enabled?: boolean;\n    /**\n     * Prettier options. Don't forget to set the parser to \"typescript\".\n     * @default { parser: \"typescript\", semi: false }\n     */\n    options?: PrettierOptions;\n  };\n};\nexport type ExpandTypeFromSourceFileOptions = ExpandTypeOptionsBase & {\n  /**\n   * Name of the source file to evaluate the type expression in.\n   */\n  sourceFileName: string;\n  /**\n   * Function that returns the source file. When not specified, the implementation of the default TypeScript compiler host is used.\n   */\n  getSourceFileFunction?: ts.CompilerHost[\"getSourceFile\"];\n};\nexport type ExpandTypeFromSourceTextOptions = ExpandTypeOptionsBase & {\n  /**\n   * TypeScript source text to evaluate the type expression in.\n   */\n  sourceText: string;\n};\nexport type ExpandMyTypeOptions =\n  | ExpandTypeFromSourceFileOptions\n  | ExpandTypeFromSourceTextOptions;\n\nexport async function expandMyType(\n  options: ExpandTypeFromSourceTextOptions,\n): Promise<string>;\nexport async function expandMyType(\n  options: ExpandTypeFromSourceFileOptions,\n): Promise<string>;\n\n/**\n * Expands a TypeScript type expression.\n *\n * @param options\n * @returns The expanded type expression.\n */\nexport async function expandMyType(options: ExpandMyTypeOptions) {\n  if (options.typeExpression.trim() === \"\") {\n    return \"never\";\n  }\n\n  if (\"sourceText\" in options) {\n    const sourceFile = ts.createSourceFile(\n      \"test.ts\",\n      options.sourceText,\n      ts.ScriptTarget.Latest,\n      true,\n    );\n\n    return expandMyType({\n      sourceFileName: \"test.ts\",\n      typeExpression: options.typeExpression,\n      getSourceFileFunction: () => sourceFile,\n      tsCompilerOptions: options.tsCompilerOptions,\n    });\n  }\n\n  const tsCompilerOptions = options.tsCompilerOptions ?? {\n    noEmit: true,\n\n    allowSyntheticDefaultImports: true,\n    allowArbitraryExtensions: true,\n    allowImportingTsExtensions: true,\n    allowJs: true,\n  };\n\n  const compilerHost = createExpanderCompilerHost(\n    options.sourceFileName,\n    options.typeExpression,\n    tsCompilerOptions,\n    options.getSourceFileFunction,\n  );\n\n  const program = ts.createProgram(\n    [options.sourceFileName],\n    tsCompilerOptions,\n    compilerHost,\n  );\n\n  const sourceFile = program.getSourceFile(options.sourceFileName);\n  if (!sourceFile) {\n    throw new Error(\"Source file not found!\");\n  }\n\n  const firstIdentifier = findTypeExpanderResultNode(sourceFile);\n  if (!firstIdentifier) {\n    throw new Error(\"No node found!\");\n  }\n\n  const typeChecker = program.getTypeChecker();\n  const expandedType = typeChecker.typeToString(\n    typeChecker.getTypeAtLocation(firstIdentifier),\n    undefined,\n    ts.TypeFormatFlags.NodeBuilderFlagsMask,\n  );\n\n  if (options.prettify && options.prettify.enabled === false) {\n    return expandedType;\n  }\n\n  const dummyTypeName = \"__TYPE_EXPANDER_RESULT__\";\n\n  return (\n    await format(\n      `type ${dummyTypeName} = ${expandedType}`,\n      options.prettify?.options ?? {\n        parser: \"typescript\",\n        semi: false,\n      },\n    )\n  )\n    .trim()\n    .substring(`type ${dummyTypeName} = `.length);\n}\n"],
  "mappings": "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",
  "names": ["ts", "sourceFile"]
}

|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "expand-my-type",
|
|
3
|
-
"version": "0.
|
|
3
|
+
"version": "0.3.0",
|
|
4
4
|
"description": "Expand TypeScript type expressions programmatically",
|
|
5
5
|
"keywords": [
|
|
6
6
|
"expand",
|
|
@@ -32,15 +32,21 @@
|
|
|
32
32
|
},
|
|
33
33
|
"main": "dist/index.cjs",
|
|
34
34
|
"types": "dist/index.d.ts",
|
|
35
|
+
"bin": {
|
|
36
|
+
"expand-my-type": "./dist/cli.js"
|
|
37
|
+
},
|
|
35
38
|
"files": [
|
|
36
39
|
"dist/index.cjs",
|
|
37
40
|
"dist/index.d.cts",
|
|
38
41
|
"dist/index.d.ts",
|
|
39
|
-
"dist/index.js"
|
|
42
|
+
"dist/index.js",
|
|
43
|
+
"dist/cli.js"
|
|
40
44
|
],
|
|
41
45
|
"scripts": {
|
|
46
|
+
"bundle-cli": "tsup ./src/cli.ts --format esm --sourcemap inline --silent",
|
|
47
|
+
"bundle-lib": "tsup ./src/index.ts --format cjs,esm --dts --clean --sourcemap inline --silent",
|
|
42
48
|
"format": "prettier --write .",
|
|
43
|
-
"prepare": "
|
|
49
|
+
"prepare": "npm run bundle-lib && npm run bundle-cli",
|
|
44
50
|
"pretest": "tsc",
|
|
45
51
|
"test": "tsx --test --test-reporter spec ./src/index.test.ts"
|
|
46
52
|
},
|