mirror of
https://github.com/kevin-DL/sapper.git
synced 2026-01-14 20:14:39 +00:00
Compare commits
138 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
46bf8f2b78 | ||
|
|
553db81b7b | ||
|
|
67cc29ed38 | ||
|
|
36f930f489 | ||
|
|
3b098caa6e | ||
|
|
d63b9437b5 | ||
|
|
e51c733e3f | ||
|
|
708fe4c74b | ||
|
|
4259fc8e58 | ||
|
|
f05a8e52a0 | ||
|
|
76cb6d97f3 | ||
|
|
5d0b7af47b | ||
|
|
bb737eeb32 | ||
|
|
86dee17040 | ||
|
|
01a709e017 | ||
|
|
f87f0e3b80 | ||
|
|
8226e9bc1f | ||
|
|
d6d0a15015 | ||
|
|
ddec58ebd4 | ||
|
|
9d904b3911 | ||
|
|
c36df0d650 | ||
|
|
ae19288797 | ||
|
|
de308d5bb0 | ||
|
|
99b096a5c4 | ||
|
|
36fc8a947b | ||
|
|
6393a30b13 | ||
|
|
458be49b35 | ||
|
|
f8d742bdd0 | ||
|
|
7e698f1613 | ||
|
|
70b5cc86dc | ||
|
|
19a5dcad1d | ||
|
|
85e25d6380 | ||
|
|
6e2383b66b | ||
|
|
200c5fcbd2 | ||
|
|
9cbb8bdc33 | ||
|
|
3d39836cfb | ||
|
|
24f2855f89 | ||
|
|
d5bf206d2a | ||
|
|
8abc01551e | ||
|
|
62b8a79e9f | ||
|
|
7f255563a4 | ||
|
|
32f4a50f25 | ||
|
|
b1a9be2dc3 | ||
|
|
c5456d3033 | ||
|
|
9b33dad589 | ||
|
|
4315a46ff2 | ||
|
|
0fb5827968 | ||
|
|
f9bf23dc43 | ||
|
|
611017fd28 | ||
|
|
72b265a35f | ||
|
|
e0d533f2ea | ||
|
|
dba83641e4 | ||
|
|
14e5c8e761 | ||
|
|
cbbf4a95db | ||
|
|
55b7ffd2ed | ||
|
|
9f4d4e70de | ||
|
|
deef1bbfcf | ||
|
|
17b0fc0d0c | ||
|
|
3c44c511e4 | ||
|
|
7cf1b9613a | ||
|
|
99e5a9601c | ||
|
|
4c9c1dccf5 | ||
|
|
2cddd5afa0 | ||
|
|
8c6a0c4773 | ||
|
|
af5063552d | ||
|
|
419d154794 | ||
|
|
abda059be5 | ||
|
|
444908cac5 | ||
|
|
c6da26e1a0 | ||
|
|
aad87857ce | ||
|
|
666c113297 | ||
|
|
84a58f34a0 | ||
|
|
75f5b5c721 | ||
|
|
a176a3b79b | ||
|
|
1627a5767a | ||
|
|
6ff3a9e9ab | ||
|
|
3ce2bd30f9 | ||
|
|
de4f99807f | ||
|
|
eae8351f77 | ||
|
|
d386308301 | ||
|
|
13afbc84d7 | ||
|
|
31327b3780 | ||
|
|
81f483d7b8 | ||
|
|
1bcf20511b | ||
|
|
003fa8ab2c | ||
|
|
d1fcd07c92 | ||
|
|
47a6d6f662 | ||
|
|
4b2b6440d0 | ||
|
|
fc855f30f8 | ||
|
|
4a75fff4ec | ||
|
|
7b7b695938 | ||
|
|
2fca2e295f | ||
|
|
eae991d369 | ||
|
|
c2b393d3fd | ||
|
|
566addd406 | ||
|
|
3d77dacbd6 | ||
|
|
51b4f9cbbf | ||
|
|
1d611be83e | ||
|
|
1782904994 | ||
|
|
e3ddbfc181 | ||
|
|
8e3830b646 | ||
|
|
b28cdff233 | ||
|
|
7f586ff1a3 | ||
|
|
731d4f535c | ||
|
|
f8c731ca21 | ||
|
|
39eb3be01e | ||
|
|
d0bb728e25 | ||
|
|
58de0f9c99 | ||
|
|
b75ae7ba96 | ||
|
|
091e38082e | ||
|
|
74acf93c7a | ||
|
|
0e3775397f | ||
|
|
8dc52a04e4 | ||
|
|
008b607c01 | ||
|
|
d01a407137 | ||
|
|
c0c717d9ec | ||
|
|
4f011bfc37 | ||
|
|
6c4ab32cf0 | ||
|
|
09b4dc1b9a | ||
|
|
bdd5a54527 | ||
|
|
b7bb4db8c1 | ||
|
|
5b5f33d3cf | ||
|
|
9611656b76 | ||
|
|
e9a71774d5 | ||
|
|
2205b8aec5 | ||
|
|
5c4e4d5d36 | ||
|
|
e87247493f | ||
|
|
0aeb63a05b | ||
|
|
57eeb5659a | ||
|
|
f821c19528 | ||
|
|
b9a120164a | ||
|
|
087356f781 | ||
|
|
31110a5326 | ||
|
|
667a68768c | ||
|
|
5075981a90 | ||
|
|
0e2c2ca101 | ||
|
|
8015be8069 | ||
|
|
e39ad59589 |
@@ -18,4 +18,4 @@ addons:
|
|||||||
install:
|
install:
|
||||||
- export DISPLAY=':99.0'
|
- export DISPLAY=':99.0'
|
||||||
- Xvfb :99 -screen 0 1024x768x24 > /dev/null 2>&1 &
|
- Xvfb :99 -screen 0 1024x768x24 > /dev/null 2>&1 &
|
||||||
- npm install
|
- npm ci || npm i
|
||||||
|
|||||||
94
CHANGELOG.md
94
CHANGELOG.md
@@ -1,5 +1,99 @@
|
|||||||
# sapper changelog
|
# sapper changelog
|
||||||
|
|
||||||
|
## 0.18.2
|
||||||
|
|
||||||
|
* Update `pkg.files`
|
||||||
|
|
||||||
|
## 0.18.1
|
||||||
|
|
||||||
|
* Add live reloading ([#385](https://github.com/sveltejs/sapper/issues/385))
|
||||||
|
|
||||||
|
## 0.18.0
|
||||||
|
|
||||||
|
* Rollup support ([#379](https://github.com/sveltejs/sapper/pull/379))
|
||||||
|
* Fail `export` if a page times out (configurable with `--timeout`) ([#378](https://github.com/sveltejs/sapper/pull/378))
|
||||||
|
|
||||||
|
## 0.17.1
|
||||||
|
|
||||||
|
* Print which file is causing build errors/warnings ([#371](https://github.com/sveltejs/sapper/pull/371))
|
||||||
|
|
||||||
|
## 0.17.0
|
||||||
|
|
||||||
|
* Use `cheap-watch` instead of `chokidar` ([#364](https://github.com/sveltejs/sapper/issues/364))
|
||||||
|
|
||||||
|
## 0.16.1
|
||||||
|
|
||||||
|
* Fix file watching regression in previous version
|
||||||
|
|
||||||
|
## 0.16.0
|
||||||
|
|
||||||
|
* Slim down installed package ([#363](https://github.com/sveltejs/sapper/pull/363))
|
||||||
|
|
||||||
|
## 0.15.8
|
||||||
|
|
||||||
|
* Only set `preloading: true` on navigation, not prefetch ([#352](https://github.com/sveltejs/sapper/issues/352))
|
||||||
|
* Provide fallback for missing preload errors ([#361](https://github.com/sveltejs/sapper/pull/361))
|
||||||
|
|
||||||
|
## 0.15.7
|
||||||
|
|
||||||
|
* Strip leading slash from redirects ([#291](https://github.com/sveltejs/sapper/issues/291))
|
||||||
|
* Pass `(req, res)` to store getter ([#344](https://github.com/sveltejs/sapper/issues/344))
|
||||||
|
|
||||||
|
## 0.15.6
|
||||||
|
|
||||||
|
* Fix exporting with custom basepath ([#342](https://github.com/sveltejs/sapper/pull/342))
|
||||||
|
|
||||||
|
## 0.15.5
|
||||||
|
|
||||||
|
* Faster `export` with more explanatory output ([#335](https://github.com/sveltejs/sapper/pull/335))
|
||||||
|
* Only blur `activeElement` if it exists ([#332](https://github.com/sveltejs/sapper/issues/332))
|
||||||
|
* Don't emit `client_info.json` or `server_info.json` ([#318](https://github.com/sveltejs/sapper/issues/318))
|
||||||
|
|
||||||
|
## 0.15.4
|
||||||
|
|
||||||
|
* Add `ignore` option ([#326](https://github.com/sveltejs/sapper/pull/326))
|
||||||
|
|
||||||
|
## 0.15.3
|
||||||
|
|
||||||
|
* Crawl pages in parallel when exporting ([#329](https://github.com/sveltejs/sapper/pull/329))
|
||||||
|
* Don't minify inline JS when exporting ([#328](https://github.com/sveltejs/sapper/pull/328))
|
||||||
|
|
||||||
|
## 0.15.2
|
||||||
|
|
||||||
|
* Collapse component chains where no intermediate layout component is specified ([#312](https://github.com/sveltejs/sapper/issues/312))
|
||||||
|
|
||||||
|
## 0.15.1
|
||||||
|
|
||||||
|
* Prevent confusing error when no root layout is specified
|
||||||
|
|
||||||
|
## 0.15.0
|
||||||
|
|
||||||
|
* Nested routes (consult [migration guide](https://sapper.svelte.technology/guide#0-14-to-0-15) and docs on [layouts](https://sapper.svelte.technology/guide#layouts)) ([#262](https://github.com/sveltejs/sapper/issues/262))
|
||||||
|
|
||||||
|
## 0.14.2
|
||||||
|
|
||||||
|
* Prevent unsafe replacements ([#307](https://github.com/sveltejs/sapper/pull/307))
|
||||||
|
|
||||||
|
## 0.14.1
|
||||||
|
|
||||||
|
* Route parameters can be qualified with regex characters ([#283](https://github.com/sveltejs/sapper/pull/283))
|
||||||
|
|
||||||
|
## 0.14.0
|
||||||
|
|
||||||
|
* `4xx.html` and `5xx.html` are replaced with `_error.html` ([#209](https://github.com/sveltejs/sapper/issues/209))
|
||||||
|
* Treat `foo/index.json.js` and `foo.json.js` as equivalents ([#297](https://github.com/sveltejs/sapper/issues/297))
|
||||||
|
* Return a promise from `goto` ([#270](https://github.com/sveltejs/sapper/issues/270))
|
||||||
|
* Use store when rendering error pages ([#293](https://github.com/sveltejs/sapper/issues/293))
|
||||||
|
* Prevent console errors when visiting an error page ([#279](https://github.com/sveltejs/sapper/issues/279))
|
||||||
|
|
||||||
|
## 0.13.6
|
||||||
|
|
||||||
|
* Fix `baseUrl` synthesis ([#296](https://github.com/sveltejs/sapper/issues/296))
|
||||||
|
|
||||||
|
## 0.13.5
|
||||||
|
|
||||||
|
* Fix handling of fatal errors ([#289](https://github.com/sveltejs/sapper/issues/289))
|
||||||
|
|
||||||
## 0.13.4
|
## 0.13.4
|
||||||
|
|
||||||
* Focus `<body>` after navigation ([#287](https://github.com/sveltejs/sapper/issues/287))
|
* Focus `<body>` after navigation ([#287](https://github.com/sveltejs/sapper/issues/287))
|
||||||
|
|||||||
38
README.md
38
README.md
@@ -31,6 +31,44 @@ npm run build
|
|||||||
npm start
|
npm start
|
||||||
```
|
```
|
||||||
|
|
||||||
|
## Development
|
||||||
|
|
||||||
|
Pull requests are encouraged and always welcome. [Pick an issue](https://github.com/sveltejs/sapper/issues?q=is%3Aissue+is%3Aopen+sort%3Aupdated-desc) and help us out!
|
||||||
|
|
||||||
|
To install and work on Sapper locally:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
git clone git@github.com:sveltejs/sapper.git
|
||||||
|
cd sapper
|
||||||
|
npm install
|
||||||
|
npm run dev
|
||||||
|
```
|
||||||
|
|
||||||
|
### Linking to a Live Project
|
||||||
|
|
||||||
|
You can make changes locally to Sapper and test it against a local Sapper project. For a quick project that takes almost no setup, use the default [sapper-template](https://github.com/sveltejs/sapper-template) project. Instruction on setup are found in that project repository.
|
||||||
|
|
||||||
|
To link Sapper to your project, from the root of your local Sapper git checkout:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
cd sapper
|
||||||
|
npm link
|
||||||
|
```
|
||||||
|
|
||||||
|
Then, to link from `sapper-template` (or any other given project):
|
||||||
|
|
||||||
|
```bash
|
||||||
|
cd sapper-template
|
||||||
|
npm link sapper
|
||||||
|
```
|
||||||
|
|
||||||
|
You should be good to test changes locally.
|
||||||
|
|
||||||
|
### Running Tests
|
||||||
|
|
||||||
|
```bash
|
||||||
|
npm run test
|
||||||
|
```
|
||||||
|
|
||||||
## License
|
## License
|
||||||
|
|
||||||
|
|||||||
2
api.js
2
api.js
@@ -1 +1 @@
|
|||||||
module.exports = require('./dist/api.ts.js');
|
module.exports = require('./dist/api.js');
|
||||||
@@ -10,11 +10,11 @@ build: off
|
|||||||
environment:
|
environment:
|
||||||
matrix:
|
matrix:
|
||||||
# node.js
|
# node.js
|
||||||
- nodejs_version: stable
|
- nodejs_version: 10.5
|
||||||
|
|
||||||
install:
|
install:
|
||||||
- ps: Install-Product node $env:nodejs_version
|
- ps: Install-Product node $env:nodejs_version
|
||||||
- npm install
|
- npm ci
|
||||||
|
|
||||||
test_script:
|
test_script:
|
||||||
- node --version && npm --version
|
- node --version && npm --version
|
||||||
|
|||||||
1
components/default-layout.html
Normal file
1
components/default-layout.html
Normal file
@@ -0,0 +1 @@
|
|||||||
|
<svelte:component this={child.component} {...child.props}/>
|
||||||
1
config/rollup.js
Normal file
1
config/rollup.js
Normal file
@@ -0,0 +1 @@
|
|||||||
|
module.exports = require('../dist/rollup.js');
|
||||||
1
config/webpack.js
Normal file
1
config/webpack.js
Normal file
@@ -0,0 +1 @@
|
|||||||
|
module.exports = require('../dist/webpack.js');
|
||||||
@@ -1,4 +1,5 @@
|
|||||||
--require source-map-support/register
|
--require source-map-support/register
|
||||||
|
--require ts-node/register
|
||||||
--recursive
|
--recursive
|
||||||
test/unit/**/*.js
|
test/unit/*/*.ts
|
||||||
test/common/test.js
|
test/common/test.js
|
||||||
7838
package-lock.json
generated
Normal file
7838
package-lock.json
generated
Normal file
File diff suppressed because it is too large
Load Diff
53
package.json
53
package.json
@@ -1,73 +1,78 @@
|
|||||||
{
|
{
|
||||||
"name": "sapper",
|
"name": "sapper",
|
||||||
"version": "0.13.4",
|
"version": "0.18.2",
|
||||||
"description": "Military-grade apps, engineered by Svelte",
|
"description": "Military-grade apps, engineered by Svelte",
|
||||||
"main": "dist/middleware.ts.js",
|
"main": "dist/middleware.js",
|
||||||
"bin": {
|
"bin": {
|
||||||
"sapper": "./sapper"
|
"sapper": "./sapper"
|
||||||
},
|
},
|
||||||
"files": [
|
"files": [
|
||||||
"*.js",
|
"*.js",
|
||||||
"*.ts.js",
|
|
||||||
"runtime",
|
"runtime",
|
||||||
"webpack",
|
"webpack",
|
||||||
|
"config",
|
||||||
"sapper",
|
"sapper",
|
||||||
|
"components",
|
||||||
"dist"
|
"dist"
|
||||||
],
|
],
|
||||||
"directories": {
|
"directories": {
|
||||||
"test": "test"
|
"test": "test"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"ansi-colors": "^2.0.1",
|
|
||||||
"cheerio": "^1.0.0-rc.2",
|
|
||||||
"chokidar": "^2.0.3",
|
|
||||||
"cookie": "^0.3.1",
|
|
||||||
"devalue": "^1.0.1",
|
|
||||||
"glob": "^7.1.2",
|
|
||||||
"html-minifier": "^3.5.16",
|
"html-minifier": "^3.5.16",
|
||||||
"mkdirp": "^0.5.1",
|
"shimport": "^0.0.9",
|
||||||
"node-fetch": "^2.1.1",
|
|
||||||
"port-authority": "^1.0.2",
|
|
||||||
"pretty-bytes": "^5.0.0",
|
|
||||||
"pretty-ms": "^3.1.0",
|
|
||||||
"require-relative": "^0.8.7",
|
|
||||||
"rimraf": "^2.6.2",
|
|
||||||
"sade": "^1.4.1",
|
|
||||||
"sander": "^0.6.0",
|
|
||||||
"source-map-support": "^0.5.6",
|
"source-map-support": "^0.5.6",
|
||||||
"tslib": "^1.9.1",
|
"tslib": "^1.9.1"
|
||||||
"url-parse": "^1.2.0",
|
|
||||||
"webpack-format-messages": "^2.0.1"
|
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@types/glob": "^5.0.34",
|
"@types/glob": "^5.0.34",
|
||||||
"@types/mkdirp": "^0.5.2",
|
"@types/mkdirp": "^0.5.2",
|
||||||
|
"@types/mocha": "^5.2.5",
|
||||||
|
"@types/node": "^10.7.1",
|
||||||
"@types/rimraf": "^2.0.2",
|
"@types/rimraf": "^2.0.2",
|
||||||
|
"cheap-watch": "^0.3.0",
|
||||||
"compression": "^1.7.1",
|
"compression": "^1.7.1",
|
||||||
|
"cookie": "^0.3.1",
|
||||||
|
"devalue": "^1.0.4",
|
||||||
"eslint": "^4.13.1",
|
"eslint": "^4.13.1",
|
||||||
"eslint-plugin-import": "^2.12.0",
|
"eslint-plugin-import": "^2.12.0",
|
||||||
"express": "^4.16.3",
|
"express": "^4.16.3",
|
||||||
|
"kleur": "^2.0.1",
|
||||||
|
"mkdirp": "^0.5.1",
|
||||||
"mocha": "^5.2.0",
|
"mocha": "^5.2.0",
|
||||||
"nightmare": "^3.0.0",
|
"nightmare": "^3.0.0",
|
||||||
|
"node-fetch": "^2.1.1",
|
||||||
"npm-run-all": "^4.1.3",
|
"npm-run-all": "^4.1.3",
|
||||||
"polka": "^0.4.0",
|
"polka": "^0.4.0",
|
||||||
"rollup": "^0.59.2",
|
"port-authority": "^1.0.5",
|
||||||
|
"pretty-bytes": "^5.0.0",
|
||||||
|
"pretty-ms": "^3.1.0",
|
||||||
|
"require-relative": "^0.8.7",
|
||||||
|
"rimraf": "^2.6.2",
|
||||||
|
"rollup": "^0.65.0",
|
||||||
"rollup-plugin-commonjs": "^9.1.3",
|
"rollup-plugin-commonjs": "^9.1.3",
|
||||||
"rollup-plugin-json": "^3.0.0",
|
"rollup-plugin-json": "^3.0.0",
|
||||||
|
"rollup-plugin-node-resolve": "^3.3.0",
|
||||||
"rollup-plugin-string": "^2.0.2",
|
"rollup-plugin-string": "^2.0.2",
|
||||||
"rollup-plugin-typescript": "^0.8.1",
|
"rollup-plugin-typescript": "^0.8.1",
|
||||||
|
"sade": "^1.4.1",
|
||||||
|
"sander": "^0.6.0",
|
||||||
"serve-static": "^1.13.2",
|
"serve-static": "^1.13.2",
|
||||||
"svelte": "^2.6.3",
|
"svelte": "^2.6.3",
|
||||||
"svelte-loader": "^2.9.0",
|
"svelte-loader": "^2.9.0",
|
||||||
|
"tiny-glob": "^0.2.2",
|
||||||
|
"ts-node": "^7.0.1",
|
||||||
"typescript": "^2.8.3",
|
"typescript": "^2.8.3",
|
||||||
|
"url-parse": "^1.2.0",
|
||||||
"walk-sync": "^0.3.2",
|
"walk-sync": "^0.3.2",
|
||||||
"webpack": "^4.8.3"
|
"webpack": "^4.8.3",
|
||||||
|
"webpack-format-messages": "^2.0.1"
|
||||||
},
|
},
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"cy:open": "cypress open",
|
"cy:open": "cypress open",
|
||||||
"test": "mocha --opts mocha.opts",
|
"test": "mocha --opts mocha.opts",
|
||||||
"pretest": "npm run build",
|
"pretest": "npm run build",
|
||||||
"build": "rollup -c",
|
"build": "rm -rf dist && rollup -c",
|
||||||
"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"
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
import typescript from 'rollup-plugin-typescript';
|
import typescript from 'rollup-plugin-typescript';
|
||||||
import string from 'rollup-plugin-string';
|
import string from 'rollup-plugin-string';
|
||||||
import json from 'rollup-plugin-json';
|
import json from 'rollup-plugin-json';
|
||||||
|
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';
|
||||||
|
|
||||||
@@ -19,7 +20,8 @@ export default [
|
|||||||
},
|
},
|
||||||
plugins: [
|
plugins: [
|
||||||
typescript({
|
typescript({
|
||||||
typescript: require('typescript')
|
typescript: require('typescript'),
|
||||||
|
target: "ES2017"
|
||||||
})
|
})
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
@@ -30,6 +32,7 @@ export default [
|
|||||||
`src/cli.ts`,
|
`src/cli.ts`,
|
||||||
`src/core.ts`,
|
`src/core.ts`,
|
||||||
`src/middleware.ts`,
|
`src/middleware.ts`,
|
||||||
|
`src/rollup.ts`,
|
||||||
`src/webpack.ts`
|
`src/webpack.ts`
|
||||||
],
|
],
|
||||||
output: {
|
output: {
|
||||||
@@ -43,12 +46,12 @@ export default [
|
|||||||
include: '**/*.md'
|
include: '**/*.md'
|
||||||
}),
|
}),
|
||||||
json(),
|
json(),
|
||||||
|
resolve(),
|
||||||
commonjs(),
|
commonjs(),
|
||||||
typescript({
|
typescript({
|
||||||
typescript: require('typescript')
|
typescript: require('typescript')
|
||||||
})
|
})
|
||||||
],
|
],
|
||||||
experimentalCodeSplitting: true,
|
experimentalCodeSplitting: true
|
||||||
experimentalDynamicImport: true
|
|
||||||
}
|
}
|
||||||
];
|
];
|
||||||
2
sapper
2
sapper
@@ -1,2 +1,2 @@
|
|||||||
#!/usr/bin/env node
|
#!/usr/bin/env node
|
||||||
require('./dist/cli.ts.js');
|
require('./dist/cli.js');
|
||||||
@@ -1,6 +1,8 @@
|
|||||||
let source;
|
let source;
|
||||||
|
|
||||||
function check() {
|
function check() {
|
||||||
|
if (typeof module === 'undefined') return;
|
||||||
|
|
||||||
if (module.hot.status() === 'idle') {
|
if (module.hot.status() === 'idle') {
|
||||||
module.hot.check(true).then(modules => {
|
module.hot.check(true).then(modules => {
|
||||||
console.log(`[SAPPER] applied HMR update`);
|
console.log(`[SAPPER] applied HMR update`);
|
||||||
|
|||||||
@@ -3,10 +3,12 @@ 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 { minify_html } from './utils/minify_html';
|
import minify_html from './utils/minify_html';
|
||||||
import { create_compilers, create_main_manifests, create_routes, create_serviceworker_manifest } from '../core'
|
import { create_compilers, create_main_manifests, create_routes, create_serviceworker_manifest } from '../core';
|
||||||
import { locations } from '../config';
|
import { Compilers, Compiler } from '../core/create_compilers';
|
||||||
import * as events from './interfaces';
|
import * as events from './interfaces';
|
||||||
|
import validate_bundler from '../cli/utils/validate_bundler';
|
||||||
|
import { copy_shimport } from './utils/copy_shimport';
|
||||||
|
|
||||||
export function build(opts: {}) {
|
export function build(opts: {}) {
|
||||||
const emitter = new EventEmitter();
|
const emitter = new EventEmitter();
|
||||||
@@ -28,11 +30,14 @@ export function build(opts: {}) {
|
|||||||
async function execute(emitter: EventEmitter, {
|
async function execute(emitter: EventEmitter, {
|
||||||
dest = 'build',
|
dest = 'build',
|
||||||
app = 'app',
|
app = 'app',
|
||||||
|
bundler,
|
||||||
webpack = 'webpack',
|
webpack = 'webpack',
|
||||||
|
rollup = 'rollup',
|
||||||
routes = 'routes'
|
routes = 'routes'
|
||||||
} = {}) {
|
} = {}) {
|
||||||
mkdirp.sync(dest);
|
|
||||||
rimraf.sync(path.join(dest, '**/*'));
|
rimraf.sync(path.join(dest, '**/*'));
|
||||||
|
mkdirp.sync(`${dest}/client`);
|
||||||
|
copy_shimport(dest);
|
||||||
|
|
||||||
// minify app/template.html
|
// minify app/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(...)
|
||||||
@@ -52,24 +57,26 @@ async function execute(emitter: EventEmitter, {
|
|||||||
// create app/manifest/client.js and app/manifest/server.js
|
// create app/manifest/client.js and app/manifest/server.js
|
||||||
create_main_manifests({ routes: route_objects });
|
create_main_manifests({ routes: route_objects });
|
||||||
|
|
||||||
const { client, server, serviceworker } = create_compilers({ webpack });
|
const { client, server, serviceworker } = create_compilers(validate_bundler(bundler), { webpack, rollup });
|
||||||
|
|
||||||
const client_stats = await compile(client);
|
const client_result = await client.compile();
|
||||||
emitter.emit('build', <events.BuildEvent>{
|
emitter.emit('build', <events.BuildEvent>{
|
||||||
type: 'client',
|
type: 'client',
|
||||||
// TODO duration/warnings
|
// TODO duration/warnings
|
||||||
webpack_stats: client_stats
|
result: client_result
|
||||||
});
|
});
|
||||||
|
|
||||||
const client_info = client_stats.toJson();
|
fs.writeFileSync(path.join(dest, 'build.json'), JSON.stringify({
|
||||||
fs.writeFileSync(path.join(dest, 'client_info.json'), JSON.stringify(client_info));
|
bundler,
|
||||||
fs.writeFileSync(path.join(dest, 'client_assets.json'), JSON.stringify(client_info.assetsByChunkName));
|
shimport: bundler === 'rollup' && require('shimport/package.json').version,
|
||||||
|
assets: client_result.assetsByChunkName
|
||||||
|
}));
|
||||||
|
|
||||||
const server_stats = await compile(server);
|
const server_stats = await server.compile();
|
||||||
emitter.emit('build', <events.BuildEvent>{
|
emitter.emit('build', <events.BuildEvent>{
|
||||||
type: 'server',
|
type: 'server',
|
||||||
// TODO duration/warnings
|
// TODO duration/warnings
|
||||||
webpack_stats: server_stats
|
result: server_stats
|
||||||
});
|
});
|
||||||
|
|
||||||
let serviceworker_stats;
|
let serviceworker_stats;
|
||||||
@@ -77,35 +84,15 @@ async function execute(emitter: EventEmitter, {
|
|||||||
if (serviceworker) {
|
if (serviceworker) {
|
||||||
create_serviceworker_manifest({
|
create_serviceworker_manifest({
|
||||||
routes: route_objects,
|
routes: route_objects,
|
||||||
client_files: client_stats.toJson().assets.map((chunk: { name: string }) => `client/${chunk.name}`)
|
client_files: client_result.assets.map((file: string) => `client/${file}`)
|
||||||
});
|
});
|
||||||
|
|
||||||
serviceworker_stats = await compile(serviceworker);
|
serviceworker_stats = await serviceworker.compile();
|
||||||
|
|
||||||
emitter.emit('build', <events.BuildEvent>{
|
emitter.emit('build', <events.BuildEvent>{
|
||||||
type: 'serviceworker',
|
type: 'serviceworker',
|
||||||
// TODO duration/warnings
|
// TODO duration/warnings
|
||||||
webpack_stats: serviceworker_stats
|
result: serviceworker_stats
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function compile(compiler: any) {
|
|
||||||
return new Promise((fulfil, reject) => {
|
|
||||||
compiler.run((err: Error, stats: any) => {
|
|
||||||
if (err) {
|
|
||||||
reject(err);
|
|
||||||
process.exit(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (stats.hasErrors()) {
|
|
||||||
console.error(stats.toString({ colors: true }));
|
|
||||||
reject(new Error(`Encountered errors while building app`));
|
|
||||||
}
|
|
||||||
|
|
||||||
else {
|
|
||||||
fulfil(stats);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
|
||||||
271
src/api/dev.ts
271
src/api/dev.ts
@@ -5,34 +5,39 @@ import * as child_process from 'child_process';
|
|||||||
import * as ports from 'port-authority';
|
import * as ports from 'port-authority';
|
||||||
import mkdirp from 'mkdirp';
|
import mkdirp from 'mkdirp';
|
||||||
import rimraf from 'rimraf';
|
import rimraf from 'rimraf';
|
||||||
import format_messages from 'webpack-format-messages';
|
|
||||||
import { locations } from '../config';
|
import { locations } from '../config';
|
||||||
import { EventEmitter } from 'events';
|
import { EventEmitter } from 'events';
|
||||||
import { create_routes, create_main_manifests, create_compilers, create_serviceworker_manifest } from '../core';
|
import { create_routes, create_main_manifests, create_compilers, create_serviceworker_manifest } from '../core';
|
||||||
|
import { Compiler, Compilers, CompileResult, CompileError } from '../core/create_compilers';
|
||||||
|
import Deferred from './utils/Deferred';
|
||||||
import * as events from './interfaces';
|
import * as events from './interfaces';
|
||||||
|
import validate_bundler from '../cli/utils/validate_bundler';
|
||||||
|
import { copy_shimport } from './utils/copy_shimport';
|
||||||
|
|
||||||
export function dev(opts) {
|
export function dev(opts) {
|
||||||
return new Watcher(opts);
|
return new Watcher(opts);
|
||||||
}
|
}
|
||||||
|
|
||||||
class Watcher extends EventEmitter {
|
class Watcher extends EventEmitter {
|
||||||
|
bundler: string;
|
||||||
dirs: {
|
dirs: {
|
||||||
app: string;
|
app: string;
|
||||||
dest: string;
|
dest: string;
|
||||||
routes: string;
|
routes: string;
|
||||||
|
rollup: string;
|
||||||
webpack: string;
|
webpack: string;
|
||||||
}
|
}
|
||||||
port: number;
|
port: number;
|
||||||
closed: boolean;
|
closed: boolean;
|
||||||
|
live: boolean;
|
||||||
|
hot: boolean;
|
||||||
|
|
||||||
dev_server: DevServer;
|
dev_server: DevServer;
|
||||||
proc: child_process.ChildProcess;
|
proc: child_process.ChildProcess;
|
||||||
filewatchers: Array<{ close: () => void }>;
|
filewatchers: Array<{ close: () => void }>;
|
||||||
deferreds: {
|
deferred: Deferred;
|
||||||
client: Deferred;
|
|
||||||
server: Deferred;
|
|
||||||
};
|
|
||||||
|
|
||||||
|
crashed: boolean;
|
||||||
restarting: boolean;
|
restarting: boolean;
|
||||||
current_build: {
|
current_build: {
|
||||||
changed: Set<string>;
|
changed: Set<string>;
|
||||||
@@ -45,21 +50,33 @@ class Watcher extends EventEmitter {
|
|||||||
app = locations.app(),
|
app = locations.app(),
|
||||||
dest = locations.dest(),
|
dest = locations.dest(),
|
||||||
routes = locations.routes(),
|
routes = locations.routes(),
|
||||||
|
live,
|
||||||
|
hot,
|
||||||
|
bundler,
|
||||||
webpack = 'webpack',
|
webpack = 'webpack',
|
||||||
|
rollup = 'rollup',
|
||||||
port = +process.env.PORT
|
port = +process.env.PORT
|
||||||
}: {
|
}: {
|
||||||
app: string,
|
app: string,
|
||||||
dest: string,
|
dest: string,
|
||||||
routes: string,
|
routes: string,
|
||||||
|
live: boolean,
|
||||||
|
hot: boolean,
|
||||||
|
bundler?: string,
|
||||||
webpack: string,
|
webpack: string,
|
||||||
|
rollup: string,
|
||||||
port: number
|
port: number
|
||||||
}) {
|
}) {
|
||||||
super();
|
super();
|
||||||
|
|
||||||
this.dirs = { app, dest, routes, webpack };
|
this.bundler = validate_bundler(bundler);
|
||||||
|
this.dirs = { app, dest, routes, webpack, rollup };
|
||||||
this.port = port;
|
this.port = port;
|
||||||
this.closed = false;
|
this.closed = false;
|
||||||
|
|
||||||
|
this.live = live;
|
||||||
|
this.hot = hot;
|
||||||
|
|
||||||
this.filewatchers = [];
|
this.filewatchers = [];
|
||||||
|
|
||||||
this.current_build = {
|
this.current_build = {
|
||||||
@@ -100,40 +117,72 @@ class Watcher extends EventEmitter {
|
|||||||
|
|
||||||
const { dest } = this.dirs;
|
const { dest } = this.dirs;
|
||||||
rimraf.sync(dest);
|
rimraf.sync(dest);
|
||||||
mkdirp.sync(dest);
|
mkdirp.sync(`${dest}/client`);
|
||||||
|
if (this.bundler === 'rollup') copy_shimport(dest);
|
||||||
|
|
||||||
const dev_port = await ports.find(10000);
|
const dev_port = await ports.find(10000);
|
||||||
|
|
||||||
const routes = create_routes();
|
try {
|
||||||
create_main_manifests({ routes, dev_port });
|
const routes = create_routes();
|
||||||
|
create_main_manifests({ routes, dev_port });
|
||||||
|
} catch (err) {
|
||||||
|
this.emit('fatal', <events.FatalEvent>{
|
||||||
|
message: err.message
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
this.dev_server = new DevServer(dev_port);
|
this.dev_server = new DevServer(dev_port);
|
||||||
|
|
||||||
this.filewatchers.push(
|
this.filewatchers.push(
|
||||||
watch_files(locations.routes(), ['add', 'unlink'], () => {
|
watch_dir(
|
||||||
const routes = create_routes();
|
locations.routes(),
|
||||||
create_main_manifests({ routes, dev_port });
|
({ path: file, stats }) => {
|
||||||
}),
|
if (stats.isDirectory()) {
|
||||||
|
return path.basename(file)[0] !== '_';
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
},
|
||||||
|
() => {
|
||||||
|
const routes = create_routes();
|
||||||
|
create_main_manifests({ routes, dev_port });
|
||||||
|
|
||||||
watch_files(`${locations.app()}/template.html`, ['change'], () => {
|
try {
|
||||||
|
const routes = create_routes();
|
||||||
|
create_main_manifests({ routes, dev_port });
|
||||||
|
} catch (err) {
|
||||||
|
this.emit('error', <events.ErrorEvent>{
|
||||||
|
message: err.message
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
),
|
||||||
|
|
||||||
|
fs.watch(`${locations.app()}/template.html`, () => {
|
||||||
this.dev_server.send({
|
this.dev_server.send({
|
||||||
action: 'reload'
|
action: 'reload'
|
||||||
});
|
});
|
||||||
})
|
})
|
||||||
);
|
);
|
||||||
|
|
||||||
this.deferreds = {
|
let deferred = new Deferred();
|
||||||
server: new Deferred(),
|
|
||||||
client: new Deferred()
|
|
||||||
};
|
|
||||||
|
|
||||||
// TODO watch the configs themselves?
|
// TODO watch the configs themselves?
|
||||||
const compilers = create_compilers({ webpack: this.dirs.webpack });
|
const compilers: Compilers = create_compilers(this.bundler, {
|
||||||
|
webpack: this.dirs.webpack,
|
||||||
|
rollup: this.dirs.rollup
|
||||||
|
});
|
||||||
|
|
||||||
|
let log = '';
|
||||||
|
|
||||||
const emitFatal = () => {
|
const emitFatal = () => {
|
||||||
this.emit('fatal', <events.FatalEvent>{
|
this.emit('fatal', <events.FatalEvent>{
|
||||||
message: `Server crashed`
|
message: `Server crashed`,
|
||||||
|
log
|
||||||
});
|
});
|
||||||
|
|
||||||
|
this.crashed = true;
|
||||||
|
this.proc = null;
|
||||||
};
|
};
|
||||||
|
|
||||||
this.watch(compilers.server, {
|
this.watch(compilers.server, {
|
||||||
@@ -141,26 +190,39 @@ class Watcher extends EventEmitter {
|
|||||||
|
|
||||||
invalid: filename => {
|
invalid: filename => {
|
||||||
this.restart(filename, 'server');
|
this.restart(filename, 'server');
|
||||||
this.deferreds.server = new Deferred();
|
|
||||||
},
|
},
|
||||||
|
|
||||||
result: info => {
|
handle_result: (result: CompileResult) => {
|
||||||
fs.writeFileSync(path.join(dest, 'server_info.json'), JSON.stringify(info, null, ' '));
|
deferred.promise.then(() => {
|
||||||
|
|
||||||
this.deferreds.client.promise.then(() => {
|
|
||||||
const restart = () => {
|
const restart = () => {
|
||||||
ports.wait(this.port).then((() => {
|
log = '';
|
||||||
this.emit('ready', <events.ReadyEvent>{
|
this.crashed = false;
|
||||||
port: this.port,
|
|
||||||
process: this.proc
|
|
||||||
});
|
|
||||||
|
|
||||||
this.deferreds.server.fulfil();
|
ports.wait(this.port)
|
||||||
|
.then((() => {
|
||||||
|
this.emit('ready', <events.ReadyEvent>{
|
||||||
|
port: this.port,
|
||||||
|
process: this.proc
|
||||||
|
});
|
||||||
|
|
||||||
this.dev_server.send({
|
if (this.hot && this.bundler === 'webpack') {
|
||||||
status: 'completed'
|
this.dev_server.send({
|
||||||
|
status: 'completed'
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
this.dev_server.send({
|
||||||
|
action: 'reload'
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}))
|
||||||
|
.catch(err => {
|
||||||
|
if (this.crashed) return;
|
||||||
|
|
||||||
|
this.emit('fatal', <events.FatalEvent>{
|
||||||
|
message: `Server is not listening on port ${this.port}`,
|
||||||
|
log
|
||||||
|
});
|
||||||
});
|
});
|
||||||
}));
|
|
||||||
};
|
};
|
||||||
|
|
||||||
if (this.proc) {
|
if (this.proc) {
|
||||||
@@ -180,10 +242,12 @@ class Watcher extends EventEmitter {
|
|||||||
});
|
});
|
||||||
|
|
||||||
this.proc.stdout.on('data', chunk => {
|
this.proc.stdout.on('data', chunk => {
|
||||||
|
log += chunk;
|
||||||
this.emit('stdout', chunk);
|
this.emit('stdout', chunk);
|
||||||
});
|
});
|
||||||
|
|
||||||
this.proc.stderr.on('data', chunk => {
|
this.proc.stderr.on('data', chunk => {
|
||||||
|
log += chunk;
|
||||||
this.emit('stderr', chunk);
|
this.emit('stderr', chunk);
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -200,32 +264,34 @@ class Watcher extends EventEmitter {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
let first = true;
|
|
||||||
|
|
||||||
this.watch(compilers.client, {
|
this.watch(compilers.client, {
|
||||||
name: 'client',
|
name: 'client',
|
||||||
|
|
||||||
invalid: filename => {
|
invalid: filename => {
|
||||||
this.restart(filename, 'client');
|
this.restart(filename, 'client');
|
||||||
this.deferreds.client = new Deferred();
|
deferred = new Deferred();
|
||||||
|
|
||||||
// TODO we should delete old assets. due to a webpack bug
|
// TODO we should delete old assets. due to a webpack bug
|
||||||
// i don't even begin to comprehend, this is apparently
|
// i don't even begin to comprehend, this is apparently
|
||||||
// quite difficult
|
// quite difficult
|
||||||
},
|
},
|
||||||
|
|
||||||
result: info => {
|
handle_result: (result: CompileResult) => {
|
||||||
fs.writeFileSync(path.join(dest, 'client_info.json'), JSON.stringify(info));
|
fs.writeFileSync(path.join(dest, 'build.json'), JSON.stringify({
|
||||||
fs.writeFileSync(path.join(dest, 'client_assets.json'), JSON.stringify(info.assetsByChunkName, null, ' '));
|
bundler: this.bundler,
|
||||||
this.deferreds.client.fulfil();
|
shimport: this.bundler === 'rollup' && require('shimport/package.json').version,
|
||||||
|
assets: result.assetsByChunkName
|
||||||
|
}, null, ' '));
|
||||||
|
|
||||||
const client_files = info.assets.map((chunk: { name: string }) => `client/${chunk.name}`);
|
const client_files = result.assets.map((file: string) => `client/${file}`);
|
||||||
|
|
||||||
create_serviceworker_manifest({
|
create_serviceworker_manifest({
|
||||||
routes: create_routes(),
|
routes: create_routes(),
|
||||||
client_files
|
client_files
|
||||||
});
|
});
|
||||||
|
|
||||||
|
deferred.fulfil();
|
||||||
|
|
||||||
// we need to wait a beat before watching the service
|
// we need to wait a beat before watching the service
|
||||||
// worker, because of some webpack nonsense
|
// worker, because of some webpack nonsense
|
||||||
setTimeout(watch_serviceworker, 100);
|
setTimeout(watch_serviceworker, 100);
|
||||||
@@ -237,11 +303,7 @@ class Watcher extends EventEmitter {
|
|||||||
watch_serviceworker = noop;
|
watch_serviceworker = noop;
|
||||||
|
|
||||||
this.watch(compilers.serviceworker, {
|
this.watch(compilers.serviceworker, {
|
||||||
name: 'service worker',
|
name: 'service worker'
|
||||||
|
|
||||||
result: info => {
|
|
||||||
fs.writeFileSync(path.join(dest, 'serviceworker_info.json'), JSON.stringify(info, null, ' '));
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
: noop;
|
: noop;
|
||||||
@@ -251,7 +313,7 @@ class Watcher extends EventEmitter {
|
|||||||
if (this.closed) return;
|
if (this.closed) return;
|
||||||
this.closed = true;
|
this.closed = true;
|
||||||
|
|
||||||
this.dev_server.close();
|
if (this.dev_server) this.dev_server.close();
|
||||||
|
|
||||||
if (this.proc) this.proc.kill();
|
if (this.proc) this.proc.kill();
|
||||||
this.filewatchers.forEach(watcher => {
|
this.filewatchers.forEach(watcher => {
|
||||||
@@ -288,95 +350,34 @@ class Watcher extends EventEmitter {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
watch(compiler: any, { name, invalid = noop, result }: {
|
watch(compiler: Compiler, { name, invalid = noop, handle_result = noop }: {
|
||||||
name: string,
|
name: string,
|
||||||
invalid?: (filename: string) => void;
|
invalid?: (filename: string) => void;
|
||||||
result: (stats: any) => void;
|
handle_result?: (result: CompileResult) => void;
|
||||||
}) {
|
}) {
|
||||||
compiler.hooks.invalid.tap('sapper', (filename: string) => {
|
compiler.oninvalid(invalid);
|
||||||
invalid(filename);
|
|
||||||
});
|
|
||||||
|
|
||||||
compiler.watch({}, (err: Error, stats: any) => {
|
compiler.watch((err?: Error, result?: CompileResult) => {
|
||||||
if (err) {
|
if (err) {
|
||||||
this.emit('error', <events.ErrorEvent>{
|
this.emit('error', <events.ErrorEvent>{
|
||||||
type: name,
|
type: name,
|
||||||
error: err
|
message: err.message
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
const messages = format_messages(stats);
|
|
||||||
const info = stats.toJson();
|
|
||||||
|
|
||||||
this.emit('build', {
|
this.emit('build', {
|
||||||
type: name,
|
type: name,
|
||||||
|
|
||||||
duration: info.time,
|
duration: result.duration,
|
||||||
|
errors: result.errors,
|
||||||
errors: messages.errors.map((message: string) => {
|
warnings: result.warnings
|
||||||
const duplicate = this.current_build.unique_errors.has(message);
|
|
||||||
this.current_build.unique_errors.add(message);
|
|
||||||
|
|
||||||
return mungeWebpackError(message, duplicate);
|
|
||||||
}),
|
|
||||||
|
|
||||||
warnings: messages.warnings.map((message: string) => {
|
|
||||||
const duplicate = this.current_build.unique_warnings.has(message);
|
|
||||||
this.current_build.unique_warnings.add(message);
|
|
||||||
|
|
||||||
return mungeWebpackError(message, duplicate);
|
|
||||||
}),
|
|
||||||
});
|
});
|
||||||
|
|
||||||
result(info);
|
handle_result(result);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const locPattern = /\((\d+):(\d+)\)$/;
|
|
||||||
|
|
||||||
function mungeWebpackError(message: string, duplicate: boolean) {
|
|
||||||
// TODO this is all a bit rube goldberg...
|
|
||||||
const lines = message.split('\n');
|
|
||||||
|
|
||||||
const file = lines.shift()
|
|
||||||
.replace('[7m', '') // careful — there is a special character at the beginning of this string
|
|
||||||
.replace('[27m', '')
|
|
||||||
.replace('./', '');
|
|
||||||
|
|
||||||
let line = null;
|
|
||||||
let column = null;
|
|
||||||
|
|
||||||
const match = locPattern.exec(lines[0]);
|
|
||||||
if (match) {
|
|
||||||
lines[0] = lines[0].replace(locPattern, '');
|
|
||||||
line = +match[1];
|
|
||||||
column = +match[2];
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
|
||||||
file,
|
|
||||||
line,
|
|
||||||
column,
|
|
||||||
message: lines.join('\n'),
|
|
||||||
originalMessage: message,
|
|
||||||
duplicate
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
class Deferred {
|
|
||||||
promise: Promise<any>;
|
|
||||||
fulfil: (value?: any) => void;
|
|
||||||
reject: (error: Error) => void;
|
|
||||||
|
|
||||||
constructor() {
|
|
||||||
this.promise = new Promise((fulfil, reject) => {
|
|
||||||
this.fulfil = fulfil;
|
|
||||||
this.reject = reject;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const INTERVAL = 10000;
|
const INTERVAL = 10000;
|
||||||
|
|
||||||
class DevServer {
|
class DevServer {
|
||||||
@@ -431,20 +432,32 @@ class DevServer {
|
|||||||
|
|
||||||
function noop() {}
|
function noop() {}
|
||||||
|
|
||||||
function watch_files(pattern: string, events: string[], callback: () => void) {
|
function watch_dir(
|
||||||
const chokidar = require('chokidar');
|
dir: string,
|
||||||
|
filter: ({ path, stats }: { path: string, stats: fs.Stats }) => boolean,
|
||||||
|
callback: () => void
|
||||||
|
) {
|
||||||
|
let watch;
|
||||||
|
let closed = false;
|
||||||
|
|
||||||
const watcher = chokidar.watch(pattern, {
|
import('cheap-watch').then(CheapWatch => {
|
||||||
persistent: true,
|
if (closed) return;
|
||||||
ignoreInitial: true,
|
|
||||||
disableGlobbing: true
|
|
||||||
});
|
|
||||||
|
|
||||||
events.forEach(event => {
|
watch = new CheapWatch({ dir, filter, debounce: 50 });
|
||||||
watcher.on(event, callback);
|
|
||||||
|
watch.on('+', ({ isNew }) => {
|
||||||
|
if (isNew) callback();
|
||||||
|
});
|
||||||
|
|
||||||
|
watch.on('-', callback);
|
||||||
|
|
||||||
|
watch.init();
|
||||||
});
|
});
|
||||||
|
|
||||||
return {
|
return {
|
||||||
close: () => watcher.close()
|
close: () => {
|
||||||
|
if (watch) watch.close();
|
||||||
|
closed = true;
|
||||||
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
@@ -1,16 +1,23 @@
|
|||||||
import * as child_process from 'child_process';
|
import * as child_process from 'child_process';
|
||||||
import * as path from 'path';
|
import * as path from 'path';
|
||||||
import * as sander from 'sander';
|
import * as sander from 'sander';
|
||||||
import cheerio from 'cheerio';
|
|
||||||
import URL from 'url-parse';
|
import URL from 'url-parse';
|
||||||
import fetch from 'node-fetch';
|
import fetch from 'node-fetch';
|
||||||
import * as ports from 'port-authority';
|
import * as ports from 'port-authority';
|
||||||
import { EventEmitter } from 'events';
|
import { EventEmitter } from 'events';
|
||||||
import { minify_html } from './utils/minify_html';
|
import clean_html from './utils/clean_html';
|
||||||
import { locations } from '../config';
|
import minify_html from './utils/minify_html';
|
||||||
|
import Deferred from './utils/Deferred';
|
||||||
import * as events from './interfaces';
|
import * as events from './interfaces';
|
||||||
|
|
||||||
export function exporter(opts: {}) {
|
type Opts = {
|
||||||
|
build: string,
|
||||||
|
dest: string,
|
||||||
|
basepath?: string,
|
||||||
|
timeout: number | false
|
||||||
|
};
|
||||||
|
|
||||||
|
export function exporter(opts: Opts) {
|
||||||
const emitter = new EventEmitter();
|
const emitter = new EventEmitter();
|
||||||
|
|
||||||
execute(emitter, opts).then(
|
execute(emitter, opts).then(
|
||||||
@@ -27,48 +34,61 @@ export function exporter(opts: {}) {
|
|||||||
return emitter;
|
return emitter;
|
||||||
}
|
}
|
||||||
|
|
||||||
async function execute(emitter: EventEmitter, {
|
async function execute(emitter: EventEmitter, opts: Opts) {
|
||||||
build = 'build',
|
const export_dir = path.join(opts.dest, opts.basepath);
|
||||||
dest = 'export',
|
|
||||||
basepath = ''
|
|
||||||
} = {}) {
|
|
||||||
const export_dir = path.join(dest, basepath);
|
|
||||||
|
|
||||||
// Prep output directory
|
// Prep output directory
|
||||||
sander.rimrafSync(export_dir);
|
sander.rimrafSync(export_dir);
|
||||||
|
|
||||||
sander.copydirSync('assets').to(export_dir);
|
sander.copydirSync('assets').to(export_dir);
|
||||||
sander.copydirSync(build, 'client').to(export_dir, 'client');
|
sander.copydirSync(opts.build, 'client').to(export_dir, 'client');
|
||||||
|
|
||||||
if (sander.existsSync(build, 'service-worker.js')) {
|
if (sander.existsSync(opts.build, 'service-worker.js')) {
|
||||||
sander.copyFileSync(build, 'service-worker.js').to(export_dir, 'service-worker.js');
|
sander.copyFileSync(opts.build, 'service-worker.js').to(export_dir, 'service-worker.js');
|
||||||
}
|
}
|
||||||
|
|
||||||
if (sander.existsSync(build, 'service-worker.js.map')) {
|
if (sander.existsSync(opts.build, 'service-worker.js.map')) {
|
||||||
sander.copyFileSync(build, 'service-worker.js.map').to(export_dir, 'service-worker.js.map');
|
sander.copyFileSync(opts.build, 'service-worker.js.map').to(export_dir, 'service-worker.js.map');
|
||||||
}
|
}
|
||||||
|
|
||||||
const port = await ports.find(3000);
|
const port = await ports.find(3000);
|
||||||
|
|
||||||
const origin = `http://localhost:${port}`;
|
const origin = `http://localhost:${port}`;
|
||||||
|
const root = new URL(opts.basepath || '', origin);
|
||||||
|
|
||||||
const proc = child_process.fork(path.resolve(`${build}/server.js`), [], {
|
emitter.emit('info', {
|
||||||
|
message: `Crawling ${root.href}`
|
||||||
|
});
|
||||||
|
|
||||||
|
const proc = child_process.fork(path.resolve(`${opts.build}/server.js`), [], {
|
||||||
cwd: process.cwd(),
|
cwd: process.cwd(),
|
||||||
env: Object.assign({
|
env: Object.assign({
|
||||||
PORT: port,
|
PORT: port,
|
||||||
NODE_ENV: 'production',
|
NODE_ENV: 'production',
|
||||||
SAPPER_DEST: build,
|
SAPPER_DEST: opts.build,
|
||||||
SAPPER_EXPORT: 'true'
|
SAPPER_EXPORT: 'true'
|
||||||
}, process.env)
|
}, process.env)
|
||||||
});
|
});
|
||||||
|
|
||||||
const seen = new Set();
|
const seen = new Set();
|
||||||
const saved = new Set();
|
const saved = new Set();
|
||||||
|
const deferreds = new Map();
|
||||||
|
|
||||||
|
function get_deferred(pathname: string) {
|
||||||
|
pathname = pathname.replace(root.pathname, '');
|
||||||
|
|
||||||
|
if (!deferreds.has(pathname)) {
|
||||||
|
deferreds.set(pathname, new Deferred());
|
||||||
|
}
|
||||||
|
|
||||||
|
return deferreds.get(pathname);
|
||||||
|
}
|
||||||
|
|
||||||
proc.on('message', message => {
|
proc.on('message', message => {
|
||||||
if (!message.__sapper__ || message.event !== 'file') return;
|
if (!message.__sapper__ || message.event !== 'file') return;
|
||||||
|
|
||||||
let file = new URL(message.url, origin).pathname.slice(1);
|
const pathname = new URL(message.url, origin).pathname;
|
||||||
|
let file = pathname.slice(1);
|
||||||
let { body } = message;
|
let { body } = message;
|
||||||
|
|
||||||
if (saved.has(file)) return;
|
if (saved.has(file)) return;
|
||||||
@@ -83,49 +103,78 @@ async function execute(emitter: EventEmitter, {
|
|||||||
|
|
||||||
emitter.emit('file', <events.FileEvent>{
|
emitter.emit('file', <events.FileEvent>{
|
||||||
file,
|
file,
|
||||||
size: body.length
|
size: body.length,
|
||||||
|
status: message.status
|
||||||
});
|
});
|
||||||
|
|
||||||
sander.writeFileSync(export_dir, file, body);
|
sander.writeFileSync(export_dir, file, body);
|
||||||
|
|
||||||
|
get_deferred(pathname).fulfil();
|
||||||
});
|
});
|
||||||
|
|
||||||
async function handle(url: URL) {
|
async function handle(url: URL) {
|
||||||
const r = await fetch(url.href);
|
const pathname = (url.pathname.replace(root.pathname, '') || '/');
|
||||||
const range = ~~(r.status / 100);
|
|
||||||
|
|
||||||
if (range >= 4) {
|
if (seen.has(pathname)) return;
|
||||||
emitter.emit('failure', <events.FailureEvent>{
|
seen.add(pathname);
|
||||||
status: r.status,
|
|
||||||
pathname: url.pathname
|
const deferred = get_deferred(pathname);
|
||||||
});
|
|
||||||
return;
|
const timeout_deferred = new Deferred();
|
||||||
}
|
const timeout = setTimeout(() => {
|
||||||
|
timeout_deferred.reject(new Error(`Timed out waiting for ${url.href}`));
|
||||||
|
}, opts.timeout);
|
||||||
|
|
||||||
|
const r = await Promise.race([
|
||||||
|
fetch(url.href),
|
||||||
|
timeout_deferred.promise
|
||||||
|
]);
|
||||||
|
|
||||||
|
clearTimeout(timeout); // prevent it hanging at the end
|
||||||
|
|
||||||
|
const range = ~~(r.status / 100);
|
||||||
|
|
||||||
if (range === 2) {
|
if (range === 2) {
|
||||||
if (r.headers.get('Content-Type') === 'text/html') {
|
if (r.headers.get('Content-Type') === 'text/html') {
|
||||||
const body = await r.text();
|
const body = await r.text();
|
||||||
const $ = cheerio.load(body);
|
|
||||||
const urls: URL[] = [];
|
const urls: URL[] = [];
|
||||||
|
|
||||||
const base = new URL($('base').attr('href') || '/', url.href);
|
const cleaned = clean_html(body);
|
||||||
|
|
||||||
$('a[href]').each((i: number, $a) => {
|
const base_match = /<base ([\s\S]+?)>/m.exec(cleaned);
|
||||||
const url = new URL($a.attribs.href, base.href);
|
const base_href = base_match && get_href(base_match[1]);
|
||||||
|
const base = new URL(base_href || '/', url.href);
|
||||||
|
|
||||||
if (url.origin === origin && !seen.has(url.pathname)) {
|
let match;
|
||||||
seen.add(url.pathname);
|
let pattern = /<a ([\s\S]+?)>/gm;
|
||||||
urls.push(url);
|
|
||||||
|
while (match = pattern.exec(cleaned)) {
|
||||||
|
const attrs = match[1];
|
||||||
|
const href = get_href(attrs);
|
||||||
|
|
||||||
|
if (href) {
|
||||||
|
const url = new URL(href, base.href);
|
||||||
|
if (url.origin === origin) urls.push(url);
|
||||||
}
|
}
|
||||||
});
|
|
||||||
|
|
||||||
for (const url of urls) {
|
|
||||||
await handle(url);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
await Promise.all(urls.map(handle));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
await deferred.promise;
|
||||||
}
|
}
|
||||||
|
|
||||||
return ports.wait(port)
|
return ports.wait(port)
|
||||||
.then(() => handle(new URL(`/${basepath}`, origin))) // TODO all static routes
|
.then(() => handle(root))
|
||||||
.then(() => proc.kill());
|
.then(() => proc.kill())
|
||||||
|
.catch(err => {
|
||||||
|
proc.kill();
|
||||||
|
throw err;
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function get_href(attrs: string) {
|
||||||
|
const match = /href\s*=\s*(?:"(.+?)"|'(.+?)'|([^\s>]+))/.exec(attrs);
|
||||||
|
return match[1] || match[2] || match[3];
|
||||||
|
}
|
||||||
@@ -1,16 +1,14 @@
|
|||||||
import * as glob from 'glob';
|
|
||||||
import { locations } from '../config';
|
import { locations } from '../config';
|
||||||
import { create_routes } from '../core';
|
import { create_routes } from '../core';
|
||||||
|
|
||||||
export function find_page(pathname: string, files: string[] = glob.sync('**/*.*', { cwd: locations.routes(), dot: true, nodir: true })) {
|
export function find_page(pathname: string, cwd = locations.routes()) {
|
||||||
const routes = create_routes({ files });
|
const { pages } = create_routes(cwd);
|
||||||
|
|
||||||
for (let i = 0; i < routes.length; i += 1) {
|
for (let i = 0; i < pages.length; i += 1) {
|
||||||
const route = routes[i];
|
const page = pages[i];
|
||||||
|
|
||||||
if (route.pattern.test(pathname)) {
|
if (page.pattern.test(pathname)) {
|
||||||
const page = route.handlers.find(handler => handler.type === 'page');
|
return page.parts[page.parts.length - 1].component.file;
|
||||||
if (page) return page.file;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -1,4 +1,5 @@
|
|||||||
import * as child_process from 'child_process';
|
import * as child_process from 'child_process';
|
||||||
|
import { CompileResult } from '../core/create_compilers';
|
||||||
|
|
||||||
export type ReadyEvent = {
|
export type ReadyEvent = {
|
||||||
port: number;
|
port: number;
|
||||||
@@ -12,6 +13,7 @@ export type ErrorEvent = {
|
|||||||
|
|
||||||
export type FatalEvent = {
|
export type FatalEvent = {
|
||||||
message: string;
|
message: string;
|
||||||
|
log?: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
export type InvalidEvent = {
|
export type InvalidEvent = {
|
||||||
@@ -25,10 +27,10 @@ export type InvalidEvent = {
|
|||||||
|
|
||||||
export type BuildEvent = {
|
export type BuildEvent = {
|
||||||
type: string;
|
type: string;
|
||||||
errors: Array<{ message: string, duplicate: boolean }>;
|
errors: Array<{ file: string, message: string, duplicate: boolean }>;
|
||||||
warnings: Array<{ message: string, duplicate: boolean }>;
|
warnings: Array<{ file: string, message: string, duplicate: boolean }>;
|
||||||
duration: number;
|
duration: number;
|
||||||
webpack_stats: any;
|
result: CompileResult;
|
||||||
}
|
}
|
||||||
|
|
||||||
export type FileEvent = {
|
export type FileEvent = {
|
||||||
|
|||||||
12
src/api/utils/Deferred.ts
Normal file
12
src/api/utils/Deferred.ts
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
export default class Deferred {
|
||||||
|
promise: Promise<any>;
|
||||||
|
fulfil: (value?: any) => void;
|
||||||
|
reject: (error: Error) => void;
|
||||||
|
|
||||||
|
constructor() {
|
||||||
|
this.promise = new Promise((fulfil, reject) => {
|
||||||
|
this.fulfil = fulfil;
|
||||||
|
this.reject = reject;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
7
src/api/utils/clean_html.ts
Normal file
7
src/api/utils/clean_html.ts
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
export default function clean_html(html: string) {
|
||||||
|
return html
|
||||||
|
.replace(/<!\[CDATA\[[\s\S]*?\]\]>/gm, '')
|
||||||
|
.replace(/(<script[\s\S]*?>)[\s\S]*?<\/script>/gm, '$1</' + 'script>')
|
||||||
|
.replace(/(<style[\s\S]*?>)[\s\S]*?<\/style>/gm, '$1</' + 'style>')
|
||||||
|
.replace(/<!--[\s\S]*?-->/gm, '');
|
||||||
|
}
|
||||||
9
src/api/utils/copy_shimport.ts
Normal file
9
src/api/utils/copy_shimport.ts
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
import * as fs from 'fs';
|
||||||
|
|
||||||
|
export function copy_shimport(dest: string) {
|
||||||
|
const shimport_version = require('shimport/package.json').version;
|
||||||
|
fs.writeFileSync(
|
||||||
|
`${dest}/client/shimport@${shimport_version}.js`,
|
||||||
|
fs.readFileSync(require.resolve('shimport/index.js'))
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
import { minify } from 'html-minifier';
|
import { minify } from 'html-minifier';
|
||||||
|
|
||||||
export function minify_html(html: string) {
|
export default function minify_html(html: string) {
|
||||||
return minify(html, {
|
return minify(html, {
|
||||||
collapseBooleanAttributes: true,
|
collapseBooleanAttributes: true,
|
||||||
collapseWhitespace: true,
|
collapseWhitespace: true,
|
||||||
@@ -8,7 +8,7 @@ export function minify_html(html: string) {
|
|||||||
decodeEntities: true,
|
decodeEntities: true,
|
||||||
html5: true,
|
html5: true,
|
||||||
minifyCSS: true,
|
minifyCSS: true,
|
||||||
minifyJS: true,
|
minifyJS: false,
|
||||||
removeAttributeQuotes: true,
|
removeAttributeQuotes: true,
|
||||||
removeComments: true,
|
removeComments: true,
|
||||||
removeOptionalTags: true,
|
removeOptionalTags: true,
|
||||||
|
|||||||
50
src/cli.ts
50
src/cli.ts
@@ -1,11 +1,8 @@
|
|||||||
import * as fs from 'fs';
|
import * as fs from 'fs';
|
||||||
import * as path from 'path';
|
import * as path from 'path';
|
||||||
import * as child_process from 'child_process';
|
|
||||||
import sade from 'sade';
|
import sade from 'sade';
|
||||||
import * as colors from 'ansi-colors';
|
import colors from 'kleur';
|
||||||
import prettyMs from 'pretty-ms';
|
import prettyMs from 'pretty-ms';
|
||||||
// import upgrade from './cli/upgrade';
|
|
||||||
import * as ports from 'port-authority';
|
|
||||||
import * as pkg from '../package.json';
|
import * as pkg from '../package.json';
|
||||||
|
|
||||||
const prog = sade('sapper').version(pkg.version);
|
const prog = sade('sapper').version(pkg.version);
|
||||||
@@ -14,7 +11,16 @@ prog.command('dev')
|
|||||||
.describe('Start a development server')
|
.describe('Start a development server')
|
||||||
.option('-p, --port', 'Specify a port')
|
.option('-p, --port', 'Specify a port')
|
||||||
.option('-o, --open', 'Open a browser window')
|
.option('-o, --open', 'Open a browser window')
|
||||||
.action(async (opts: { port: number, open: boolean }) => {
|
.option('--hot', 'Use hot module replacement (requires webpack)', true)
|
||||||
|
.option('-l --live', 'Reload on changes if not using --hot', true)
|
||||||
|
.option('--bundler', 'Specify a bundler (rollup or webpack)')
|
||||||
|
.action(async (opts: {
|
||||||
|
port: number,
|
||||||
|
open: boolean,
|
||||||
|
live: boolean,
|
||||||
|
hot: boolean,
|
||||||
|
bundler?: string
|
||||||
|
}) => {
|
||||||
const { dev } = await import('./cli/dev');
|
const { dev } = await import('./cli/dev');
|
||||||
dev(opts);
|
dev(opts);
|
||||||
});
|
});
|
||||||
@@ -22,8 +28,9 @@ prog.command('dev')
|
|||||||
prog.command('build [dest]')
|
prog.command('build [dest]')
|
||||||
.describe('Create a production-ready version of your app')
|
.describe('Create a production-ready version of your app')
|
||||||
.option('-p, --port', 'Default of process.env.PORT', '3000')
|
.option('-p, --port', 'Default of process.env.PORT', '3000')
|
||||||
|
.option('--bundler', 'Specify a bundler (rollup or webpack, blank for auto)')
|
||||||
.example(`build custom-dir -p 4567`)
|
.example(`build custom-dir -p 4567`)
|
||||||
.action(async (dest = 'build', opts: { port: string }) => {
|
.action(async (dest = 'build', opts: { port: string, bundler?: string }) => {
|
||||||
console.log(`> Building...`);
|
console.log(`> Building...`);
|
||||||
|
|
||||||
process.env.NODE_ENV = process.env.NODE_ENV || 'production';
|
process.env.NODE_ENV = process.env.NODE_ENV || 'production';
|
||||||
@@ -33,7 +40,7 @@ prog.command('build [dest]')
|
|||||||
|
|
||||||
try {
|
try {
|
||||||
const { build } = await import('./cli/build');
|
const { build } = await import('./cli/build');
|
||||||
await build();
|
await build(opts);
|
||||||
|
|
||||||
const launcher = path.resolve(dest, 'index.js');
|
const launcher = path.resolve(dest, 'index.js');
|
||||||
|
|
||||||
@@ -49,7 +56,7 @@ prog.command('build [dest]')
|
|||||||
|
|
||||||
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.`);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
console.error(err ? err.details || err.stack || err.message || err : 'Unknown error');
|
console.log(`${colors.bold.red(`> ${err.message}`)}`);
|
||||||
process.exit(1);
|
process.exit(1);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@@ -65,25 +72,36 @@ 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-dir', 'Specify a custom temporary build directory', '.sapper/prod')
|
||||||
.option('--basepath', 'Specify a base path')
|
.option('--basepath', 'Specify a base path')
|
||||||
.action(async (dest = 'export', opts: { basepath?: string }) => {
|
.option('--timeout', 'Milliseconds to wait for a page (--no-timeout to disable)', 5000)
|
||||||
console.log(`> Building...`);
|
.option('--bundler', 'Specify a bundler (rollup or webpack, blank for auto)')
|
||||||
|
.action(async (dest = 'export', opts: {
|
||||||
|
build: boolean,
|
||||||
|
bundler?: string,
|
||||||
|
'build-dir': string,
|
||||||
|
basepath?: string,
|
||||||
|
timeout: number | false
|
||||||
|
}) => {
|
||||||
process.env.NODE_ENV = 'production';
|
process.env.NODE_ENV = 'production';
|
||||||
process.env.SAPPER_DEST = '.sapper/.export';
|
process.env.SAPPER_DEST = opts['build-dir'];
|
||||||
|
|
||||||
const start = Date.now();
|
const start = Date.now();
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const { build } = await import('./cli/build');
|
if (opts.build) {
|
||||||
await build();
|
console.log(`> Building...`);
|
||||||
console.error(`\n> Built in ${elapsed(start)}. Crawling site...`);
|
const { build } = await import('./cli/build');
|
||||||
|
await build(opts);
|
||||||
|
console.error(`\n> Built in ${elapsed(start)}`);
|
||||||
|
}
|
||||||
|
|
||||||
const { exporter } = await import('./cli/export');
|
const { exporter } = await import('./cli/export');
|
||||||
await exporter(dest, opts);
|
await exporter(dest, opts);
|
||||||
console.error(`\n> Finished in ${elapsed(start)}. Type ${colors.bold.cyan(`npx serve ${dest}`)} to run the app.`);
|
console.error(`\n> Finished in ${elapsed(start)}. Type ${colors.bold.cyan(`npx serve ${dest}`)} to run the app.`);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
console.error(err ? err.details || err.stack || err.message || err : 'Unknown error');
|
console.error(colors.bold.red(`> ${err.message}`));
|
||||||
process.exit(1);
|
process.exit(1);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,20 +1,39 @@
|
|||||||
import { build as _build } from '../api/build';
|
import { build as _build } from '../api/build';
|
||||||
import * as colors from 'ansi-colors';
|
import colors from 'kleur';
|
||||||
import { locations } from '../config';
|
import { locations } from '../config';
|
||||||
|
import validate_bundler from './utils/validate_bundler';
|
||||||
|
import { repeat } from '../utils';
|
||||||
|
|
||||||
|
export function build(opts: { bundler?: string }) {
|
||||||
|
const bundler = validate_bundler(opts.bundler);
|
||||||
|
|
||||||
export function build() {
|
|
||||||
return new Promise((fulfil, reject) => {
|
return new Promise((fulfil, reject) => {
|
||||||
try {
|
try {
|
||||||
const emitter = _build({
|
const emitter = _build({
|
||||||
dest: locations.dest(),
|
dest: locations.dest(),
|
||||||
app: locations.app(),
|
app: locations.app(),
|
||||||
routes: locations.routes(),
|
routes: locations.routes(),
|
||||||
webpack: 'webpack'
|
bundler,
|
||||||
|
webpack: 'webpack',
|
||||||
|
rollup: 'rollup'
|
||||||
});
|
});
|
||||||
|
|
||||||
emitter.on('build', event => {
|
emitter.on('build', event => {
|
||||||
console.log(colors.inverse(`\nbuilt ${event.type}`));
|
let banner = `built ${event.type}`;
|
||||||
console.log(event.webpack_stats.toString({ colors: true }));
|
let c = colors.cyan;
|
||||||
|
|
||||||
|
const { warnings } = event.result;
|
||||||
|
if (warnings.length > 0) {
|
||||||
|
banner += ` with ${warnings.length} ${warnings.length === 1 ? 'warning' : 'warnings'}`;
|
||||||
|
c = colors.yellow;
|
||||||
|
}
|
||||||
|
|
||||||
|
console.log();
|
||||||
|
console.log(c(`┌─${repeat('─', banner.length)}─┐`));
|
||||||
|
console.log(c(`│ ${colors.bold(banner) } │`));
|
||||||
|
console.log(c(`└─${repeat('─', banner.length)}─┘`));
|
||||||
|
|
||||||
|
console.log(event.result.print());
|
||||||
});
|
});
|
||||||
|
|
||||||
emitter.on('error', event => {
|
emitter.on('error', event => {
|
||||||
@@ -25,8 +44,7 @@ export function build() {
|
|||||||
fulfil();
|
fulfil();
|
||||||
});
|
});
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
console.log(`${colors.bold.red(`> ${err.message}`)}`);
|
reject(err);
|
||||||
process.exit(1);
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -1,11 +1,11 @@
|
|||||||
import * as path from 'path';
|
import * as path from 'path';
|
||||||
import * as colors from 'ansi-colors';
|
import colors from 'kleur';
|
||||||
import * as child_process from 'child_process';
|
import * as child_process from 'child_process';
|
||||||
import prettyMs from 'pretty-ms';
|
import prettyMs from 'pretty-ms';
|
||||||
import { dev as _dev } from '../api/dev';
|
import { dev as _dev } from '../api/dev';
|
||||||
import * as events from '../api/interfaces';
|
import * as events from '../api/interfaces';
|
||||||
|
|
||||||
export function dev(opts: { port: number, open: boolean }) {
|
export function dev(opts: { port: number, open: boolean, bundler?: string }) {
|
||||||
try {
|
try {
|
||||||
const watcher = _dev(opts);
|
const watcher = _dev(opts);
|
||||||
|
|
||||||
@@ -13,7 +13,7 @@ export function dev(opts: { port: number, open: boolean }) {
|
|||||||
|
|
||||||
watcher.on('ready', (event: events.ReadyEvent) => {
|
watcher.on('ready', (event: events.ReadyEvent) => {
|
||||||
if (first) {
|
if (first) {
|
||||||
console.log(`${colors.bold.cyan(`> Listening on http://localhost:${event.port}`)}`);
|
console.log(colors.bold.cyan(`> Listening on http://localhost:${event.port}`));
|
||||||
if (opts.open) child_process.exec(`open http://localhost:${event.port}`);
|
if (opts.open) child_process.exec(`open http://localhost:${event.port}`);
|
||||||
first = false;
|
first = false;
|
||||||
}
|
}
|
||||||
@@ -35,19 +35,21 @@ export function dev(opts: { port: number, open: boolean }) {
|
|||||||
});
|
});
|
||||||
|
|
||||||
watcher.on('error', (event: events.ErrorEvent) => {
|
watcher.on('error', (event: events.ErrorEvent) => {
|
||||||
console.log(`${colors.red(`✗ ${event.type}`)}`);
|
console.log(colors.red(`✗ ${event.type}`));
|
||||||
console.log(`${colors.red(event.error.message)}`);
|
console.log(colors.red(event.message));
|
||||||
});
|
});
|
||||||
|
|
||||||
watcher.on('fatal', (event: events.FatalEvent) => {
|
watcher.on('fatal', (event: events.FatalEvent) => {
|
||||||
console.log(`${colors.bold.red(`> ${event.error.message}`)}`);
|
console.log(colors.bold.red(`> ${event.message}`));
|
||||||
|
if (event.log) console.log(event.log);
|
||||||
});
|
});
|
||||||
|
|
||||||
watcher.on('build', (event: events.BuildEvent) => {
|
watcher.on('build', (event: events.BuildEvent) => {
|
||||||
if (event.errors.length) {
|
if (event.errors.length) {
|
||||||
console.log(`${colors.bold.red(`✗ ${event.type}`)}`);
|
console.log(colors.bold.red(`✗ ${event.type}`));
|
||||||
|
|
||||||
event.errors.filter(e => !e.duplicate).forEach(error => {
|
event.errors.filter(e => !e.duplicate).forEach(error => {
|
||||||
|
if (error.file) console.log(colors.bold(error.file));
|
||||||
console.log(error.message);
|
console.log(error.message);
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -56,9 +58,10 @@ export function dev(opts: { port: number, open: boolean }) {
|
|||||||
console.log(`${hidden} duplicate ${hidden === 1 ? 'error' : 'errors'} hidden\n`);
|
console.log(`${hidden} duplicate ${hidden === 1 ? 'error' : 'errors'} hidden\n`);
|
||||||
}
|
}
|
||||||
} else if (event.warnings.length) {
|
} else if (event.warnings.length) {
|
||||||
console.log(`${colors.bold.yellow(`• ${event.type}`)}`);
|
console.log(colors.bold.yellow(`• ${event.type}`));
|
||||||
|
|
||||||
event.warnings.filter(e => !e.duplicate).forEach(warning => {
|
event.warnings.filter(e => !e.duplicate).forEach(warning => {
|
||||||
|
if (warning.file) console.log(colors.bold(warning.file));
|
||||||
console.log(warning.message);
|
console.log(warning.message);
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -71,7 +74,7 @@ export function dev(opts: { port: number, open: boolean }) {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
console.log(`${colors.bold.red(`> ${err.message}`)}`);
|
console.log(colors.bold.red(`> ${err.message}`));
|
||||||
process.exit(1);
|
process.exit(1);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -1,23 +1,38 @@
|
|||||||
import { exporter as _exporter } from '../api/export';
|
import { exporter as _exporter } from '../api/export';
|
||||||
import * as colors from 'ansi-colors';
|
import colors from 'kleur';
|
||||||
import prettyBytes from 'pretty-bytes';
|
import pb from 'pretty-bytes';
|
||||||
import { locations } from '../config';
|
import { locations } from '../config';
|
||||||
|
import { left_pad } from '../utils';
|
||||||
|
|
||||||
export function exporter(export_dir: string, { basepath = '' }) {
|
export function exporter(export_dir: string, {
|
||||||
|
basepath = '',
|
||||||
|
timeout
|
||||||
|
}: {
|
||||||
|
basepath: string,
|
||||||
|
timeout: number | false
|
||||||
|
}) {
|
||||||
return new Promise((fulfil, reject) => {
|
return new Promise((fulfil, reject) => {
|
||||||
try {
|
try {
|
||||||
const emitter = _exporter({
|
const emitter = _exporter({
|
||||||
build: locations.dest(),
|
build: locations.dest(),
|
||||||
dest: export_dir,
|
dest: export_dir,
|
||||||
basepath
|
basepath,
|
||||||
|
timeout
|
||||||
});
|
});
|
||||||
|
|
||||||
emitter.on('file', event => {
|
emitter.on('file', event => {
|
||||||
console.log(`${colors.bold.cyan(event.file)} ${colors.gray(`(${prettyBytes(event.size)})`)}`);
|
const size_color = event.size > 150000 ? colors.bold.red : event.size > 50000 ? colors.bold.yellow : colors.bold.gray;
|
||||||
|
const size_label = size_color(left_pad(pb(event.size), 10));
|
||||||
|
|
||||||
|
const file_label = event.status === 200
|
||||||
|
? event.file
|
||||||
|
: colors.bold[event.status >= 400 ? 'red' : 'yellow'](`(${event.status}) ${event.file}`);
|
||||||
|
|
||||||
|
console.log(`${size_label} ${file_label}`);
|
||||||
});
|
});
|
||||||
|
|
||||||
emitter.on('failure', event => {
|
emitter.on('info', event => {
|
||||||
console.log(`${colors.red(`> Received ${event.status} response when fetching ${event.pathname}`)}`);
|
console.log(colors.bold.cyan(`> ${event.message}`));
|
||||||
});
|
});
|
||||||
|
|
||||||
emitter.on('error', event => {
|
emitter.on('error', event => {
|
||||||
|
|||||||
@@ -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 * as child_process from 'child_process';
|
import * as child_process from 'child_process';
|
||||||
import * as colors from 'ansi-colors';
|
import colors from 'kleur';
|
||||||
import * as ports from 'port-authority';
|
import * as ports from 'port-authority';
|
||||||
|
|
||||||
export async function start(dir: string, opts: { port: number, open: boolean }) {
|
export async function start(dir: string, opts: { port: number, open: boolean }) {
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import * as fs from 'fs';
|
import * as fs from 'fs';
|
||||||
import * as colors from 'ansi-colors';
|
import colors from 'kleur';
|
||||||
|
|
||||||
export default async function upgrade() {
|
export default async function upgrade() {
|
||||||
const upgraded = [
|
const upgraded = [
|
||||||
|
|||||||
21
src/cli/utils/validate_bundler.ts
Normal file
21
src/cli/utils/validate_bundler.ts
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
import * as fs from 'fs';
|
||||||
|
|
||||||
|
export default function validate_bundler(bundler?: string) {
|
||||||
|
if (!bundler) {
|
||||||
|
bundler = (
|
||||||
|
fs.existsSync('rollup') ? 'rollup' :
|
||||||
|
fs.existsSync('webpack') ? 'webpack' :
|
||||||
|
null
|
||||||
|
);
|
||||||
|
|
||||||
|
if (!bundler) {
|
||||||
|
throw new Error(`Could not find a 'rollup' or 'webpack' directory`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (bundler !== 'rollup' && bundler !== 'webpack') {
|
||||||
|
throw new Error(`'${bundler}' is not a valid option for --bundler — must be either 'rollup' or 'webpack'`);
|
||||||
|
}
|
||||||
|
|
||||||
|
return bundler;
|
||||||
|
}
|
||||||
@@ -6,5 +6,5 @@ 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'),
|
app: () => path.resolve(process.env.SAPPER_BASE || '', process.env.SAPPER_APP || 'app'),
|
||||||
routes: () => path.resolve(process.env.SAPPER_BASE || '', process.env.SAPPER_ROUTES || 'routes'),
|
routes: () => path.resolve(process.env.SAPPER_BASE || '', process.env.SAPPER_ROUTES || 'routes'),
|
||||||
dest: () => path.resolve(process.env.SAPPER_BASE || '', process.env.SAPPER_DEST || '.sapper')
|
dest: () => path.resolve(process.env.SAPPER_BASE || '', process.env.SAPPER_DEST || `.sapper/${dev() ? 'dev' : 'prod'}`)
|
||||||
};
|
};
|
||||||
@@ -1,29 +1,378 @@
|
|||||||
|
import * as fs from 'fs';
|
||||||
import * as path from 'path';
|
import * as path from 'path';
|
||||||
|
import colors from 'kleur';
|
||||||
|
import pb from 'pretty-bytes';
|
||||||
import relative from 'require-relative';
|
import relative from 'require-relative';
|
||||||
|
import { left_pad } from '../utils';
|
||||||
|
|
||||||
export default function create_compilers({ webpack }: { webpack: string }) {
|
let r: any;
|
||||||
const wp = relative('webpack', process.cwd());
|
let wp: any;
|
||||||
|
|
||||||
const serviceworker_config = try_require(path.resolve(`${webpack}/service-worker.config.js`));
|
export class CompileError {
|
||||||
|
file: string;
|
||||||
|
message: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export class CompileResult {
|
||||||
|
duration: number;
|
||||||
|
errors: CompileError[];
|
||||||
|
warnings: CompileError[];
|
||||||
|
assets: string[];
|
||||||
|
assetsByChunkName: Record<string, string>;
|
||||||
|
}
|
||||||
|
|
||||||
|
class RollupResult extends CompileResult {
|
||||||
|
summary: string;
|
||||||
|
|
||||||
|
constructor(duration: number, compiler: RollupCompiler) {
|
||||||
|
super();
|
||||||
|
|
||||||
|
this.duration = duration;
|
||||||
|
|
||||||
|
this.errors = compiler.errors.map(munge_rollup_warning_or_error);
|
||||||
|
this.warnings = compiler.warnings.map(munge_rollup_warning_or_error); // TODO emit this as they happen
|
||||||
|
|
||||||
|
this.assets = compiler.chunks.map(chunk => chunk.fileName);
|
||||||
|
|
||||||
|
// TODO populate this properly. We don't have namedcompiler. chunks, as in
|
||||||
|
// webpack, but we can have a route -> [chunk] map or something
|
||||||
|
this.assetsByChunkName = {};
|
||||||
|
|
||||||
|
compiler.chunks.forEach(chunk => {
|
||||||
|
if (compiler.input in chunk.modules) {
|
||||||
|
this.assetsByChunkName.main = chunk.fileName;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
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_label = left_pad(pb(chunk.code.length), 10);
|
||||||
|
|
||||||
|
const lines = [size_color(`${size_label} ${chunk.fileName}`)];
|
||||||
|
|
||||||
|
const deps = Object.keys(chunk.modules)
|
||||||
|
.map(file => {
|
||||||
|
return {
|
||||||
|
file: path.relative(process.cwd(), file),
|
||||||
|
size: chunk.modules[file].renderedLength
|
||||||
|
};
|
||||||
|
})
|
||||||
|
.filter(dep => dep.size > 0)
|
||||||
|
.sort((a, b) => b.size - a.size);
|
||||||
|
|
||||||
|
const total_unminified = deps.reduce((t, d) => t + d.size, 0);
|
||||||
|
|
||||||
|
deps.forEach((dep, i) => {
|
||||||
|
const c = i === deps.length - 1 ? '└' : '│';
|
||||||
|
let line = ` ${c} ${dep.file}`;
|
||||||
|
|
||||||
|
if (deps.length > 1) {
|
||||||
|
const p = (100 * dep.size / total_unminified).toFixed(1);
|
||||||
|
line += ` (${p}%)`;
|
||||||
|
}
|
||||||
|
|
||||||
|
lines.push(colors.gray(line));
|
||||||
|
});
|
||||||
|
|
||||||
|
return lines.join('\n');
|
||||||
|
}).join('\n');
|
||||||
|
}
|
||||||
|
|
||||||
|
print() {
|
||||||
|
const blocks: string[] = this.warnings.map(warning => {
|
||||||
|
return warning.file
|
||||||
|
? `> ${colors.bold(warning.file)}\n${warning.message}`
|
||||||
|
: `> ${warning.message}`;
|
||||||
|
});
|
||||||
|
|
||||||
|
blocks.push(this.summary);
|
||||||
|
|
||||||
|
return blocks.join('\n\n');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class WebpackResult extends CompileResult {
|
||||||
|
stats: any;
|
||||||
|
|
||||||
|
constructor(stats: any) {
|
||||||
|
super();
|
||||||
|
|
||||||
|
this.stats = stats;
|
||||||
|
|
||||||
|
const info = stats.toJson();
|
||||||
|
|
||||||
|
// TODO use import()
|
||||||
|
const format_messages = require('webpack-format-messages');
|
||||||
|
const messages = format_messages(stats);
|
||||||
|
|
||||||
|
this.errors = messages.errors.map(munge_webpack_warning_or_error);
|
||||||
|
this.warnings = messages.warnings.map(munge_webpack_warning_or_error);
|
||||||
|
|
||||||
|
this.duration = info.time;
|
||||||
|
|
||||||
|
this.assets = info.assets.map((chunk: { name: string }) => chunk.name);
|
||||||
|
this.assetsByChunkName = info.assetsByChunkName;
|
||||||
|
}
|
||||||
|
|
||||||
|
print() {
|
||||||
|
return this.stats.toString({ colors: true });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export class RollupCompiler {
|
||||||
|
_: Promise<any>;
|
||||||
|
_oninvalid: (filename: string) => void;
|
||||||
|
_start: number;
|
||||||
|
input: string;
|
||||||
|
warnings: any[];
|
||||||
|
errors: any[];
|
||||||
|
chunks: any[]; // TODO types
|
||||||
|
|
||||||
|
constructor(config: any) {
|
||||||
|
this._ = this.get_config(path.resolve(config));
|
||||||
|
this.input = null;
|
||||||
|
this.warnings = [];
|
||||||
|
this.errors = [];
|
||||||
|
this.chunks = [];
|
||||||
|
}
|
||||||
|
|
||||||
|
async get_config(input: string) {
|
||||||
|
const bundle = await r.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({
|
||||||
|
name: 'sapper-internal',
|
||||||
|
options: (opts: any) => {
|
||||||
|
this.input = opts.input;
|
||||||
|
},
|
||||||
|
renderChunk: (code: string, chunk: any) => {
|
||||||
|
if (chunk.isEntry) {
|
||||||
|
this.chunks.push(chunk);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
const onwarn = mod.onwarn || ((warning: any, handler: (warning: any) => void) => {
|
||||||
|
handler(warning);
|
||||||
|
});
|
||||||
|
|
||||||
|
mod.onwarn = (warning: any) => {
|
||||||
|
onwarn(warning, (warning: any) => {
|
||||||
|
this.warnings.push(warning);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
return mod;
|
||||||
|
}
|
||||||
|
|
||||||
|
oninvalid(cb: (filename: string) => void) {
|
||||||
|
this._oninvalid = cb;
|
||||||
|
}
|
||||||
|
|
||||||
|
async compile(): Promise<CompileResult> {
|
||||||
|
const config = await this._;
|
||||||
|
|
||||||
|
const start = Date.now();
|
||||||
|
|
||||||
|
try {
|
||||||
|
const bundle = await r.rollup(config);
|
||||||
|
await bundle.write(config.output);
|
||||||
|
|
||||||
|
return new RollupResult(Date.now() - start, this);
|
||||||
|
} catch (err) {
|
||||||
|
if (err.filename) {
|
||||||
|
// TODO this is a bit messy. Also, can
|
||||||
|
// Rollup emit other kinds of error?
|
||||||
|
err.message = [
|
||||||
|
`Failed to build — error in ${err.filename}: ${err.message}`,
|
||||||
|
err.frame
|
||||||
|
].filter(Boolean).join('\n');
|
||||||
|
}
|
||||||
|
|
||||||
|
throw err;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async watch(cb: (err?: Error, stats?: any) => void) {
|
||||||
|
const config = await this._;
|
||||||
|
|
||||||
|
const watcher = r.watch(config);
|
||||||
|
|
||||||
|
watcher.on('change', (id: string) => {
|
||||||
|
this.chunks = [];
|
||||||
|
this.warnings = [];
|
||||||
|
this.errors = [];
|
||||||
|
this._oninvalid(id);
|
||||||
|
});
|
||||||
|
|
||||||
|
watcher.on('event', (event: any) => {
|
||||||
|
switch (event.code) {
|
||||||
|
case 'FATAL':
|
||||||
|
// TODO kill the process?
|
||||||
|
if (event.error.filename) {
|
||||||
|
// TODO this is a bit messy. Also, can
|
||||||
|
// Rollup emit other kinds of error?
|
||||||
|
event.error.message = [
|
||||||
|
`Failed to build — error in ${event.error.filename}: ${event.error.message}`,
|
||||||
|
event.error.frame
|
||||||
|
].filter(Boolean).join('\n');
|
||||||
|
}
|
||||||
|
|
||||||
|
cb(event.error);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'ERROR':
|
||||||
|
this.errors.push(event.error);
|
||||||
|
cb(null, new RollupResult(Date.now() - this._start, this));
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'START':
|
||||||
|
case 'END':
|
||||||
|
// TODO is there anything to do with this info?
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'BUNDLE_START':
|
||||||
|
this._start = Date.now();
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'BUNDLE_END':
|
||||||
|
cb(null, new RollupResult(Date.now() - this._start, this));
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
console.log(`Unexpected event ${event.code}`);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export class WebpackCompiler {
|
||||||
|
_: any;
|
||||||
|
|
||||||
|
constructor(config: any) {
|
||||||
|
this._ = wp(require(path.resolve(config)));
|
||||||
|
}
|
||||||
|
|
||||||
|
oninvalid(cb: (filename: string) => void) {
|
||||||
|
this._.hooks.invalid.tap('sapper', cb);
|
||||||
|
}
|
||||||
|
|
||||||
|
compile(): Promise<CompileResult> {
|
||||||
|
return new Promise((fulfil, reject) => {
|
||||||
|
this._.run((err: Error, stats: any) => {
|
||||||
|
if (err) {
|
||||||
|
reject(err);
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
const result = new WebpackResult(stats);
|
||||||
|
|
||||||
|
if (result.errors.length) {
|
||||||
|
// TODO print errors
|
||||||
|
// console.error(stats.toString({ colors: true }));
|
||||||
|
reject(new Error(`Encountered errors while building app`));
|
||||||
|
}
|
||||||
|
|
||||||
|
else {
|
||||||
|
fulfil(result);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
watch(cb: (err?: Error, stats?: any) => void) {
|
||||||
|
this._.watch({}, (err?: Error, stats?: any) => {
|
||||||
|
cb(err, stats && new WebpackResult(stats));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export type Compiler = RollupCompiler | WebpackCompiler;
|
||||||
|
|
||||||
|
export type Compilers = {
|
||||||
|
client: Compiler;
|
||||||
|
server: Compiler;
|
||||||
|
serviceworker?: Compiler;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function create_compilers(bundler: string, { webpack, rollup }: { webpack: string, rollup: string }): Compilers {
|
||||||
|
if (bundler === 'rollup') {
|
||||||
|
if (!r) r = relative('rollup', process.cwd());
|
||||||
|
|
||||||
|
const sw = `${rollup}/service-worker.config.js`;
|
||||||
|
|
||||||
|
return {
|
||||||
|
client: new RollupCompiler(`${rollup}/client.config.js`),
|
||||||
|
server: new RollupCompiler(`${rollup}/server.config.js`),
|
||||||
|
serviceworker: fs.existsSync(sw) && new RollupCompiler(sw)
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
if (bundler === 'webpack') {
|
||||||
|
if (!wp) wp = relative('webpack', process.cwd());
|
||||||
|
|
||||||
|
const sw = `${webpack}/service-worker.config.js`;
|
||||||
|
|
||||||
|
return {
|
||||||
|
client: new WebpackCompiler(`${webpack}/client.config.js`),
|
||||||
|
server: new WebpackCompiler(`${webpack}/server.config.js`),
|
||||||
|
serviceworker: fs.existsSync(sw) && new WebpackCompiler(sw)
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
// this shouldn't be possible...
|
||||||
|
throw new Error(`Invalid bundler option '${bundler}'`);
|
||||||
|
}
|
||||||
|
|
||||||
|
const locPattern = /\((\d+):(\d+)\)$/;
|
||||||
|
|
||||||
|
function munge_webpack_warning_or_error(message: string) {
|
||||||
|
// TODO this is all a bit rube goldberg...
|
||||||
|
const lines = message.split('\n');
|
||||||
|
|
||||||
|
const file = lines.shift()
|
||||||
|
.replace('[7m', '') // careful — there is a special character at the beginning of this string
|
||||||
|
.replace('[27m', '')
|
||||||
|
.replace('./', '');
|
||||||
|
|
||||||
|
let line = null;
|
||||||
|
let column = null;
|
||||||
|
|
||||||
|
const match = locPattern.exec(lines[0]);
|
||||||
|
if (match) {
|
||||||
|
lines[0] = lines[0].replace(locPattern, '');
|
||||||
|
line = +match[1];
|
||||||
|
column = +match[2];
|
||||||
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
client: wp(
|
file,
|
||||||
require(path.resolve(`${webpack}/client.config.js`))
|
message: lines.join('\n')
|
||||||
),
|
|
||||||
|
|
||||||
server: wp(
|
|
||||||
require(path.resolve(`${webpack}/server.config.js`))
|
|
||||||
),
|
|
||||||
|
|
||||||
serviceworker: serviceworker_config && wp(serviceworker_config)
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
function try_require(specifier: string) {
|
function munge_rollup_warning_or_error(warning_or_error: any) {
|
||||||
try {
|
return {
|
||||||
return require(specifier);
|
file: warning_or_error.filename,
|
||||||
} catch (err) {
|
message: [warning_or_error.message, warning_or_error.frame].filter(Boolean).join('\n')
|
||||||
if (err.code === 'MODULE_NOT_FOUND') return null;
|
};
|
||||||
throw err;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
@@ -1,28 +1,35 @@
|
|||||||
import * as fs from 'fs';
|
import * as fs from 'fs';
|
||||||
import * as path from 'path';
|
import * as path from 'path';
|
||||||
import * as glob from 'glob';
|
import glob from 'tiny-glob/sync.js';
|
||||||
import { posixify, write_if_changed } from './utils';
|
import { posixify, write_if_changed } from './utils';
|
||||||
import { dev, locations } from '../config';
|
import { dev, locations } from '../config';
|
||||||
import { Route } from '../interfaces';
|
import { Page, PageComponent, ServerRoute } from '../interfaces';
|
||||||
|
|
||||||
export function create_main_manifests({ routes, dev_port }: {
|
export function create_main_manifests({ routes, dev_port }: {
|
||||||
routes: Route[];
|
routes: { components: PageComponent[], pages: Page[], server_routes: ServerRoute[] };
|
||||||
dev_port?: number;
|
dev_port?: number;
|
||||||
}) {
|
}) {
|
||||||
const path_to_routes = path.relative(`${locations.app()}/manifest`, locations.routes());
|
const manifest_dir = path.join(locations.app(), 'manifest');
|
||||||
|
if (!fs.existsSync(manifest_dir)) fs.mkdirSync(manifest_dir);
|
||||||
|
|
||||||
|
const path_to_routes = path.relative(manifest_dir, locations.routes());
|
||||||
|
|
||||||
const client_manifest = generate_client(routes, path_to_routes, dev_port);
|
const client_manifest = generate_client(routes, path_to_routes, dev_port);
|
||||||
const server_manifest = generate_server(routes, path_to_routes);
|
const server_manifest = generate_server(routes, path_to_routes);
|
||||||
|
|
||||||
write_if_changed(`${locations.app()}/manifest/client.js`, client_manifest);
|
write_if_changed(
|
||||||
write_if_changed(`${locations.app()}/manifest/server.js`, server_manifest);
|
`${manifest_dir}/default-layout.html`,
|
||||||
|
`<svelte:component this={child.component} {...child.props}/>`
|
||||||
|
);
|
||||||
|
write_if_changed(`${manifest_dir}/client.js`, client_manifest);
|
||||||
|
write_if_changed(`${manifest_dir}/server.js`, server_manifest);
|
||||||
}
|
}
|
||||||
|
|
||||||
export function create_serviceworker_manifest({ routes, client_files }: {
|
export function create_serviceworker_manifest({ routes, client_files }: {
|
||||||
routes: Route[];
|
routes: { components: PageComponent[], pages: Page[], server_routes: ServerRoute[] };
|
||||||
client_files: string[];
|
client_files: string[];
|
||||||
}) {
|
}) {
|
||||||
const assets = glob.sync('**', { cwd: 'assets', nodir: true });
|
const assets = glob('**', { cwd: 'assets', filesOnly: true });
|
||||||
|
|
||||||
let code = `
|
let code = `
|
||||||
// This file is generated by Sapper — do not edit it!
|
// This file is generated by Sapper — do not edit it!
|
||||||
@@ -32,38 +39,62 @@ export function create_serviceworker_manifest({ routes, client_files }: {
|
|||||||
|
|
||||||
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) => `"${x}"`).join(',\n\t')}\n];
|
||||||
|
|
||||||
export const routes = [\n\t${routes.filter((r: Route) => r.type === 'page' && !/^_[45]xx$/.test(r.id)).map((r: Route) => `{ pattern: ${r.pattern} }`).join(',\n\t')}\n];
|
export const routes = [\n\t${routes.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(`${locations.app()}/manifest/service-worker.js`, code);
|
||||||
}
|
}
|
||||||
|
|
||||||
function generate_client(routes: Route[], path_to_routes: string, dev_port?: number) {
|
function generate_client(
|
||||||
|
routes: { root: PageComponent, components: PageComponent[], pages: Page[], server_routes: ServerRoute[] },
|
||||||
|
path_to_routes: string,
|
||||||
|
dev_port?: number
|
||||||
|
) {
|
||||||
|
const page_ids = new Set(routes.pages.map(page =>
|
||||||
|
page.pattern.toString()));
|
||||||
|
|
||||||
|
const server_routes_to_ignore = routes.server_routes.filter(route =>
|
||||||
|
!page_ids.has(route.pattern.toString()));
|
||||||
|
|
||||||
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 routes = [
|
import root from '${get_file(path_to_routes, routes.root)}';
|
||||||
${routes
|
import error from '${posixify(`${path_to_routes}/_error.html`)}';
|
||||||
.map(route => {
|
|
||||||
const page = route.handlers.find(({ type }) => type === 'page');
|
|
||||||
|
|
||||||
if (!page) {
|
${routes.components.map(component =>
|
||||||
return `{ pattern: ${route.pattern}, ignore: true }`;
|
`const ${component.name} = () =>
|
||||||
}
|
import(/* webpackChunkName: "${component.name}" */ '${get_file(path_to_routes, component)}');`)
|
||||||
|
.join('\n')}
|
||||||
|
|
||||||
const file = posixify(`${path_to_routes}/${page.file}`);
|
export const manifest = {
|
||||||
|
ignore: [${server_routes_to_ignore.map(route => route.pattern).join(', ')}],
|
||||||
|
|
||||||
if (route.id === '_4xx' || route.id === '_5xx') {
|
pages: [
|
||||||
return `{ error: '${route.id.slice(1)}', load: () => import(/* webpackChunkName: "${route.id}" */ '${file}') }`;
|
${routes.pages.map(page => `{
|
||||||
}
|
// ${page.parts[page.parts.length - 1].component.file}
|
||||||
|
pattern: ${page.pattern},
|
||||||
|
parts: [
|
||||||
|
${page.parts.map(part => {
|
||||||
|
if (part === null) return 'null';
|
||||||
|
|
||||||
const params = route.params.length === 0
|
if (part.params.length > 0) {
|
||||||
? '{}'
|
const props = part.params.map((param, i) => `${param}: match[${i + 1}]`);
|
||||||
: `{ ${route.params.map((part, i) => `${part}: match[${i + 1}]`).join(', ')} }`;
|
return `{ component: ${part.component.name}, params: match => ({ ${props.join(', ')} }) }`;
|
||||||
|
}
|
||||||
|
|
||||||
return `{ pattern: ${route.pattern}, params: ${route.params.length > 0 ? `match` : `()`} => (${params}), load: () => import(/* webpackChunkName: "${route.id}" */ '${file}') }`;
|
return `{ component: ${part.component.name} }`;
|
||||||
})
|
}).join(',\n\t\t\t\t\t\t')}
|
||||||
.join(',\n\t')}
|
]
|
||||||
];`.replace(/^\t\t/gm, '').trim();
|
}`).join(',\n\n\t\t\t\t')}
|
||||||
|
],
|
||||||
|
|
||||||
|
root,
|
||||||
|
|
||||||
|
error
|
||||||
|
};
|
||||||
|
|
||||||
|
// this is included for legacy reasons
|
||||||
|
export const routes = {};`.replace(/^\t\t/gm, '').trim();
|
||||||
|
|
||||||
if (dev()) {
|
if (dev()) {
|
||||||
const sapper_dev_client = posixify(
|
const sapper_dev_client = posixify(
|
||||||
@@ -72,54 +103,82 @@ function generate_client(routes: Route[], path_to_routes: string, dev_port?: num
|
|||||||
|
|
||||||
code += `
|
code += `
|
||||||
|
|
||||||
if (module.hot) {
|
import('${sapper_dev_client}').then(client => {
|
||||||
import('${sapper_dev_client}').then(client => {
|
client.connect(${dev_port});
|
||||||
client.connect(${dev_port});
|
});`.replace(/^\t{3}/gm, '');
|
||||||
});
|
|
||||||
}`.replace(/^\t{3}/gm, '');
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return code;
|
return code;
|
||||||
}
|
}
|
||||||
|
|
||||||
function generate_server(routes: Route[], path_to_routes: string) {
|
function generate_server(
|
||||||
|
routes: { root: PageComponent, components: PageComponent[], pages: Page[], server_routes: ServerRoute[] },
|
||||||
|
path_to_routes: string
|
||||||
|
) {
|
||||||
|
const imports = [].concat(
|
||||||
|
routes.server_routes.map(route =>
|
||||||
|
`import * as ${route.name} from '${posixify(`${path_to_routes}/${route.file}`)}';`),
|
||||||
|
routes.components.map(component =>
|
||||||
|
`import ${component.name} from '${get_file(path_to_routes, component)}';`),
|
||||||
|
`import root from '${get_file(path_to_routes, routes.root)}';`,
|
||||||
|
`import error from '${posixify(`${path_to_routes}/_error.html`)}';`
|
||||||
|
);
|
||||||
|
|
||||||
let code = `
|
let code = `
|
||||||
// This file is generated by Sapper — do not edit it!
|
// This file is generated by Sapper — do not edit it!
|
||||||
${routes
|
${imports.join('\n')}
|
||||||
.map(route =>
|
|
||||||
route.handlers
|
|
||||||
.map(({ type, file }, index) => {
|
|
||||||
const module = posixify(`${path_to_routes}/${file}`);
|
|
||||||
|
|
||||||
return type === 'page'
|
export const manifest = {
|
||||||
? `import ${route.id}${index} from '${module}';`
|
server_routes: [
|
||||||
: `import * as ${route.id}${index} from '${module}';`;
|
${routes.server_routes.map(route => `{
|
||||||
})
|
// ${route.file}
|
||||||
.join('\n')
|
pattern: ${route.pattern},
|
||||||
)
|
handlers: ${route.name},
|
||||||
.join('\n')}
|
params: ${route.params.length > 0
|
||||||
|
? `match => ({ ${route.params.map((param, i) => `${param}: match[${i + 1}]`).join(', ')} })`
|
||||||
|
: `() => ({})`}
|
||||||
|
}`).join(',\n\n\t\t\t\t')}
|
||||||
|
],
|
||||||
|
|
||||||
export const routes = [
|
pages: [
|
||||||
${routes
|
${routes.pages.map(page => `{
|
||||||
.map(route => {
|
// ${page.parts[page.parts.length - 1].component.file}
|
||||||
const handlers = route.handlers
|
pattern: ${page.pattern},
|
||||||
.map(({ type }, index) =>
|
parts: [
|
||||||
`{ type: '${type}', module: ${route.id}${index} }`)
|
${page.parts.map(part => {
|
||||||
.join(', ');
|
if (part === null) return 'null';
|
||||||
|
|
||||||
if (route.id === '_4xx' || route.id === '_5xx') {
|
const props = [
|
||||||
return `{ error: '${route.id.slice(1)}', handlers: [${handlers}] }`;
|
`name: "${part.component.name}"`,
|
||||||
}
|
`component: ${part.component.name}`
|
||||||
|
];
|
||||||
|
|
||||||
const params = route.params.length === 0
|
if (part.params.length > 0) {
|
||||||
? '{}'
|
const params = part.params.map((param, i) => `${param}: match[${i + 1}]`);
|
||||||
: `{ ${route.params.map((part, i) => `${part}: match[${i + 1}]`).join(', ')} }`;
|
props.push(`params: match => ({ ${params.join(', ')} })`);
|
||||||
|
}
|
||||||
|
|
||||||
return `{ id: '${route.id}', pattern: ${route.pattern}, params: ${route.params.length > 0 ? `match` : `()`} => (${params}), handlers: [${handlers}] }`;
|
return `{ ${props.join(', ')} }`;
|
||||||
})
|
}).join(',\n\t\t\t\t\t\t')}
|
||||||
.join(',\n\t')
|
]
|
||||||
}
|
}`).join(',\n\n\t\t\t\t')}
|
||||||
];`.replace(/^\t\t/gm, '').trim();
|
],
|
||||||
|
|
||||||
|
root,
|
||||||
|
|
||||||
|
error
|
||||||
|
};
|
||||||
|
|
||||||
|
// this is included for legacy reasons
|
||||||
|
export const routes = {};`.replace(/^\t\t/gm, '').trim();
|
||||||
|
|
||||||
return code;
|
return code;
|
||||||
|
}
|
||||||
|
|
||||||
|
function get_file(path_to_routes: string, component: PageComponent) {
|
||||||
|
if (component.default) {
|
||||||
|
return `./default-layout.html`;
|
||||||
|
}
|
||||||
|
|
||||||
|
return posixify(`${path_to_routes}/${component.file}`);
|
||||||
}
|
}
|
||||||
@@ -1,154 +1,295 @@
|
|||||||
|
import * as fs from 'fs';
|
||||||
import * as path from 'path';
|
import * as path from 'path';
|
||||||
import glob from 'glob';
|
|
||||||
import { locations } from '../config';
|
import { locations } from '../config';
|
||||||
import { Route } from '../interfaces';
|
import { Page, PageComponent, ServerRoute } from '../interfaces';
|
||||||
|
import { posixify } from './utils';
|
||||||
|
|
||||||
export default function create_routes({ files } = { files: glob.sync('**/*.*', { cwd: locations.routes(), dot: true, nodir: true }) }) {
|
export default function create_routes(cwd = locations.routes()) {
|
||||||
const routes: Route[] = files
|
const components: PageComponent[] = [];
|
||||||
.filter((file: string) => !/(^|\/|\\)_/.test(file))
|
const pages: Page[] = [];
|
||||||
.map((file: string) => {
|
const server_routes: ServerRoute[] = [];
|
||||||
if (/(^|\/|\\)(_|\.(?!well-known))/.test(file)) return;
|
|
||||||
|
|
||||||
if (/]\[/.test(file)) {
|
const default_layout: PageComponent = {
|
||||||
throw new Error(`Invalid route ${file} — parameters must be separated`);
|
default: true,
|
||||||
}
|
name: '_default_layout',
|
||||||
|
file: null
|
||||||
|
};
|
||||||
|
|
||||||
const base = file.replace(/\.[^/.]+$/, '');
|
function walk(
|
||||||
const parts = base.split('/'); // glob output is always posix-style
|
dir: string,
|
||||||
if (parts[parts.length - 1] === 'index') parts.pop();
|
parent_segments: Part[][],
|
||||||
|
parent_params: string[],
|
||||||
|
stack: Array<{
|
||||||
|
component: PageComponent,
|
||||||
|
params: string[]
|
||||||
|
}>
|
||||||
|
) {
|
||||||
|
const items = fs.readdirSync(dir)
|
||||||
|
.map(basename => {
|
||||||
|
const resolved = path.join(dir, basename);
|
||||||
|
const file = path.relative(cwd, resolved);
|
||||||
|
const is_dir = fs.statSync(resolved).isDirectory();
|
||||||
|
|
||||||
return {
|
const segment = is_dir
|
||||||
files: [file],
|
? basename
|
||||||
base,
|
: basename.slice(0, -path.extname(basename).length);
|
||||||
parts
|
|
||||||
};
|
|
||||||
})
|
|
||||||
.filter(Boolean)
|
|
||||||
.filter((a, index, array) => {
|
|
||||||
const found = array.slice(index + 1).find(b => a.base === b.base);
|
|
||||||
if (found) found.files.push(...a.files);
|
|
||||||
return !found;
|
|
||||||
})
|
|
||||||
.sort((a, b) => {
|
|
||||||
if (a.parts[0] === '4xx' || a.parts[0] === '5xx') return -1;
|
|
||||||
if (b.parts[0] === '4xx' || b.parts[0] === '5xx') return 1;
|
|
||||||
|
|
||||||
const max = Math.max(a.parts.length, b.parts.length);
|
const parts = get_parts(segment);
|
||||||
|
const is_index = is_dir ? false : basename.startsWith('index.');
|
||||||
|
const is_page = path.extname(basename) === '.html';
|
||||||
|
|
||||||
for (let i = 0; i < max; i += 1) {
|
parts.forEach(part => {
|
||||||
const a_part = a.parts[i];
|
if (/\]\[/.test(part.content)) {
|
||||||
const b_part = b.parts[i];
|
throw new Error(`Invalid route ${file} — parameters must be separated`);
|
||||||
|
|
||||||
if (!a_part) return -1;
|
|
||||||
if (!b_part) return 1;
|
|
||||||
|
|
||||||
const a_sub_parts = get_sub_parts(a_part);
|
|
||||||
const b_sub_parts = get_sub_parts(b_part);
|
|
||||||
const max = Math.max(a_sub_parts.length, b_sub_parts.length);
|
|
||||||
|
|
||||||
for (let i = 0; i < max; i += 1) {
|
|
||||||
const a_sub_part = a_sub_parts[i];
|
|
||||||
const b_sub_part = b_sub_parts[i];
|
|
||||||
|
|
||||||
if (!a_sub_part) return 1; // b is more specific, so goes first
|
|
||||||
if (!b_sub_part) return -1;
|
|
||||||
|
|
||||||
if (a_sub_part.dynamic !== b_sub_part.dynamic) {
|
|
||||||
return a_sub_part.dynamic ? 1 : -1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!a_sub_part.dynamic && a_sub_part.content !== b_sub_part.content) {
|
if (part.qualifier && /[\(\)\?\:]/.test(part.qualifier.slice(1, -1))) {
|
||||||
return (
|
throw new Error(`Invalid route ${file} — cannot use (, ), ? or : in route qualifiers`);
|
||||||
(b_sub_part.content.length - a_sub_part.content.length) ||
|
|
||||||
(a_sub_part.content < b_sub_part.content ? -1 : 1)
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
throw new Error(`The ${a.base} and ${b.base} routes clash`);
|
|
||||||
})
|
|
||||||
.map(({ files, base, parts }) => {
|
|
||||||
const id = (
|
|
||||||
parts.join('_').replace(/[[\]]/g, '$').replace(/^\d/, '_$&').replace(/[^a-zA-Z0-9_$]/g, '_')
|
|
||||||
) || '_';
|
|
||||||
|
|
||||||
const params: string[] = [];
|
|
||||||
const param_pattern = /\[([^\]]+)\]/g;
|
|
||||||
let match;
|
|
||||||
while (match = param_pattern.exec(base)) {
|
|
||||||
params.push(match[1]);
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO can we do all this with sub-parts? or does
|
|
||||||
// nesting make that impossible?
|
|
||||||
let pattern_string = '';
|
|
||||||
let i = parts.length;
|
|
||||||
let nested = true;
|
|
||||||
while (i--) {
|
|
||||||
const part = encodeURI(parts[i].normalize()).replace(/\?/g, '%3F').replace(/#/g, '%23').replace(/%5B/g, '[').replace(/%5D/g, ']');
|
|
||||||
const dynamic = ~part.indexOf('[');
|
|
||||||
|
|
||||||
if (dynamic) {
|
|
||||||
const matcher = part.replace(param_pattern, `([^\/]+?)`);
|
|
||||||
pattern_string = nested ? `(?:\\/${matcher}${pattern_string})?` : `\\/${matcher}${pattern_string}`;
|
|
||||||
} else {
|
|
||||||
nested = false;
|
|
||||||
pattern_string = `\\/${part}${pattern_string}`;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const pattern = new RegExp(`^${pattern_string}\\/?$`);
|
|
||||||
|
|
||||||
const test = (url: string) => pattern.test(url);
|
|
||||||
|
|
||||||
const exec = (url: string) => {
|
|
||||||
const match = pattern.exec(url);
|
|
||||||
if (!match) return;
|
|
||||||
|
|
||||||
const result: Record<string, string> = {};
|
|
||||||
params.forEach((param, i) => {
|
|
||||||
result[param] = match[i + 1];
|
|
||||||
});
|
});
|
||||||
|
|
||||||
return result;
|
return {
|
||||||
};
|
basename,
|
||||||
|
parts,
|
||||||
|
file: posixify(file),
|
||||||
|
is_dir,
|
||||||
|
is_index,
|
||||||
|
is_page
|
||||||
|
};
|
||||||
|
})
|
||||||
|
.sort(comparator);
|
||||||
|
|
||||||
return {
|
items.forEach(item => {
|
||||||
id,
|
if (item.basename[0] === '_') return;
|
||||||
handlers: files.map(file => ({
|
|
||||||
type: path.extname(file) === '.html' ? 'page' : 'route',
|
if (item.basename[0] === '.') {
|
||||||
file
|
if (item.file !== '.well-known') return;
|
||||||
})).sort((a, b) => {
|
}
|
||||||
if (a.type === 'page' && b.type === 'route') {
|
|
||||||
return 1;
|
const segments = parent_segments.slice();
|
||||||
|
|
||||||
|
if (item.is_index && segments.length > 0) {
|
||||||
|
const last_segment = segments[segments.length - 1].slice();
|
||||||
|
const suffix = item.basename
|
||||||
|
.slice(0, -path.extname(item.basename).length).
|
||||||
|
replace('index', '');
|
||||||
|
|
||||||
|
if (suffix) {
|
||||||
|
const last_part = last_segment[last_segment.length - 1];
|
||||||
|
if (last_part.dynamic) {
|
||||||
|
last_segment.push({ dynamic: false, content: suffix });
|
||||||
|
} else {
|
||||||
|
last_segment[last_segment.length - 1] = {
|
||||||
|
dynamic: false,
|
||||||
|
content: `${last_part.content}${suffix}`
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
if (a.type === 'route' && b.type === 'page') {
|
segments[segments.length - 1] = last_segment;
|
||||||
return -1;
|
}
|
||||||
}
|
} else {
|
||||||
|
segments.push(item.parts);
|
||||||
|
}
|
||||||
|
|
||||||
return 0;
|
const params = parent_params.slice();
|
||||||
}),
|
params.push(...item.parts.filter(p => p.dynamic).map(p => p.content));
|
||||||
pattern,
|
|
||||||
test,
|
if (item.is_dir) {
|
||||||
exec,
|
const index = path.join(dir, item.basename, '_layout.html');
|
||||||
parts,
|
|
||||||
params
|
const component = fs.existsSync(index) && {
|
||||||
};
|
name: `${get_slug(item.file)}__layout`,
|
||||||
|
file: `${item.file}/_layout.html`
|
||||||
|
};
|
||||||
|
|
||||||
|
if (component) components.push(component);
|
||||||
|
|
||||||
|
walk(
|
||||||
|
path.join(dir, item.basename),
|
||||||
|
segments,
|
||||||
|
params,
|
||||||
|
component
|
||||||
|
? stack.concat({ component, params })
|
||||||
|
: stack.concat(null)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
else if (item.is_page) {
|
||||||
|
const component = {
|
||||||
|
name: get_slug(item.file),
|
||||||
|
file: item.file
|
||||||
|
};
|
||||||
|
|
||||||
|
const parts = stack.concat({
|
||||||
|
component,
|
||||||
|
params
|
||||||
|
});
|
||||||
|
|
||||||
|
components.push(component);
|
||||||
|
if (item.basename === 'index.html') {
|
||||||
|
pages.push({
|
||||||
|
pattern: get_pattern(parent_segments),
|
||||||
|
parts
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
pages.push({
|
||||||
|
pattern: get_pattern(segments),
|
||||||
|
parts
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
else {
|
||||||
|
server_routes.push({
|
||||||
|
name: `route_${get_slug(item.file)}`,
|
||||||
|
pattern: get_pattern(segments),
|
||||||
|
file: item.file,
|
||||||
|
params: params
|
||||||
|
});
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
}
|
||||||
|
|
||||||
return routes;
|
const root_file = path.join(cwd, '_layout.html');
|
||||||
|
const root = fs.existsSync(root_file)
|
||||||
|
? {
|
||||||
|
name: 'main',
|
||||||
|
file: '_layout.html'
|
||||||
|
}
|
||||||
|
: default_layout;
|
||||||
|
|
||||||
|
walk(cwd, [], [], []);
|
||||||
|
|
||||||
|
// check for clashes
|
||||||
|
const seen_pages: Map<string, Page> = new Map();
|
||||||
|
pages.forEach(page => {
|
||||||
|
const pattern = page.pattern.toString();
|
||||||
|
if (seen_pages.has(pattern)) {
|
||||||
|
const file = page.parts.pop().component.file;
|
||||||
|
const other_page = seen_pages.get(pattern);
|
||||||
|
const other_file = other_page.parts.pop().component.file;
|
||||||
|
|
||||||
|
throw new Error(`The ${other_file} and ${file} pages clash`);
|
||||||
|
}
|
||||||
|
|
||||||
|
seen_pages.set(pattern, page);
|
||||||
|
});
|
||||||
|
|
||||||
|
const seen_routes: Map<string, ServerRoute> = new Map();
|
||||||
|
server_routes.forEach(route => {
|
||||||
|
const pattern = route.pattern.toString();
|
||||||
|
if (seen_routes.has(pattern)) {
|
||||||
|
const other_route = seen_routes.get(pattern);
|
||||||
|
throw new Error(`The ${other_route.file} and ${route.file} routes clash`);
|
||||||
|
}
|
||||||
|
|
||||||
|
seen_routes.set(pattern, route);
|
||||||
|
});
|
||||||
|
|
||||||
|
return {
|
||||||
|
root,
|
||||||
|
components,
|
||||||
|
pages,
|
||||||
|
server_routes
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
function get_sub_parts(part: string) {
|
type Part = {
|
||||||
return part.split(/[\[\]]/)
|
content: string;
|
||||||
.map((content, i) => {
|
dynamic: boolean;
|
||||||
if (!content) return null;
|
qualifier?: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
function comparator(
|
||||||
|
a: { basename: string, parts: Part[], file: string, is_index: boolean },
|
||||||
|
b: { basename: string, parts: Part[], file: string, is_index: boolean }
|
||||||
|
) {
|
||||||
|
if (a.is_index !== b.is_index) return a.is_index ? -1 : 1;
|
||||||
|
|
||||||
|
const max = Math.max(a.parts.length, b.parts.length);
|
||||||
|
|
||||||
|
for (let i = 0; i < max; i += 1) {
|
||||||
|
const a_sub_part = a.parts[i];
|
||||||
|
const b_sub_part = b.parts[i];
|
||||||
|
|
||||||
|
if (!a_sub_part) return 1; // b is more specific, so goes first
|
||||||
|
if (!b_sub_part) return -1;
|
||||||
|
|
||||||
|
if (a_sub_part.dynamic !== b_sub_part.dynamic) {
|
||||||
|
return a_sub_part.dynamic ? 1 : -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!a_sub_part.dynamic && a_sub_part.content !== b_sub_part.content) {
|
||||||
|
return (
|
||||||
|
(b_sub_part.content.length - a_sub_part.content.length) ||
|
||||||
|
(a_sub_part.content < b_sub_part.content ? -1 : 1)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// If both parts dynamic, check for regexp patterns
|
||||||
|
if (a_sub_part.dynamic && b_sub_part.dynamic) {
|
||||||
|
const regexp_pattern = /\((.*?)\)/;
|
||||||
|
const a_match = regexp_pattern.exec(a_sub_part.content);
|
||||||
|
const b_match = regexp_pattern.exec(b_sub_part.content);
|
||||||
|
|
||||||
|
if (!a_match && b_match) {
|
||||||
|
return 1; // No regexp, so less specific than b
|
||||||
|
}
|
||||||
|
if (!b_match && a_match) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
if (a_match && b_match && a_match[1] !== b_match[1]) {
|
||||||
|
return b_match[1].length - a_match[1].length;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function get_parts(part: string): Part[] {
|
||||||
|
return part.split(/\[(.+)\]/)
|
||||||
|
.map((str, i) => {
|
||||||
|
if (!str) return null;
|
||||||
|
const dynamic = i % 2 === 1;
|
||||||
|
|
||||||
|
const [, content, qualifier] = dynamic
|
||||||
|
? /([^(]+)(\(.+\))?$/.exec(str)
|
||||||
|
: [, str, null];
|
||||||
|
|
||||||
return {
|
return {
|
||||||
content,
|
content,
|
||||||
dynamic: i % 2 === 1
|
dynamic,
|
||||||
|
qualifier
|
||||||
};
|
};
|
||||||
})
|
})
|
||||||
.filter(Boolean);
|
.filter(Boolean);
|
||||||
|
}
|
||||||
|
|
||||||
|
function get_slug(file: string) {
|
||||||
|
return file
|
||||||
|
.replace(/[\\\/]index/, '')
|
||||||
|
.replace(/_default([\/\\index])?\.html$/, 'index')
|
||||||
|
.replace(/[\/\\]/g, '_')
|
||||||
|
.replace(/\.\w+$/, '')
|
||||||
|
.replace(/\[([^(]+)(?:\([^(]+\))?\]/, '$$$1')
|
||||||
|
.replace(/[^a-zA-Z0-9_$]/g, c => {
|
||||||
|
return c === '.' ? '_' : `$${c.charCodeAt(0)}`
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function get_pattern(segments: Part[][]) {
|
||||||
|
return new RegExp(
|
||||||
|
`^` +
|
||||||
|
segments.map(segment => {
|
||||||
|
return '\\/' + segment.map(part => {
|
||||||
|
return part.dynamic
|
||||||
|
? part.qualifier || '([^\\/]+?)'
|
||||||
|
: encodeURI(part.content.normalize())
|
||||||
|
.replace(/\?/g, '%3F')
|
||||||
|
.replace(/#/g, '%23')
|
||||||
|
.replace(/%5B/g, '[')
|
||||||
|
.replace(/%5D/g, ']');
|
||||||
|
}).join('');
|
||||||
|
}).join('') +
|
||||||
|
'\\\/?$'
|
||||||
|
);
|
||||||
}
|
}
|
||||||
@@ -1,11 +1,11 @@
|
|||||||
import * as sander from 'sander';
|
import * as fs from 'fs';
|
||||||
|
|
||||||
const previous_contents = new Map();
|
const previous_contents = new Map();
|
||||||
|
|
||||||
export function write_if_changed(file: string, code: string) {
|
export function write_if_changed(file: string, code: string) {
|
||||||
if (code !== previous_contents.get(file)) {
|
if (code !== previous_contents.get(file)) {
|
||||||
previous_contents.set(file, code);
|
previous_contents.set(file, code);
|
||||||
sander.writeFileSync(file, code);
|
fs.writeFileSync(file, code);
|
||||||
fudge_mtime(file);
|
fudge_mtime(file);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -16,8 +16,8 @@ export function posixify(file: string) {
|
|||||||
|
|
||||||
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 } = sander.statSync(file);
|
const { atime, mtime } = fs.statSync(file);
|
||||||
sander.utimesSync(
|
fs.utimesSync(
|
||||||
file,
|
file,
|
||||||
new Date(atime.getTime() - 999999),
|
new Date(atime.getTime() - 999999),
|
||||||
new Date(mtime.getTime() - 999999)
|
new Date(mtime.getTime() - 999999)
|
||||||
|
|||||||
@@ -18,4 +18,25 @@ export type Template = {
|
|||||||
|
|
||||||
export type Store = {
|
export type Store = {
|
||||||
get: () => any;
|
get: () => any;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type PageComponent = {
|
||||||
|
default?: boolean;
|
||||||
|
name: string;
|
||||||
|
file: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type Page = {
|
||||||
|
pattern: RegExp;
|
||||||
|
parts: Array<{
|
||||||
|
component: PageComponent;
|
||||||
|
params: string[];
|
||||||
|
}>
|
||||||
|
};
|
||||||
|
|
||||||
|
export type ServerRoute = {
|
||||||
|
name: string;
|
||||||
|
pattern: RegExp;
|
||||||
|
file: string;
|
||||||
|
params: string[];
|
||||||
};
|
};
|
||||||
@@ -11,13 +11,26 @@ import sourceMapSupport from 'source-map-support';
|
|||||||
|
|
||||||
sourceMapSupport.install();
|
sourceMapSupport.install();
|
||||||
|
|
||||||
type RouteObject = {
|
type ServerRoute = {
|
||||||
id: string;
|
|
||||||
type: 'page' | 'route';
|
|
||||||
pattern: RegExp;
|
pattern: RegExp;
|
||||||
|
handlers: Record<string, Handler>;
|
||||||
params: (match: RegExpMatchArray) => Record<string, string>;
|
params: (match: RegExpMatchArray) => Record<string, string>;
|
||||||
module: Component;
|
};
|
||||||
error?: 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 Handler = (req: Req, res: ServerResponse, next: () => void) => void;
|
||||||
@@ -26,6 +39,20 @@ type Store = {
|
|||||||
get: () => any
|
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 {
|
interface Req extends ClientRequest {
|
||||||
url: string;
|
url: string;
|
||||||
baseUrl: string;
|
baseUrl: string;
|
||||||
@@ -33,6 +60,7 @@ interface Req extends ClientRequest {
|
|||||||
method: string;
|
method: string;
|
||||||
path: string;
|
path: string;
|
||||||
params: Record<string, string>;
|
params: Record<string, string>;
|
||||||
|
query: Record<string, string>;
|
||||||
headers: Record<string, string>;
|
headers: Record<string, string>;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -45,24 +73,47 @@ interface Component {
|
|||||||
preload: (data: any) => any | Promise<any>
|
preload: (data: any) => any | Promise<any>
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function middleware({ App, routes, store }: {
|
const IGNORE = '__SAPPER__IGNORE__';
|
||||||
App: Component,
|
function toIgnore(uri: string, val: any) {
|
||||||
routes: RouteObject[],
|
if (Array.isArray(val)) return val.some(x => toIgnore(uri, x));
|
||||||
store: (req: Req) => Store
|
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 (!App) {
|
if (opts.routes) {
|
||||||
throw new Error(`As of 0.12, you must supply an App component to Sapper — see https://sapper.svelte.technology/guide#0-11-to-0-12 for more information`);
|
throw new Error(`As of Sapper 0.15, opts.routes should be opts.manifest`);
|
||||||
}
|
}
|
||||||
|
|
||||||
const output = locations.dest();
|
const output = locations.dest();
|
||||||
|
|
||||||
|
const { manifest, store, ignore } = opts;
|
||||||
|
|
||||||
let emitted_basepath = false;
|
let emitted_basepath = false;
|
||||||
|
|
||||||
const middleware = compose_handlers([
|
const middleware = compose_handlers([
|
||||||
|
ignore && ((req: Req, res: ServerResponse, next: () => void) => {
|
||||||
|
req[IGNORE] = toIgnore(req.path, ignore);
|
||||||
|
next();
|
||||||
|
}),
|
||||||
|
|
||||||
(req: Req, res: ServerResponse, next: () => void) => {
|
(req: Req, res: ServerResponse, next: () => void) => {
|
||||||
|
if (req[IGNORE]) return next();
|
||||||
|
|
||||||
if (req.baseUrl === undefined) {
|
if (req.baseUrl === undefined) {
|
||||||
req.baseUrl = req.originalUrl
|
let { originalUrl } = req;
|
||||||
? req.originalUrl.slice(0, -req.url.length)
|
if (req.url === '/' && originalUrl[originalUrl.length - 1] !== '/') {
|
||||||
|
originalUrl += '/';
|
||||||
|
}
|
||||||
|
|
||||||
|
req.baseUrl = originalUrl
|
||||||
|
? originalUrl.slice(0, -req.url.length)
|
||||||
: '';
|
: '';
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -100,10 +151,11 @@ export default function middleware({ App, routes, store }: {
|
|||||||
|
|
||||||
serve({
|
serve({
|
||||||
prefix: '/client/',
|
prefix: '/client/',
|
||||||
cache_control: 'max-age=31536000'
|
cache_control: dev() ? 'no-cache' : 'max-age=31536000'
|
||||||
}),
|
}),
|
||||||
|
|
||||||
get_route_handler(App, routes, store)
|
get_server_route_handler(manifest.server_routes),
|
||||||
|
get_page_handler(manifest, store)
|
||||||
].filter(Boolean));
|
].filter(Boolean));
|
||||||
|
|
||||||
return middleware;
|
return middleware;
|
||||||
@@ -127,6 +179,8 @@ function serve({ prefix, pathname, cache_control }: {
|
|||||||
: (file: string) => (cache.has(file) ? cache : cache.set(file, fs.readFileSync(path.resolve(output, file)))).get(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) => {
|
return (req: Req, res: ServerResponse, next: () => void) => {
|
||||||
|
if (req[IGNORE]) return next();
|
||||||
|
|
||||||
if (filter(req)) {
|
if (filter(req)) {
|
||||||
const type = lookup(req.path);
|
const type = lookup(req.path);
|
||||||
|
|
||||||
@@ -146,300 +200,351 @@ function serve({ prefix, pathname, cache_control }: {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
function get_route_handler(App: Component, routes: RouteObject[], store_getter: (req: Req) => Store) {
|
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 output = locations.dest();
|
||||||
|
|
||||||
const get_chunks = dev()
|
const get_build_info = dev()
|
||||||
? () => JSON.parse(fs.readFileSync(path.join(output, 'client_assets.json'), 'utf-8'))
|
? () => JSON.parse(fs.readFileSync(path.join(output, 'build.json'), 'utf-8'))
|
||||||
: (assets => () => assets)(JSON.parse(fs.readFileSync(path.join(output, 'client_assets.json'), 'utf-8')));
|
: (assets => () => assets)(JSON.parse(fs.readFileSync(path.join(output, 'build.json'), 'utf-8')));
|
||||||
|
|
||||||
const template = dev()
|
const template = dev()
|
||||||
? () => fs.readFileSync(`${locations.app()}/template.html`, 'utf-8')
|
? () => fs.readFileSync(`${locations.app()}/template.html`, 'utf-8')
|
||||||
: (str => () => str)(fs.readFileSync(`${locations.dest()}/template.html`, 'utf-8'));
|
: (str => () => str)(fs.readFileSync(`${locations.dest()}/template.html`, 'utf-8'));
|
||||||
|
|
||||||
function handle_route(route: RouteObject, req: Req, res: ServerResponse) {
|
const { server_routes, pages } = manifest;
|
||||||
req.params = route.params(route.pattern.exec(req.path));
|
const error_route = manifest.error;
|
||||||
|
|
||||||
const handlers = route.handlers[Symbol.iterator]();
|
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 next() {
|
function handle_page(page: Page, req: Req, res: ServerResponse, status = 200, error: Error | string = null) {
|
||||||
const chunks: Record<string, string> = get_chunks();
|
const build_info: {
|
||||||
|
bundler: 'rollup' | 'webpack',
|
||||||
|
shimport: string | null,
|
||||||
|
assets: Record<string, string | string[]>
|
||||||
|
} = get_build_info();
|
||||||
|
|
||||||
try {
|
res.setHeader('Content-Type', 'text/html');
|
||||||
const { value: handler, done } = handlers.next();
|
|
||||||
|
|
||||||
if (done) {
|
// preload main.js and current route
|
||||||
handle_error(req, res, 404, 'Not found');
|
// TODO detect other stuff we can preload? images, CSS, fonts?
|
||||||
return;
|
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;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const mod = handler.module;
|
return fetch(parsed.href, opts);
|
||||||
|
},
|
||||||
|
store
|
||||||
|
};
|
||||||
|
|
||||||
if (handler.type === 'page') {
|
const root_preloaded = manifest.root.preload
|
||||||
res.setHeader('Content-Type', 'text/html');
|
? manifest.root.preload.call(preload_context, {
|
||||||
|
path: req.path,
|
||||||
|
query: req.query,
|
||||||
|
params: {}
|
||||||
|
})
|
||||||
|
: {};
|
||||||
|
|
||||||
// preload main.js and current route
|
const match = error ? null : page.pattern.exec(req.path);
|
||||||
// TODO detect other stuff we can preload? images, CSS, fonts?
|
|
||||||
const link = []
|
|
||||||
.concat(chunks.main, chunks[route.id])
|
|
||||||
.filter(file => !file.match(/\.map$/))
|
|
||||||
.map(file => `<${req.baseUrl}/client/${file}>;rel="preload";as="script"`)
|
|
||||||
.join(', ');
|
|
||||||
|
|
||||||
res.setHeader('Link', link);
|
Promise.all([root_preloaded].concat(page.parts.map(part => {
|
||||||
|
if (!part) return null;
|
||||||
|
|
||||||
const store = store_getter ? store_getter(req) : null;
|
return part.component.preload
|
||||||
const props = { params: req.params, query: req.query, path: req.path };
|
? 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}`;
|
||||||
|
|
||||||
let redirect: { statusCode: number, location: string };
|
res.statusCode = redirect.statusCode;
|
||||||
let error: { statusCode: number, message: Error | string };
|
res.setHeader('Location', location);
|
||||||
|
res.end();
|
||||||
|
|
||||||
Promise.resolve(
|
if (process.send) {
|
||||||
mod.preload ? mod.preload.call({
|
process.send({
|
||||||
redirect: (statusCode: number, location: string) => {
|
__sapper__: true,
|
||||||
redirect = { statusCode, location };
|
event: 'file',
|
||||||
},
|
url: req.url,
|
||||||
error: (statusCode: number, message: Error | string) => {
|
method: req.method,
|
||||||
error = { statusCode, message };
|
status: redirect.statusCode,
|
||||||
},
|
type: 'text/html',
|
||||||
fetch: (url: string, opts?: any) => {
|
body: `<script>window.location.href = "${location}"</script>`
|
||||||
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) {
|
|
||||||
const cookies: Record<string, string> = {};
|
|
||||||
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
|
|
||||||
}, req) : {}
|
|
||||||
).catch(err => {
|
|
||||||
error = { statusCode: 500, message: err };
|
|
||||||
}).then(preloaded => {
|
|
||||||
if (redirect) {
|
|
||||||
res.statusCode = redirect.statusCode;
|
|
||||||
res.setHeader('Location', `${req.baseUrl}/${redirect.location}`);
|
|
||||||
res.end();
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (error) {
|
|
||||||
handle_error(req, res, error.statusCode, error.message);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const serialized = {
|
|
||||||
preloaded: mod.preload && try_serialize(preloaded),
|
|
||||||
store: store && try_serialize(store.get())
|
|
||||||
};
|
|
||||||
Object.assign(props, preloaded);
|
|
||||||
|
|
||||||
const { html, head, css } = App.render({ Page: mod, props }, {
|
|
||||||
store
|
|
||||||
});
|
|
||||||
|
|
||||||
let scripts = []
|
|
||||||
.concat(chunks.main) // chunks main might be an array. it might not! thanks, webpack
|
|
||||||
.filter(file => !file.match(/\.map$/))
|
|
||||||
.map(file => `<script src='${req.baseUrl}/client/${file}'></script>`)
|
|
||||||
.join('');
|
|
||||||
|
|
||||||
let inline_script = `__SAPPER__={${[
|
|
||||||
`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) {
|
|
||||||
inline_script += `if ('serviceWorker' in navigator) navigator.serviceWorker.register('${req.baseUrl}/service-worker.js');`;
|
|
||||||
}
|
|
||||||
|
|
||||||
const page = template()
|
|
||||||
.replace('%sapper.base%', `<base href="${req.baseUrl}/">`)
|
|
||||||
.replace('%sapper.scripts%', `<script>${inline_script}</script>${scripts}`)
|
|
||||||
.replace('%sapper.html%', html)
|
|
||||||
.replace('%sapper.head%', `<noscript id='sapper-head-start'></noscript>${head}<noscript id='sapper-head-end'></noscript>`)
|
|
||||||
.replace('%sapper.styles%', (css && css.code ? `<style>${css.code}</style>` : ''));
|
|
||||||
|
|
||||||
res.end(page);
|
|
||||||
|
|
||||||
if (process.send) {
|
|
||||||
process.send({
|
|
||||||
__sapper__: true,
|
|
||||||
event: 'file',
|
|
||||||
url: req.url,
|
|
||||||
method: req.method,
|
|
||||||
status: 200,
|
|
||||||
type: 'text/html',
|
|
||||||
body: page
|
|
||||||
});
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
else {
|
return;
|
||||||
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 = mod[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(new Buffer(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(new Buffer(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_bad_result = (err?: Error) => {
|
|
||||||
if (err) {
|
|
||||||
console.error(err.stack);
|
|
||||||
res.statusCode = 500;
|
|
||||||
res.end(err.message);
|
|
||||||
} else {
|
|
||||||
process.nextTick(next);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
try {
|
|
||||||
handle_method(req, res, handle_bad_result);
|
|
||||||
} catch (err) {
|
|
||||||
handle_bad_result(err);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// no matching handler for method
|
|
||||||
process.nextTick(next);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} catch (error) {
|
|
||||||
handle_error(req, res, 500, error);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
next();
|
|
||||||
}
|
|
||||||
|
|
||||||
const not_found_route = routes.find((route: RouteObject) => route.error === '4xx');
|
|
||||||
const error_route = routes.find((route: RouteObject) => route.error === '5xx');
|
|
||||||
|
|
||||||
function handle_error(req: Req, res: ServerResponse, statusCode: number, message: Error | string) {
|
|
||||||
res.statusCode = statusCode;
|
|
||||||
res.setHeader('Content-Type', 'text/html');
|
|
||||||
|
|
||||||
const error = message instanceof Error ? message : new Error(message);
|
|
||||||
|
|
||||||
const not_found = statusCode >= 400 && statusCode < 500;
|
|
||||||
|
|
||||||
const route = not_found
|
|
||||||
? not_found_route
|
|
||||||
: error_route;
|
|
||||||
|
|
||||||
function render_page({ head, css, html }) {
|
|
||||||
const page = template()
|
|
||||||
.replace('%sapper.base%', `<base href="${req.baseUrl}/">`)
|
|
||||||
.replace('%sapper.scripts%', `<script>__SAPPER__={baseUrl: "${req.baseUrl}"}</script><script src='${req.baseUrl}/client/${get_chunks().main}'></script>`)
|
|
||||||
.replace('%sapper.html%', html)
|
|
||||||
.replace('%sapper.head%', `<noscript id='sapper-head-start'></noscript>${head}<noscript id='sapper-head-end'></noscript>`)
|
|
||||||
.replace('%sapper.styles%', (css && css.code ? `<style>${css.code}</style>` : ''));
|
|
||||||
|
|
||||||
res.end(page);
|
|
||||||
}
|
|
||||||
|
|
||||||
function handle_notfound() {
|
|
||||||
const title: string = not_found
|
|
||||||
? 'Not found'
|
|
||||||
: `Internal server error: ${error.message}`;
|
|
||||||
|
|
||||||
render_page({ head: '', css: null, html: title });
|
|
||||||
}
|
|
||||||
|
|
||||||
if (route) {
|
|
||||||
const handlers = route.handlers[Symbol.iterator]();
|
|
||||||
|
|
||||||
function next() {
|
|
||||||
const { value: handler, done } = handlers.next();
|
|
||||||
|
|
||||||
if (done) {
|
|
||||||
handle_notfound();
|
|
||||||
} else if (handler.type === 'page') {
|
|
||||||
render_page(handler.module.render({
|
|
||||||
status: statusCode,
|
|
||||||
error
|
|
||||||
}, {
|
|
||||||
store: store_getter && store_getter(req)
|
|
||||||
}));
|
|
||||||
} else {
|
|
||||||
const handle_method = mod[method_export];
|
|
||||||
if (handle_method) {
|
|
||||||
handle_method(req, res, next);
|
|
||||||
} else {
|
|
||||||
next();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
next();
|
if (preload_error) {
|
||||||
} else {
|
handle_error(req, res, preload_error.statusCode, preload_error.message);
|
||||||
handle_notfound();
|
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
|
||||||
|
});
|
||||||
|
|
||||||
|
const file = [].concat(build_info.assets.main).filter(file => file && /\.js$/.test(file))[0];
|
||||||
|
const main = `${req.baseUrl}/client/${file}`;
|
||||||
|
|
||||||
|
const script = build_info.bundler === 'rollup'
|
||||||
|
? `<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);}</script>`
|
||||||
|
: `<script src="${main}"></script>`;
|
||||||
|
|
||||||
|
let inline_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) {
|
||||||
|
inline_script += `if ('serviceWorker' in navigator) navigator.serviceWorker.register('${req.baseUrl}/service-worker.js');`;
|
||||||
|
}
|
||||||
|
|
||||||
|
const body = template()
|
||||||
|
.replace('%sapper.base%', () => `<base href="${req.baseUrl}/">`)
|
||||||
|
.replace('%sapper.scripts%', () => `<script>${inline_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%', () => (css && css.code ? `<style>${css.code}</style>` : ''));
|
||||||
|
|
||||||
|
res.statusCode = status;
|
||||||
|
res.end(body);
|
||||||
|
|
||||||
|
if (process.send) {
|
||||||
|
process.send({
|
||||||
|
__sapper__: true,
|
||||||
|
event: 'file',
|
||||||
|
url: req.url,
|
||||||
|
method: req.method,
|
||||||
|
status,
|
||||||
|
type: 'text/html',
|
||||||
|
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) {
|
return function find_route(req: Req, res: ServerResponse, next: () => void) {
|
||||||
for (const route of routes) {
|
if (req[IGNORE]) return next();
|
||||||
if (!route.error && route.pattern.test(req.path)) return handle_route(route, req, res);
|
|
||||||
|
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');
|
handle_error(req, res, 404, 'Not found');
|
||||||
@@ -473,3 +578,15 @@ function try_serialize(data: any) {
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function escape_html(html: string) {
|
||||||
|
const chars: Record<string, string> = {
|
||||||
|
'"' : 'quot',
|
||||||
|
"'": '#39',
|
||||||
|
'&': 'amp',
|
||||||
|
'<' : 'lt',
|
||||||
|
'>' : 'gt'
|
||||||
|
};
|
||||||
|
|
||||||
|
return html.replace(/["'&<>]/g, c => `&${chars[c]};`);
|
||||||
|
}
|
||||||
|
|||||||
46
src/rollup.ts
Normal file
46
src/rollup.ts
Normal file
@@ -0,0 +1,46 @@
|
|||||||
|
import { locations, dev } from './config';
|
||||||
|
|
||||||
|
export default {
|
||||||
|
dev: dev(),
|
||||||
|
|
||||||
|
client: {
|
||||||
|
input: () => {
|
||||||
|
return `${locations.app()}/client.js`
|
||||||
|
},
|
||||||
|
|
||||||
|
output: () => {
|
||||||
|
return {
|
||||||
|
dir: `${locations.dest()}/client`,
|
||||||
|
entryFileNames: '[name].[hash].js',
|
||||||
|
chunkFileNames: '[name].[hash].js',
|
||||||
|
format: 'esm'
|
||||||
|
};
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
server: {
|
||||||
|
input: () => {
|
||||||
|
return `${locations.app()}/server.js`
|
||||||
|
},
|
||||||
|
|
||||||
|
output: () => {
|
||||||
|
return {
|
||||||
|
dir: locations.dest(),
|
||||||
|
format: 'cjs'
|
||||||
|
};
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
serviceworker: {
|
||||||
|
input: () => {
|
||||||
|
return `${locations.app()}/service-worker.js`;
|
||||||
|
},
|
||||||
|
|
||||||
|
output: () => {
|
||||||
|
return {
|
||||||
|
dir: locations.dest(),
|
||||||
|
format: 'iife'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
@@ -1,14 +1,39 @@
|
|||||||
import { detach, findAnchor, scroll_state, which } from './utils';
|
import { detach, findAnchor, scroll_state, which } from './utils';
|
||||||
import { Component, ComponentConstructor, Params, Query, Route, RouteData, ScrollPosition, Store, Target } from './interfaces';
|
import { Component, ComponentConstructor, Params, Query, Redirect, Manifest, RouteData, ScrollPosition, Store, Target } from './interfaces';
|
||||||
|
|
||||||
const manifest = typeof window !== 'undefined' && window.__SAPPER__;
|
const initial_data = typeof window !== 'undefined' && window.__SAPPER__;
|
||||||
|
|
||||||
export let App: ComponentConstructor;
|
export let root: Component;
|
||||||
export let component: Component;
|
|
||||||
let target: Node;
|
let target: Node;
|
||||||
let store: Store;
|
let store: Store;
|
||||||
let routes: Route[];
|
let manifest: Manifest;
|
||||||
let errors: { '4xx': Route, '5xx': Route };
|
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 : {
|
const history = typeof window !== 'undefined' ? window.history : {
|
||||||
pushState: (state: any, title: string, href: string) => {},
|
pushState: (state: any, title: string, href: string) => {},
|
||||||
@@ -26,42 +51,50 @@ if ('scrollRestoration' in history) {
|
|||||||
|
|
||||||
function select_route(url: URL): Target {
|
function select_route(url: URL): Target {
|
||||||
if (url.origin !== window.location.origin) return null;
|
if (url.origin !== window.location.origin) return null;
|
||||||
if (!url.pathname.startsWith(manifest.baseUrl)) return null;
|
if (!url.pathname.startsWith(initial_data.baseUrl)) return null;
|
||||||
|
|
||||||
const path = url.pathname.slice(manifest.baseUrl.length);
|
const path = url.pathname.slice(initial_data.baseUrl.length);
|
||||||
|
|
||||||
for (const route of routes) {
|
// avoid accidental clashes between server routes and pages
|
||||||
const match = route.pattern.exec(path);
|
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) {
|
if (match) {
|
||||||
if (route.ignore) return null;
|
|
||||||
|
|
||||||
const params = route.params(match);
|
|
||||||
|
|
||||||
const query: Record<string, string | true> = {};
|
const query: Record<string, string | true> = {};
|
||||||
if (url.search.length > 0) {
|
if (url.search.length > 0) {
|
||||||
url.search.slice(1).split('&').forEach(searchParam => {
|
url.search.slice(1).split('&').forEach(searchParam => {
|
||||||
const [, key, value] = /([^=]+)=(.*)/.exec(searchParam);
|
const [, key, value] = /([^=]+)=(.*)/.exec(searchParam);
|
||||||
query[key] = value || true;
|
query[key] = value || true;
|
||||||
})
|
});
|
||||||
}
|
}
|
||||||
return { url, route, props: { params, query, path } };
|
return { url, path, page, match, query };
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let current_token: {};
|
let current_token: {};
|
||||||
|
|
||||||
function render(Page: ComponentConstructor, props: any, scroll: ScrollPosition, token: {}) {
|
function render(data: any, nullable_depth: number, scroll: ScrollPosition, token: {}) {
|
||||||
if (current_token !== token) return;
|
if (current_token !== token) return;
|
||||||
|
|
||||||
const data = {
|
if (root) {
|
||||||
Page,
|
// first, clear out highest-level root component
|
||||||
props,
|
let level = data.child;
|
||||||
preloading: false
|
for (let i = 0; i < nullable_depth; i += 1) {
|
||||||
};
|
if (i === nullable_depth) break;
|
||||||
|
level = level.props.child;
|
||||||
|
}
|
||||||
|
|
||||||
if (component) {
|
const { component } = level;
|
||||||
component.set(data);
|
level.component = null;
|
||||||
|
root.set({ child: data.child });
|
||||||
|
|
||||||
|
// then render new stuff
|
||||||
|
level.component = component;
|
||||||
|
root.set(data);
|
||||||
} else {
|
} else {
|
||||||
// first load — remove SSR'd <head> contents
|
// first load — remove SSR'd <head> contents
|
||||||
const start = document.querySelector('#sapper-head-start');
|
const start = document.querySelector('#sapper-head-start');
|
||||||
@@ -73,7 +106,9 @@ function render(Page: ComponentConstructor, props: any, scroll: ScrollPosition,
|
|||||||
detach(end);
|
detach(end);
|
||||||
}
|
}
|
||||||
|
|
||||||
component = new App({
|
Object.assign(data, root_data);
|
||||||
|
|
||||||
|
root = new manifest.root({
|
||||||
target,
|
target,
|
||||||
data,
|
data,
|
||||||
store,
|
store,
|
||||||
@@ -84,56 +119,155 @@ function render(Page: ComponentConstructor, props: any, scroll: ScrollPosition,
|
|||||||
if (scroll) {
|
if (scroll) {
|
||||||
window.scrollTo(scroll.x, scroll.y);
|
window.scrollTo(scroll.x, scroll.y);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Object.assign(root_props, data);
|
||||||
|
ready = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
function prepare_route(Page: ComponentConstructor, props: RouteData) {
|
function changed(a: Record<string, string | true>, b: Record<string, string | true>) {
|
||||||
let redirect: { statusCode: number, location: string } = null;
|
return JSON.stringify(a) !== JSON.stringify(b);
|
||||||
|
}
|
||||||
|
|
||||||
|
let root_preload: Promise<any>;
|
||||||
|
let root_data: any;
|
||||||
|
|
||||||
|
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;
|
let error: { statusCode: number, message: Error | string } = null;
|
||||||
|
|
||||||
if (!Page.preload) {
|
const preload_context = {
|
||||||
return { Page, props, redirect, error };
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!component && manifest.preloaded) {
|
|
||||||
return { Page, props: Object.assign(props, manifest.preloaded), redirect, error };
|
|
||||||
}
|
|
||||||
|
|
||||||
if (component) {
|
|
||||||
component.set({
|
|
||||||
preloading: true
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
return Promise.resolve(Page.preload.call({
|
|
||||||
store,
|
store,
|
||||||
fetch: (url: string, opts?: any) => window.fetch(url, opts),
|
fetch: (url: string, opts?: any) => window.fetch(url, opts),
|
||||||
redirect: (statusCode: number, location: string) => {
|
redirect: (statusCode: number, location: string) => {
|
||||||
|
if (redirect && (redirect.statusCode !== statusCode || redirect.location !== location)) {
|
||||||
|
throw new Error(`Conflicting redirects`);
|
||||||
|
}
|
||||||
redirect = { statusCode, location };
|
redirect = { statusCode, location };
|
||||||
},
|
},
|
||||||
error: (statusCode: number, message: Error | string) => {
|
error: (statusCode: number, message: Error | string) => {
|
||||||
error = { statusCode, message };
|
error = { statusCode, message };
|
||||||
}
|
}
|
||||||
}, props)).catch(err => {
|
};
|
||||||
error = { statusCode: 500, message: err };
|
|
||||||
}).then(preloaded => {
|
|
||||||
if (error) {
|
|
||||||
const route = error.statusCode >= 400 && error.statusCode < 500
|
|
||||||
? errors['4xx']
|
|
||||||
: errors['5xx'];
|
|
||||||
|
|
||||||
return route.load().then(({ default: Page }: { default: ComponentConstructor }) => {
|
if (!root_preload) {
|
||||||
const err = error.message instanceof Error ? error.message : new Error(error.message);
|
root_preload = manifest.root.preload
|
||||||
Object.assign(props, { status: error.statusCode, error: err });
|
? initial_data.preloaded[0] || manifest.root.preload.call(preload_context, {
|
||||||
return { Page, props, redirect: null };
|
path,
|
||||||
});
|
query,
|
||||||
|
params: {}
|
||||||
|
})
|
||||||
|
: {};
|
||||||
|
}
|
||||||
|
|
||||||
|
return Promise.all(page.parts.map(async (part, i) => {
|
||||||
|
if (i < changed_from) return null;
|
||||||
|
if (!part) return null;
|
||||||
|
|
||||||
|
const { default: Component } = await 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 };
|
||||||
}
|
}
|
||||||
|
|
||||||
Object.assign(props, preloaded)
|
segments = new_segments;
|
||||||
return { Page, props, redirect };
|
|
||||||
|
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 };
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function navigate(target: Target, id: number) {
|
async function navigate(target: Target, id: number): Promise<any> {
|
||||||
if (id) {
|
if (id) {
|
||||||
// popstate or initial navigation
|
// popstate or initial navigation
|
||||||
cid = id;
|
cid = id;
|
||||||
@@ -147,22 +281,24 @@ function navigate(target: Target, id: number) {
|
|||||||
|
|
||||||
cid = id;
|
cid = id;
|
||||||
|
|
||||||
|
if (root) {
|
||||||
|
root.set({ preloading: true });
|
||||||
|
}
|
||||||
const loaded = prefetching && prefetching.href === target.url.href ?
|
const loaded = prefetching && prefetching.href === target.url.href ?
|
||||||
prefetching.promise :
|
prefetching.promise :
|
||||||
target.route.load().then(mod => prepare_route(mod.default, target.props));
|
prepare_page(target);
|
||||||
|
|
||||||
prefetching = null;
|
prefetching = null;
|
||||||
|
|
||||||
const token = current_token = {};
|
const token = current_token = {};
|
||||||
|
const { redirect, data, nullable_depth } = await loaded;
|
||||||
|
|
||||||
return loaded.then(({ Page, props, redirect }) => {
|
if (redirect) {
|
||||||
if (redirect) {
|
await goto(redirect.location, { replaceState: true });
|
||||||
return goto(redirect.location, { replaceState: true });
|
} else {
|
||||||
}
|
render(data, nullable_depth, scroll_history[id], token);
|
||||||
|
if (document.activeElement) document.activeElement.blur();
|
||||||
render(Page, props, scroll_history[id], token);
|
}
|
||||||
document.activeElement.blur();
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function handle_click(event: MouseEvent) {
|
function handle_click(event: MouseEvent) {
|
||||||
@@ -212,7 +348,11 @@ function handle_popstate(event: PopStateEvent) {
|
|||||||
if (event.state) {
|
if (event.state) {
|
||||||
const url = new URL(window.location.href);
|
const url = new URL(window.location.href);
|
||||||
const target = select_route(url);
|
const target = select_route(url);
|
||||||
navigate(target, event.state.id);
|
if (target) {
|
||||||
|
navigate(target, event.state.id);
|
||||||
|
} else {
|
||||||
|
window.location.href = window.location.href;
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
// hashchange
|
// hashchange
|
||||||
cid = ++uid;
|
cid = ++uid;
|
||||||
@@ -222,16 +362,16 @@ function handle_popstate(event: PopStateEvent) {
|
|||||||
|
|
||||||
let prefetching: {
|
let prefetching: {
|
||||||
href: string;
|
href: string;
|
||||||
promise: Promise<{ Page: ComponentConstructor, props: any }>;
|
promise: Promise<{ redirect?: Redirect, data?: any, nullable_depth?: number }>;
|
||||||
} = null;
|
} = null;
|
||||||
|
|
||||||
export function prefetch(href: string) {
|
export function prefetch(href: string) {
|
||||||
const selected = select_route(new URL(href, document.baseURI));
|
const target: Target = select_route(new URL(href, document.baseURI));
|
||||||
|
|
||||||
if (selected && (!prefetching || href !== prefetching.href)) {
|
if (target && (!prefetching || href !== prefetching.href)) {
|
||||||
prefetching = {
|
prefetching = {
|
||||||
href,
|
href,
|
||||||
promise: selected.route.load().then(mod => prepare_route(mod.default, selected.props))
|
promise: prepare_page(target)
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -253,22 +393,28 @@ function trigger_prefetch(event: MouseEvent | TouchEvent) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
let inited: boolean;
|
let inited: boolean;
|
||||||
|
let ready = false;
|
||||||
|
|
||||||
export function init(opts: { App: ComponentConstructor, target: Node, routes: Route[], store?: (data: any) => Store }) {
|
export function init(opts: {
|
||||||
|
App: ComponentConstructor,
|
||||||
|
target: Node,
|
||||||
|
manifest: Manifest,
|
||||||
|
store?: (data: any) => Store,
|
||||||
|
routes?: any // legacy
|
||||||
|
}) {
|
||||||
if (opts instanceof HTMLElement) {
|
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`);
|
throw new Error(`The signature of init(...) has changed — see https://sapper.svelte.technology/guide#0-11-to-0-12 for more information`);
|
||||||
}
|
}
|
||||||
|
|
||||||
App = opts.App;
|
if (opts.routes) {
|
||||||
|
throw new Error(`As of Sapper 0.15, opts.routes should be opts.manifest`);
|
||||||
|
}
|
||||||
|
|
||||||
target = opts.target;
|
target = opts.target;
|
||||||
routes = opts.routes.filter(r => !r.error);
|
manifest = opts.manifest;
|
||||||
errors = {
|
|
||||||
'4xx': opts.routes.find(r => r.error === '4xx'),
|
|
||||||
'5xx': opts.routes.find(r => r.error === '5xx')
|
|
||||||
};
|
|
||||||
|
|
||||||
if (opts && opts.store) {
|
if (opts && opts.store) {
|
||||||
store = opts.store(manifest.store);
|
store = opts.store(initial_data.store);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!inited) { // this check makes HMR possible
|
if (!inited) { // this check makes HMR possible
|
||||||
@@ -292,33 +438,35 @@ export function init(opts: { App: ComponentConstructor, target: Node, routes: Ro
|
|||||||
|
|
||||||
history.replaceState({ id: uid }, '', href);
|
history.replaceState({ id: uid }, '', href);
|
||||||
|
|
||||||
const target = select_route(new URL(window.location.href));
|
if (!initial_data.error) {
|
||||||
return navigate(target, uid);
|
const target = select_route(new URL(window.location.href));
|
||||||
|
if (target) return navigate(target, uid);
|
||||||
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
export function goto(href: string, opts = { replaceState: false }) {
|
export function goto(href: string, opts = { replaceState: false }) {
|
||||||
const target = select_route(new URL(href, document.baseURI));
|
const target = select_route(new URL(href, document.baseURI));
|
||||||
|
let promise;
|
||||||
|
|
||||||
if (target) {
|
if (target) {
|
||||||
navigate(target, null);
|
promise = navigate(target, null);
|
||||||
if (history) history[opts.replaceState ? 'replaceState' : 'pushState']({ id: cid }, '', href);
|
if (history) history[opts.replaceState ? 'replaceState' : 'pushState']({ id: cid }, '', href);
|
||||||
} else {
|
} else {
|
||||||
window.location.href = href;
|
window.location.href = href;
|
||||||
|
promise = new Promise(f => {}); // never resolves
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return promise;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function prefetchRoutes(pathnames: string[]) {
|
export function prefetchRoutes(pathnames: string[]) {
|
||||||
if (!routes) throw new Error(`You must call init() first`);
|
if (!manifest) throw new Error(`You must call init() first`);
|
||||||
|
|
||||||
return routes
|
return manifest.pages
|
||||||
.filter(route => {
|
.filter(route => {
|
||||||
if (!pathnames) return true;
|
if (!pathnames) return true;
|
||||||
return pathnames.some(pathname => {
|
return pathnames.some(pathname => route.pattern.test(pathname));
|
||||||
return route.error
|
|
||||||
? route.error === pathname
|
|
||||||
: route.pattern.test(pathname)
|
|
||||||
});
|
|
||||||
})
|
})
|
||||||
.reduce((promise: Promise<any>, route) => {
|
.reduce((promise: Promise<any>, route) => {
|
||||||
return promise.then(route.load);
|
return promise.then(route.load);
|
||||||
|
|||||||
@@ -11,15 +11,23 @@ export interface ComponentConstructor {
|
|||||||
};
|
};
|
||||||
|
|
||||||
export interface Component {
|
export interface Component {
|
||||||
|
set: (data: any) => void;
|
||||||
destroy: () => void;
|
destroy: () => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
export type Route = {
|
export type Page = {
|
||||||
pattern: RegExp;
|
pattern: RegExp;
|
||||||
load: () => Promise<{ default: ComponentConstructor }>;
|
parts: Array<{
|
||||||
error?: string;
|
component: () => Promise<{ default: ComponentConstructor }>;
|
||||||
params?: (match: RegExpExecArray) => Record<string, string>;
|
params?: (match: RegExpExecArray) => Record<string, string>;
|
||||||
ignore?: boolean;
|
}>;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type Manifest = {
|
||||||
|
ignore: RegExp[];
|
||||||
|
root: ComponentConstructor;
|
||||||
|
error: () => Promise<{ default: ComponentConstructor }>;
|
||||||
|
pages: Page[]
|
||||||
};
|
};
|
||||||
|
|
||||||
export type ScrollPosition = {
|
export type ScrollPosition = {
|
||||||
@@ -29,6 +37,13 @@ export type ScrollPosition = {
|
|||||||
|
|
||||||
export type Target = {
|
export type Target = {
|
||||||
url: URL;
|
url: URL;
|
||||||
route: Route;
|
path: string;
|
||||||
props: RouteData;
|
page: Page;
|
||||||
|
match: RegExpExecArray;
|
||||||
|
query: Record<string, string | true>;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type Redirect = {
|
||||||
|
statusCode: number;
|
||||||
|
location: string;
|
||||||
};
|
};
|
||||||
10
src/utils.ts
Normal file
10
src/utils.ts
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
export function left_pad(str: string, len: number) {
|
||||||
|
while (str.length < len) str = ` ${str}`;
|
||||||
|
return str;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function repeat(str: string, i: number) {
|
||||||
|
let result = '';
|
||||||
|
while (i--) result += str;
|
||||||
|
return result;
|
||||||
|
}
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
{#if preloading}
|
|
||||||
<progress class='preloading-progress' value=0.5/>
|
|
||||||
{/if}
|
|
||||||
|
|
||||||
<svelte:component this={Page} {...props}/>
|
|
||||||
|
|
||||||
@@ -1,13 +1,11 @@
|
|||||||
import { init, prefetchRoutes } from '../../../runtime.js';
|
import { init, prefetchRoutes } from '../../../runtime.js';
|
||||||
import { Store } from 'svelte/store.js';
|
import { Store } from 'svelte/store.js';
|
||||||
import { routes } from './manifest/client.js';
|
import { manifest } from './manifest/client.js';
|
||||||
import App from './App.html';
|
|
||||||
|
|
||||||
window.init = () => {
|
window.init = () => {
|
||||||
return init({
|
return init({
|
||||||
target: document.querySelector('#sapper'),
|
target: document.querySelector('#sapper'),
|
||||||
App,
|
manifest,
|
||||||
routes,
|
|
||||||
store: data => new Store(data)
|
store: data => new Store(data)
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -2,10 +2,9 @@ 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.ts.js';
|
import sapper from '../../../dist/middleware.js';
|
||||||
import { Store } from 'svelte/store.js';
|
import { Store } from 'svelte/store.js';
|
||||||
import { routes } from './manifest/server.js';
|
import { manifest } from './manifest/server.js';
|
||||||
import App from './App.html'
|
|
||||||
|
|
||||||
let pending;
|
let pending;
|
||||||
let ended;
|
let ended;
|
||||||
@@ -86,15 +85,27 @@ const middlewares = [
|
|||||||
next();
|
next();
|
||||||
},
|
},
|
||||||
|
|
||||||
|
// set up some values for the store
|
||||||
|
(req, res, next) => {
|
||||||
|
req.hello = 'hello';
|
||||||
|
res.locals = { name: 'world' };
|
||||||
|
next();
|
||||||
|
},
|
||||||
|
|
||||||
sapper({
|
sapper({
|
||||||
App,
|
manifest,
|
||||||
routes,
|
store: (req, res) => {
|
||||||
store: () => {
|
|
||||||
return new Store({
|
return new Store({
|
||||||
title: 'Stored title'
|
title: `${req.hello} ${res.locals.name}`
|
||||||
});
|
});
|
||||||
}
|
},
|
||||||
})
|
ignore: [
|
||||||
|
/foobar/i,
|
||||||
|
'/buzz',
|
||||||
|
'fizz',
|
||||||
|
x => x === '/hello'
|
||||||
|
]
|
||||||
|
}),
|
||||||
];
|
];
|
||||||
|
|
||||||
if (BASEPATH) {
|
if (BASEPATH) {
|
||||||
@@ -103,4 +114,8 @@ if (BASEPATH) {
|
|||||||
app.use(...middlewares);
|
app.use(...middlewares);
|
||||||
}
|
}
|
||||||
|
|
||||||
app.listen(PORT);
|
['foobar', 'buzz', 'fizzer', 'hello'].forEach(uri => {
|
||||||
|
app.get('/'+uri, (req, res) => res.end(uri));
|
||||||
|
});
|
||||||
|
|
||||||
|
app.listen(PORT);
|
||||||
|
|||||||
@@ -1,6 +0,0 @@
|
|||||||
<svelte:head>
|
|
||||||
<title>Internal server error</title>
|
|
||||||
</svelte:head>
|
|
||||||
|
|
||||||
<h1>Internal server error</h1>
|
|
||||||
<p>{error.message}</p>
|
|
||||||
20
test/app/routes/[x]/[y]/[z].html
Normal file
20
test/app/routes/[x]/[y]/[z].html
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
<span>z: {segment} {count}</span>
|
||||||
|
<a href="foo/bar/qux"></a>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import counts from '../_counts.js';
|
||||||
|
|
||||||
|
export default {
|
||||||
|
preload() {
|
||||||
|
return {
|
||||||
|
count: counts.z += 1
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
oncreate() {
|
||||||
|
this.set({
|
||||||
|
segment: this.get().params.z
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
</script>
|
||||||
22
test/app/routes/[x]/[y]/_layout.html
Normal file
22
test/app/routes/[x]/[y]/_layout.html
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
<span>y: {segment} {count}</span>
|
||||||
|
<svelte:component this={child.component} {...child.props}/>
|
||||||
|
|
||||||
|
<span>child segment: {child.segment}</span>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import counts from '../_counts.js';
|
||||||
|
|
||||||
|
export default {
|
||||||
|
preload() {
|
||||||
|
return {
|
||||||
|
count: counts.y += 1
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
oncreate() {
|
||||||
|
this.set({
|
||||||
|
segment: this.get().params.y
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
</script>
|
||||||
5
test/app/routes/[x]/_counts.js
Normal file
5
test/app/routes/[x]/_counts.js
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
export default {
|
||||||
|
x: process.browser ? 1 : 0,
|
||||||
|
y: process.browser ? 1 : 0,
|
||||||
|
z: process.browser ? 1 : 0
|
||||||
|
};
|
||||||
@@ -2,5 +2,5 @@
|
|||||||
<title>{status}</title>
|
<title>{status}</title>
|
||||||
</svelte:head>
|
</svelte:head>
|
||||||
|
|
||||||
<h1>Not found</h1>
|
<h1>{status}</h1>
|
||||||
<p>{error.message}</p>
|
<p>{error.message}</p>
|
||||||
15
test/app/routes/_layout.html
Normal file
15
test/app/routes/_layout.html
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
{#if preloading}
|
||||||
|
<progress class='preloading-progress' value=0.5/>
|
||||||
|
{/if}
|
||||||
|
|
||||||
|
<svelte:component this={child.component} {rootPreloadFunctionRan} {...child.props}/>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
export default {
|
||||||
|
preload() {
|
||||||
|
return {
|
||||||
|
rootPreloadFunctionRan: true
|
||||||
|
};
|
||||||
|
}
|
||||||
|
};
|
||||||
|
</script>
|
||||||
@@ -6,15 +6,21 @@
|
|||||||
|
|
||||||
<p>This is the 'about' page. There's not much here.</p>
|
<p>This is the 'about' page. There's not much here.</p>
|
||||||
|
|
||||||
<button class='goto' on:click='goto("blog/what-is-sapper")'>What is Sapper?</button>
|
|
||||||
<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 { goto, prefetch } from '../../../runtime.js';
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
|
oncreate() {
|
||||||
|
window.goto = goto;
|
||||||
|
},
|
||||||
|
|
||||||
|
ondestroy() {
|
||||||
|
window.goto = null;
|
||||||
|
},
|
||||||
|
|
||||||
methods: {
|
methods: {
|
||||||
goto,
|
|
||||||
prefetch
|
prefetch
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import posts from './blog/_posts.js';
|
import posts from './_posts.js';
|
||||||
|
|
||||||
const contents = JSON.stringify(posts.map(post => {
|
const contents = JSON.stringify(posts.map(post => {
|
||||||
return {
|
return {
|
||||||
@@ -8,6 +8,7 @@
|
|||||||
<a href='about'>about</a>
|
<a href='about'>about</a>
|
||||||
<a href='slow-preload'>slow preload</a>
|
<a href='slow-preload'>slow preload</a>
|
||||||
<a href='redirect-from'>redirect</a>
|
<a href='redirect-from'>redirect</a>
|
||||||
|
<a href='redirect-root'>redirect (root)</a>
|
||||||
<a href='blog/nope'>broken link</a>
|
<a href='blog/nope'>broken link</a>
|
||||||
<a href='blog/throw-an-error'>error link</a>
|
<a href='blog/throw-an-error'>error link</a>
|
||||||
<a href='credentials?creds=include'>credentials</a>
|
<a href='credentials?creds=include'>credentials</a>
|
||||||
|
|||||||
1
test/app/routes/missing-index/ok.html
Normal file
1
test/app/routes/missing-index/ok.html
Normal file
@@ -0,0 +1 @@
|
|||||||
|
<h1>it works</h1>
|
||||||
1
test/app/routes/preload-root/index.html
Normal file
1
test/app/routes/preload-root/index.html
Normal file
@@ -0,0 +1 @@
|
|||||||
|
<h1>root preload function ran: {rootPreloadFunctionRan}</h1>
|
||||||
1
test/app/routes/preload-values/index.html
Normal file
1
test/app/routes/preload-values/index.html
Normal file
@@ -0,0 +1 @@
|
|||||||
|
<svelte:component this={child.component} {...child.props}/>
|
||||||
7
test/app/routes/redirect-root.html
Normal file
7
test/app/routes/redirect-root.html
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
<script>
|
||||||
|
export default {
|
||||||
|
preload() {
|
||||||
|
this.redirect(301, '/');
|
||||||
|
}
|
||||||
|
};
|
||||||
|
</script>
|
||||||
@@ -1,9 +0,0 @@
|
|||||||
<p>URL is {url}</p>
|
|
||||||
|
|
||||||
<script>
|
|
||||||
export default {
|
|
||||||
preload({ url }) {
|
|
||||||
if (url) return { url };
|
|
||||||
}
|
|
||||||
};
|
|
||||||
</script>
|
|
||||||
9
test/app/routes/unsafe-replacement.html
Normal file
9
test/app/routes/unsafe-replacement.html
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
$&
|
||||||
|
|
||||||
|
<script>
|
||||||
|
export default {
|
||||||
|
preload() {
|
||||||
|
return '$&';
|
||||||
|
}
|
||||||
|
};
|
||||||
|
</script>
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
const config = require('../../../webpack/config.js');
|
const config = require('../../../config/webpack.js');
|
||||||
const webpack = require('webpack');
|
const webpack = require('webpack');
|
||||||
|
|
||||||
const mode = process.env.NODE_ENV;
|
const mode = process.env.NODE_ENV;
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
const config = require('../../../webpack/config.js');
|
const config = require('../../../config/webpack.js');
|
||||||
const sapper_pkg = require('../../../package.json');
|
const sapper_pkg = require('../../../package.json');
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
const config = require('../../../webpack/config.js');
|
const config = require('../../../config/webpack.js');
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
entry: config.serviceworker.entry(),
|
entry: config.serviceworker.entry(),
|
||||||
|
|||||||
@@ -32,6 +32,8 @@ Nightmare.action('prefetchRoutes', function(done) {
|
|||||||
|
|
||||||
const cli = path.resolve(__dirname, '../../sapper');
|
const cli = path.resolve(__dirname, '../../sapper');
|
||||||
|
|
||||||
|
const wait = ms => new Promise(f => setTimeout(f, ms));
|
||||||
|
|
||||||
describe('sapper', function() {
|
describe('sapper', function() {
|
||||||
process.chdir(path.resolve(__dirname, '../app'));
|
process.chdir(path.resolve(__dirname, '../app'));
|
||||||
|
|
||||||
@@ -41,7 +43,7 @@ describe('sapper', function() {
|
|||||||
rimraf.sync('.sapper');
|
rimraf.sync('.sapper');
|
||||||
rimraf.sync('start.js');
|
rimraf.sync('start.js');
|
||||||
|
|
||||||
this.timeout(process.env.CI ? 30000 : 10000);
|
this.timeout(process.env.CI ? 30000 : 15000);
|
||||||
|
|
||||||
// TODO reinstate dev tests
|
// TODO reinstate dev tests
|
||||||
// run({
|
// run({
|
||||||
@@ -57,9 +59,19 @@ describe('sapper', function() {
|
|||||||
basepath: '/custom-basepath'
|
basepath: '/custom-basepath'
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('export', () => {
|
testExport({});
|
||||||
|
|
||||||
|
testExport({ basepath: '/custom-basepath' });
|
||||||
|
});
|
||||||
|
|
||||||
|
function testExport({ basepath = '' }) {
|
||||||
|
describe(basepath ? `export --basepath ${basepath}` : 'export', () => {
|
||||||
before(() => {
|
before(() => {
|
||||||
return exec(`node ${cli} export`);
|
if (basepath) {
|
||||||
|
process.env.BASEPATH = basepath;
|
||||||
|
}
|
||||||
|
|
||||||
|
return exec(`node ${cli} export ${basepath ? `--basepath ${basepath}` : ''}`);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('export all pages', () => {
|
it('export all pages', () => {
|
||||||
@@ -94,16 +106,18 @@ describe('sapper', function() {
|
|||||||
'service-worker.js',
|
'service-worker.js',
|
||||||
'svelte-logo-192.png',
|
'svelte-logo-192.png',
|
||||||
'svelte-logo-512.png',
|
'svelte-logo-512.png',
|
||||||
];
|
].map(file => {
|
||||||
|
return basepath ? `${basepath.replace(/^[\/\\]/, '')}/${file}` : file;
|
||||||
|
});
|
||||||
|
|
||||||
// Client scripts that should show up in the extraction directory.
|
// Client scripts that should show up in the extraction directory.
|
||||||
const expectedClientRegexes = [
|
const expectedClientRegexes = [
|
||||||
/client\/[^/]+\/_(\.\d+)?\.js/,
|
|
||||||
/client\/[^/]+\/about(\.\d+)?\.js/,
|
|
||||||
/client\/[^/]+\/blog_\$slug\$(\.\d+)?\.js/,
|
|
||||||
/client\/[^/]+\/blog(\.\d+)?\.js/,
|
|
||||||
/client\/[^/]+\/main(\.\d+)?\.js/,
|
/client\/[^/]+\/main(\.\d+)?\.js/,
|
||||||
/client\/[^/]+\/show_url(\.\d+)?\.js/,
|
/client\/[^/]+\/index(\.\d+)?\.js/,
|
||||||
/client\/[^/]+\/slow_preload(\.\d+)?\.js/,
|
/client\/[^/]+\/about(\.\d+)?\.js/,
|
||||||
|
/client\/[^/]+\/blog_\$slug(\.\d+)?\.js/,
|
||||||
|
/client\/[^/]+\/blog(\.\d+)?\.js/,
|
||||||
|
/client\/[^/]+\/slow\$45preload(\.\d+)?\.js/,
|
||||||
];
|
];
|
||||||
const allPages = walkSync(dest);
|
const allPages = walkSync(dest);
|
||||||
|
|
||||||
@@ -125,7 +139,7 @@ describe('sapper', function() {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
}
|
||||||
|
|
||||||
function run({ mode, basepath = '' }) {
|
function run({ mode, basepath = '' }) {
|
||||||
describe(`mode=${mode}`, function () {
|
describe(`mode=${mode}`, function () {
|
||||||
@@ -266,8 +280,9 @@ function run({ mode, basepath = '' }) {
|
|||||||
})
|
})
|
||||||
.then(requests => {
|
.then(requests => {
|
||||||
assert.deepEqual(requests.map(r => r.url), []);
|
assert.deepEqual(requests.map(r => r.url), []);
|
||||||
return nightmare.path();
|
|
||||||
})
|
})
|
||||||
|
.then(() => wait(100))
|
||||||
|
.then(() => nightmare.path())
|
||||||
.then(path => {
|
.then(path => {
|
||||||
assert.equal(path, `${basepath}/about`);
|
assert.equal(path, `${basepath}/about`);
|
||||||
return nightmare.title();
|
return nightmare.title();
|
||||||
@@ -281,9 +296,7 @@ function run({ mode, basepath = '' }) {
|
|||||||
return nightmare
|
return nightmare
|
||||||
.goto(`${base}/about`)
|
.goto(`${base}/about`)
|
||||||
.init()
|
.init()
|
||||||
.click('.goto')
|
.evaluate(() => window.goto('blog/what-is-sapper'))
|
||||||
.wait(url => window.location.pathname === url, `${basepath}/blog/what-is-sapper`)
|
|
||||||
.wait(100)
|
|
||||||
.title()
|
.title()
|
||||||
.then(title => {
|
.then(title => {
|
||||||
assert.equal(title, 'What is Sapper?');
|
assert.equal(title, 'What is Sapper?');
|
||||||
@@ -369,16 +382,6 @@ function run({ mode, basepath = '' }) {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
it('passes entire request object to preload', () => {
|
|
||||||
return nightmare
|
|
||||||
.goto(`${base}/show-url`)
|
|
||||||
.init()
|
|
||||||
.evaluate(() => document.querySelector('p').innerHTML)
|
|
||||||
.then(html => {
|
|
||||||
assert.equal(html, `URL is /show-url`);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('calls a delete handler', () => {
|
it('calls a delete handler', () => {
|
||||||
return nightmare
|
return nightmare
|
||||||
.goto(`${base}/delete-test`)
|
.goto(`${base}/delete-test`)
|
||||||
@@ -433,6 +436,33 @@ function run({ mode, basepath = '' }) {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('redirects on server (root)', () => {
|
||||||
|
return nightmare.goto(`${base}/redirect-root`)
|
||||||
|
.path()
|
||||||
|
.then(path => {
|
||||||
|
assert.equal(path, `${basepath}/`);
|
||||||
|
})
|
||||||
|
.then(() => nightmare.page.title())
|
||||||
|
.then(title => {
|
||||||
|
assert.equal(title, 'Great success!');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('redirects in client (root)', () => {
|
||||||
|
return nightmare.goto(base)
|
||||||
|
.wait('[href="redirect-root"]')
|
||||||
|
.click('[href="redirect-root"]')
|
||||||
|
.wait(200)
|
||||||
|
.path()
|
||||||
|
.then(path => {
|
||||||
|
assert.equal(path, `${basepath}/`);
|
||||||
|
})
|
||||||
|
.then(() => nightmare.page.title())
|
||||||
|
.then(title => {
|
||||||
|
assert.equal(title, 'Great success!');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
it('handles 4xx error on server', () => {
|
it('handles 4xx error on server', () => {
|
||||||
return nightmare.goto(`${base}/blog/nope`)
|
return nightmare.goto(`${base}/blog/nope`)
|
||||||
.path()
|
.path()
|
||||||
@@ -441,7 +471,7 @@ function run({ mode, basepath = '' }) {
|
|||||||
})
|
})
|
||||||
.then(() => nightmare.page.title())
|
.then(() => nightmare.page.title())
|
||||||
.then(title => {
|
.then(title => {
|
||||||
assert.equal(title, 'Not found')
|
assert.equal(title, '404')
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -456,7 +486,7 @@ function run({ mode, basepath = '' }) {
|
|||||||
})
|
})
|
||||||
.then(() => nightmare.page.title())
|
.then(() => nightmare.page.title())
|
||||||
.then(title => {
|
.then(title => {
|
||||||
assert.equal(title, 'Not found');
|
assert.equal(title, '404');
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -468,7 +498,7 @@ function run({ mode, basepath = '' }) {
|
|||||||
})
|
})
|
||||||
.then(() => nightmare.page.title())
|
.then(() => nightmare.page.title())
|
||||||
.then(title => {
|
.then(title => {
|
||||||
assert.equal(title, 'Internal server error')
|
assert.equal(title, '500')
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -483,10 +513,46 @@ function run({ mode, basepath = '' }) {
|
|||||||
})
|
})
|
||||||
.then(() => nightmare.page.title())
|
.then(() => nightmare.page.title())
|
||||||
.then(title => {
|
.then(title => {
|
||||||
assert.equal(title, 'Internal server error');
|
assert.equal(title, '500');
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// Ignores are meant for top-level escape.
|
||||||
|
// ~> Sapper **should** own the entire {basepath} when designated.
|
||||||
|
if (!basepath) {
|
||||||
|
it('respects `options.ignore` values (RegExp)', () => {
|
||||||
|
return nightmare.goto(`${base}/foobar`)
|
||||||
|
.evaluate(() => document.documentElement.textContent)
|
||||||
|
.then(text => {
|
||||||
|
assert.equal(text, 'foobar');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('respects `options.ignore` values (String #1)', () => {
|
||||||
|
return nightmare.goto(`${base}/buzz`)
|
||||||
|
.evaluate(() => document.documentElement.textContent)
|
||||||
|
.then(text => {
|
||||||
|
assert.equal(text, 'buzz');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('respects `options.ignore` values (String #2)', () => {
|
||||||
|
return nightmare.goto(`${base}/fizzer`)
|
||||||
|
.evaluate(() => document.documentElement.textContent)
|
||||||
|
.then(text => {
|
||||||
|
assert.equal(text, 'fizzer');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('respects `options.ignore` values (Function)', () => {
|
||||||
|
return nightmare.goto(`${base}/hello`)
|
||||||
|
.evaluate(() => document.documentElement.textContent)
|
||||||
|
.then(text => {
|
||||||
|
assert.equal(text, 'hello');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
it('does not attempt client-side navigation to server routes', () => {
|
it('does not attempt client-side navigation to server routes', () => {
|
||||||
return nightmare.goto(`${base}/blog/how-is-sapper-different-from-next`)
|
return nightmare.goto(`${base}/blog/how-is-sapper-different-from-next`)
|
||||||
.init()
|
.init()
|
||||||
@@ -542,11 +608,11 @@ function run({ mode, basepath = '' }) {
|
|||||||
return nightmare.goto(`${base}/store`)
|
return nightmare.goto(`${base}/store`)
|
||||||
.page.title()
|
.page.title()
|
||||||
.then(title => {
|
.then(title => {
|
||||||
assert.equal(title, 'Stored title');
|
assert.equal(title, 'hello world');
|
||||||
return nightmare.init().page.title();
|
return nightmare.init().page.title();
|
||||||
})
|
})
|
||||||
.then(title => {
|
.then(title => {
|
||||||
assert.equal(title, 'Stored title');
|
assert.equal(title, 'hello world');
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -621,6 +687,59 @@ function run({ mode, basepath = '' }) {
|
|||||||
assert.equal(name, 'BODY');
|
assert.equal(name, 'BODY');
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('replaces %sapper.xxx% tags safely', () => {
|
||||||
|
return nightmare
|
||||||
|
.goto(`${base}/unsafe-replacement`)
|
||||||
|
.init()
|
||||||
|
.page.html()
|
||||||
|
.then(html => {
|
||||||
|
assert.equal(html.indexOf('%sapper'), -1);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('only recreates components when necessary', () => {
|
||||||
|
return nightmare
|
||||||
|
.goto(`${base}/foo/bar/baz`)
|
||||||
|
.init()
|
||||||
|
.evaluate(() => document.querySelector('#sapper').textContent)
|
||||||
|
.then(text => {
|
||||||
|
assert.deepEqual(text.split('\n').filter(Boolean), [
|
||||||
|
'y: bar 1',
|
||||||
|
'z: baz 1',
|
||||||
|
'child segment: baz'
|
||||||
|
]);
|
||||||
|
|
||||||
|
return nightmare.click(`a`)
|
||||||
|
.then(() => wait(100))
|
||||||
|
.then(() => {
|
||||||
|
return nightmare.evaluate(() => document.querySelector('#sapper').textContent);
|
||||||
|
});
|
||||||
|
})
|
||||||
|
.then(text => {
|
||||||
|
assert.deepEqual(text.split('\n').filter(Boolean), [
|
||||||
|
'y: bar 1',
|
||||||
|
'z: qux 2',
|
||||||
|
'child segment: qux'
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('uses a fallback index component if none is provided', () => {
|
||||||
|
return nightmare.goto(`${base}/missing-index/ok`)
|
||||||
|
.page.title()
|
||||||
|
.then(title => {
|
||||||
|
assert.equal(title, 'it works');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('runs preload in root component', () => {
|
||||||
|
return nightmare.goto(`${base}/preload-root`)
|
||||||
|
.page.title()
|
||||||
|
.then(title => {
|
||||||
|
assert.equal(title, 'root preload function ran: true');
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('headers', () => {
|
describe('headers', () => {
|
||||||
@@ -633,7 +752,7 @@ function run({ mode, basepath = '' }) {
|
|||||||
'text/html'
|
'text/html'
|
||||||
);
|
);
|
||||||
|
|
||||||
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"`;
|
||||||
})
|
})
|
||||||
|
|||||||
26
test/unit/clean_html/index.ts
Normal file
26
test/unit/clean_html/index.ts
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
import * as fs from 'fs';
|
||||||
|
import * as path from 'path';
|
||||||
|
import * as assert from 'assert';
|
||||||
|
import clean_html from '../../../src/api/utils/clean_html';
|
||||||
|
|
||||||
|
describe('clean_html', () => {
|
||||||
|
const samples = path.join(__dirname, 'samples');
|
||||||
|
|
||||||
|
fs.readdirSync(samples).forEach(dir => {
|
||||||
|
if (dir[0] === '.') return;
|
||||||
|
|
||||||
|
it(dir, () => {
|
||||||
|
const input = fs.readFileSync(`${samples}/${dir}/input.html`, 'utf-8');
|
||||||
|
const expected = fs.readFileSync(`${samples}/${dir}/output.html`, 'utf-8');
|
||||||
|
|
||||||
|
const actual = clean_html(input);
|
||||||
|
|
||||||
|
fs.writeFileSync(`${samples}/${dir}/.actual.html`, actual);
|
||||||
|
|
||||||
|
assert.equal(
|
||||||
|
actual.replace(/\s+$/gm, ''),
|
||||||
|
expected.replace(/\s+$/gm, '')
|
||||||
|
);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
14
test/unit/clean_html/samples/removes-cdata/.actual.html
Normal file
14
test/unit/clean_html/samples/removes-cdata/.actual.html
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
<!doctype html>
|
||||||
|
<html lang="en">
|
||||||
|
<head>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<math>
|
||||||
|
<ms></ms>
|
||||||
|
<mo>+</mo>
|
||||||
|
<mn>3</mn>
|
||||||
|
<mo>=</mo>
|
||||||
|
<ms></ms>
|
||||||
|
</math>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
14
test/unit/clean_html/samples/removes-cdata/input.html
Normal file
14
test/unit/clean_html/samples/removes-cdata/input.html
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
<!doctype html>
|
||||||
|
<html lang="en">
|
||||||
|
<head>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<math>
|
||||||
|
<ms><![CDATA[x<y]]></ms>
|
||||||
|
<mo>+</mo>
|
||||||
|
<mn>3</mn>
|
||||||
|
<mo>=</mo>
|
||||||
|
<ms><![CDATA[x<y3]]></ms>
|
||||||
|
</math>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
14
test/unit/clean_html/samples/removes-cdata/output.html
Normal file
14
test/unit/clean_html/samples/removes-cdata/output.html
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
<!doctype html>
|
||||||
|
<html lang="en">
|
||||||
|
<head>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<math>
|
||||||
|
<ms></ms>
|
||||||
|
<mo>+</mo>
|
||||||
|
<mn>3</mn>
|
||||||
|
<mo>=</mo>
|
||||||
|
<ms></ms>
|
||||||
|
</math>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
@@ -0,0 +1,9 @@
|
|||||||
|
<!doctype html>
|
||||||
|
<html lang="en">
|
||||||
|
<head>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<a href="keep-me">keep me</a>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
9
test/unit/clean_html/samples/removes-comments/input.html
Normal file
9
test/unit/clean_html/samples/removes-comments/input.html
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
<!doctype html>
|
||||||
|
<html lang="en">
|
||||||
|
<head>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<a href="keep-me">keep me</a>
|
||||||
|
<!-- <a href="delete-me">delete me</a> -->
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
@@ -0,0 +1,9 @@
|
|||||||
|
<!doctype html>
|
||||||
|
<html lang="en">
|
||||||
|
<head>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<a href="keep-me">keep me</a>
|
||||||
|
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
@@ -0,0 +1,12 @@
|
|||||||
|
<!doctype html>
|
||||||
|
<html lang="en">
|
||||||
|
<head>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<script></script>
|
||||||
|
|
||||||
|
<script></script>
|
||||||
|
|
||||||
|
<script src="attributes-are-preserved.js"></script>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
@@ -0,0 +1,16 @@
|
|||||||
|
<!doctype html>
|
||||||
|
<html lang="en">
|
||||||
|
<head>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<script>
|
||||||
|
console.log('this should be deleted');
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
console.log('so should this');
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<script src="attributes-are-preserved.js"></script>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
@@ -0,0 +1,12 @@
|
|||||||
|
<!doctype html>
|
||||||
|
<html lang="en">
|
||||||
|
<head>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<script></script>
|
||||||
|
|
||||||
|
<script></script>
|
||||||
|
|
||||||
|
<script src="attributes-are-preserved.js"></script>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
@@ -1,287 +0,0 @@
|
|||||||
const assert = require('assert');
|
|
||||||
const { create_routes } = require('../../dist/core.ts.js');
|
|
||||||
|
|
||||||
describe('create_routes', () => {
|
|
||||||
it('sorts handlers correctly', () => {
|
|
||||||
const routes = create_routes({
|
|
||||||
files: ['foo.html', 'foo.js']
|
|
||||||
});
|
|
||||||
|
|
||||||
assert.deepEqual(
|
|
||||||
routes.map(r => r.handlers),
|
|
||||||
[
|
|
||||||
[
|
|
||||||
{
|
|
||||||
type: 'route',
|
|
||||||
file: 'foo.js'
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: 'page',
|
|
||||||
file: 'foo.html'
|
|
||||||
}
|
|
||||||
]
|
|
||||||
]
|
|
||||||
)
|
|
||||||
});
|
|
||||||
|
|
||||||
it('encodes characters not allowed in path', () => {
|
|
||||||
const routes = create_routes({
|
|
||||||
files: [
|
|
||||||
'"',
|
|
||||||
'#',
|
|
||||||
'?'
|
|
||||||
]
|
|
||||||
});
|
|
||||||
|
|
||||||
assert.deepEqual(
|
|
||||||
routes.map(r => r.pattern),
|
|
||||||
[
|
|
||||||
/^\/%22\/?$/,
|
|
||||||
/^\/%23\/?$/,
|
|
||||||
/^\/%3F\/?$/
|
|
||||||
]
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('sorts routes correctly', () => {
|
|
||||||
const routes = create_routes({
|
|
||||||
files: ['index.html', 'about.html', 'post/f[xx].html', '[wildcard].html', 'post/foo.html', 'post/[id].html', 'post/bar.html', 'post/[id].json.js']
|
|
||||||
});
|
|
||||||
|
|
||||||
assert.deepEqual(
|
|
||||||
routes.map(r => r.handlers[0].file),
|
|
||||||
[
|
|
||||||
'index.html',
|
|
||||||
'about.html',
|
|
||||||
'post/bar.html',
|
|
||||||
'post/foo.html',
|
|
||||||
'post/f[xx].html',
|
|
||||||
'post/[id].json.js',
|
|
||||||
'post/[id].html',
|
|
||||||
'[wildcard].html'
|
|
||||||
]
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('prefers index page to nested route', () => {
|
|
||||||
let routes = create_routes({
|
|
||||||
files: [
|
|
||||||
'api/examples/[slug].js',
|
|
||||||
'api/examples/index.js',
|
|
||||||
'blog/[slug].html',
|
|
||||||
'api/gists/[id].js',
|
|
||||||
'api/gists/index.js',
|
|
||||||
'4xx.html',
|
|
||||||
'5xx.html',
|
|
||||||
'blog/index.html',
|
|
||||||
'blog/rss.xml.js',
|
|
||||||
'guide/index.html',
|
|
||||||
'index.html'
|
|
||||||
]
|
|
||||||
});
|
|
||||||
|
|
||||||
assert.deepEqual(
|
|
||||||
routes.map(r => r.handlers[0].file),
|
|
||||||
[
|
|
||||||
'4xx.html',
|
|
||||||
'5xx.html',
|
|
||||||
'index.html',
|
|
||||||
'guide/index.html',
|
|
||||||
'blog/index.html',
|
|
||||||
'blog/rss.xml.js',
|
|
||||||
'blog/[slug].html',
|
|
||||||
'api/examples/index.js',
|
|
||||||
'api/examples/[slug].js',
|
|
||||||
'api/gists/index.js',
|
|
||||||
'api/gists/[id].js',
|
|
||||||
]
|
|
||||||
);
|
|
||||||
|
|
||||||
routes = create_routes({
|
|
||||||
files: [
|
|
||||||
'4xx.html',
|
|
||||||
'5xx.html',
|
|
||||||
'api/blog/[slug].js',
|
|
||||||
'api/blog/index.js',
|
|
||||||
'api/guide/contents.js',
|
|
||||||
'api/guide/index.js',
|
|
||||||
'blog/[slug].html',
|
|
||||||
'blog/index.html',
|
|
||||||
'blog/rss.xml.js',
|
|
||||||
'gist/[id].js',
|
|
||||||
'gist/create.js',
|
|
||||||
'guide/index.html',
|
|
||||||
'index.html',
|
|
||||||
'repl/index.html'
|
|
||||||
]
|
|
||||||
});
|
|
||||||
|
|
||||||
assert.deepEqual(
|
|
||||||
routes.map(r => r.handlers[0].file),
|
|
||||||
[
|
|
||||||
'4xx.html',
|
|
||||||
'5xx.html',
|
|
||||||
'index.html',
|
|
||||||
'guide/index.html',
|
|
||||||
'blog/index.html',
|
|
||||||
'blog/rss.xml.js',
|
|
||||||
'blog/[slug].html',
|
|
||||||
'gist/create.js',
|
|
||||||
'gist/[id].js',
|
|
||||||
'repl/index.html',
|
|
||||||
'api/guide/index.js',
|
|
||||||
'api/guide/contents.js',
|
|
||||||
'api/blog/index.js',
|
|
||||||
'api/blog/[slug].js',
|
|
||||||
]
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('generates params', () => {
|
|
||||||
const routes = create_routes({
|
|
||||||
files: ['index.html', 'about.html', '[wildcard].html', 'post/[id].html']
|
|
||||||
});
|
|
||||||
|
|
||||||
let file;
|
|
||||||
let params;
|
|
||||||
for (let i = 0; i < routes.length; i += 1) {
|
|
||||||
const route = routes[i];
|
|
||||||
if (params = route.exec('/post/123')) {
|
|
||||||
file = route.handlers[0].file;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
assert.equal(file, 'post/[id].html');
|
|
||||||
assert.deepEqual(params, {
|
|
||||||
id: '123'
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('ignores files and directories with leading underscores', () => {
|
|
||||||
const routes = create_routes({
|
|
||||||
files: ['index.html', '_foo.html', 'a/_b/c/d.html', 'e/f/g/h.html', 'i/_j.html']
|
|
||||||
});
|
|
||||||
|
|
||||||
assert.deepEqual(
|
|
||||||
routes.map(r => r.handlers[0].file),
|
|
||||||
[
|
|
||||||
'index.html',
|
|
||||||
'e/f/g/h.html'
|
|
||||||
]
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('ignores files and directories with leading dots except .well-known', () => {
|
|
||||||
const routes = create_routes({
|
|
||||||
files: ['.well-known', '.unknown']
|
|
||||||
});
|
|
||||||
|
|
||||||
assert.deepEqual(
|
|
||||||
routes.map(r => r.handlers[0].file),
|
|
||||||
['.well-known']
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('matches /foo/:bar before /:baz/qux', () => {
|
|
||||||
const a = create_routes({
|
|
||||||
files: ['foo/[bar].html', '[baz]/qux.html']
|
|
||||||
});
|
|
||||||
const b = create_routes({
|
|
||||||
files: ['[baz]/qux.html', 'foo/[bar].html']
|
|
||||||
});
|
|
||||||
|
|
||||||
assert.deepEqual(
|
|
||||||
a.map(r => r.handlers[0].file),
|
|
||||||
['foo/[bar].html', '[baz]/qux.html']
|
|
||||||
);
|
|
||||||
|
|
||||||
assert.deepEqual(
|
|
||||||
b.map(r => r.handlers[0].file),
|
|
||||||
['foo/[bar].html', '[baz]/qux.html']
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('fails if routes are indistinguishable', () => {
|
|
||||||
assert.throws(() => {
|
|
||||||
create_routes({
|
|
||||||
files: ['[foo].html', '[bar]/index.html']
|
|
||||||
});
|
|
||||||
}, /The \[foo\] and \[bar\]\/index routes clash/);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('matches nested routes', () => {
|
|
||||||
const route = create_routes({
|
|
||||||
files: ['settings/[submenu].html']
|
|
||||||
})[0];
|
|
||||||
|
|
||||||
assert.deepEqual(route.exec('/settings/foo'), {
|
|
||||||
submenu: 'foo'
|
|
||||||
});
|
|
||||||
|
|
||||||
assert.deepEqual(route.exec('/settings'), {
|
|
||||||
submenu: null
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('prefers index routes to nested routes', () => {
|
|
||||||
const routes = create_routes({
|
|
||||||
files: ['settings/[submenu].html', 'settings.html']
|
|
||||||
});
|
|
||||||
|
|
||||||
assert.deepEqual(
|
|
||||||
routes.map(r => r.handlers[0].file),
|
|
||||||
['settings.html', 'settings/[submenu].html']
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('matches deeply nested routes', () => {
|
|
||||||
const route = create_routes({
|
|
||||||
files: ['settings/[a]/[b]/index.html']
|
|
||||||
})[0];
|
|
||||||
|
|
||||||
assert.deepEqual(route.exec('/settings/foo/bar'), {
|
|
||||||
a: 'foo',
|
|
||||||
b: 'bar'
|
|
||||||
});
|
|
||||||
|
|
||||||
assert.deepEqual(route.exec('/settings/foo'), {
|
|
||||||
a: 'foo',
|
|
||||||
b: null
|
|
||||||
});
|
|
||||||
|
|
||||||
assert.deepEqual(route.exec('/settings'), {
|
|
||||||
a: null,
|
|
||||||
b: null
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('matches a dynamic part within a part', () => {
|
|
||||||
const route = create_routes({
|
|
||||||
files: ['things/[slug].json.js']
|
|
||||||
})[0];
|
|
||||||
|
|
||||||
assert.deepEqual(route.exec('/things/foo.json'), {
|
|
||||||
slug: 'foo'
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('matches multiple dynamic parts within a part', () => {
|
|
||||||
const route = create_routes({
|
|
||||||
files: ['things/[id]_[slug].json.js']
|
|
||||||
})[0];
|
|
||||||
|
|
||||||
assert.deepEqual(route.exec('/things/someid_someslug.json'), {
|
|
||||||
id: 'someid',
|
|
||||||
slug: 'someslug'
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('fails if dynamic params are not separated', () => {
|
|
||||||
assert.throws(() => {
|
|
||||||
create_routes({
|
|
||||||
files: ['[foo][bar].js']
|
|
||||||
});
|
|
||||||
}, /Invalid route \[foo\]\[bar\]\.js — parameters must be separated/);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
157
test/unit/create_routes/index.ts
Normal file
157
test/unit/create_routes/index.ts
Normal file
@@ -0,0 +1,157 @@
|
|||||||
|
import * as path from 'path';
|
||||||
|
import * as assert from 'assert';
|
||||||
|
import create_routes from '../../../src/core/create_routes';
|
||||||
|
|
||||||
|
describe('create_routes', () => {
|
||||||
|
it('creates routes', () => {
|
||||||
|
const { components, pages, server_routes } = create_routes(path.join(__dirname, 'samples/basic'));
|
||||||
|
|
||||||
|
const index = { name: 'index', file: 'index.html' };
|
||||||
|
const about = { name: 'about', file: 'about.html' };
|
||||||
|
const blog = { name: 'blog', file: 'blog/index.html' };
|
||||||
|
const blog_$slug = { name: 'blog_$slug', file: 'blog/[slug].html' };
|
||||||
|
|
||||||
|
assert.deepEqual(components, [
|
||||||
|
index,
|
||||||
|
about,
|
||||||
|
blog,
|
||||||
|
blog_$slug
|
||||||
|
]);
|
||||||
|
|
||||||
|
assert.deepEqual(pages, [
|
||||||
|
{
|
||||||
|
pattern: /^\/?$/,
|
||||||
|
parts: [
|
||||||
|
{ component: index, params: [] }
|
||||||
|
]
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
pattern: /^\/about\/?$/,
|
||||||
|
parts: [
|
||||||
|
{ component: about, params: [] }
|
||||||
|
]
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
pattern: /^\/blog\/?$/,
|
||||||
|
parts: [
|
||||||
|
null,
|
||||||
|
{ component: blog, params: [] }
|
||||||
|
]
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
pattern: /^\/blog\/([^\/]+?)\/?$/,
|
||||||
|
parts: [
|
||||||
|
null,
|
||||||
|
{ component: blog_$slug, params: ['slug'] }
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]);
|
||||||
|
|
||||||
|
assert.deepEqual(server_routes, [
|
||||||
|
{
|
||||||
|
name: 'route_blog_json',
|
||||||
|
pattern: /^\/blog.json\/?$/,
|
||||||
|
file: 'blog/index.json.js',
|
||||||
|
params: []
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
name: 'route_blog_$slug_json',
|
||||||
|
pattern: /^\/blog\/([^\/]+?).json\/?$/,
|
||||||
|
file: 'blog/[slug].json.js',
|
||||||
|
params: ['slug']
|
||||||
|
}
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('encodes invalid characters', () => {
|
||||||
|
const { components, pages } = create_routes(path.join(__dirname, 'samples/encoding'));
|
||||||
|
|
||||||
|
// had to remove ? and " because windows
|
||||||
|
|
||||||
|
// const quote = { name: '$34', file: '".html' };
|
||||||
|
const hash = { name: '$35', file: '#.html' };
|
||||||
|
// const question_mark = { name: '$63', file: '?.html' };
|
||||||
|
|
||||||
|
assert.deepEqual(components, [
|
||||||
|
// quote,
|
||||||
|
hash,
|
||||||
|
// question_mark
|
||||||
|
]);
|
||||||
|
|
||||||
|
assert.deepEqual(pages.map(p => p.pattern), [
|
||||||
|
// /^\/%22\/?$/,
|
||||||
|
/^\/%23\/?$/,
|
||||||
|
// /^\/%3F\/?$/
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('allows regex qualifiers', () => {
|
||||||
|
const { pages } = create_routes(path.join(__dirname, 'samples/qualifiers'));
|
||||||
|
|
||||||
|
assert.deepEqual(pages.map(p => p.pattern), [
|
||||||
|
/^\/([0-9-a-z]{3,})\/?$/,
|
||||||
|
/^\/([a-z]{2})\/?$/,
|
||||||
|
/^\/([^\/]+?)\/?$/
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('sorts routes correctly', () => {
|
||||||
|
const { pages } = create_routes(path.join(__dirname, 'samples/sorting'));
|
||||||
|
|
||||||
|
assert.deepEqual(pages.map(p => p.parts.map(part => part && part.component.file)), [
|
||||||
|
['index.html'],
|
||||||
|
['about.html'],
|
||||||
|
[null, 'post/index.html'],
|
||||||
|
[null, 'post/bar.html'],
|
||||||
|
[null, 'post/foo.html'],
|
||||||
|
[null, 'post/f[xx].html'],
|
||||||
|
[null, 'post/[id([0-9-a-z]{3,})].html'],
|
||||||
|
[null, 'post/[id].html'],
|
||||||
|
['[wildcard].html']
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('ignores files and directories with leading underscores', () => {
|
||||||
|
const { server_routes } = create_routes(path.join(__dirname, 'samples/hidden-underscore'));
|
||||||
|
|
||||||
|
assert.deepEqual(server_routes.map(r => r.file), [
|
||||||
|
'index.js',
|
||||||
|
'e/f/g/h.js'
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('ignores files and directories with leading dots except .well-known', () => {
|
||||||
|
const { server_routes } = create_routes(path.join(__dirname, 'samples/hidden-dot'));
|
||||||
|
|
||||||
|
assert.deepEqual(server_routes.map(r => r.file), [
|
||||||
|
'.well-known/dnt-policy.txt.js'
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('fails on clashes', () => {
|
||||||
|
assert.throws(() => {
|
||||||
|
const { pages } = create_routes(path.join(__dirname, 'samples/clash-pages'));
|
||||||
|
}, /The \[bar\]\/index\.html and \[foo\]\.html pages clash/);
|
||||||
|
|
||||||
|
assert.throws(() => {
|
||||||
|
const { server_routes } = create_routes(path.join(__dirname, 'samples/clash-routes'));
|
||||||
|
console.log(server_routes);
|
||||||
|
}, /The \[bar\]\/index\.js and \[foo\]\.js routes clash/);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('fails if dynamic params are not separated', () => {
|
||||||
|
assert.throws(() => {
|
||||||
|
create_routes(path.join(__dirname, 'samples/invalid-params'));
|
||||||
|
}, /Invalid route \[foo\]\[bar\]\.js — parameters must be separated/);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('errors when trying to use reserved characters in route regexp', () => {
|
||||||
|
assert.throws(() => {
|
||||||
|
create_routes(path.join(__dirname, 'samples/invalid-qualifier'));
|
||||||
|
}, /Invalid route \[foo\(\[a-z\]\(\[0-9\]\)\)\].js — cannot use \(, \), \? or \: in route qualifiers/);
|
||||||
|
});
|
||||||
|
});
|
||||||
0
test/unit/create_routes/samples/basic/about.html
Normal file
0
test/unit/create_routes/samples/basic/about.html
Normal file
0
test/unit/create_routes/samples/basic/index.html
Normal file
0
test/unit/create_routes/samples/basic/index.html
Normal file
0
test/unit/create_routes/samples/encoding/#.html
Normal file
0
test/unit/create_routes/samples/encoding/#.html
Normal file
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user