-
Notifications
You must be signed in to change notification settings - Fork 22
/
assets_context.js
104 lines (85 loc) · 2.59 KB
/
assets_context.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
'use strict';
const assert = require('assert');
const utility = require('utility');
const CONTEXT_TEMPLATE_ID = 'context' + utility.sha1(String(Date.now()));
// URL consists of host, resourceBase and entry,
// E.X. http://127.0.0.1:7001/public/index.js
// host is http://127.0.0.1:7001
// resourceBase is /public/
// entry is index.js
class Assets {
constructor(ctx) {
this.ctx = ctx;
this.config = ctx.app.config.assets;
this.isLocal = this.config.isLocal;
this.manifest = this.config.manifest;
// publicPath should contain trailing / and leading /
this.publicPath = this.isLocal ? '/' : normalizePublicPath(this.config.publicPath);
}
get host() {
return this.config.url;
}
get resourceBase() {
return `${this.host}${this.publicPath}`;
}
getStyle(entry) {
entry = entry || this.entryCss;
return linkTpl({ url: this.getURL(entry) });
}
getScript(entry) {
entry = entry || this.entry;
let script = '';
if (this.publicPath) {
script = `<script>window.__webpack_public_path__ = '${this.publicPath}';</script>`;
}
script += scriptTpl({ url: this.getURL(entry) });
return script;
}
getContext(data) {
data = safeStringify(data || this.assetsContext);
let ret = `<div id="${CONTEXT_TEMPLATE_ID}" style="display:none">${data}</div>\n`;
ret += `<script>window.${this.config.contextKey} = JSON.parse(document.getElementById('${CONTEXT_TEMPLATE_ID}').textContent || '{}');</script>`;
return ret;
}
getURL(entry) {
let urlpath = entry;
if (!this.isLocal) {
urlpath = this.manifest[urlpath];
assert(urlpath, `Don't find ${entry} in manifest.json`);
}
return `${this.resourceBase}${urlpath}`;
}
setEntry(entry) {
this.entry = entry.replace(/\.jsx?$/, '.js');
this.entryCss = entry.replace(/\.jsx?$/, '.css');
}
setContext(context) {
this.assetsContext = context;
}
}
module.exports = Assets;
function linkTpl({ url }) {
return `<link rel="stylesheet" href="${url}"></link>`;
}
function scriptTpl({ url }) {
return `<script src="${url}"></script>`;
}
const escapeMap = {
'<': '<',
'>': '>',
};
function safeStringify(data) {
if (!data) return '';
return JSON.stringify(data)
.replace(/[<>]/g, function(ch) {
return escapeMap[ch];
});
}
function normalizePublicPath(publicPath) {
if (!publicPath) return '/';
let firstIndex = 0;
if (publicPath[firstIndex] === '/') firstIndex++;
let lastIndex = publicPath.length - 1;
if (publicPath[lastIndex] === '/') lastIndex--;
return '/' + publicPath.slice(firstIndex, lastIndex + 1) + '/';
}