Add more tests

This commit is contained in:
Michael Jackson 2019-07-09 23:50:00 -07:00
parent d84a0296b2
commit 681fc99a68
9 changed files with 169 additions and 56 deletions

View File

@ -0,0 +1,34 @@
import request from 'supertest';
import createServer from '../createServer.js';
describe('A request that targets a directory with a trailing slash', () => {
let server;
beforeEach(() => {
server = createServer();
});
describe('when the directory exists', () => {
it('returns an HTML page', done => {
request(server)
.get('/react@16.8.0/umd/')
.end((err, res) => {
expect(res.statusCode).toBe(200);
expect(res.headers['content-type']).toMatch(/\btext\/html\b/);
done();
});
});
});
describe('when the directory does not exist', () => {
it('returns a 404 text error', done => {
request(server)
.get('/react@16.8.0/not-here/')
.end((err, res) => {
expect(res.statusCode).toBe(404);
expect(res.headers['content-type']).toMatch(/\btext\/plain\b/);
done();
});
});
});
});

View File

@ -0,0 +1,40 @@
import request from 'supertest';
import createServer from '../createServer.js';
describe('A request for a directory', () => {
let server;
beforeEach(() => {
server = createServer();
});
describe('when a .js file exists with the same name', () => {
it('is redirected to the .js file', done => {
request(server)
.get('/preact@8.4.2/devtools')
.end((err, res) => {
expect(res.statusCode).toBe(302);
expect(res.headers.location).toEqual('/preact@8.4.2/devtools.js');
done();
});
});
});
describe('when a .json file exists with the same name', () => {
it('is redirected to the .json file');
});
describe('when it contains an index.js file', () => {
it('is redirected to the index.js file', done => {
request(server)
.get('/preact@8.4.2/src/dom')
.end((err, res) => {
expect(res.statusCode).toBe(302);
expect(res.headers.location).toEqual(
'/preact@8.4.2/src/dom/index.js'
);
done();
});
});
});
});

View File

@ -0,0 +1,20 @@
import request from 'supertest';
import createServer from '../createServer.js';
describe('A request for a non-existent file', () => {
let server;
beforeEach(() => {
server = createServer();
});
it('returns a 404 text error', done => {
request(server)
.get('/preact@8.4.2/not-here.js')
.end((err, res) => {
expect(res.statusCode).toBe(404);
expect(res.headers['content-type']).toMatch(/\btext\/plain\b/);
done();
});
});
});

View File

@ -4,13 +4,8 @@ import semver from 'semver';
import AutoIndexApp from '../client/autoIndex/App.js';
import MainTemplate from './utils/MainTemplate.js';
import getEntryPoint from './utils/getEntryPoint.js';
import getGlobalScripts from './utils/getGlobalScripts.js';
import {
createElement,
createHTML,
createScript
} from './utils/markupHelpers.js';
import getScripts from './utils/getScripts.js';
import { createElement, createHTML } from './utils/markupHelpers.js';
const doctype = '<!DOCTYPE html>';
const globalURLs =
@ -40,10 +35,7 @@ export default function serveAutoIndexPage(req, res) {
entries: req.entries
};
const content = createHTML(renderToString(createElement(AutoIndexApp, data)));
const entryPoint = getEntryPoint('autoIndex', 'iife');
const elements = getGlobalScripts(entryPoint, globalURLs).concat(
createScript(entryPoint.code)
);
const elements = getScripts('autoIndex', 'iife', globalURLs);
const html =
doctype +

View File

@ -3,13 +3,8 @@ import { renderToString, renderToStaticMarkup } from 'react-dom/server';
import MainApp from '../client/main/App.js';
import MainTemplate from './utils/MainTemplate.js';
import getEntryPoint from './utils/getEntryPoint.js';
import getGlobalScripts from './utils/getGlobalScripts.js';
import {
createElement,
createHTML,
createScript
} from './utils/markupHelpers.js';
import getScripts from './utils/getScripts.js';
import { createElement, createHTML } from './utils/markupHelpers.js';
const doctype = '<!DOCTYPE html>';
const globalURLs =
@ -27,10 +22,7 @@ const globalURLs =
export default function serveMainPage(req, res) {
const content = createHTML(renderToString(createElement(MainApp)));
const entryPoint = getEntryPoint('main', 'iife');
const elements = getGlobalScripts(entryPoint, globalURLs).concat(
createScript(entryPoint.code)
);
const elements = getScripts('main', 'iife', globalURLs);
const html =
doctype +

View File

@ -1,18 +0,0 @@
// Virtual module id; see rollup.config.js
// eslint-disable-next-line import/no-unresolved
import entryManifest from 'entry-manifest';
export default function getEntryPoint(name, format) {
let entryPoints;
entryManifest.forEach(manifest => {
if (name in manifest) {
entryPoints = manifest[name];
}
});
if (entryPoints) {
return entryPoints.find(e => e.format === format);
}
return null;
}

View File

@ -1,13 +0,0 @@
import { createElement } from './markupHelpers.js';
export default function getGlobalScripts(entryPoint, globalURLs) {
return entryPoint.globalImports.map(id => {
if (process.env.NODE_ENV !== 'production') {
if (!globalURLs[id]) {
throw new Error('Missing global URL for id "%s"', id);
}
}
return createElement('script', { src: globalURLs[id] });
});
}

View File

@ -0,0 +1,42 @@
// Virtual module id; see rollup.config.js
// eslint-disable-next-line import/no-unresolved
import entryManifest from 'entry-manifest';
import { createElement, createScript } from './markupHelpers.js';
function getEntryPoint(name, format) {
let entryPoints;
entryManifest.forEach(manifest => {
if (name in manifest) {
entryPoints = manifest[name];
}
});
if (entryPoints) {
return entryPoints.find(e => e.format === format);
}
return null;
}
function getGlobalScripts(entryPoint, globalURLs) {
return entryPoint.globalImports.map(id => {
if (process.env.NODE_ENV !== 'production') {
if (!globalURLs[id]) {
throw new Error('Missing global URL for id "%s"', id);
}
}
return createElement('script', { src: globalURLs[id] });
});
}
export default function getScripts(entryName, format, globalURLs) {
const entryPoint = getEntryPoint(entryName, format);
if (!entryPoint) return [];
return getGlobalScripts(entryPoint, globalURLs).concat(
createScript(entryPoint.code)
);
}

View File

@ -7,6 +7,25 @@ import { fetchPackage as fetchNpmPackage } from '../utils/npm.js';
import getIntegrity from '../utils/getIntegrity.js';
import getContentType from '../utils/getContentType.js';
function fileRedirect(req, res, entry) {
// Redirect to the file with the extension so it's more
// clear which file is being served.
res
.set({
'Cache-Control': 'public, max-age=31536000', // 1 year
'Cache-Tag': 'redirect, file-redirect'
})
.redirect(
302,
createPackageURL(
req.packageName,
req.packageVersion,
addLeadingSlash(entry.name),
createSearch(req.query)
)
);
}
function indexRedirect(req, res, entry) {
// Redirect to the index file so relative imports
// resolve correctly.
@ -104,7 +123,9 @@ function searchEntries(tarballStream, entryName, wantsIndex) {
const chunks = [];
stream
.on('data', chunk => chunks.push(chunk))
.on('data', chunk => {
chunks.push(chunk);
})
.on('end', () => {
const content = Buffer.concat(chunks);
@ -170,6 +191,10 @@ export default async function findFile(req, res, next) {
.send(`Cannot find "${req.filename}" in ${req.packageSpec}`);
}
if (foundEntry.type === 'file' && foundEntry.name !== entryName) {
return fileRedirect(req, res, foundEntry);
}
// If the foundEntry is a directory and there is no trailing slash
// on the request path, we need to redirect to some "index" file
// inside that directory. This is so our URLs work in a similar way
@ -177,8 +202,7 @@ export default async function findFile(req, res, next) {
// and `lib/index.json` when `lib` is a directory.
if (foundEntry.type === 'directory' && !wantsIndex) {
const indexEntry =
entries[path.join(entryName, 'index.js')] ||
entries[path.join(entryName, 'index.json')];
entries[`${entryName}/index.js`] || entries[`${entryName}/index.json`];
if (indexEntry && indexEntry.type === 'file') {
return indexRedirect(req, res, indexEntry);