Skip to content

Commit 0c4d4dd

Browse files
feat(create-email): Testing export, type checking and starter creation with respective fixes (#2038)
Co-authored-by: graphite-app[bot] <96075541+graphite-app[bot]@users.noreply.github.com>
1 parent c31b73e commit 0c4d4dd

12 files changed

+114
-12
lines changed

.changeset/itchy-coats-doubt.md

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
---
2+
"create-email": patch
3+
---
4+
5+
fix undefined behavior when template has already been created, fix error on tree when using custom project name

biome.json

+1
Original file line numberDiff line numberDiff line change
@@ -67,6 +67,7 @@
6767
"./**/node_modules/**/*",
6868
"./**/*.d.ts",
6969
"./**/**/prism.ts",
70+
"out",
7071
".turbo"
7172
]
7273
}

packages/create-email/.gitignore

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
.test

packages/create-email/.npmignore

+1
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,4 @@
1+
.test
12
template/node_modules
23
template/CHANGELOG.md
34
template/.turbo

packages/create-email/package.json

+5-1
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,9 @@
55
"main": "src/index.js",
66
"type": "module",
77
"license": "MIT",
8+
"scripts": {
9+
"test": "vitest run"
10+
},
811
"dependencies": {
912
"commander": "9.4.1",
1013
"fs-extra": "11.1.1",
@@ -24,6 +27,7 @@
2427
"create-email": "src/index.js"
2528
},
2629
"devDependencies": {
27-
"tsconfig": "workspace:*"
30+
"tsconfig": "workspace:*",
31+
"typescript": "5.8.2"
2832
}
2933
}

packages/create-email/src/index.js

+16-3
Original file line numberDiff line numberDiff line change
@@ -9,8 +9,6 @@ import ora from 'ora';
99
import { tree } from './tree.js';
1010

1111
const init = async (name) => {
12-
const spinner = ora('Preparing files...\n').start();
13-
1412
let projectPath = name;
1513

1614
if (!projectPath) {
@@ -25,6 +23,15 @@ const init = async (name) => {
2523
const templatePath = path.resolve(__dirname, '../template');
2624
const resolvedProjectPath = path.resolve(projectPath);
2725

26+
if (fse.existsSync(resolvedProjectPath)) {
27+
console.error(`Project called ${projectPath} already exists!`);
28+
process.exit(1);
29+
}
30+
31+
const spinner = ora({
32+
text: 'Preparing files...\n',
33+
}).start();
34+
2835
fse.copySync(templatePath, resolvedProjectPath, {
2936
recursive: true,
3037
});
@@ -60,7 +67,13 @@ const init = async (name) => {
6067
});
6168

6269
// eslint-disable-next-line no-console
63-
console.log(await tree('./react-email-starter', 4));
70+
console.info(
71+
await tree(resolvedProjectPath, 4, (dirent) => {
72+
return !path
73+
.join(dirent.parentPath, dirent.name)
74+
.includes('node_modules');
75+
}),
76+
);
6477
};
6578

6679
new Command()
+56
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,56 @@
1+
import { spawnSync } from 'node:child_process';
2+
import { promises as fs } from 'node:fs';
3+
import path from 'node:path';
4+
5+
describe('automatic setup', () => {
6+
const starterPath = path.resolve(__dirname, '../.test');
7+
test.sequential('creation', async () => {
8+
await fs.rm(starterPath, { recursive: true });
9+
10+
const createEmailProcess = spawnSync(
11+
'node',
12+
[path.resolve(__dirname, './index.js'), '.test'],
13+
{
14+
shell: true,
15+
cwd: path.resolve(__dirname, '../'),
16+
stdio: 'pipe',
17+
},
18+
);
19+
if (createEmailProcess.stderr) {
20+
console.log(createEmailProcess.stderr.toString());
21+
}
22+
expect(createEmailProcess.status, 'starter creation should return 0').toBe(
23+
0,
24+
);
25+
});
26+
27+
test.sequential('export', () => {
28+
const exportProcess = spawnSync('npm', ['run export'], {
29+
shell: true,
30+
cwd: starterPath,
31+
stdio: 'pipe',
32+
});
33+
if (exportProcess.stderr) {
34+
console.log(exportProcess.stderr.toString());
35+
}
36+
expect(exportProcess.status, 'export should return status code 0').toBe(0);
37+
});
38+
39+
test.sequential('type checking', () => {
40+
const typecheckingProcess = spawnSync('npx', ['tsc'], {
41+
shell: true,
42+
cwd: starterPath,
43+
stdio: 'pipe',
44+
});
45+
if (typecheckingProcess.stderr) {
46+
console.log(typecheckingProcess.stderr.toString());
47+
}
48+
if (typecheckingProcess.stdout) {
49+
console.log(typecheckingProcess.stdout.toString());
50+
}
51+
expect(
52+
typecheckingProcess.status,
53+
'type checking should return status code 0',
54+
).toBe(0);
55+
});
56+
});

packages/create-email/src/tree.js

+8-3
Original file line numberDiff line numberDiff line change
@@ -12,7 +12,7 @@ const SYMBOLS = {
1212
VERTICAL: '│ ',
1313
};
1414

15-
const getTreeLines = async (dirPath, depth, currentDepth = 0) => {
15+
const getTreeLines = async (dirPath, depth, filter, currentDepth = 0) => {
1616
const base = process.cwd();
1717
const dirFullpath = path.resolve(base, dirPath);
1818
const dirname = path.basename(dirFullpath);
@@ -44,13 +44,18 @@ const getTreeLines = async (dirPath, depth, currentDepth = 0) => {
4444
const branchingSymbol = isLast ? SYMBOLS.LAST_BRANCH : SYMBOLS.BRANCH;
4545
const verticalSymbol = isLast ? SYMBOLS.INDENT : SYMBOLS.VERTICAL;
4646

47+
if (filter?.(dirent) === false) {
48+
continue;
49+
}
50+
4751
if (dirent.isFile()) {
4852
lines.push(`${branchingSymbol}${dirent.name}`);
4953
} else {
5054
const pathToDirectory = path.join(dirFullpath, dirent.name);
5155
const treeLinesForSubDirectory = await getTreeLines(
5256
pathToDirectory,
5357
depth,
58+
filter,
5459
currentDepth + 1,
5560
);
5661
lines = lines.concat(
@@ -67,7 +72,7 @@ const getTreeLines = async (dirPath, depth, currentDepth = 0) => {
6772
return lines;
6873
};
6974

70-
export const tree = async (dirPath, depth) => {
71-
const lines = await getTreeLines(dirPath, depth);
75+
export const tree = async (dirPath, depth, filter) => {
76+
const lines = await getTreeLines(dirPath, depth, filter);
7277
return lines.join(os.EOL);
7378
};

packages/create-email/template/tsconfig.json

+3-2
Original file line numberDiff line numberDiff line change
@@ -4,8 +4,9 @@
44
"jsx": "react-jsx",
55
"skipLibCheck": true,
66
"strict": true,
7+
"noEmit": true,
78
"strictNullChecks": true
89
},
9-
"include": ["."],
10-
"exclude": ["node_modules"]
10+
"include": ["**/*.ts", "**/*.tsx"],
11+
"exclude": ["node_modules", ".react-email"]
1112
}

packages/create-email/tsconfig.json

+6-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,8 @@
11
{
2-
"extends": "tsconfig/react-library.json",
3-
"include": ["."],
4-
"exclude": ["dist", "build", "node_modules"]
2+
"extends": "tsconfig/base.json",
3+
"include": ["**/*.ts", "**/*.tsx"],
4+
"exclude": ["dist", "build", "node_modules", ".test"],
5+
"compilerOptions": {
6+
"noEmit": true
7+
}
58
}
+9
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
import { defineConfig } from 'vitest/config';
2+
3+
export default defineConfig({
4+
test: {
5+
globals: true,
6+
environment: 'happy-dom',
7+
exclude: ['.test/**/*', '**/node_modules'],
8+
},
9+
});

pnpm-lock.yaml

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

0 commit comments

Comments
 (0)