-
-
Notifications
You must be signed in to change notification settings - Fork 71
/
Copy pathrollup-folder-zip-string.js
136 lines (112 loc) · 4.17 KB
/
rollup-folder-zip-string.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
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
/* eslint-disable no-console */
import JSZip from 'jszip';
import fs from 'fs';
import path from 'path';
import minify from 'minify';
import MagicString from 'magic-string';
const importSuffix = "!zipStringEncoded";
const stringMap = new Map();
let counter = 0;
export default function zipStringEncoded() {
return {
name: 'zip-string-encoded', // this name will show up in warnings and errors
resolveId ( source ) {
if (source.endsWith(importSuffix)) {
return source; // this signals that rollup should not ask other plugins or check the file system to find this id
}
return null; // other ids should be handled as usually
},
async load ( id ) {
if (id.endsWith(importSuffix)) {
const folder = id.substr(0, id.length - importSuffix.length);
const zip = await getZipOfFolder(folder);
const theString = await zip.generateAsync({type: "string", compression: "DEFLATE", compressionOptions: {level: 9}});
const placeholder = `rollupZipStringEncodedNo${counter}`;
const replacementCode = theString.replaceAll("*", "* ");
stringMap.set(placeholder, replacementCode);
return `const a = function(){/*@preserve${placeholder}*/};const s=a.toString();const s2 = s.substring(22,s.length-3); const res= s2.replaceAll("* ", "*"); export default res;`; // the source code for "virtual-module"
}
return null; // other ids should be handled as usually
},
renderChunk(code) {
const magicString = new MagicString(code);
for (const [placeholder, replacementCode] of stringMap.entries()) {
const pattern = new RegExp(placeholder, 'gm');
let match;
let i = 0;
while ((match = pattern.exec(code)) !== null) {
const pattern = new RegExp(placeholder);
if (match.index === pattern.lastIndex) {
pattern.lastIndex++;
}
const start = match.index;
const end = start + match[0].length;
magicString.overwrite(start, end, replacementCode);
}
}
return {code: magicString.toString(), map: magicString.generateMap({ hires: true })};
}
};
}
const getFilePathsRecursively = (dir) => {
// returns a flat array of absolute paths of all files recursively contained in the dir
let results = [];
let list = fs.readdirSync(dir);
var pending = list.length;
if (!pending) return results;
for (let file of list) {
file = path.resolve(dir, file);
let stat = fs.lstatSync(file);
if (stat && stat.isDirectory()) {
results = results.concat(getFilePathsRecursively(file));
} else {
results.push(file);
}
if (!--pending) return results;
}
return results;
};
const compressibleFormats = new Set(["html", "js", "img"]);
const getZipOfFolder = async (dir) => {
// returns a JSZip instance filled with contents of dir.
let allPaths = getFilePathsRecursively(dir);
let zip = new JSZip();
for (let filePath of allPaths) {
let addPath = slash(path.relative(dir, filePath)); // use this instead if you don't want the source folder itself in the zip
let ext = filePath.split('.').pop().trim();
let data;
if(compressibleFormats.has(ext)) {
try {
console.log("minifying ", filePath)
data = await minify(filePath)
} catch(e) {
console.log("Minification Failed for ", filePath)
}
}
if(!data) {
data = fs.readFileSync(filePath);
}
let stat = fs.lstatSync(filePath);
let permissions = stat.mode;
if (stat.isSymbolicLink()) {
zip.file(addPath, fs.readlinkSync(filePath), {
unixPermissions: parseInt('120755', 8), // This permission can be more permissive than necessary for non-executables but we don't mind.
dir: stat.isDirectory()
});
} else {
zip.file(addPath, data, {
unixPermissions: permissions,
dir: stat.isDirectory()
});
}
}
return zip;
};
function slash(path) {
const isExtendedLengthPath = /^\\\\\?\\/.test(path);
const hasNonAscii = /[^\u0000-\u0080]+/.test(path); // eslint-disable-line no-control-regex
if (isExtendedLengthPath || hasNonAscii) {
return path;
}
return path.replace(/\\/g, '/');
}