expand-my-type 0.1.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/LICENSE +21 -0
- package/README.md +100 -0
- package/dist/index.cjs +150 -0
- package/dist/index.d.cts +50 -0
- package/dist/index.d.ts +50 -0
- package/dist/index.js +114 -0
- package/package.json +59 -0
package/LICENSE
ADDED
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
MIT License
|
|
2
|
+
|
|
3
|
+
Copyright (c) 2024 Fardjad Davari
|
|
4
|
+
|
|
5
|
+
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
6
|
+
of this software and associated documentation files (the "Software"), to deal
|
|
7
|
+
in the Software without restriction, including without limitation the rights
|
|
8
|
+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
9
|
+
copies of the Software, and to permit persons to whom the Software is
|
|
10
|
+
furnished to do so, subject to the following conditions:
|
|
11
|
+
|
|
12
|
+
The above copyright notice and this permission notice shall be included in all
|
|
13
|
+
copies or substantial portions of the Software.
|
|
14
|
+
|
|
15
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
16
|
+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
17
|
+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
18
|
+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
19
|
+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
20
|
+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
21
|
+
SOFTWARE.
|
package/README.md
ADDED
|
@@ -0,0 +1,100 @@
|
|
|
1
|
+
# Expand My Type
|
|
2
|
+
|
|
3
|
+
<div class="paragraph">
|
|
4
|
+
|
|
5
|
+
<span class="image"><a href="https://www.npmjs.com/package/expand-my-type" class="image"><img src="https://img.shields.io/npm/v/expand-my-type" alt="NPM Version" /></a></span> <span class="image"><a href="https://www.npmjs.com/package/expand-my-type" class="image"><img src="https://img.shields.io/npm/dm/expand-my-type" alt="Monthly Downloads" /></a></span> <span class="image"><a href="https://github.com/fardjad/node-expand-my-type/actions" class="image"><img src="https://img.shields.io/github/actions/workflow/status/fardjad/node-expand-my-type/test-and-release.yml?branch=main" alt="test-and-release Workflow Status" /></a></span>
|
|
6
|
+
|
|
7
|
+
</div>
|
|
8
|
+
|
|
9
|
+
In TypeScript, expanding a type (also known as computing, resolving,
|
|
10
|
+
simplifying, unpacking, or unwrapping) refers to converting a type expression
|
|
11
|
+
into a flattened type that doesn't reference other types. That can be achieved
|
|
12
|
+
by defining [a utility type][compute-type]:
|
|
13
|
+
|
|
14
|
+
```typescript
|
|
15
|
+
type Compute<A> = { [K in keyof A]: Compute<A[K]> } & {};
|
|
16
|
+
```
|
|
17
|
+
|
|
18
|
+
And wrapping the type expression with it:
|
|
19
|
+
|
|
20
|
+
```typescript
|
|
21
|
+
type ExpandedType = Compute<SomeType>;
|
|
22
|
+
```
|
|
23
|
+
|
|
24
|
+
This is mostly done to improve the readability of type hints shown in editors/IDEs.
|
|
25
|
+
|
|
26
|
+
**Expand My Type** provides a programmatic way to do the same. It gets the
|
|
27
|
+
source code as input, along with a type expression and returns the expanded
|
|
28
|
+
form as a string. That can be useful for code-generation, testing, and debugging
|
|
29
|
+
complex type errors.
|
|
30
|
+
|
|
31
|
+
Under the hood, it uses the [TypeScript Compiler API][ts-compiler-api] to expand
|
|
32
|
+
the type expression and optionally formats the output using
|
|
33
|
+
[Prettier][prettier].
|
|
34
|
+
|
|
35
|
+
# Usage
|
|
36
|
+
|
|
37
|
+
This package exports a function named `expandMyType` that can be used in one of
|
|
38
|
+
the following ways:
|
|
39
|
+
|
|
40
|
+
1. Expand the type expression in a source file:
|
|
41
|
+
|
|
42
|
+
Given a file named `example.ts` with the following content:
|
|
43
|
+
|
|
44
|
+
```typescript
|
|
45
|
+
interface SomeInterface {
|
|
46
|
+
a: number;
|
|
47
|
+
b?: string;
|
|
48
|
+
c: number | undefined;
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
type SomeType = {
|
|
52
|
+
d: number;
|
|
53
|
+
e: SomeInterface;
|
|
54
|
+
f: () => void;
|
|
55
|
+
};
|
|
56
|
+
```
|
|
57
|
+
|
|
58
|
+
Running the following code expands the type expression `SomeType`:
|
|
59
|
+
|
|
60
|
+
```typescript
|
|
61
|
+
import { expandMyType } from "expand-my-type";
|
|
62
|
+
|
|
63
|
+
const expandedType = await expandMyType({
|
|
64
|
+
typeExpression: "SomeType",
|
|
65
|
+
sourceFileName: "./example.ts",
|
|
66
|
+
});
|
|
67
|
+
|
|
68
|
+
console.log(expandedType);
|
|
69
|
+
/* {
|
|
70
|
+
d: number
|
|
71
|
+
e: { a: number; b?: string; c: number }
|
|
72
|
+
f: {}
|
|
73
|
+
} */
|
|
74
|
+
```
|
|
75
|
+
|
|
76
|
+
2. Expand the type expression in a source text:
|
|
77
|
+
|
|
78
|
+
```typescript
|
|
79
|
+
import { expandMyType } from "expand-my-type";
|
|
80
|
+
|
|
81
|
+
const expandedType = await expandMyType({
|
|
82
|
+
typeExpression: "A<number>",
|
|
83
|
+
sourceText: `
|
|
84
|
+
type A<T> = {
|
|
85
|
+
a: string;
|
|
86
|
+
} & B<T>;
|
|
87
|
+
|
|
88
|
+
type B<T> = {
|
|
89
|
+
b: T;
|
|
90
|
+
};
|
|
91
|
+
`,
|
|
92
|
+
});
|
|
93
|
+
|
|
94
|
+
console.log(expandedType);
|
|
95
|
+
/* { a: string; b: number } */
|
|
96
|
+
```
|
|
97
|
+
|
|
98
|
+
[compute-type]: https://github.com/microsoft/TypeScript/blob/main/tests/cases/compiler/computedTypesKeyofNoIndexSignatureType.ts
|
|
99
|
+
[prettier]: https://prettier.io
|
|
100
|
+
[ts-compiler-api]: https://github.com/microsoft/TypeScript/wiki/Using-the-Compiler-API
|
package/dist/index.cjs
ADDED
|
@@ -0,0 +1,150 @@
|
|
|
1
|
+
var __create = Object.create;
|
|
2
|
+
var __defProp = Object.defineProperty;
|
|
3
|
+
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
4
|
+
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
5
|
+
var __getProtoOf = Object.getPrototypeOf;
|
|
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 });
|
|
10
|
+
};
|
|
11
|
+
var __copyProps = (to, from, except, desc) => {
|
|
12
|
+
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 });
|
|
16
|
+
}
|
|
17
|
+
return to;
|
|
18
|
+
};
|
|
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
|
+
|
|
29
|
+
// src/index.ts
|
|
30
|
+
var src_exports = {};
|
|
31
|
+
__export(src_exports, {
|
|
32
|
+
expandMyType: () => expandMyType
|
|
33
|
+
});
|
|
34
|
+
module.exports = __toCommonJS(src_exports);
|
|
35
|
+
|
|
36
|
+
// src/create-expander-compiler-host.ts
|
|
37
|
+
var import_typescript = __toESM(require("typescript"), 1);
|
|
38
|
+
var createExpanderCompilerHost = (sourceFileName, typeExpression, compilerOptions, getSourceFileFunction) => {
|
|
39
|
+
const customCompilerHost = import_typescript.default.createCompilerHost(compilerOptions ?? {});
|
|
40
|
+
const originalGetSourceFile = customCompilerHost.getSourceFile;
|
|
41
|
+
customCompilerHost.getSourceFile = (fileName, ...args) => {
|
|
42
|
+
if (fileName === sourceFileName) {
|
|
43
|
+
const sourceFile = (getSourceFileFunction ?? originalGetSourceFile)(
|
|
44
|
+
fileName,
|
|
45
|
+
...args
|
|
46
|
+
);
|
|
47
|
+
if (sourceFile === void 0) {
|
|
48
|
+
return void 0;
|
|
49
|
+
}
|
|
50
|
+
const sourceText = [
|
|
51
|
+
...[
|
|
52
|
+
"type __TYPE_EXPANDER_RESULT__ = __TYPE_EXPANDER_EXPAND__<__TYPE_EXPANDER_EXPRESSION__>;",
|
|
53
|
+
`type __TYPE_EXPANDER_EXPRESSION__ = ${typeExpression};`,
|
|
54
|
+
"type __TYPE_EXPANDER_EXPAND__<T> = {",
|
|
55
|
+
" [K in keyof T]: __TYPE_EXPANDER_EXPAND__<T[K]>;",
|
|
56
|
+
"} & {};"
|
|
57
|
+
],
|
|
58
|
+
sourceFile.getFullText()
|
|
59
|
+
].join("\n");
|
|
60
|
+
return import_typescript.default.createSourceFile(
|
|
61
|
+
fileName,
|
|
62
|
+
sourceText,
|
|
63
|
+
sourceFile.languageVersion,
|
|
64
|
+
true
|
|
65
|
+
);
|
|
66
|
+
}
|
|
67
|
+
return originalGetSourceFile(fileName, ...args);
|
|
68
|
+
};
|
|
69
|
+
return customCompilerHost;
|
|
70
|
+
};
|
|
71
|
+
|
|
72
|
+
// src/index.ts
|
|
73
|
+
var import_prettier = require("prettier");
|
|
74
|
+
var import_typescript2 = __toESM(require("typescript"), 1);
|
|
75
|
+
var findTypeExpanderResultNode = (node) => {
|
|
76
|
+
if (node.getChildCount() == 0) {
|
|
77
|
+
if (!import_typescript2.default.isIdentifier(node)) {
|
|
78
|
+
return void 0;
|
|
79
|
+
}
|
|
80
|
+
return node;
|
|
81
|
+
}
|
|
82
|
+
return import_typescript2.default.forEachChild(node, findTypeExpanderResultNode);
|
|
83
|
+
};
|
|
84
|
+
async function expandMyType(options) {
|
|
85
|
+
if (options.typeExpression.trim() === "") {
|
|
86
|
+
return "never";
|
|
87
|
+
}
|
|
88
|
+
if ("sourceText" in options) {
|
|
89
|
+
const sourceFile2 = import_typescript2.default.createSourceFile(
|
|
90
|
+
"test.ts",
|
|
91
|
+
options.sourceText,
|
|
92
|
+
import_typescript2.default.ScriptTarget.Latest,
|
|
93
|
+
true
|
|
94
|
+
);
|
|
95
|
+
return expandMyType({
|
|
96
|
+
sourceFileName: "test.ts",
|
|
97
|
+
typeExpression: options.typeExpression,
|
|
98
|
+
getSourceFileFunction: () => sourceFile2,
|
|
99
|
+
tsCompilerOptions: options.tsCompilerOptions
|
|
100
|
+
});
|
|
101
|
+
}
|
|
102
|
+
const tsCompilerOptions = options.tsCompilerOptions ?? {
|
|
103
|
+
noEmit: true,
|
|
104
|
+
allowSyntheticDefaultImports: true,
|
|
105
|
+
allowArbitraryExtensions: true,
|
|
106
|
+
allowImportingTsExtensions: true,
|
|
107
|
+
allowJs: true
|
|
108
|
+
};
|
|
109
|
+
const compilerHost = createExpanderCompilerHost(
|
|
110
|
+
options.sourceFileName,
|
|
111
|
+
options.typeExpression,
|
|
112
|
+
tsCompilerOptions,
|
|
113
|
+
options.getSourceFileFunction
|
|
114
|
+
);
|
|
115
|
+
const program = import_typescript2.default.createProgram(
|
|
116
|
+
[options.sourceFileName],
|
|
117
|
+
tsCompilerOptions,
|
|
118
|
+
compilerHost
|
|
119
|
+
);
|
|
120
|
+
const sourceFile = program.getSourceFile(options.sourceFileName);
|
|
121
|
+
if (!sourceFile) {
|
|
122
|
+
throw new Error("Source file not found!");
|
|
123
|
+
}
|
|
124
|
+
const firstIdentifier = findTypeExpanderResultNode(sourceFile);
|
|
125
|
+
if (!firstIdentifier) {
|
|
126
|
+
throw new Error("No node found!");
|
|
127
|
+
}
|
|
128
|
+
const typeChecker = program.getTypeChecker();
|
|
129
|
+
const expandedType = typeChecker.typeToString(
|
|
130
|
+
typeChecker.getTypeAtLocation(firstIdentifier),
|
|
131
|
+
void 0,
|
|
132
|
+
import_typescript2.default.TypeFormatFlags.NodeBuilderFlagsMask
|
|
133
|
+
);
|
|
134
|
+
if (options.prettify && options.prettify.enabled === false) {
|
|
135
|
+
return expandedType;
|
|
136
|
+
}
|
|
137
|
+
const dummyTypeName = "__TYPE_EXPANDER_RESULT__";
|
|
138
|
+
return (await (0, import_prettier.format)(
|
|
139
|
+
`type ${dummyTypeName} = ${expandedType}`,
|
|
140
|
+
options.prettify?.options ?? {
|
|
141
|
+
parser: "typescript",
|
|
142
|
+
semi: false
|
|
143
|
+
}
|
|
144
|
+
)).trim().substring(`type ${dummyTypeName} = `.length);
|
|
145
|
+
}
|
|
146
|
+
// Annotate the CommonJS export names for ESM import in node:
|
|
147
|
+
0 && (module.exports = {
|
|
148
|
+
expandMyType
|
|
149
|
+
});
|
|
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"]
}

|
package/dist/index.d.cts
ADDED
|
@@ -0,0 +1,50 @@
|
|
|
1
|
+
import { Options } from 'prettier';
|
|
2
|
+
import ts from 'typescript';
|
|
3
|
+
|
|
4
|
+
type ExpandTypeOptionsBase = {
|
|
5
|
+
/**
|
|
6
|
+
* The type expression to expand.
|
|
7
|
+
* @example "ReturnType<typeof myFunction>"
|
|
8
|
+
*/
|
|
9
|
+
typeExpression: string;
|
|
10
|
+
/**
|
|
11
|
+
* TypeScript compiler options.
|
|
12
|
+
*/
|
|
13
|
+
tsCompilerOptions?: ts.CompilerOptions;
|
|
14
|
+
/**
|
|
15
|
+
* Prettier options.
|
|
16
|
+
*/
|
|
17
|
+
prettify?: {
|
|
18
|
+
/**
|
|
19
|
+
* Whether to prettify the output.
|
|
20
|
+
* @default true
|
|
21
|
+
*/
|
|
22
|
+
enabled?: boolean;
|
|
23
|
+
/**
|
|
24
|
+
* Prettier options. Don't forget to set the parser to "typescript".
|
|
25
|
+
* @default { parser: "typescript", semi: false }
|
|
26
|
+
*/
|
|
27
|
+
options?: Options;
|
|
28
|
+
};
|
|
29
|
+
};
|
|
30
|
+
type ExpandTypeFromSourceFileOptions = ExpandTypeOptionsBase & {
|
|
31
|
+
/**
|
|
32
|
+
* Name of the source file to evaluate the type expression in.
|
|
33
|
+
*/
|
|
34
|
+
sourceFileName: string;
|
|
35
|
+
/**
|
|
36
|
+
* Function that returns the source file. When not specified, the implementation of the default TypeScript compiler host is used.
|
|
37
|
+
*/
|
|
38
|
+
getSourceFileFunction?: ts.CompilerHost["getSourceFile"];
|
|
39
|
+
};
|
|
40
|
+
type ExpandTypeFromSourceTextOptions = ExpandTypeOptionsBase & {
|
|
41
|
+
/**
|
|
42
|
+
* TypeScript source text to evaluate the type expression in.
|
|
43
|
+
*/
|
|
44
|
+
sourceText: string;
|
|
45
|
+
};
|
|
46
|
+
type ExpandMyTypeOptions = ExpandTypeFromSourceFileOptions | ExpandTypeFromSourceTextOptions;
|
|
47
|
+
declare function expandMyType(options: ExpandTypeFromSourceTextOptions): Promise<string>;
|
|
48
|
+
declare function expandMyType(options: ExpandTypeFromSourceFileOptions): Promise<string>;
|
|
49
|
+
|
|
50
|
+
export { type ExpandMyTypeOptions, type ExpandTypeFromSourceFileOptions, type ExpandTypeFromSourceTextOptions, type ExpandTypeOptionsBase, expandMyType };
|
package/dist/index.d.ts
ADDED
|
@@ -0,0 +1,50 @@
|
|
|
1
|
+
import { Options } from 'prettier';
|
|
2
|
+
import ts from 'typescript';
|
|
3
|
+
|
|
4
|
+
type ExpandTypeOptionsBase = {
|
|
5
|
+
/**
|
|
6
|
+
* The type expression to expand.
|
|
7
|
+
* @example "ReturnType<typeof myFunction>"
|
|
8
|
+
*/
|
|
9
|
+
typeExpression: string;
|
|
10
|
+
/**
|
|
11
|
+
* TypeScript compiler options.
|
|
12
|
+
*/
|
|
13
|
+
tsCompilerOptions?: ts.CompilerOptions;
|
|
14
|
+
/**
|
|
15
|
+
* Prettier options.
|
|
16
|
+
*/
|
|
17
|
+
prettify?: {
|
|
18
|
+
/**
|
|
19
|
+
* Whether to prettify the output.
|
|
20
|
+
* @default true
|
|
21
|
+
*/
|
|
22
|
+
enabled?: boolean;
|
|
23
|
+
/**
|
|
24
|
+
* Prettier options. Don't forget to set the parser to "typescript".
|
|
25
|
+
* @default { parser: "typescript", semi: false }
|
|
26
|
+
*/
|
|
27
|
+
options?: Options;
|
|
28
|
+
};
|
|
29
|
+
};
|
|
30
|
+
type ExpandTypeFromSourceFileOptions = ExpandTypeOptionsBase & {
|
|
31
|
+
/**
|
|
32
|
+
* Name of the source file to evaluate the type expression in.
|
|
33
|
+
*/
|
|
34
|
+
sourceFileName: string;
|
|
35
|
+
/**
|
|
36
|
+
* Function that returns the source file. When not specified, the implementation of the default TypeScript compiler host is used.
|
|
37
|
+
*/
|
|
38
|
+
getSourceFileFunction?: ts.CompilerHost["getSourceFile"];
|
|
39
|
+
};
|
|
40
|
+
type ExpandTypeFromSourceTextOptions = ExpandTypeOptionsBase & {
|
|
41
|
+
/**
|
|
42
|
+
* TypeScript source text to evaluate the type expression in.
|
|
43
|
+
*/
|
|
44
|
+
sourceText: string;
|
|
45
|
+
};
|
|
46
|
+
type ExpandMyTypeOptions = ExpandTypeFromSourceFileOptions | ExpandTypeFromSourceTextOptions;
|
|
47
|
+
declare function expandMyType(options: ExpandTypeFromSourceTextOptions): Promise<string>;
|
|
48
|
+
declare function expandMyType(options: ExpandTypeFromSourceFileOptions): Promise<string>;
|
|
49
|
+
|
|
50
|
+
export { type ExpandMyTypeOptions, type ExpandTypeFromSourceFileOptions, type ExpandTypeFromSourceTextOptions, type ExpandTypeOptionsBase, expandMyType };
|
package/dist/index.js
ADDED
|
@@ -0,0 +1,114 @@
|
|
|
1
|
+
// src/create-expander-compiler-host.ts
|
|
2
|
+
import ts from "typescript";
|
|
3
|
+
var createExpanderCompilerHost = (sourceFileName, typeExpression, compilerOptions, getSourceFileFunction) => {
|
|
4
|
+
const customCompilerHost = ts.createCompilerHost(compilerOptions ?? {});
|
|
5
|
+
const originalGetSourceFile = customCompilerHost.getSourceFile;
|
|
6
|
+
customCompilerHost.getSourceFile = (fileName, ...args) => {
|
|
7
|
+
if (fileName === sourceFileName) {
|
|
8
|
+
const sourceFile = (getSourceFileFunction ?? originalGetSourceFile)(
|
|
9
|
+
fileName,
|
|
10
|
+
...args
|
|
11
|
+
);
|
|
12
|
+
if (sourceFile === void 0) {
|
|
13
|
+
return void 0;
|
|
14
|
+
}
|
|
15
|
+
const sourceText = [
|
|
16
|
+
...[
|
|
17
|
+
"type __TYPE_EXPANDER_RESULT__ = __TYPE_EXPANDER_EXPAND__<__TYPE_EXPANDER_EXPRESSION__>;",
|
|
18
|
+
`type __TYPE_EXPANDER_EXPRESSION__ = ${typeExpression};`,
|
|
19
|
+
"type __TYPE_EXPANDER_EXPAND__<T> = {",
|
|
20
|
+
" [K in keyof T]: __TYPE_EXPANDER_EXPAND__<T[K]>;",
|
|
21
|
+
"} & {};"
|
|
22
|
+
],
|
|
23
|
+
sourceFile.getFullText()
|
|
24
|
+
].join("\n");
|
|
25
|
+
return ts.createSourceFile(
|
|
26
|
+
fileName,
|
|
27
|
+
sourceText,
|
|
28
|
+
sourceFile.languageVersion,
|
|
29
|
+
true
|
|
30
|
+
);
|
|
31
|
+
}
|
|
32
|
+
return originalGetSourceFile(fileName, ...args);
|
|
33
|
+
};
|
|
34
|
+
return customCompilerHost;
|
|
35
|
+
};
|
|
36
|
+
|
|
37
|
+
// src/index.ts
|
|
38
|
+
import { format } from "prettier";
|
|
39
|
+
import ts2 from "typescript";
|
|
40
|
+
var findTypeExpanderResultNode = (node) => {
|
|
41
|
+
if (node.getChildCount() == 0) {
|
|
42
|
+
if (!ts2.isIdentifier(node)) {
|
|
43
|
+
return void 0;
|
|
44
|
+
}
|
|
45
|
+
return node;
|
|
46
|
+
}
|
|
47
|
+
return ts2.forEachChild(node, findTypeExpanderResultNode);
|
|
48
|
+
};
|
|
49
|
+
async function expandMyType(options) {
|
|
50
|
+
if (options.typeExpression.trim() === "") {
|
|
51
|
+
return "never";
|
|
52
|
+
}
|
|
53
|
+
if ("sourceText" in options) {
|
|
54
|
+
const sourceFile2 = ts2.createSourceFile(
|
|
55
|
+
"test.ts",
|
|
56
|
+
options.sourceText,
|
|
57
|
+
ts2.ScriptTarget.Latest,
|
|
58
|
+
true
|
|
59
|
+
);
|
|
60
|
+
return expandMyType({
|
|
61
|
+
sourceFileName: "test.ts",
|
|
62
|
+
typeExpression: options.typeExpression,
|
|
63
|
+
getSourceFileFunction: () => sourceFile2,
|
|
64
|
+
tsCompilerOptions: options.tsCompilerOptions
|
|
65
|
+
});
|
|
66
|
+
}
|
|
67
|
+
const tsCompilerOptions = options.tsCompilerOptions ?? {
|
|
68
|
+
noEmit: true,
|
|
69
|
+
allowSyntheticDefaultImports: true,
|
|
70
|
+
allowArbitraryExtensions: true,
|
|
71
|
+
allowImportingTsExtensions: true,
|
|
72
|
+
allowJs: true
|
|
73
|
+
};
|
|
74
|
+
const compilerHost = createExpanderCompilerHost(
|
|
75
|
+
options.sourceFileName,
|
|
76
|
+
options.typeExpression,
|
|
77
|
+
tsCompilerOptions,
|
|
78
|
+
options.getSourceFileFunction
|
|
79
|
+
);
|
|
80
|
+
const program = ts2.createProgram(
|
|
81
|
+
[options.sourceFileName],
|
|
82
|
+
tsCompilerOptions,
|
|
83
|
+
compilerHost
|
|
84
|
+
);
|
|
85
|
+
const sourceFile = program.getSourceFile(options.sourceFileName);
|
|
86
|
+
if (!sourceFile) {
|
|
87
|
+
throw new Error("Source file not found!");
|
|
88
|
+
}
|
|
89
|
+
const firstIdentifier = findTypeExpanderResultNode(sourceFile);
|
|
90
|
+
if (!firstIdentifier) {
|
|
91
|
+
throw new Error("No node found!");
|
|
92
|
+
}
|
|
93
|
+
const typeChecker = program.getTypeChecker();
|
|
94
|
+
const expandedType = typeChecker.typeToString(
|
|
95
|
+
typeChecker.getTypeAtLocation(firstIdentifier),
|
|
96
|
+
void 0,
|
|
97
|
+
ts2.TypeFormatFlags.NodeBuilderFlagsMask
|
|
98
|
+
);
|
|
99
|
+
if (options.prettify && options.prettify.enabled === false) {
|
|
100
|
+
return expandedType;
|
|
101
|
+
}
|
|
102
|
+
const dummyTypeName = "__TYPE_EXPANDER_RESULT__";
|
|
103
|
+
return (await format(
|
|
104
|
+
`type ${dummyTypeName} = ${expandedType}`,
|
|
105
|
+
options.prettify?.options ?? {
|
|
106
|
+
parser: "typescript",
|
|
107
|
+
semi: false
|
|
108
|
+
}
|
|
109
|
+
)).trim().substring(`type ${dummyTypeName} = `.length);
|
|
110
|
+
}
|
|
111
|
+
export {
|
|
112
|
+
expandMyType
|
|
113
|
+
};
|
|
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"]
}

|
package/package.json
ADDED
|
@@ -0,0 +1,59 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "expand-my-type",
|
|
3
|
+
"version": "0.1.0",
|
|
4
|
+
"description": "Expand TypeScript type expressions programmatically",
|
|
5
|
+
"keywords": [
|
|
6
|
+
"expand",
|
|
7
|
+
"typescript",
|
|
8
|
+
"types"
|
|
9
|
+
],
|
|
10
|
+
"homepage": "https://github.com/fardjad/node-expand-my-type",
|
|
11
|
+
"bugs": {
|
|
12
|
+
"url": "https://github.com/fardjad/node-expand-my-type/issues"
|
|
13
|
+
},
|
|
14
|
+
"repository": {
|
|
15
|
+
"type": "git",
|
|
16
|
+
"url": "git+ssh://git@github.com/fardjad/node-expand-my-type.git"
|
|
17
|
+
},
|
|
18
|
+
"license": "MIT",
|
|
19
|
+
"author": "Fardjad Davari <public@fardjad.com>",
|
|
20
|
+
"type": "module",
|
|
21
|
+
"exports": {
|
|
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
|
+
}
|
|
31
|
+
}
|
|
32
|
+
},
|
|
33
|
+
"main": "dist/index.cjs",
|
|
34
|
+
"types": "dist/index.d.ts",
|
|
35
|
+
"files": [
|
|
36
|
+
"dist/index.cjs",
|
|
37
|
+
"dist/index.d.cts",
|
|
38
|
+
"dist/index.d.ts",
|
|
39
|
+
"dist/index.js"
|
|
40
|
+
],
|
|
41
|
+
"scripts": {
|
|
42
|
+
"format": "prettier --write .",
|
|
43
|
+
"prepare": "tsup ./src/index.ts --format esm,cjs --dts --clean --sourcemap inline --silent",
|
|
44
|
+
"pretest": "tsc",
|
|
45
|
+
"test": "tsx --test --test-reporter spec ./src/index.test.ts"
|
|
46
|
+
},
|
|
47
|
+
"dependencies": {
|
|
48
|
+
"prettier": "^3.2.5",
|
|
49
|
+
"typescript": "^5.4.5"
|
|
50
|
+
},
|
|
51
|
+
"devDependencies": {
|
|
52
|
+
"@trivago/prettier-plugin-sort-imports": "^4.3.0",
|
|
53
|
+
"@types/node": "^20.12.12",
|
|
54
|
+
"npm-check-updates": "^16.14.18",
|
|
55
|
+
"prettier-plugin-packagejson": "^2.4.14",
|
|
56
|
+
"tsup": "^8.0.2",
|
|
57
|
+
"tsx": "^4.10.5"
|
|
58
|
+
}
|
|
59
|
+
}
|