Skip to content

Commit 62f8a98

Browse files
author
infodusha
committed
feat!: ts path alias support
1 parent 9cec5e2 commit 62f8a98

File tree

7 files changed

+91
-15
lines changed

7 files changed

+91
-15
lines changed

examples/general/src/app/Inner.jsx

+3-1
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,5 @@
1+
import { Inner2 } from './Inner2'
2+
13
export function Inner() {
2-
return <div>Inner</div>;
4+
return <div>Inner <Inner2 /> </div>;
35
}

examples/general/src/app/Inner2.jsx

+4
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
2+
export function Inner2() {
3+
return <div>Inner 2</div>;
4+
}

examples/general/src/app/error.jsx

+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
'use client';
2+
3+
import { Inner } from "./Inner";
4+
5+
export default function Error() {
6+
7+
return <Inner/>
8+
}

package-lock.json

+15-2
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

packages/next-rsc-error-handler/package.json

+2-1
Original file line numberDiff line numberDiff line change
@@ -28,6 +28,7 @@
2828
"@babel/traverse": "^7.24.5",
2929
"@babel/types": "^7.24.5",
3030
"client-only": "^0.0.1",
31-
"server-only": "^0.0.1"
31+
"server-only": "^0.0.1",
32+
"typescript": "~5.4.5"
3233
}
3334
}

packages/next-rsc-error-handler/src/loader.js

+8-11
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,6 @@ import parser from "@babel/parser";
22
import traverse from "@babel/traverse";
33
import generate from "@babel/generator";
44
import * as t from "@babel/types";
5-
import path from "node:path";
65

76
import {
87
getRelativePath,
@@ -11,6 +10,7 @@ import {
1110
wrapArrowFunction,
1211
getOptionsExpressionLiteral,
1312
} from "./utils.js";
13+
import { resolveTsAlias } from "./ts-path-alias.js";
1414

1515
const WRAPPER_NAME = "__rscWrapper";
1616
const WRAPPER_PATH = "next-rsc-error-handler/inserted/wrapper";
@@ -37,9 +37,14 @@ export default function (source) {
3737

3838
let hasComponents = false;
3939

40+
const getImportRelativePath = (p) => {
41+
const val = resolveTsAlias(p.node.source.value);
42+
return getRelativePath(this.utils.absolutify(this.context, val));
43+
};
44+
4045
traverse.default(ast, {
4146
ImportDeclaration(p) {
42-
clientComponents.add(getImportRelativePath(resourcePath, p));
47+
clientComponents.add(getImportRelativePath(p));
4348
},
4449
// TODO add FunctionExpression
4550
FunctionDeclaration(p) {
@@ -148,13 +153,5 @@ function addClientOnlyImport(ast) {
148153
}
149154

150155
function dropExtension(relativePath) {
151-
return relativePath.replace(/\.[^/.]+$/, "");
152-
}
153-
154-
function getImportRelativePath(resourcePath, p) {
155-
return dropExtension(
156-
getRelativePath(
157-
path.resolve(path.dirname(resourcePath), p.node.source.value)
158-
)
159-
);
156+
return relativePath.replace(/\.[^\/.]+$/, "");
160157
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,51 @@
1+
import { dirname, resolve, relative } from "node:path";
2+
import typescript from "typescript";
3+
4+
const { findConfigFile, parseJsonConfigFileContent, readConfigFile, sys } =
5+
typescript;
6+
7+
function loadTSConfig() {
8+
const configFileName = findConfigFile(process.cwd(), sys.fileExists);
9+
if (!configFileName) {
10+
return null;
11+
}
12+
const configFile = readConfigFile(configFileName, sys.readFile);
13+
const directory = dirname(configFileName);
14+
const options = parseJsonConfigFileContent(configFile.config, sys, directory);
15+
return options;
16+
}
17+
18+
const tsConfig = loadTSConfig();
19+
20+
const fns = tsConfig
21+
? Object.entries(tsConfig.options.paths).map(prepareRegex)
22+
: [];
23+
24+
function prepareRegex(paths) {
25+
const [key, [value]] = paths;
26+
const from = new RegExp(escapeRegExp(key).replace("*", "(.*)"));
27+
const to = relative(
28+
process.cwd(),
29+
resolve(tsConfig.options.baseUrl, value.replace("*", "$$1"))
30+
);
31+
return (x) => {
32+
if (from.test(x)) {
33+
return x.replace(from, to);
34+
}
35+
return null;
36+
};
37+
}
38+
39+
function escapeRegExp(text) {
40+
return text.replace(/[-[\]{}()+?.,\\^$|#\s]/g, "\\$&");
41+
}
42+
43+
export function resolveTsAlias(value) {
44+
for (const fn of fns) {
45+
const result = fn(value);
46+
if (result) {
47+
return result;
48+
}
49+
}
50+
return value;
51+
}

0 commit comments

Comments
 (0)