mirror of
https://github.com/kevin-DL/sapper.git
synced 2026-01-15 04:14:46 +00:00
Compare commits
42 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 |
17
CHANGELOG.md
17
CHANGELOG.md
@@ -1,5 +1,22 @@
|
|||||||
# 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
|
## 0.17.0
|
||||||
|
|
||||||
* Use `cheap-watch` instead of `chokidar` ([#364](https://github.com/sveltejs/sapper/issues/364))
|
* Use `cheap-watch` instead of `chokidar` ([#364](https://github.com/sveltejs/sapper/issues/364))
|
||||||
|
|||||||
2
api.js
2
api.js
@@ -1 +1 @@
|
|||||||
module.exports = require('./dist/api.ts.js');
|
module.exports = require('./dist/api.js');
|
||||||
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');
|
||||||
1521
package-lock.json
generated
1521
package-lock.json
generated
File diff suppressed because it is too large
Load Diff
@@ -1,16 +1,16 @@
|
|||||||
{
|
{
|
||||||
"name": "sapper",
|
"name": "sapper",
|
||||||
"version": "0.17.0",
|
"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",
|
"components",
|
||||||
"dist"
|
"dist"
|
||||||
@@ -20,6 +20,7 @@
|
|||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"html-minifier": "^3.5.16",
|
"html-minifier": "^3.5.16",
|
||||||
|
"shimport": "^0.0.9",
|
||||||
"source-map-support": "^0.5.6",
|
"source-map-support": "^0.5.6",
|
||||||
"tslib": "^1.9.1"
|
"tslib": "^1.9.1"
|
||||||
},
|
},
|
||||||
@@ -48,7 +49,7 @@
|
|||||||
"pretty-ms": "^3.1.0",
|
"pretty-ms": "^3.1.0",
|
||||||
"require-relative": "^0.8.7",
|
"require-relative": "^0.8.7",
|
||||||
"rimraf": "^2.6.2",
|
"rimraf": "^2.6.2",
|
||||||
"rollup": "^0.59.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-node-resolve": "^3.3.0",
|
||||||
|
|||||||
@@ -32,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: {
|
||||||
@@ -51,7 +52,6 @@ export default [
|
|||||||
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`);
|
||||||
|
|||||||
@@ -5,7 +5,10 @@ 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 { 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();
|
||||||
@@ -27,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(...)
|
||||||
@@ -51,23 +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_assets.json'), JSON.stringify(client_info.assetsByChunkName));
|
bundler,
|
||||||
|
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;
|
||||||
@@ -75,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);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|||||||
146
src/api/dev.ts
146
src/api/dev.ts
@@ -5,34 +5,37 @@ 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 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;
|
crashed: boolean;
|
||||||
restarting: boolean;
|
restarting: boolean;
|
||||||
@@ -47,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 = {
|
||||||
@@ -102,7 +117,8 @@ 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);
|
||||||
|
|
||||||
@@ -149,13 +165,13 @@ class Watcher extends EventEmitter {
|
|||||||
})
|
})
|
||||||
);
|
);
|
||||||
|
|
||||||
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 = '';
|
let log = '';
|
||||||
|
|
||||||
@@ -174,11 +190,10 @@ 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) => {
|
||||||
this.deferreds.client.promise.then(() => {
|
deferred.promise.then(() => {
|
||||||
const restart = () => {
|
const restart = () => {
|
||||||
log = '';
|
log = '';
|
||||||
this.crashed = false;
|
this.crashed = false;
|
||||||
@@ -190,11 +205,15 @@ class Watcher extends EventEmitter {
|
|||||||
process: this.proc
|
process: this.proc
|
||||||
});
|
});
|
||||||
|
|
||||||
this.deferreds.server.fulfil();
|
if (this.hot && this.bundler === 'webpack') {
|
||||||
|
this.dev_server.send({
|
||||||
this.dev_server.send({
|
status: 'completed'
|
||||||
status: 'completed'
|
});
|
||||||
});
|
} else {
|
||||||
|
this.dev_server.send({
|
||||||
|
action: 'reload'
|
||||||
|
});
|
||||||
|
}
|
||||||
}))
|
}))
|
||||||
.catch(err => {
|
.catch(err => {
|
||||||
if (this.crashed) return;
|
if (this.crashed) return;
|
||||||
@@ -245,31 +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_assets.json'), JSON.stringify(info.assetsByChunkName, null, ' '));
|
fs.writeFileSync(path.join(dest, 'build.json'), JSON.stringify({
|
||||||
this.deferreds.client.fulfil();
|
bundler: this.bundler,
|
||||||
|
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);
|
||||||
@@ -281,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;
|
||||||
@@ -332,82 +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,
|
||||||
message: err.message
|
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
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
const INTERVAL = 10000;
|
const INTERVAL = 10000;
|
||||||
|
|
||||||
class DevServer {
|
class DevServer {
|
||||||
|
|||||||
@@ -10,7 +10,14 @@ import minify_html from './utils/minify_html';
|
|||||||
import Deferred from './utils/Deferred';
|
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,42 +34,38 @@ 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(basepath || '', origin);
|
const root = new URL(opts.basepath || '', origin);
|
||||||
|
|
||||||
emitter.emit('info', {
|
emitter.emit('info', {
|
||||||
message: `Crawling ${root.href}`
|
message: `Crawling ${root.href}`
|
||||||
});
|
});
|
||||||
|
|
||||||
const proc = child_process.fork(path.resolve(`${build}/server.js`), [], {
|
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)
|
||||||
});
|
});
|
||||||
@@ -117,7 +120,18 @@ async function execute(emitter: EventEmitter, {
|
|||||||
|
|
||||||
const deferred = get_deferred(pathname);
|
const deferred = get_deferred(pathname);
|
||||||
|
|
||||||
const r = await fetch(url.href);
|
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);
|
const range = ~~(r.status / 100);
|
||||||
|
|
||||||
if (range === 2) {
|
if (range === 2) {
|
||||||
@@ -152,11 +166,12 @@ async function execute(emitter: EventEmitter, {
|
|||||||
}
|
}
|
||||||
|
|
||||||
return ports.wait(port)
|
return ports.wait(port)
|
||||||
.then(() => {
|
.then(() => handle(root))
|
||||||
// TODO all static routes
|
.then(() => proc.kill())
|
||||||
return handle(root);
|
.catch(err => {
|
||||||
})
|
proc.kill();
|
||||||
.then(() => proc.kill());
|
throw err;
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function get_href(attrs: string) {
|
function get_href(attrs: string) {
|
||||||
|
|||||||
@@ -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;
|
||||||
@@ -26,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 = {
|
||||||
|
|||||||
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'))
|
||||||
|
);
|
||||||
|
}
|
||||||
32
src/cli.ts
32
src/cli.ts
@@ -11,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);
|
||||||
});
|
});
|
||||||
@@ -19,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';
|
||||||
@@ -30,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');
|
||||||
|
|
||||||
@@ -46,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,7 +75,15 @@ prog.command('export [dest]')
|
|||||||
.option('--build', '(Re)build app before exporting', true)
|
.option('--build', '(Re)build app before exporting', true)
|
||||||
.option('--build-dir', 'Specify a custom temporary build directory', '.sapper/prod')
|
.option('--build-dir', 'Specify a custom temporary build directory', '.sapper/prod')
|
||||||
.option('--basepath', 'Specify a base path')
|
.option('--basepath', 'Specify a base path')
|
||||||
.action(async (dest = 'export', opts: { build: boolean, 'build-dir': string, basepath?: string }) => {
|
.option('--timeout', 'Milliseconds to wait for a page (--no-timeout to disable)', 5000)
|
||||||
|
.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 = opts['build-dir'];
|
process.env.SAPPER_DEST = opts['build-dir'];
|
||||||
|
|
||||||
@@ -75,7 +93,7 @@ prog.command('export [dest]')
|
|||||||
if (opts.build) {
|
if (opts.build) {
|
||||||
console.log(`> Building...`);
|
console.log(`> Building...`);
|
||||||
const { build } = await import('./cli/build');
|
const { build } = await import('./cli/build');
|
||||||
await build();
|
await build(opts);
|
||||||
console.error(`\n> Built in ${elapsed(start)}`);
|
console.error(`\n> Built in ${elapsed(start)}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -83,7 +101,7 @@ prog.command('export [dest]')
|
|||||||
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 colors from 'kleur';
|
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);
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -5,7 +5,7 @@ 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,20 +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.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.message}`)}`);
|
console.log(colors.bold.red(`> ${event.message}`));
|
||||||
if (event.log) console.log(event.log);
|
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);
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -57,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);
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -72,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,26 +1,28 @@
|
|||||||
import { exporter as _exporter } from '../api/export';
|
import { exporter as _exporter } from '../api/export';
|
||||||
import colors from 'kleur';
|
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';
|
||||||
|
|
||||||
function left_pad(str: string, len: number) {
|
export function exporter(export_dir: string, {
|
||||||
while (str.length < len) str = ` ${str}`;
|
basepath = '',
|
||||||
return str;
|
timeout
|
||||||
}
|
}: {
|
||||||
|
basepath: string,
|
||||||
export function exporter(export_dir: string, { basepath = '' }) {
|
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 => {
|
||||||
const pb = prettyBytes(event.size);
|
|
||||||
const size_color = event.size > 150000 ? colors.bold.red : event.size > 50000 ? colors.bold.yellow : colors.bold.gray;
|
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(prettyBytes(event.size), 10));
|
const size_label = size_color(left_pad(pb(event.size), 10));
|
||||||
|
|
||||||
const file_label = event.status === 200
|
const file_label = event.status === 200
|
||||||
? event.file
|
? event.file
|
||||||
|
|||||||
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;
|
||||||
|
}
|
||||||
@@ -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;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
@@ -56,8 +56,6 @@ function generate_client(
|
|||||||
const server_routes_to_ignore = routes.server_routes.filter(route =>
|
const server_routes_to_ignore = routes.server_routes.filter(route =>
|
||||||
!page_ids.has(route.pattern.toString()));
|
!page_ids.has(route.pattern.toString()));
|
||||||
|
|
||||||
const len = Math.max(...routes.components.map(c => c.name.length));
|
|
||||||
|
|
||||||
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, routes.root)}';
|
import root from '${get_file(path_to_routes, routes.root)}';
|
||||||
@@ -105,11 +103,9 @@ function generate_client(
|
|||||||
|
|
||||||
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;
|
||||||
|
|||||||
@@ -151,7 +151,7 @@ export default function middleware(opts: {
|
|||||||
|
|
||||||
serve({
|
serve({
|
||||||
prefix: '/client/',
|
prefix: '/client/',
|
||||||
cache_control: 'max-age=31536000'
|
cache_control: dev() ? 'no-cache' : 'max-age=31536000'
|
||||||
}),
|
}),
|
||||||
|
|
||||||
get_server_route_handler(manifest.server_routes),
|
get_server_route_handler(manifest.server_routes),
|
||||||
@@ -217,7 +217,7 @@ function get_server_route_handler(routes: ServerRoute[]) {
|
|||||||
|
|
||||||
// intercept data so that it can be exported
|
// intercept data so that it can be exported
|
||||||
res.write = function(chunk: any) {
|
res.write = function(chunk: any) {
|
||||||
chunks.push(new Buffer(chunk));
|
chunks.push(Buffer.from(chunk));
|
||||||
write.apply(res, arguments);
|
write.apply(res, arguments);
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -227,7 +227,7 @@ function get_server_route_handler(routes: ServerRoute[]) {
|
|||||||
};
|
};
|
||||||
|
|
||||||
res.end = function(chunk?: any) {
|
res.end = function(chunk?: any) {
|
||||||
if (chunk) chunks.push(new Buffer(chunk));
|
if (chunk) chunks.push(Buffer.from(chunk));
|
||||||
end.apply(res, arguments);
|
end.apply(res, arguments);
|
||||||
|
|
||||||
process.send({
|
process.send({
|
||||||
@@ -282,9 +282,9 @@ function get_page_handler(
|
|||||||
) {
|
) {
|
||||||
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')
|
||||||
@@ -303,24 +303,28 @@ function get_page_handler(
|
|||||||
}
|
}
|
||||||
|
|
||||||
function handle_page(page: Page, req: Req, res: ServerResponse, status = 200, error: Error | string = null) {
|
function handle_page(page: Page, req: Req, res: ServerResponse, status = 200, error: Error | string = null) {
|
||||||
const chunks: Record<string, string | string[]> = get_chunks();
|
const build_info: {
|
||||||
|
bundler: 'rollup' | 'webpack',
|
||||||
|
shimport: string | null,
|
||||||
|
assets: Record<string, string | string[]>
|
||||||
|
} = get_build_info();
|
||||||
|
|
||||||
res.setHeader('Content-Type', 'text/html');
|
res.setHeader('Content-Type', 'text/html');
|
||||||
|
|
||||||
// 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?
|
||||||
let preloaded_chunks = Array.isArray(chunks.main) ? chunks.main : [chunks.main];
|
let preloaded_chunks = Array.isArray(build_info.assets.main) ? build_info.assets.main : [build_info.assets.main];
|
||||||
if (!error) {
|
if (!error) {
|
||||||
page.parts.forEach(part => {
|
page.parts.forEach(part => {
|
||||||
if (!part) return;
|
if (!part) return;
|
||||||
|
|
||||||
// using concat because it could be a string or an array. thanks webpack!
|
// using concat because it could be a string or an array. thanks webpack!
|
||||||
preloaded_chunks = preloaded_chunks.concat(chunks[part.name]);
|
preloaded_chunks = preloaded_chunks.concat(build_info.assets[part.name]);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
const link = preloaded_chunks
|
const link = preloaded_chunks
|
||||||
.filter(file => !file.match(/\.map$/))
|
.filter(file => file && !file.match(/\.map$/))
|
||||||
.map(file => `<${req.baseUrl}/client/${file}>;rel="preload";as="script"`)
|
.map(file => `<${req.baseUrl}/client/${file}>;rel="preload";as="script"`)
|
||||||
.join(', ');
|
.join(', ');
|
||||||
|
|
||||||
@@ -354,7 +358,6 @@ function get_page_handler(
|
|||||||
);
|
);
|
||||||
|
|
||||||
if (include_cookies) {
|
if (include_cookies) {
|
||||||
const cookies: Record<string, string> = {};
|
|
||||||
if (!opts.headers) opts.headers = {};
|
if (!opts.headers) opts.headers = {};
|
||||||
|
|
||||||
const str = []
|
const str = []
|
||||||
@@ -481,11 +484,12 @@ function get_page_handler(
|
|||||||
store
|
store
|
||||||
});
|
});
|
||||||
|
|
||||||
let scripts = []
|
const file = [].concat(build_info.assets.main).filter(file => file && /\.js$/.test(file))[0];
|
||||||
.concat(chunks.main) // chunks main might be an array. it might not! thanks, webpack
|
const main = `${req.baseUrl}/client/${file}`;
|
||||||
.filter(file => !file.match(/\.map$/))
|
|
||||||
.map(file => `<script src='${req.baseUrl}/client/${file}'></script>`)
|
const script = build_info.bundler === 'rollup'
|
||||||
.join('');
|
? `<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__={${[
|
let inline_script = `__SAPPER__={${[
|
||||||
error && `error:1`,
|
error && `error:1`,
|
||||||
@@ -501,7 +505,7 @@ function get_page_handler(
|
|||||||
|
|
||||||
const body = template()
|
const body = template()
|
||||||
.replace('%sapper.base%', () => `<base href="${req.baseUrl}/">`)
|
.replace('%sapper.base%', () => `<base href="${req.baseUrl}/">`)
|
||||||
.replace('%sapper.scripts%', () => `<script>${inline_script}</script>${scripts}`)
|
.replace('%sapper.scripts%', () => `<script>${inline_script}</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%', () => (css && css.code ? `<style>${css.code}</style>` : ''));
|
.replace('%sapper.styles%', () => (css && css.code ? `<style>${css.code}</style>` : ''));
|
||||||
|
|||||||
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'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
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;
|
||||||
|
}
|
||||||
@@ -2,7 +2,7 @@ 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 { manifest } from './manifest/server.js';
|
import { manifest } from './manifest/server.js';
|
||||||
|
|
||||||
|
|||||||
@@ -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(),
|
||||||
|
|||||||
@@ -1,2 +1,2 @@
|
|||||||
// TODO write to this file, instead of webpack.ts.js
|
console.error(`[DEPRECATION] As of Sapper 0.18, webpack config should be loaded from sapper/config/webpack.js`);
|
||||||
module.exports = require('./dist/webpack.ts.js');
|
module.exports = require('./dist/webpack.js');
|
||||||
@@ -1,2 +1,2 @@
|
|||||||
// TODO deprecate this file in favour of sapper/webpack.js
|
console.error(`[DEPRECATION] As of Sapper 0.18, webpack config should be loaded from sapper/config/webpack.js`);
|
||||||
module.exports = require('../dist/webpack.ts.js');
|
module.exports = require('../dist/webpack.js');
|
||||||
Reference in New Issue
Block a user