Skip to content

Commit d08c83e

Browse files
authored
fix(@astrojs/node): handler should work with express (#8176)
1 parent 95120ef commit d08c83e

File tree

7 files changed

+221
-16
lines changed

7 files changed

+221
-16
lines changed

.changeset/rich-tigers-march.md

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
---
2+
'@astrojs/node': patch
3+
---
4+
5+
Fix an issue where `express` couldn't use the `handler` in `middleware` mode.

packages/astro/src/cli/add/index.ts

+1
Original file line numberDiff line numberDiff line change
@@ -709,6 +709,7 @@ async function tryToInstallIntegrations({
709709
} catch (err) {
710710
spinner.fail();
711711
debug('add', 'Error installing dependencies', err);
712+
// eslint-disable no-console
712713
console.error('\n', (err as any).stdout, '\n');
713714
return UpdateResult.failure;
714715
}

packages/integrations/node/package.json

+2-1
Original file line numberDiff line numberDiff line change
@@ -50,6 +50,7 @@
5050
"cheerio": "1.0.0-rc.12",
5151
"mocha": "^9.2.2",
5252
"node-mocks-http": "^1.13.0",
53-
"undici": "^5.22.1"
53+
"undici": "^5.22.1",
54+
"express": "^4.18.2"
5455
}
5556
}

packages/integrations/node/src/nodeMiddleware.ts

+12-5
Original file line numberDiff line numberDiff line change
@@ -6,15 +6,22 @@ import { responseIterator } from './response-iterator';
66
import type { ErrorHandlerParams, Options, RequestHandlerParams } from './types';
77

88
// Disable no-unused-vars to avoid breaking signature change
9-
// eslint-disable-next-line @typescript-eslint/no-unused-vars
10-
export default function (app: NodeApp, _mode: Options['mode']) {
9+
export default function (app: NodeApp, mode: Options['mode']) {
1110
return async function (...args: RequestHandlerParams | ErrorHandlerParams) {
1211
let error = null;
13-
let [req, res, next, locals] = args as RequestHandlerParams;
12+
let locals;
13+
let [req, res, next] = args as RequestHandlerParams;
14+
if (mode === 'middleware') {
15+
let { [3]: _locals } = args;
16+
locals = _locals;
17+
}
1418

1519
if (args[0] instanceof Error) {
16-
[error, req, res, next, locals] = args as ErrorHandlerParams;
17-
20+
[error, req, res, next] = args as ErrorHandlerParams;
21+
if (mode === 'middleware') {
22+
let { [4]: _locals } = args as ErrorHandlerParams;
23+
locals = _locals;
24+
}
1825
if (error) {
1926
if (next) {
2027
return next(error);
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
export async function get() {
2+
let number = Math.random();
3+
return {
4+
body: JSON.stringify({
5+
number,
6+
message: `Here's a random number: ${number}`,
7+
}),
8+
};
9+
}

packages/integrations/node/test/node-middleware.test.js

+41-1
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@ import nodejs from '../dist/index.js';
22
import { loadFixture } from './test-utils.js';
33
import { expect } from 'chai';
44
import * as cheerio from 'cheerio';
5+
import express from 'express';
56

67
/**
78
* @typedef {import('../../../astro/test/test-utils').Fixture} Fixture
@@ -14,7 +15,7 @@ async function load() {
1415
return mod;
1516
}
1617

17-
describe('behavior from middleware', () => {
18+
describe('behavior from middleware, standalone', () => {
1819
/** @type {import('./test-utils').Fixture} */
1920
let fixture;
2021
let server;
@@ -53,3 +54,42 @@ describe('behavior from middleware', () => {
5354
});
5455
});
5556
});
57+
58+
describe('behavior from middleware, middleware', () => {
59+
/** @type {import('./test-utils').Fixture} */
60+
let fixture;
61+
let server;
62+
63+
before(async () => {
64+
process.env.ASTRO_NODE_AUTOSTART = 'disabled';
65+
process.env.PRERENDER = false;
66+
fixture = await loadFixture({
67+
root: './fixtures/node-middleware/',
68+
output: 'server',
69+
adapter: nodejs({ mode: 'middleware' }),
70+
});
71+
await fixture.build();
72+
const { handler } = await load();
73+
const app = express();
74+
app.use(handler);
75+
server = app.listen(8888);
76+
});
77+
78+
after(async () => {
79+
server.close();
80+
await fixture.clean();
81+
delete process.env.PRERENDER;
82+
});
83+
84+
it('when mode is standalone', async () => {
85+
const res = await fetch(`http://localhost:8888/ssr`);
86+
87+
expect(res.status).to.equal(200);
88+
89+
const html = await res.text();
90+
const $ = cheerio.load(html);
91+
92+
const body = $('body');
93+
expect(body.text()).to.contain("Here's a random number");
94+
});
95+
});

0 commit comments

Comments
 (0)