Compare commits
74 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
9403799393 | ||
|
|
472c0c198a | ||
|
|
02256ae214 | ||
|
|
e2d325ec9f | ||
|
|
954bcba333 | ||
|
|
709c9992e3 | ||
|
|
9773781262 | ||
|
|
48b1fafc33 | ||
|
|
d1624add66 | ||
|
|
e2206d0e0d | ||
|
|
9cd4da4c39 | ||
|
|
6ded1a5975 | ||
|
|
584ddd1c85 | ||
|
|
4071acf7c0 | ||
|
|
e8773d3196 | ||
|
|
01a519a4d9 | ||
|
|
d9ad1d1b10 | ||
|
|
0826a58995 | ||
|
|
6a74097b0c | ||
|
|
278be67228 | ||
|
|
64921dfc3c | ||
|
|
c8962ccf8c | ||
|
|
664c093391 | ||
|
|
4375feac83 | ||
|
|
4d7d448597 | ||
|
|
2e2b8dcd83 | ||
|
|
b915bab070 | ||
|
|
8530d06d00 | ||
|
|
a43764a971 | ||
|
|
4f6efbda79 | ||
|
|
5573258a10 | ||
|
|
2185f89669 | ||
|
|
e30842caa8 | ||
|
|
ff24877d8f | ||
|
|
9cf90ce01d | ||
|
|
e7f9ddae86 | ||
|
|
ffa1e1f704 | ||
|
|
80bb958b47 | ||
|
|
532f559fc5 | ||
|
|
0bd1b0b8e2 | ||
|
|
10c5ff4169 | ||
|
|
273823dfd7 | ||
|
|
8f064fe5ac | ||
|
|
f29e7efbd6 | ||
|
|
e66e3cd7eb | ||
|
|
ff415b391b | ||
|
|
91182ad0a2 | ||
|
|
467041a3cd | ||
|
|
520949c5e1 | ||
|
|
8c07d9d2ac | ||
|
|
7bd684a80e | ||
|
|
cbb5e8755b | ||
|
|
7ef72dbb77 | ||
|
|
87ff9c2aeb | ||
|
|
2d1f535314 | ||
|
|
cd1b53b80d | ||
|
|
0a7be736c0 | ||
|
|
5ee53a98c6 | ||
|
|
0e8ed6612c | ||
|
|
5ec748b95d | ||
|
|
64b16715cd | ||
|
|
9ea5e5e251 | ||
|
|
68b78f56d6 | ||
|
|
68e93a8fa0 | ||
|
|
e377515867 | ||
|
|
99ae39b8a8 | ||
|
|
1b489f4687 | ||
|
|
91f2c6e49c | ||
|
|
f5e07e9f78 | ||
|
|
17297a9794 | ||
|
|
9ef4f33e38 | ||
|
|
30966ee7f2 | ||
|
|
ae90f774e1 | ||
|
|
0706b5f50a |
6
.gitignore
vendored
@@ -4,10 +4,12 @@ yarn-error.log
|
|||||||
node_modules
|
node_modules
|
||||||
cypress/screenshots
|
cypress/screenshots
|
||||||
test/app/.sapper
|
test/app/.sapper
|
||||||
test/app/app/manifest
|
test/app/src/manifest
|
||||||
|
__sapper__
|
||||||
test/app/export
|
test/app/export
|
||||||
test/app/build
|
test/app/build
|
||||||
sapper
|
sapper
|
||||||
runtime.js
|
runtime.js
|
||||||
dist
|
dist
|
||||||
!rollup.config.js
|
!rollup.config.js
|
||||||
|
templates/*.js
|
||||||
73
CHANGELOG.md
@@ -1,5 +1,78 @@
|
|||||||
# sapper changelog
|
# sapper changelog
|
||||||
|
|
||||||
|
## 0.22.8
|
||||||
|
|
||||||
|
* Ensure CSS placeholders are overwritten ([#462](https://github.com/sveltejs/sapper/pull/462))
|
||||||
|
|
||||||
|
## 0.22.7
|
||||||
|
|
||||||
|
* Fix cookies ([#460](https://github.com/sveltejs/sapper/pull/460))
|
||||||
|
|
||||||
|
## 0.22.6
|
||||||
|
|
||||||
|
* Normalise chunk filenames on Windows ([#456](https://github.com/sveltejs/sapper/pull/456))
|
||||||
|
* Load modules with credentials ([#458](https://github.com/sveltejs/sapper/pull/458))
|
||||||
|
|
||||||
|
## 0.22.5
|
||||||
|
|
||||||
|
* Fix `sapper dev`. Oops.
|
||||||
|
|
||||||
|
## 0.22.4
|
||||||
|
|
||||||
|
* Ensure launcher does not overwrite a module ([#455](https://github.com/sveltejs/sapper/pull/455))
|
||||||
|
|
||||||
|
## 0.22.3
|
||||||
|
|
||||||
|
* Prevent server from accidentally importing dev client
|
||||||
|
|
||||||
|
## 0.22.2
|
||||||
|
|
||||||
|
* Make paths in generated code relative to project
|
||||||
|
|
||||||
|
## 0.22.1
|
||||||
|
|
||||||
|
* Fix `pkg.files`
|
||||||
|
|
||||||
|
## 0.22.0
|
||||||
|
|
||||||
|
* Move generated files into `__sapper__` ([#453](https://github.com/sveltejs/sapper/pull/453))
|
||||||
|
* Change default build and export directories to `__sapper__/build` and `__sapper__/export` ([#453](https://github.com/sveltejs/sapper/pull/453))
|
||||||
|
|
||||||
|
## 0.21.1
|
||||||
|
|
||||||
|
* Read template from build directory in production
|
||||||
|
|
||||||
|
## 0.21.0
|
||||||
|
|
||||||
|
* Change project folder structure ([#432](https://github.com/sveltejs/sapper/issues/432))
|
||||||
|
* Escape filenames ([#446](https://github.com/sveltejs/sapper/pull/446/))
|
||||||
|
|
||||||
|
## 0.20.4
|
||||||
|
|
||||||
|
* Fix legacy build CSS ([#439](https://github.com/sveltejs/sapper/issues/439))
|
||||||
|
* Enable debugging in Chrome and VSCode ([#435](https://github.com/sveltejs/sapper/issues/435))
|
||||||
|
|
||||||
|
## 0.20.3
|
||||||
|
|
||||||
|
* Inject `nonce` attribute if `res.locals.nonce` is present ([#424](https://github.com/sveltejs/sapper/pull/424))
|
||||||
|
* Prevent service worker caching ([#428](https://github.com/sveltejs/sapper/pull/428))
|
||||||
|
* Consistent caching for HTML responses ([#429](https://github.com/sveltejs/sapper/pull/429))
|
||||||
|
|
||||||
|
## 0.20.2
|
||||||
|
|
||||||
|
* Add `immutable` cache control header for hashed assets ([#425](https://github.com/sveltejs/sapper/pull/425))
|
||||||
|
* Handle value-less query string params ([#426](https://github.com/sveltejs/sapper/issues/426))
|
||||||
|
|
||||||
|
## 0.20.1
|
||||||
|
|
||||||
|
* Update shimport
|
||||||
|
|
||||||
|
## 0.20.0
|
||||||
|
|
||||||
|
* Decode `req.params` and `req.query` ([#417](https://github.com/sveltejs/sapper/issues/417))
|
||||||
|
* Decode URLs before writing files in `sapper export` ([#414](https://github.com/sveltejs/sapper/pull/414))
|
||||||
|
* Generate server sourcemaps for Rollup apps in dev mode ([#418](https://github.com/sveltejs/sapper/pull/418))
|
||||||
|
|
||||||
## 0.19.3
|
## 0.19.3
|
||||||
|
|
||||||
* Better unicode route handling ([#347](https://github.com/sveltejs/sapper/issues/347))
|
* Better unicode route handling ([#347](https://github.com/sveltejs/sapper/issues/347))
|
||||||
|
|||||||
@@ -1 +0,0 @@
|
|||||||
<svelte:component this={child.component} {...child.props}/>
|
|
||||||
1
index.js
Normal file
@@ -0,0 +1 @@
|
|||||||
|
throw new Error(`As of Sapper 0.22, you should not import 'sapper' directly. See https://sapper.svelte.technology/guide#0-21-to-0-22 for more information`);
|
||||||
1015
package-lock.json
generated
12
package.json
@@ -1,26 +1,24 @@
|
|||||||
{
|
{
|
||||||
"name": "sapper",
|
"name": "sapper",
|
||||||
"version": "0.19.3",
|
"version": "0.22.8",
|
||||||
"description": "Military-grade apps, engineered by Svelte",
|
"description": "Military-grade apps, engineered by Svelte",
|
||||||
"main": "dist/middleware.js",
|
|
||||||
"bin": {
|
"bin": {
|
||||||
"sapper": "./sapper"
|
"sapper": "./sapper"
|
||||||
},
|
},
|
||||||
"files": [
|
"files": [
|
||||||
"*.js",
|
"*.js",
|
||||||
"runtime",
|
|
||||||
"webpack",
|
"webpack",
|
||||||
"config",
|
"config",
|
||||||
"sapper",
|
"sapper",
|
||||||
"components",
|
"dist/*.js",
|
||||||
"dist/*.js"
|
"templates/*.js"
|
||||||
],
|
],
|
||||||
"directories": {
|
"directories": {
|
||||||
"test": "test"
|
"test": "test"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"html-minifier": "^3.5.16",
|
"html-minifier": "^3.5.16",
|
||||||
"shimport": "^0.0.10",
|
"shimport": "0.0.11",
|
||||||
"source-map-support": "^0.5.6",
|
"source-map-support": "^0.5.6",
|
||||||
"sourcemap-codec": "^1.4.1",
|
"sourcemap-codec": "^1.4.1",
|
||||||
"string-hash": "^1.1.3",
|
"string-hash": "^1.1.3",
|
||||||
@@ -32,6 +30,7 @@
|
|||||||
"@types/mocha": "^5.2.5",
|
"@types/mocha": "^5.2.5",
|
||||||
"@types/node": "^10.7.1",
|
"@types/node": "^10.7.1",
|
||||||
"@types/rimraf": "^2.0.2",
|
"@types/rimraf": "^2.0.2",
|
||||||
|
"agadoo": "^1.0.1",
|
||||||
"cheap-watch": "^0.3.0",
|
"cheap-watch": "^0.3.0",
|
||||||
"compression": "^1.7.1",
|
"compression": "^1.7.1",
|
||||||
"cookie": "^0.3.1",
|
"cookie": "^0.3.1",
|
||||||
@@ -74,6 +73,7 @@
|
|||||||
"test": "mocha --opts mocha.opts",
|
"test": "mocha --opts mocha.opts",
|
||||||
"pretest": "npm run build",
|
"pretest": "npm run build",
|
||||||
"build": "rm -rf dist && rollup -c",
|
"build": "rm -rf dist && rollup -c",
|
||||||
|
"prepare": "npm run build",
|
||||||
"dev": "rollup -cw",
|
"dev": "rollup -cw",
|
||||||
"prepublishOnly": "npm test",
|
"prepublishOnly": "npm test",
|
||||||
"update_mime_types": "curl http://svn.apache.org/repos/asf/httpd/httpd/trunk/docs/conf/mime.types | grep -e \"^[^#]\" > src/middleware/mime-types.md"
|
"update_mime_types": "curl http://svn.apache.org/repos/asf/httpd/httpd/trunk/docs/conf/mime.types | grep -e \"^[^#]\" > src/middleware/mime-types.md"
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ import json from 'rollup-plugin-json';
|
|||||||
import resolve from 'rollup-plugin-node-resolve';
|
import resolve from 'rollup-plugin-node-resolve';
|
||||||
import commonjs from 'rollup-plugin-commonjs';
|
import commonjs from 'rollup-plugin-commonjs';
|
||||||
import pkg from './package.json';
|
import pkg from './package.json';
|
||||||
|
import { builtinModules } from 'module';
|
||||||
|
|
||||||
const external = [].concat(
|
const external = [].concat(
|
||||||
Object.keys(pkg.dependencies),
|
Object.keys(pkg.dependencies),
|
||||||
@@ -11,27 +12,37 @@ const external = [].concat(
|
|||||||
'sapper/core.js'
|
'sapper/core.js'
|
||||||
);
|
);
|
||||||
|
|
||||||
export default [
|
function template(kind, external) {
|
||||||
{
|
return {
|
||||||
input: `src/runtime/index.ts`,
|
input: `templates/src/${kind}/index.ts`,
|
||||||
output: {
|
output: {
|
||||||
file: `runtime.js`,
|
file: `templates/${kind}.js`,
|
||||||
format: 'es'
|
format: 'es'
|
||||||
},
|
},
|
||||||
|
external,
|
||||||
plugins: [
|
plugins: [
|
||||||
|
resolve(),
|
||||||
|
commonjs(),
|
||||||
|
string({
|
||||||
|
include: '**/*.md'
|
||||||
|
}),
|
||||||
typescript({
|
typescript({
|
||||||
typescript: require('typescript'),
|
typescript: require('typescript'),
|
||||||
target: "ES2017"
|
target: "ES2017"
|
||||||
})
|
})
|
||||||
]
|
]
|
||||||
},
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export default [
|
||||||
|
template('client', ['__ROOT__', '__ERROR__']),
|
||||||
|
template('server', builtinModules),
|
||||||
|
|
||||||
{
|
{
|
||||||
input: [
|
input: [
|
||||||
`src/api.ts`,
|
`src/api.ts`,
|
||||||
`src/cli.ts`,
|
`src/cli.ts`,
|
||||||
`src/core.ts`,
|
`src/core.ts`,
|
||||||
`src/middleware.ts`,
|
|
||||||
`src/rollup.ts`,
|
`src/rollup.ts`,
|
||||||
`src/webpack.ts`
|
`src/webpack.ts`
|
||||||
],
|
],
|
||||||
@@ -42,9 +53,6 @@ export default [
|
|||||||
},
|
},
|
||||||
external,
|
external,
|
||||||
plugins: [
|
plugins: [
|
||||||
string({
|
|
||||||
include: '**/*.md'
|
|
||||||
}),
|
|
||||||
json(),
|
json(),
|
||||||
resolve(),
|
resolve(),
|
||||||
commonjs(),
|
commonjs(),
|
||||||
|
|||||||
@@ -1 +0,0 @@
|
|||||||
This directory exists for legacy reasons and should be deleted before releasing version 1.
|
|
||||||
@@ -1,2 +0,0 @@
|
|||||||
console.error('sapper/runtime/app.js has been deprecated in favour of sapper/runtime.js');
|
|
||||||
export * from '../runtime.js';
|
|
||||||
@@ -3,18 +3,16 @@ import * as path from 'path';
|
|||||||
import mkdirp from 'mkdirp';
|
import mkdirp from 'mkdirp';
|
||||||
import rimraf from 'rimraf';
|
import rimraf from 'rimraf';
|
||||||
import { EventEmitter } from 'events';
|
import { EventEmitter } from 'events';
|
||||||
import * as codec from 'sourcemap-codec';
|
|
||||||
import hash from 'string-hash';
|
|
||||||
import minify_html from './utils/minify_html';
|
import minify_html from './utils/minify_html';
|
||||||
import { create_compilers, create_main_manifests, create_manifest_data, create_serviceworker_manifest } from '../core';
|
import { create_compilers, create_main_manifests, create_manifest_data, create_serviceworker_manifest } from '../core';
|
||||||
import * as events from './interfaces';
|
import * as events from './interfaces';
|
||||||
import { copy_shimport } from './utils/copy_shimport';
|
import { copy_shimport } from './utils/copy_shimport';
|
||||||
import { Dirs, PageComponent } from '../interfaces';
|
import { Dirs } from '../interfaces';
|
||||||
import { CompileResult } from '../core/create_compilers/interfaces';
|
import read_template from '../core/read_template';
|
||||||
|
|
||||||
type Opts = {
|
type Opts = {
|
||||||
legacy: boolean;
|
legacy: boolean;
|
||||||
bundler: string;
|
bundler: 'rollup' | 'webpack';
|
||||||
};
|
};
|
||||||
|
|
||||||
export function build(opts: Opts, dirs: Dirs) {
|
export function build(opts: Opts, dirs: Dirs) {
|
||||||
@@ -39,9 +37,9 @@ async function execute(emitter: EventEmitter, opts: Opts, dirs: Dirs) {
|
|||||||
mkdirp.sync(`${dirs.dest}/client`);
|
mkdirp.sync(`${dirs.dest}/client`);
|
||||||
copy_shimport(dirs.dest);
|
copy_shimport(dirs.dest);
|
||||||
|
|
||||||
// minify app/template.html
|
// minify src/template.html
|
||||||
// TODO compile this to a function? could be quicker than str.replace(...).replace(...).replace(...)
|
// TODO compile this to a function? could be quicker than str.replace(...).replace(...).replace(...)
|
||||||
const template = fs.readFileSync(`${dirs.app}/template.html`, 'utf-8');
|
const template = read_template();
|
||||||
|
|
||||||
// remove this in a future version
|
// remove this in a future version
|
||||||
if (template.indexOf('%sapper.base%') === -1) {
|
if (template.indexOf('%sapper.base%') === -1) {
|
||||||
@@ -54,10 +52,10 @@ async function execute(emitter: EventEmitter, opts: Opts, dirs: Dirs) {
|
|||||||
|
|
||||||
const manifest_data = create_manifest_data();
|
const manifest_data = create_manifest_data();
|
||||||
|
|
||||||
// create app/manifest/client.js and app/manifest/server.js
|
// create src/manifest/client.js and src/manifest/server.js
|
||||||
create_main_manifests({ bundler: opts.bundler, manifest_data });
|
create_main_manifests({ bundler: opts.bundler, manifest_data });
|
||||||
|
|
||||||
const { client, server, serviceworker } = create_compilers(opts.bundler, dirs);
|
const { client, server, serviceworker } = await create_compilers(opts.bundler);
|
||||||
|
|
||||||
const client_result = await client.compile();
|
const client_result = await client.compile();
|
||||||
emitter.emit('build', <events.BuildEvent>{
|
emitter.emit('build', <events.BuildEvent>{
|
||||||
@@ -70,7 +68,7 @@ async function execute(emitter: EventEmitter, opts: Opts, dirs: Dirs) {
|
|||||||
|
|
||||||
if (opts.legacy) {
|
if (opts.legacy) {
|
||||||
process.env.SAPPER_LEGACY_BUILD = 'true';
|
process.env.SAPPER_LEGACY_BUILD = 'true';
|
||||||
const { client } = create_compilers(opts.bundler, dirs);
|
const { client } = await create_compilers(opts.bundler);
|
||||||
|
|
||||||
const client_result = await client.compile();
|
const client_result = await client.compile();
|
||||||
|
|
||||||
@@ -80,6 +78,7 @@ async function execute(emitter: EventEmitter, opts: Opts, dirs: Dirs) {
|
|||||||
result: client_result
|
result: client_result
|
||||||
});
|
});
|
||||||
|
|
||||||
|
client_result.to_json(manifest_data, dirs);
|
||||||
build_info.legacy_assets = client_result.assets;
|
build_info.legacy_assets = client_result.assets;
|
||||||
delete process.env.SAPPER_LEGACY_BUILD;
|
delete process.env.SAPPER_LEGACY_BUILD;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -15,6 +15,7 @@ import * as events from './interfaces';
|
|||||||
import validate_bundler from '../cli/utils/validate_bundler';
|
import validate_bundler from '../cli/utils/validate_bundler';
|
||||||
import { copy_shimport } from './utils/copy_shimport';
|
import { copy_shimport } from './utils/copy_shimport';
|
||||||
import { ManifestData } from '../interfaces';
|
import { ManifestData } from '../interfaces';
|
||||||
|
import read_template from '../core/read_template';
|
||||||
|
|
||||||
export function dev(opts) {
|
export function dev(opts) {
|
||||||
return new Watcher(opts);
|
return new Watcher(opts);
|
||||||
@@ -23,7 +24,7 @@ export function dev(opts) {
|
|||||||
class Watcher extends EventEmitter {
|
class Watcher extends EventEmitter {
|
||||||
bundler: string;
|
bundler: string;
|
||||||
dirs: {
|
dirs: {
|
||||||
app: string;
|
src: string;
|
||||||
dest: string;
|
dest: string;
|
||||||
routes: string;
|
routes: string;
|
||||||
rollup: string;
|
rollup: string;
|
||||||
@@ -36,6 +37,8 @@ class Watcher extends EventEmitter {
|
|||||||
live: boolean;
|
live: boolean;
|
||||||
hot: boolean;
|
hot: boolean;
|
||||||
|
|
||||||
|
devtools_port: number;
|
||||||
|
|
||||||
dev_server: DevServer;
|
dev_server: DevServer;
|
||||||
proc: child_process.ChildProcess;
|
proc: child_process.ChildProcess;
|
||||||
filewatchers: Array<{ close: () => void }>;
|
filewatchers: Array<{ close: () => void }>;
|
||||||
@@ -51,23 +54,25 @@ class Watcher extends EventEmitter {
|
|||||||
}
|
}
|
||||||
|
|
||||||
constructor({
|
constructor({
|
||||||
app = locations.app(),
|
src = locations.src(),
|
||||||
dest = locations.dest(),
|
dest = locations.dest(),
|
||||||
routes = locations.routes(),
|
routes = locations.routes(),
|
||||||
'dev-port': dev_port,
|
'dev-port': dev_port,
|
||||||
live,
|
live,
|
||||||
hot,
|
hot,
|
||||||
|
'devtools-port': devtools_port,
|
||||||
bundler,
|
bundler,
|
||||||
webpack = 'webpack',
|
webpack = 'webpack',
|
||||||
rollup = 'rollup',
|
rollup = 'rollup',
|
||||||
port = +process.env.PORT
|
port = +process.env.PORT
|
||||||
}: {
|
}: {
|
||||||
app: string,
|
src: string,
|
||||||
dest: string,
|
dest: string,
|
||||||
routes: string,
|
routes: string,
|
||||||
'dev-port': number,
|
'dev-port': number,
|
||||||
live: boolean,
|
live: boolean,
|
||||||
hot: boolean,
|
hot: boolean,
|
||||||
|
'devtools-port': number,
|
||||||
bundler?: string,
|
bundler?: string,
|
||||||
webpack: string,
|
webpack: string,
|
||||||
rollup: string,
|
rollup: string,
|
||||||
@@ -76,7 +81,7 @@ class Watcher extends EventEmitter {
|
|||||||
super();
|
super();
|
||||||
|
|
||||||
this.bundler = validate_bundler(bundler);
|
this.bundler = validate_bundler(bundler);
|
||||||
this.dirs = { app, dest, routes, webpack, rollup };
|
this.dirs = { src, dest, routes, webpack, rollup };
|
||||||
this.port = port;
|
this.port = port;
|
||||||
this.closed = false;
|
this.closed = false;
|
||||||
|
|
||||||
@@ -84,6 +89,8 @@ class Watcher extends EventEmitter {
|
|||||||
this.live = live;
|
this.live = live;
|
||||||
this.hot = hot;
|
this.hot = hot;
|
||||||
|
|
||||||
|
this.devtools_port = devtools_port;
|
||||||
|
|
||||||
this.filewatchers = [];
|
this.filewatchers = [];
|
||||||
|
|
||||||
this.current_build = {
|
this.current_build = {
|
||||||
@@ -94,7 +101,7 @@ class Watcher extends EventEmitter {
|
|||||||
};
|
};
|
||||||
|
|
||||||
// remove this in a future version
|
// remove this in a future version
|
||||||
const template = fs.readFileSync(path.join(app, 'template.html'), 'utf-8');
|
const template = read_template();
|
||||||
if (template.indexOf('%sapper.base%') === -1) {
|
if (template.indexOf('%sapper.base%') === -1) {
|
||||||
const error = new Error(`As of Sapper v0.10, your template.html file must include %sapper.base% in the <head>`);
|
const error = new Error(`As of Sapper v0.10, your template.html file must include %sapper.base% in the <head>`);
|
||||||
error.code = `missing-sapper-base`;
|
error.code = `missing-sapper-base`;
|
||||||
@@ -129,6 +136,9 @@ class Watcher extends EventEmitter {
|
|||||||
|
|
||||||
if (!this.dev_port) this.dev_port = await ports.find(10000);
|
if (!this.dev_port) this.dev_port = await ports.find(10000);
|
||||||
|
|
||||||
|
// Chrome looks for debugging targets on ports 9222 and 9229 by default
|
||||||
|
if (!this.devtools_port) this.devtools_port = await ports.find(9222);
|
||||||
|
|
||||||
let manifest_data: ManifestData;
|
let manifest_data: ManifestData;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
@@ -166,7 +176,7 @@ class Watcher extends EventEmitter {
|
|||||||
}
|
}
|
||||||
),
|
),
|
||||||
|
|
||||||
fs.watch(`${locations.app()}/template.html`, () => {
|
fs.watch(`${locations.src()}/template.html`, () => {
|
||||||
this.dev_server.send({
|
this.dev_server.send({
|
||||||
action: 'reload'
|
action: 'reload'
|
||||||
});
|
});
|
||||||
@@ -176,7 +186,7 @@ class Watcher extends EventEmitter {
|
|||||||
let deferred = new Deferred();
|
let deferred = new Deferred();
|
||||||
|
|
||||||
// TODO watch the configs themselves?
|
// TODO watch the configs themselves?
|
||||||
const compilers: Compilers = create_compilers(this.bundler, this.dirs);
|
const compilers: Compilers = await create_compilers(this.bundler, this.dirs);
|
||||||
|
|
||||||
let log = '';
|
let log = '';
|
||||||
|
|
||||||
@@ -238,12 +248,21 @@ class Watcher extends EventEmitter {
|
|||||||
restart();
|
restart();
|
||||||
}
|
}
|
||||||
|
|
||||||
this.proc = child_process.fork(`${dest}/server.js`, [], {
|
// we need to give the child process its own DevTools port,
|
||||||
|
// otherwise Node will try to use the parent's (and fail)
|
||||||
|
const debugArgRegex = /--inspect(?:-brk|-port)?|--debug-port/;
|
||||||
|
const execArgv = process.execArgv.slice();
|
||||||
|
if (execArgv.some((arg: string) => !!arg.match(debugArgRegex))) {
|
||||||
|
execArgv.push(`--inspect-port=${this.devtools_port}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
this.proc = child_process.fork(`${dest}/server/server.js`, [], {
|
||||||
cwd: process.cwd(),
|
cwd: process.cwd(),
|
||||||
env: Object.assign({
|
env: Object.assign({
|
||||||
PORT: this.port
|
PORT: this.port
|
||||||
}, process.env),
|
}, process.env),
|
||||||
stdio: ['ipc']
|
stdio: ['ipc'],
|
||||||
|
execArgv
|
||||||
});
|
});
|
||||||
|
|
||||||
this.proc.stdout.on('data', chunk => {
|
this.proc.stdout.on('data', chunk => {
|
||||||
|
|||||||
@@ -13,6 +13,7 @@ import * as events from './interfaces';
|
|||||||
type Opts = {
|
type Opts = {
|
||||||
build: string,
|
build: string,
|
||||||
dest: string,
|
dest: string,
|
||||||
|
static: string,
|
||||||
basepath?: string,
|
basepath?: string,
|
||||||
timeout: number | false
|
timeout: number | false
|
||||||
};
|
};
|
||||||
@@ -46,7 +47,7 @@ async function execute(emitter: EventEmitter, opts: Opts) {
|
|||||||
// Prep output directory
|
// Prep output directory
|
||||||
sander.rimrafSync(export_dir);
|
sander.rimrafSync(export_dir);
|
||||||
|
|
||||||
sander.copydirSync('assets').to(export_dir);
|
sander.copydirSync(opts.static).to(export_dir);
|
||||||
sander.copydirSync(opts.build, 'client').to(export_dir, 'client');
|
sander.copydirSync(opts.build, 'client').to(export_dir, 'client');
|
||||||
|
|
||||||
if (sander.existsSync(opts.build, 'service-worker.js')) {
|
if (sander.existsSync(opts.build, 'service-worker.js')) {
|
||||||
@@ -70,7 +71,7 @@ async function execute(emitter: EventEmitter, opts: Opts) {
|
|||||||
message: `Crawling ${root.href}`
|
message: `Crawling ${root.href}`
|
||||||
});
|
});
|
||||||
|
|
||||||
const proc = child_process.fork(path.resolve(`${opts.build}/server.js`), [], {
|
const proc = child_process.fork(path.resolve(`${opts.build}/server/server.js`), [], {
|
||||||
cwd: process.cwd(),
|
cwd: process.cwd(),
|
||||||
env: Object.assign({
|
env: Object.assign({
|
||||||
PORT: port,
|
PORT: port,
|
||||||
@@ -85,7 +86,7 @@ async function execute(emitter: EventEmitter, opts: Opts) {
|
|||||||
|
|
||||||
function save(path: string, status: number, type: string, body: string) {
|
function save(path: string, status: number, type: string, body: string) {
|
||||||
const { pathname } = resolve(origin, path);
|
const { pathname } = resolve(origin, path);
|
||||||
let file = pathname.slice(1);
|
let file = decodeURIComponent(pathname.slice(1));
|
||||||
|
|
||||||
if (saved.has(file)) return;
|
if (saved.has(file)) return;
|
||||||
saved.add(file);
|
saved.add(file);
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import * as child_process from 'child_process';
|
import * as child_process from 'child_process';
|
||||||
import { CompileResult } from '../core/create_compilers';
|
import { CompileResult } from '../core/create_compilers/interfaces';
|
||||||
|
|
||||||
export type ReadyEvent = {
|
export type ReadyEvent = {
|
||||||
port: number;
|
port: number;
|
||||||
|
|||||||
@@ -33,7 +33,7 @@ prog.command('build [dest]')
|
|||||||
.option('--bundler', 'Specify a bundler (rollup or webpack, blank for auto)')
|
.option('--bundler', 'Specify a bundler (rollup or webpack, blank for auto)')
|
||||||
.option('--legacy', 'Create separate legacy build')
|
.option('--legacy', 'Create separate legacy build')
|
||||||
.example(`build custom-dir -p 4567`)
|
.example(`build custom-dir -p 4567`)
|
||||||
.action(async (dest = 'build', opts: {
|
.action(async (dest = '__sapper__/build', opts: {
|
||||||
port: string,
|
port: string,
|
||||||
legacy: boolean,
|
legacy: boolean,
|
||||||
bundler?: string
|
bundler?: string
|
||||||
@@ -58,7 +58,7 @@ prog.command('build [dest]')
|
|||||||
process.env.PORT = process.env.PORT || ${opts.port || 3000};
|
process.env.PORT = process.env.PORT || ${opts.port || 3000};
|
||||||
|
|
||||||
console.log('Starting server on port ' + process.env.PORT);
|
console.log('Starting server on port ' + process.env.PORT);
|
||||||
require('./server.js');
|
require('./server/server.js');
|
||||||
`.replace(/^\t+/gm, '').trim());
|
`.replace(/^\t+/gm, '').trim());
|
||||||
|
|
||||||
console.error(`\n> Finished in ${elapsed(start)}. Type ${colors.bold.cyan(`node ${dest}`)} to run the app.`);
|
console.error(`\n> Finished in ${elapsed(start)}. Type ${colors.bold.cyan(`node ${dest}`)} to run the app.`);
|
||||||
@@ -80,12 +80,12 @@ prog.command('start [dir]')
|
|||||||
prog.command('export [dest]')
|
prog.command('export [dest]')
|
||||||
.describe('Export your app as static files (if possible)')
|
.describe('Export your app as static files (if possible)')
|
||||||
.option('--build', '(Re)build app before exporting', true)
|
.option('--build', '(Re)build app before exporting', true)
|
||||||
.option('--build-dir', 'Specify a custom temporary build directory', '.sapper/prod')
|
.option('--build-dir', 'Specify a custom temporary build directory', '__sapper__/build')
|
||||||
.option('--basepath', 'Specify a base path')
|
.option('--basepath', 'Specify a base path')
|
||||||
.option('--timeout', 'Milliseconds to wait for a page (--no-timeout to disable)', 5000)
|
.option('--timeout', 'Milliseconds to wait for a page (--no-timeout to disable)', 5000)
|
||||||
.option('--legacy', 'Create separate legacy build')
|
.option('--legacy', 'Create separate legacy build')
|
||||||
.option('--bundler', 'Specify a bundler (rollup or webpack, blank for auto)')
|
.option('--bundler', 'Specify a bundler (rollup or webpack, blank for auto)')
|
||||||
.action(async (dest = 'export', opts: {
|
.action(async (dest = '__sapper__/export', opts: {
|
||||||
build: boolean,
|
build: boolean,
|
||||||
legacy: boolean,
|
legacy: boolean,
|
||||||
bundler?: string,
|
bundler?: string,
|
||||||
|
|||||||
@@ -18,7 +18,7 @@ export function build(opts: { bundler?: string, legacy?: boolean }) {
|
|||||||
bundler
|
bundler
|
||||||
}, {
|
}, {
|
||||||
dest: locations.dest(),
|
dest: locations.dest(),
|
||||||
app: locations.app(),
|
src: locations.src(),
|
||||||
routes: locations.routes(),
|
routes: locations.routes(),
|
||||||
webpack: 'webpack',
|
webpack: 'webpack',
|
||||||
rollup: 'rollup'
|
rollup: 'rollup'
|
||||||
|
|||||||
@@ -15,6 +15,7 @@ export function exporter(export_dir: string, {
|
|||||||
try {
|
try {
|
||||||
const emitter = _exporter({
|
const emitter = _exporter({
|
||||||
build: locations.dest(),
|
build: locations.dest(),
|
||||||
|
static: locations.static(),
|
||||||
dest: export_dir,
|
dest: export_dir,
|
||||||
basepath,
|
basepath,
|
||||||
timeout
|
timeout
|
||||||
|
|||||||
@@ -1,15 +1,19 @@
|
|||||||
import * as fs from 'fs';
|
import * as fs from 'fs';
|
||||||
|
|
||||||
export default function validate_bundler(bundler?: string) {
|
export default function validate_bundler(bundler?: 'rollup' | 'webpack') {
|
||||||
if (!bundler) {
|
if (!bundler) {
|
||||||
bundler = (
|
bundler = (
|
||||||
fs.existsSync('rollup') ? 'rollup' :
|
fs.existsSync('rollup.config.js') ? 'rollup' :
|
||||||
fs.existsSync('webpack') ? 'webpack' :
|
fs.existsSync('webpack.config.js') ? 'webpack' :
|
||||||
null
|
null
|
||||||
);
|
);
|
||||||
|
|
||||||
if (!bundler) {
|
if (!bundler) {
|
||||||
throw new Error(`Could not find a 'rollup' or 'webpack' directory`);
|
// TODO remove in a future version
|
||||||
|
deprecate_dir('rollup');
|
||||||
|
deprecate_dir('webpack');
|
||||||
|
|
||||||
|
throw new Error(`Could not find rollup.config.js or webpack.config.js`);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -18,4 +22,17 @@ export default function validate_bundler(bundler?: string) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
return bundler;
|
return bundler;
|
||||||
|
}
|
||||||
|
|
||||||
|
function deprecate_dir(bundler: 'rollup' | 'webpack') {
|
||||||
|
try {
|
||||||
|
const stats = fs.statSync(bundler);
|
||||||
|
if (!stats.isDirectory()) return;
|
||||||
|
} catch (err) {
|
||||||
|
// do nothing
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO link to docs, once those docs exist
|
||||||
|
throw new Error(`As of Sapper 0.21, build configuration should be placed in a single ${bundler}.config.js file`);
|
||||||
}
|
}
|
||||||
@@ -4,7 +4,8 @@ export const dev = () => process.env.NODE_ENV !== 'production';
|
|||||||
|
|
||||||
export const locations = {
|
export const locations = {
|
||||||
base: () => path.resolve(process.env.SAPPER_BASE || ''),
|
base: () => path.resolve(process.env.SAPPER_BASE || ''),
|
||||||
app: () => path.resolve(process.env.SAPPER_BASE || '', process.env.SAPPER_APP || 'app'),
|
src: () => path.resolve(process.env.SAPPER_BASE || '', process.env.SAPPER_SRC || 'src'),
|
||||||
routes: () => path.resolve(process.env.SAPPER_BASE || '', process.env.SAPPER_ROUTES || 'routes'),
|
static: () => path.resolve(process.env.SAPPER_BASE || '', process.env.SAPPER_STATIC || 'static'),
|
||||||
dest: () => path.resolve(process.env.SAPPER_BASE || '', process.env.SAPPER_DEST || `.sapper/${dev() ? 'dev' : 'prod'}`)
|
routes: () => path.resolve(process.env.SAPPER_BASE || '', process.env.SAPPER_ROUTES || 'src/routes'),
|
||||||
|
dest: () => path.resolve(process.env.SAPPER_BASE || '', process.env.SAPPER_DEST || `__sapper__/${dev() ? 'dev' : 'build'}`)
|
||||||
};
|
};
|
||||||
@@ -15,8 +15,8 @@ export default class RollupCompiler {
|
|||||||
chunks: any[];
|
chunks: any[];
|
||||||
css_files: Array<{ id: string, code: string }>;
|
css_files: Array<{ id: string, code: string }>;
|
||||||
|
|
||||||
constructor(config: string) {
|
constructor(config: any) {
|
||||||
this._ = this.get_config(path.resolve(config));
|
this._ = this.get_config(config);
|
||||||
this.input = null;
|
this.input = null;
|
||||||
this.warnings = [];
|
this.warnings = [];
|
||||||
this.errors = [];
|
this.errors = [];
|
||||||
@@ -24,31 +24,8 @@ export default class RollupCompiler {
|
|||||||
this.css_files = [];
|
this.css_files = [];
|
||||||
}
|
}
|
||||||
|
|
||||||
async get_config(input: string) {
|
async get_config(mod: any) {
|
||||||
if (!rollup) rollup = relative('rollup', process.cwd());
|
// TODO this is hacky, and doesn't need to apply to all three compilers
|
||||||
|
|
||||||
const bundle = await rollup.rollup({
|
|
||||||
input,
|
|
||||||
external: (id: string) => {
|
|
||||||
return (id[0] !== '.' && !path.isAbsolute(id)) || id.slice(-5, id.length) === '.json';
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
const { code } = await bundle.generate({ format: 'cjs' });
|
|
||||||
|
|
||||||
// temporarily override require
|
|
||||||
const defaultLoader = require.extensions['.js'];
|
|
||||||
require.extensions['.js'] = (module: any, filename: string) => {
|
|
||||||
if (filename === input) {
|
|
||||||
module._compile(code, filename);
|
|
||||||
} else {
|
|
||||||
defaultLoader(module, filename);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const mod: any = require(input);
|
|
||||||
delete require.cache[input];
|
|
||||||
|
|
||||||
(mod.plugins || (mod.plugins = [])).push({
|
(mod.plugins || (mod.plugins = [])).push({
|
||||||
name: 'sapper-internal',
|
name: 'sapper-internal',
|
||||||
options: (opts: any) => {
|
options: (opts: any) => {
|
||||||
@@ -157,4 +134,34 @@ export default class RollupCompiler {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static async load_config() {
|
||||||
|
if (!rollup) rollup = relative('rollup', process.cwd());
|
||||||
|
|
||||||
|
const input = path.resolve('rollup.config.js');
|
||||||
|
|
||||||
|
const bundle = await rollup.rollup({
|
||||||
|
input,
|
||||||
|
external: (id: string) => {
|
||||||
|
return (id[0] !== '.' && !path.isAbsolute(id)) || id.slice(-5, id.length) === '.json';
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
const { code } = await bundle.generate({ format: 'cjs' });
|
||||||
|
|
||||||
|
// temporarily override require
|
||||||
|
const defaultLoader = require.extensions['.js'];
|
||||||
|
require.extensions['.js'] = (module: any, filename: string) => {
|
||||||
|
if (filename === input) {
|
||||||
|
module._compile(code, filename);
|
||||||
|
} else {
|
||||||
|
defaultLoader(module, filename);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const config: any = require(input);
|
||||||
|
delete require.cache[input];
|
||||||
|
|
||||||
|
return config;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
@@ -38,11 +38,18 @@ export default class RollupResult implements CompileResult {
|
|||||||
// webpack, but we can have a route -> [chunk] map or something
|
// webpack, but we can have a route -> [chunk] map or something
|
||||||
this.assets = {};
|
this.assets = {};
|
||||||
|
|
||||||
compiler.chunks.forEach(chunk => {
|
if (typeof compiler.input === 'string') {
|
||||||
if (compiler.input in chunk.modules) {
|
compiler.chunks.forEach(chunk => {
|
||||||
this.assets.main = chunk.fileName;
|
if (compiler.input in chunk.modules) {
|
||||||
|
this.assets.main = chunk.fileName;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
for (const name in compiler.input) {
|
||||||
|
const file = compiler.input[name];
|
||||||
|
this.assets[name] = compiler.chunks.find(chunk => file in chunk.modules).fileName;
|
||||||
}
|
}
|
||||||
});
|
}
|
||||||
|
|
||||||
this.summary = compiler.chunks.map(chunk => {
|
this.summary = compiler.chunks.map(chunk => {
|
||||||
const size_color = chunk.code.length > 150000 ? colors.bold.red : chunk.code.length > 50000 ? colors.bold.yellow : colors.bold.white;
|
const size_color = chunk.code.length > 150000 ? colors.bold.red : chunk.code.length > 50000 ? colors.bold.yellow : colors.bold.white;
|
||||||
|
|||||||
@@ -1,4 +1,3 @@
|
|||||||
import * as path from 'path';
|
|
||||||
import relative from 'require-relative';
|
import relative from 'require-relative';
|
||||||
import { CompileResult } from './interfaces';
|
import { CompileResult } from './interfaces';
|
||||||
import WebpackResult from './WebpackResult';
|
import WebpackResult from './WebpackResult';
|
||||||
@@ -8,9 +7,9 @@ let webpack: any;
|
|||||||
export class WebpackCompiler {
|
export class WebpackCompiler {
|
||||||
_: any;
|
_: any;
|
||||||
|
|
||||||
constructor(config: string) {
|
constructor(config: any) {
|
||||||
if (!webpack) webpack = relative('webpack', process.cwd());
|
if (!webpack) webpack = relative('webpack', process.cwd());
|
||||||
this._ = webpack(require(path.resolve(config)));
|
this._ = webpack(config);
|
||||||
}
|
}
|
||||||
|
|
||||||
oninvalid(cb: (filename: string) => void) {
|
oninvalid(cb: (filename: string) => void) {
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ import hash from 'string-hash';
|
|||||||
import * as codec from 'sourcemap-codec';
|
import * as codec from 'sourcemap-codec';
|
||||||
import { PageComponent, Dirs } from '../../interfaces';
|
import { PageComponent, Dirs } from '../../interfaces';
|
||||||
import { CompileResult } from './interfaces';
|
import { CompileResult } from './interfaces';
|
||||||
|
import { posixify } from '../utils'
|
||||||
|
|
||||||
const inline_sourcemap_header = 'data:application/json;charset=utf-8;base64,';
|
const inline_sourcemap_header = 'data:application/json;charset=utf-8;base64,';
|
||||||
|
|
||||||
@@ -75,7 +76,7 @@ export default function extract_css(client_result: CompileResult, components: Pa
|
|||||||
const component_owners = new Map();
|
const component_owners = new Map();
|
||||||
client_result.chunks.forEach(chunk => {
|
client_result.chunks.forEach(chunk => {
|
||||||
chunk.modules.forEach(module => {
|
chunk.modules.forEach(module => {
|
||||||
const component = path.relative(dirs.routes, module);
|
const component = posixify(path.relative(dirs.routes, module));
|
||||||
component_owners.set(component, chunk);
|
component_owners.set(component, chunk);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@@ -169,7 +170,8 @@ export default function extract_css(client_result: CompileResult, components: Pa
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
const main = client_result.assets.main;
|
let main = client_result.assets.main;
|
||||||
|
if (process.env.SAPPER_LEGACY_BUILD) main = `legacy/${main}`;
|
||||||
const entry = fs.readFileSync(`${dirs.dest}/client/${main}`, 'utf-8');
|
const entry = fs.readFileSync(`${dirs.dest}/client/${main}`, 'utf-8');
|
||||||
|
|
||||||
const replacements = new Map();
|
const replacements = new Map();
|
||||||
@@ -203,11 +205,15 @@ export default function extract_css(client_result: CompileResult, components: Pa
|
|||||||
result.chunks[component.file] = files;
|
result.chunks[component.file] = files;
|
||||||
});
|
});
|
||||||
|
|
||||||
const replaced = entry.replace(/["']__SAPPER_CSS_PLACEHOLDER:(.+?)__["']/g, (m, route) => {
|
fs.readdirSync(`${dirs.dest}/client`).forEach(file => {
|
||||||
return JSON.stringify(replacements.get(route));
|
const source = fs.readFileSync(`${dirs.dest}/client/${file}`, 'utf-8');
|
||||||
});
|
|
||||||
|
|
||||||
fs.writeFileSync(`${dirs.dest}/client/${main}`, replaced);
|
const replaced = source.replace(/["']__SAPPER_CSS_PLACEHOLDER:(.+?)__["']/g, (m, route) => {
|
||||||
|
return JSON.stringify(replacements.get(route));
|
||||||
|
});
|
||||||
|
|
||||||
|
fs.writeFileSync(`${dirs.dest}/client/${file}`, replaced);
|
||||||
|
});
|
||||||
|
|
||||||
const leftover = get_css_from_modules(Array.from(unaccounted_for));
|
const leftover = get_css_from_modules(Array.from(unaccounted_for));
|
||||||
if (leftover) {
|
if (leftover) {
|
||||||
|
|||||||
@@ -1,5 +1,4 @@
|
|||||||
import * as fs from 'fs';
|
import * as path from 'path';
|
||||||
import { Dirs } from '../../interfaces';
|
|
||||||
import RollupCompiler from './RollupCompiler';
|
import RollupCompiler from './RollupCompiler';
|
||||||
import { WebpackCompiler } from './WebpackCompiler';
|
import { WebpackCompiler } from './WebpackCompiler';
|
||||||
|
|
||||||
@@ -11,27 +10,52 @@ export type Compilers = {
|
|||||||
serviceworker?: Compiler;
|
serviceworker?: Compiler;
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function create_compilers(bundler: string, dirs: Dirs): Compilers {
|
export default async function create_compilers(bundler: 'rollup' | 'webpack'): Promise<Compilers> {
|
||||||
if (bundler === 'rollup') {
|
if (bundler === 'rollup') {
|
||||||
const sw = `${dirs.rollup}/service-worker.config.js`;
|
const config = await RollupCompiler.load_config();
|
||||||
|
validate_config(config, 'rollup');
|
||||||
|
|
||||||
|
normalize_rollup_config(config.client);
|
||||||
|
normalize_rollup_config(config.server);
|
||||||
|
|
||||||
|
if (config.serviceworker) {
|
||||||
|
normalize_rollup_config(config.serviceworker);
|
||||||
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
client: new RollupCompiler(`${dirs.rollup}/client.config.js`),
|
client: new RollupCompiler(config.client),
|
||||||
server: new RollupCompiler(`${dirs.rollup}/server.config.js`),
|
server: new RollupCompiler(config.server),
|
||||||
serviceworker: fs.existsSync(sw) && new RollupCompiler(sw)
|
serviceworker: config.serviceworker && new RollupCompiler(config.serviceworker)
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
if (bundler === 'webpack') {
|
if (bundler === 'webpack') {
|
||||||
const sw = `${dirs.webpack}/service-worker.config.js`;
|
const config = require(path.resolve('webpack.config.js'));
|
||||||
|
validate_config(config, 'webpack');
|
||||||
|
|
||||||
return {
|
return {
|
||||||
client: new WebpackCompiler(`${dirs.webpack}/client.config.js`),
|
client: new WebpackCompiler(config.client),
|
||||||
server: new WebpackCompiler(`${dirs.webpack}/server.config.js`),
|
server: new WebpackCompiler(config.server),
|
||||||
serviceworker: fs.existsSync(sw) && new WebpackCompiler(sw)
|
serviceworker: config.serviceworker && new WebpackCompiler(config.serviceworker)
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
// this shouldn't be possible...
|
// this shouldn't be possible...
|
||||||
throw new Error(`Invalid bundler option '${bundler}'`);
|
throw new Error(`Invalid bundler option '${bundler}'`);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function validate_config(config: any, bundler: 'rollup' | 'webpack') {
|
||||||
|
if (!config.client || !config.server) {
|
||||||
|
throw new Error(`${bundler}.config.js must export a { client, server, serviceworker? } object`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function normalize_rollup_config(config: any) {
|
||||||
|
if (typeof config.input === 'string') {
|
||||||
|
config.input = path.normalize(config.input);
|
||||||
|
} else {
|
||||||
|
for (const name in config.input) {
|
||||||
|
config.input[name] = path.normalize(config.input[name]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -5,6 +5,11 @@ import { Page, PageComponent, ServerRoute, ManifestData } from '../interfaces';
|
|||||||
import { posixify, reserved_words } from './utils';
|
import { posixify, reserved_words } from './utils';
|
||||||
|
|
||||||
export default function create_manifest_data(cwd = locations.routes()): ManifestData {
|
export default function create_manifest_data(cwd = locations.routes()): ManifestData {
|
||||||
|
// TODO remove in a future version
|
||||||
|
if (!fs.existsSync(cwd)) {
|
||||||
|
throw new Error(`As of Sapper 0.21, the routes/ directory should become src/routes/`);
|
||||||
|
}
|
||||||
|
|
||||||
const components: PageComponent[] = [];
|
const components: PageComponent[] = [];
|
||||||
const pages: Page[] = [];
|
const pages: Page[] = [];
|
||||||
const server_routes: ServerRoute[] = [];
|
const server_routes: ServerRoute[] = [];
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import * as fs from 'fs';
|
import * as fs from 'fs';
|
||||||
import * as path from 'path';
|
import * as path from 'path';
|
||||||
import glob from 'tiny-glob/sync.js';
|
import glob from 'tiny-glob/sync.js';
|
||||||
import { posixify, write_if_changed } from './utils';
|
import { posixify, stringify, write_if_changed } from './utils';
|
||||||
import { dev, locations } from '../config';
|
import { dev, locations } from '../config';
|
||||||
import { Page, PageComponent, ServerRoute, ManifestData } from '../interfaces';
|
import { Page, PageComponent, ServerRoute, ManifestData } from '../interfaces';
|
||||||
|
|
||||||
@@ -10,7 +10,7 @@ export function create_main_manifests({ bundler, manifest_data, dev_port }: {
|
|||||||
manifest_data: ManifestData;
|
manifest_data: ManifestData;
|
||||||
dev_port?: number;
|
dev_port?: number;
|
||||||
}) {
|
}) {
|
||||||
const manifest_dir = path.join(locations.app(), 'manifest');
|
const manifest_dir = '__sapper__';
|
||||||
if (!fs.existsSync(manifest_dir)) fs.mkdirSync(manifest_dir);
|
if (!fs.existsSync(manifest_dir)) fs.mkdirSync(manifest_dir);
|
||||||
|
|
||||||
const path_to_routes = path.relative(manifest_dir, locations.routes());
|
const path_to_routes = path.relative(manifest_dir, locations.routes());
|
||||||
@@ -30,20 +30,32 @@ export function create_serviceworker_manifest({ manifest_data, client_files }: {
|
|||||||
manifest_data: ManifestData;
|
manifest_data: ManifestData;
|
||||||
client_files: string[];
|
client_files: string[];
|
||||||
}) {
|
}) {
|
||||||
const assets = glob('**', { cwd: 'assets', filesOnly: true });
|
let files;
|
||||||
|
|
||||||
|
// TODO remove in a future version
|
||||||
|
if (fs.existsSync(locations.static())) {
|
||||||
|
files = glob('**', { cwd: locations.static(), filesOnly: true });
|
||||||
|
} else {
|
||||||
|
if (fs.existsSync('assets')) {
|
||||||
|
throw new Error(`As of Sapper 0.21, the assets/ directory should become static/`);
|
||||||
|
}
|
||||||
|
|
||||||
|
files = [];
|
||||||
|
}
|
||||||
|
|
||||||
let code = `
|
let code = `
|
||||||
// This file is generated by Sapper — do not edit it!
|
// This file is generated by Sapper — do not edit it!
|
||||||
export const timestamp = ${Date.now()};
|
export const timestamp = ${Date.now()};
|
||||||
|
|
||||||
export const assets = [\n\t${assets.map((x: string) => `"${x}"`).join(',\n\t')}\n];
|
export const files = [\n\t${files.map((x: string) => stringify(x)).join(',\n\t')}\n];
|
||||||
|
export { files as assets }; // legacy
|
||||||
|
|
||||||
export const shell = [\n\t${client_files.map((x: string) => `"${x}"`).join(',\n\t')}\n];
|
export const shell = [\n\t${client_files.map((x: string) => stringify(x)).join(',\n\t')}\n];
|
||||||
|
|
||||||
export const routes = [\n\t${manifest_data.pages.map((r: Page) => `{ pattern: ${r.pattern} }`).join(',\n\t')}\n];
|
export const routes = [\n\t${manifest_data.pages.map((r: Page) => `{ pattern: ${r.pattern} }`).join(',\n\t')}\n];
|
||||||
`.replace(/^\t\t/gm, '').trim();
|
`.replace(/^\t\t/gm, '').trim();
|
||||||
|
|
||||||
write_if_changed(`${locations.app()}/manifest/service-worker.js`, code);
|
write_if_changed(`__sapper__/service-worker.js`, code);
|
||||||
}
|
}
|
||||||
|
|
||||||
function generate_client(
|
function generate_client(
|
||||||
@@ -52,100 +64,114 @@ function generate_client(
|
|||||||
bundler: string,
|
bundler: string,
|
||||||
dev_port?: number
|
dev_port?: number
|
||||||
) {
|
) {
|
||||||
|
const template_file = path.resolve(__dirname, '../templates/client.js');
|
||||||
|
const template = fs.readFileSync(template_file, 'utf-8');
|
||||||
|
|
||||||
const page_ids = new Set(manifest_data.pages.map(page =>
|
const page_ids = new Set(manifest_data.pages.map(page =>
|
||||||
page.pattern.toString()));
|
page.pattern.toString()));
|
||||||
|
|
||||||
const server_routes_to_ignore = manifest_data.server_routes.filter(route =>
|
const server_routes_to_ignore = manifest_data.server_routes.filter(route =>
|
||||||
!page_ids.has(route.pattern.toString()));
|
!page_ids.has(route.pattern.toString()));
|
||||||
|
|
||||||
let code = `
|
const component_indexes: Record<string, number> = {};
|
||||||
// This file is generated by Sapper — do not edit it!
|
|
||||||
import root from '${get_file(path_to_routes, manifest_data.root)}';
|
|
||||||
import error from '${posixify(`${path_to_routes}/_error.html`)}';
|
|
||||||
|
|
||||||
${manifest_data.components.map(component => {
|
const components = `[
|
||||||
|
${manifest_data.components.map((component, i) => {
|
||||||
const annotation = bundler === 'webpack'
|
const annotation = bundler === 'webpack'
|
||||||
? `/* webpackChunkName: "${component.name}" */ `
|
? `/* webpackChunkName: "${component.name}" */ `
|
||||||
: '';
|
: '';
|
||||||
|
|
||||||
const source = get_file(path_to_routes, component);
|
const source = get_file(path_to_routes, component);
|
||||||
|
|
||||||
return `const ${component.name} = {
|
component_indexes[component.name] = i;
|
||||||
js: () => import(${annotation}'${source}'),
|
|
||||||
css: "__SAPPER_CSS_PLACEHOLDER:${component.file}__"
|
|
||||||
};`;
|
|
||||||
}).join('\n')}
|
|
||||||
|
|
||||||
export const manifest = {
|
return `{
|
||||||
ignore: [${server_routes_to_ignore.map(route => route.pattern).join(', ')}],
|
js: () => import(${annotation}${stringify(source)}),
|
||||||
|
css: "__SAPPER_CSS_PLACEHOLDER:${stringify(component.file, false)}__"
|
||||||
|
}`;
|
||||||
|
}).join(',\n\t\t')}
|
||||||
|
]`.replace(/^\t/gm, '').trim();
|
||||||
|
|
||||||
pages: [
|
let needs_decode = false;
|
||||||
${manifest_data.pages.map(page => `{
|
|
||||||
// ${page.parts[page.parts.length - 1].component.file}
|
|
||||||
pattern: ${page.pattern},
|
|
||||||
parts: [
|
|
||||||
${page.parts.map(part => {
|
|
||||||
if (part === null) return 'null';
|
|
||||||
|
|
||||||
if (part.params.length > 0) {
|
let pages = `[
|
||||||
const props = part.params.map((param, i) => `${param}: match[${i + 1}]`);
|
${manifest_data.pages.map(page => `{
|
||||||
return `{ component: ${part.component.name}, params: match => ({ ${props.join(', ')} }) }`;
|
// ${page.parts[page.parts.length - 1].component.file}
|
||||||
}
|
pattern: ${page.pattern},
|
||||||
|
parts: [
|
||||||
|
${page.parts.map(part => {
|
||||||
|
if (part === null) return 'null';
|
||||||
|
|
||||||
return `{ component: ${part.component.name} }`;
|
if (part.params.length > 0) {
|
||||||
}).join(',\n\t\t\t\t\t\t')}
|
needs_decode = true;
|
||||||
]
|
const props = part.params.map((param, i) => `${param}: d(match[${i + 1}])`);
|
||||||
}`).join(',\n\n\t\t\t\t')}
|
return `{ i: ${component_indexes[part.component.name]}, params: match => ({ ${props.join(', ')} }) }`;
|
||||||
],
|
}
|
||||||
|
|
||||||
root,
|
return `{ i: ${component_indexes[part.component.name]} }`;
|
||||||
|
}).join(',\n\t\t\t\t')}
|
||||||
|
]
|
||||||
|
}`).join(',\n\n\t\t')}
|
||||||
|
]`.replace(/^\t/gm, '').trim();
|
||||||
|
|
||||||
error
|
if (needs_decode) {
|
||||||
};
|
pages = `(d => ${pages})(decodeURIComponent)`
|
||||||
|
}
|
||||||
|
|
||||||
// this is included for legacy reasons
|
let footer = '';
|
||||||
export const routes = {};`.replace(/^\t\t/gm, '').trim();
|
|
||||||
|
|
||||||
if (dev()) {
|
if (dev()) {
|
||||||
const sapper_dev_client = posixify(
|
const sapper_dev_client = posixify(
|
||||||
path.resolve(__dirname, '../sapper-dev-client.js')
|
path.resolve(__dirname, '../sapper-dev-client.js')
|
||||||
);
|
);
|
||||||
|
|
||||||
code += `
|
footer = `
|
||||||
|
|
||||||
import('${sapper_dev_client}').then(client => {
|
if (typeof window !== 'undefined') {
|
||||||
client.connect(${dev_port});
|
import(${stringify(sapper_dev_client)}).then(client => {
|
||||||
});`.replace(/^\t{3}/gm, '');
|
client.connect(${dev_port});
|
||||||
|
});
|
||||||
|
}`.replace(/^\t{3}/gm, '');
|
||||||
}
|
}
|
||||||
|
|
||||||
return code;
|
return `// This file is generated by Sapper — do not edit it!\n` + template
|
||||||
|
.replace('__ROOT__', stringify(get_file(path_to_routes, manifest_data.root), false))
|
||||||
|
.replace('__ERROR__', stringify(posixify(`${path_to_routes}/_error.html`), false))
|
||||||
|
.replace('__IGNORE__', `[${server_routes_to_ignore.map(route => route.pattern).join(', ')}]`)
|
||||||
|
.replace('__COMPONENTS__', components)
|
||||||
|
.replace('__PAGES__', pages) +
|
||||||
|
footer;
|
||||||
}
|
}
|
||||||
|
|
||||||
function generate_server(
|
function generate_server(
|
||||||
manifest_data: ManifestData,
|
manifest_data: ManifestData,
|
||||||
path_to_routes: string
|
path_to_routes: string
|
||||||
) {
|
) {
|
||||||
|
const template_file = path.resolve(__dirname, '../templates/server.js');
|
||||||
|
const template = fs.readFileSync(template_file, 'utf-8');
|
||||||
|
|
||||||
const imports = [].concat(
|
const imports = [].concat(
|
||||||
manifest_data.server_routes.map(route =>
|
manifest_data.server_routes.map(route =>
|
||||||
`import * as ${route.name} from '${posixify(`${path_to_routes}/${route.file}`)}';`),
|
`import * as __${route.name} from ${stringify(posixify(`${path_to_routes}/${route.file}`))};`),
|
||||||
manifest_data.components.map(component =>
|
manifest_data.components.map(component =>
|
||||||
`import ${component.name} from '${get_file(path_to_routes, component)}';`),
|
`import __${component.name} from ${stringify(get_file(path_to_routes, component))};`),
|
||||||
`import root from '${get_file(path_to_routes, manifest_data.root)}';`,
|
`import root from ${stringify(get_file(path_to_routes, manifest_data.root))};`,
|
||||||
`import error from '${posixify(`${path_to_routes}/_error.html`)}';`
|
`import error from ${stringify(posixify(`${path_to_routes}/_error.html`))};`
|
||||||
);
|
);
|
||||||
|
|
||||||
let code = `
|
let code = `
|
||||||
// This file is generated by Sapper — do not edit it!
|
|
||||||
${imports.join('\n')}
|
${imports.join('\n')}
|
||||||
|
|
||||||
|
const d = decodeURIComponent;
|
||||||
|
|
||||||
export const manifest = {
|
export const manifest = {
|
||||||
server_routes: [
|
server_routes: [
|
||||||
${manifest_data.server_routes.map(route => `{
|
${manifest_data.server_routes.map(route => `{
|
||||||
// ${route.file}
|
// ${route.file}
|
||||||
pattern: ${route.pattern},
|
pattern: ${route.pattern},
|
||||||
handlers: ${route.name},
|
handlers: __${route.name},
|
||||||
params: ${route.params.length > 0
|
params: ${route.params.length > 0
|
||||||
? `match => ({ ${route.params.map((param, i) => `${param}: match[${i + 1}]`).join(', ')} })`
|
? `match => ({ ${route.params.map((param, i) => `${param}: d(match[${i + 1}])`).join(', ')} })`
|
||||||
: `() => ({})`}
|
: `() => ({})`}
|
||||||
}`).join(',\n\n\t\t\t\t')}
|
}`).join(',\n\n\t\t\t\t')}
|
||||||
],
|
],
|
||||||
@@ -160,12 +186,12 @@ function generate_server(
|
|||||||
|
|
||||||
const props = [
|
const props = [
|
||||||
`name: "${part.component.name}"`,
|
`name: "${part.component.name}"`,
|
||||||
`file: "${part.component.file}"`,
|
`file: ${stringify(part.component.file)}`,
|
||||||
`component: ${part.component.name}`
|
`component: __${part.component.name}`
|
||||||
];
|
];
|
||||||
|
|
||||||
if (part.params.length > 0) {
|
if (part.params.length > 0) {
|
||||||
const params = part.params.map((param, i) => `${param}: match[${i + 1}]`);
|
const params = part.params.map((param, i) => `${param}: d(match[${i + 1}])`);
|
||||||
props.push(`params: match => ({ ${params.join(', ')} })`);
|
props.push(`params: match => ({ ${params.join(', ')} })`);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -178,12 +204,16 @@ function generate_server(
|
|||||||
root,
|
root,
|
||||||
|
|
||||||
error
|
error
|
||||||
};
|
};`.replace(/^\t\t/gm, '').trim();
|
||||||
|
|
||||||
// this is included for legacy reasons
|
const build_dir = path.relative(process.cwd(), locations.dest());
|
||||||
export const routes = {};`.replace(/^\t\t/gm, '').trim();
|
const src_dir = path.relative(process.cwd(), locations.src());
|
||||||
|
|
||||||
return code;
|
return `// This file is generated by Sapper — do not edit it!\n` + template
|
||||||
|
.replace('__BUILD__DIR__', JSON.stringify(build_dir))
|
||||||
|
.replace('__SRC__DIR__', JSON.stringify(src_dir))
|
||||||
|
.replace('__DEV__', dev() ? 'true' : 'false')
|
||||||
|
.replace(/const manifest = __MANIFEST__;/, code);
|
||||||
}
|
}
|
||||||
|
|
||||||
function get_file(path_to_routes: string, component: PageComponent) {
|
function get_file(path_to_routes: string, component: PageComponent) {
|
||||||
|
|||||||
17
src/core/read_template.ts
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
import * as fs from 'fs';
|
||||||
|
import { locations } from '../config';
|
||||||
|
|
||||||
|
export default function read_template(dir = locations.src()) {
|
||||||
|
try {
|
||||||
|
return fs.readFileSync(`${dir}/template.html`, 'utf-8');
|
||||||
|
} catch (err) {
|
||||||
|
if (fs.existsSync(`app/template.html`)) {
|
||||||
|
throw new Error(`As of Sapper 0.21, the default folder structure has been changed:
|
||||||
|
app/ --> src/
|
||||||
|
routes/ --> src/routes/
|
||||||
|
assets/ --> static/`);
|
||||||
|
}
|
||||||
|
|
||||||
|
throw err;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -14,6 +14,11 @@ export function posixify(file: string) {
|
|||||||
return file.replace(/[/\\]/g, '/');
|
return file.replace(/[/\\]/g, '/');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function stringify(string: string, includeQuotes: boolean = true) {
|
||||||
|
const quoted = JSON.stringify(string);
|
||||||
|
return includeQuotes ? quoted : quoted.slice(1, -1);
|
||||||
|
}
|
||||||
|
|
||||||
export function fudge_mtime(file: string) {
|
export function fudge_mtime(file: string) {
|
||||||
// need to fudge the mtime so that webpack doesn't go doolally
|
// need to fudge the mtime so that webpack doesn't go doolally
|
||||||
const { atime, mtime } = fs.statSync(file);
|
const { atime, mtime } = fs.statSync(file);
|
||||||
|
|||||||
@@ -43,7 +43,7 @@ export type ServerRoute = {
|
|||||||
|
|
||||||
export type Dirs = {
|
export type Dirs = {
|
||||||
dest: string,
|
dest: string,
|
||||||
app: string,
|
src: string,
|
||||||
routes: string,
|
routes: string,
|
||||||
webpack: string,
|
webpack: string,
|
||||||
rollup: string
|
rollup: string
|
||||||
|
|||||||
@@ -1,601 +0,0 @@
|
|||||||
import * as fs from 'fs';
|
|
||||||
import * as path from 'path';
|
|
||||||
import { URL } from 'url';
|
|
||||||
import { ClientRequest, ServerResponse } from 'http';
|
|
||||||
import cookie from 'cookie';
|
|
||||||
import devalue from 'devalue';
|
|
||||||
import fetch from 'node-fetch';
|
|
||||||
import { lookup } from './middleware/mime';
|
|
||||||
import { locations, dev } from './config';
|
|
||||||
import sourceMapSupport from 'source-map-support';
|
|
||||||
|
|
||||||
sourceMapSupport.install();
|
|
||||||
|
|
||||||
type ServerRoute = {
|
|
||||||
pattern: RegExp;
|
|
||||||
handlers: Record<string, Handler>;
|
|
||||||
params: (match: RegExpMatchArray) => Record<string, string>;
|
|
||||||
};
|
|
||||||
|
|
||||||
type Page = {
|
|
||||||
pattern: RegExp;
|
|
||||||
parts: Array<{
|
|
||||||
name: string;
|
|
||||||
component: Component;
|
|
||||||
params?: (match: RegExpMatchArray) => Record<string, string>;
|
|
||||||
}>
|
|
||||||
};
|
|
||||||
|
|
||||||
type Manifest = {
|
|
||||||
server_routes: ServerRoute[];
|
|
||||||
pages: Page[];
|
|
||||||
root: Component;
|
|
||||||
error: Component;
|
|
||||||
}
|
|
||||||
|
|
||||||
type Handler = (req: Req, res: ServerResponse, next: () => void) => void;
|
|
||||||
|
|
||||||
type Store = {
|
|
||||||
get: () => any
|
|
||||||
};
|
|
||||||
|
|
||||||
type Props = {
|
|
||||||
path: string;
|
|
||||||
query: Record<string, string>;
|
|
||||||
params: Record<string, string>;
|
|
||||||
error?: { message: string };
|
|
||||||
status?: number;
|
|
||||||
child: {
|
|
||||||
segment: string;
|
|
||||||
component: Component;
|
|
||||||
props: Props;
|
|
||||||
};
|
|
||||||
[key: string]: any;
|
|
||||||
};
|
|
||||||
|
|
||||||
interface Req extends ClientRequest {
|
|
||||||
url: string;
|
|
||||||
baseUrl: string;
|
|
||||||
originalUrl: string;
|
|
||||||
method: string;
|
|
||||||
path: string;
|
|
||||||
params: Record<string, string>;
|
|
||||||
query: Record<string, string>;
|
|
||||||
headers: Record<string, string>;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface Component {
|
|
||||||
render: (data: any, opts: { store: Store }) => {
|
|
||||||
head: string;
|
|
||||||
css: { code: string, map: any };
|
|
||||||
html: string
|
|
||||||
},
|
|
||||||
preload: (data: any) => any | Promise<any>
|
|
||||||
}
|
|
||||||
|
|
||||||
const IGNORE = '__SAPPER__IGNORE__';
|
|
||||||
function toIgnore(uri: string, val: any) {
|
|
||||||
if (Array.isArray(val)) return val.some(x => toIgnore(uri, x));
|
|
||||||
if (val instanceof RegExp) return val.test(uri);
|
|
||||||
if (typeof val === 'function') return val(uri);
|
|
||||||
return uri.startsWith(val.charCodeAt(0) === 47 ? val : `/${val}`);
|
|
||||||
}
|
|
||||||
|
|
||||||
export default function middleware(opts: {
|
|
||||||
manifest: Manifest,
|
|
||||||
store: (req: Req, res: ServerResponse) => Store,
|
|
||||||
ignore?: any,
|
|
||||||
routes?: any // legacy
|
|
||||||
}) {
|
|
||||||
if (opts.routes) {
|
|
||||||
throw new Error(`As of Sapper 0.15, opts.routes should be opts.manifest`);
|
|
||||||
}
|
|
||||||
|
|
||||||
const output = locations.dest();
|
|
||||||
|
|
||||||
const { manifest, store, ignore } = opts;
|
|
||||||
|
|
||||||
let emitted_basepath = false;
|
|
||||||
|
|
||||||
const middleware = compose_handlers([
|
|
||||||
ignore && ((req: Req, res: ServerResponse, next: () => void) => {
|
|
||||||
req[IGNORE] = toIgnore(req.path, ignore);
|
|
||||||
next();
|
|
||||||
}),
|
|
||||||
|
|
||||||
(req: Req, res: ServerResponse, next: () => void) => {
|
|
||||||
if (req[IGNORE]) return next();
|
|
||||||
|
|
||||||
if (req.baseUrl === undefined) {
|
|
||||||
let { originalUrl } = req;
|
|
||||||
if (req.url === '/' && originalUrl[originalUrl.length - 1] !== '/') {
|
|
||||||
originalUrl += '/';
|
|
||||||
}
|
|
||||||
|
|
||||||
req.baseUrl = originalUrl
|
|
||||||
? originalUrl.slice(0, -req.url.length)
|
|
||||||
: '';
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!emitted_basepath && process.send) {
|
|
||||||
process.send({
|
|
||||||
__sapper__: true,
|
|
||||||
event: 'basepath',
|
|
||||||
basepath: req.baseUrl
|
|
||||||
});
|
|
||||||
|
|
||||||
emitted_basepath = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (req.path === undefined) {
|
|
||||||
req.path = req.url.replace(/\?.*/, '');
|
|
||||||
}
|
|
||||||
|
|
||||||
next();
|
|
||||||
},
|
|
||||||
|
|
||||||
fs.existsSync(path.join(output, 'index.html')) && serve({
|
|
||||||
pathname: '/index.html',
|
|
||||||
cache_control: 'max-age=600'
|
|
||||||
}),
|
|
||||||
|
|
||||||
fs.existsSync(path.join(output, 'service-worker.js')) && serve({
|
|
||||||
pathname: '/service-worker.js',
|
|
||||||
cache_control: 'max-age=600'
|
|
||||||
}),
|
|
||||||
|
|
||||||
fs.existsSync(path.join(output, 'service-worker.js.map')) && serve({
|
|
||||||
pathname: '/service-worker.js.map',
|
|
||||||
cache_control: 'max-age=600'
|
|
||||||
}),
|
|
||||||
|
|
||||||
serve({
|
|
||||||
prefix: '/client/',
|
|
||||||
cache_control: dev() ? 'no-cache' : 'max-age=31536000'
|
|
||||||
}),
|
|
||||||
|
|
||||||
get_server_route_handler(manifest.server_routes),
|
|
||||||
get_page_handler(manifest, store)
|
|
||||||
].filter(Boolean));
|
|
||||||
|
|
||||||
return middleware;
|
|
||||||
}
|
|
||||||
|
|
||||||
function serve({ prefix, pathname, cache_control }: {
|
|
||||||
prefix?: string,
|
|
||||||
pathname?: string,
|
|
||||||
cache_control: string
|
|
||||||
}) {
|
|
||||||
const filter = pathname
|
|
||||||
? (req: Req) => req.path === pathname
|
|
||||||
: (req: Req) => req.path.startsWith(prefix);
|
|
||||||
|
|
||||||
const output = locations.dest();
|
|
||||||
|
|
||||||
const cache: Map<string, Buffer> = new Map();
|
|
||||||
|
|
||||||
const read = dev()
|
|
||||||
? (file: string) => fs.readFileSync(path.resolve(output, file))
|
|
||||||
: (file: string) => (cache.has(file) ? cache : cache.set(file, fs.readFileSync(path.resolve(output, file)))).get(file)
|
|
||||||
|
|
||||||
return (req: Req, res: ServerResponse, next: () => void) => {
|
|
||||||
if (req[IGNORE]) return next();
|
|
||||||
|
|
||||||
if (filter(req)) {
|
|
||||||
const type = lookup(req.path);
|
|
||||||
|
|
||||||
try {
|
|
||||||
const file = decodeURIComponent(req.path.slice(1));
|
|
||||||
const data = read(file);
|
|
||||||
|
|
||||||
res.setHeader('Content-Type', type);
|
|
||||||
res.setHeader('Cache-Control', cache_control);
|
|
||||||
res.end(data);
|
|
||||||
} catch (err) {
|
|
||||||
res.statusCode = 404;
|
|
||||||
res.end('not found');
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
next();
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
function get_server_route_handler(routes: ServerRoute[]) {
|
|
||||||
function handle_route(route: ServerRoute, req: Req, res: ServerResponse, next: () => void) {
|
|
||||||
req.params = route.params(route.pattern.exec(req.path));
|
|
||||||
|
|
||||||
const method = req.method.toLowerCase();
|
|
||||||
// 'delete' cannot be exported from a module because it is a keyword,
|
|
||||||
// so check for 'del' instead
|
|
||||||
const method_export = method === 'delete' ? 'del' : method;
|
|
||||||
const handle_method = route.handlers[method_export];
|
|
||||||
if (handle_method) {
|
|
||||||
if (process.env.SAPPER_EXPORT) {
|
|
||||||
const { write, end, setHeader } = res;
|
|
||||||
const chunks: any[] = [];
|
|
||||||
const headers: Record<string, string> = {};
|
|
||||||
|
|
||||||
// intercept data so that it can be exported
|
|
||||||
res.write = function(chunk: any) {
|
|
||||||
chunks.push(Buffer.from(chunk));
|
|
||||||
write.apply(res, arguments);
|
|
||||||
};
|
|
||||||
|
|
||||||
res.setHeader = function(name: string, value: string) {
|
|
||||||
headers[name.toLowerCase()] = value;
|
|
||||||
setHeader.apply(res, arguments);
|
|
||||||
};
|
|
||||||
|
|
||||||
res.end = function(chunk?: any) {
|
|
||||||
if (chunk) chunks.push(Buffer.from(chunk));
|
|
||||||
end.apply(res, arguments);
|
|
||||||
|
|
||||||
process.send({
|
|
||||||
__sapper__: true,
|
|
||||||
event: 'file',
|
|
||||||
url: req.url,
|
|
||||||
method: req.method,
|
|
||||||
status: res.statusCode,
|
|
||||||
type: headers['content-type'],
|
|
||||||
body: Buffer.concat(chunks).toString()
|
|
||||||
});
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
const handle_next = (err?: Error) => {
|
|
||||||
if (err) {
|
|
||||||
res.statusCode = 500;
|
|
||||||
res.end(err.message);
|
|
||||||
} else {
|
|
||||||
process.nextTick(next);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
try {
|
|
||||||
handle_method(req, res, handle_next);
|
|
||||||
} catch (err) {
|
|
||||||
handle_next(err);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// no matching handler for method
|
|
||||||
process.nextTick(next);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return function find_route(req: Req, res: ServerResponse, next: () => void) {
|
|
||||||
if (req[IGNORE]) return next();
|
|
||||||
|
|
||||||
for (const route of routes) {
|
|
||||||
if (route.pattern.test(req.path)) {
|
|
||||||
handle_route(route, req, res, next);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
next();
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
function get_page_handler(
|
|
||||||
manifest: Manifest,
|
|
||||||
store_getter: (req: Req, res: ServerResponse) => Store
|
|
||||||
) {
|
|
||||||
const output = locations.dest();
|
|
||||||
|
|
||||||
const get_build_info = dev()
|
|
||||||
? () => JSON.parse(fs.readFileSync(path.join(output, 'build.json'), 'utf-8'))
|
|
||||||
: (assets => () => assets)(JSON.parse(fs.readFileSync(path.join(output, 'build.json'), 'utf-8')));
|
|
||||||
|
|
||||||
const template = dev()
|
|
||||||
? () => fs.readFileSync(`${locations.app()}/template.html`, 'utf-8')
|
|
||||||
: (str => () => str)(fs.readFileSync(`${locations.dest()}/template.html`, 'utf-8'));
|
|
||||||
|
|
||||||
const { server_routes, pages } = manifest;
|
|
||||||
const error_route = manifest.error;
|
|
||||||
|
|
||||||
function handle_error(req: Req, res: ServerResponse, statusCode: number, error: Error | string) {
|
|
||||||
handle_page({
|
|
||||||
pattern: null,
|
|
||||||
parts: [
|
|
||||||
{ name: null, component: error_route }
|
|
||||||
]
|
|
||||||
}, req, res, statusCode, error || new Error('Unknown error in preload function'));
|
|
||||||
}
|
|
||||||
|
|
||||||
function handle_page(page: Page, req: Req, res: ServerResponse, status = 200, error: Error | string = null) {
|
|
||||||
const build_info: {
|
|
||||||
bundler: 'rollup' | 'webpack',
|
|
||||||
shimport: string | null,
|
|
||||||
assets: Record<string, string | string[]>,
|
|
||||||
legacy_assets?: Record<string, string>
|
|
||||||
} = get_build_info();
|
|
||||||
|
|
||||||
res.setHeader('Content-Type', 'text/html');
|
|
||||||
|
|
||||||
// preload main.js and current route
|
|
||||||
// TODO detect other stuff we can preload? images, CSS, fonts?
|
|
||||||
let preloaded_chunks = Array.isArray(build_info.assets.main) ? build_info.assets.main : [build_info.assets.main];
|
|
||||||
if (!error) {
|
|
||||||
page.parts.forEach(part => {
|
|
||||||
if (!part) return;
|
|
||||||
|
|
||||||
// using concat because it could be a string or an array. thanks webpack!
|
|
||||||
preloaded_chunks = preloaded_chunks.concat(build_info.assets[part.name]);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
const link = preloaded_chunks
|
|
||||||
.filter(file => file && !file.match(/\.map$/))
|
|
||||||
.map(file => `<${req.baseUrl}/client/${file}>;rel="preload";as="script"`)
|
|
||||||
.join(', ');
|
|
||||||
|
|
||||||
res.setHeader('Link', link);
|
|
||||||
|
|
||||||
const store = store_getter ? store_getter(req, res) : null;
|
|
||||||
|
|
||||||
let redirect: { statusCode: number, location: string };
|
|
||||||
let preload_error: { statusCode: number, message: Error | string };
|
|
||||||
|
|
||||||
const preload_context = {
|
|
||||||
redirect: (statusCode: number, location: string) => {
|
|
||||||
if (redirect && (redirect.statusCode !== statusCode || redirect.location !== location)) {
|
|
||||||
throw new Error(`Conflicting redirects`);
|
|
||||||
}
|
|
||||||
location = location.replace(/^\//g, ''); // leading slash (only)
|
|
||||||
redirect = { statusCode, location };
|
|
||||||
},
|
|
||||||
error: (statusCode: number, message: Error | string) => {
|
|
||||||
preload_error = { statusCode, message };
|
|
||||||
},
|
|
||||||
fetch: (url: string, opts?: any) => {
|
|
||||||
const parsed = new URL(url, `http://127.0.0.1:${process.env.PORT}${req.baseUrl ? req.baseUrl + '/' :''}`);
|
|
||||||
|
|
||||||
if (opts) {
|
|
||||||
opts = Object.assign({}, opts);
|
|
||||||
|
|
||||||
const include_cookies = (
|
|
||||||
opts.credentials === 'include' ||
|
|
||||||
opts.credentials === 'same-origin' && parsed.origin === `http://127.0.0.1:${process.env.PORT}`
|
|
||||||
);
|
|
||||||
|
|
||||||
if (include_cookies) {
|
|
||||||
if (!opts.headers) opts.headers = {};
|
|
||||||
|
|
||||||
const str = []
|
|
||||||
.concat(
|
|
||||||
cookie.parse(req.headers.cookie || ''),
|
|
||||||
cookie.parse(opts.headers.cookie || ''),
|
|
||||||
cookie.parse(res.getHeader('Set-Cookie') || '')
|
|
||||||
)
|
|
||||||
.map(cookie => {
|
|
||||||
return Object.keys(cookie)
|
|
||||||
.map(name => `${name}=${encodeURIComponent(cookie[name])}`)
|
|
||||||
.join('; ');
|
|
||||||
})
|
|
||||||
.filter(Boolean)
|
|
||||||
.join(', ');
|
|
||||||
|
|
||||||
opts.headers.cookie = str;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return fetch(parsed.href, opts);
|
|
||||||
},
|
|
||||||
store
|
|
||||||
};
|
|
||||||
|
|
||||||
const root_preloaded = manifest.root.preload
|
|
||||||
? manifest.root.preload.call(preload_context, {
|
|
||||||
path: req.path,
|
|
||||||
query: req.query,
|
|
||||||
params: {}
|
|
||||||
})
|
|
||||||
: {};
|
|
||||||
|
|
||||||
const match = error ? null : page.pattern.exec(req.path);
|
|
||||||
|
|
||||||
Promise.all([root_preloaded].concat(page.parts.map(part => {
|
|
||||||
if (!part) return null;
|
|
||||||
|
|
||||||
return part.component.preload
|
|
||||||
? part.component.preload.call(preload_context, {
|
|
||||||
path: req.path,
|
|
||||||
query: req.query,
|
|
||||||
params: part.params ? part.params(match) : {}
|
|
||||||
})
|
|
||||||
: {};
|
|
||||||
}))).catch(err => {
|
|
||||||
preload_error = { statusCode: 500, message: err };
|
|
||||||
return []; // appease TypeScript
|
|
||||||
}).then(preloaded => {
|
|
||||||
if (redirect) {
|
|
||||||
const location = `${req.baseUrl}/${redirect.location}`;
|
|
||||||
|
|
||||||
res.statusCode = redirect.statusCode;
|
|
||||||
res.setHeader('Location', location);
|
|
||||||
res.end();
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (preload_error) {
|
|
||||||
handle_error(req, res, preload_error.statusCode, preload_error.message);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const serialized = {
|
|
||||||
preloaded: `[${preloaded.map(data => try_serialize(data)).join(',')}]`,
|
|
||||||
store: store && try_serialize(store.get())
|
|
||||||
};
|
|
||||||
|
|
||||||
const segments = req.path.split('/').filter(Boolean);
|
|
||||||
|
|
||||||
const props: Props = {
|
|
||||||
path: req.path,
|
|
||||||
query: req.query,
|
|
||||||
params: {},
|
|
||||||
child: null
|
|
||||||
};
|
|
||||||
|
|
||||||
if (error) {
|
|
||||||
props.error = error instanceof Error ? error : { message: error };
|
|
||||||
props.status = status;
|
|
||||||
}
|
|
||||||
|
|
||||||
const data = Object.assign({}, props, preloaded[0], {
|
|
||||||
params: {},
|
|
||||||
child: {
|
|
||||||
segment: segments[0]
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
let level = data.child;
|
|
||||||
for (let i = 0; i < page.parts.length; i += 1) {
|
|
||||||
const part = page.parts[i];
|
|
||||||
if (!part) continue;
|
|
||||||
|
|
||||||
const get_params = part.params || (() => ({}));
|
|
||||||
|
|
||||||
Object.assign(level, {
|
|
||||||
component: part.component,
|
|
||||||
props: Object.assign({}, props, {
|
|
||||||
params: get_params(match)
|
|
||||||
}, preloaded[i + 1])
|
|
||||||
});
|
|
||||||
|
|
||||||
level.props.child = <Props["child"]>{
|
|
||||||
segment: segments[i + 1]
|
|
||||||
};
|
|
||||||
level = level.props.child;
|
|
||||||
}
|
|
||||||
|
|
||||||
const { html, head, css } = manifest.root.render(data, {
|
|
||||||
store
|
|
||||||
});
|
|
||||||
|
|
||||||
let script = `__SAPPER__={${[
|
|
||||||
error && `error:1`,
|
|
||||||
`baseUrl:"${req.baseUrl}"`,
|
|
||||||
serialized.preloaded && `preloaded:${serialized.preloaded}`,
|
|
||||||
serialized.store && `store:${serialized.store}`
|
|
||||||
].filter(Boolean).join(',')}};`;
|
|
||||||
|
|
||||||
const has_service_worker = fs.existsSync(path.join(locations.dest(), 'service-worker.js'));
|
|
||||||
if (has_service_worker) {
|
|
||||||
script += `if('serviceWorker' in navigator)navigator.serviceWorker.register('${req.baseUrl}/service-worker.js');`;
|
|
||||||
}
|
|
||||||
|
|
||||||
const file = [].concat(build_info.assets.main).filter(file => file && /\.js$/.test(file))[0];
|
|
||||||
const main = `${req.baseUrl}/client/${file}`;
|
|
||||||
|
|
||||||
if (build_info.bundler === 'rollup') {
|
|
||||||
if (build_info.legacy_assets) {
|
|
||||||
const legacy_main = `${req.baseUrl}/client/legacy/${build_info.legacy_assets.main}`;
|
|
||||||
script += `(function(){try{eval("async function x(){}");var main="${main}"}catch(e){main="${legacy_main}"};try{new Function("import('"+main+"')")();}catch(e){var s=document.createElement("script");s.src="${req.baseUrl}/client/shimport@${build_info.shimport}.js";s.setAttribute("data-main",main);document.head.appendChild(s);}}());`;
|
|
||||||
} else {
|
|
||||||
script += `try{new Function("import('${main}')")();}catch(e){var s=document.createElement("script");s.src="${req.baseUrl}/client/shimport@${build_info.shimport}.js";s.setAttribute("data-main","${main}");document.head.appendChild(s);}`;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
script += `</script><script src="${main}">`;
|
|
||||||
}
|
|
||||||
|
|
||||||
let styles: string;
|
|
||||||
|
|
||||||
// TODO make this consistent across apps
|
|
||||||
if (build_info.css && build_info.css.main) {
|
|
||||||
const css_chunks = new Set();
|
|
||||||
if (build_info.css.main) css_chunks.add(build_info.css.main);
|
|
||||||
page.parts.forEach(part => {
|
|
||||||
if (!part) return;
|
|
||||||
const css_chunks_for_part = build_info.css.chunks[part.file];
|
|
||||||
|
|
||||||
if (css_chunks_for_part) {
|
|
||||||
css_chunks_for_part.forEach(file => {
|
|
||||||
css_chunks.add(file);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
styles = Array.from(css_chunks)
|
|
||||||
.map(href => `<link rel="stylesheet" href="client/${href}">`)
|
|
||||||
.join('')
|
|
||||||
} else {
|
|
||||||
styles = (css && css.code ? `<style>${css.code}</style>` : '');
|
|
||||||
}
|
|
||||||
|
|
||||||
const body = template()
|
|
||||||
.replace('%sapper.base%', () => `<base href="${req.baseUrl}/">`)
|
|
||||||
.replace('%sapper.scripts%', () => `<script>${script}</script>`)
|
|
||||||
.replace('%sapper.html%', () => html)
|
|
||||||
.replace('%sapper.head%', () => `<noscript id='sapper-head-start'></noscript>${head}<noscript id='sapper-head-end'></noscript>`)
|
|
||||||
.replace('%sapper.styles%', () => styles);
|
|
||||||
|
|
||||||
res.statusCode = status;
|
|
||||||
res.end(body);
|
|
||||||
}).catch(err => {
|
|
||||||
if (error) {
|
|
||||||
// we encountered an error while rendering the error page — oops
|
|
||||||
res.statusCode = 500;
|
|
||||||
res.end(`<pre>${escape_html(err.message)}</pre>`);
|
|
||||||
} else {
|
|
||||||
handle_error(req, res, 500, err);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
return function find_route(req: Req, res: ServerResponse, next: () => void) {
|
|
||||||
if (req[IGNORE]) return next();
|
|
||||||
|
|
||||||
if (!server_routes.some(route => route.pattern.test(req.path))) {
|
|
||||||
for (const page of pages) {
|
|
||||||
if (page.pattern.test(req.path)) {
|
|
||||||
handle_page(page, req, res);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
handle_error(req, res, 404, 'Not found');
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
function compose_handlers(handlers: Handler[]) {
|
|
||||||
return (req: Req, res: ServerResponse, next: () => void) => {
|
|
||||||
let i = 0;
|
|
||||||
function go() {
|
|
||||||
const handler = handlers[i];
|
|
||||||
|
|
||||||
if (handler) {
|
|
||||||
handler(req, res, () => {
|
|
||||||
i += 1;
|
|
||||||
go();
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
next();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
go();
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
function try_serialize(data: any) {
|
|
||||||
try {
|
|
||||||
return devalue(data);
|
|
||||||
} catch (err) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function escape_html(html: string) {
|
|
||||||
const chars: Record<string, string> = {
|
|
||||||
'"' : 'quot',
|
|
||||||
"'": '#39',
|
|
||||||
'&': 'amp',
|
|
||||||
'<' : 'lt',
|
|
||||||
'>' : 'gt'
|
|
||||||
};
|
|
||||||
|
|
||||||
return html.replace(/["'&<>]/g, c => `&${chars[c]};`);
|
|
||||||
}
|
|
||||||
@@ -5,7 +5,7 @@ export default {
|
|||||||
|
|
||||||
client: {
|
client: {
|
||||||
input: () => {
|
input: () => {
|
||||||
return `${locations.app()}/client.js`
|
return `${locations.src()}/client.js`
|
||||||
},
|
},
|
||||||
|
|
||||||
output: () => {
|
output: () => {
|
||||||
@@ -24,20 +24,23 @@ export default {
|
|||||||
|
|
||||||
server: {
|
server: {
|
||||||
input: () => {
|
input: () => {
|
||||||
return `${locations.app()}/server.js`
|
return {
|
||||||
|
server: `${locations.src()}/server.js`
|
||||||
|
};
|
||||||
},
|
},
|
||||||
|
|
||||||
output: () => {
|
output: () => {
|
||||||
return {
|
return {
|
||||||
dir: locations.dest(),
|
dir: `${locations.dest()}/server`,
|
||||||
format: 'cjs'
|
format: 'cjs',
|
||||||
|
sourcemap: dev()
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
serviceworker: {
|
serviceworker: {
|
||||||
input: () => {
|
input: () => {
|
||||||
return `${locations.app()}/service-worker.js`;
|
return `${locations.src()}/service-worker.js`;
|
||||||
},
|
},
|
||||||
|
|
||||||
output: () => {
|
output: () => {
|
||||||
|
|||||||
@@ -1,504 +0,0 @@
|
|||||||
import { detach, findAnchor, scroll_state, which } from './utils';
|
|
||||||
import { Component, ComponentConstructor, Params, Query, Redirect, Manifest, RouteData, ScrollPosition, Store, Target, ComponentLoader } from './interfaces';
|
|
||||||
|
|
||||||
const initial_data = typeof window !== 'undefined' && window.__SAPPER__;
|
|
||||||
|
|
||||||
export let root: Component;
|
|
||||||
let target: Node;
|
|
||||||
let store: Store;
|
|
||||||
let manifest: Manifest;
|
|
||||||
let segments: string[] = [];
|
|
||||||
|
|
||||||
type RootProps = {
|
|
||||||
path: string;
|
|
||||||
params: Record<string, string>;
|
|
||||||
query: Record<string, string>;
|
|
||||||
child: Child;
|
|
||||||
};
|
|
||||||
|
|
||||||
type Child = {
|
|
||||||
segment?: string;
|
|
||||||
props?: any;
|
|
||||||
component?: Component;
|
|
||||||
};
|
|
||||||
|
|
||||||
const root_props: RootProps = {
|
|
||||||
path: null,
|
|
||||||
params: null,
|
|
||||||
query: null,
|
|
||||||
child: {
|
|
||||||
segment: null,
|
|
||||||
component: null,
|
|
||||||
props: {}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
export { root as component }; // legacy reasons — drop in a future version
|
|
||||||
|
|
||||||
const history = typeof window !== 'undefined' ? window.history : {
|
|
||||||
pushState: (state: any, title: string, href: string) => {},
|
|
||||||
replaceState: (state: any, title: string, href: string) => {},
|
|
||||||
scrollRestoration: ''
|
|
||||||
};
|
|
||||||
|
|
||||||
const scroll_history: Record<string, ScrollPosition> = {};
|
|
||||||
let uid = 1;
|
|
||||||
let cid: number;
|
|
||||||
|
|
||||||
if ('scrollRestoration' in history) {
|
|
||||||
history.scrollRestoration = 'manual';
|
|
||||||
}
|
|
||||||
|
|
||||||
function select_route(url: URL): Target {
|
|
||||||
if (url.origin !== window.location.origin) return null;
|
|
||||||
if (!url.pathname.startsWith(initial_data.baseUrl)) return null;
|
|
||||||
|
|
||||||
const path = url.pathname.slice(initial_data.baseUrl.length);
|
|
||||||
|
|
||||||
// avoid accidental clashes between server routes and pages
|
|
||||||
if (manifest.ignore.some(pattern => pattern.test(path))) return;
|
|
||||||
|
|
||||||
for (let i = 0; i < manifest.pages.length; i += 1) {
|
|
||||||
const page = manifest.pages[i];
|
|
||||||
|
|
||||||
const match = page.pattern.exec(path);
|
|
||||||
if (match) {
|
|
||||||
const query: Record<string, string | true> = {};
|
|
||||||
if (url.search.length > 0) {
|
|
||||||
url.search.slice(1).split('&').forEach(searchParam => {
|
|
||||||
const [, key, value] = /([^=]+)=(.*)/.exec(searchParam);
|
|
||||||
query[key] = value || true;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
return { url, path, page, match, query };
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
let current_token: {};
|
|
||||||
|
|
||||||
function render(data: any, nullable_depth: number, scroll: ScrollPosition, token: {}) {
|
|
||||||
if (current_token !== token) return;
|
|
||||||
|
|
||||||
if (root) {
|
|
||||||
// first, clear out highest-level root component
|
|
||||||
let level = data.child;
|
|
||||||
for (let i = 0; i < nullable_depth; i += 1) {
|
|
||||||
if (i === nullable_depth) break;
|
|
||||||
level = level.props.child;
|
|
||||||
}
|
|
||||||
|
|
||||||
const { component } = level;
|
|
||||||
level.component = null;
|
|
||||||
root.set({ child: data.child });
|
|
||||||
|
|
||||||
// then render new stuff
|
|
||||||
level.component = component;
|
|
||||||
root.set(data);
|
|
||||||
} else {
|
|
||||||
// first load — remove SSR'd <head> contents
|
|
||||||
const start = document.querySelector('#sapper-head-start');
|
|
||||||
const end = document.querySelector('#sapper-head-end');
|
|
||||||
|
|
||||||
if (start && end) {
|
|
||||||
while (start.nextSibling !== end) detach(start.nextSibling);
|
|
||||||
detach(start);
|
|
||||||
detach(end);
|
|
||||||
}
|
|
||||||
|
|
||||||
Object.assign(data, root_data);
|
|
||||||
|
|
||||||
root = new manifest.root({
|
|
||||||
target,
|
|
||||||
data,
|
|
||||||
store,
|
|
||||||
hydrate: true
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
if (scroll) {
|
|
||||||
window.scrollTo(scroll.x, scroll.y);
|
|
||||||
}
|
|
||||||
|
|
||||||
Object.assign(root_props, data);
|
|
||||||
ready = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
function changed(a: Record<string, string | true>, b: Record<string, string | true>) {
|
|
||||||
return JSON.stringify(a) !== JSON.stringify(b);
|
|
||||||
}
|
|
||||||
|
|
||||||
let root_preload: Promise<any>;
|
|
||||||
let root_data: any;
|
|
||||||
|
|
||||||
function load_css(chunk: string) {
|
|
||||||
const href = `${initial_data.baseUrl}client/${chunk}`;
|
|
||||||
if (document.querySelector(`link[href="${href}"]`)) return;
|
|
||||||
|
|
||||||
return new Promise((fulfil, reject) => {
|
|
||||||
const link = document.createElement('link');
|
|
||||||
link.rel = 'stylesheet';
|
|
||||||
link.href = href;
|
|
||||||
|
|
||||||
link.onload = () => fulfil();
|
|
||||||
link.onerror = reject;
|
|
||||||
|
|
||||||
document.head.appendChild(link);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
function load_component(component: ComponentLoader): Promise<ComponentConstructor> {
|
|
||||||
// TODO this is temporary — once placeholders are
|
|
||||||
// always rewritten, scratch the ternary
|
|
||||||
const promises: Array<Promise<any>> = (typeof component.css === 'string' ? [] : component.css.map(load_css));
|
|
||||||
promises.unshift(component.js());
|
|
||||||
return Promise.all(promises).then(values => values[0].default);
|
|
||||||
}
|
|
||||||
|
|
||||||
function prepare_page(target: Target): Promise<{
|
|
||||||
redirect?: Redirect;
|
|
||||||
data?: any;
|
|
||||||
nullable_depth?: number;
|
|
||||||
}> {
|
|
||||||
const { page, path, query } = target;
|
|
||||||
const new_segments = path.split('/').filter(Boolean);
|
|
||||||
let changed_from = 0;
|
|
||||||
|
|
||||||
while (
|
|
||||||
segments[changed_from] &&
|
|
||||||
new_segments[changed_from] &&
|
|
||||||
segments[changed_from] === new_segments[changed_from]
|
|
||||||
) changed_from += 1;
|
|
||||||
|
|
||||||
let redirect: Redirect = null;
|
|
||||||
let error: { statusCode: number, message: Error | string } = null;
|
|
||||||
|
|
||||||
const preload_context = {
|
|
||||||
store,
|
|
||||||
fetch: (url: string, opts?: any) => window.fetch(url, opts),
|
|
||||||
redirect: (statusCode: number, location: string) => {
|
|
||||||
if (redirect && (redirect.statusCode !== statusCode || redirect.location !== location)) {
|
|
||||||
throw new Error(`Conflicting redirects`);
|
|
||||||
}
|
|
||||||
redirect = { statusCode, location };
|
|
||||||
},
|
|
||||||
error: (statusCode: number, message: Error | string) => {
|
|
||||||
error = { statusCode, message };
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
if (!root_preload) {
|
|
||||||
root_preload = manifest.root.preload
|
|
||||||
? initial_data.preloaded[0] || manifest.root.preload.call(preload_context, {
|
|
||||||
path,
|
|
||||||
query,
|
|
||||||
params: {}
|
|
||||||
})
|
|
||||||
: {};
|
|
||||||
}
|
|
||||||
|
|
||||||
return Promise.all(page.parts.map(async (part, i) => {
|
|
||||||
if (i < changed_from) return null;
|
|
||||||
if (!part) return null;
|
|
||||||
|
|
||||||
const Component = await load_component(part.component);
|
|
||||||
|
|
||||||
const req = {
|
|
||||||
path,
|
|
||||||
query,
|
|
||||||
params: part.params ? part.params(target.match) : {}
|
|
||||||
};
|
|
||||||
|
|
||||||
const preloaded = ready || !initial_data.preloaded[i + 1]
|
|
||||||
? Component.preload ? await Component.preload.call(preload_context, req) : {}
|
|
||||||
: initial_data.preloaded[i + 1];
|
|
||||||
|
|
||||||
return { Component, preloaded };
|
|
||||||
})).catch(err => {
|
|
||||||
error = { statusCode: 500, message: err };
|
|
||||||
return [];
|
|
||||||
}).then(async results => {
|
|
||||||
if (!root_data) root_data = await root_preload;
|
|
||||||
|
|
||||||
if (redirect) {
|
|
||||||
return { redirect };
|
|
||||||
}
|
|
||||||
|
|
||||||
segments = new_segments;
|
|
||||||
|
|
||||||
const get_params = page.parts[page.parts.length - 1].params || (() => ({}));
|
|
||||||
const params = get_params(target.match);
|
|
||||||
|
|
||||||
if (error) {
|
|
||||||
const props = {
|
|
||||||
path,
|
|
||||||
query,
|
|
||||||
params,
|
|
||||||
error: typeof error.message === 'string' ? new Error(error.message) : error.message,
|
|
||||||
status: error.statusCode
|
|
||||||
};
|
|
||||||
|
|
||||||
return {
|
|
||||||
data: Object.assign({}, props, {
|
|
||||||
preloading: false,
|
|
||||||
child: {
|
|
||||||
component: manifest.error,
|
|
||||||
props
|
|
||||||
}
|
|
||||||
})
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
const props = { path, query };
|
|
||||||
const data = {
|
|
||||||
path,
|
|
||||||
preloading: false,
|
|
||||||
child: Object.assign({}, root_props.child, {
|
|
||||||
segment: segments[0]
|
|
||||||
})
|
|
||||||
};
|
|
||||||
if (changed(query, root_props.query)) data.query = query;
|
|
||||||
if (changed(params, root_props.params)) data.params = params;
|
|
||||||
|
|
||||||
let level = data.child;
|
|
||||||
let nullable_depth = 0;
|
|
||||||
|
|
||||||
for (let i = 0; i < page.parts.length; i += 1) {
|
|
||||||
const part = page.parts[i];
|
|
||||||
if (!part) continue;
|
|
||||||
|
|
||||||
const get_params = part.params || (() => ({}));
|
|
||||||
|
|
||||||
if (i < changed_from) {
|
|
||||||
level.props.path = path;
|
|
||||||
level.props.query = query;
|
|
||||||
level.props.child = Object.assign({}, level.props.child);
|
|
||||||
|
|
||||||
nullable_depth += 1;
|
|
||||||
} else {
|
|
||||||
level.component = results[i].Component;
|
|
||||||
level.props = Object.assign({}, level.props, props, {
|
|
||||||
params: get_params(target.match),
|
|
||||||
}, results[i].preloaded);
|
|
||||||
|
|
||||||
level.props.child = {};
|
|
||||||
}
|
|
||||||
|
|
||||||
level = level.props.child;
|
|
||||||
level.segment = segments[i + 1];
|
|
||||||
}
|
|
||||||
|
|
||||||
return { data, nullable_depth };
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
async function navigate(target: Target, id: number): Promise<any> {
|
|
||||||
if (id) {
|
|
||||||
// popstate or initial navigation
|
|
||||||
cid = id;
|
|
||||||
} else {
|
|
||||||
// clicked on a link. preserve scroll state
|
|
||||||
scroll_history[cid] = scroll_state();
|
|
||||||
|
|
||||||
id = cid = ++uid;
|
|
||||||
scroll_history[cid] = { x: 0, y: 0 };
|
|
||||||
}
|
|
||||||
|
|
||||||
cid = id;
|
|
||||||
|
|
||||||
if (root) {
|
|
||||||
root.set({ preloading: true });
|
|
||||||
}
|
|
||||||
const loaded = prefetching && prefetching.href === target.url.href ?
|
|
||||||
prefetching.promise :
|
|
||||||
prepare_page(target);
|
|
||||||
|
|
||||||
prefetching = null;
|
|
||||||
|
|
||||||
const token = current_token = {};
|
|
||||||
const { redirect, data, nullable_depth } = await loaded;
|
|
||||||
|
|
||||||
if (redirect) {
|
|
||||||
await goto(redirect.location, { replaceState: true });
|
|
||||||
} else {
|
|
||||||
render(data, nullable_depth, scroll_history[id], token);
|
|
||||||
if (document.activeElement) document.activeElement.blur();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function handle_click(event: MouseEvent) {
|
|
||||||
// Adapted from https://github.com/visionmedia/page.js
|
|
||||||
// MIT license https://github.com/visionmedia/page.js#license
|
|
||||||
if (which(event) !== 1) return;
|
|
||||||
if (event.metaKey || event.ctrlKey || event.shiftKey) return;
|
|
||||||
if (event.defaultPrevented) return;
|
|
||||||
|
|
||||||
const a: HTMLAnchorElement | SVGAElement = <HTMLAnchorElement | SVGAElement>findAnchor(<Node>event.target);
|
|
||||||
if (!a) return;
|
|
||||||
|
|
||||||
if (!a.href) return;
|
|
||||||
|
|
||||||
// check if link is inside an svg
|
|
||||||
// in this case, both href and target are always inside an object
|
|
||||||
const svg = typeof a.href === 'object' && a.href.constructor.name === 'SVGAnimatedString';
|
|
||||||
const href = String(svg ? (<SVGAElement>a).href.baseVal : a.href);
|
|
||||||
|
|
||||||
if (href === window.location.href) {
|
|
||||||
event.preventDefault();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Ignore if tag has
|
|
||||||
// 1. 'download' attribute
|
|
||||||
// 2. rel='external' attribute
|
|
||||||
if (a.hasAttribute('download') || a.getAttribute('rel') === 'external') return;
|
|
||||||
|
|
||||||
// Ignore if <a> has a target
|
|
||||||
if (svg ? (<SVGAElement>a).target.baseVal : a.target) return;
|
|
||||||
|
|
||||||
const url = new URL(href);
|
|
||||||
|
|
||||||
// Don't handle hash changes
|
|
||||||
if (url.pathname === window.location.pathname && url.search === window.location.search) return;
|
|
||||||
|
|
||||||
const target = select_route(url);
|
|
||||||
if (target) {
|
|
||||||
navigate(target, null);
|
|
||||||
event.preventDefault();
|
|
||||||
history.pushState({ id: cid }, '', url.href);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function handle_popstate(event: PopStateEvent) {
|
|
||||||
scroll_history[cid] = scroll_state();
|
|
||||||
|
|
||||||
if (event.state) {
|
|
||||||
const url = new URL(window.location.href);
|
|
||||||
const target = select_route(url);
|
|
||||||
if (target) {
|
|
||||||
navigate(target, event.state.id);
|
|
||||||
} else {
|
|
||||||
window.location.href = window.location.href;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// hashchange
|
|
||||||
cid = ++uid;
|
|
||||||
history.replaceState({ id: cid }, '', window.location.href);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
let prefetching: {
|
|
||||||
href: string;
|
|
||||||
promise: Promise<{ redirect?: Redirect, data?: any, nullable_depth?: number }>;
|
|
||||||
} = null;
|
|
||||||
|
|
||||||
export function prefetch(href: string) {
|
|
||||||
const target: Target = select_route(new URL(href, document.baseURI));
|
|
||||||
|
|
||||||
if (target && (!prefetching || href !== prefetching.href)) {
|
|
||||||
prefetching = {
|
|
||||||
href,
|
|
||||||
promise: prepare_page(target)
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
let mousemove_timeout: NodeJS.Timer;
|
|
||||||
|
|
||||||
function handle_mousemove(event: MouseEvent) {
|
|
||||||
clearTimeout(mousemove_timeout);
|
|
||||||
mousemove_timeout = setTimeout(() => {
|
|
||||||
trigger_prefetch(event);
|
|
||||||
}, 20);
|
|
||||||
}
|
|
||||||
|
|
||||||
function trigger_prefetch(event: MouseEvent | TouchEvent) {
|
|
||||||
const a: HTMLAnchorElement = <HTMLAnchorElement>findAnchor(<Node>event.target);
|
|
||||||
if (!a || a.rel !== 'prefetch') return;
|
|
||||||
|
|
||||||
prefetch(a.href);
|
|
||||||
}
|
|
||||||
|
|
||||||
let inited: boolean;
|
|
||||||
let ready = false;
|
|
||||||
|
|
||||||
export function init(opts: {
|
|
||||||
App: ComponentConstructor,
|
|
||||||
target: Node,
|
|
||||||
manifest: Manifest,
|
|
||||||
store?: (data: any) => Store,
|
|
||||||
routes?: any // legacy
|
|
||||||
}) {
|
|
||||||
if (opts instanceof HTMLElement) {
|
|
||||||
throw new Error(`The signature of init(...) has changed — see https://sapper.svelte.technology/guide#0-11-to-0-12 for more information`);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (opts.routes) {
|
|
||||||
throw new Error(`As of Sapper 0.15, opts.routes should be opts.manifest`);
|
|
||||||
}
|
|
||||||
|
|
||||||
target = opts.target;
|
|
||||||
manifest = opts.manifest;
|
|
||||||
|
|
||||||
if (opts && opts.store) {
|
|
||||||
store = opts.store(initial_data.store);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!inited) { // this check makes HMR possible
|
|
||||||
window.addEventListener('click', handle_click);
|
|
||||||
window.addEventListener('popstate', handle_popstate);
|
|
||||||
|
|
||||||
// prefetch
|
|
||||||
window.addEventListener('touchstart', trigger_prefetch);
|
|
||||||
window.addEventListener('mousemove', handle_mousemove);
|
|
||||||
|
|
||||||
inited = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
return Promise.resolve().then(() => {
|
|
||||||
const { hash, href } = window.location;
|
|
||||||
|
|
||||||
const deep_linked = hash && document.getElementById(hash.slice(1));
|
|
||||||
scroll_history[uid] = deep_linked ?
|
|
||||||
{ x: 0, y: deep_linked.getBoundingClientRect().top } :
|
|
||||||
scroll_state();
|
|
||||||
|
|
||||||
history.replaceState({ id: uid }, '', href);
|
|
||||||
|
|
||||||
if (!initial_data.error) {
|
|
||||||
const target = select_route(new URL(window.location.href));
|
|
||||||
if (target) return navigate(target, uid);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
export function goto(href: string, opts = { replaceState: false }) {
|
|
||||||
const target = select_route(new URL(href, document.baseURI));
|
|
||||||
let promise;
|
|
||||||
|
|
||||||
if (target) {
|
|
||||||
promise = navigate(target, null);
|
|
||||||
if (history) history[opts.replaceState ? 'replaceState' : 'pushState']({ id: cid }, '', href);
|
|
||||||
} else {
|
|
||||||
window.location.href = href;
|
|
||||||
promise = new Promise(f => {}); // never resolves
|
|
||||||
}
|
|
||||||
|
|
||||||
return promise;
|
|
||||||
}
|
|
||||||
|
|
||||||
export function prefetchRoutes(pathnames: string[]) {
|
|
||||||
if (!manifest) throw new Error(`You must call init() first`);
|
|
||||||
|
|
||||||
return manifest.pages
|
|
||||||
.filter(route => {
|
|
||||||
if (!pathnames) return true;
|
|
||||||
return pathnames.some(pathname => route.pattern.test(pathname));
|
|
||||||
})
|
|
||||||
.reduce((promise: Promise<any>, route) => promise.then(() => {
|
|
||||||
return Promise.all(route.parts.map(part => part && load_component(part.component)));
|
|
||||||
}), Promise.resolve());
|
|
||||||
}
|
|
||||||
|
|
||||||
// remove this in 0.9
|
|
||||||
export { prefetchRoutes as preloadRoutes };
|
|
||||||
@@ -1,19 +0,0 @@
|
|||||||
export function detach(node: Node) {
|
|
||||||
node.parentNode.removeChild(node);
|
|
||||||
}
|
|
||||||
|
|
||||||
export function findAnchor(node: Node) {
|
|
||||||
while (node && node.nodeName.toUpperCase() !== 'A') node = node.parentNode; // SVG <a> elements have a lowercase name
|
|
||||||
return node;
|
|
||||||
}
|
|
||||||
|
|
||||||
export function which(event: MouseEvent) {
|
|
||||||
return event.which === null ? event.button : event.which;
|
|
||||||
}
|
|
||||||
|
|
||||||
export function scroll_state() {
|
|
||||||
return {
|
|
||||||
x: window.scrollX,
|
|
||||||
y: window.scrollY
|
|
||||||
};
|
|
||||||
}
|
|
||||||
@@ -6,7 +6,7 @@ export default {
|
|||||||
client: {
|
client: {
|
||||||
entry: () => {
|
entry: () => {
|
||||||
return {
|
return {
|
||||||
main: `${locations.app()}/client`
|
main: `${locations.src()}/client`
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
|
|
||||||
@@ -23,13 +23,13 @@ export default {
|
|||||||
server: {
|
server: {
|
||||||
entry: () => {
|
entry: () => {
|
||||||
return {
|
return {
|
||||||
server: `${locations.app()}/server`
|
server: `${locations.src()}/server`
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
|
|
||||||
output: () => {
|
output: () => {
|
||||||
return {
|
return {
|
||||||
path: locations.dest(),
|
path: `${locations.dest()}/server`,
|
||||||
filename: '[name].js',
|
filename: '[name].js',
|
||||||
chunkFilename: '[hash]/[name].[id].js',
|
chunkFilename: '[hash]/[name].[id].js',
|
||||||
libraryTarget: 'commonjs2'
|
libraryTarget: 'commonjs2'
|
||||||
@@ -40,7 +40,7 @@ export default {
|
|||||||
serviceworker: {
|
serviceworker: {
|
||||||
entry: () => {
|
entry: () => {
|
||||||
return {
|
return {
|
||||||
'service-worker': `${locations.app()}/service-worker`
|
'service-worker': `${locations.src()}/service-worker`
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|||||||
372
templates/src/client/app.ts
Normal file
@@ -0,0 +1,372 @@
|
|||||||
|
import RootComponent from '__ROOT__';
|
||||||
|
import ErrorComponent from '__ERROR__';
|
||||||
|
import {
|
||||||
|
Target,
|
||||||
|
ScrollPosition,
|
||||||
|
Component,
|
||||||
|
Redirect,
|
||||||
|
ComponentLoader,
|
||||||
|
ComponentConstructor,
|
||||||
|
RootProps,
|
||||||
|
Page
|
||||||
|
} from './types';
|
||||||
|
import goto from './goto';
|
||||||
|
|
||||||
|
const ignore = __IGNORE__;
|
||||||
|
export const components: ComponentLoader[] = __COMPONENTS__;
|
||||||
|
export const pages: Page[] = __PAGES__;
|
||||||
|
|
||||||
|
let ready = false;
|
||||||
|
let root_component: Component;
|
||||||
|
let segments: string[] = [];
|
||||||
|
let current_token: {};
|
||||||
|
let root_preload: Promise<any>;
|
||||||
|
let root_data: any;
|
||||||
|
|
||||||
|
const root_props: RootProps = {
|
||||||
|
path: null,
|
||||||
|
params: null,
|
||||||
|
query: null,
|
||||||
|
child: {
|
||||||
|
segment: null,
|
||||||
|
component: null,
|
||||||
|
props: {}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export let prefetching: {
|
||||||
|
href: string;
|
||||||
|
promise: Promise<{ redirect?: Redirect, data?: any, nullable_depth?: number }>;
|
||||||
|
} = null;
|
||||||
|
export function set_prefetching(href, promise) {
|
||||||
|
prefetching = { href, promise };
|
||||||
|
}
|
||||||
|
|
||||||
|
export let store;
|
||||||
|
export function set_store(fn) {
|
||||||
|
store = fn(initial_data.store);
|
||||||
|
}
|
||||||
|
|
||||||
|
export let target: Node;
|
||||||
|
export function set_target(element) {
|
||||||
|
target = element;
|
||||||
|
}
|
||||||
|
|
||||||
|
export let uid = 1;
|
||||||
|
export function set_uid(n) {
|
||||||
|
uid = n;
|
||||||
|
}
|
||||||
|
|
||||||
|
export let cid: number;
|
||||||
|
export function set_cid(n) {
|
||||||
|
cid = n;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const initial_data = typeof __SAPPER__ !== 'undefined' && __SAPPER__;
|
||||||
|
|
||||||
|
const _history = typeof history !== 'undefined' ? history : {
|
||||||
|
pushState: (state: any, title: string, href: string) => {},
|
||||||
|
replaceState: (state: any, title: string, href: string) => {},
|
||||||
|
scrollRestoration: ''
|
||||||
|
};
|
||||||
|
export { _history as history };
|
||||||
|
|
||||||
|
export const scroll_history: Record<string, ScrollPosition> = {};
|
||||||
|
|
||||||
|
export function select_route(url: URL): Target {
|
||||||
|
if (url.origin !== location.origin) return null;
|
||||||
|
if (!url.pathname.startsWith(initial_data.baseUrl)) return null;
|
||||||
|
|
||||||
|
const path = url.pathname.slice(initial_data.baseUrl.length);
|
||||||
|
|
||||||
|
// avoid accidental clashes between server routes and pages
|
||||||
|
if (ignore.some(pattern => pattern.test(path))) return;
|
||||||
|
|
||||||
|
for (let i = 0; i < pages.length; i += 1) {
|
||||||
|
const page = pages[i];
|
||||||
|
|
||||||
|
const match = page.pattern.exec(path);
|
||||||
|
if (match) {
|
||||||
|
const query: Record<string, string | true> = {};
|
||||||
|
if (url.search.length > 0) {
|
||||||
|
url.search.slice(1).split('&').forEach(searchParam => {
|
||||||
|
const [, key, value] = /([^=]+)(?:=(.*))?/.exec(searchParam);
|
||||||
|
query[key] = decodeURIComponent((value || '').replace(/\+/g, ' '));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
return { url, path, page, match, query };
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export function scroll_state() {
|
||||||
|
return {
|
||||||
|
x: scrollX,
|
||||||
|
y: scrollY
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function navigate(target: Target, id: number): Promise<any> {
|
||||||
|
if (id) {
|
||||||
|
// popstate or initial navigation
|
||||||
|
cid = id;
|
||||||
|
} else {
|
||||||
|
// clicked on a link. preserve scroll state
|
||||||
|
scroll_history[cid] = scroll_state();
|
||||||
|
|
||||||
|
id = cid = ++uid;
|
||||||
|
scroll_history[cid] = { x: 0, y: 0 };
|
||||||
|
}
|
||||||
|
|
||||||
|
cid = id;
|
||||||
|
|
||||||
|
if (root_component) {
|
||||||
|
root_component.set({ preloading: true });
|
||||||
|
}
|
||||||
|
const loaded = prefetching && prefetching.href === target.url.href ?
|
||||||
|
prefetching.promise :
|
||||||
|
prepare_page(target);
|
||||||
|
|
||||||
|
prefetching = null;
|
||||||
|
|
||||||
|
const token = current_token = {};
|
||||||
|
|
||||||
|
return loaded.then(({ redirect, data, nullable_depth }) => {
|
||||||
|
if (redirect) {
|
||||||
|
return goto(redirect.location, { replaceState: true });
|
||||||
|
}
|
||||||
|
|
||||||
|
render(data, nullable_depth, scroll_history[id], token);
|
||||||
|
if (document.activeElement) document.activeElement.blur();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function render(data: any, nullable_depth: number, scroll: ScrollPosition, token: {}) {
|
||||||
|
if (current_token !== token) return;
|
||||||
|
|
||||||
|
if (root_component) {
|
||||||
|
// first, clear out highest-level root component
|
||||||
|
let level = data.child;
|
||||||
|
for (let i = 0; i < nullable_depth; i += 1) {
|
||||||
|
if (i === nullable_depth) break;
|
||||||
|
level = level.props.child;
|
||||||
|
}
|
||||||
|
|
||||||
|
const { component } = level;
|
||||||
|
level.component = null;
|
||||||
|
root_component.set({ child: data.child });
|
||||||
|
|
||||||
|
// then render new stuff
|
||||||
|
level.component = component;
|
||||||
|
root_component.set(data);
|
||||||
|
} else {
|
||||||
|
// first load — remove SSR'd <head> contents
|
||||||
|
const start = document.querySelector('#sapper-head-start');
|
||||||
|
const end = document.querySelector('#sapper-head-end');
|
||||||
|
|
||||||
|
if (start && end) {
|
||||||
|
while (start.nextSibling !== end) detach(start.nextSibling);
|
||||||
|
detach(start);
|
||||||
|
detach(end);
|
||||||
|
}
|
||||||
|
|
||||||
|
Object.assign(data, root_data);
|
||||||
|
|
||||||
|
root_component = new RootComponent({
|
||||||
|
target,
|
||||||
|
data,
|
||||||
|
store,
|
||||||
|
hydrate: true
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (scroll) {
|
||||||
|
scrollTo(scroll.x, scroll.y);
|
||||||
|
}
|
||||||
|
|
||||||
|
Object.assign(root_props, data);
|
||||||
|
ready = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function prepare_page(target: Target): Promise<{
|
||||||
|
redirect?: Redirect;
|
||||||
|
data?: any;
|
||||||
|
nullable_depth?: number;
|
||||||
|
}> {
|
||||||
|
const { page, path, query } = target;
|
||||||
|
const new_segments = path.split('/').filter(Boolean);
|
||||||
|
let changed_from = 0;
|
||||||
|
|
||||||
|
while (
|
||||||
|
segments[changed_from] &&
|
||||||
|
new_segments[changed_from] &&
|
||||||
|
segments[changed_from] === new_segments[changed_from]
|
||||||
|
) changed_from += 1;
|
||||||
|
|
||||||
|
let redirect: Redirect = null;
|
||||||
|
let error: { statusCode: number, message: Error | string } = null;
|
||||||
|
|
||||||
|
const preload_context = {
|
||||||
|
store,
|
||||||
|
fetch: (url: string, opts?: any) => fetch(url, opts),
|
||||||
|
redirect: (statusCode: number, location: string) => {
|
||||||
|
if (redirect && (redirect.statusCode !== statusCode || redirect.location !== location)) {
|
||||||
|
throw new Error(`Conflicting redirects`);
|
||||||
|
}
|
||||||
|
redirect = { statusCode, location };
|
||||||
|
},
|
||||||
|
error: (statusCode: number, message: Error | string) => {
|
||||||
|
error = { statusCode, message };
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
if (!root_preload) {
|
||||||
|
root_preload = RootComponent.preload
|
||||||
|
? initial_data.preloaded[0] || RootComponent.preload.call(preload_context, {
|
||||||
|
path,
|
||||||
|
query,
|
||||||
|
params: {}
|
||||||
|
})
|
||||||
|
: {};
|
||||||
|
}
|
||||||
|
|
||||||
|
return Promise.all(page.parts.map((part, i) => {
|
||||||
|
if (i < changed_from) return null;
|
||||||
|
if (!part) return null;
|
||||||
|
|
||||||
|
return load_component(components[part.i]).then(Component => {
|
||||||
|
const req = {
|
||||||
|
path,
|
||||||
|
query,
|
||||||
|
params: part.params ? part.params(target.match) : {}
|
||||||
|
};
|
||||||
|
|
||||||
|
let preloaded;
|
||||||
|
if (ready || !initial_data.preloaded[i + 1]) {
|
||||||
|
preloaded = Component.preload
|
||||||
|
? Component.preload.call(preload_context, req)
|
||||||
|
: {};
|
||||||
|
} else {
|
||||||
|
preloaded = initial_data.preloaded[i + 1];
|
||||||
|
}
|
||||||
|
|
||||||
|
return Promise.resolve(preloaded).then(preloaded => {
|
||||||
|
return { Component, preloaded };
|
||||||
|
});
|
||||||
|
});
|
||||||
|
})).catch(err => {
|
||||||
|
error = { statusCode: 500, message: err };
|
||||||
|
return [];
|
||||||
|
}).then(results => {
|
||||||
|
if (root_data) {
|
||||||
|
return results;
|
||||||
|
} else {
|
||||||
|
return Promise.resolve(root_preload).then(value => {
|
||||||
|
root_data = value;
|
||||||
|
return results;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}).then(results => {
|
||||||
|
if (redirect) {
|
||||||
|
return { redirect };
|
||||||
|
}
|
||||||
|
|
||||||
|
segments = new_segments;
|
||||||
|
|
||||||
|
const get_params = page.parts[page.parts.length - 1].params || (() => ({}));
|
||||||
|
const params = get_params(target.match);
|
||||||
|
|
||||||
|
if (error) {
|
||||||
|
const props = {
|
||||||
|
path,
|
||||||
|
query,
|
||||||
|
params,
|
||||||
|
error: typeof error.message === 'string' ? new Error(error.message) : error.message,
|
||||||
|
status: error.statusCode
|
||||||
|
};
|
||||||
|
|
||||||
|
return {
|
||||||
|
data: Object.assign({}, props, {
|
||||||
|
preloading: false,
|
||||||
|
child: {
|
||||||
|
component: ErrorComponent,
|
||||||
|
props
|
||||||
|
}
|
||||||
|
})
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const props = { path, query };
|
||||||
|
const data = {
|
||||||
|
path,
|
||||||
|
preloading: false,
|
||||||
|
child: Object.assign({}, root_props.child, {
|
||||||
|
segment: segments[0]
|
||||||
|
})
|
||||||
|
};
|
||||||
|
if (changed(query, root_props.query)) data.query = query;
|
||||||
|
if (changed(params, root_props.params)) data.params = params;
|
||||||
|
|
||||||
|
let level = data.child;
|
||||||
|
let nullable_depth = 0;
|
||||||
|
|
||||||
|
for (let i = 0; i < page.parts.length; i += 1) {
|
||||||
|
const part = page.parts[i];
|
||||||
|
if (!part) continue;
|
||||||
|
|
||||||
|
const get_params = part.params || (() => ({}));
|
||||||
|
|
||||||
|
if (i < changed_from) {
|
||||||
|
level.props.path = path;
|
||||||
|
level.props.query = query;
|
||||||
|
level.props.child = Object.assign({}, level.props.child);
|
||||||
|
|
||||||
|
nullable_depth += 1;
|
||||||
|
} else {
|
||||||
|
level.component = results[i].Component;
|
||||||
|
level.props = Object.assign({}, level.props, props, {
|
||||||
|
params: get_params(target.match),
|
||||||
|
}, results[i].preloaded);
|
||||||
|
|
||||||
|
level.props.child = {};
|
||||||
|
}
|
||||||
|
|
||||||
|
level = level.props.child;
|
||||||
|
level.segment = segments[i + 1];
|
||||||
|
}
|
||||||
|
|
||||||
|
return { data, nullable_depth };
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function load_css(chunk: string) {
|
||||||
|
const href = `${initial_data.baseUrl}client/${chunk}`;
|
||||||
|
if (document.querySelector(`link[href="${href}"]`)) return;
|
||||||
|
|
||||||
|
return new Promise((fulfil, reject) => {
|
||||||
|
const link = document.createElement('link');
|
||||||
|
link.rel = 'stylesheet';
|
||||||
|
link.href = href;
|
||||||
|
|
||||||
|
link.onload = () => fulfil();
|
||||||
|
link.onerror = reject;
|
||||||
|
|
||||||
|
document.head.appendChild(link);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export function load_component(component: ComponentLoader): Promise<ComponentConstructor> {
|
||||||
|
// TODO this is temporary — once placeholders are
|
||||||
|
// always rewritten, scratch the ternary
|
||||||
|
const promises: Array<Promise<any>> = (typeof component.css === 'string' ? [] : component.css.map(load_css));
|
||||||
|
promises.unshift(component.js());
|
||||||
|
return Promise.all(promises).then(values => values[0].default);
|
||||||
|
}
|
||||||
|
|
||||||
|
function detach(node: Node) {
|
||||||
|
node.parentNode.removeChild(node);
|
||||||
|
}
|
||||||
|
|
||||||
|
function changed(a: Record<string, string | true>, b: Record<string, string | true>) {
|
||||||
|
return JSON.stringify(a) !== JSON.stringify(b);
|
||||||
|
}
|
||||||
13
templates/src/client/goto/index.ts
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
import { history, select_route, navigate, cid } from '../app';
|
||||||
|
|
||||||
|
export default function goto(href: string, opts = { replaceState: false }) {
|
||||||
|
const target = select_route(new URL(href, document.baseURI));
|
||||||
|
|
||||||
|
if (target) {
|
||||||
|
history[opts.replaceState ? 'replaceState' : 'pushState']({ id: cid }, '', href);
|
||||||
|
return navigate(target, null).then(() => {});
|
||||||
|
}
|
||||||
|
|
||||||
|
location.href = href;
|
||||||
|
return new Promise(f => {}); // never resolves
|
||||||
|
}
|
||||||
4
templates/src/client/index.ts
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
export { default as start } from './start/index';
|
||||||
|
export { default as goto } from './goto/index';
|
||||||
|
export { default as prefetch } from './prefetch/index';
|
||||||
|
export { default as prefetchRoutes } from './prefetchRoutes/index';
|
||||||
10
templates/src/client/prefetch/index.ts
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
import { select_route, prefetching, set_prefetching, prepare_page } from '../app';
|
||||||
|
import { Target } from '../types';
|
||||||
|
|
||||||
|
export default function prefetch(href: string) {
|
||||||
|
const target: Target = select_route(new URL(href, document.baseURI));
|
||||||
|
|
||||||
|
if (target && (!prefetching || href !== prefetching.href)) {
|
||||||
|
set_prefetching(href, prepare_page(target));
|
||||||
|
}
|
||||||
|
}
|
||||||
12
templates/src/client/prefetchRoutes/index.ts
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
import { components, pages, load_component } from "../app";
|
||||||
|
|
||||||
|
export default function prefetchRoutes(pathnames: string[]) {
|
||||||
|
return pages
|
||||||
|
.filter(route => {
|
||||||
|
if (!pathnames) return true;
|
||||||
|
return pathnames.some(pathname => route.pattern.test(pathname));
|
||||||
|
})
|
||||||
|
.reduce((promise: Promise<any>, route) => promise.then(() => {
|
||||||
|
return Promise.all(route.parts.map(part => part && load_component(components[part.i])));
|
||||||
|
}), Promise.resolve());
|
||||||
|
}
|
||||||
138
templates/src/client/start/index.ts
Normal file
@@ -0,0 +1,138 @@
|
|||||||
|
import {
|
||||||
|
cid,
|
||||||
|
history,
|
||||||
|
initial_data,
|
||||||
|
navigate,
|
||||||
|
scroll_history,
|
||||||
|
scroll_state,
|
||||||
|
select_route,
|
||||||
|
set_store,
|
||||||
|
set_target,
|
||||||
|
uid,
|
||||||
|
set_uid,
|
||||||
|
set_cid
|
||||||
|
} from '../app';
|
||||||
|
import prefetch from '../prefetch/index';
|
||||||
|
import { Store } from '../types';
|
||||||
|
|
||||||
|
export default function start(opts: {
|
||||||
|
target: Node,
|
||||||
|
store?: (data: any) => Store
|
||||||
|
}) {
|
||||||
|
if ('scrollRestoration' in history) {
|
||||||
|
history.scrollRestoration = 'manual';
|
||||||
|
}
|
||||||
|
|
||||||
|
set_target(opts.target);
|
||||||
|
if (opts.store) set_store(opts.store);
|
||||||
|
|
||||||
|
addEventListener('click', handle_click);
|
||||||
|
addEventListener('popstate', handle_popstate);
|
||||||
|
|
||||||
|
// prefetch
|
||||||
|
addEventListener('touchstart', trigger_prefetch);
|
||||||
|
addEventListener('mousemove', handle_mousemove);
|
||||||
|
|
||||||
|
return Promise.resolve().then(() => {
|
||||||
|
const { hash, href } = location;
|
||||||
|
|
||||||
|
const deep_linked = hash && document.getElementById(hash.slice(1));
|
||||||
|
scroll_history[uid] = deep_linked ?
|
||||||
|
{ x: 0, y: deep_linked.getBoundingClientRect().top } :
|
||||||
|
scroll_state();
|
||||||
|
|
||||||
|
history.replaceState({ id: uid }, '', href);
|
||||||
|
|
||||||
|
if (!initial_data.error) {
|
||||||
|
const target = select_route(new URL(location.href));
|
||||||
|
if (target) return navigate(target, uid);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
let mousemove_timeout: NodeJS.Timer;
|
||||||
|
|
||||||
|
function handle_mousemove(event: MouseEvent) {
|
||||||
|
clearTimeout(mousemove_timeout);
|
||||||
|
mousemove_timeout = setTimeout(() => {
|
||||||
|
trigger_prefetch(event);
|
||||||
|
}, 20);
|
||||||
|
}
|
||||||
|
|
||||||
|
function trigger_prefetch(event: MouseEvent | TouchEvent) {
|
||||||
|
const a: HTMLAnchorElement = <HTMLAnchorElement>find_anchor(<Node>event.target);
|
||||||
|
if (!a || a.rel !== 'prefetch') return;
|
||||||
|
|
||||||
|
prefetch(a.href);
|
||||||
|
}
|
||||||
|
|
||||||
|
function handle_click(event: MouseEvent) {
|
||||||
|
// Adapted from https://github.com/visionmedia/page.js
|
||||||
|
// MIT license https://github.com/visionmedia/page.js#license
|
||||||
|
if (which(event) !== 1) return;
|
||||||
|
if (event.metaKey || event.ctrlKey || event.shiftKey) return;
|
||||||
|
if (event.defaultPrevented) return;
|
||||||
|
|
||||||
|
const a: HTMLAnchorElement | SVGAElement = <HTMLAnchorElement | SVGAElement>find_anchor(<Node>event.target);
|
||||||
|
if (!a) return;
|
||||||
|
|
||||||
|
if (!a.href) return;
|
||||||
|
|
||||||
|
// check if link is inside an svg
|
||||||
|
// in this case, both href and target are always inside an object
|
||||||
|
const svg = typeof a.href === 'object' && a.href.constructor.name === 'SVGAnimatedString';
|
||||||
|
const href = String(svg ? (<SVGAElement>a).href.baseVal : a.href);
|
||||||
|
|
||||||
|
if (href === location.href) {
|
||||||
|
event.preventDefault();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ignore if tag has
|
||||||
|
// 1. 'download' attribute
|
||||||
|
// 2. rel='external' attribute
|
||||||
|
if (a.hasAttribute('download') || a.getAttribute('rel') === 'external') return;
|
||||||
|
|
||||||
|
// Ignore if <a> has a target
|
||||||
|
if (svg ? (<SVGAElement>a).target.baseVal : a.target) return;
|
||||||
|
|
||||||
|
const url = new URL(href);
|
||||||
|
|
||||||
|
// Don't handle hash changes
|
||||||
|
if (url.pathname === location.pathname && url.search === location.search) return;
|
||||||
|
|
||||||
|
const target = select_route(url);
|
||||||
|
if (target) {
|
||||||
|
navigate(target, null);
|
||||||
|
event.preventDefault();
|
||||||
|
history.pushState({ id: cid }, '', url.href);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function which(event: MouseEvent) {
|
||||||
|
return event.which === null ? event.button : event.which;
|
||||||
|
}
|
||||||
|
|
||||||
|
function find_anchor(node: Node) {
|
||||||
|
while (node && node.nodeName.toUpperCase() !== 'A') node = node.parentNode; // SVG <a> elements have a lowercase name
|
||||||
|
return node;
|
||||||
|
}
|
||||||
|
|
||||||
|
function handle_popstate(event: PopStateEvent) {
|
||||||
|
scroll_history[cid] = scroll_state();
|
||||||
|
|
||||||
|
if (event.state) {
|
||||||
|
const url = new URL(location.href);
|
||||||
|
const target = select_route(url);
|
||||||
|
if (target) {
|
||||||
|
navigate(target, event.state.id);
|
||||||
|
} else {
|
||||||
|
location.href = location.href;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// hashchange
|
||||||
|
set_uid(uid + 1);
|
||||||
|
set_cid(uid);
|
||||||
|
history.replaceState({ id: cid }, '', location.href);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,10 +1,20 @@
|
|||||||
import { Store } from '../interfaces';
|
|
||||||
|
|
||||||
export { Store };
|
|
||||||
export type Params = Record<string, string>;
|
export type Params = Record<string, string>;
|
||||||
export type Query = Record<string, string | true>;
|
export type Query = Record<string, string | true>;
|
||||||
export type RouteData = { params: Params, query: Query, path: string };
|
export type RouteData = { params: Params, query: Query, path: string };
|
||||||
|
|
||||||
|
type Child = {
|
||||||
|
segment?: string;
|
||||||
|
props?: any;
|
||||||
|
component?: Component;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type RootProps = {
|
||||||
|
path: string;
|
||||||
|
params: Record<string, string>;
|
||||||
|
query: Record<string, string>;
|
||||||
|
child: Child;
|
||||||
|
};
|
||||||
|
|
||||||
export interface ComponentConstructor {
|
export interface ComponentConstructor {
|
||||||
new (options: { target: Node, data: any, store: Store, hydrate: boolean }): Component;
|
new (options: { target: Node, data: any, store: Store, hydrate: boolean }): Component;
|
||||||
preload: (props: { params: Params, query: Query }) => Promise<any>;
|
preload: (props: { params: Params, query: Query }) => Promise<any>;
|
||||||
@@ -23,7 +33,7 @@ export type ComponentLoader = {
|
|||||||
export type Page = {
|
export type Page = {
|
||||||
pattern: RegExp;
|
pattern: RegExp;
|
||||||
parts: Array<{
|
parts: Array<{
|
||||||
component: ComponentLoader;
|
i: number;
|
||||||
params?: (match: RegExpExecArray) => Record<string, string>;
|
params?: (match: RegExpExecArray) => Record<string, string>;
|
||||||
}>;
|
}>;
|
||||||
};
|
};
|
||||||
@@ -51,4 +61,8 @@ export type Target = {
|
|||||||
export type Redirect = {
|
export type Redirect = {
|
||||||
statusCode: number;
|
statusCode: number;
|
||||||
location: string;
|
location: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export type Store = {
|
||||||
|
get: () => any;
|
||||||
|
}
|
||||||
1
templates/src/server/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export { default as middleware } from './middleware/index';
|
||||||
321
templates/src/server/middleware/get_page_handler.ts
Normal file
@@ -0,0 +1,321 @@
|
|||||||
|
import * as fs from 'fs';
|
||||||
|
import * as path from 'path';
|
||||||
|
import cookie from 'cookie';
|
||||||
|
import devalue from 'devalue';
|
||||||
|
import fetch from 'node-fetch';
|
||||||
|
import { URL } from 'url';
|
||||||
|
import { build_dir, dev, src_dir, IGNORE } from '../placeholders';
|
||||||
|
import { Manifest, Page, Props, Req, Res, Store } from './types';
|
||||||
|
|
||||||
|
export function get_page_handler(
|
||||||
|
manifest: Manifest,
|
||||||
|
store_getter: (req: Req, res: Res) => Store
|
||||||
|
) {
|
||||||
|
const get_build_info = dev
|
||||||
|
? () => JSON.parse(fs.readFileSync(path.join(build_dir, 'build.json'), 'utf-8'))
|
||||||
|
: (assets => () => assets)(JSON.parse(fs.readFileSync(path.join(build_dir, 'build.json'), 'utf-8')));
|
||||||
|
|
||||||
|
const template = dev
|
||||||
|
? () => read_template(src_dir)
|
||||||
|
: (str => () => str)(read_template(build_dir));
|
||||||
|
|
||||||
|
const has_service_worker = fs.existsSync(path.join(build_dir, 'service-worker.js'));
|
||||||
|
|
||||||
|
const { server_routes, pages } = manifest;
|
||||||
|
const error_route = manifest.error;
|
||||||
|
|
||||||
|
function handle_error(req: Req, res: Res, statusCode: number, error: Error | string) {
|
||||||
|
handle_page({
|
||||||
|
pattern: null,
|
||||||
|
parts: [
|
||||||
|
{ name: null, component: error_route }
|
||||||
|
]
|
||||||
|
}, req, res, statusCode, error || new Error('Unknown error in preload function'));
|
||||||
|
}
|
||||||
|
|
||||||
|
function handle_page(page: Page, req: Req, res: Res, status = 200, error: Error | string = null) {
|
||||||
|
const build_info: {
|
||||||
|
bundler: 'rollup' | 'webpack',
|
||||||
|
shimport: string | null,
|
||||||
|
assets: Record<string, string | string[]>,
|
||||||
|
legacy_assets?: Record<string, string>
|
||||||
|
} = get_build_info();
|
||||||
|
|
||||||
|
res.setHeader('Content-Type', 'text/html');
|
||||||
|
res.setHeader('Cache-Control', dev ? 'no-cache' : 'max-age=600');
|
||||||
|
|
||||||
|
// preload main.js and current route
|
||||||
|
// TODO detect other stuff we can preload? images, CSS, fonts?
|
||||||
|
let preloaded_chunks = Array.isArray(build_info.assets.main) ? build_info.assets.main : [build_info.assets.main];
|
||||||
|
if (!error) {
|
||||||
|
page.parts.forEach(part => {
|
||||||
|
if (!part) return;
|
||||||
|
|
||||||
|
// using concat because it could be a string or an array. thanks webpack!
|
||||||
|
preloaded_chunks = preloaded_chunks.concat(build_info.assets[part.name]);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
const link = preloaded_chunks
|
||||||
|
.filter(file => file && !file.match(/\.map$/))
|
||||||
|
.map(file => `<${req.baseUrl}/client/${file}>;rel="preload";as="script"`)
|
||||||
|
.join(', ');
|
||||||
|
|
||||||
|
res.setHeader('Link', link);
|
||||||
|
|
||||||
|
const store = store_getter ? store_getter(req, res) : null;
|
||||||
|
|
||||||
|
let redirect: { statusCode: number, location: string };
|
||||||
|
let preload_error: { statusCode: number, message: Error | string };
|
||||||
|
|
||||||
|
const preload_context = {
|
||||||
|
redirect: (statusCode: number, location: string) => {
|
||||||
|
if (redirect && (redirect.statusCode !== statusCode || redirect.location !== location)) {
|
||||||
|
throw new Error(`Conflicting redirects`);
|
||||||
|
}
|
||||||
|
location = location.replace(/^\//g, ''); // leading slash (only)
|
||||||
|
redirect = { statusCode, location };
|
||||||
|
},
|
||||||
|
error: (statusCode: number, message: Error | string) => {
|
||||||
|
preload_error = { statusCode, message };
|
||||||
|
},
|
||||||
|
fetch: (url: string, opts?: any) => {
|
||||||
|
const parsed = new URL(url, `http://127.0.0.1:${process.env.PORT}${req.baseUrl ? req.baseUrl + '/' :''}`);
|
||||||
|
|
||||||
|
if (opts) {
|
||||||
|
opts = Object.assign({}, opts);
|
||||||
|
|
||||||
|
const include_cookies = (
|
||||||
|
opts.credentials === 'include' ||
|
||||||
|
opts.credentials === 'same-origin' && parsed.origin === `http://127.0.0.1:${process.env.PORT}`
|
||||||
|
);
|
||||||
|
|
||||||
|
if (include_cookies) {
|
||||||
|
if (!opts.headers) opts.headers = {};
|
||||||
|
|
||||||
|
const cookies = Object.assign(
|
||||||
|
{},
|
||||||
|
cookie.parse(req.headers.cookie || ''),
|
||||||
|
cookie.parse(opts.headers.cookie || '')
|
||||||
|
);
|
||||||
|
|
||||||
|
const set_cookie = res.getHeader('Set-Cookie');
|
||||||
|
(Array.isArray(set_cookie) ? set_cookie : [set_cookie]).forEach(str => {
|
||||||
|
const match = /([^=]+)=([^;]+)/.exec(<string>str);
|
||||||
|
if (match) cookies[match[1]] = match[2];
|
||||||
|
});
|
||||||
|
|
||||||
|
const str = Object.keys(cookies)
|
||||||
|
.map(key => `${key}=${cookies[key]}`)
|
||||||
|
.join('; ');
|
||||||
|
|
||||||
|
opts.headers.cookie = str;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return fetch(parsed.href, opts);
|
||||||
|
},
|
||||||
|
store
|
||||||
|
};
|
||||||
|
|
||||||
|
const root_preloaded = manifest.root.preload
|
||||||
|
? manifest.root.preload.call(preload_context, {
|
||||||
|
path: req.path,
|
||||||
|
query: req.query,
|
||||||
|
params: {}
|
||||||
|
})
|
||||||
|
: {};
|
||||||
|
|
||||||
|
const match = error ? null : page.pattern.exec(req.path);
|
||||||
|
|
||||||
|
Promise.all([root_preloaded].concat(page.parts.map(part => {
|
||||||
|
if (!part) return null;
|
||||||
|
|
||||||
|
return part.component.preload
|
||||||
|
? part.component.preload.call(preload_context, {
|
||||||
|
path: req.path,
|
||||||
|
query: req.query,
|
||||||
|
params: part.params ? part.params(match) : {}
|
||||||
|
})
|
||||||
|
: {};
|
||||||
|
}))).catch(err => {
|
||||||
|
preload_error = { statusCode: 500, message: err };
|
||||||
|
return []; // appease TypeScript
|
||||||
|
}).then(preloaded => {
|
||||||
|
if (redirect) {
|
||||||
|
const location = `${req.baseUrl}/${redirect.location}`;
|
||||||
|
|
||||||
|
res.statusCode = redirect.statusCode;
|
||||||
|
res.setHeader('Location', location);
|
||||||
|
res.end();
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (preload_error) {
|
||||||
|
handle_error(req, res, preload_error.statusCode, preload_error.message);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const serialized = {
|
||||||
|
preloaded: `[${preloaded.map(data => try_serialize(data)).join(',')}]`,
|
||||||
|
store: store && try_serialize(store.get())
|
||||||
|
};
|
||||||
|
|
||||||
|
const segments = req.path.split('/').filter(Boolean);
|
||||||
|
|
||||||
|
const props: Props = {
|
||||||
|
path: req.path,
|
||||||
|
query: req.query,
|
||||||
|
params: {},
|
||||||
|
child: null
|
||||||
|
};
|
||||||
|
|
||||||
|
if (error) {
|
||||||
|
props.error = error instanceof Error ? error : { message: error };
|
||||||
|
props.status = status;
|
||||||
|
}
|
||||||
|
|
||||||
|
const data = Object.assign({}, props, preloaded[0], {
|
||||||
|
params: {},
|
||||||
|
child: {
|
||||||
|
segment: segments[0]
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
let level = data.child;
|
||||||
|
for (let i = 0; i < page.parts.length; i += 1) {
|
||||||
|
const part = page.parts[i];
|
||||||
|
if (!part) continue;
|
||||||
|
|
||||||
|
const get_params = part.params || (() => ({}));
|
||||||
|
|
||||||
|
Object.assign(level, {
|
||||||
|
component: part.component,
|
||||||
|
props: Object.assign({}, props, {
|
||||||
|
params: get_params(match)
|
||||||
|
}, preloaded[i + 1])
|
||||||
|
});
|
||||||
|
|
||||||
|
level.props.child = <Props["child"]>{
|
||||||
|
segment: segments[i + 1]
|
||||||
|
};
|
||||||
|
level = level.props.child;
|
||||||
|
}
|
||||||
|
|
||||||
|
const { html, head, css } = manifest.root.render(data, {
|
||||||
|
store
|
||||||
|
});
|
||||||
|
|
||||||
|
let script = `__SAPPER__={${[
|
||||||
|
error && `error:1`,
|
||||||
|
`baseUrl:"${req.baseUrl}"`,
|
||||||
|
serialized.preloaded && `preloaded:${serialized.preloaded}`,
|
||||||
|
serialized.store && `store:${serialized.store}`
|
||||||
|
].filter(Boolean).join(',')}};`;
|
||||||
|
|
||||||
|
if (has_service_worker) {
|
||||||
|
script += `if('serviceWorker' in navigator)navigator.serviceWorker.register('${req.baseUrl}/service-worker.js');`;
|
||||||
|
}
|
||||||
|
|
||||||
|
const file = [].concat(build_info.assets.main).filter(file => file && /\.js$/.test(file))[0];
|
||||||
|
const main = `${req.baseUrl}/client/${file}`;
|
||||||
|
|
||||||
|
if (build_info.bundler === 'rollup') {
|
||||||
|
if (build_info.legacy_assets) {
|
||||||
|
const legacy_main = `${req.baseUrl}/client/legacy/${build_info.legacy_assets.main}`;
|
||||||
|
script += `(function(){try{eval("async function x(){}");var main="${main}"}catch(e){main="${legacy_main}"};var s=document.createElement("script");try{new Function("if(0)import('')")();s.src=main;s.type="module";s.crossOrigin="use-credentials";}catch(e){s.src="${req.baseUrl}/client/shimport@${build_info.shimport}.js";s.setAttribute("data-main",main);}document.head.appendChild(s);}());`;
|
||||||
|
} else {
|
||||||
|
script += `var s=document.createElement("script");try{new Function("if(0)import('')")();s.src="${main}";s.type="module";s.crossOrigin="use-credentials";}catch(e){console.error(e);s.src="${req.baseUrl}/client/shimport@${build_info.shimport}.js";s.setAttribute("data-main","${main}")}document.head.appendChild(s)`;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
script += `</script><script src="${main}">`;
|
||||||
|
}
|
||||||
|
|
||||||
|
let styles: string;
|
||||||
|
|
||||||
|
// TODO make this consistent across apps
|
||||||
|
// TODO embed build_info in placeholder.ts
|
||||||
|
if (build_info.css && build_info.css.main) {
|
||||||
|
const css_chunks = new Set();
|
||||||
|
if (build_info.css.main) css_chunks.add(build_info.css.main);
|
||||||
|
page.parts.forEach(part => {
|
||||||
|
if (!part) return;
|
||||||
|
const css_chunks_for_part = build_info.css.chunks[part.file];
|
||||||
|
|
||||||
|
if (css_chunks_for_part) {
|
||||||
|
css_chunks_for_part.forEach(file => {
|
||||||
|
css_chunks.add(file);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
styles = Array.from(css_chunks)
|
||||||
|
.map(href => `<link rel="stylesheet" href="client/${href}">`)
|
||||||
|
.join('')
|
||||||
|
} else {
|
||||||
|
styles = (css && css.code ? `<style>${css.code}</style>` : '');
|
||||||
|
}
|
||||||
|
|
||||||
|
// users can set a CSP nonce using res.locals.nonce
|
||||||
|
const nonce_attr = (res.locals && res.locals.nonce) ? ` nonce="${res.locals.nonce}"` : '';
|
||||||
|
|
||||||
|
const body = template()
|
||||||
|
.replace('%sapper.base%', () => `<base href="${req.baseUrl}/">`)
|
||||||
|
.replace('%sapper.scripts%', () => `<script${nonce_attr}>${script}</script>`)
|
||||||
|
.replace('%sapper.html%', () => html)
|
||||||
|
.replace('%sapper.head%', () => `<noscript id='sapper-head-start'></noscript>${head}<noscript id='sapper-head-end'></noscript>`)
|
||||||
|
.replace('%sapper.styles%', () => styles);
|
||||||
|
|
||||||
|
res.statusCode = status;
|
||||||
|
res.end(body);
|
||||||
|
}).catch(err => {
|
||||||
|
if (error) {
|
||||||
|
// we encountered an error while rendering the error page — oops
|
||||||
|
res.statusCode = 500;
|
||||||
|
res.end(`<pre>${escape_html(err.message)}</pre>`);
|
||||||
|
} else {
|
||||||
|
handle_error(req, res, 500, err);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return function find_route(req: Req, res: Res, next: () => void) {
|
||||||
|
if (req[IGNORE]) return next();
|
||||||
|
|
||||||
|
if (!server_routes.some(route => route.pattern.test(req.path))) {
|
||||||
|
for (const page of pages) {
|
||||||
|
if (page.pattern.test(req.path)) {
|
||||||
|
handle_page(page, req, res);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
handle_error(req, res, 404, 'Not found');
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function read_template(dir = build_dir) {
|
||||||
|
return fs.readFileSync(`${dir}/template.html`, 'utf-8');
|
||||||
|
}
|
||||||
|
|
||||||
|
function try_serialize(data: any) {
|
||||||
|
try {
|
||||||
|
return devalue(data);
|
||||||
|
} catch (err) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function escape_html(html: string) {
|
||||||
|
const chars: Record<string, string> = {
|
||||||
|
'"' : 'quot',
|
||||||
|
"'": '#39',
|
||||||
|
'&': 'amp',
|
||||||
|
'<' : 'lt',
|
||||||
|
'>' : 'gt'
|
||||||
|
};
|
||||||
|
|
||||||
|
return html.replace(/["'&<>]/g, c => `&${chars[c]};`);
|
||||||
|
}
|
||||||
78
templates/src/server/middleware/get_server_route_handler.ts
Normal file
@@ -0,0 +1,78 @@
|
|||||||
|
import { IGNORE } from '../placeholders';
|
||||||
|
import { Req, Res, ServerRoute } from './types';
|
||||||
|
|
||||||
|
export function get_server_route_handler(routes: ServerRoute[]) {
|
||||||
|
function handle_route(route: ServerRoute, req: Req, res: Res, next: () => void) {
|
||||||
|
req.params = route.params(route.pattern.exec(req.path));
|
||||||
|
|
||||||
|
const method = req.method.toLowerCase();
|
||||||
|
// 'delete' cannot be exported from a module because it is a keyword,
|
||||||
|
// so check for 'del' instead
|
||||||
|
const method_export = method === 'delete' ? 'del' : method;
|
||||||
|
const handle_method = route.handlers[method_export];
|
||||||
|
if (handle_method) {
|
||||||
|
if (process.env.SAPPER_EXPORT) {
|
||||||
|
const { write, end, setHeader } = res;
|
||||||
|
const chunks: any[] = [];
|
||||||
|
const headers: Record<string, string> = {};
|
||||||
|
|
||||||
|
// intercept data so that it can be exported
|
||||||
|
res.write = function(chunk: any) {
|
||||||
|
chunks.push(Buffer.from(chunk));
|
||||||
|
write.apply(res, arguments);
|
||||||
|
};
|
||||||
|
|
||||||
|
res.setHeader = function(name: string, value: string) {
|
||||||
|
headers[name.toLowerCase()] = value;
|
||||||
|
setHeader.apply(res, arguments);
|
||||||
|
};
|
||||||
|
|
||||||
|
res.end = function(chunk?: any) {
|
||||||
|
if (chunk) chunks.push(Buffer.from(chunk));
|
||||||
|
end.apply(res, arguments);
|
||||||
|
|
||||||
|
process.send({
|
||||||
|
__sapper__: true,
|
||||||
|
event: 'file',
|
||||||
|
url: req.url,
|
||||||
|
method: req.method,
|
||||||
|
status: res.statusCode,
|
||||||
|
type: headers['content-type'],
|
||||||
|
body: Buffer.concat(chunks).toString()
|
||||||
|
});
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const handle_next = (err?: Error) => {
|
||||||
|
if (err) {
|
||||||
|
res.statusCode = 500;
|
||||||
|
res.end(err.message);
|
||||||
|
} else {
|
||||||
|
process.nextTick(next);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
try {
|
||||||
|
handle_method(req, res, handle_next);
|
||||||
|
} catch (err) {
|
||||||
|
handle_next(err);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// no matching handler for method
|
||||||
|
process.nextTick(next);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return function find_route(req: Req, res: Res, next: () => void) {
|
||||||
|
if (req[IGNORE]) return next();
|
||||||
|
|
||||||
|
for (const route of routes) {
|
||||||
|
if (route.pattern.test(req.path)) {
|
||||||
|
handle_route(route, req, res, next);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
next();
|
||||||
|
};
|
||||||
|
}
|
||||||
143
templates/src/server/middleware/index.ts
Normal file
@@ -0,0 +1,143 @@
|
|||||||
|
import * as fs from 'fs';
|
||||||
|
import * as path from 'path';
|
||||||
|
import { build_dir, dev, manifest, IGNORE } from '../placeholders';
|
||||||
|
import { Handler, Req, Res, Store } from './types';
|
||||||
|
import { get_server_route_handler } from './get_server_route_handler';
|
||||||
|
import { get_page_handler } from './get_page_handler';
|
||||||
|
import { lookup } from './mime';
|
||||||
|
|
||||||
|
export default function middleware(opts: {
|
||||||
|
store?: (req: Req, res: Res) => Store,
|
||||||
|
ignore?: any
|
||||||
|
} = {}) {
|
||||||
|
const { store, ignore } = opts;
|
||||||
|
|
||||||
|
let emitted_basepath = false;
|
||||||
|
|
||||||
|
return compose_handlers([
|
||||||
|
ignore && ((req: Req, res: Res, next: () => void) => {
|
||||||
|
req[IGNORE] = should_ignore(req.path, ignore);
|
||||||
|
next();
|
||||||
|
}),
|
||||||
|
|
||||||
|
(req: Req, res: Res, next: () => void) => {
|
||||||
|
if (req[IGNORE]) return next();
|
||||||
|
|
||||||
|
if (req.baseUrl === undefined) {
|
||||||
|
let { originalUrl } = req;
|
||||||
|
if (req.url === '/' && originalUrl[originalUrl.length - 1] !== '/') {
|
||||||
|
originalUrl += '/';
|
||||||
|
}
|
||||||
|
|
||||||
|
req.baseUrl = originalUrl
|
||||||
|
? originalUrl.slice(0, -req.url.length)
|
||||||
|
: '';
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!emitted_basepath && process.send) {
|
||||||
|
process.send({
|
||||||
|
__sapper__: true,
|
||||||
|
event: 'basepath',
|
||||||
|
basepath: req.baseUrl
|
||||||
|
});
|
||||||
|
|
||||||
|
emitted_basepath = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (req.path === undefined) {
|
||||||
|
req.path = req.url.replace(/\?.*/, '');
|
||||||
|
}
|
||||||
|
|
||||||
|
next();
|
||||||
|
},
|
||||||
|
|
||||||
|
fs.existsSync(path.join(build_dir, 'index.html')) && serve({
|
||||||
|
pathname: '/index.html',
|
||||||
|
cache_control: dev ? 'no-cache' : 'max-age=600'
|
||||||
|
}),
|
||||||
|
|
||||||
|
fs.existsSync(path.join(build_dir, 'service-worker.js')) && serve({
|
||||||
|
pathname: '/service-worker.js',
|
||||||
|
cache_control: 'no-cache, no-store, must-revalidate'
|
||||||
|
}),
|
||||||
|
|
||||||
|
fs.existsSync(path.join(build_dir, 'service-worker.js.map')) && serve({
|
||||||
|
pathname: '/service-worker.js.map',
|
||||||
|
cache_control: 'no-cache, no-store, must-revalidate'
|
||||||
|
}),
|
||||||
|
|
||||||
|
serve({
|
||||||
|
prefix: '/client/',
|
||||||
|
cache_control: dev ? 'no-cache' : 'max-age=31536000, immutable'
|
||||||
|
}),
|
||||||
|
|
||||||
|
get_server_route_handler(manifest.server_routes),
|
||||||
|
|
||||||
|
get_page_handler(manifest, store)
|
||||||
|
].filter(Boolean));
|
||||||
|
}
|
||||||
|
|
||||||
|
export function compose_handlers(handlers: Handler[]) {
|
||||||
|
return (req: Req, res: Res, next: () => void) => {
|
||||||
|
let i = 0;
|
||||||
|
function go() {
|
||||||
|
const handler = handlers[i];
|
||||||
|
|
||||||
|
if (handler) {
|
||||||
|
handler(req, res, () => {
|
||||||
|
i += 1;
|
||||||
|
go();
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
next();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
go();
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function should_ignore(uri: string, val: any) {
|
||||||
|
if (Array.isArray(val)) return val.some(x => should_ignore(uri, x));
|
||||||
|
if (val instanceof RegExp) return val.test(uri);
|
||||||
|
if (typeof val === 'function') return val(uri);
|
||||||
|
return uri.startsWith(val.charCodeAt(0) === 47 ? val : `/${val}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function serve({ prefix, pathname, cache_control }: {
|
||||||
|
prefix?: string,
|
||||||
|
pathname?: string,
|
||||||
|
cache_control: string
|
||||||
|
}) {
|
||||||
|
const filter = pathname
|
||||||
|
? (req: Req) => req.path === pathname
|
||||||
|
: (req: Req) => req.path.startsWith(prefix);
|
||||||
|
|
||||||
|
const cache: Map<string, Buffer> = new Map();
|
||||||
|
|
||||||
|
const read = dev
|
||||||
|
? (file: string) => fs.readFileSync(path.resolve(build_dir, file))
|
||||||
|
: (file: string) => (cache.has(file) ? cache : cache.set(file, fs.readFileSync(path.resolve(build_dir, file)))).get(file)
|
||||||
|
|
||||||
|
return (req: Req, res: Res, next: () => void) => {
|
||||||
|
if (req[IGNORE]) return next();
|
||||||
|
|
||||||
|
if (filter(req)) {
|
||||||
|
const type = lookup(req.path);
|
||||||
|
|
||||||
|
try {
|
||||||
|
const file = decodeURIComponent(req.path.slice(1));
|
||||||
|
const data = read(file);
|
||||||
|
|
||||||
|
res.setHeader('Content-Type', type);
|
||||||
|
res.setHeader('Cache-Control', cache_control);
|
||||||
|
res.end(data);
|
||||||
|
} catch (err) {
|
||||||
|
res.statusCode = 404;
|
||||||
|
res.end('not found');
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
next();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
69
templates/src/server/middleware/types.ts
Normal file
@@ -0,0 +1,69 @@
|
|||||||
|
import { ClientRequest, ServerResponse } from 'http';
|
||||||
|
|
||||||
|
export type ServerRoute = {
|
||||||
|
pattern: RegExp;
|
||||||
|
handlers: Record<string, Handler>;
|
||||||
|
params: (match: RegExpMatchArray) => Record<string, string>;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type Page = {
|
||||||
|
pattern: RegExp;
|
||||||
|
parts: Array<{
|
||||||
|
name: string;
|
||||||
|
component: Component;
|
||||||
|
params?: (match: RegExpMatchArray) => Record<string, string>;
|
||||||
|
}>
|
||||||
|
};
|
||||||
|
|
||||||
|
export type Manifest = {
|
||||||
|
server_routes: ServerRoute[];
|
||||||
|
pages: Page[];
|
||||||
|
root: Component;
|
||||||
|
error: Component;
|
||||||
|
}
|
||||||
|
|
||||||
|
export type Handler = (req: Req, res: Res, next: () => void) => void;
|
||||||
|
|
||||||
|
export type Store = {
|
||||||
|
get: () => any
|
||||||
|
};
|
||||||
|
|
||||||
|
export type Props = {
|
||||||
|
path: string;
|
||||||
|
query: Record<string, string>;
|
||||||
|
params: Record<string, string>;
|
||||||
|
error?: { message: string };
|
||||||
|
status?: number;
|
||||||
|
child: {
|
||||||
|
segment: string;
|
||||||
|
component: Component;
|
||||||
|
props: Props;
|
||||||
|
};
|
||||||
|
[key: string]: any;
|
||||||
|
};
|
||||||
|
|
||||||
|
export interface Req extends ClientRequest {
|
||||||
|
url: string;
|
||||||
|
baseUrl: string;
|
||||||
|
originalUrl: string;
|
||||||
|
method: string;
|
||||||
|
path: string;
|
||||||
|
params: Record<string, string>;
|
||||||
|
query: Record<string, string>;
|
||||||
|
headers: Record<string, string>;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface Res extends ServerResponse {
|
||||||
|
write: (data: any) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export { ServerResponse };
|
||||||
|
|
||||||
|
interface Component {
|
||||||
|
render: (data: any, opts: { store: Store }) => {
|
||||||
|
head: string;
|
||||||
|
css: { code: string, map: any };
|
||||||
|
html: string
|
||||||
|
},
|
||||||
|
preload: (data: any) => any | Promise<any>
|
||||||
|
}
|
||||||
11
templates/src/server/placeholders.ts
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
import { Manifest } from './types';
|
||||||
|
|
||||||
|
export const manifest: Manifest = __MANIFEST__;
|
||||||
|
|
||||||
|
export const build_dir = __BUILD__DIR__;
|
||||||
|
|
||||||
|
export const src_dir = __SRC__DIR__;
|
||||||
|
|
||||||
|
export const dev = __DEV__;
|
||||||
|
|
||||||
|
export const IGNORE = '__SAPPER__IGNORE__';
|
||||||
@@ -1,13 +0,0 @@
|
|||||||
import { init, prefetchRoutes } from '../../../runtime.js';
|
|
||||||
import { Store } from 'svelte/store.js';
|
|
||||||
import { manifest } from './manifest/client.js';
|
|
||||||
|
|
||||||
window.init = () => {
|
|
||||||
return init({
|
|
||||||
target: document.querySelector('#sapper'),
|
|
||||||
manifest,
|
|
||||||
store: data => new Store(data)
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
window.prefetchRoutes = prefetchRoutes;
|
|
||||||
@@ -1,28 +0,0 @@
|
|||||||
export function get(req, res) {
|
|
||||||
const cookies = req.headers.cookie
|
|
||||||
? req.headers.cookie.split(/,\s+/).reduce((cookies, cookie) => {
|
|
||||||
const [pair] = cookie.split('; ');
|
|
||||||
const [name, value] = pair.split('=');
|
|
||||||
cookies[name] = value;
|
|
||||||
return cookies;
|
|
||||||
}, {})
|
|
||||||
: {};
|
|
||||||
|
|
||||||
if (cookies.test) {
|
|
||||||
res.writeHead(200, {
|
|
||||||
'Content-Type': 'application/json'
|
|
||||||
});
|
|
||||||
|
|
||||||
res.end(JSON.stringify({
|
|
||||||
message: cookies.test
|
|
||||||
}));
|
|
||||||
} else {
|
|
||||||
res.writeHead(403, {
|
|
||||||
'Content-Type': 'application/json'
|
|
||||||
});
|
|
||||||
|
|
||||||
res.end(JSON.stringify({
|
|
||||||
message: 'unauthorized'
|
|
||||||
}));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
12
test/app/src/client.js
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
import { Store } from 'svelte/store.js';
|
||||||
|
import * as sapper from '../__sapper__/client.js';
|
||||||
|
|
||||||
|
window.init = () => {
|
||||||
|
return sapper.start({
|
||||||
|
target: document.querySelector('#sapper'),
|
||||||
|
store: data => new Store(data)
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
window.prefetchRoutes = sapper.prefetchRoutes;
|
||||||
|
window.goto = sapper.goto;
|
||||||
@@ -9,17 +9,9 @@
|
|||||||
<button class='prefetch' on:click='prefetch("blog/why-the-name")'>Why the name?</button>
|
<button class='prefetch' on:click='prefetch("blog/why-the-name")'>Why the name?</button>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
import { goto, prefetch } from '../../../runtime.js';
|
import { prefetch } from '../../__sapper__/client.js';
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
oncreate() {
|
|
||||||
window.goto = goto;
|
|
||||||
},
|
|
||||||
|
|
||||||
ondestroy() {
|
|
||||||
window.goto = null;
|
|
||||||
},
|
|
||||||
|
|
||||||
methods: {
|
methods: {
|
||||||
prefetch
|
prefetch
|
||||||
}
|
}
|
||||||
@@ -106,6 +106,14 @@ const posts = [
|
|||||||
<p>If you didn't have adult onset diabetes, I wouldn't mind giving you a little sugar. Everybody dance NOW. And the soup of the day is bread. Great, now I'm gonna smell to high heaven like a tuna melt!</p>
|
<p>If you didn't have adult onset diabetes, I wouldn't mind giving you a little sugar. Everybody dance NOW. And the soup of the day is bread. Great, now I'm gonna smell to high heaven like a tuna melt!</p>
|
||||||
<p>That's how Tony Wonder lost a nut. She calls it a Mayonegg. Go ahead, touch the Cornballer. There's a new daddy in town. A discipline daddy.</p>
|
<p>That's how Tony Wonder lost a nut. She calls it a Mayonegg. Go ahead, touch the Cornballer. There's a new daddy in town. A discipline daddy.</p>
|
||||||
`
|
`
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
title: 'Encödïng test',
|
||||||
|
slug: 'encödïng-test',
|
||||||
|
html: `
|
||||||
|
<p>It works</p>
|
||||||
|
`
|
||||||
}
|
}
|
||||||
];
|
];
|
||||||
|
|
||||||
23
test/app/src/routes/credentials/test.json.js
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
import cookie from 'cookie';
|
||||||
|
|
||||||
|
export function get(req, res) {
|
||||||
|
if (req.headers.cookie) {
|
||||||
|
const cookies = cookie.parse(req.headers.cookie);
|
||||||
|
|
||||||
|
res.writeHead(200, {
|
||||||
|
'Content-Type': 'application/json'
|
||||||
|
});
|
||||||
|
|
||||||
|
res.end(JSON.stringify({
|
||||||
|
message: `a: ${cookies.a}, b: ${cookies.b}, max-age: ${cookies['max-age']}`
|
||||||
|
}));
|
||||||
|
} else {
|
||||||
|
res.writeHead(403, {
|
||||||
|
'Content-Type': 'application/json'
|
||||||
|
});
|
||||||
|
|
||||||
|
res.end(JSON.stringify({
|
||||||
|
message: 'unauthorized'
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
}
|
||||||
12
test/app/src/routes/echo/page/[slug].html
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
<h1>{slug} ({message})</h1>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
export default {
|
||||||
|
preload({ params, query }) {
|
||||||
|
return {
|
||||||
|
slug: params.slug,
|
||||||
|
message: query.message
|
||||||
|
};
|
||||||
|
}
|
||||||
|
};
|
||||||
|
</script>
|
||||||
15
test/app/src/routes/echo/server-route/[slug].js
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
export function get(req, res) {
|
||||||
|
res.writeHead(200, {
|
||||||
|
'Content-Type': 'text/html'
|
||||||
|
});
|
||||||
|
|
||||||
|
res.end(`
|
||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head><meta charset="utf-8"></head>
|
||||||
|
<body>
|
||||||
|
<h1>${req.params.slug}</h1>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
`);
|
||||||
|
}
|
||||||
@@ -15,6 +15,8 @@
|
|||||||
<a href='credentials?creds=include'>credentials</a>
|
<a href='credentials?creds=include'>credentials</a>
|
||||||
<a rel=prefetch class='{page === "blog" ? "selected" : ""}' href='blog'>blog</a>
|
<a rel=prefetch class='{page === "blog" ? "selected" : ""}' href='blog'>blog</a>
|
||||||
<a href="const">const</a>
|
<a href="const">const</a>
|
||||||
|
<a href="echo/page/encöded?message=hëllö+wörld">echo/page/encöded?message=hëllö+wörld</a>
|
||||||
|
<a href="echo/page/empty?message">echo/page/empty?message</a>
|
||||||
|
|
||||||
<div class='hydrate-test'></div>
|
<div class='hydrate-test'></div>
|
||||||
|
|
||||||
@@ -2,9 +2,8 @@ import fs from 'fs';
|
|||||||
import { resolve } from 'url';
|
import { resolve } from 'url';
|
||||||
import express from 'express';
|
import express from 'express';
|
||||||
import serve from 'serve-static';
|
import serve from 'serve-static';
|
||||||
import sapper from '../../../dist/middleware.js';
|
|
||||||
import { Store } from 'svelte/store.js';
|
import { Store } from 'svelte/store.js';
|
||||||
import { manifest } from './manifest/server.js';
|
import * as sapper from '../__sapper__/server.js';
|
||||||
|
|
||||||
let pending;
|
let pending;
|
||||||
let ended;
|
let ended;
|
||||||
@@ -45,7 +44,7 @@ const middlewares = [
|
|||||||
|
|
||||||
// set test cookie
|
// set test cookie
|
||||||
(req, res, next) => {
|
(req, res, next) => {
|
||||||
res.setHeader('Set-Cookie', 'test=woohoo!; Max-Age=3600');
|
res.setHeader('Set-Cookie', ['a=1; Path=/', 'b=2; Path=/']);
|
||||||
next();
|
next();
|
||||||
},
|
},
|
||||||
|
|
||||||
@@ -92,8 +91,7 @@ const middlewares = [
|
|||||||
next();
|
next();
|
||||||
},
|
},
|
||||||
|
|
||||||
sapper({
|
sapper.middleware({
|
||||||
manifest,
|
|
||||||
store: (req, res) => {
|
store: (req, res) => {
|
||||||
return new Store({
|
return new Store({
|
||||||
title: `${req.hello} ${res.locals.name}`
|
title: `${req.hello} ${res.locals.name}`
|
||||||
@@ -1,10 +1,10 @@
|
|||||||
import { assets, shell, timestamp, routes } from './manifest/service-worker.js';
|
import { files, shell, timestamp, routes } from '../__sapper__/service-worker.js';
|
||||||
|
|
||||||
const ASSETS = `cachetimestamp`;
|
const ASSETS = `cachetimestamp`;
|
||||||
|
|
||||||
// `shell` is an array of all the files generated by webpack,
|
// `shell` is an array of all the files generated by webpack,
|
||||||
// `assets` is an array of everything in the `assets` directory
|
// `assets` is an array of everything in the `assets` directory
|
||||||
const to_cache = shell.concat(assets);
|
const to_cache = shell.concat(files);
|
||||||
const cached = new Set(to_cache);
|
const cached = new Set(to_cache);
|
||||||
|
|
||||||
self.addEventListener('install', event => {
|
self.addEventListener('install', event => {
|
||||||
|
Before Width: | Height: | Size: 1.9 KiB After Width: | Height: | Size: 1.9 KiB |
|
Before Width: | Height: | Size: 80 KiB After Width: | Height: | Size: 80 KiB |
|
Before Width: | Height: | Size: 20 KiB After Width: | Height: | Size: 20 KiB |
|
Before Width: | Height: | Size: 6.3 KiB After Width: | Height: | Size: 6.3 KiB |
84
test/app/webpack.config.js
Normal file
@@ -0,0 +1,84 @@
|
|||||||
|
const path = require('path');
|
||||||
|
const webpack = require('webpack');
|
||||||
|
const config = require('../../config/webpack.js');
|
||||||
|
const sapper_pkg = require('../../package.json');
|
||||||
|
|
||||||
|
const mode = process.env.NODE_ENV;
|
||||||
|
const isDev = mode === 'development';
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
client: {
|
||||||
|
entry: config.client.entry(),
|
||||||
|
output: config.client.output(),
|
||||||
|
resolve: {
|
||||||
|
extensions: ['.js', '.html']
|
||||||
|
},
|
||||||
|
module: {
|
||||||
|
rules: [
|
||||||
|
{
|
||||||
|
test: /\.html$/,
|
||||||
|
exclude: /node_modules/,
|
||||||
|
use: {
|
||||||
|
loader: 'svelte-loader',
|
||||||
|
options: {
|
||||||
|
hydratable: true,
|
||||||
|
cascade: false,
|
||||||
|
store: true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
mode,
|
||||||
|
optimization: {
|
||||||
|
minimize: false
|
||||||
|
},
|
||||||
|
plugins: [
|
||||||
|
isDev && new webpack.HotModuleReplacementPlugin()
|
||||||
|
].filter(Boolean),
|
||||||
|
devtool: isDev && 'inline-source-map'
|
||||||
|
},
|
||||||
|
|
||||||
|
server: {
|
||||||
|
entry: config.server.entry(),
|
||||||
|
output: config.server.output(),
|
||||||
|
target: 'node',
|
||||||
|
resolve: {
|
||||||
|
extensions: ['.js', '.html']
|
||||||
|
},
|
||||||
|
externals: [].concat(
|
||||||
|
Object.keys(sapper_pkg.dependencies),
|
||||||
|
Object.keys(sapper_pkg.devDependencies)
|
||||||
|
),
|
||||||
|
module: {
|
||||||
|
rules: [
|
||||||
|
{
|
||||||
|
test: /\.html$/,
|
||||||
|
exclude: /node_modules/,
|
||||||
|
use: {
|
||||||
|
loader: 'svelte-loader',
|
||||||
|
options: {
|
||||||
|
css: false,
|
||||||
|
cascade: false,
|
||||||
|
store: true,
|
||||||
|
generate: 'ssr'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
mode,
|
||||||
|
optimization: {
|
||||||
|
minimize: false
|
||||||
|
},
|
||||||
|
performance: {
|
||||||
|
hints: false // it doesn't matter if server.js is large
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
serviceworker: {
|
||||||
|
entry: config.serviceworker.entry(),
|
||||||
|
output: config.serviceworker.output(),
|
||||||
|
mode
|
||||||
|
}
|
||||||
|
};
|
||||||
@@ -1,34 +0,0 @@
|
|||||||
const config = require('../../../config/webpack.js');
|
|
||||||
const webpack = require('webpack');
|
|
||||||
|
|
||||||
const mode = process.env.NODE_ENV;
|
|
||||||
const isDev = mode === 'development';
|
|
||||||
|
|
||||||
module.exports = {
|
|
||||||
entry: config.client.entry(),
|
|
||||||
output: config.client.output(),
|
|
||||||
resolve: {
|
|
||||||
extensions: ['.js', '.html']
|
|
||||||
},
|
|
||||||
module: {
|
|
||||||
rules: [
|
|
||||||
{
|
|
||||||
test: /\.html$/,
|
|
||||||
exclude: /node_modules/,
|
|
||||||
use: {
|
|
||||||
loader: 'svelte-loader',
|
|
||||||
options: {
|
|
||||||
hydratable: true,
|
|
||||||
cascade: false,
|
|
||||||
store: true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
mode,
|
|
||||||
plugins: [
|
|
||||||
isDev && new webpack.HotModuleReplacementPlugin()
|
|
||||||
].filter(Boolean),
|
|
||||||
devtool: isDev && 'inline-source-map'
|
|
||||||
};
|
|
||||||
@@ -1,36 +0,0 @@
|
|||||||
const config = require('../../../config/webpack.js');
|
|
||||||
const sapper_pkg = require('../../../package.json');
|
|
||||||
|
|
||||||
module.exports = {
|
|
||||||
entry: config.server.entry(),
|
|
||||||
output: config.server.output(),
|
|
||||||
target: 'node',
|
|
||||||
resolve: {
|
|
||||||
extensions: ['.js', '.html']
|
|
||||||
},
|
|
||||||
externals: [].concat(
|
|
||||||
Object.keys(sapper_pkg.dependencies),
|
|
||||||
Object.keys(sapper_pkg.devDependencies)
|
|
||||||
),
|
|
||||||
module: {
|
|
||||||
rules: [
|
|
||||||
{
|
|
||||||
test: /\.html$/,
|
|
||||||
exclude: /node_modules/,
|
|
||||||
use: {
|
|
||||||
loader: 'svelte-loader',
|
|
||||||
options: {
|
|
||||||
css: false,
|
|
||||||
cascade: false,
|
|
||||||
store: true,
|
|
||||||
generate: 'ssr'
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
mode: process.env.NODE_ENV,
|
|
||||||
performance: {
|
|
||||||
hints: false // it doesn't matter if server.js is large
|
|
||||||
}
|
|
||||||
};
|
|
||||||
@@ -1,7 +0,0 @@
|
|||||||
const config = require('../../../config/webpack.js');
|
|
||||||
|
|
||||||
module.exports = {
|
|
||||||
entry: config.serviceworker.entry(),
|
|
||||||
output: config.serviceworker.output(),
|
|
||||||
mode: process.env.NODE_ENV
|
|
||||||
};
|
|
||||||
@@ -5,6 +5,7 @@ const Nightmare = require('nightmare');
|
|||||||
const walkSync = require('walk-sync');
|
const walkSync = require('walk-sync');
|
||||||
const rimraf = require('rimraf');
|
const rimraf = require('rimraf');
|
||||||
const ports = require('port-authority');
|
const ports = require('port-authority');
|
||||||
|
const fetch = require('node-fetch');
|
||||||
|
|
||||||
Nightmare.action('page', {
|
Nightmare.action('page', {
|
||||||
title(done) {
|
title(done) {
|
||||||
@@ -36,10 +37,7 @@ describe('sapper', function() {
|
|||||||
process.chdir(path.resolve(__dirname, '../app'));
|
process.chdir(path.resolve(__dirname, '../app'));
|
||||||
|
|
||||||
// clean up after previous test runs
|
// clean up after previous test runs
|
||||||
rimraf.sync('export');
|
rimraf.sync('__sapper__');
|
||||||
rimraf.sync('build');
|
|
||||||
rimraf.sync('.sapper');
|
|
||||||
rimraf.sync('start.js');
|
|
||||||
|
|
||||||
this.timeout(process.env.CI ? 30000 : 15000);
|
this.timeout(process.env.CI ? 30000 : 15000);
|
||||||
|
|
||||||
@@ -73,7 +71,7 @@ function testExport({ basepath = '' }) {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it('export all pages', () => {
|
it('export all pages', () => {
|
||||||
const dest = path.resolve(__dirname, '../app/export');
|
const dest = path.resolve(__dirname, '../app/__sapper__/export');
|
||||||
|
|
||||||
// Pages that should show up in the extraction directory.
|
// Pages that should show up in the extraction directory.
|
||||||
const expectedPages = [
|
const expectedPages = [
|
||||||
@@ -93,6 +91,7 @@ function testExport({ basepath = '' }) {
|
|||||||
'blog/how-to-use-sapper/index.html',
|
'blog/how-to-use-sapper/index.html',
|
||||||
'blog/what-is-sapper/index.html',
|
'blog/what-is-sapper/index.html',
|
||||||
'blog/why-the-name/index.html',
|
'blog/why-the-name/index.html',
|
||||||
|
'blog/encödïng-test/index.html',
|
||||||
|
|
||||||
'blog.json',
|
'blog.json',
|
||||||
'blog/a-very-long-post.json',
|
'blog/a-very-long-post.json',
|
||||||
@@ -101,6 +100,7 @@ function testExport({ basepath = '' }) {
|
|||||||
'blog/how-to-use-sapper.json',
|
'blog/how-to-use-sapper.json',
|
||||||
'blog/what-is-sapper.json',
|
'blog/what-is-sapper.json',
|
||||||
'blog/why-the-name.json',
|
'blog/why-the-name.json',
|
||||||
|
'blog/encödïng-test.json',
|
||||||
|
|
||||||
'favicon.png',
|
'favicon.png',
|
||||||
'global.css',
|
'global.css',
|
||||||
@@ -178,13 +178,13 @@ function run({ mode, basepath = '' }) {
|
|||||||
base = `http://localhost:${port}`;
|
base = `http://localhost:${port}`;
|
||||||
if (basepath) base += basepath;
|
if (basepath) base += basepath;
|
||||||
|
|
||||||
const dir = mode === 'production' ? 'build' : '.sapper';
|
const dir = mode === 'production' ? '__sapper__/build' : '__sapper__/dev';
|
||||||
|
|
||||||
if (mode === 'production') {
|
if (mode === 'production') {
|
||||||
assert.ok(fs.existsSync('build/index.js'));
|
assert.ok(fs.existsSync('__sapper__/build/index.js'));
|
||||||
}
|
}
|
||||||
|
|
||||||
proc = require('child_process').fork(`${dir}/server.js`, {
|
proc = require('child_process').fork(`${dir}/server/server.js`, {
|
||||||
cwd: process.cwd(),
|
cwd: process.cwd(),
|
||||||
env: {
|
env: {
|
||||||
NODE_ENV: mode,
|
NODE_ENV: mode,
|
||||||
@@ -623,7 +623,7 @@ function run({ mode, basepath = '' }) {
|
|||||||
return nightmare.goto(`${base}/credentials?creds=include`)
|
return nightmare.goto(`${base}/credentials?creds=include`)
|
||||||
.page.title()
|
.page.title()
|
||||||
.then(title => {
|
.then(title => {
|
||||||
assert.equal(title, 'woohoo!');
|
assert.equal(title, 'a: 1, b: 2, max-age: undefined');
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -641,7 +641,7 @@ function run({ mode, basepath = '' }) {
|
|||||||
.wait(100)
|
.wait(100)
|
||||||
.page.title()
|
.page.title()
|
||||||
.then(title => {
|
.then(title => {
|
||||||
assert.equal(title, 'woohoo!');
|
assert.equal(title, 'a: 1, b: 2, max-age: undefined');
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -751,18 +751,67 @@ function run({ mode, basepath = '' }) {
|
|||||||
assert.equal(title, 'reserved words are okay as routes');
|
assert.equal(title, 'reserved words are okay as routes');
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('encodes req.params and req.query for server-rendered pages', () => {
|
||||||
|
return nightmare.goto(`${base}/echo/page/encöded?message=hëllö+wörld`)
|
||||||
|
.page.title()
|
||||||
|
.then(title => {
|
||||||
|
assert.equal(title, 'encöded (hëllö wörld)');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('encodes req.params and req.query for client-rendered pages', () => {
|
||||||
|
return nightmare.goto(base).init()
|
||||||
|
.click('a[href="echo/page/encöded?message=hëllö+wörld"]')
|
||||||
|
.wait(100)
|
||||||
|
.page.title()
|
||||||
|
.then(title => {
|
||||||
|
assert.equal(title, 'encöded (hëllö wörld)');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('accepts value-less query string parameter on server', () => {
|
||||||
|
return nightmare.goto(`${base}/echo/page/empty?message`)
|
||||||
|
.page.title()
|
||||||
|
.then(title => {
|
||||||
|
assert.equal(title, 'empty ()');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('accepts value-less query string parameter on client', () => {
|
||||||
|
return nightmare.goto(base).init()
|
||||||
|
.click('a[href="echo/page/empty?message"]')
|
||||||
|
.wait(100)
|
||||||
|
.page.title()
|
||||||
|
.then(title => {
|
||||||
|
assert.equal(title, 'empty ()');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('encodes req.params for server routes', () => {
|
||||||
|
return nightmare.goto(`${base}/echo/server-route/encöded`)
|
||||||
|
.page.title()
|
||||||
|
.then(title => {
|
||||||
|
assert.equal(title, 'encöded');
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('headers', () => {
|
describe('headers', () => {
|
||||||
it('sets Content-Type and Link...preload headers', () => {
|
it('sets Content-Type, Link...preload, and Cache-Control headers', () => {
|
||||||
return capture(() => nightmare.goto(base)).then(requests => {
|
return capture(() => fetch(base)).then(responses => {
|
||||||
const { headers } = requests[0];
|
const { headers } = responses[0];
|
||||||
|
|
||||||
assert.equal(
|
assert.equal(
|
||||||
headers['content-type'],
|
headers['content-type'],
|
||||||
'text/html'
|
'text/html'
|
||||||
);
|
);
|
||||||
|
|
||||||
|
assert.equal(
|
||||||
|
headers['cache-control'],
|
||||||
|
'max-age=600'
|
||||||
|
);
|
||||||
|
|
||||||
const str = ['main', '.+?\\.\\d+']
|
const str = ['main', '.+?\\.\\d+']
|
||||||
.map(file => {
|
.map(file => {
|
||||||
return `<${basepath}/client/[^/]+/${file}\\.js>;rel="preload";as="script"`;
|
return `<${basepath}/client/[^/]+/${file}\\.js>;rel="preload";as="script"`;
|
||||||
|
|||||||