expand-my-type 0.6.7 → 0.7.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 CHANGED
@@ -137,6 +137,15 @@ the following ways:
137
137
  /* { a: string; b: number } */
138
138
  ```
139
139
 
140
+ For local development in this repository, use Bun:
141
+
142
+ ```sh
143
+ bun install
144
+ bun run check
145
+ bun run build
146
+ bun run deps:update
147
+ ```
148
+
140
149
  ## How It Works
141
150
 
142
151
  Expand My Type uses the following utility types to expand the type expression:
@@ -0,0 +1,16 @@
1
+ import ts from "typescript";
2
+ type ExtractFunctions<T, K extends keyof T = keyof T> = {
3
+ [P in K]: T[P] extends (...args: any[]) => any ? T[P] : never;
4
+ };
5
+ export type CompilerHostFunctionOverrides = Partial<ExtractFunctions<ts.CompilerHost>>;
6
+ /**
7
+ * Creates a custom compiler host that augments the specified source file for expanding a type expression.
8
+ *
9
+ * @param sourceFileName Name of the source file to augment.
10
+ * @param codeToAdd Type expression.
11
+ * @param compilerOptions TypeScript compiler options.
12
+ * @param compilerHostFunctionOverrides A record of functions to override in the compiler host. Useful for mocking.
13
+ * @returns A custom compiler host that returns an augmented source file that can be used to expand the type expression.
14
+ */
15
+ export declare const createAugmenterCompilerHost: (sourceFileName: string, codeToAdd: string, compilerOptions?: ts.CompilerOptions, compilerHostFunctionOverrides?: CompilerHostFunctionOverrides) => ts.CompilerHost;
16
+ export {};
package/dist/cli.d.ts ADDED
@@ -0,0 +1,2 @@
1
+ #!/usr/bin/env node
2
+ export {};
package/dist/cli.js CHANGED
@@ -2,18 +2,22 @@
2
2
 
3
3
  // src/augmenter-compiler-host.ts
4
4
  import ts from "typescript";
5
- var createAugmenterCompilerHost = (sourceFileName2, codeToAdd, compilerOptions, compilerHostFunctionOverrides) => {
5
+ var createAugmenterCompilerHost = (sourceFileName, codeToAdd, compilerOptions, compilerHostFunctionOverrides) => {
6
6
  const customCompilerHost = ts.createCompilerHost(compilerOptions ?? {}, true);
7
- for (const key of Object.keys(compilerHostFunctionOverrides ?? {})) {
8
- customCompilerHost[key] = compilerHostFunctionOverrides[key];
7
+ const overrides = compilerHostFunctionOverrides ?? {};
8
+ for (const key of Object.keys(overrides)) {
9
+ const override = overrides[key];
10
+ if (override) {
11
+ customCompilerHost[key] = override;
12
+ }
9
13
  }
10
14
  const originalReadFile = customCompilerHost.readFile;
11
15
  customCompilerHost.readFile = (fileName) => {
12
16
  const contents = originalReadFile(fileName);
13
- if (contents === void 0) {
17
+ if (contents === undefined) {
14
18
  return contents;
15
19
  }
16
- if (fileName !== sourceFileName2) {
20
+ if (fileName !== sourceFileName) {
17
21
  return contents;
18
22
  }
19
23
  return `${codeToAdd}
@@ -25,9 +29,9 @@ ${contents}`;
25
29
  // src/code-generator.ts
26
30
  import { format } from "prettier";
27
31
  var identifierPrefix = "__EXPAND_MY_TYPE__";
28
- var createExpandCodeBlock = (typeExpression2) => {
32
+ var createExpandCodeBlock = (typeExpression) => {
29
33
  return `type ${identifierPrefix}Result = ${identifierPrefix}Expand<${identifierPrefix}Expression>;
30
- type ${identifierPrefix}Expression = ${typeExpression2};
34
+ type ${identifierPrefix}Expression = ${typeExpression};
31
35
 
32
36
  type ${identifierPrefix}Expand<T> =
33
37
  T extends (...args: infer A) => infer R ? (...args: ${identifierPrefix}Expand<A>) => ${identifierPrefix}Expand<R>
@@ -42,13 +46,10 @@ var createExpandCodeBlock = (typeExpression2) => {
42
46
  type ${identifierPrefix}RemoveUnderscore<T extends string> = T extends \`\${infer U}_\` ? U : never;`;
43
47
  };
44
48
  var formatTypeExpression = async (code, prettierOptions) => {
45
- return (await format(
46
- `type ${identifierPrefix} = ${code}`,
47
- prettierOptions ?? {
48
- parser: "typescript",
49
- semi: false
50
- }
51
- )).trim().substring(`type ${identifierPrefix} = `.length);
49
+ return (await format(`type ${identifierPrefix} = ${code}`, prettierOptions ?? {
50
+ parser: "typescript",
51
+ semi: false
52
+ })).trim().substring(`type ${identifierPrefix} = `.length);
52
53
  };
53
54
 
54
55
  // src/index.ts
@@ -57,7 +58,7 @@ import ts2 from "typescript";
57
58
  var findResultIdentifierNode = (node) => {
58
59
  if (node.getChildCount() == 0) {
59
60
  if (!ts2.isIdentifier(node)) {
60
- return void 0;
61
+ return;
61
62
  }
62
63
  return node;
63
64
  }
@@ -96,17 +97,8 @@ async function expandMyType(options) {
96
97
  if (!tsCompilerOptions.strictNullChecks) {
97
98
  throw new Error("strictNullChecks must be enabled!");
98
99
  }
99
- const compilerHost = createAugmenterCompilerHost(
100
- resolvedSourceFileName,
101
- createExpandCodeBlock(options.typeExpression),
102
- tsCompilerOptions,
103
- options.compilerHostFunctionOverrides
104
- );
105
- const program = ts2.createProgram(
106
- [resolvedSourceFileName],
107
- tsCompilerOptions,
108
- compilerHost
109
- );
100
+ const compilerHost = createAugmenterCompilerHost(resolvedSourceFileName, createExpandCodeBlock(options.typeExpression), tsCompilerOptions, options.compilerHostFunctionOverrides);
101
+ const program = ts2.createProgram([resolvedSourceFileName], tsCompilerOptions, compilerHost);
110
102
  const sourceFile = program.getSourceFile(resolvedSourceFileName);
111
103
  if (!sourceFile) {
112
104
  throw new Error("Source file not found!");
@@ -116,11 +108,7 @@ async function expandMyType(options) {
116
108
  throw new Error("No node found!");
117
109
  }
118
110
  const typeChecker = program.getTypeChecker();
119
- const expandedTypeString = typeChecker.typeToString(
120
- typeChecker.getTypeAtLocation(resultIdentifierNode),
121
- void 0,
122
- ts2.TypeFormatFlags.NodeBuilderFlagsMask
123
- );
111
+ const expandedTypeString = typeChecker.typeToString(typeChecker.getTypeAtLocation(resultIdentifierNode), undefined, ts2.TypeFormatFlags.NodeBuilderFlagsMask);
124
112
  if (options.prettify && options.prettify.enabled === false) {
125
113
  return expandedTypeString;
126
114
  }
@@ -131,37 +119,37 @@ async function expandMyType(options) {
131
119
  import { parseArgs } from "node:util";
132
120
  import ts3 from "typescript";
133
121
  var tryParse = (options) => {
134
- let result2;
122
+ let result;
135
123
  try {
136
- result2 = parseArgs({
124
+ result = parseArgs({
137
125
  options,
138
126
  tokens: true,
139
127
  allowPositionals: true
140
128
  });
141
- } catch (error2) {
129
+ } catch (error) {
142
130
  return {
143
- error: error2,
131
+ error: error instanceof Error ? error : new Error(String(error)),
144
132
  values: {},
145
133
  positionals: []
146
134
  };
147
135
  }
148
- const { tokens, values: values2, positionals: positionals2 } = result2;
136
+ const { tokens, values, positionals } = result;
149
137
  for (const token of tokens ?? []) {
150
138
  if (token.kind === "option-terminator" || token.kind === "positional") {
151
139
  continue;
152
140
  }
153
141
  if (token.name.startsWith("no-")) {
154
142
  const positiveName = token.name.slice("no-".length);
155
- values2[positiveName] = false;
156
- delete values2[token.name];
143
+ values[positiveName] = false;
144
+ delete values[token.name];
157
145
  } else {
158
- values2[token.name] = token.value ?? true;
146
+ values[token.name] = token.value ?? true;
159
147
  }
160
148
  }
161
149
  return {
162
- error: void 0,
163
- values: values2,
164
- positionals: positionals2
150
+ error: undefined,
151
+ values,
152
+ positionals
165
153
  };
166
154
  };
167
155
  var { error, values, positionals } = tryParse({
@@ -192,11 +180,12 @@ var usagePrompt = [
192
180
  " expand-my-type [options] <source-file> <expression>",
193
181
  "",
194
182
  "Options:",
195
- " -h, --help Show this help message",
196
- " -p, --prettify Prettify the output (default)",
197
- " -P, --no-prettify Do not prettify the output",
198
- " -c, --tsconfig <file> Use the specified tsconfig.json file"
199
- ].join("\n");
183
+ " -h, --help\t\t\tShow this help message",
184
+ " -p, --prettify\t\tPrettify the output (default)",
185
+ " -P, --no-prettify\t\tDo not prettify the output",
186
+ " -c, --tsconfig <file>\t\tUse the specified tsconfig.json file"
187
+ ].join(`
188
+ `);
200
189
  if (positionals.length !== 2) {
201
190
  console.error(usagePrompt);
202
191
  process.exit(1);
@@ -210,11 +199,7 @@ var { prettify, tsconfig: tsConfigFileName } = values;
210
199
  var tsParsedCommandLine;
211
200
  if (tsConfigFileName) {
212
201
  const configFile = ts3.readConfigFile(tsConfigFileName, ts3.sys.readFile);
213
- const compilerOptions = ts3.parseJsonConfigFileContent(
214
- configFile.config,
215
- ts3.sys,
216
- "./"
217
- );
202
+ const compilerOptions = ts3.parseJsonConfigFileContent(configFile.config, ts3.sys, "./");
218
203
  tsParsedCommandLine = compilerOptions;
219
204
  }
220
205
  var result = await expandMyType({
@@ -226,4 +211,6 @@ var result = await expandMyType({
226
211
  tsCompilerOptions: tsParsedCommandLine?.options
227
212
  });
228
213
  console.log(result);
229
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/augmenter-compiler-host.ts", "../src/code-generator.ts", "../src/index.ts", "../src/cli.ts"],
  "sourcesContent": ["import ts from \"typescript\";\n\ntype ExtractFunctions<T, K extends keyof T = keyof T> = {\n  [P in K]: T[P] extends (...args: any[]) => any ? T[P] : never;\n};\nexport type CompilerHostFunctionOverrides = Partial<\n  ExtractFunctions<ts.CompilerHost>\n>;\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 codeToAdd Type expression.\n * @param compilerOptions TypeScript compiler options.\n * @param compilerHostFunctionOverrides A record of functions to override in the compiler host. Useful for mocking.\n * @returns A custom compiler host that returns an augmented source file that can be used to expand the type expression.\n */\nexport const createAugmenterCompilerHost = (\n  sourceFileName: string,\n  codeToAdd: string,\n  compilerOptions?: ts.CompilerOptions,\n  compilerHostFunctionOverrides?: CompilerHostFunctionOverrides,\n) => {\n  const customCompilerHost = ts.createCompilerHost(compilerOptions ?? {}, true);\n  for (const key of Object.keys(compilerHostFunctionOverrides ?? {})) {\n    customCompilerHost[key] = compilerHostFunctionOverrides![key];\n  }\n\n  const originalReadFile = customCompilerHost.readFile;\n\n  customCompilerHost.readFile = (fileName) => {\n    const contents = originalReadFile(fileName);\n\n    if (contents === undefined) {\n      return contents;\n    }\n\n    if (fileName !== sourceFileName) {\n      return contents;\n    }\n\n    return `${codeToAdd}\\n${contents}`;\n  };\n\n  return customCompilerHost;\n};\n", "import type { Options as PrettierOptions } from \"prettier\";\nimport { format } from \"prettier\";\n\nconst identifierPrefix = \"__EXPAND_MY_TYPE__\";\n\nexport const createExpandCodeBlock = (typeExpression: string) => {\n  // https://github.com/microsoft/TypeScript/blob/main/tests/cases/compiler/computedTypesKeyofNoIndexSignatureType.ts\n  return `type ${identifierPrefix}Result = ${identifierPrefix}Expand<${identifierPrefix}Expression>;\n    type ${identifierPrefix}Expression = ${typeExpression};\n\n    type ${identifierPrefix}Expand<T> = \n        T extends (...args: infer A) => infer R ? (...args: ${identifierPrefix}Expand<A>) => ${identifierPrefix}Expand<R>\n      : T extends Promise<infer U> ? Promise<${identifierPrefix}ExpandTypeArgument<U>>\n      : { [K in keyof T]: T[K] extends string ? ${identifierPrefix}ExpandString<T[K]> : ${identifierPrefix}Expand<T[K]>; } & {};\n\n    type ${identifierPrefix}ExpandTypeArgument<T> = [T & {}] extends [never] ? T : T & {} extends void ? T : ${identifierPrefix}Expand<T & {}>;\n\n    // Forces a union of string literal types to be expanded\n    type ${identifierPrefix}ExpandString<T extends string> = ${identifierPrefix}RemoveUnderscore<${identifierPrefix}AppendUnderscore<T>>;\n    type ${identifierPrefix}AppendUnderscore<T extends string> = \\`\\${T}_\\` extends string ? \\`\\${T}_\\` : never;\n    type ${identifierPrefix}RemoveUnderscore<T extends string> = T extends \\`\\${infer U}_\\` ? U : never;`;\n};\n\nexport const formatTypeExpression = async (\n  code: string,\n  prettierOptions?: PrettierOptions,\n) => {\n  return (\n    await format(\n      `type ${identifierPrefix} = ${code}`,\n      prettierOptions ?? {\n        parser: \"typescript\",\n        semi: false,\n      },\n    )\n  )\n    .trim()\n    .substring(`type ${identifierPrefix} = `.length);\n};\n", "import { createAugmenterCompilerHost } from \"./augmenter-compiler-host.ts\";\nimport { type CompilerHostFunctionOverrides } from \"./augmenter-compiler-host.ts\";\nimport {\n  createExpandCodeBlock,\n  formatTypeExpression,\n} from \"./code-generator.ts\";\nimport path from \"node:path\";\nimport type { Options as PrettierOptions } from \"prettier\";\nimport ts from \"typescript\";\n\n/**\n * Finds the result type identifier node.\n *\n * @param node Node in which type type should be searched.\n * @returns The result type identifier node.\n */\nconst findResultIdentifierNode = (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 __<IDENTIFIER>__ 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, findResultIdentifierNode);\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\n  /**\n   * A record of functions to override in the compiler host. Useful for mocking\n   */\n  compilerHostFunctionOverrides?: CompilerHostFunctionOverrides;\n};\nexport type ExpandTypeFromSourceFileOptions = ExpandTypeOptionsBase & {\n  /**\n   * Name of the source file to evaluate the type expression in.\n   */\n  sourceFileName: string;\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 dummyFileName = \"expand-my-type-dummy.ts\";\n\n    return expandMyType({\n      sourceFileName: dummyFileName,\n      typeExpression: options.typeExpression,\n      compilerHostFunctionOverrides: {\n        readFile(fileName: string) {\n          if (path.basename(fileName) === dummyFileName) {\n            return options.sourceText;\n          }\n\n          return ts.sys.readFile(fileName);\n        },\n      },\n      tsCompilerOptions: options.tsCompilerOptions,\n      prettify: options.prettify,\n    });\n  }\n\n  const resolvedSourceFileName = path.resolve(options.sourceFileName);\n\n  const tsCompilerOptions = options.tsCompilerOptions ?? {\n    noEmit: true,\n\n    strictNullChecks: true,\n    allowSyntheticDefaultImports: true,\n    allowArbitraryExtensions: true,\n    allowImportingTsExtensions: true,\n    allowJs: true,\n  };\n\n  if (!tsCompilerOptions.strictNullChecks) {\n    throw new Error(\"strictNullChecks must be enabled!\");\n  }\n\n  const compilerHost = createAugmenterCompilerHost(\n    resolvedSourceFileName,\n    createExpandCodeBlock(options.typeExpression),\n    tsCompilerOptions,\n    options.compilerHostFunctionOverrides,\n  );\n\n  const program = ts.createProgram(\n    [resolvedSourceFileName],\n    tsCompilerOptions,\n    compilerHost,\n  );\n\n  const sourceFile = program.getSourceFile(resolvedSourceFileName);\n  if (!sourceFile) {\n    throw new Error(\"Source file not found!\");\n  }\n\n  const resultIdentifierNode = findResultIdentifierNode(sourceFile);\n  if (!resultIdentifierNode) {\n    throw new Error(\"No node found!\");\n  }\n\n  const typeChecker = program.getTypeChecker();\n  const expandedTypeString = typeChecker.typeToString(\n    typeChecker.getTypeAtLocation(resultIdentifierNode),\n    undefined,\n    ts.TypeFormatFlags.NodeBuilderFlagsMask,\n  );\n\n  if (options.prettify && options.prettify.enabled === false) {\n    return expandedTypeString;\n  }\n\n  return formatTypeExpression(expandedTypeString, options.prettify?.options);\n}\n", "#!/usr/bin/env node\nimport { expandMyType } from \"./index.ts\";\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,\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", "ts", "result", "error", "values", "positionals"]
}

214
+
215
+ //# debugId=CDA842588C4AEC4564756E2164756E21
216
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/augmenter-compiler-host.ts", "../src/code-generator.ts", "../src/index.ts", "../src/cli.ts"],
  "sourcesContent": [
    "import ts from \"typescript\";\n\ntype ExtractFunctions<T, K extends keyof T = keyof T> = {\n  [P in K]: T[P] extends (...args: any[]) => any ? T[P] : never;\n};\nexport type CompilerHostFunctionOverrides = Partial<\n  ExtractFunctions<ts.CompilerHost>\n>;\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 codeToAdd Type expression.\n * @param compilerOptions TypeScript compiler options.\n * @param compilerHostFunctionOverrides A record of functions to override in the compiler host. Useful for mocking.\n * @returns A custom compiler host that returns an augmented source file that can be used to expand the type expression.\n */\nexport const createAugmenterCompilerHost = (\n  sourceFileName: string,\n  codeToAdd: string,\n  compilerOptions?: ts.CompilerOptions,\n  compilerHostFunctionOverrides?: CompilerHostFunctionOverrides,\n) => {\n  const customCompilerHost = ts.createCompilerHost(compilerOptions ?? {}, true);\n  const overrides = compilerHostFunctionOverrides ?? {};\n\n  for (const key of Object.keys(overrides) as Array<\n    keyof CompilerHostFunctionOverrides\n  >) {\n    const override = overrides[key];\n\n    if (override) {\n      (customCompilerHost as unknown as Record<string, unknown>)[key] =\n        override;\n    }\n  }\n\n  const originalReadFile = customCompilerHost.readFile;\n\n  customCompilerHost.readFile = (fileName) => {\n    const contents = originalReadFile(fileName);\n\n    if (contents === undefined) {\n      return contents;\n    }\n\n    if (fileName !== sourceFileName) {\n      return contents;\n    }\n\n    return `${codeToAdd}\\n${contents}`;\n  };\n\n  return customCompilerHost;\n};\n",
    "import type { Options as PrettierOptions } from \"prettier\";\nimport { format } from \"prettier\";\n\nconst identifierPrefix = \"__EXPAND_MY_TYPE__\";\n\nexport const createExpandCodeBlock = (typeExpression: string) => {\n  // https://github.com/microsoft/TypeScript/blob/main/tests/cases/compiler/computedTypesKeyofNoIndexSignatureType.ts\n  return `type ${identifierPrefix}Result = ${identifierPrefix}Expand<${identifierPrefix}Expression>;\n    type ${identifierPrefix}Expression = ${typeExpression};\n\n    type ${identifierPrefix}Expand<T> = \n        T extends (...args: infer A) => infer R ? (...args: ${identifierPrefix}Expand<A>) => ${identifierPrefix}Expand<R>\n      : T extends Promise<infer U> ? Promise<${identifierPrefix}ExpandTypeArgument<U>>\n      : { [K in keyof T]: T[K] extends string ? ${identifierPrefix}ExpandString<T[K]> : ${identifierPrefix}Expand<T[K]>; } & {};\n\n    type ${identifierPrefix}ExpandTypeArgument<T> = [T & {}] extends [never] ? T : T & {} extends void ? T : ${identifierPrefix}Expand<T & {}>;\n\n    // Forces a union of string literal types to be expanded\n    type ${identifierPrefix}ExpandString<T extends string> = ${identifierPrefix}RemoveUnderscore<${identifierPrefix}AppendUnderscore<T>>;\n    type ${identifierPrefix}AppendUnderscore<T extends string> = \\`\\${T}_\\` extends string ? \\`\\${T}_\\` : never;\n    type ${identifierPrefix}RemoveUnderscore<T extends string> = T extends \\`\\${infer U}_\\` ? U : never;`;\n};\n\nexport const formatTypeExpression = async (\n  code: string,\n  prettierOptions?: PrettierOptions,\n) => {\n  return (\n    await format(\n      `type ${identifierPrefix} = ${code}`,\n      prettierOptions ?? {\n        parser: \"typescript\",\n        semi: false,\n      },\n    )\n  )\n    .trim()\n    .substring(`type ${identifierPrefix} = `.length);\n};\n",
    "import { createAugmenterCompilerHost } from \"./augmenter-compiler-host.js\";\nimport { type CompilerHostFunctionOverrides } from \"./augmenter-compiler-host.js\";\nimport {\n  createExpandCodeBlock,\n  formatTypeExpression,\n} from \"./code-generator.js\";\nimport path from \"node:path\";\nimport type { Options as PrettierOptions } from \"prettier\";\nimport ts from \"typescript\";\n\n/**\n * Finds the result type identifier node.\n *\n * @param node Node in which type type should be searched.\n * @returns The result type identifier node.\n */\nconst findResultIdentifierNode = (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 __<IDENTIFIER>__ 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, findResultIdentifierNode);\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\n  /**\n   * A record of functions to override in the compiler host. Useful for mocking\n   */\n  compilerHostFunctionOverrides?: CompilerHostFunctionOverrides;\n};\nexport type ExpandTypeFromSourceFileOptions = ExpandTypeOptionsBase & {\n  /**\n   * Name of the source file to evaluate the type expression in.\n   */\n  sourceFileName: string;\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 dummyFileName = \"expand-my-type-dummy.ts\";\n\n    return expandMyType({\n      sourceFileName: dummyFileName,\n      typeExpression: options.typeExpression,\n      compilerHostFunctionOverrides: {\n        readFile(fileName: string) {\n          if (path.basename(fileName) === dummyFileName) {\n            return options.sourceText;\n          }\n\n          return ts.sys.readFile(fileName);\n        },\n      },\n      tsCompilerOptions: options.tsCompilerOptions,\n      prettify: options.prettify,\n    });\n  }\n\n  const resolvedSourceFileName = path.resolve(options.sourceFileName);\n\n  const tsCompilerOptions = options.tsCompilerOptions ?? {\n    noEmit: true,\n\n    strictNullChecks: true,\n    allowSyntheticDefaultImports: true,\n    allowArbitraryExtensions: true,\n    allowImportingTsExtensions: true,\n    allowJs: true,\n  };\n\n  if (!tsCompilerOptions.strictNullChecks) {\n    throw new Error(\"strictNullChecks must be enabled!\");\n  }\n\n  const compilerHost = createAugmenterCompilerHost(\n    resolvedSourceFileName,\n    createExpandCodeBlock(options.typeExpression),\n    tsCompilerOptions,\n    options.compilerHostFunctionOverrides,\n  );\n\n  const program = ts.createProgram(\n    [resolvedSourceFileName],\n    tsCompilerOptions,\n    compilerHost,\n  );\n\n  const sourceFile = program.getSourceFile(resolvedSourceFileName);\n  if (!sourceFile) {\n    throw new Error(\"Source file not found!\");\n  }\n\n  const resultIdentifierNode = findResultIdentifierNode(sourceFile);\n  if (!resultIdentifierNode) {\n    throw new Error(\"No node found!\");\n  }\n\n  const typeChecker = program.getTypeChecker();\n  const expandedTypeString = typeChecker.typeToString(\n    typeChecker.getTypeAtLocation(resultIdentifierNode),\n    undefined,\n    ts.TypeFormatFlags.NodeBuilderFlagsMask,\n  );\n\n  if (options.prettify && options.prettify.enabled === false) {\n    return expandedTypeString;\n  }\n\n  return formatTypeExpression(expandedTypeString, options.prettify?.options);\n}\n",
    "#!/usr/bin/env node\nimport { expandMyType } from \"./index.js\";\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: error instanceof Error ? error : new Error(String(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,\n  typeExpression,\n  prettify: {\n    enabled: prettify,\n  },\n  tsCompilerOptions: tsParsedCommandLine?.options,\n});\n\nconsole.log(result);\n"
  ],
  "mappings": "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",
  "debugId": "CDA842588C4AEC4564756E2164756E21",
  "names": []
}
@@ -0,0 +1,3 @@
1
+ import type { Options as PrettierOptions } from "prettier";
2
+ export declare const createExpandCodeBlock: (typeExpression: string) => string;
3
+ export declare const formatTypeExpression: (code: string, prettierOptions?: PrettierOptions) => Promise<string>;
package/dist/index.cjs CHANGED
@@ -1,49 +1,87 @@
1
1
  var __create = Object.create;
2
+ var __getProtoOf = Object.getPrototypeOf;
2
3
  var __defProp = Object.defineProperty;
3
- var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
4
4
  var __getOwnPropNames = Object.getOwnPropertyNames;
5
- var __getProtoOf = Object.getPrototypeOf;
5
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
6
6
  var __hasOwnProp = Object.prototype.hasOwnProperty;
7
- var __export = (target, all) => {
8
- for (var name in all)
9
- __defProp(target, name, { get: all[name], enumerable: true });
7
+ function __accessProp(key) {
8
+ return this[key];
9
+ }
10
+ var __toESMCache_node;
11
+ var __toESMCache_esm;
12
+ var __toESM = (mod, isNodeMode, target) => {
13
+ var canCache = mod != null && typeof mod === "object";
14
+ if (canCache) {
15
+ var cache = isNodeMode ? __toESMCache_node ??= new WeakMap : __toESMCache_esm ??= new WeakMap;
16
+ var cached = cache.get(mod);
17
+ if (cached)
18
+ return cached;
19
+ }
20
+ target = mod != null ? __create(__getProtoOf(mod)) : {};
21
+ const to = isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target;
22
+ for (let key of __getOwnPropNames(mod))
23
+ if (!__hasOwnProp.call(to, key))
24
+ __defProp(to, key, {
25
+ get: __accessProp.bind(mod, key),
26
+ enumerable: true
27
+ });
28
+ if (canCache)
29
+ cache.set(mod, to);
30
+ return to;
10
31
  };
11
- var __copyProps = (to, from, except, desc) => {
32
+ var __toCommonJS = (from) => {
33
+ var entry = (__moduleCache ??= new WeakMap).get(from), desc;
34
+ if (entry)
35
+ return entry;
36
+ entry = __defProp({}, "__esModule", { value: true });
12
37
  if (from && typeof from === "object" || typeof from === "function") {
13
- for (let key of __getOwnPropNames(from))
14
- if (!__hasOwnProp.call(to, key) && key !== except)
15
- __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
38
+ for (var key of __getOwnPropNames(from))
39
+ if (!__hasOwnProp.call(entry, key))
40
+ __defProp(entry, key, {
41
+ get: __accessProp.bind(from, key),
42
+ enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable
43
+ });
16
44
  }
17
- return to;
45
+ __moduleCache.set(from, entry);
46
+ return entry;
47
+ };
48
+ var __moduleCache;
49
+ var __returnValue = (v) => v;
50
+ function __exportSetter(name, newValue) {
51
+ this[name] = __returnValue.bind(null, newValue);
52
+ }
53
+ var __export = (target, all) => {
54
+ for (var name in all)
55
+ __defProp(target, name, {
56
+ get: all[name],
57
+ enumerable: true,
58
+ configurable: true,
59
+ set: __exportSetter.bind(all, name)
60
+ });
18
61
  };
19
- var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
20
- // If the importer is in node compatibility mode or this is not an ESM
21
- // file that has been converted to a CommonJS file using a Babel-
22
- // compatible transform (i.e. "__esModule" has not been set), then set
23
- // "default" to the CommonJS "module.exports" for node compatibility.
24
- isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
25
- mod
26
- ));
27
- var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
28
62
 
29
63
  // src/index.ts
30
- var src_exports = {};
31
- __export(src_exports, {
64
+ var exports_src = {};
65
+ __export(exports_src, {
32
66
  expandMyType: () => expandMyType
33
67
  });
34
- module.exports = __toCommonJS(src_exports);
68
+ module.exports = __toCommonJS(exports_src);
35
69
 
36
70
  // src/augmenter-compiler-host.ts
37
- var import_typescript = __toESM(require("typescript"), 1);
71
+ var import_typescript = __toESM(require("typescript"));
38
72
  var createAugmenterCompilerHost = (sourceFileName, codeToAdd, compilerOptions, compilerHostFunctionOverrides) => {
39
73
  const customCompilerHost = import_typescript.default.createCompilerHost(compilerOptions ?? {}, true);
40
- for (const key of Object.keys(compilerHostFunctionOverrides ?? {})) {
41
- customCompilerHost[key] = compilerHostFunctionOverrides[key];
74
+ const overrides = compilerHostFunctionOverrides ?? {};
75
+ for (const key of Object.keys(overrides)) {
76
+ const override = overrides[key];
77
+ if (override) {
78
+ customCompilerHost[key] = override;
79
+ }
42
80
  }
43
81
  const originalReadFile = customCompilerHost.readFile;
44
82
  customCompilerHost.readFile = (fileName) => {
45
83
  const contents = originalReadFile(fileName);
46
- if (contents === void 0) {
84
+ if (contents === undefined) {
47
85
  return contents;
48
86
  }
49
87
  if (fileName !== sourceFileName) {
@@ -75,22 +113,19 @@ var createExpandCodeBlock = (typeExpression) => {
75
113
  type ${identifierPrefix}RemoveUnderscore<T extends string> = T extends \`\${infer U}_\` ? U : never;`;
76
114
  };
77
115
  var formatTypeExpression = async (code, prettierOptions) => {
78
- return (await (0, import_prettier.format)(
79
- `type ${identifierPrefix} = ${code}`,
80
- prettierOptions ?? {
81
- parser: "typescript",
82
- semi: false
83
- }
84
- )).trim().substring(`type ${identifierPrefix} = `.length);
116
+ return (await import_prettier.format(`type ${identifierPrefix} = ${code}`, prettierOptions ?? {
117
+ parser: "typescript",
118
+ semi: false
119
+ })).trim().substring(`type ${identifierPrefix} = `.length);
85
120
  };
86
121
 
87
122
  // src/index.ts
88
- var import_node_path = __toESM(require("path"), 1);
89
- var import_typescript2 = __toESM(require("typescript"), 1);
123
+ var import_node_path = __toESM(require("node:path"));
124
+ var import_typescript2 = __toESM(require("typescript"));
90
125
  var findResultIdentifierNode = (node) => {
91
126
  if (node.getChildCount() == 0) {
92
127
  if (!import_typescript2.default.isIdentifier(node)) {
93
- return void 0;
128
+ return;
94
129
  }
95
130
  return node;
96
131
  }
@@ -129,17 +164,8 @@ async function expandMyType(options) {
129
164
  if (!tsCompilerOptions.strictNullChecks) {
130
165
  throw new Error("strictNullChecks must be enabled!");
131
166
  }
132
- const compilerHost = createAugmenterCompilerHost(
133
- resolvedSourceFileName,
134
- createExpandCodeBlock(options.typeExpression),
135
- tsCompilerOptions,
136
- options.compilerHostFunctionOverrides
137
- );
138
- const program = import_typescript2.default.createProgram(
139
- [resolvedSourceFileName],
140
- tsCompilerOptions,
141
- compilerHost
142
- );
167
+ const compilerHost = createAugmenterCompilerHost(resolvedSourceFileName, createExpandCodeBlock(options.typeExpression), tsCompilerOptions, options.compilerHostFunctionOverrides);
168
+ const program = import_typescript2.default.createProgram([resolvedSourceFileName], tsCompilerOptions, compilerHost);
143
169
  const sourceFile = program.getSourceFile(resolvedSourceFileName);
144
170
  if (!sourceFile) {
145
171
  throw new Error("Source file not found!");
@@ -149,18 +175,12 @@ async function expandMyType(options) {
149
175
  throw new Error("No node found!");
150
176
  }
151
177
  const typeChecker = program.getTypeChecker();
152
- const expandedTypeString = typeChecker.typeToString(
153
- typeChecker.getTypeAtLocation(resultIdentifierNode),
154
- void 0,
155
- import_typescript2.default.TypeFormatFlags.NodeBuilderFlagsMask
156
- );
178
+ const expandedTypeString = typeChecker.typeToString(typeChecker.getTypeAtLocation(resultIdentifierNode), undefined, import_typescript2.default.TypeFormatFlags.NodeBuilderFlagsMask);
157
179
  if (options.prettify && options.prettify.enabled === false) {
158
180
  return expandedTypeString;
159
181
  }
160
182
  return formatTypeExpression(expandedTypeString, options.prettify?.options);
161
183
  }
162
- // Annotate the CommonJS export names for ESM import in node:
163
- 0 && (module.exports = {
164
- expandMyType
165
- });
166
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/index.ts", "../src/augmenter-compiler-host.ts", "../src/code-generator.ts"],
  "sourcesContent": ["import { createAugmenterCompilerHost } from \"./augmenter-compiler-host.ts\";\nimport { type CompilerHostFunctionOverrides } from \"./augmenter-compiler-host.ts\";\nimport {\n  createExpandCodeBlock,\n  formatTypeExpression,\n} from \"./code-generator.ts\";\nimport path from \"node:path\";\nimport type { Options as PrettierOptions } from \"prettier\";\nimport ts from \"typescript\";\n\n/**\n * Finds the result type identifier node.\n *\n * @param node Node in which type type should be searched.\n * @returns The result type identifier node.\n */\nconst findResultIdentifierNode = (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 __<IDENTIFIER>__ 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, findResultIdentifierNode);\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\n  /**\n   * A record of functions to override in the compiler host. Useful for mocking\n   */\n  compilerHostFunctionOverrides?: CompilerHostFunctionOverrides;\n};\nexport type ExpandTypeFromSourceFileOptions = ExpandTypeOptionsBase & {\n  /**\n   * Name of the source file to evaluate the type expression in.\n   */\n  sourceFileName: string;\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 dummyFileName = \"expand-my-type-dummy.ts\";\n\n    return expandMyType({\n      sourceFileName: dummyFileName,\n      typeExpression: options.typeExpression,\n      compilerHostFunctionOverrides: {\n        readFile(fileName: string) {\n          if (path.basename(fileName) === dummyFileName) {\n            return options.sourceText;\n          }\n\n          return ts.sys.readFile(fileName);\n        },\n      },\n      tsCompilerOptions: options.tsCompilerOptions,\n      prettify: options.prettify,\n    });\n  }\n\n  const resolvedSourceFileName = path.resolve(options.sourceFileName);\n\n  const tsCompilerOptions = options.tsCompilerOptions ?? {\n    noEmit: true,\n\n    strictNullChecks: true,\n    allowSyntheticDefaultImports: true,\n    allowArbitraryExtensions: true,\n    allowImportingTsExtensions: true,\n    allowJs: true,\n  };\n\n  if (!tsCompilerOptions.strictNullChecks) {\n    throw new Error(\"strictNullChecks must be enabled!\");\n  }\n\n  const compilerHost = createAugmenterCompilerHost(\n    resolvedSourceFileName,\n    createExpandCodeBlock(options.typeExpression),\n    tsCompilerOptions,\n    options.compilerHostFunctionOverrides,\n  );\n\n  const program = ts.createProgram(\n    [resolvedSourceFileName],\n    tsCompilerOptions,\n    compilerHost,\n  );\n\n  const sourceFile = program.getSourceFile(resolvedSourceFileName);\n  if (!sourceFile) {\n    throw new Error(\"Source file not found!\");\n  }\n\n  const resultIdentifierNode = findResultIdentifierNode(sourceFile);\n  if (!resultIdentifierNode) {\n    throw new Error(\"No node found!\");\n  }\n\n  const typeChecker = program.getTypeChecker();\n  const expandedTypeString = typeChecker.typeToString(\n    typeChecker.getTypeAtLocation(resultIdentifierNode),\n    undefined,\n    ts.TypeFormatFlags.NodeBuilderFlagsMask,\n  );\n\n  if (options.prettify && options.prettify.enabled === false) {\n    return expandedTypeString;\n  }\n\n  return formatTypeExpression(expandedTypeString, options.prettify?.options);\n}\n", "import ts from \"typescript\";\n\ntype ExtractFunctions<T, K extends keyof T = keyof T> = {\n  [P in K]: T[P] extends (...args: any[]) => any ? T[P] : never;\n};\nexport type CompilerHostFunctionOverrides = Partial<\n  ExtractFunctions<ts.CompilerHost>\n>;\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 codeToAdd Type expression.\n * @param compilerOptions TypeScript compiler options.\n * @param compilerHostFunctionOverrides A record of functions to override in the compiler host. Useful for mocking.\n * @returns A custom compiler host that returns an augmented source file that can be used to expand the type expression.\n */\nexport const createAugmenterCompilerHost = (\n  sourceFileName: string,\n  codeToAdd: string,\n  compilerOptions?: ts.CompilerOptions,\n  compilerHostFunctionOverrides?: CompilerHostFunctionOverrides,\n) => {\n  const customCompilerHost = ts.createCompilerHost(compilerOptions ?? {}, true);\n  for (const key of Object.keys(compilerHostFunctionOverrides ?? {})) {\n    customCompilerHost[key] = compilerHostFunctionOverrides![key];\n  }\n\n  const originalReadFile = customCompilerHost.readFile;\n\n  customCompilerHost.readFile = (fileName) => {\n    const contents = originalReadFile(fileName);\n\n    if (contents === undefined) {\n      return contents;\n    }\n\n    if (fileName !== sourceFileName) {\n      return contents;\n    }\n\n    return `${codeToAdd}\\n${contents}`;\n  };\n\n  return customCompilerHost;\n};\n", "import type { Options as PrettierOptions } from \"prettier\";\nimport { format } from \"prettier\";\n\nconst identifierPrefix = \"__EXPAND_MY_TYPE__\";\n\nexport const createExpandCodeBlock = (typeExpression: string) => {\n  // https://github.com/microsoft/TypeScript/blob/main/tests/cases/compiler/computedTypesKeyofNoIndexSignatureType.ts\n  return `type ${identifierPrefix}Result = ${identifierPrefix}Expand<${identifierPrefix}Expression>;\n    type ${identifierPrefix}Expression = ${typeExpression};\n\n    type ${identifierPrefix}Expand<T> = \n        T extends (...args: infer A) => infer R ? (...args: ${identifierPrefix}Expand<A>) => ${identifierPrefix}Expand<R>\n      : T extends Promise<infer U> ? Promise<${identifierPrefix}ExpandTypeArgument<U>>\n      : { [K in keyof T]: T[K] extends string ? ${identifierPrefix}ExpandString<T[K]> : ${identifierPrefix}Expand<T[K]>; } & {};\n\n    type ${identifierPrefix}ExpandTypeArgument<T> = [T & {}] extends [never] ? T : T & {} extends void ? T : ${identifierPrefix}Expand<T & {}>;\n\n    // Forces a union of string literal types to be expanded\n    type ${identifierPrefix}ExpandString<T extends string> = ${identifierPrefix}RemoveUnderscore<${identifierPrefix}AppendUnderscore<T>>;\n    type ${identifierPrefix}AppendUnderscore<T extends string> = \\`\\${T}_\\` extends string ? \\`\\${T}_\\` : never;\n    type ${identifierPrefix}RemoveUnderscore<T extends string> = T extends \\`\\${infer U}_\\` ? U : never;`;\n};\n\nexport const formatTypeExpression = async (\n  code: string,\n  prettierOptions?: PrettierOptions,\n) => {\n  return (\n    await format(\n      `type ${identifierPrefix} = ${code}`,\n      prettierOptions ?? {\n        parser: \"typescript\",\n        semi: false,\n      },\n    )\n  )\n    .trim()\n    .substring(`type ${identifierPrefix} = `.length);\n};\n"],
  "mappings": "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",
  "names": ["ts", "import_typescript", "ts", "path"]
}

184
+
185
+ //# debugId=FD61D95A1BB3AC7F64756E2164756E21
186
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["src/augmenter-compiler-host.ts", "src/code-generator.ts", "src/index.ts"],
  "sourcesContent": [
    "import ts from \"typescript\";\n\ntype ExtractFunctions<T, K extends keyof T = keyof T> = {\n  [P in K]: T[P] extends (...args: any[]) => any ? T[P] : never;\n};\nexport type CompilerHostFunctionOverrides = Partial<\n  ExtractFunctions<ts.CompilerHost>\n>;\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 codeToAdd Type expression.\n * @param compilerOptions TypeScript compiler options.\n * @param compilerHostFunctionOverrides A record of functions to override in the compiler host. Useful for mocking.\n * @returns A custom compiler host that returns an augmented source file that can be used to expand the type expression.\n */\nexport const createAugmenterCompilerHost = (\n  sourceFileName: string,\n  codeToAdd: string,\n  compilerOptions?: ts.CompilerOptions,\n  compilerHostFunctionOverrides?: CompilerHostFunctionOverrides,\n) => {\n  const customCompilerHost = ts.createCompilerHost(compilerOptions ?? {}, true);\n  const overrides = compilerHostFunctionOverrides ?? {};\n\n  for (const key of Object.keys(overrides) as Array<\n    keyof CompilerHostFunctionOverrides\n  >) {\n    const override = overrides[key];\n\n    if (override) {\n      (customCompilerHost as unknown as Record<string, unknown>)[key] =\n        override;\n    }\n  }\n\n  const originalReadFile = customCompilerHost.readFile;\n\n  customCompilerHost.readFile = (fileName) => {\n    const contents = originalReadFile(fileName);\n\n    if (contents === undefined) {\n      return contents;\n    }\n\n    if (fileName !== sourceFileName) {\n      return contents;\n    }\n\n    return `${codeToAdd}\\n${contents}`;\n  };\n\n  return customCompilerHost;\n};\n",
    "import type { Options as PrettierOptions } from \"prettier\";\nimport { format } from \"prettier\";\n\nconst identifierPrefix = \"__EXPAND_MY_TYPE__\";\n\nexport const createExpandCodeBlock = (typeExpression: string) => {\n  // https://github.com/microsoft/TypeScript/blob/main/tests/cases/compiler/computedTypesKeyofNoIndexSignatureType.ts\n  return `type ${identifierPrefix}Result = ${identifierPrefix}Expand<${identifierPrefix}Expression>;\n    type ${identifierPrefix}Expression = ${typeExpression};\n\n    type ${identifierPrefix}Expand<T> = \n        T extends (...args: infer A) => infer R ? (...args: ${identifierPrefix}Expand<A>) => ${identifierPrefix}Expand<R>\n      : T extends Promise<infer U> ? Promise<${identifierPrefix}ExpandTypeArgument<U>>\n      : { [K in keyof T]: T[K] extends string ? ${identifierPrefix}ExpandString<T[K]> : ${identifierPrefix}Expand<T[K]>; } & {};\n\n    type ${identifierPrefix}ExpandTypeArgument<T> = [T & {}] extends [never] ? T : T & {} extends void ? T : ${identifierPrefix}Expand<T & {}>;\n\n    // Forces a union of string literal types to be expanded\n    type ${identifierPrefix}ExpandString<T extends string> = ${identifierPrefix}RemoveUnderscore<${identifierPrefix}AppendUnderscore<T>>;\n    type ${identifierPrefix}AppendUnderscore<T extends string> = \\`\\${T}_\\` extends string ? \\`\\${T}_\\` : never;\n    type ${identifierPrefix}RemoveUnderscore<T extends string> = T extends \\`\\${infer U}_\\` ? U : never;`;\n};\n\nexport const formatTypeExpression = async (\n  code: string,\n  prettierOptions?: PrettierOptions,\n) => {\n  return (\n    await format(\n      `type ${identifierPrefix} = ${code}`,\n      prettierOptions ?? {\n        parser: \"typescript\",\n        semi: false,\n      },\n    )\n  )\n    .trim()\n    .substring(`type ${identifierPrefix} = `.length);\n};\n",
    "import { createAugmenterCompilerHost } from \"./augmenter-compiler-host.js\";\nimport { type CompilerHostFunctionOverrides } from \"./augmenter-compiler-host.js\";\nimport {\n  createExpandCodeBlock,\n  formatTypeExpression,\n} from \"./code-generator.js\";\nimport path from \"node:path\";\nimport type { Options as PrettierOptions } from \"prettier\";\nimport ts from \"typescript\";\n\n/**\n * Finds the result type identifier node.\n *\n * @param node Node in which type type should be searched.\n * @returns The result type identifier node.\n */\nconst findResultIdentifierNode = (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 __<IDENTIFIER>__ 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, findResultIdentifierNode);\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\n  /**\n   * A record of functions to override in the compiler host. Useful for mocking\n   */\n  compilerHostFunctionOverrides?: CompilerHostFunctionOverrides;\n};\nexport type ExpandTypeFromSourceFileOptions = ExpandTypeOptionsBase & {\n  /**\n   * Name of the source file to evaluate the type expression in.\n   */\n  sourceFileName: string;\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 dummyFileName = \"expand-my-type-dummy.ts\";\n\n    return expandMyType({\n      sourceFileName: dummyFileName,\n      typeExpression: options.typeExpression,\n      compilerHostFunctionOverrides: {\n        readFile(fileName: string) {\n          if (path.basename(fileName) === dummyFileName) {\n            return options.sourceText;\n          }\n\n          return ts.sys.readFile(fileName);\n        },\n      },\n      tsCompilerOptions: options.tsCompilerOptions,\n      prettify: options.prettify,\n    });\n  }\n\n  const resolvedSourceFileName = path.resolve(options.sourceFileName);\n\n  const tsCompilerOptions = options.tsCompilerOptions ?? {\n    noEmit: true,\n\n    strictNullChecks: true,\n    allowSyntheticDefaultImports: true,\n    allowArbitraryExtensions: true,\n    allowImportingTsExtensions: true,\n    allowJs: true,\n  };\n\n  if (!tsCompilerOptions.strictNullChecks) {\n    throw new Error(\"strictNullChecks must be enabled!\");\n  }\n\n  const compilerHost = createAugmenterCompilerHost(\n    resolvedSourceFileName,\n    createExpandCodeBlock(options.typeExpression),\n    tsCompilerOptions,\n    options.compilerHostFunctionOverrides,\n  );\n\n  const program = ts.createProgram(\n    [resolvedSourceFileName],\n    tsCompilerOptions,\n    compilerHost,\n  );\n\n  const sourceFile = program.getSourceFile(resolvedSourceFileName);\n  if (!sourceFile) {\n    throw new Error(\"Source file not found!\");\n  }\n\n  const resultIdentifierNode = findResultIdentifierNode(sourceFile);\n  if (!resultIdentifierNode) {\n    throw new Error(\"No node found!\");\n  }\n\n  const typeChecker = program.getTypeChecker();\n  const expandedTypeString = typeChecker.typeToString(\n    typeChecker.getTypeAtLocation(resultIdentifierNode),\n    undefined,\n    ts.TypeFormatFlags.NodeBuilderFlagsMask,\n  );\n\n  if (options.prettify && options.prettify.enabled === false) {\n    return expandedTypeString;\n  }\n\n  return formatTypeExpression(expandedTypeString, options.prettify?.options);\n}\n"
  ],
  "mappings": "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",
  "debugId": "FD61D95A1BB3AC7F64756E2164756E21",
  "names": []
}
package/dist/index.d.ts CHANGED
@@ -1,12 +1,7 @@
1
- import ts from 'typescript';
2
- import { Options } from 'prettier';
3
-
4
- type ExtractFunctions<T, K extends keyof T = keyof T> = {
5
- [P in K]: T[P] extends (...args: any[]) => any ? T[P] : never;
6
- };
7
- type CompilerHostFunctionOverrides = Partial<ExtractFunctions<ts.CompilerHost>>;
8
-
9
- type ExpandTypeOptionsBase = {
1
+ import { type CompilerHostFunctionOverrides } from "./augmenter-compiler-host.js";
2
+ import type { Options as PrettierOptions } from "prettier";
3
+ import ts from "typescript";
4
+ export type ExpandTypeOptionsBase = {
10
5
  /**
11
6
  * The type expression to expand.
12
7
  * @example "ReturnType<typeof myFunction>"
@@ -29,27 +24,25 @@ type ExpandTypeOptionsBase = {
29
24
  * Prettier options. Don't forget to set the parser to "typescript".
30
25
  * @default { parser: "typescript", semi: false }
31
26
  */
32
- options?: Options;
27
+ options?: PrettierOptions;
33
28
  };
34
29
  /**
35
30
  * A record of functions to override in the compiler host. Useful for mocking
36
31
  */
37
32
  compilerHostFunctionOverrides?: CompilerHostFunctionOverrides;
38
33
  };
39
- type ExpandTypeFromSourceFileOptions = ExpandTypeOptionsBase & {
34
+ export type ExpandTypeFromSourceFileOptions = ExpandTypeOptionsBase & {
40
35
  /**
41
36
  * Name of the source file to evaluate the type expression in.
42
37
  */
43
38
  sourceFileName: string;
44
39
  };
45
- type ExpandTypeFromSourceTextOptions = ExpandTypeOptionsBase & {
40
+ export type ExpandTypeFromSourceTextOptions = ExpandTypeOptionsBase & {
46
41
  /**
47
42
  * TypeScript source text to evaluate the type expression in.
48
43
  */
49
44
  sourceText: string;
50
45
  };
51
- type ExpandMyTypeOptions = ExpandTypeFromSourceFileOptions | ExpandTypeFromSourceTextOptions;
52
- declare function expandMyType(options: ExpandTypeFromSourceTextOptions): Promise<string>;
53
- declare function expandMyType(options: ExpandTypeFromSourceFileOptions): Promise<string>;
54
-
55
- export { type ExpandMyTypeOptions, type ExpandTypeFromSourceFileOptions, type ExpandTypeFromSourceTextOptions, type ExpandTypeOptionsBase, expandMyType };
46
+ export type ExpandMyTypeOptions = ExpandTypeFromSourceFileOptions | ExpandTypeFromSourceTextOptions;
47
+ export declare function expandMyType(options: ExpandTypeFromSourceTextOptions): Promise<string>;
48
+ export declare function expandMyType(options: ExpandTypeFromSourceFileOptions): Promise<string>;
package/dist/index.js CHANGED
@@ -2,13 +2,17 @@
2
2
  import ts from "typescript";
3
3
  var createAugmenterCompilerHost = (sourceFileName, codeToAdd, compilerOptions, compilerHostFunctionOverrides) => {
4
4
  const customCompilerHost = ts.createCompilerHost(compilerOptions ?? {}, true);
5
- for (const key of Object.keys(compilerHostFunctionOverrides ?? {})) {
6
- customCompilerHost[key] = compilerHostFunctionOverrides[key];
5
+ const overrides = compilerHostFunctionOverrides ?? {};
6
+ for (const key of Object.keys(overrides)) {
7
+ const override = overrides[key];
8
+ if (override) {
9
+ customCompilerHost[key] = override;
10
+ }
7
11
  }
8
12
  const originalReadFile = customCompilerHost.readFile;
9
13
  customCompilerHost.readFile = (fileName) => {
10
14
  const contents = originalReadFile(fileName);
11
- if (contents === void 0) {
15
+ if (contents === undefined) {
12
16
  return contents;
13
17
  }
14
18
  if (fileName !== sourceFileName) {
@@ -40,13 +44,10 @@ var createExpandCodeBlock = (typeExpression) => {
40
44
  type ${identifierPrefix}RemoveUnderscore<T extends string> = T extends \`\${infer U}_\` ? U : never;`;
41
45
  };
42
46
  var formatTypeExpression = async (code, prettierOptions) => {
43
- return (await format(
44
- `type ${identifierPrefix} = ${code}`,
45
- prettierOptions ?? {
46
- parser: "typescript",
47
- semi: false
48
- }
49
- )).trim().substring(`type ${identifierPrefix} = `.length);
47
+ return (await format(`type ${identifierPrefix} = ${code}`, prettierOptions ?? {
48
+ parser: "typescript",
49
+ semi: false
50
+ })).trim().substring(`type ${identifierPrefix} = `.length);
50
51
  };
51
52
 
52
53
  // src/index.ts
@@ -55,7 +56,7 @@ import ts2 from "typescript";
55
56
  var findResultIdentifierNode = (node) => {
56
57
  if (node.getChildCount() == 0) {
57
58
  if (!ts2.isIdentifier(node)) {
58
- return void 0;
59
+ return;
59
60
  }
60
61
  return node;
61
62
  }
@@ -94,17 +95,8 @@ async function expandMyType(options) {
94
95
  if (!tsCompilerOptions.strictNullChecks) {
95
96
  throw new Error("strictNullChecks must be enabled!");
96
97
  }
97
- const compilerHost = createAugmenterCompilerHost(
98
- resolvedSourceFileName,
99
- createExpandCodeBlock(options.typeExpression),
100
- tsCompilerOptions,
101
- options.compilerHostFunctionOverrides
102
- );
103
- const program = ts2.createProgram(
104
- [resolvedSourceFileName],
105
- tsCompilerOptions,
106
- compilerHost
107
- );
98
+ const compilerHost = createAugmenterCompilerHost(resolvedSourceFileName, createExpandCodeBlock(options.typeExpression), tsCompilerOptions, options.compilerHostFunctionOverrides);
99
+ const program = ts2.createProgram([resolvedSourceFileName], tsCompilerOptions, compilerHost);
108
100
  const sourceFile = program.getSourceFile(resolvedSourceFileName);
109
101
  if (!sourceFile) {
110
102
  throw new Error("Source file not found!");
@@ -114,11 +106,7 @@ async function expandMyType(options) {
114
106
  throw new Error("No node found!");
115
107
  }
116
108
  const typeChecker = program.getTypeChecker();
117
- const expandedTypeString = typeChecker.typeToString(
118
- typeChecker.getTypeAtLocation(resultIdentifierNode),
119
- void 0,
120
- ts2.TypeFormatFlags.NodeBuilderFlagsMask
121
- );
109
+ const expandedTypeString = typeChecker.typeToString(typeChecker.getTypeAtLocation(resultIdentifierNode), undefined, ts2.TypeFormatFlags.NodeBuilderFlagsMask);
122
110
  if (options.prettify && options.prettify.enabled === false) {
123
111
  return expandedTypeString;
124
112
  }
@@ -127,4 +115,6 @@ async function expandMyType(options) {
127
115
  export {
128
116
  expandMyType
129
117
  };
130
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/augmenter-compiler-host.ts", "../src/code-generator.ts", "../src/index.ts"],
  "sourcesContent": ["import ts from \"typescript\";\n\ntype ExtractFunctions<T, K extends keyof T = keyof T> = {\n  [P in K]: T[P] extends (...args: any[]) => any ? T[P] : never;\n};\nexport type CompilerHostFunctionOverrides = Partial<\n  ExtractFunctions<ts.CompilerHost>\n>;\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 codeToAdd Type expression.\n * @param compilerOptions TypeScript compiler options.\n * @param compilerHostFunctionOverrides A record of functions to override in the compiler host. Useful for mocking.\n * @returns A custom compiler host that returns an augmented source file that can be used to expand the type expression.\n */\nexport const createAugmenterCompilerHost = (\n  sourceFileName: string,\n  codeToAdd: string,\n  compilerOptions?: ts.CompilerOptions,\n  compilerHostFunctionOverrides?: CompilerHostFunctionOverrides,\n) => {\n  const customCompilerHost = ts.createCompilerHost(compilerOptions ?? {}, true);\n  for (const key of Object.keys(compilerHostFunctionOverrides ?? {})) {\n    customCompilerHost[key] = compilerHostFunctionOverrides![key];\n  }\n\n  const originalReadFile = customCompilerHost.readFile;\n\n  customCompilerHost.readFile = (fileName) => {\n    const contents = originalReadFile(fileName);\n\n    if (contents === undefined) {\n      return contents;\n    }\n\n    if (fileName !== sourceFileName) {\n      return contents;\n    }\n\n    return `${codeToAdd}\\n${contents}`;\n  };\n\n  return customCompilerHost;\n};\n", "import type { Options as PrettierOptions } from \"prettier\";\nimport { format } from \"prettier\";\n\nconst identifierPrefix = \"__EXPAND_MY_TYPE__\";\n\nexport const createExpandCodeBlock = (typeExpression: string) => {\n  // https://github.com/microsoft/TypeScript/blob/main/tests/cases/compiler/computedTypesKeyofNoIndexSignatureType.ts\n  return `type ${identifierPrefix}Result = ${identifierPrefix}Expand<${identifierPrefix}Expression>;\n    type ${identifierPrefix}Expression = ${typeExpression};\n\n    type ${identifierPrefix}Expand<T> = \n        T extends (...args: infer A) => infer R ? (...args: ${identifierPrefix}Expand<A>) => ${identifierPrefix}Expand<R>\n      : T extends Promise<infer U> ? Promise<${identifierPrefix}ExpandTypeArgument<U>>\n      : { [K in keyof T]: T[K] extends string ? ${identifierPrefix}ExpandString<T[K]> : ${identifierPrefix}Expand<T[K]>; } & {};\n\n    type ${identifierPrefix}ExpandTypeArgument<T> = [T & {}] extends [never] ? T : T & {} extends void ? T : ${identifierPrefix}Expand<T & {}>;\n\n    // Forces a union of string literal types to be expanded\n    type ${identifierPrefix}ExpandString<T extends string> = ${identifierPrefix}RemoveUnderscore<${identifierPrefix}AppendUnderscore<T>>;\n    type ${identifierPrefix}AppendUnderscore<T extends string> = \\`\\${T}_\\` extends string ? \\`\\${T}_\\` : never;\n    type ${identifierPrefix}RemoveUnderscore<T extends string> = T extends \\`\\${infer U}_\\` ? U : never;`;\n};\n\nexport const formatTypeExpression = async (\n  code: string,\n  prettierOptions?: PrettierOptions,\n) => {\n  return (\n    await format(\n      `type ${identifierPrefix} = ${code}`,\n      prettierOptions ?? {\n        parser: \"typescript\",\n        semi: false,\n      },\n    )\n  )\n    .trim()\n    .substring(`type ${identifierPrefix} = `.length);\n};\n", "import { createAugmenterCompilerHost } from \"./augmenter-compiler-host.ts\";\nimport { type CompilerHostFunctionOverrides } from \"./augmenter-compiler-host.ts\";\nimport {\n  createExpandCodeBlock,\n  formatTypeExpression,\n} from \"./code-generator.ts\";\nimport path from \"node:path\";\nimport type { Options as PrettierOptions } from \"prettier\";\nimport ts from \"typescript\";\n\n/**\n * Finds the result type identifier node.\n *\n * @param node Node in which type type should be searched.\n * @returns The result type identifier node.\n */\nconst findResultIdentifierNode = (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 __<IDENTIFIER>__ 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, findResultIdentifierNode);\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\n  /**\n   * A record of functions to override in the compiler host. Useful for mocking\n   */\n  compilerHostFunctionOverrides?: CompilerHostFunctionOverrides;\n};\nexport type ExpandTypeFromSourceFileOptions = ExpandTypeOptionsBase & {\n  /**\n   * Name of the source file to evaluate the type expression in.\n   */\n  sourceFileName: string;\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 dummyFileName = \"expand-my-type-dummy.ts\";\n\n    return expandMyType({\n      sourceFileName: dummyFileName,\n      typeExpression: options.typeExpression,\n      compilerHostFunctionOverrides: {\n        readFile(fileName: string) {\n          if (path.basename(fileName) === dummyFileName) {\n            return options.sourceText;\n          }\n\n          return ts.sys.readFile(fileName);\n        },\n      },\n      tsCompilerOptions: options.tsCompilerOptions,\n      prettify: options.prettify,\n    });\n  }\n\n  const resolvedSourceFileName = path.resolve(options.sourceFileName);\n\n  const tsCompilerOptions = options.tsCompilerOptions ?? {\n    noEmit: true,\n\n    strictNullChecks: true,\n    allowSyntheticDefaultImports: true,\n    allowArbitraryExtensions: true,\n    allowImportingTsExtensions: true,\n    allowJs: true,\n  };\n\n  if (!tsCompilerOptions.strictNullChecks) {\n    throw new Error(\"strictNullChecks must be enabled!\");\n  }\n\n  const compilerHost = createAugmenterCompilerHost(\n    resolvedSourceFileName,\n    createExpandCodeBlock(options.typeExpression),\n    tsCompilerOptions,\n    options.compilerHostFunctionOverrides,\n  );\n\n  const program = ts.createProgram(\n    [resolvedSourceFileName],\n    tsCompilerOptions,\n    compilerHost,\n  );\n\n  const sourceFile = program.getSourceFile(resolvedSourceFileName);\n  if (!sourceFile) {\n    throw new Error(\"Source file not found!\");\n  }\n\n  const resultIdentifierNode = findResultIdentifierNode(sourceFile);\n  if (!resultIdentifierNode) {\n    throw new Error(\"No node found!\");\n  }\n\n  const typeChecker = program.getTypeChecker();\n  const expandedTypeString = typeChecker.typeToString(\n    typeChecker.getTypeAtLocation(resultIdentifierNode),\n    undefined,\n    ts.TypeFormatFlags.NodeBuilderFlagsMask,\n  );\n\n  if (options.prettify && options.prettify.enabled === false) {\n    return expandedTypeString;\n  }\n\n  return formatTypeExpression(expandedTypeString, options.prettify?.options);\n}\n"],
  "mappings": "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",
  "names": ["ts"]
}

118
+
119
+ //# debugId=FA40E2FF4FF7499A64756E2164756E21
120
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/augmenter-compiler-host.ts", "../src/code-generator.ts", "../src/index.ts"],
  "sourcesContent": [
    "import ts from \"typescript\";\n\ntype ExtractFunctions<T, K extends keyof T = keyof T> = {\n  [P in K]: T[P] extends (...args: any[]) => any ? T[P] : never;\n};\nexport type CompilerHostFunctionOverrides = Partial<\n  ExtractFunctions<ts.CompilerHost>\n>;\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 codeToAdd Type expression.\n * @param compilerOptions TypeScript compiler options.\n * @param compilerHostFunctionOverrides A record of functions to override in the compiler host. Useful for mocking.\n * @returns A custom compiler host that returns an augmented source file that can be used to expand the type expression.\n */\nexport const createAugmenterCompilerHost = (\n  sourceFileName: string,\n  codeToAdd: string,\n  compilerOptions?: ts.CompilerOptions,\n  compilerHostFunctionOverrides?: CompilerHostFunctionOverrides,\n) => {\n  const customCompilerHost = ts.createCompilerHost(compilerOptions ?? {}, true);\n  const overrides = compilerHostFunctionOverrides ?? {};\n\n  for (const key of Object.keys(overrides) as Array<\n    keyof CompilerHostFunctionOverrides\n  >) {\n    const override = overrides[key];\n\n    if (override) {\n      (customCompilerHost as unknown as Record<string, unknown>)[key] =\n        override;\n    }\n  }\n\n  const originalReadFile = customCompilerHost.readFile;\n\n  customCompilerHost.readFile = (fileName) => {\n    const contents = originalReadFile(fileName);\n\n    if (contents === undefined) {\n      return contents;\n    }\n\n    if (fileName !== sourceFileName) {\n      return contents;\n    }\n\n    return `${codeToAdd}\\n${contents}`;\n  };\n\n  return customCompilerHost;\n};\n",
    "import type { Options as PrettierOptions } from \"prettier\";\nimport { format } from \"prettier\";\n\nconst identifierPrefix = \"__EXPAND_MY_TYPE__\";\n\nexport const createExpandCodeBlock = (typeExpression: string) => {\n  // https://github.com/microsoft/TypeScript/blob/main/tests/cases/compiler/computedTypesKeyofNoIndexSignatureType.ts\n  return `type ${identifierPrefix}Result = ${identifierPrefix}Expand<${identifierPrefix}Expression>;\n    type ${identifierPrefix}Expression = ${typeExpression};\n\n    type ${identifierPrefix}Expand<T> = \n        T extends (...args: infer A) => infer R ? (...args: ${identifierPrefix}Expand<A>) => ${identifierPrefix}Expand<R>\n      : T extends Promise<infer U> ? Promise<${identifierPrefix}ExpandTypeArgument<U>>\n      : { [K in keyof T]: T[K] extends string ? ${identifierPrefix}ExpandString<T[K]> : ${identifierPrefix}Expand<T[K]>; } & {};\n\n    type ${identifierPrefix}ExpandTypeArgument<T> = [T & {}] extends [never] ? T : T & {} extends void ? T : ${identifierPrefix}Expand<T & {}>;\n\n    // Forces a union of string literal types to be expanded\n    type ${identifierPrefix}ExpandString<T extends string> = ${identifierPrefix}RemoveUnderscore<${identifierPrefix}AppendUnderscore<T>>;\n    type ${identifierPrefix}AppendUnderscore<T extends string> = \\`\\${T}_\\` extends string ? \\`\\${T}_\\` : never;\n    type ${identifierPrefix}RemoveUnderscore<T extends string> = T extends \\`\\${infer U}_\\` ? U : never;`;\n};\n\nexport const formatTypeExpression = async (\n  code: string,\n  prettierOptions?: PrettierOptions,\n) => {\n  return (\n    await format(\n      `type ${identifierPrefix} = ${code}`,\n      prettierOptions ?? {\n        parser: \"typescript\",\n        semi: false,\n      },\n    )\n  )\n    .trim()\n    .substring(`type ${identifierPrefix} = `.length);\n};\n",
    "import { createAugmenterCompilerHost } from \"./augmenter-compiler-host.js\";\nimport { type CompilerHostFunctionOverrides } from \"./augmenter-compiler-host.js\";\nimport {\n  createExpandCodeBlock,\n  formatTypeExpression,\n} from \"./code-generator.js\";\nimport path from \"node:path\";\nimport type { Options as PrettierOptions } from \"prettier\";\nimport ts from \"typescript\";\n\n/**\n * Finds the result type identifier node.\n *\n * @param node Node in which type type should be searched.\n * @returns The result type identifier node.\n */\nconst findResultIdentifierNode = (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 __<IDENTIFIER>__ 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, findResultIdentifierNode);\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\n  /**\n   * A record of functions to override in the compiler host. Useful for mocking\n   */\n  compilerHostFunctionOverrides?: CompilerHostFunctionOverrides;\n};\nexport type ExpandTypeFromSourceFileOptions = ExpandTypeOptionsBase & {\n  /**\n   * Name of the source file to evaluate the type expression in.\n   */\n  sourceFileName: string;\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 dummyFileName = \"expand-my-type-dummy.ts\";\n\n    return expandMyType({\n      sourceFileName: dummyFileName,\n      typeExpression: options.typeExpression,\n      compilerHostFunctionOverrides: {\n        readFile(fileName: string) {\n          if (path.basename(fileName) === dummyFileName) {\n            return options.sourceText;\n          }\n\n          return ts.sys.readFile(fileName);\n        },\n      },\n      tsCompilerOptions: options.tsCompilerOptions,\n      prettify: options.prettify,\n    });\n  }\n\n  const resolvedSourceFileName = path.resolve(options.sourceFileName);\n\n  const tsCompilerOptions = options.tsCompilerOptions ?? {\n    noEmit: true,\n\n    strictNullChecks: true,\n    allowSyntheticDefaultImports: true,\n    allowArbitraryExtensions: true,\n    allowImportingTsExtensions: true,\n    allowJs: true,\n  };\n\n  if (!tsCompilerOptions.strictNullChecks) {\n    throw new Error(\"strictNullChecks must be enabled!\");\n  }\n\n  const compilerHost = createAugmenterCompilerHost(\n    resolvedSourceFileName,\n    createExpandCodeBlock(options.typeExpression),\n    tsCompilerOptions,\n    options.compilerHostFunctionOverrides,\n  );\n\n  const program = ts.createProgram(\n    [resolvedSourceFileName],\n    tsCompilerOptions,\n    compilerHost,\n  );\n\n  const sourceFile = program.getSourceFile(resolvedSourceFileName);\n  if (!sourceFile) {\n    throw new Error(\"Source file not found!\");\n  }\n\n  const resultIdentifierNode = findResultIdentifierNode(sourceFile);\n  if (!resultIdentifierNode) {\n    throw new Error(\"No node found!\");\n  }\n\n  const typeChecker = program.getTypeChecker();\n  const expandedTypeString = typeChecker.typeToString(\n    typeChecker.getTypeAtLocation(resultIdentifierNode),\n    undefined,\n    ts.TypeFormatFlags.NodeBuilderFlagsMask,\n  );\n\n  if (options.prettify && options.prettify.enabled === false) {\n    return expandedTypeString;\n  }\n\n  return formatTypeExpression(expandedTypeString, options.prettify?.options);\n}\n"
  ],
  "mappings": "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",
  "debugId": "FA40E2FF4FF7499A64756E2164756E21",
  "names": []
}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "expand-my-type",
3
- "version": "0.6.7",
3
+ "version": "0.7.0",
4
4
  "description": "Expand TypeScript type expressions programmatically",
5
5
  "keywords": [
6
6
  "expand",
@@ -20,14 +20,9 @@
20
20
  "type": "module",
21
21
  "exports": {
22
22
  ".": {
23
- "require": {
24
- "default": "./dist/index.cjs",
25
- "types": "./dist/index.d.cts"
26
- },
27
- "import": {
28
- "default": "./dist/index.js",
29
- "types": "./dist/index.d.ts"
30
- }
23
+ "types": "./dist/index.d.ts",
24
+ "require": "./dist/index.cjs",
25
+ "import": "./dist/index.js"
31
26
  }
32
27
  },
33
28
  "main": "dist/index.cjs",
@@ -36,30 +31,28 @@
36
31
  "expand-my-type": "./dist/cli.js"
37
32
  },
38
33
  "files": [
39
- "dist/index.cjs",
40
- "dist/index.d.cts",
41
- "dist/index.d.ts",
42
- "dist/index.js",
43
- "dist/cli.js"
34
+ "dist"
44
35
  ],
45
36
  "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",
37
+ "build": "bun run clean && bun run build:esm && bun run build:cjs && bun run build:types",
38
+ "build:cjs": "bun build ./src/index.ts --target node --format cjs --packages external --outfile ./dist/index.cjs --sourcemap=inline",
39
+ "build:esm": "bun build ./src/index.ts ./src/cli.ts --target node --format esm --packages external --outdir ./dist --root ./src --sourcemap=inline",
40
+ "build:types": "tsc -p tsconfig.build.json",
41
+ "check": "bun run typecheck && bun test ./src/index.test.ts",
42
+ "clean": "rm -rf dist",
43
+ "deps:update": "bun update --latest",
48
44
  "format": "prettier --write .",
49
- "prepare": "npm run bundle-lib && npm run bundle-cli",
50
- "pretest": "tsc",
51
- "test": "tsx --test --test-reporter spec ./src/index.test.ts"
45
+ "test": "bun test ./src/index.test.ts",
46
+ "typecheck": "tsc --noEmit"
52
47
  },
53
48
  "dependencies": {
54
- "prettier": "^3.3.3",
55
- "typescript": "^5.6.3"
49
+ "prettier": "^3.8.1",
50
+ "typescript": "^6.0.2"
56
51
  },
57
52
  "devDependencies": {
58
- "@trivago/prettier-plugin-sort-imports": "^4.3.0",
59
- "@types/node": "^22.5.5",
60
- "npm-check-updates": "^17.1.1",
61
- "prettier-plugin-packagejson": "^2.5.2",
62
- "tsup": "^8.2.4",
63
- "tsx": "^4.19.1"
64
- }
53
+ "@trivago/prettier-plugin-sort-imports": "^6.0.2",
54
+ "@types/node": "^22.19.11",
55
+ "prettier-plugin-packagejson": "^2.5.22"
56
+ },
57
+ "packageManager": "bun@1.3.11"
65
58
  }
package/dist/index.d.cts DELETED
@@ -1,55 +0,0 @@
1
- import ts from 'typescript';
2
- import { Options } from 'prettier';
3
-
4
- type ExtractFunctions<T, K extends keyof T = keyof T> = {
5
- [P in K]: T[P] extends (...args: any[]) => any ? T[P] : never;
6
- };
7
- type CompilerHostFunctionOverrides = Partial<ExtractFunctions<ts.CompilerHost>>;
8
-
9
- type ExpandTypeOptionsBase = {
10
- /**
11
- * The type expression to expand.
12
- * @example "ReturnType<typeof myFunction>"
13
- */
14
- typeExpression: string;
15
- /**
16
- * TypeScript compiler options.
17
- */
18
- tsCompilerOptions?: ts.CompilerOptions;
19
- /**
20
- * Prettier options.
21
- */
22
- prettify?: {
23
- /**
24
- * Whether to prettify the output.
25
- * @default true
26
- */
27
- enabled?: boolean;
28
- /**
29
- * Prettier options. Don't forget to set the parser to "typescript".
30
- * @default { parser: "typescript", semi: false }
31
- */
32
- options?: Options;
33
- };
34
- /**
35
- * A record of functions to override in the compiler host. Useful for mocking
36
- */
37
- compilerHostFunctionOverrides?: CompilerHostFunctionOverrides;
38
- };
39
- type ExpandTypeFromSourceFileOptions = ExpandTypeOptionsBase & {
40
- /**
41
- * Name of the source file to evaluate the type expression in.
42
- */
43
- sourceFileName: string;
44
- };
45
- type ExpandTypeFromSourceTextOptions = ExpandTypeOptionsBase & {
46
- /**
47
- * TypeScript source text to evaluate the type expression in.
48
- */
49
- sourceText: string;
50
- };
51
- type ExpandMyTypeOptions = ExpandTypeFromSourceFileOptions | ExpandTypeFromSourceTextOptions;
52
- declare function expandMyType(options: ExpandTypeFromSourceTextOptions): Promise<string>;
53
- declare function expandMyType(options: ExpandTypeFromSourceFileOptions): Promise<string>;
54
-
55
- export { type ExpandMyTypeOptions, type ExpandTypeFromSourceFileOptions, type ExpandTypeFromSourceTextOptions, type ExpandTypeOptionsBase, expandMyType };