Compare commits

...

40 Commits

Author SHA1 Message Date
Rich Harris
9cf90ce01d -> v0.21.0 2018-09-23 21:48:53 -04:00
Rich Harris
e7f9ddae86 Merge pull request #446 from mrkishi/escaping-issues
Fix filenames
2018-09-23 21:47:28 -04:00
Rich Harris
ffa1e1f704 Merge branch 'master' into escaping-issues 2018-09-23 21:41:02 -04:00
Rich Harris
80bb958b47 Merge pull request #443 from sveltejs/gh-432
update folder structure
2018-09-23 21:39:05 -04:00
mrkishi
532f559fc5 Update package-lock.json 2018-09-21 15:40:12 -03:00
mrkishi
0bd1b0b8e2 Fix filename escaping issues 2018-09-21 15:34:59 -03:00
Rich Harris
10c5ff4169 remove test app manifest from repo 2018-09-19 17:51:56 -04:00
Rich Harris
273823dfd7 move config into single file, add errors to help people migrate 2018-09-19 16:27:30 -04:00
Rich Harris
8f064fe5ac update folder structure (#432) 2018-09-19 12:02:11 -04:00
Rich Harris
f29e7efbd6 const -> let 2018-09-19 11:25:04 -04:00
Rich Harris
e66e3cd7eb -> v0.20.4 2018-09-19 11:11:42 -04:00
Rich Harris
ff415b391b Merge pull request #436 from nsivertsen/devtools
Enable debugging in Chrome and VS Code - fixes #435
2018-09-19 11:09:19 -04:00
Rich Harris
91182ad0a2 Merge pull request #441 from silentworks/bugfix/legacy-manifest
Fix for legacy manifest file
2018-09-19 10:43:50 -04:00
Andrew Smith
467041a3cd Fix for legacy manifest file 2018-09-13 19:45:30 +01:00
Nikolai Sivertsen
520949c5e1 Enable debugging in Chrome and VS Code - fixes 435 2018-09-12 16:55:45 +02:00
Rich Harris
8c07d9d2ac -> v0.20.3 2018-09-08 10:24:21 -04:00
Rich Harris
7bd684a80e Merge pull request #428 from nolanlawson/cache-control-service-worker
Set service-worker max-age to 0
2018-09-08 10:20:04 -04:00
Rich Harris
cbb5e8755b Use MDN recommendation for preventing SW caching 2018-09-08 10:12:55 -04:00
Rich Harris
7ef72dbb77 Merge pull request #429 from nolanlawson/consistent-cache-control
Use consistent cache-control:max-age=600 for HTML pages
2018-09-08 09:40:56 -04:00
Nolan Lawson
87ff9c2aeb fix test 2018-09-07 17:25:17 -07:00
Nolan Lawson
2d1f535314 use consistent cache-control:max-age=600 for HTML pages 2018-09-07 16:46:40 -07:00
Rich Harris
cd1b53b80d Merge pull request #424 from nolanlawson/csp-nonce
Allow scripts to contain a CSP nonce
2018-09-07 19:46:07 -04:00
Rich Harris
0a7be736c0 snake_case 2018-09-07 19:45:40 -04:00
Nolan Lawson
5ee53a98c6 set service-worker max-age to 0 2018-09-07 16:28:38 -07:00
Rich Harris
0e8ed6612c -> v0.20.2 2018-09-07 18:19:39 -04:00
Rich Harris
5ec748b95d Merge pull request #427 from sveltejs/gh-426
handle value-less query string parameters
2018-09-07 18:18:28 -04:00
Rich Harris
64b16715cd handle value-less query string parameters - fixes #426 2018-09-07 18:02:58 -04:00
Rich Harris
9ea5e5e251 Merge pull request #425 from nolanlawson/cache-control-immutable
Add cache-control:immutable for immutable assets
2018-09-07 17:12:23 -04:00
Rich Harris
68b78f56d6 remove unused file 2018-09-07 16:42:56 -04:00
Nolan Lawson
68e93a8fa0 add cache-control:immutable for immutable assets 2018-09-07 11:07:50 -07:00
Nolan Lawson
e377515867 allow scripts to contain a CSP nonce 2018-09-07 09:46:53 -07:00
Rich Harris
99ae39b8a8 -> v0.20.1 2018-09-06 18:57:11 -04:00
Rich Harris
1b489f4687 -> v0.20.0 2018-09-04 07:35:24 -04:00
Rich Harris
91f2c6e49c Merge pull request #418 from nsivertsen/sourcemaps
Enable source maps by default in dev mode when using rollup
2018-09-04 07:32:15 -04:00
Rich Harris
f5e07e9f78 Merge pull request #419 from sveltejs/gh-417
decode req.params
2018-09-04 07:31:53 -04:00
Rich Harris
17297a9794 Merge pull request #414 from sveltejs/gh-347-b
decode req.path during export
2018-09-04 07:31:36 -04:00
Rich Harris
9ef4f33e38 decode query params 2018-09-03 20:09:25 -04:00
Rich Harris
30966ee7f2 decode req.params - fixes #417 2018-09-03 18:36:07 -04:00
Nikolai Sivertsen
ae90f774e1 Enable source maps by default in dev mode when using rollup 2018-09-04 00:26:18 +02:00
Rich Harris
0706b5f50a decode req.path during export 2018-09-03 09:02:13 -04:00
75 changed files with 419 additions and 193 deletions

2
.gitignore vendored
View File

@@ -4,7 +4,7 @@ yarn-error.log
node_modules node_modules
cypress/screenshots cypress/screenshots
test/app/.sapper test/app/.sapper
test/app/app/manifest test/app/src/manifest
test/app/export test/app/export
test/app/build test/app/build
sapper sapper

View File

@@ -1,5 +1,36 @@
# sapper changelog # sapper changelog
## 0.21.0
* Change project folder structure ([#432](https://github.com/sveltejs/sapper/issues/432))
* Escape filenames ([#446](https://github.com/sveltejs/sapper/pull/446/))
## 0.20.4
* Fix legacy build CSS ([#439](https://github.com/sveltejs/sapper/issues/439))
* Enable debugging in Chrome and VSCode ([#435](https://github.com/sveltejs/sapper/issues/435))
## 0.20.3
* Inject `nonce` attribute if `res.locals.nonce` is present ([#424](https://github.com/sveltejs/sapper/pull/424))
* Prevent service worker caching ([#428](https://github.com/sveltejs/sapper/pull/428))
* Consistent caching for HTML responses ([#429](https://github.com/sveltejs/sapper/pull/429))
## 0.20.2
* Add `immutable` cache control header for hashed assets ([#425](https://github.com/sveltejs/sapper/pull/425))
* Handle value-less query string params ([#426](https://github.com/sveltejs/sapper/issues/426))
## 0.20.1
* Update shimport
## 0.20.0
* Decode `req.params` and `req.query` ([#417](https://github.com/sveltejs/sapper/issues/417))
* Decode URLs before writing files in `sapper export` ([#414](https://github.com/sveltejs/sapper/pull/414))
* Generate server sourcemaps for Rollup apps in dev mode ([#418](https://github.com/sveltejs/sapper/pull/418))
## 0.19.3 ## 0.19.3
* Better unicode route handling ([#347](https://github.com/sveltejs/sapper/issues/347)) * Better unicode route handling ([#347](https://github.com/sveltejs/sapper/issues/347))

View File

@@ -1 +0,0 @@
<svelte:component this={child.component} {...child.props}/>

8
package-lock.json generated
View File

@@ -1,6 +1,6 @@
{ {
"name": "sapper", "name": "sapper",
"version": "0.19.0", "version": "0.20.4",
"lockfileVersion": 1, "lockfileVersion": 1,
"requires": true, "requires": true,
"dependencies": { "dependencies": {
@@ -6287,9 +6287,9 @@
} }
}, },
"shimport": { "shimport": {
"version": "0.0.10", "version": "0.0.11",
"resolved": "https://registry.npmjs.org/shimport/-/shimport-0.0.10.tgz", "resolved": "https://registry.npmjs.org/shimport/-/shimport-0.0.11.tgz",
"integrity": "sha512-3xPFDLmcLj87sx0OwA60qbloMQUsu6VGF97IG4RqxTf91sGeiaaXOPxM1PoQHbaTm4TOhH8zosokqLAZtuNGnA==" "integrity": "sha512-wRlG/wMuV/czrzJEWBUPjydU/Ve0kTrTH8wHLRjuY6S2BDB+qDDXkTY/WrNc/7t5jnd0LPVO1sRIE7Ga6uXTpw=="
}, },
"signal-exit": { "signal-exit": {
"version": "3.0.2", "version": "3.0.2",

View File

@@ -1,6 +1,6 @@
{ {
"name": "sapper", "name": "sapper",
"version": "0.19.3", "version": "0.21.0",
"description": "Military-grade apps, engineered by Svelte", "description": "Military-grade apps, engineered by Svelte",
"main": "dist/middleware.js", "main": "dist/middleware.js",
"bin": { "bin": {
@@ -20,7 +20,7 @@
}, },
"dependencies": { "dependencies": {
"html-minifier": "^3.5.16", "html-minifier": "^3.5.16",
"shimport": "^0.0.10", "shimport": "0.0.11",
"source-map-support": "^0.5.6", "source-map-support": "^0.5.6",
"sourcemap-codec": "^1.4.1", "sourcemap-codec": "^1.4.1",
"string-hash": "^1.1.3", "string-hash": "^1.1.3",
@@ -74,6 +74,7 @@
"test": "mocha --opts mocha.opts", "test": "mocha --opts mocha.opts",
"pretest": "npm run build", "pretest": "npm run build",
"build": "rm -rf dist && rollup -c", "build": "rm -rf dist && rollup -c",
"prepare": "npm run build",
"dev": "rollup -cw", "dev": "rollup -cw",
"prepublishOnly": "npm test", "prepublishOnly": "npm test",
"update_mime_types": "curl http://svn.apache.org/repos/asf/httpd/httpd/trunk/docs/conf/mime.types | grep -e \"^[^#]\" > src/middleware/mime-types.md" "update_mime_types": "curl http://svn.apache.org/repos/asf/httpd/httpd/trunk/docs/conf/mime.types | grep -e \"^[^#]\" > src/middleware/mime-types.md"

View File

@@ -11,6 +11,7 @@ import * as events from './interfaces';
import { copy_shimport } from './utils/copy_shimport'; import { copy_shimport } from './utils/copy_shimport';
import { Dirs, PageComponent } from '../interfaces'; import { Dirs, PageComponent } from '../interfaces';
import { CompileResult } from '../core/create_compilers/interfaces'; import { CompileResult } from '../core/create_compilers/interfaces';
import read_template from '../core/read_template';
type Opts = { type Opts = {
legacy: boolean; legacy: boolean;
@@ -39,9 +40,9 @@ async function execute(emitter: EventEmitter, opts: Opts, dirs: Dirs) {
mkdirp.sync(`${dirs.dest}/client`); mkdirp.sync(`${dirs.dest}/client`);
copy_shimport(dirs.dest); copy_shimport(dirs.dest);
// minify app/template.html // minify src/template.html
// TODO compile this to a function? could be quicker than str.replace(...).replace(...).replace(...) // TODO compile this to a function? could be quicker than str.replace(...).replace(...).replace(...)
const template = fs.readFileSync(`${dirs.app}/template.html`, 'utf-8'); const template = read_template();
// remove this in a future version // remove this in a future version
if (template.indexOf('%sapper.base%') === -1) { if (template.indexOf('%sapper.base%') === -1) {
@@ -54,10 +55,10 @@ async function execute(emitter: EventEmitter, opts: Opts, dirs: Dirs) {
const manifest_data = create_manifest_data(); const manifest_data = create_manifest_data();
// create app/manifest/client.js and app/manifest/server.js // create src/manifest/client.js and src/manifest/server.js
create_main_manifests({ bundler: opts.bundler, manifest_data }); create_main_manifests({ bundler: opts.bundler, manifest_data });
const { client, server, serviceworker } = create_compilers(opts.bundler, dirs); const { client, server, serviceworker } = await create_compilers(opts.bundler, dirs);
const client_result = await client.compile(); const client_result = await client.compile();
emitter.emit('build', <events.BuildEvent>{ emitter.emit('build', <events.BuildEvent>{
@@ -70,7 +71,7 @@ async function execute(emitter: EventEmitter, opts: Opts, dirs: Dirs) {
if (opts.legacy) { if (opts.legacy) {
process.env.SAPPER_LEGACY_BUILD = 'true'; process.env.SAPPER_LEGACY_BUILD = 'true';
const { client } = create_compilers(opts.bundler, dirs); const { client } = await create_compilers(opts.bundler, dirs);
const client_result = await client.compile(); const client_result = await client.compile();
@@ -80,6 +81,7 @@ async function execute(emitter: EventEmitter, opts: Opts, dirs: Dirs) {
result: client_result result: client_result
}); });
client_result.to_json(manifest_data, dirs);
build_info.legacy_assets = client_result.assets; build_info.legacy_assets = client_result.assets;
delete process.env.SAPPER_LEGACY_BUILD; delete process.env.SAPPER_LEGACY_BUILD;
} }

View File

@@ -15,6 +15,7 @@ import * as events from './interfaces';
import validate_bundler from '../cli/utils/validate_bundler'; import validate_bundler from '../cli/utils/validate_bundler';
import { copy_shimport } from './utils/copy_shimport'; import { copy_shimport } from './utils/copy_shimport';
import { ManifestData } from '../interfaces'; import { ManifestData } from '../interfaces';
import read_template from '../core/read_template';
export function dev(opts) { export function dev(opts) {
return new Watcher(opts); return new Watcher(opts);
@@ -23,7 +24,7 @@ export function dev(opts) {
class Watcher extends EventEmitter { class Watcher extends EventEmitter {
bundler: string; bundler: string;
dirs: { dirs: {
app: string; src: string;
dest: string; dest: string;
routes: string; routes: string;
rollup: string; rollup: string;
@@ -36,6 +37,8 @@ class Watcher extends EventEmitter {
live: boolean; live: boolean;
hot: boolean; hot: boolean;
devtools_port: number;
dev_server: DevServer; dev_server: DevServer;
proc: child_process.ChildProcess; proc: child_process.ChildProcess;
filewatchers: Array<{ close: () => void }>; filewatchers: Array<{ close: () => void }>;
@@ -51,23 +54,25 @@ class Watcher extends EventEmitter {
} }
constructor({ constructor({
app = locations.app(), src = locations.src(),
dest = locations.dest(), dest = locations.dest(),
routes = locations.routes(), routes = locations.routes(),
'dev-port': dev_port, 'dev-port': dev_port,
live, live,
hot, hot,
'devtools-port': devtools_port,
bundler, bundler,
webpack = 'webpack', webpack = 'webpack',
rollup = 'rollup', rollup = 'rollup',
port = +process.env.PORT port = +process.env.PORT
}: { }: {
app: string, src: string,
dest: string, dest: string,
routes: string, routes: string,
'dev-port': number, 'dev-port': number,
live: boolean, live: boolean,
hot: boolean, hot: boolean,
'devtools-port': number,
bundler?: string, bundler?: string,
webpack: string, webpack: string,
rollup: string, rollup: string,
@@ -76,7 +81,7 @@ class Watcher extends EventEmitter {
super(); super();
this.bundler = validate_bundler(bundler); this.bundler = validate_bundler(bundler);
this.dirs = { app, dest, routes, webpack, rollup }; this.dirs = { src, dest, routes, webpack, rollup };
this.port = port; this.port = port;
this.closed = false; this.closed = false;
@@ -84,6 +89,8 @@ class Watcher extends EventEmitter {
this.live = live; this.live = live;
this.hot = hot; this.hot = hot;
this.devtools_port = devtools_port;
this.filewatchers = []; this.filewatchers = [];
this.current_build = { this.current_build = {
@@ -94,7 +101,7 @@ class Watcher extends EventEmitter {
}; };
// remove this in a future version // remove this in a future version
const template = fs.readFileSync(path.join(app, 'template.html'), 'utf-8'); const template = read_template();
if (template.indexOf('%sapper.base%') === -1) { if (template.indexOf('%sapper.base%') === -1) {
const error = new Error(`As of Sapper v0.10, your template.html file must include %sapper.base% in the <head>`); const error = new Error(`As of Sapper v0.10, your template.html file must include %sapper.base% in the <head>`);
error.code = `missing-sapper-base`; error.code = `missing-sapper-base`;
@@ -129,6 +136,9 @@ class Watcher extends EventEmitter {
if (!this.dev_port) this.dev_port = await ports.find(10000); if (!this.dev_port) this.dev_port = await ports.find(10000);
// Chrome looks for debugging targets on ports 9222 and 9229 by default
if (!this.devtools_port) this.devtools_port = await ports.find(9222);
let manifest_data: ManifestData; let manifest_data: ManifestData;
try { try {
@@ -166,7 +176,7 @@ class Watcher extends EventEmitter {
} }
), ),
fs.watch(`${locations.app()}/template.html`, () => { fs.watch(`${locations.src()}/template.html`, () => {
this.dev_server.send({ this.dev_server.send({
action: 'reload' action: 'reload'
}); });
@@ -176,7 +186,7 @@ class Watcher extends EventEmitter {
let deferred = new Deferred(); let deferred = new Deferred();
// TODO watch the configs themselves? // TODO watch the configs themselves?
const compilers: Compilers = create_compilers(this.bundler, this.dirs); const compilers: Compilers = await create_compilers(this.bundler, this.dirs);
let log = ''; let log = '';
@@ -238,12 +248,21 @@ class Watcher extends EventEmitter {
restart(); restart();
} }
// we need to give the child process its own DevTools port,
// otherwise Node will try to use the parent's (and fail)
const debugArgRegex = /--inspect(?:-brk|-port)?|--debug-port/;
const execArgv = process.execArgv.slice();
if (execArgv.some((arg: string) => !!arg.match(debugArgRegex))) {
execArgv.push(`--inspect-port=${this.devtools_port}`);
}
this.proc = child_process.fork(`${dest}/server.js`, [], { this.proc = child_process.fork(`${dest}/server.js`, [], {
cwd: process.cwd(), cwd: process.cwd(),
env: Object.assign({ env: Object.assign({
PORT: this.port PORT: this.port
}, process.env), }, process.env),
stdio: ['ipc'] stdio: ['ipc'],
execArgv
}); });
this.proc.stdout.on('data', chunk => { this.proc.stdout.on('data', chunk => {

View File

@@ -13,6 +13,7 @@ import * as events from './interfaces';
type Opts = { type Opts = {
build: string, build: string,
dest: string, dest: string,
static: string,
basepath?: string, basepath?: string,
timeout: number | false timeout: number | false
}; };
@@ -46,7 +47,7 @@ async function execute(emitter: EventEmitter, opts: Opts) {
// Prep output directory // Prep output directory
sander.rimrafSync(export_dir); sander.rimrafSync(export_dir);
sander.copydirSync('assets').to(export_dir); sander.copydirSync(opts.static).to(export_dir);
sander.copydirSync(opts.build, 'client').to(export_dir, 'client'); sander.copydirSync(opts.build, 'client').to(export_dir, 'client');
if (sander.existsSync(opts.build, 'service-worker.js')) { if (sander.existsSync(opts.build, 'service-worker.js')) {
@@ -85,7 +86,7 @@ async function execute(emitter: EventEmitter, opts: Opts) {
function save(path: string, status: number, type: string, body: string) { function save(path: string, status: number, type: string, body: string) {
const { pathname } = resolve(origin, path); const { pathname } = resolve(origin, path);
let file = pathname.slice(1); let file = decodeURIComponent(pathname.slice(1));
if (saved.has(file)) return; if (saved.has(file)) return;
saved.add(file); saved.add(file);

View File

@@ -1,5 +1,5 @@
import * as child_process from 'child_process'; import * as child_process from 'child_process';
import { CompileResult } from '../core/create_compilers'; import { CompileResult } from '../core/create_compilers/interfaces';
export type ReadyEvent = { export type ReadyEvent = {
port: number; port: number;

View File

@@ -18,7 +18,7 @@ export function build(opts: { bundler?: string, legacy?: boolean }) {
bundler bundler
}, { }, {
dest: locations.dest(), dest: locations.dest(),
app: locations.app(), src: locations.src(),
routes: locations.routes(), routes: locations.routes(),
webpack: 'webpack', webpack: 'webpack',
rollup: 'rollup' rollup: 'rollup'

View File

@@ -15,6 +15,7 @@ export function exporter(export_dir: string, {
try { try {
const emitter = _exporter({ const emitter = _exporter({
build: locations.dest(), build: locations.dest(),
static: locations.static(),
dest: export_dir, dest: export_dir,
basepath, basepath,
timeout timeout

View File

@@ -1,15 +1,19 @@
import * as fs from 'fs'; import * as fs from 'fs';
export default function validate_bundler(bundler?: string) { export default function validate_bundler(bundler?: 'rollup' | 'webpack') {
if (!bundler) { if (!bundler) {
bundler = ( bundler = (
fs.existsSync('rollup') ? 'rollup' : fs.existsSync('rollup.config.js') ? 'rollup' :
fs.existsSync('webpack') ? 'webpack' : fs.existsSync('webpack.config.js') ? 'webpack' :
null null
); );
if (!bundler) { if (!bundler) {
throw new Error(`Could not find a 'rollup' or 'webpack' directory`); // TODO remove in a future version
deprecate_dir('rollup');
deprecate_dir('webpack');
throw new Error(`Could not find rollup.config.js or webpack.config.js`);
} }
} }
@@ -18,4 +22,17 @@ export default function validate_bundler(bundler?: string) {
} }
return bundler; return bundler;
}
function deprecate_dir(bundler: 'rollup' | 'webpack') {
try {
const stats = fs.statSync(bundler);
if (!stats.isDirectory()) return;
} catch (err) {
// do nothing
return;
}
// TODO link to docs, once those docs exist
throw new Error(`As of Sapper 0.21, build configuration should be placed in a single ${bundler}.config.js file`);
} }

View File

@@ -4,7 +4,8 @@ export const dev = () => process.env.NODE_ENV !== 'production';
export const locations = { export const locations = {
base: () => path.resolve(process.env.SAPPER_BASE || ''), base: () => path.resolve(process.env.SAPPER_BASE || ''),
app: () => path.resolve(process.env.SAPPER_BASE || '', process.env.SAPPER_APP || 'app'), src: () => path.resolve(process.env.SAPPER_BASE || '', process.env.SAPPER_SRC || 'src'),
routes: () => path.resolve(process.env.SAPPER_BASE || '', process.env.SAPPER_ROUTES || 'routes'), static: () => path.resolve(process.env.SAPPER_BASE || '', process.env.SAPPER_STATIC || 'static'),
routes: () => path.resolve(process.env.SAPPER_BASE || '', process.env.SAPPER_ROUTES || 'src/routes'),
dest: () => path.resolve(process.env.SAPPER_BASE || '', process.env.SAPPER_DEST || `.sapper/${dev() ? 'dev' : 'prod'}`) dest: () => path.resolve(process.env.SAPPER_BASE || '', process.env.SAPPER_DEST || `.sapper/${dev() ? 'dev' : 'prod'}`)
}; };

View File

@@ -15,8 +15,8 @@ export default class RollupCompiler {
chunks: any[]; chunks: any[];
css_files: Array<{ id: string, code: string }>; css_files: Array<{ id: string, code: string }>;
constructor(config: string) { constructor(config: any) {
this._ = this.get_config(path.resolve(config)); this._ = this.get_config(config);
this.input = null; this.input = null;
this.warnings = []; this.warnings = [];
this.errors = []; this.errors = [];
@@ -24,31 +24,8 @@ export default class RollupCompiler {
this.css_files = []; this.css_files = [];
} }
async get_config(input: string) { async get_config(mod: any) {
if (!rollup) rollup = relative('rollup', process.cwd()); // TODO this is hacky, and doesn't need to apply to all three compilers
const bundle = await rollup.rollup({
input,
external: (id: string) => {
return (id[0] !== '.' && !path.isAbsolute(id)) || id.slice(-5, id.length) === '.json';
}
});
const { code } = await bundle.generate({ format: 'cjs' });
// temporarily override require
const defaultLoader = require.extensions['.js'];
require.extensions['.js'] = (module: any, filename: string) => {
if (filename === input) {
module._compile(code, filename);
} else {
defaultLoader(module, filename);
}
};
const mod: any = require(input);
delete require.cache[input];
(mod.plugins || (mod.plugins = [])).push({ (mod.plugins || (mod.plugins = [])).push({
name: 'sapper-internal', name: 'sapper-internal',
options: (opts: any) => { options: (opts: any) => {
@@ -157,4 +134,34 @@ export default class RollupCompiler {
} }
}); });
} }
static async load_config() {
if (!rollup) rollup = relative('rollup', process.cwd());
const input = path.resolve('rollup.config.js');
const bundle = await rollup.rollup({
input,
external: (id: string) => {
return (id[0] !== '.' && !path.isAbsolute(id)) || id.slice(-5, id.length) === '.json';
}
});
const { code } = await bundle.generate({ format: 'cjs' });
// temporarily override require
const defaultLoader = require.extensions['.js'];
require.extensions['.js'] = (module: any, filename: string) => {
if (filename === input) {
module._compile(code, filename);
} else {
defaultLoader(module, filename);
}
};
const config: any = require(input);
delete require.cache[input];
return config;
}
} }

View File

@@ -1,4 +1,3 @@
import * as path from 'path';
import relative from 'require-relative'; import relative from 'require-relative';
import { CompileResult } from './interfaces'; import { CompileResult } from './interfaces';
import WebpackResult from './WebpackResult'; import WebpackResult from './WebpackResult';
@@ -8,9 +7,9 @@ let webpack: any;
export class WebpackCompiler { export class WebpackCompiler {
_: any; _: any;
constructor(config: string) { constructor(config: any) {
if (!webpack) webpack = relative('webpack', process.cwd()); if (!webpack) webpack = relative('webpack', process.cwd());
this._ = webpack(require(path.resolve(config))); this._ = webpack(config);
} }
oninvalid(cb: (filename: string) => void) { oninvalid(cb: (filename: string) => void) {

View File

@@ -4,6 +4,7 @@ import hash from 'string-hash';
import * as codec from 'sourcemap-codec'; import * as codec from 'sourcemap-codec';
import { PageComponent, Dirs } from '../../interfaces'; import { PageComponent, Dirs } from '../../interfaces';
import { CompileResult } from './interfaces'; import { CompileResult } from './interfaces';
import { posixify } from '../utils'
const inline_sourcemap_header = 'data:application/json;charset=utf-8;base64,'; const inline_sourcemap_header = 'data:application/json;charset=utf-8;base64,';
@@ -75,7 +76,7 @@ export default function extract_css(client_result: CompileResult, components: Pa
const component_owners = new Map(); const component_owners = new Map();
client_result.chunks.forEach(chunk => { client_result.chunks.forEach(chunk => {
chunk.modules.forEach(module => { chunk.modules.forEach(module => {
const component = path.relative(dirs.routes, module); const component = posixify(path.relative(dirs.routes, module));
component_owners.set(component, chunk); component_owners.set(component, chunk);
}); });
}); });
@@ -169,7 +170,8 @@ export default function extract_css(client_result: CompileResult, components: Pa
return null; return null;
} }
const main = client_result.assets.main; let main = client_result.assets.main;
if (process.env.SAPPER_LEGACY_BUILD) main = `legacy/${main}`;
const entry = fs.readFileSync(`${dirs.dest}/client/${main}`, 'utf-8'); const entry = fs.readFileSync(`${dirs.dest}/client/${main}`, 'utf-8');
const replacements = new Map(); const replacements = new Map();

View File

@@ -1,5 +1,4 @@
import * as fs from 'fs'; import * as path from 'path';
import { Dirs } from '../../interfaces';
import RollupCompiler from './RollupCompiler'; import RollupCompiler from './RollupCompiler';
import { WebpackCompiler } from './WebpackCompiler'; import { WebpackCompiler } from './WebpackCompiler';
@@ -11,27 +10,35 @@ export type Compilers = {
serviceworker?: Compiler; serviceworker?: Compiler;
} }
export default function create_compilers(bundler: string, dirs: Dirs): Compilers { export default async function create_compilers(bundler: 'rollup' | 'webpack'): Promise<Compilers> {
if (bundler === 'rollup') { if (bundler === 'rollup') {
const sw = `${dirs.rollup}/service-worker.config.js`; const config = await RollupCompiler.load_config();
validate_config(config, 'rollup');
return { return {
client: new RollupCompiler(`${dirs.rollup}/client.config.js`), client: new RollupCompiler(config.client),
server: new RollupCompiler(`${dirs.rollup}/server.config.js`), server: new RollupCompiler(config.server),
serviceworker: fs.existsSync(sw) && new RollupCompiler(sw) serviceworker: config.serviceworker && new RollupCompiler(config.serviceworker)
}; };
} }
if (bundler === 'webpack') { if (bundler === 'webpack') {
const sw = `${dirs.webpack}/service-worker.config.js`; const config = require(path.resolve('webpack.config.js'));
validate_config(config, 'webpack');
return { return {
client: new WebpackCompiler(`${dirs.webpack}/client.config.js`), client: new WebpackCompiler(config.client),
server: new WebpackCompiler(`${dirs.webpack}/server.config.js`), server: new WebpackCompiler(config.server),
serviceworker: fs.existsSync(sw) && new WebpackCompiler(sw) serviceworker: config.serviceworker && new WebpackCompiler(config.serviceworker)
}; };
} }
// this shouldn't be possible... // this shouldn't be possible...
throw new Error(`Invalid bundler option '${bundler}'`); throw new Error(`Invalid bundler option '${bundler}'`);
}
function validate_config(config: any, bundler: 'rollup' | 'webpack') {
if (!config.client || !config.server) {
throw new Error(`${bundler}.config.js must export a { client, server, serviceworker? } object`);
}
} }

View File

@@ -5,6 +5,11 @@ import { Page, PageComponent, ServerRoute, ManifestData } from '../interfaces';
import { posixify, reserved_words } from './utils'; import { posixify, reserved_words } from './utils';
export default function create_manifest_data(cwd = locations.routes()): ManifestData { export default function create_manifest_data(cwd = locations.routes()): ManifestData {
// TODO remove in a future version
if (!fs.existsSync(cwd)) {
throw new Error(`As of Sapper 0.21, the routes/ directory should become src/routes/`);
}
const components: PageComponent[] = []; const components: PageComponent[] = [];
const pages: Page[] = []; const pages: Page[] = [];
const server_routes: ServerRoute[] = []; const server_routes: ServerRoute[] = [];

View File

@@ -1,7 +1,7 @@
import * as fs from 'fs'; import * as fs from 'fs';
import * as path from 'path'; import * as path from 'path';
import glob from 'tiny-glob/sync.js'; import glob from 'tiny-glob/sync.js';
import { posixify, write_if_changed } from './utils'; import { posixify, stringify, write_if_changed } from './utils';
import { dev, locations } from '../config'; import { dev, locations } from '../config';
import { Page, PageComponent, ServerRoute, ManifestData } from '../interfaces'; import { Page, PageComponent, ServerRoute, ManifestData } from '../interfaces';
@@ -10,7 +10,7 @@ export function create_main_manifests({ bundler, manifest_data, dev_port }: {
manifest_data: ManifestData; manifest_data: ManifestData;
dev_port?: number; dev_port?: number;
}) { }) {
const manifest_dir = path.join(locations.app(), 'manifest'); const manifest_dir = path.join(locations.src(), 'manifest');
if (!fs.existsSync(manifest_dir)) fs.mkdirSync(manifest_dir); if (!fs.existsSync(manifest_dir)) fs.mkdirSync(manifest_dir);
const path_to_routes = path.relative(manifest_dir, locations.routes()); const path_to_routes = path.relative(manifest_dir, locations.routes());
@@ -30,20 +30,32 @@ export function create_serviceworker_manifest({ manifest_data, client_files }: {
manifest_data: ManifestData; manifest_data: ManifestData;
client_files: string[]; client_files: string[];
}) { }) {
const assets = glob('**', { cwd: 'assets', filesOnly: true }); let files;
// TODO remove in a future version
if (fs.existsSync(locations.static())) {
files = glob('**', { cwd: locations.static(), filesOnly: true });
} else {
if (fs.existsSync('assets')) {
throw new Error(`As of Sapper 0.21, the assets/ directory should become static/`);
}
files = [];
}
let code = ` let code = `
// This file is generated by Sapper — do not edit it! // This file is generated by Sapper — do not edit it!
export const timestamp = ${Date.now()}; export const timestamp = ${Date.now()};
export const assets = [\n\t${assets.map((x: string) => `"${x}"`).join(',\n\t')}\n]; export const files = [\n\t${files.map((x: string) => stringify(x)).join(',\n\t')}\n];
export { files as assets }; // legacy
export const shell = [\n\t${client_files.map((x: string) => `"${x}"`).join(',\n\t')}\n]; export const shell = [\n\t${client_files.map((x: string) => stringify(x)).join(',\n\t')}\n];
export const routes = [\n\t${manifest_data.pages.map((r: Page) => `{ pattern: ${r.pattern} }`).join(',\n\t')}\n]; export const routes = [\n\t${manifest_data.pages.map((r: Page) => `{ pattern: ${r.pattern} }`).join(',\n\t')}\n];
`.replace(/^\t\t/gm, '').trim(); `.replace(/^\t\t/gm, '').trim();
write_if_changed(`${locations.app()}/manifest/service-worker.js`, code); write_if_changed(`${locations.src()}/manifest/service-worker.js`, code);
} }
function generate_client( function generate_client(
@@ -60,8 +72,10 @@ function generate_client(
let code = ` let code = `
// This file is generated by Sapper — do not edit it! // This file is generated by Sapper — do not edit it!
import root from '${get_file(path_to_routes, manifest_data.root)}'; import root from ${stringify(get_file(path_to_routes, manifest_data.root))};
import error from '${posixify(`${path_to_routes}/_error.html`)}'; import error from ${stringify(posixify(`${path_to_routes}/_error.html`))};
const d = decodeURIComponent;
${manifest_data.components.map(component => { ${manifest_data.components.map(component => {
const annotation = bundler === 'webpack' const annotation = bundler === 'webpack'
@@ -71,8 +85,8 @@ function generate_client(
const source = get_file(path_to_routes, component); const source = get_file(path_to_routes, component);
return `const ${component.name} = { return `const ${component.name} = {
js: () => import(${annotation}'${source}'), js: () => import(${annotation}${stringify(source)}),
css: "__SAPPER_CSS_PLACEHOLDER:${component.file}__" css: "__SAPPER_CSS_PLACEHOLDER:${stringify(component.file, false)}__"
};`; };`;
}).join('\n')} }).join('\n')}
@@ -88,7 +102,7 @@ function generate_client(
if (part === null) return 'null'; if (part === null) return 'null';
if (part.params.length > 0) { if (part.params.length > 0) {
const props = part.params.map((param, i) => `${param}: match[${i + 1}]`); const props = part.params.map((param, i) => `${param}: d(match[${i + 1}])`);
return `{ component: ${part.component.name}, params: match => ({ ${props.join(', ')} }) }`; return `{ component: ${part.component.name}, params: match => ({ ${props.join(', ')} }) }`;
} }
@@ -113,7 +127,7 @@ function generate_client(
code += ` code += `
import('${sapper_dev_client}').then(client => { import(${stringify(sapper_dev_client)}).then(client => {
client.connect(${dev_port}); client.connect(${dev_port});
});`.replace(/^\t{3}/gm, ''); });`.replace(/^\t{3}/gm, '');
} }
@@ -127,17 +141,19 @@ function generate_server(
) { ) {
const imports = [].concat( const imports = [].concat(
manifest_data.server_routes.map(route => manifest_data.server_routes.map(route =>
`import * as ${route.name} from '${posixify(`${path_to_routes}/${route.file}`)}';`), `import * as ${route.name} from ${stringify(posixify(`${path_to_routes}/${route.file}`))};`),
manifest_data.components.map(component => manifest_data.components.map(component =>
`import ${component.name} from '${get_file(path_to_routes, component)}';`), `import ${component.name} from ${stringify(get_file(path_to_routes, component))};`),
`import root from '${get_file(path_to_routes, manifest_data.root)}';`, `import root from ${stringify(get_file(path_to_routes, manifest_data.root))};`,
`import error from '${posixify(`${path_to_routes}/_error.html`)}';` `import error from ${stringify(posixify(`${path_to_routes}/_error.html`))};`
); );
let code = ` let code = `
// This file is generated by Sapper — do not edit it! // This file is generated by Sapper — do not edit it!
${imports.join('\n')} ${imports.join('\n')}
const d = decodeURIComponent;
export const manifest = { export const manifest = {
server_routes: [ server_routes: [
${manifest_data.server_routes.map(route => `{ ${manifest_data.server_routes.map(route => `{
@@ -145,7 +161,7 @@ function generate_server(
pattern: ${route.pattern}, pattern: ${route.pattern},
handlers: ${route.name}, handlers: ${route.name},
params: ${route.params.length > 0 params: ${route.params.length > 0
? `match => ({ ${route.params.map((param, i) => `${param}: match[${i + 1}]`).join(', ')} })` ? `match => ({ ${route.params.map((param, i) => `${param}: d(match[${i + 1}])`).join(', ')} })`
: `() => ({})`} : `() => ({})`}
}`).join(',\n\n\t\t\t\t')} }`).join(',\n\n\t\t\t\t')}
], ],
@@ -160,12 +176,12 @@ function generate_server(
const props = [ const props = [
`name: "${part.component.name}"`, `name: "${part.component.name}"`,
`file: "${part.component.file}"`, `file: ${stringify(part.component.file)}`,
`component: ${part.component.name}` `component: ${part.component.name}`
]; ];
if (part.params.length > 0) { if (part.params.length > 0) {
const params = part.params.map((param, i) => `${param}: match[${i + 1}]`); const params = part.params.map((param, i) => `${param}: d(match[${i + 1}])`);
props.push(`params: match => ({ ${params.join(', ')} })`); props.push(`params: match => ({ ${params.join(', ')} })`);
} }

17
src/core/read_template.ts Normal file
View File

@@ -0,0 +1,17 @@
import * as fs from 'fs';
import { locations } from '../config';
export default function read_template() {
try {
return fs.readFileSync(`${locations.src()}/template.html`, 'utf-8');
} catch (err) {
if (fs.existsSync(`app/template.html`)) {
throw new Error(`As of Sapper 0.21, the default folder structure has been changed:
app/ --> src/
routes/ --> src/routes/
assets/ --> static/`);
}
throw err;
}
}

View File

@@ -14,6 +14,11 @@ export function posixify(file: string) {
return file.replace(/[/\\]/g, '/'); return file.replace(/[/\\]/g, '/');
} }
export function stringify(string: string, includeQuotes: boolean = true) {
const quoted = JSON.stringify(string);
return includeQuotes ? quoted : quoted.slice(1, -1);
}
export function fudge_mtime(file: string) { export function fudge_mtime(file: string) {
// need to fudge the mtime so that webpack doesn't go doolally // need to fudge the mtime so that webpack doesn't go doolally
const { atime, mtime } = fs.statSync(file); const { atime, mtime } = fs.statSync(file);

View File

@@ -43,7 +43,7 @@ export type ServerRoute = {
export type Dirs = { export type Dirs = {
dest: string, dest: string,
app: string, src: string,
routes: string, routes: string,
webpack: string, webpack: string,
rollup: string rollup: string

View File

@@ -8,6 +8,7 @@ import fetch from 'node-fetch';
import { lookup } from './middleware/mime'; import { lookup } from './middleware/mime';
import { locations, dev } from './config'; import { locations, dev } from './config';
import sourceMapSupport from 'source-map-support'; import sourceMapSupport from 'source-map-support';
import read_template from './core/read_template';
sourceMapSupport.install(); sourceMapSupport.install();
@@ -136,22 +137,22 @@ export default function middleware(opts: {
fs.existsSync(path.join(output, 'index.html')) && serve({ fs.existsSync(path.join(output, 'index.html')) && serve({
pathname: '/index.html', pathname: '/index.html',
cache_control: 'max-age=600' cache_control: dev() ? 'no-cache' : 'max-age=600'
}), }),
fs.existsSync(path.join(output, 'service-worker.js')) && serve({ fs.existsSync(path.join(output, 'service-worker.js')) && serve({
pathname: '/service-worker.js', pathname: '/service-worker.js',
cache_control: 'max-age=600' cache_control: 'no-cache, no-store, must-revalidate'
}), }),
fs.existsSync(path.join(output, 'service-worker.js.map')) && serve({ fs.existsSync(path.join(output, 'service-worker.js.map')) && serve({
pathname: '/service-worker.js.map', pathname: '/service-worker.js.map',
cache_control: 'max-age=600' cache_control: 'no-cache, no-store, must-revalidate'
}), }),
serve({ serve({
prefix: '/client/', prefix: '/client/',
cache_control: dev() ? 'no-cache' : 'max-age=31536000' cache_control: dev() ? 'no-cache' : 'max-age=31536000, immutable'
}), }),
get_server_route_handler(manifest.server_routes), get_server_route_handler(manifest.server_routes),
@@ -288,8 +289,8 @@ function get_page_handler(
: (assets => () => assets)(JSON.parse(fs.readFileSync(path.join(output, 'build.json'), 'utf-8'))); : (assets => () => assets)(JSON.parse(fs.readFileSync(path.join(output, 'build.json'), 'utf-8')));
const template = dev() const template = dev()
? () => fs.readFileSync(`${locations.app()}/template.html`, 'utf-8') ? () => read_template()
: (str => () => str)(fs.readFileSync(`${locations.dest()}/template.html`, 'utf-8')); : (str => () => str)(read_template());
const { server_routes, pages } = manifest; const { server_routes, pages } = manifest;
const error_route = manifest.error; const error_route = manifest.error;
@@ -312,6 +313,7 @@ function get_page_handler(
} = get_build_info(); } = get_build_info();
res.setHeader('Content-Type', 'text/html'); res.setHeader('Content-Type', 'text/html');
res.setHeader('Cache-Control', dev() ? 'no-cache' : 'max-age=600');
// preload main.js and current route // preload main.js and current route
// TODO detect other stuff we can preload? images, CSS, fonts? // TODO detect other stuff we can preload? images, CSS, fonts?
@@ -524,9 +526,12 @@ function get_page_handler(
styles = (css && css.code ? `<style>${css.code}</style>` : ''); styles = (css && css.code ? `<style>${css.code}</style>` : '');
} }
// users can set a CSP nonce using res.locals.nonce
const nonce_attr = (res.locals && res.locals.nonce) ? ` nonce="${res.locals.nonce}"` : '';
const body = template() const body = template()
.replace('%sapper.base%', () => `<base href="${req.baseUrl}/">`) .replace('%sapper.base%', () => `<base href="${req.baseUrl}/">`)
.replace('%sapper.scripts%', () => `<script>${script}</script>`) .replace('%sapper.scripts%', () => `<script${nonce_attr}>${script}</script>`)
.replace('%sapper.html%', () => html) .replace('%sapper.html%', () => html)
.replace('%sapper.head%', () => `<noscript id='sapper-head-start'></noscript>${head}<noscript id='sapper-head-end'></noscript>`) .replace('%sapper.head%', () => `<noscript id='sapper-head-start'></noscript>${head}<noscript id='sapper-head-end'></noscript>`)
.replace('%sapper.styles%', () => styles); .replace('%sapper.styles%', () => styles);

View File

@@ -5,7 +5,7 @@ export default {
client: { client: {
input: () => { input: () => {
return `${locations.app()}/client.js` return `${locations.src()}/client.js`
}, },
output: () => { output: () => {
@@ -24,20 +24,21 @@ export default {
server: { server: {
input: () => { input: () => {
return `${locations.app()}/server.js` return `${locations.src()}/server.js`
}, },
output: () => { output: () => {
return { return {
dir: locations.dest(), dir: locations.dest(),
format: 'cjs' format: 'cjs',
sourcemap: dev()
}; };
} }
}, },
serviceworker: { serviceworker: {
input: () => { input: () => {
return `${locations.app()}/service-worker.js`; return `${locations.src()}/service-worker.js`;
}, },
output: () => { output: () => {

View File

@@ -66,8 +66,8 @@ function select_route(url: URL): Target {
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] = decodeURIComponent((value || '').replace(/\+/g, ' '));
}); });
} }
return { url, path, page, match, query }; return { url, path, page, match, query };

View File

@@ -6,7 +6,7 @@ export default {
client: { client: {
entry: () => { entry: () => {
return { return {
main: `${locations.app()}/client` main: `${locations.src()}/client`
}; };
}, },
@@ -23,7 +23,7 @@ export default {
server: { server: {
entry: () => { entry: () => {
return { return {
server: `${locations.app()}/server` server: `${locations.src()}/server`
}; };
}, },
@@ -40,7 +40,7 @@ export default {
serviceworker: { serviceworker: {
entry: () => { entry: () => {
return { return {
'service-worker': `${locations.app()}/service-worker` 'service-worker': `${locations.src()}/service-worker`
}; };
}, },

View File

@@ -1,4 +1,4 @@
import { init, prefetchRoutes } from '../../../runtime.js'; import { init, goto, prefetchRoutes } from '../../../runtime.js';
import { Store } from 'svelte/store.js'; import { Store } from 'svelte/store.js';
import { manifest } from './manifest/client.js'; import { manifest } from './manifest/client.js';
@@ -10,4 +10,5 @@ window.init = () => {
}); });
}; };
window.prefetchRoutes = prefetchRoutes; window.prefetchRoutes = prefetchRoutes;
window.goto = goto;

View File

@@ -9,7 +9,7 @@
<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() { oncreate() {

View File

@@ -106,6 +106,14 @@ const posts = [
<p>If you didn't have adult onset diabetes, I wouldn't mind giving you a little sugar. Everybody dance NOW. And the soup of the day is bread. Great, now I'm gonna smell to high heaven like a tuna melt!</p> <p>If you didn't have adult onset diabetes, I wouldn't mind giving you a little sugar. Everybody dance NOW. And the soup of the day is bread. Great, now I'm gonna smell to high heaven like a tuna melt!</p>
<p>That's how Tony Wonder lost a nut. She calls it a Mayonegg. Go ahead, touch the Cornballer. There's a new daddy in town. A discipline daddy.</p> <p>That's how Tony Wonder lost a nut. She calls it a Mayonegg. Go ahead, touch the Cornballer. There's a new daddy in town. A discipline daddy.</p>
` `
},
{
title: 'Encödïng test',
slug: 'encödïng-test',
html: `
<p>It works</p>
`
} }
]; ];

View File

@@ -0,0 +1,12 @@
<h1>{slug} ({message})</h1>
<script>
export default {
preload({ params, query }) {
return {
slug: params.slug,
message: query.message
};
}
};
</script>

View File

@@ -0,0 +1,15 @@
export function get(req, res) {
res.writeHead(200, {
'Content-Type': 'text/html'
});
res.end(`
<!doctype html>
<html>
<head><meta charset="utf-8"></head>
<body>
<h1>${req.params.slug}</h1>
</body>
</html>
`);
}

View File

@@ -15,6 +15,8 @@
<a href='credentials?creds=include'>credentials</a> <a href='credentials?creds=include'>credentials</a>
<a rel=prefetch class='{page === "blog" ? "selected" : ""}' href='blog'>blog</a> <a rel=prefetch class='{page === "blog" ? "selected" : ""}' href='blog'>blog</a>
<a href="const">const</a> <a href="const">const</a>
<a href="echo/page/encöded?message=hëllö+wörld">echo/page/encöded?message=hëllö+wörld</a>
<a href="echo/page/empty?message">echo/page/empty?message</a>
<div class='hydrate-test'></div> <div class='hydrate-test'></div>

View File

Before

Width:  |  Height:  |  Size: 1.9 KiB

After

Width:  |  Height:  |  Size: 1.9 KiB

View File

Before

Width:  |  Height:  |  Size: 80 KiB

After

Width:  |  Height:  |  Size: 80 KiB

View File

Before

Width:  |  Height:  |  Size: 20 KiB

After

Width:  |  Height:  |  Size: 20 KiB

View File

Before

Width:  |  Height:  |  Size: 6.3 KiB

After

Width:  |  Height:  |  Size: 6.3 KiB

View File

@@ -0,0 +1,77 @@
const webpack = require('webpack');
const config = require('../../config/webpack.js');
const sapper_pkg = require('../../package.json');
const mode = process.env.NODE_ENV;
const isDev = mode === 'development';
module.exports = {
client: {
entry: config.client.entry(),
output: config.client.output(),
resolve: {
extensions: ['.js', '.html']
},
module: {
rules: [
{
test: /\.html$/,
exclude: /node_modules/,
use: {
loader: 'svelte-loader',
options: {
hydratable: true,
cascade: false,
store: true
}
}
}
]
},
mode,
plugins: [
isDev && new webpack.HotModuleReplacementPlugin()
].filter(Boolean),
devtool: isDev && 'inline-source-map'
},
server: {
entry: config.server.entry(),
output: config.server.output(),
target: 'node',
resolve: {
extensions: ['.js', '.html']
},
externals: [].concat(
Object.keys(sapper_pkg.dependencies),
Object.keys(sapper_pkg.devDependencies)
),
module: {
rules: [
{
test: /\.html$/,
exclude: /node_modules/,
use: {
loader: 'svelte-loader',
options: {
css: false,
cascade: false,
store: true,
generate: 'ssr'
}
}
}
]
},
mode,
performance: {
hints: false // it doesn't matter if server.js is large
}
},
serviceworker: {
entry: config.serviceworker.entry(),
output: config.serviceworker.output(),
mode
}
};

View File

@@ -1,34 +0,0 @@
const config = require('../../../config/webpack.js');
const webpack = require('webpack');
const mode = process.env.NODE_ENV;
const isDev = mode === 'development';
module.exports = {
entry: config.client.entry(),
output: config.client.output(),
resolve: {
extensions: ['.js', '.html']
},
module: {
rules: [
{
test: /\.html$/,
exclude: /node_modules/,
use: {
loader: 'svelte-loader',
options: {
hydratable: true,
cascade: false,
store: true
}
}
}
]
},
mode,
plugins: [
isDev && new webpack.HotModuleReplacementPlugin()
].filter(Boolean),
devtool: isDev && 'inline-source-map'
};

View File

@@ -1,36 +0,0 @@
const config = require('../../../config/webpack.js');
const sapper_pkg = require('../../../package.json');
module.exports = {
entry: config.server.entry(),
output: config.server.output(),
target: 'node',
resolve: {
extensions: ['.js', '.html']
},
externals: [].concat(
Object.keys(sapper_pkg.dependencies),
Object.keys(sapper_pkg.devDependencies)
),
module: {
rules: [
{
test: /\.html$/,
exclude: /node_modules/,
use: {
loader: 'svelte-loader',
options: {
css: false,
cascade: false,
store: true,
generate: 'ssr'
}
}
}
]
},
mode: process.env.NODE_ENV,
performance: {
hints: false // it doesn't matter if server.js is large
}
};

View File

@@ -1,7 +0,0 @@
const config = require('../../../config/webpack.js');
module.exports = {
entry: config.serviceworker.entry(),
output: config.serviceworker.output(),
mode: process.env.NODE_ENV
};

View File

@@ -5,6 +5,7 @@ const Nightmare = require('nightmare');
const walkSync = require('walk-sync'); const walkSync = require('walk-sync');
const rimraf = require('rimraf'); const rimraf = require('rimraf');
const ports = require('port-authority'); const ports = require('port-authority');
const fetch = require('node-fetch');
Nightmare.action('page', { Nightmare.action('page', {
title(done) { title(done) {
@@ -93,6 +94,7 @@ function testExport({ basepath = '' }) {
'blog/how-to-use-sapper/index.html', 'blog/how-to-use-sapper/index.html',
'blog/what-is-sapper/index.html', 'blog/what-is-sapper/index.html',
'blog/why-the-name/index.html', 'blog/why-the-name/index.html',
'blog/encödïng-test/index.html',
'blog.json', 'blog.json',
'blog/a-very-long-post.json', 'blog/a-very-long-post.json',
@@ -101,6 +103,7 @@ function testExport({ basepath = '' }) {
'blog/how-to-use-sapper.json', 'blog/how-to-use-sapper.json',
'blog/what-is-sapper.json', 'blog/what-is-sapper.json',
'blog/why-the-name.json', 'blog/why-the-name.json',
'blog/encödïng-test.json',
'favicon.png', 'favicon.png',
'global.css', 'global.css',
@@ -751,18 +754,67 @@ function run({ mode, basepath = '' }) {
assert.equal(title, 'reserved words are okay as routes'); assert.equal(title, 'reserved words are okay as routes');
}); });
}); });
it('encodes req.params and req.query for server-rendered pages', () => {
return nightmare.goto(`${base}/echo/page/encöded?message=hëllö+wörld`)
.page.title()
.then(title => {
assert.equal(title, 'encöded (hëllö wörld)');
});
});
it('encodes req.params and req.query for client-rendered pages', () => {
return nightmare.goto(base).init()
.click('a[href="echo/page/encöded?message=hëllö+wörld"]')
.wait(100)
.page.title()
.then(title => {
assert.equal(title, 'encöded (hëllö wörld)');
});
});
it('accepts value-less query string parameter on server', () => {
return nightmare.goto(`${base}/echo/page/empty?message`)
.page.title()
.then(title => {
assert.equal(title, 'empty ()');
});
});
it('accepts value-less query string parameter on client', () => {
return nightmare.goto(base).init()
.click('a[href="echo/page/empty?message"]')
.wait(100)
.page.title()
.then(title => {
assert.equal(title, 'empty ()');
});
});
it('encodes req.params for server routes', () => {
return nightmare.goto(`${base}/echo/server-route/encöded`)
.page.title()
.then(title => {
assert.equal(title, 'encöded');
});
});
}); });
describe('headers', () => { describe('headers', () => {
it('sets Content-Type and Link...preload headers', () => { it('sets Content-Type, Link...preload, and Cache-Control headers', () => {
return capture(() => nightmare.goto(base)).then(requests => { return capture(() => fetch(base)).then(responses => {
const { headers } = requests[0]; const { headers } = responses[0];
assert.equal( assert.equal(
headers['content-type'], headers['content-type'],
'text/html' 'text/html'
); );
assert.equal(
headers['cache-control'],
'max-age=600'
);
const str = ['main', '.+?\\.\\d+'] const str = ['main', '.+?\\.\\d+']
.map(file => { .map(file => {
return `<${basepath}/client/[^/]+/${file}\\.js>;rel="preload";as="script"`; return `<${basepath}/client/[^/]+/${file}\\.js>;rel="preload";as="script"`;