Skip to content

Commit 98df54a

Browse files
panteliselefwobsoriano
authored andcommitted
test(nextjs): E2E tests for middleware file placement (#5037)
1 parent 9f62d0d commit 98df54a

File tree

4 files changed

+138
-10
lines changed

4 files changed

+138
-10
lines changed

.changeset/selfish-news-invent.md

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
---
2+
---

integration/models/application.ts

+9
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,7 @@ export const application = (
2020
const stdoutFilePath = path.resolve(appDirPath, `e2e.${now}.log`);
2121
const stderrFilePath = path.resolve(appDirPath, `e2e.${now}.err.log`);
2222
let buildOutput = '';
23+
let serveOutput = '';
2324

2425
const self = {
2526
name,
@@ -93,7 +94,11 @@ export const application = (
9394
get buildOutput() {
9495
return buildOutput;
9596
},
97+
get serveOutput() {
98+
return serveOutput;
99+
},
96100
serve: async (opts: { port?: number; manualStart?: boolean } = {}) => {
101+
const log = logger.child({ prefix: 'serve' }).info;
97102
const port = opts.port || (await getPort());
98103
// TODO: get serverUrl as in dev()
99104
const serverUrl = `http://localhost:${port}`;
@@ -102,6 +107,10 @@ export const application = (
102107
const proc = run(scripts.serve, {
103108
cwd: appDirPath,
104109
env: { PORT: port.toString() },
110+
log: (msg: string) => {
111+
serveOutput += `\n${msg}`;
112+
log(msg);
113+
},
105114
});
106115
cleanupFns.push(() => awaitableTreekill(proc.pid, 'SIGKILL'));
107116
await waitForIdleProcess(proc);

integration/models/applicationConfig.ts

+28-10
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,7 @@ type Scripts = { dev: string; build: string; setup: string; serve: string };
1313
export const applicationConfig = () => {
1414
let name = '';
1515
let serverUrl = '';
16-
const templates: string[] = [];
16+
let template: string;
1717
const files = new Map<string, string>();
1818
const scripts: Scripts = { dev: 'pnpm dev', serve: 'pnpm serve', build: 'pnpm build', setup: 'pnpm install' };
1919
const envFormatters = { public: (key: string) => key, private: (key: string) => key };
@@ -26,7 +26,9 @@ export const applicationConfig = () => {
2626
clone.setName(name);
2727
clone.setEnvFormatter('public', envFormatters.public);
2828
clone.setEnvFormatter('private', envFormatters.private);
29-
templates.forEach(t => clone.useTemplate(t));
29+
if (template) {
30+
clone.useTemplate(template);
31+
}
3032
dependencies.forEach((v, k) => clone.addDependency(k, v));
3133
Object.entries(scripts).forEach(([k, v]) => clone.addScript(k as keyof typeof scripts, v));
3234
files.forEach((v, k) => clone.addFile(k, () => v));
@@ -44,8 +46,12 @@ export const applicationConfig = () => {
4446
files.set(filePath, cbOrPath(helpers));
4547
return self;
4648
},
49+
removeFile: (filePath: string) => {
50+
files.set(filePath, '');
51+
return self;
52+
},
4753
useTemplate: (path: string) => {
48-
templates.push(path);
54+
template = path;
4955
return self;
5056
},
5157
setEnvFormatter: (type: keyof typeof envFormatters, formatter: typeof envFormatters.public) => {
@@ -76,20 +82,32 @@ export const applicationConfig = () => {
7682
const appDirPath = path.resolve(constants.TMP_DIR, appDirName);
7783

7884
// Copy template files
79-
for (const template of templates) {
85+
if (template) {
8086
logger.info(`Copying template "${path.basename(template)}" -> ${appDirPath}`);
8187
await fs.ensureDir(appDirPath);
8288
await fs.copy(template, appDirPath, { overwrite: true, filter: (p: string) => !p.includes('node_modules') });
8389
}
8490

91+
await Promise.all(
92+
[...files]
93+
.filter(([, contents]) => !contents)
94+
.map(async ([pathname]) => {
95+
const dest = path.resolve(appDirPath, pathname);
96+
logger.info(`Deleting file ${dest}`);
97+
await fs.remove(dest);
98+
}),
99+
);
100+
85101
// Create individual files
86102
await Promise.all(
87-
[...files].map(async ([pathname, contents]) => {
88-
const dest = path.resolve(appDirPath, pathname);
89-
logger.info(`Copying file "${pathname}" -> ${dest}`);
90-
await fs.ensureFile(dest);
91-
await fs.writeFile(dest, contents);
92-
}),
103+
[...files]
104+
.filter(([, contents]) => contents)
105+
.map(async ([pathname, contents]) => {
106+
const dest = path.resolve(appDirPath, pathname);
107+
logger.info(`Copying file "${pathname}" -> ${dest}`);
108+
await fs.ensureFile(dest);
109+
await fs.writeFile(dest, contents);
110+
}),
93111
);
94112

95113
// Adjust package.json dependencies
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,99 @@
1+
import { expect, test } from '@playwright/test';
2+
3+
import type { Application } from '../models/application';
4+
import { appConfigs } from '../presets';
5+
import { createTestUtils } from '../testUtils';
6+
7+
const middlewareFileContents = `
8+
import { clerkMiddleware } from '@clerk/nextjs/server';
9+
export default clerkMiddleware();
10+
11+
export const config = {
12+
matcher: ['/((?!.*\\..*|_next).*)', '/', '/(api|trpc)(.*)'],
13+
};
14+
`;
15+
16+
const commonSetup = appConfigs.next.appRouterQuickstart.clone().removeFile('src/middleware.ts');
17+
18+
test.describe('next start - missing middleware @quickstart', () => {
19+
test.describe.configure({ mode: 'parallel' });
20+
let app: Application;
21+
22+
test.beforeAll(async () => {
23+
app = await commonSetup.commit();
24+
await app.setup();
25+
await app.withEnv(appConfigs.envs.withEmailCodesQuickstart);
26+
await app.build();
27+
await app.serve();
28+
});
29+
30+
test.afterAll(async () => {
31+
await app.teardown();
32+
});
33+
34+
test('Display error for missing middleware', async ({ page, context }) => {
35+
const u = createTestUtils({ app, page, context });
36+
await u.page.goToAppHome();
37+
38+
expect(app.serveOutput).toContain('Your Middleware exists at ./src/middleware.(ts|js)');
39+
});
40+
});
41+
42+
test.describe('next start - invalid middleware at root on src/ @quickstart', () => {
43+
test.describe.configure({ mode: 'parallel' });
44+
let app: Application;
45+
46+
test.beforeAll(async () => {
47+
app = await commonSetup.addFile('middleware.ts', () => middlewareFileContents).commit();
48+
await app.setup();
49+
await app.withEnv(appConfigs.envs.withEmailCodesQuickstart);
50+
await app.build();
51+
await app.serve();
52+
});
53+
54+
test.afterAll(async () => {
55+
await app.teardown();
56+
});
57+
58+
test('Display suggestion for moving middleware to from `./middleware.ts` to `./src/middleware.ts`', async ({
59+
page,
60+
context,
61+
}) => {
62+
const u = createTestUtils({ app, page, context });
63+
await u.page.goToAppHome();
64+
65+
expect(app.serveOutput).not.toContain('Your Middleware exists at ./src/middleware.(ts|js)');
66+
expect(app.serveOutput).toContain(
67+
'Clerk: clerkMiddleware() was not run, your middleware file might be misplaced. Move your middleware file to ./src/middleware.ts. Currently located at ./middleware.ts',
68+
);
69+
});
70+
});
71+
72+
test.describe('next start - invalid middleware inside app on src/ @quickstart', () => {
73+
test.describe.configure({ mode: 'parallel' });
74+
let app: Application;
75+
76+
test.beforeAll(async () => {
77+
app = await commonSetup.addFile('src/app/middleware.ts', () => middlewareFileContents).commit();
78+
await app.setup();
79+
await app.withEnv(appConfigs.envs.withEmailCodesQuickstart);
80+
await app.build();
81+
await app.serve();
82+
});
83+
84+
test.afterAll(async () => {
85+
await app.teardown();
86+
});
87+
88+
test('Display suggestion for moving middleware to from `./src/app/middleware.ts` to `./src/middleware.ts`', async ({
89+
page,
90+
context,
91+
}) => {
92+
const u = createTestUtils({ app, page, context });
93+
await u.page.goToAppHome();
94+
expect(app.serveOutput).not.toContain('Your Middleware exists at ./src/middleware.(ts|js)');
95+
expect(app.serveOutput).toContain(
96+
'Clerk: clerkMiddleware() was not run, your middleware file might be misplaced. Move your middleware file to ./src/middleware.ts. Currently located at ./src/app/middleware.ts',
97+
);
98+
});
99+
});

0 commit comments

Comments
 (0)