├── .gitignore
├── .vscode
├── launch.json
└── tasks.json
├── LICENSE
├── README.md
├── src
├── home
│ ├── 404.html
│ ├── README.md
│ ├── assets
│ │ ├── bundle.debug.js
│ │ ├── cors_v1.txt
│ │ ├── ico
│ │ │ ├── blogger.png
│ │ │ ├── facebook.png
│ │ │ ├── flickr.png
│ │ │ ├── gist.png
│ │ │ ├── google.png
│ │ │ ├── quora.png
│ │ │ ├── reddit.png
│ │ │ ├── twitch.png
│ │ │ ├── twitter.png
│ │ │ ├── wiki.png
│ │ │ └── youtube.png
│ │ └── index_v3.html
│ ├── build.sh
│ ├── conf.js
│ └── sw.js
└── proxy
│ ├── .eslintrc.json
│ ├── .package-lock.json
│ ├── .package.json
│ ├── README.md
│ ├── debug.sh
│ ├── gen-tld
│ ├── README.md
│ └── gen.js
│ ├── jsconfig.json
│ ├── release.sh
│ └── src
│ ├── cdn.js
│ ├── client.js
│ ├── cookie.js
│ ├── database.js
│ ├── env.js
│ ├── fakeloc.js
│ ├── hook.js
│ ├── index.js
│ ├── inject.js
│ ├── jsfilter.js
│ ├── msg.js
│ ├── network.js
│ ├── page.js
│ ├── path.js
│ ├── route.js
│ ├── signal.js
│ ├── storage.js
│ ├── sw.js
│ ├── tld-data.js
│ ├── tld.js
│ ├── urlx.js
│ └── util.js
└── www
├── 404.html
├── assets
├── README.md
├── cors_v1.txt
├── ico
│ ├── blogger.png
│ ├── facebook.png
│ ├── flickr.png
│ ├── gist.png
│ ├── google.png
│ ├── quora.png
│ ├── reddit.png
│ ├── twitch.png
│ ├── twitter.png
│ ├── wiki.png
│ └── youtube.png
└── index_v3.html
├── conf.js
└── sw.js
/.gitignore:
--------------------------------------------------------------------------------
1 | ._*
2 | node_modules
3 | src/home/assets/bundle.debug.js
4 | www/assets/bundle.*.js
--------------------------------------------------------------------------------
/.vscode/launch.json:
--------------------------------------------------------------------------------
1 | {
2 | // 使用 IntelliSense 了解相关属性。
3 | // 悬停以查看现有属性的描述。
4 | // 欲了解更多信息,请访问: https://go.microsoft.com/fwlink/?linkid=830387
5 | "version": "0.2.0",
6 | "configurations": [
7 | {
8 | "name": "clang build and debug active file",
9 | "type": "cppdbg",
10 | "request": "launch",
11 | "program": "${fileDirname}/${fileBasenameNoExtension}",
12 | "args": [],
13 | "stopAtEntry": false,
14 | "cwd": "${workspaceFolder}",
15 | "environment": [],
16 | "externalConsole": false,
17 | "MIMode": "lldb",
18 | "preLaunchTask": "clang build active file"
19 | }
20 | ]
21 | }
--------------------------------------------------------------------------------
/.vscode/tasks.json:
--------------------------------------------------------------------------------
1 | {
2 | "tasks": [
3 | {
4 | "type": "shell",
5 | "label": "clang build active file",
6 | "command": "/usr/bin/clang",
7 | "args": [
8 | "-g",
9 | "${file}",
10 | "-o",
11 | "${fileDirname}/${fileBasenameNoExtension}"
12 | ],
13 | "options": {
14 | "cwd": "/usr/bin"
15 | }
16 | }
17 | ],
18 | "version": "2.0.0"
19 | }
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | MIT License
2 |
3 | Copyright (c) 2019 EtherDream
4 |
5 | Permission is hereby granted, free of charge, to any person obtaining a copy
6 | of this software and associated documentation files (the "Software"), to deal
7 | in the Software without restriction, including without limitation the rights
8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9 | copies of the Software, and to permit persons to whom the Software is
10 | furnished to do so, subject to the following conditions:
11 |
12 | The above copyright notice and this permission notice shall be included in all
13 | copies or substantial portions of the Software.
14 |
15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21 | SOFTWARE.
22 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | JsProxy 浏览器端程序
2 |
3 |
4 | # 修改配置
5 |
6 | 修改 `www/conf.js` 配置,发布 `www` 目录即可。
7 |
8 |
9 | ## TODO
10 |
11 | * 增加网络错误重试、优先选择空闲节点功能
12 |
13 | * 在 SW 中替换 HTML 的 URL 属性,并支持流模式
14 |
15 | * CDN 文件使用 brotli 压缩
16 |
17 | * 使用 AST 修改 JS 代码
18 |
19 | * 动态页面压缩传输(反模板引擎,只传输变量,模板本身存储在 CDN)
20 |
21 | * 使用更多的免费图床作为 CDN 资源存储,并支持 Hash 校验
22 |
23 | * 计算程序使用 wasm 实现
24 |
25 | * 支持 blob/data/javascript 协议
26 |
27 | * 增加 qos 功能,优先满足资料查询网站流量
28 |
29 | * 改进同源策略的安全性,增加部分 API 的授权界面
30 |
31 | * 重新设计首页,增加更多功能
32 |
33 | * 完整的测试案例
34 |
35 |
36 | # 已知问题
37 |
38 | * 文件下载对话框取消后 SW 仍在下载(fetch.signal 的 onabort 未触发,可能是浏览器 BUG)
39 |
40 | * Chrome 图片无法保存
41 |
42 | * 非 UTF8 编码的 JS 会出现乱码(MIME 未指定 charset 的情况下出现)
43 |
44 | * Google 登陆页无法下一步
45 |
46 | * Google reCAPTCHA 无法执行
47 |
48 | * Google Maps 切换到卫星地图后卡死
49 |
50 | * iOS Safari 无法播放 Youtube 视频
51 |
52 | * twitter 在 Chrome 普通模式下无法登陆,但隐身模式可以
53 |
54 | * twitter iframe 经常加载不出来
55 |
56 | * SVG 脚本没有处理
57 |
58 | * Youtube 视频全屏播放会卡住
59 |
60 | * twitch.tv 首页报错(JS 代码修改导致错误,需要在 AST 层面修改)
--------------------------------------------------------------------------------
/src/home/404.html:
--------------------------------------------------------------------------------
1 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
74 |
75 |
--------------------------------------------------------------------------------
/src/home/README.md:
--------------------------------------------------------------------------------
1 | ## 404.html
2 |
3 | 该页面用于安装 Service Worker。
4 |
5 | 因为本项目只有一个 404.html,所以访问任意路径都会触发该页面。通过该页面安装 Service Worker 然后自动刷新,后续所有流量都可被 Service Worker 接管。
6 |
7 |
8 | ## sw.js
9 |
10 | Service Worker 脚本。
11 |
12 | 虽然 Web 标准规定 Service Worker 脚本必须和站点同源,但是 JS 可通过 `importScripts` 加载站外脚本。因此可将主体代码部署在 CDN,该脚本仅仅作为一个加载器。
13 |
14 |
15 | ## build.sh
16 |
17 | 压缩当前目录 404.html 到 ../../www/404.html
18 |
19 |
20 | ## conf.js
21 |
22 | 配置文件。该文件首次运行时动态加载
23 |
24 |
25 | ## assets
26 |
27 | 该目录存放静态资源,可部署到 CDN。
--------------------------------------------------------------------------------
/src/home/assets/cors_v1.txt:
--------------------------------------------------------------------------------
1 | # HTTP 返回头存在 access-control-allow-origin: * 的站点,不走代理直接连接
2 | # 收集了部分,实验中...
3 |
4 | # google
5 | ssl.google-analytics.com
6 |
7 | # [public]
8 | cdn.jsdelivr.net
9 | unpkg.com
10 | cdnjs.cloudflare.com
11 | cdn.bootcss.com
12 | use.fontawesome.com
13 | fast.fonts.net
14 | script.hotjar.com
15 |
16 | # github
17 | github.githubassets.com
18 | avatars0.githubusercontent.com
19 | avatars1.githubusercontent.com
20 | avatars2.githubusercontent.com
21 | avatars3.githubusercontent.com
22 |
23 | desktop.github.com
24 |
25 | # flickr
26 | status.flickr.net
27 |
28 | # ali
29 | at.alicdn.com
30 | img.alicdn.com
31 | g.alicdn.com
32 | i.alicdn.com
33 | atanx.alicdn.com
34 | wwc.alicdn.com
35 | gw.alicdn.com
36 | assets.alicdn.com
37 | aeis.alicdn.com
38 | atanx.alicdn.com
39 | hudong.alicdn.com
40 | gma.alicdn.com
41 |
42 | sc01.alicdn.com
43 | sc02.alicdn.com
44 | sc03.alicdn.com
45 | sc04.alicdn.com
46 |
47 | cbu01.alicdn.com
48 | cbu02.alicdn.com
49 | cbu03.alicdn.com
50 | cbu04.alicdn.com
51 |
52 | # baidu
53 | # img*.bdimg.com
54 | img0.bdimg.com
55 | img1.bdimg.com
56 | img2.bdimg.com
57 | img3.bdimg.com
58 | img4.bdimg.com
59 | img5.bdimg.com
60 |
61 | webmap0.bdimg.com
62 | webmap1.bdimg.com
63 | iknowpc.bdimg.com
64 | bkssl.bdimg.com
65 | baikebcs.bdimg.com
66 | gh.bdstatic.com
67 |
68 | # qq
69 | 3gimg.qq.com
70 | combo.b.qq.com
71 |
72 | # taotiao
73 | images.taboola.com
74 | images.taboola.com.cn
75 | images-dup.taboola.com
76 |
77 | # zhihu
78 | static.zhihu.com
79 | pic1.zhimg.com
80 | pic2.zhimg.com
81 | pic3.zhimg.com
82 | pic4.zhimg.com
83 | pic5.zhimg.com
84 | pic7.zhimg.com
85 |
86 | # jd
87 | img11.360buyimg.com
88 |
89 | # jianshu
90 | upload.jianshu.io
91 | upload-images.jianshu.io
92 | cdn2.jianshu.io
93 |
94 | # 163
95 | urswebzj.nosdn.127.net
96 | static.ws.126.net
97 | img1.cache.netease.com
98 | img2.cache.netease.com
99 | img3.cache.netease.com
100 | img4.cache.netease.com
101 | img5.cache.netease.com
102 | img6.cache.netease.com
103 |
104 | # sina
105 | js.t.sinajs.cn
106 | mjs.sinaimg.cn
107 | h5.sinaimg.cn
108 |
109 | # sohu
110 | 0d077ef9e74d8.cdn.sohucs.com
111 | 39d0825d09f05.cdn.sohucs.com
112 | 5b0988e595225.cdn.sohucs.com
113 | caaceed4aeaf2.cdn.sohucs.com
114 |
115 | img01.sogoucdn.com
116 | img02.sogoucdn.com
117 | img03.sogoucdn.com
118 | img04.sogoucdn.com
119 | img05.sogoucdn.com
120 |
121 | # hupu
122 | w1.hoopchina.com.cn
123 | w2.hoopchina.com.cn
124 | w3.hoopchina.com.cn
125 | w4.hoopchina.com.cn
126 | shihuo.hupucdn.com
127 |
128 | # uc
129 | image.uc.cn
130 |
131 | # ...
132 | static.cnodejs.org
133 | static2.cnodejs.org
134 | 2b.zol-img.com.cn
135 | img.pconline.com.cn
136 | angular.cn
137 | img1.dxycdn.com
138 | cdn.kastatic.org
139 | static.geetest.com
140 | cdn.registerdisney.go.com
141 | secure-us.imrworldwide.com
142 | img1.doubanio.com
143 | qnwww2.autoimg.cn
144 | qnwww3.autoimg.cn
145 | s.autoimg.cn
146 |
147 | hb.imgix.net
148 | main.qcloudimg.com
149 | vz-cdn2.contentabc.com
150 | twemoji.maxcdn.com
151 | fgn.cdn.serverable.com
152 |
153 | s1.hdslb.com
154 | s2.hdslb.com
155 | s3.hdslb.com
156 |
157 | # cnblogs
158 | common.cnblogs.com
159 | mathjax.cnblogs.com
160 |
161 | # csdn
162 | csdnimg.cn
163 | g.csdnimg.cn
164 | img-ads.csdn.net
165 | img-bss.csdn.net
166 | img-blog.csdn.net
167 |
168 | # ...
169 | static.geekbang.org
170 | static001.infoq.cn
171 | static.docs.com
172 | cdn1.developermedia.com
173 | cdn2.developermedia.com
174 | cdn.optimizely.com
175 | cdn.ampproject.org
176 |
177 | camshowverse.to
178 | static.camshowhub-cdn.to
179 |
180 | xqimg.imedao.com
181 | xavatar.imedao.com
182 |
183 | # ???
184 | img-l3.xvideos-cdn.com
185 | static-egc.xvideos-cdn.com
186 | img-hw.xvideos-cdn.com
187 | img-hw.xnxx-cdn.com
188 | static-egc.xnxx-cdn.com
189 | di.phncdn.com
190 | cv.phncdn.com
191 | roomimg.stream.highwebmedia.com
192 | w3.cdn.anvato.net
--------------------------------------------------------------------------------
/src/home/assets/ico/blogger.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/EtherDream/jsproxy-browser/a7a6d740e2c64b1a029fe9ced8ffe9f786e154dd/src/home/assets/ico/blogger.png
--------------------------------------------------------------------------------
/src/home/assets/ico/facebook.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/EtherDream/jsproxy-browser/a7a6d740e2c64b1a029fe9ced8ffe9f786e154dd/src/home/assets/ico/facebook.png
--------------------------------------------------------------------------------
/src/home/assets/ico/flickr.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/EtherDream/jsproxy-browser/a7a6d740e2c64b1a029fe9ced8ffe9f786e154dd/src/home/assets/ico/flickr.png
--------------------------------------------------------------------------------
/src/home/assets/ico/gist.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/EtherDream/jsproxy-browser/a7a6d740e2c64b1a029fe9ced8ffe9f786e154dd/src/home/assets/ico/gist.png
--------------------------------------------------------------------------------
/src/home/assets/ico/google.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/EtherDream/jsproxy-browser/a7a6d740e2c64b1a029fe9ced8ffe9f786e154dd/src/home/assets/ico/google.png
--------------------------------------------------------------------------------
/src/home/assets/ico/quora.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/EtherDream/jsproxy-browser/a7a6d740e2c64b1a029fe9ced8ffe9f786e154dd/src/home/assets/ico/quora.png
--------------------------------------------------------------------------------
/src/home/assets/ico/reddit.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/EtherDream/jsproxy-browser/a7a6d740e2c64b1a029fe9ced8ffe9f786e154dd/src/home/assets/ico/reddit.png
--------------------------------------------------------------------------------
/src/home/assets/ico/twitch.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/EtherDream/jsproxy-browser/a7a6d740e2c64b1a029fe9ced8ffe9f786e154dd/src/home/assets/ico/twitch.png
--------------------------------------------------------------------------------
/src/home/assets/ico/twitter.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/EtherDream/jsproxy-browser/a7a6d740e2c64b1a029fe9ced8ffe9f786e154dd/src/home/assets/ico/twitter.png
--------------------------------------------------------------------------------
/src/home/assets/ico/wiki.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/EtherDream/jsproxy-browser/a7a6d740e2c64b1a029fe9ced8ffe9f786e154dd/src/home/assets/ico/wiki.png
--------------------------------------------------------------------------------
/src/home/assets/ico/youtube.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/EtherDream/jsproxy-browser/a7a6d740e2c64b1a029fe9ced8ffe9f786e154dd/src/home/assets/ico/youtube.png
--------------------------------------------------------------------------------
/src/home/assets/index_v3.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | Page Sandbox
5 |
6 |
7 |
8 |
29 |
30 |
31 |
32 |
33 |
34 |
35 |
36 |
37 |
38 | 切换线路:
39 |
40 |
41 |
42 |
43 |
167 |
168 |
--------------------------------------------------------------------------------
/src/home/build.sh:
--------------------------------------------------------------------------------
1 | # 压缩 404.html
2 | html-minifier \
3 | --collapse-whitespace \
4 | --remove-comments \
5 | --remove-redundant-attributes \
6 | --remove-script-type-attributes \
7 | --remove-tag-whitespace \
8 | --use-short-doctype \
9 | --remove-attribute-quotes \
10 | --minify-css true \
11 | --minify-js '{"toplevel": true, "ie8": true}' \
12 | -o ../../www/404.html \
13 | 404.html
14 |
--------------------------------------------------------------------------------
/src/home/conf.js:
--------------------------------------------------------------------------------
1 | jsproxy_config({
2 | // 当前配置的版本(记录在日志中,用于排查问题)
3 | // 每次修改配置,该值需要增加,否则不会生效。
4 | // 默认每隔 5 分钟自动下载配置,若想立即验证,可通过隐私模式访问。
5 | ver: '105',
6 |
7 | // 通过 CDN 加速常用网站的静态资源(实验中)
8 | static_boost: {
9 | enable: true,
10 | ver: 56
11 | },
12 |
13 | // 节点配置
14 | node_map: {
15 | 'demo-hk': {
16 | label: '演示服务-香港节点',
17 | lines: {
18 | // 主机:权重
19 | 'node-aliyun-hk-0.etherdream.com:8443': 1,
20 | 'node-aliyun-hk-1.etherdream.com:8443': 1,
21 | 'node-aliyun-hk-2.etherdream.com:8443': 1,
22 | }
23 | },
24 | 'demo-sg': {
25 | label: '演示服务-新加坡节点',
26 | lines: {
27 | 'node-aliyun-sg.etherdream.com:8443': 1,
28 | },
29 | },
30 | 'demo-la': {
31 | label: '演示服务-洛杉矶节点',
32 | lines: {
33 | 'node-bwh-la.etherdream.com:8443': 1,
34 | },
35 | },
36 | 'mysite': {
37 | label: '当前站点',
38 | lines: {
39 | [location.host]: 1,
40 | }
41 | },
42 | // 该节点用于加载大体积的静态资源
43 | 'cfworker': {
44 | label: '',
45 | hidden: true,
46 | lines: {
47 | 'node-cfworker-2.etherdream.com': 1,
48 | }
49 | }
50 | },
51 |
52 | /**
53 | * 默认节点
54 | */
55 | node_default: 'demo-hk',
56 |
57 | /**
58 | * 加速节点
59 | */
60 | node_acc: 'cfworker',
61 |
62 | /**
63 | * 静态资源 CDN 地址
64 | * 用于加速 `assets` 目录中的资源访问
65 | */
66 | // assets_cdn: 'https://cdn.jsdelivr.net/gh/zjcqoo/zjcqoo.github.io@master/assets/',
67 |
68 | // 本地测试时打开,否则访问的是线上的
69 | assets_cdn: 'assets/',
70 |
71 | // 首页路径
72 | index_path: 'index_v3.html',
73 |
74 | // 支持 CORS 的站点列表(实验中...)
75 | direct_host_list: 'cors_v1.txt',
76 |
77 | /**
78 | * 自定义注入页面的 HTML
79 | */
80 | inject_html: '',
81 |
82 | /**
83 | * URL 自定义处理(设计中)
84 | */
85 | url_handler: {
86 | 'https://www.baidu.com/img/baidu_resultlogo@2.png': {
87 | replace: 'https://www.google.com/images/branding/googlelogo/2x/googlelogo_color_92x30dp.png'
88 | },
89 | 'https://www.pornhub.com/': {
90 | redir: 'https://php.net/'
91 | },
92 | 'http://haha.com/': {
93 | content: 'Hello World'
94 | },
95 | }
96 | })
--------------------------------------------------------------------------------
/src/home/sw.js:
--------------------------------------------------------------------------------
1 | jsproxy_config=x=>{__CONF__=x;importScripts(__FILE__=x.assets_cdn+'bundle.debug.js')};importScripts('conf.js')
--------------------------------------------------------------------------------
/src/proxy/.eslintrc.json:
--------------------------------------------------------------------------------
1 | {
2 | "env": {
3 | "browser": true,
4 | "es6": true,
5 | "serviceworker": true
6 | },
7 | "extends": "eslint:recommended",
8 | "globals": {
9 | "Atomics": "readonly",
10 | "SharedArrayBuffer": "readonly"
11 | },
12 | "parserOptions": {
13 | "ecmaVersion": 2017,
14 | "sourceType": "module"
15 | },
16 | "rules": {
17 | "no-console": "warn",
18 | "no-empty": "warn",
19 | "no-unused-vars": "warn",
20 | "no-debugger": "warn",
21 | "no-constant-condition": "warn"
22 | }
23 | }
--------------------------------------------------------------------------------
/src/proxy/.package-lock.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "jsproxy-client",
3 | "version": "0.0.1",
4 | "lockfileVersion": 1,
5 | "requires": true,
6 | "dependencies": {
7 | "@babel/code-frame": {
8 | "version": "7.0.0",
9 | "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.0.0.tgz",
10 | "integrity": "sha512-OfC2uemaknXr87bdLUkWog7nYuliM9Ij5HUcajsVcMCpQrcLmtxRbVFTIqmcSkSeYRBFBRxs2FiUqFJDLdiebA==",
11 | "dev": true,
12 | "requires": {
13 | "@babel/highlight": "^7.0.0"
14 | }
15 | },
16 | "@babel/highlight": {
17 | "version": "7.0.0",
18 | "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.0.0.tgz",
19 | "integrity": "sha512-UFMC4ZeFC48Tpvj7C8UgLvtkaUuovQX+5xNWrsIoMG8o2z+XFKjKaN9iVmS84dPwVN00W4wPmqvYoZF3EGAsfw==",
20 | "dev": true,
21 | "requires": {
22 | "chalk": "^2.0.0",
23 | "esutils": "^2.0.2",
24 | "js-tokens": "^4.0.0"
25 | }
26 | },
27 | "acorn": {
28 | "version": "6.1.1",
29 | "resolved": "https://registry.npmjs.org/acorn/-/acorn-6.1.1.tgz",
30 | "integrity": "sha512-jPTiwtOxaHNaAPg/dmrJ/beuzLRnXtB0kQPQ8JpotKJgTB6rX6c8mlf315941pyjBSaPg8NHXS9fhP4u17DpGA==",
31 | "dev": true
32 | },
33 | "acorn-jsx": {
34 | "version": "5.0.1",
35 | "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.0.1.tgz",
36 | "integrity": "sha512-HJ7CfNHrfJLlNTzIEUTj43LNWGkqpRLxm3YjAlcD0ACydk9XynzYsCBHxut+iqt+1aBXkx9UP/w/ZqMr13XIzg==",
37 | "dev": true
38 | },
39 | "ajv": {
40 | "version": "6.10.0",
41 | "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.10.0.tgz",
42 | "integrity": "sha512-nffhOpkymDECQyR0mnsUtoCE8RlX38G0rYP+wgLWFyZuUyuuojSSvi/+euOiQBIn63whYwYVIIH1TvE3tu4OEg==",
43 | "dev": true,
44 | "requires": {
45 | "fast-deep-equal": "^2.0.1",
46 | "fast-json-stable-stringify": "^2.0.0",
47 | "json-schema-traverse": "^0.4.1",
48 | "uri-js": "^4.2.2"
49 | }
50 | },
51 | "ansi-escapes": {
52 | "version": "3.2.0",
53 | "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-3.2.0.tgz",
54 | "integrity": "sha512-cBhpre4ma+U0T1oM5fXg7Dy1Jw7zzwv7lt/GoCpr+hDQJoYnKVPLL4dCvSEFMmQurOQvSrwT7SL/DAlhBI97RQ==",
55 | "dev": true
56 | },
57 | "ansi-regex": {
58 | "version": "3.0.0",
59 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.0.tgz",
60 | "integrity": "sha1-7QMXwyIGT3lGbAKWa922Bas32Zg=",
61 | "dev": true
62 | },
63 | "ansi-styles": {
64 | "version": "3.2.1",
65 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz",
66 | "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==",
67 | "dev": true,
68 | "requires": {
69 | "color-convert": "^1.9.0"
70 | }
71 | },
72 | "argparse": {
73 | "version": "1.0.10",
74 | "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz",
75 | "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==",
76 | "dev": true,
77 | "requires": {
78 | "sprintf-js": "~1.0.2"
79 | }
80 | },
81 | "astral-regex": {
82 | "version": "1.0.0",
83 | "resolved": "https://registry.npmjs.org/astral-regex/-/astral-regex-1.0.0.tgz",
84 | "integrity": "sha512-+Ryf6g3BKoRc7jfp7ad8tM4TtMiaWvbF/1/sQcZPkkS7ag3D5nMBCe2UfOTONtAkaG0tO0ij3C5Lwmf1EiyjHg==",
85 | "dev": true
86 | },
87 | "balanced-match": {
88 | "version": "1.0.0",
89 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz",
90 | "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=",
91 | "dev": true
92 | },
93 | "brace-expansion": {
94 | "version": "1.1.11",
95 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz",
96 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==",
97 | "dev": true,
98 | "requires": {
99 | "balanced-match": "^1.0.0",
100 | "concat-map": "0.0.1"
101 | }
102 | },
103 | "callsites": {
104 | "version": "3.1.0",
105 | "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz",
106 | "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==",
107 | "dev": true
108 | },
109 | "chalk": {
110 | "version": "2.4.2",
111 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz",
112 | "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==",
113 | "dev": true,
114 | "requires": {
115 | "ansi-styles": "^3.2.1",
116 | "escape-string-regexp": "^1.0.5",
117 | "supports-color": "^5.3.0"
118 | }
119 | },
120 | "chardet": {
121 | "version": "0.7.0",
122 | "resolved": "https://registry.npmjs.org/chardet/-/chardet-0.7.0.tgz",
123 | "integrity": "sha512-mT8iDcrh03qDGRRmoA2hmBJnxpllMR+0/0qlzjqZES6NdiWDcZkCNAk4rPFZ9Q85r27unkiNNg8ZOiwZXBHwcA==",
124 | "dev": true
125 | },
126 | "cli-cursor": {
127 | "version": "2.1.0",
128 | "resolved": "https://registry.npmjs.org/cli-cursor/-/cli-cursor-2.1.0.tgz",
129 | "integrity": "sha1-s12sN2R5+sw+lHR9QdDQ9SOP/LU=",
130 | "dev": true,
131 | "requires": {
132 | "restore-cursor": "^2.0.0"
133 | }
134 | },
135 | "cli-width": {
136 | "version": "2.2.0",
137 | "resolved": "https://registry.npmjs.org/cli-width/-/cli-width-2.2.0.tgz",
138 | "integrity": "sha1-/xnt6Kml5XkyQUewwR8PvLq+1jk=",
139 | "dev": true
140 | },
141 | "color-convert": {
142 | "version": "1.9.3",
143 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz",
144 | "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==",
145 | "dev": true,
146 | "requires": {
147 | "color-name": "1.1.3"
148 | }
149 | },
150 | "color-name": {
151 | "version": "1.1.3",
152 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz",
153 | "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=",
154 | "dev": true
155 | },
156 | "concat-map": {
157 | "version": "0.0.1",
158 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz",
159 | "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=",
160 | "dev": true
161 | },
162 | "cross-spawn": {
163 | "version": "6.0.5",
164 | "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-6.0.5.tgz",
165 | "integrity": "sha512-eTVLrBSt7fjbDygz805pMnstIs2VTBNkRm0qxZd+M7A5XDdxVRWO5MxGBXZhjY4cqLYLdtrGqRf8mBPmzwSpWQ==",
166 | "dev": true,
167 | "requires": {
168 | "nice-try": "^1.0.4",
169 | "path-key": "^2.0.1",
170 | "semver": "^5.5.0",
171 | "shebang-command": "^1.2.0",
172 | "which": "^1.2.9"
173 | }
174 | },
175 | "debug": {
176 | "version": "4.1.1",
177 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.1.1.tgz",
178 | "integrity": "sha512-pYAIzeRo8J6KPEaJ0VWOh5Pzkbw/RetuzehGM7QRRX5he4fPHx2rdKMB256ehJCkX+XRQm16eZLqLNS8RSZXZw==",
179 | "dev": true,
180 | "requires": {
181 | "ms": "^2.1.1"
182 | }
183 | },
184 | "deep-is": {
185 | "version": "0.1.3",
186 | "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.3.tgz",
187 | "integrity": "sha1-s2nW+128E+7PUk+RsHD+7cNXzzQ=",
188 | "dev": true
189 | },
190 | "doctrine": {
191 | "version": "3.0.0",
192 | "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-3.0.0.tgz",
193 | "integrity": "sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==",
194 | "dev": true,
195 | "requires": {
196 | "esutils": "^2.0.2"
197 | }
198 | },
199 | "emoji-regex": {
200 | "version": "7.0.3",
201 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-7.0.3.tgz",
202 | "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==",
203 | "dev": true
204 | },
205 | "escape-string-regexp": {
206 | "version": "1.0.5",
207 | "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz",
208 | "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=",
209 | "dev": true
210 | },
211 | "eslint": {
212 | "version": "5.16.0",
213 | "resolved": "https://registry.npmjs.org/eslint/-/eslint-5.16.0.tgz",
214 | "integrity": "sha512-S3Rz11i7c8AA5JPv7xAH+dOyq/Cu/VXHiHXBPOU1k/JAM5dXqQPt3qcrhpHSorXmrpu2g0gkIBVXAqCpzfoZIg==",
215 | "dev": true,
216 | "requires": {
217 | "@babel/code-frame": "^7.0.0",
218 | "ajv": "^6.9.1",
219 | "chalk": "^2.1.0",
220 | "cross-spawn": "^6.0.5",
221 | "debug": "^4.0.1",
222 | "doctrine": "^3.0.0",
223 | "eslint-scope": "^4.0.3",
224 | "eslint-utils": "^1.3.1",
225 | "eslint-visitor-keys": "^1.0.0",
226 | "espree": "^5.0.1",
227 | "esquery": "^1.0.1",
228 | "esutils": "^2.0.2",
229 | "file-entry-cache": "^5.0.1",
230 | "functional-red-black-tree": "^1.0.1",
231 | "glob": "^7.1.2",
232 | "globals": "^11.7.0",
233 | "ignore": "^4.0.6",
234 | "import-fresh": "^3.0.0",
235 | "imurmurhash": "^0.1.4",
236 | "inquirer": "^6.2.2",
237 | "js-yaml": "^3.13.0",
238 | "json-stable-stringify-without-jsonify": "^1.0.1",
239 | "levn": "^0.3.0",
240 | "lodash": "^4.17.11",
241 | "minimatch": "^3.0.4",
242 | "mkdirp": "^0.5.1",
243 | "natural-compare": "^1.4.0",
244 | "optionator": "^0.8.2",
245 | "path-is-inside": "^1.0.2",
246 | "progress": "^2.0.0",
247 | "regexpp": "^2.0.1",
248 | "semver": "^5.5.1",
249 | "strip-ansi": "^4.0.0",
250 | "strip-json-comments": "^2.0.1",
251 | "table": "^5.2.3",
252 | "text-table": "^0.2.0"
253 | }
254 | },
255 | "eslint-scope": {
256 | "version": "4.0.3",
257 | "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-4.0.3.tgz",
258 | "integrity": "sha512-p7VutNr1O/QrxysMo3E45FjYDTeXBy0iTltPFNSqKAIfjDSXC+4dj+qfyuD8bfAXrW/y6lW3O76VaYNPKfpKrg==",
259 | "dev": true,
260 | "requires": {
261 | "esrecurse": "^4.1.0",
262 | "estraverse": "^4.1.1"
263 | }
264 | },
265 | "eslint-utils": {
266 | "version": "1.3.1",
267 | "resolved": "https://registry.npmjs.org/eslint-utils/-/eslint-utils-1.3.1.tgz",
268 | "integrity": "sha512-Z7YjnIldX+2XMcjr7ZkgEsOj/bREONV60qYeB/bjMAqqqZ4zxKyWX+BOUkdmRmA9riiIPVvo5x86m5elviOk0Q==",
269 | "dev": true
270 | },
271 | "eslint-visitor-keys": {
272 | "version": "1.0.0",
273 | "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-1.0.0.tgz",
274 | "integrity": "sha512-qzm/XxIbxm/FHyH341ZrbnMUpe+5Bocte9xkmFMzPMjRaZMcXww+MpBptFvtU+79L362nqiLhekCxCxDPaUMBQ==",
275 | "dev": true
276 | },
277 | "espree": {
278 | "version": "5.0.1",
279 | "resolved": "https://registry.npmjs.org/espree/-/espree-5.0.1.tgz",
280 | "integrity": "sha512-qWAZcWh4XE/RwzLJejfcofscgMc9CamR6Tn1+XRXNzrvUSSbiAjGOI/fggztjIi7y9VLPqnICMIPiGyr8JaZ0A==",
281 | "dev": true,
282 | "requires": {
283 | "acorn": "^6.0.7",
284 | "acorn-jsx": "^5.0.0",
285 | "eslint-visitor-keys": "^1.0.0"
286 | }
287 | },
288 | "esprima": {
289 | "version": "4.0.1",
290 | "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz",
291 | "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==",
292 | "dev": true
293 | },
294 | "esquery": {
295 | "version": "1.0.1",
296 | "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.0.1.tgz",
297 | "integrity": "sha512-SmiyZ5zIWH9VM+SRUReLS5Q8a7GxtRdxEBVZpm98rJM7Sb+A9DVCndXfkeFUd3byderg+EbDkfnevfCwynWaNA==",
298 | "dev": true,
299 | "requires": {
300 | "estraverse": "^4.0.0"
301 | }
302 | },
303 | "esrecurse": {
304 | "version": "4.2.1",
305 | "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.2.1.tgz",
306 | "integrity": "sha512-64RBB++fIOAXPw3P9cy89qfMlvZEXZkqqJkjqqXIvzP5ezRZjW+lPWjw35UX/3EhUPFYbg5ER4JYgDw4007/DQ==",
307 | "dev": true,
308 | "requires": {
309 | "estraverse": "^4.1.0"
310 | }
311 | },
312 | "estraverse": {
313 | "version": "4.2.0",
314 | "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.2.0.tgz",
315 | "integrity": "sha1-De4/7TH81GlhjOc0IJn8GvoL2xM=",
316 | "dev": true
317 | },
318 | "esutils": {
319 | "version": "2.0.2",
320 | "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.2.tgz",
321 | "integrity": "sha1-Cr9PHKpbyx96nYrMbepPqqBLrJs=",
322 | "dev": true
323 | },
324 | "external-editor": {
325 | "version": "3.0.3",
326 | "resolved": "https://registry.npmjs.org/external-editor/-/external-editor-3.0.3.tgz",
327 | "integrity": "sha512-bn71H9+qWoOQKyZDo25mOMVpSmXROAsTJVVVYzrrtol3d4y+AsKjf4Iwl2Q+IuT0kFSQ1qo166UuIwqYq7mGnA==",
328 | "dev": true,
329 | "requires": {
330 | "chardet": "^0.7.0",
331 | "iconv-lite": "^0.4.24",
332 | "tmp": "^0.0.33"
333 | }
334 | },
335 | "fast-deep-equal": {
336 | "version": "2.0.1",
337 | "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-2.0.1.tgz",
338 | "integrity": "sha1-ewUhjd+WZ79/Nwv3/bLLFf3Qqkk=",
339 | "dev": true
340 | },
341 | "fast-json-stable-stringify": {
342 | "version": "2.0.0",
343 | "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.0.0.tgz",
344 | "integrity": "sha1-1RQsDK7msRifh9OnYREGT4bIu/I=",
345 | "dev": true
346 | },
347 | "fast-levenshtein": {
348 | "version": "2.0.6",
349 | "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz",
350 | "integrity": "sha1-PYpcZog6FqMMqGQ+hR8Zuqd5eRc=",
351 | "dev": true
352 | },
353 | "figures": {
354 | "version": "2.0.0",
355 | "resolved": "https://registry.npmjs.org/figures/-/figures-2.0.0.tgz",
356 | "integrity": "sha1-OrGi0qYsi/tDGgyUy3l6L84nyWI=",
357 | "dev": true,
358 | "requires": {
359 | "escape-string-regexp": "^1.0.5"
360 | }
361 | },
362 | "file-entry-cache": {
363 | "version": "5.0.1",
364 | "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-5.0.1.tgz",
365 | "integrity": "sha512-bCg29ictuBaKUwwArK4ouCaqDgLZcysCFLmM/Yn/FDoqndh/9vNuQfXRDvTuXKLxfD/JtZQGKFT8MGcJBK644g==",
366 | "dev": true,
367 | "requires": {
368 | "flat-cache": "^2.0.1"
369 | }
370 | },
371 | "flat-cache": {
372 | "version": "2.0.1",
373 | "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-2.0.1.tgz",
374 | "integrity": "sha512-LoQe6yDuUMDzQAEH8sgmh4Md6oZnc/7PjtwjNFSzveXqSHt6ka9fPBuso7IGf9Rz4uqnSnWiFH2B/zj24a5ReA==",
375 | "dev": true,
376 | "requires": {
377 | "flatted": "^2.0.0",
378 | "rimraf": "2.6.3",
379 | "write": "1.0.3"
380 | }
381 | },
382 | "flatted": {
383 | "version": "2.0.0",
384 | "resolved": "https://registry.npmjs.org/flatted/-/flatted-2.0.0.tgz",
385 | "integrity": "sha512-R+H8IZclI8AAkSBRQJLVOsxwAoHd6WC40b4QTNWIjzAa6BXOBfQcM587MXDTVPeYaopFNWHUFLx7eNmHDSxMWg==",
386 | "dev": true
387 | },
388 | "fs.realpath": {
389 | "version": "1.0.0",
390 | "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz",
391 | "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=",
392 | "dev": true
393 | },
394 | "functional-red-black-tree": {
395 | "version": "1.0.1",
396 | "resolved": "https://registry.npmjs.org/functional-red-black-tree/-/functional-red-black-tree-1.0.1.tgz",
397 | "integrity": "sha1-GwqzvVU7Kg1jmdKcDj6gslIHgyc=",
398 | "dev": true
399 | },
400 | "glob": {
401 | "version": "7.1.3",
402 | "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.3.tgz",
403 | "integrity": "sha512-vcfuiIxogLV4DlGBHIUOwI0IbrJ8HWPc4MU7HzviGeNho/UJDfi6B5p3sHeWIQ0KGIU0Jpxi5ZHxemQfLkkAwQ==",
404 | "dev": true,
405 | "requires": {
406 | "fs.realpath": "^1.0.0",
407 | "inflight": "^1.0.4",
408 | "inherits": "2",
409 | "minimatch": "^3.0.4",
410 | "once": "^1.3.0",
411 | "path-is-absolute": "^1.0.0"
412 | }
413 | },
414 | "globals": {
415 | "version": "11.12.0",
416 | "resolved": "https://registry.npmjs.org/globals/-/globals-11.12.0.tgz",
417 | "integrity": "sha512-WOBp/EEGUiIsJSp7wcv/y6MO+lV9UoncWqxuFfm8eBwzWNgyfBd6Gz+IeKQ9jCmyhoH99g15M3T+QaVHFjizVA==",
418 | "dev": true
419 | },
420 | "has-flag": {
421 | "version": "3.0.0",
422 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz",
423 | "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=",
424 | "dev": true
425 | },
426 | "iconv-lite": {
427 | "version": "0.4.24",
428 | "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz",
429 | "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==",
430 | "dev": true,
431 | "requires": {
432 | "safer-buffer": ">= 2.1.2 < 3"
433 | }
434 | },
435 | "ignore": {
436 | "version": "4.0.6",
437 | "resolved": "https://registry.npmjs.org/ignore/-/ignore-4.0.6.tgz",
438 | "integrity": "sha512-cyFDKrqc/YdcWFniJhzI42+AzS+gNwmUzOSFcRCQYwySuBBBy/KjuxWLZ/FHEH6Moq1NizMOBWyTcv8O4OZIMg==",
439 | "dev": true
440 | },
441 | "import-fresh": {
442 | "version": "3.0.0",
443 | "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.0.0.tgz",
444 | "integrity": "sha512-pOnA9tfM3Uwics+SaBLCNyZZZbK+4PTu0OPZtLlMIrv17EdBoC15S9Kn8ckJ9TZTyKb3ywNE5y1yeDxxGA7nTQ==",
445 | "dev": true,
446 | "requires": {
447 | "parent-module": "^1.0.0",
448 | "resolve-from": "^4.0.0"
449 | }
450 | },
451 | "imurmurhash": {
452 | "version": "0.1.4",
453 | "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz",
454 | "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=",
455 | "dev": true
456 | },
457 | "inflight": {
458 | "version": "1.0.6",
459 | "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz",
460 | "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=",
461 | "dev": true,
462 | "requires": {
463 | "once": "^1.3.0",
464 | "wrappy": "1"
465 | }
466 | },
467 | "inherits": {
468 | "version": "2.0.3",
469 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz",
470 | "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=",
471 | "dev": true
472 | },
473 | "inquirer": {
474 | "version": "6.3.1",
475 | "resolved": "https://registry.npmjs.org/inquirer/-/inquirer-6.3.1.tgz",
476 | "integrity": "sha512-MmL624rfkFt4TG9y/Jvmt8vdmOo836U7Y0Hxr2aFk3RelZEGX4Igk0KabWrcaaZaTv9uzglOqWh1Vly+FAWAXA==",
477 | "dev": true,
478 | "requires": {
479 | "ansi-escapes": "^3.2.0",
480 | "chalk": "^2.4.2",
481 | "cli-cursor": "^2.1.0",
482 | "cli-width": "^2.0.0",
483 | "external-editor": "^3.0.3",
484 | "figures": "^2.0.0",
485 | "lodash": "^4.17.11",
486 | "mute-stream": "0.0.7",
487 | "run-async": "^2.2.0",
488 | "rxjs": "^6.4.0",
489 | "string-width": "^2.1.0",
490 | "strip-ansi": "^5.1.0",
491 | "through": "^2.3.6"
492 | },
493 | "dependencies": {
494 | "ansi-regex": {
495 | "version": "4.1.0",
496 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz",
497 | "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==",
498 | "dev": true
499 | },
500 | "strip-ansi": {
501 | "version": "5.2.0",
502 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz",
503 | "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==",
504 | "dev": true,
505 | "requires": {
506 | "ansi-regex": "^4.1.0"
507 | }
508 | }
509 | }
510 | },
511 | "is-fullwidth-code-point": {
512 | "version": "2.0.0",
513 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz",
514 | "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=",
515 | "dev": true
516 | },
517 | "is-promise": {
518 | "version": "2.1.0",
519 | "resolved": "https://registry.npmjs.org/is-promise/-/is-promise-2.1.0.tgz",
520 | "integrity": "sha1-eaKp7OfwlugPNtKy87wWwf9L8/o=",
521 | "dev": true
522 | },
523 | "isexe": {
524 | "version": "2.0.0",
525 | "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz",
526 | "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=",
527 | "dev": true
528 | },
529 | "js-tokens": {
530 | "version": "4.0.0",
531 | "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz",
532 | "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==",
533 | "dev": true
534 | },
535 | "js-yaml": {
536 | "version": "3.13.1",
537 | "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.13.1.tgz",
538 | "integrity": "sha512-YfbcO7jXDdyj0DGxYVSlSeQNHbD7XPWvrVWeVUujrQEoZzWJIRrCPoyk6kL6IAjAG2IolMK4T0hNUe0HOUs5Jw==",
539 | "dev": true,
540 | "requires": {
541 | "argparse": "^1.0.7",
542 | "esprima": "^4.0.0"
543 | }
544 | },
545 | "json-schema-traverse": {
546 | "version": "0.4.1",
547 | "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz",
548 | "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==",
549 | "dev": true
550 | },
551 | "json-stable-stringify-without-jsonify": {
552 | "version": "1.0.1",
553 | "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz",
554 | "integrity": "sha1-nbe1lJatPzz+8wp1FC0tkwrXJlE=",
555 | "dev": true
556 | },
557 | "levn": {
558 | "version": "0.3.0",
559 | "resolved": "https://registry.npmjs.org/levn/-/levn-0.3.0.tgz",
560 | "integrity": "sha1-OwmSTt+fCDwEkP3UwLxEIeBHZO4=",
561 | "dev": true,
562 | "requires": {
563 | "prelude-ls": "~1.1.2",
564 | "type-check": "~0.3.2"
565 | }
566 | },
567 | "lodash": {
568 | "version": "4.17.11",
569 | "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.11.tgz",
570 | "integrity": "sha512-cQKh8igo5QUhZ7lg38DYWAxMvjSAKG0A8wGSVimP07SIUEK2UO+arSRKbRZWtelMtN5V0Hkwh5ryOto/SshYIg==",
571 | "dev": true
572 | },
573 | "mimic-fn": {
574 | "version": "1.2.0",
575 | "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-1.2.0.tgz",
576 | "integrity": "sha512-jf84uxzwiuiIVKiOLpfYk7N46TSy8ubTonmneY9vrpHNAnp0QBt2BxWV9dO3/j+BoVAb+a5G6YDPW3M5HOdMWQ==",
577 | "dev": true
578 | },
579 | "minimatch": {
580 | "version": "3.0.4",
581 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz",
582 | "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==",
583 | "dev": true,
584 | "requires": {
585 | "brace-expansion": "^1.1.7"
586 | }
587 | },
588 | "minimist": {
589 | "version": "0.0.8",
590 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-0.0.8.tgz",
591 | "integrity": "sha1-hX/Kv8M5fSYluCKCYuhqp6ARsF0=",
592 | "dev": true
593 | },
594 | "mkdirp": {
595 | "version": "0.5.1",
596 | "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.1.tgz",
597 | "integrity": "sha1-MAV0OOrGz3+MR2fzhkjWaX11yQM=",
598 | "dev": true,
599 | "requires": {
600 | "minimist": "0.0.8"
601 | }
602 | },
603 | "ms": {
604 | "version": "2.1.1",
605 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz",
606 | "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg==",
607 | "dev": true
608 | },
609 | "mute-stream": {
610 | "version": "0.0.7",
611 | "resolved": "https://registry.npmjs.org/mute-stream/-/mute-stream-0.0.7.tgz",
612 | "integrity": "sha1-MHXOk7whuPq0PhvE2n6BFe0ee6s=",
613 | "dev": true
614 | },
615 | "natural-compare": {
616 | "version": "1.4.0",
617 | "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz",
618 | "integrity": "sha1-Sr6/7tdUHywnrPspvbvRXI1bpPc=",
619 | "dev": true
620 | },
621 | "nice-try": {
622 | "version": "1.0.5",
623 | "resolved": "https://registry.npmjs.org/nice-try/-/nice-try-1.0.5.tgz",
624 | "integrity": "sha512-1nh45deeb5olNY7eX82BkPO7SSxR5SSYJiPTrTdFUVYwAl8CKMA5N9PjTYkHiRjisVcxcQ1HXdLhx2qxxJzLNQ==",
625 | "dev": true
626 | },
627 | "once": {
628 | "version": "1.4.0",
629 | "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz",
630 | "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=",
631 | "dev": true,
632 | "requires": {
633 | "wrappy": "1"
634 | }
635 | },
636 | "onetime": {
637 | "version": "2.0.1",
638 | "resolved": "https://registry.npmjs.org/onetime/-/onetime-2.0.1.tgz",
639 | "integrity": "sha1-BnQoIw/WdEOyeUsiu6UotoZ5YtQ=",
640 | "dev": true,
641 | "requires": {
642 | "mimic-fn": "^1.0.0"
643 | }
644 | },
645 | "optionator": {
646 | "version": "0.8.2",
647 | "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.8.2.tgz",
648 | "integrity": "sha1-NkxeQJ0/TWMB1sC0wFu6UBgK62Q=",
649 | "dev": true,
650 | "requires": {
651 | "deep-is": "~0.1.3",
652 | "fast-levenshtein": "~2.0.4",
653 | "levn": "~0.3.0",
654 | "prelude-ls": "~1.1.2",
655 | "type-check": "~0.3.2",
656 | "wordwrap": "~1.0.0"
657 | }
658 | },
659 | "os-tmpdir": {
660 | "version": "1.0.2",
661 | "resolved": "https://registry.npmjs.org/os-tmpdir/-/os-tmpdir-1.0.2.tgz",
662 | "integrity": "sha1-u+Z0BseaqFxc/sdm/lc0VV36EnQ=",
663 | "dev": true
664 | },
665 | "parent-module": {
666 | "version": "1.0.1",
667 | "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz",
668 | "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==",
669 | "dev": true,
670 | "requires": {
671 | "callsites": "^3.0.0"
672 | }
673 | },
674 | "path-is-absolute": {
675 | "version": "1.0.1",
676 | "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz",
677 | "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=",
678 | "dev": true
679 | },
680 | "path-is-inside": {
681 | "version": "1.0.2",
682 | "resolved": "https://registry.npmjs.org/path-is-inside/-/path-is-inside-1.0.2.tgz",
683 | "integrity": "sha1-NlQX3t5EQw0cEa9hAn+s8HS9/FM=",
684 | "dev": true
685 | },
686 | "path-key": {
687 | "version": "2.0.1",
688 | "resolved": "https://registry.npmjs.org/path-key/-/path-key-2.0.1.tgz",
689 | "integrity": "sha1-QRyttXTFoUDTpLGRDUDYDMn0C0A=",
690 | "dev": true
691 | },
692 | "prelude-ls": {
693 | "version": "1.1.2",
694 | "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.1.2.tgz",
695 | "integrity": "sha1-IZMqVJ9eUv/ZqCf1cOBL5iqX2lQ=",
696 | "dev": true
697 | },
698 | "progress": {
699 | "version": "2.0.3",
700 | "resolved": "https://registry.npmjs.org/progress/-/progress-2.0.3.tgz",
701 | "integrity": "sha512-7PiHtLll5LdnKIMw100I+8xJXR5gW2QwWYkT6iJva0bXitZKa/XMrSbdmg3r2Xnaidz9Qumd0VPaMrZlF9V9sA==",
702 | "dev": true
703 | },
704 | "punycode": {
705 | "version": "2.1.1",
706 | "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.1.1.tgz",
707 | "integrity": "sha512-XRsRjdf+j5ml+y/6GKHPZbrF/8p2Yga0JPtdqTIY2Xe5ohJPD9saDJJLPvp9+NSBprVvevdXZybnj2cv8OEd0A==",
708 | "dev": true
709 | },
710 | "regexpp": {
711 | "version": "2.0.1",
712 | "resolved": "https://registry.npmjs.org/regexpp/-/regexpp-2.0.1.tgz",
713 | "integrity": "sha512-lv0M6+TkDVniA3aD1Eg0DVpfU/booSu7Eev3TDO/mZKHBfVjgCGTV4t4buppESEYDtkArYFOxTJWv6S5C+iaNw==",
714 | "dev": true
715 | },
716 | "resolve-from": {
717 | "version": "4.0.0",
718 | "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz",
719 | "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==",
720 | "dev": true
721 | },
722 | "restore-cursor": {
723 | "version": "2.0.0",
724 | "resolved": "https://registry.npmjs.org/restore-cursor/-/restore-cursor-2.0.0.tgz",
725 | "integrity": "sha1-n37ih/gv0ybU/RYpI9YhKe7g368=",
726 | "dev": true,
727 | "requires": {
728 | "onetime": "^2.0.0",
729 | "signal-exit": "^3.0.2"
730 | }
731 | },
732 | "rimraf": {
733 | "version": "2.6.3",
734 | "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.6.3.tgz",
735 | "integrity": "sha512-mwqeW5XsA2qAejG46gYdENaxXjx9onRNCfn7L0duuP4hCuTIi/QO7PDK07KJfp1d+izWPrzEJDcSqBa0OZQriA==",
736 | "dev": true,
737 | "requires": {
738 | "glob": "^7.1.3"
739 | }
740 | },
741 | "run-async": {
742 | "version": "2.3.0",
743 | "resolved": "https://registry.npmjs.org/run-async/-/run-async-2.3.0.tgz",
744 | "integrity": "sha1-A3GrSuC91yDUFm19/aZP96RFpsA=",
745 | "dev": true,
746 | "requires": {
747 | "is-promise": "^2.1.0"
748 | }
749 | },
750 | "rxjs": {
751 | "version": "6.5.1",
752 | "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-6.5.1.tgz",
753 | "integrity": "sha512-y0j31WJc83wPu31vS1VlAFW5JGrnGC+j+TtGAa1fRQphy48+fDYiDmX8tjGloToEsMkxnouOg/1IzXGKkJnZMg==",
754 | "dev": true,
755 | "requires": {
756 | "tslib": "^1.9.0"
757 | }
758 | },
759 | "safer-buffer": {
760 | "version": "2.1.2",
761 | "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz",
762 | "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==",
763 | "dev": true
764 | },
765 | "semver": {
766 | "version": "5.7.0",
767 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.0.tgz",
768 | "integrity": "sha512-Ya52jSX2u7QKghxeoFGpLwCtGlt7j0oY9DYb5apt9nPlJ42ID+ulTXESnt/qAQcoSERyZ5sl3LDIOw0nAn/5DA==",
769 | "dev": true
770 | },
771 | "shebang-command": {
772 | "version": "1.2.0",
773 | "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-1.2.0.tgz",
774 | "integrity": "sha1-RKrGW2lbAzmJaMOfNj/uXer98eo=",
775 | "dev": true,
776 | "requires": {
777 | "shebang-regex": "^1.0.0"
778 | }
779 | },
780 | "shebang-regex": {
781 | "version": "1.0.0",
782 | "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-1.0.0.tgz",
783 | "integrity": "sha1-2kL0l0DAtC2yypcoVxyxkMmO/qM=",
784 | "dev": true
785 | },
786 | "signal-exit": {
787 | "version": "3.0.2",
788 | "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.2.tgz",
789 | "integrity": "sha1-tf3AjxKH6hF4Yo5BXiUTK3NkbG0=",
790 | "dev": true
791 | },
792 | "slice-ansi": {
793 | "version": "2.1.0",
794 | "resolved": "https://registry.npmjs.org/slice-ansi/-/slice-ansi-2.1.0.tgz",
795 | "integrity": "sha512-Qu+VC3EwYLldKa1fCxuuvULvSJOKEgk9pi8dZeCVK7TqBfUNTH4sFkk4joj8afVSfAYgJoSOetjx9QWOJ5mYoQ==",
796 | "dev": true,
797 | "requires": {
798 | "ansi-styles": "^3.2.0",
799 | "astral-regex": "^1.0.0",
800 | "is-fullwidth-code-point": "^2.0.0"
801 | }
802 | },
803 | "sprintf-js": {
804 | "version": "1.0.3",
805 | "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz",
806 | "integrity": "sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw=",
807 | "dev": true
808 | },
809 | "string-width": {
810 | "version": "2.1.1",
811 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.1.1.tgz",
812 | "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==",
813 | "dev": true,
814 | "requires": {
815 | "is-fullwidth-code-point": "^2.0.0",
816 | "strip-ansi": "^4.0.0"
817 | }
818 | },
819 | "strip-ansi": {
820 | "version": "4.0.0",
821 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz",
822 | "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=",
823 | "dev": true,
824 | "requires": {
825 | "ansi-regex": "^3.0.0"
826 | }
827 | },
828 | "strip-json-comments": {
829 | "version": "2.0.1",
830 | "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz",
831 | "integrity": "sha1-PFMZQukIwml8DsNEhYwobHygpgo=",
832 | "dev": true
833 | },
834 | "supports-color": {
835 | "version": "5.5.0",
836 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz",
837 | "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==",
838 | "dev": true,
839 | "requires": {
840 | "has-flag": "^3.0.0"
841 | }
842 | },
843 | "table": {
844 | "version": "5.2.3",
845 | "resolved": "https://registry.npmjs.org/table/-/table-5.2.3.tgz",
846 | "integrity": "sha512-N2RsDAMvDLvYwFcwbPyF3VmVSSkuF+G1e+8inhBLtHpvwXGw4QRPEZhihQNeEN0i1up6/f6ObCJXNdlRG3YVyQ==",
847 | "dev": true,
848 | "requires": {
849 | "ajv": "^6.9.1",
850 | "lodash": "^4.17.11",
851 | "slice-ansi": "^2.1.0",
852 | "string-width": "^3.0.0"
853 | },
854 | "dependencies": {
855 | "ansi-regex": {
856 | "version": "4.1.0",
857 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz",
858 | "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==",
859 | "dev": true
860 | },
861 | "string-width": {
862 | "version": "3.1.0",
863 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz",
864 | "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==",
865 | "dev": true,
866 | "requires": {
867 | "emoji-regex": "^7.0.1",
868 | "is-fullwidth-code-point": "^2.0.0",
869 | "strip-ansi": "^5.1.0"
870 | }
871 | },
872 | "strip-ansi": {
873 | "version": "5.2.0",
874 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz",
875 | "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==",
876 | "dev": true,
877 | "requires": {
878 | "ansi-regex": "^4.1.0"
879 | }
880 | }
881 | }
882 | },
883 | "text-table": {
884 | "version": "0.2.0",
885 | "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz",
886 | "integrity": "sha1-f17oI66AUgfACvLfSoTsP8+lcLQ=",
887 | "dev": true
888 | },
889 | "through": {
890 | "version": "2.3.8",
891 | "resolved": "https://registry.npmjs.org/through/-/through-2.3.8.tgz",
892 | "integrity": "sha1-DdTJ/6q8NXlgsbckEV1+Doai4fU=",
893 | "dev": true
894 | },
895 | "tmp": {
896 | "version": "0.0.33",
897 | "resolved": "https://registry.npmjs.org/tmp/-/tmp-0.0.33.tgz",
898 | "integrity": "sha512-jRCJlojKnZ3addtTOjdIqoRuPEKBvNXcGYqzO6zWZX8KfKEpnGY5jfggJQ3EjKuu8D4bJRr0y+cYJFmYbImXGw==",
899 | "dev": true,
900 | "requires": {
901 | "os-tmpdir": "~1.0.2"
902 | }
903 | },
904 | "tslib": {
905 | "version": "1.9.3",
906 | "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.9.3.tgz",
907 | "integrity": "sha512-4krF8scpejhaOgqzBEcGM7yDIEfi0/8+8zDRZhNZZ2kjmHJ4hv3zCbQWxoJGz1iw5U0Jl0nma13xzHXcncMavQ==",
908 | "dev": true
909 | },
910 | "type-check": {
911 | "version": "0.3.2",
912 | "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.3.2.tgz",
913 | "integrity": "sha1-WITKtRLPHTVeP7eE8wgEsrUg23I=",
914 | "dev": true,
915 | "requires": {
916 | "prelude-ls": "~1.1.2"
917 | }
918 | },
919 | "uri-js": {
920 | "version": "4.2.2",
921 | "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.2.2.tgz",
922 | "integrity": "sha512-KY9Frmirql91X2Qgjry0Wd4Y+YTdrdZheS8TFwvkbLWf/G5KNJDCh6pKL5OZctEW4+0Baa5idK2ZQuELRwPznQ==",
923 | "dev": true,
924 | "requires": {
925 | "punycode": "^2.1.0"
926 | }
927 | },
928 | "which": {
929 | "version": "1.3.1",
930 | "resolved": "https://registry.npmjs.org/which/-/which-1.3.1.tgz",
931 | "integrity": "sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==",
932 | "dev": true,
933 | "requires": {
934 | "isexe": "^2.0.0"
935 | }
936 | },
937 | "wordwrap": {
938 | "version": "1.0.0",
939 | "resolved": "https://registry.npmjs.org/wordwrap/-/wordwrap-1.0.0.tgz",
940 | "integrity": "sha1-J1hIEIkUVqQXHI0CJkQa3pDLyus=",
941 | "dev": true
942 | },
943 | "wrappy": {
944 | "version": "1.0.2",
945 | "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz",
946 | "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=",
947 | "dev": true
948 | },
949 | "write": {
950 | "version": "1.0.3",
951 | "resolved": "https://registry.npmjs.org/write/-/write-1.0.3.tgz",
952 | "integrity": "sha512-/lg70HAjtkUgWPVZhZcm+T4hkL8Zbtp1nFNOn3lRrxnlv50SRBv7cR7RqR+GMsd3hUXy9hWBo4CHTbFTcOYwig==",
953 | "dev": true,
954 | "requires": {
955 | "mkdirp": "^0.5.1"
956 | }
957 | }
958 | }
959 | }
960 |
--------------------------------------------------------------------------------
/src/proxy/.package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "jsproxy-client",
3 | "version": "0.0.1",
4 | "description": "",
5 | "main": "boot.js",
6 | "directories": {
7 | "lib": "lib"
8 | },
9 | "scripts": {
10 | "test": "echo \"Error: no test specified\" && exit 1"
11 | },
12 | "keywords": [],
13 | "author": "EtherDream",
14 | "license": "MIT",
15 | "dependencies": {},
16 | "devDependencies": {
17 | "eslint": "^5.16.0"
18 | }
19 | }
20 |
--------------------------------------------------------------------------------
/src/proxy/README.md:
--------------------------------------------------------------------------------
1 | 浏览器脚本的源文件
2 |
3 | # src
4 |
5 | 源文件目录
6 |
7 |
8 | # debug.sh
9 |
10 | 开发模式。
11 |
12 | # release.sh
13 |
14 | 发布脚本。
15 |
--------------------------------------------------------------------------------
/src/proxy/debug.sh:
--------------------------------------------------------------------------------
1 | webpack \
2 | --o "../home/assets/bundle.debug.js" \
3 | --mode development -w
--------------------------------------------------------------------------------
/src/proxy/gen-tld/README.md:
--------------------------------------------------------------------------------
1 | 顶级域名后缀生成脚本
2 |
3 | # 生成
4 |
5 | ```bash
6 | node gen
7 | ```
8 |
9 | # TODO
10 |
11 | 该列表较大,导致最终 JS 增加上百 KB 的体积。
12 |
13 | 未考虑 `!` 和 `*.` 开头的域名。
14 |
15 | 未来考虑将结果保存为单独的文件,JS 运行时动态加载。
--------------------------------------------------------------------------------
/src/proxy/gen-tld/gen.js:
--------------------------------------------------------------------------------
1 | const https = require('https')
2 | const fs = require('fs')
3 | const {stdout} = process
4 |
5 | const SRC_PATH = 'https://publicsuffix.org/list/effective_tld_names.dat'
6 | const DST_PATH = '../src/tld-data.js'
7 |
8 |
9 | stdout.write('loading')
10 |
11 | https.get(SRC_PATH, res => {
12 | let str = ''
13 | res.on('data', s => {
14 | str += s
15 | stdout.write('.');
16 | }).on('end', _ => {
17 | proc(str)
18 | }).setEncoding('utf8')
19 | })
20 |
21 |
22 | function proc(str) {
23 | const list = str
24 | .split('\n')
25 | .filter(v => v)
26 | .filter(v => !v.startsWith('//'))
27 | .filter(v => !v.startsWith('!'))
28 | .map(v => v.replace('*.', ''))
29 | .sort((a, b) => a > b ? 1 : -1)
30 | .join(',')
31 |
32 | const result = `\
33 | // THIS FILE WAS GENERATED BY './tools/tld/gen.js'
34 | export default '${list}'
35 | `
36 | fs.writeFileSync(DST_PATH, result)
37 |
38 | console.log('\nok')
39 | }
40 |
--------------------------------------------------------------------------------
/src/proxy/jsconfig.json:
--------------------------------------------------------------------------------
1 | {
2 | "compilerOptions": {
3 | "checkJs": true,
4 | "lib": ["webworker", "dom", "es2017"],
5 | "target": "es2017"
6 | },
7 | "include": [
8 | "src/*"
9 | ]
10 | }
--------------------------------------------------------------------------------
/src/proxy/release.sh:
--------------------------------------------------------------------------------
1 | DST=../../www/assets
2 |
3 | rm $DST/bundle.*.js
4 |
5 | webpack \
6 | --o "$DST/bundle.[hash:8].js" \
7 | --mode production
8 |
9 | cd $DST
10 |
11 | for i in bundle.*.js; do
12 | printf "\
13 | jsproxy_config=\
14 | x=>{\
15 | __CONF__=x;\
16 | importScripts(__FILE__=x.assets_cdn+'$i')\
17 | };\
18 | importScripts('conf.js')\
19 | " > ../sw.js
20 | done
--------------------------------------------------------------------------------
/src/proxy/src/cdn.js:
--------------------------------------------------------------------------------
1 | import * as util from './util'
2 |
3 | // 暂时先用 jsdelivr 试验。之后换成速度很快、容量更大的免费图床
4 | const CDN = 'https://cdn.jsdelivr.net/npm/jsproxy-cache-01@0.0.'
5 |
6 | let mCurVer = -1
7 |
8 | /** @type {Map} */
9 | let mUrlHashVerMap = new Map()
10 |
11 | /** @type {Set} */
12 | let mDirectHostSet = new Set()
13 |
14 |
15 | async function loadDirectList(conf) {
16 | const url = conf.assets_cdn + conf.direct_host_list
17 | const res = await fetch(url)
18 | const txt = await res.text()
19 |
20 | for (const host of txt.split('\n')) {
21 | if (host && host[0] !== '#') {
22 | mDirectHostSet.add(host)
23 | }
24 | }
25 | }
26 |
27 | async function loadStaticList(conf) {
28 | const info = conf.static_boost
29 | if (!info || !info.enable) {
30 | return
31 | }
32 | const latest = info.ver
33 | if (mCurVer >= latest) {
34 | return
35 | }
36 | mCurVer = latest
37 | console.log('[jsproxy] cdn cache ver:', latest)
38 |
39 | const res = await fetch(CDN + latest + '/full')
40 | const buf = await res.arrayBuffer()
41 | const u32 = new Uint32Array(buf)
42 |
43 | let p = 0
44 | for (let ver = 0; ver <= latest; ver++) {
45 | const num = u32[p++]
46 |
47 | for (let i = 0; i < num; i++) {
48 | const urlHash = u32[p++]
49 | mUrlHashVerMap.set(urlHash, ver)
50 | }
51 | }
52 | }
53 |
54 |
55 | export function setConf(conf) {
56 | return Promise.all([
57 | loadStaticList(conf),
58 | loadDirectList(conf),
59 | ])
60 | }
61 |
62 | /**
63 | * @param {string} host
64 | */
65 | export function isDirectHost(host) {
66 | return mDirectHostSet.has(host)
67 | }
68 |
69 |
70 | /**
71 | * @param {string} url
72 | */
73 | export async function proxyDirect(url) {
74 | try {
75 | const res = await fetch(url, {
76 | referrerPolicy: 'no-referrer',
77 | })
78 | const {status} = res
79 | if (status === 200 || status === 206) {
80 | return res
81 | }
82 | console.warn('direct status:', status, url)
83 | } catch (err) {
84 | console.warn('direct fail:', url)
85 | }
86 | }
87 |
88 |
89 | /**
90 | * @param {number} urlHash
91 | */
92 | export function getFileVer(urlHash) {
93 | return mUrlHashVerMap.get(urlHash)
94 | }
95 |
96 |
97 | /**
98 | * @param {number} urlHash
99 | * @param {number} urlVer
100 | */
101 | async function proxyStaticMain(urlHash, urlVer) {
102 | const hashHex = util.numToHex(urlHash, 8)
103 | const res = await fetch(CDN + urlVer + '/' + hashHex + '.txt')
104 | if (res.status !== 200) {
105 | throw 'bad status: ' + res.status
106 | }
107 | const buf = await res.arrayBuffer()
108 | const b = new Uint8Array(buf)
109 |
110 | const hdrLen = b[0] << 8 | b[1]
111 | const hdrBuf = b.subarray(2, 2 + hdrLen)
112 | const hdrStr = util.bytesToStr(hdrBuf)
113 | const hdrObj = JSON.parse(hdrStr)
114 |
115 | const body = b.subarray(2 + hdrLen)
116 | hdrObj['date'] = new Date().toUTCString()
117 |
118 | return new Response(body, {
119 | headers: hdrObj
120 | })
121 | }
122 |
123 |
124 | /**
125 | * @param {number} urlHash
126 | * @param {number} urlVer
127 | */
128 | export async function proxyStatic(urlHash, urlVer) {
129 | // TODO: 使用多个 CDN
130 | try {
131 | return await proxyStaticMain(urlHash, urlVer)
132 | } catch(err) {
133 | console.warn('cdn fail:', err)
134 | }
135 | }
136 |
--------------------------------------------------------------------------------
/src/proxy/src/client.js:
--------------------------------------------------------------------------------
1 | import * as urlx from './urlx.js'
2 | import * as route from './route.js'
3 | import * as env from './env.js'
4 | import * as hook from './hook.js'
5 | import {createFakeLoc} from './fakeloc.js'
6 | import {createStorage} from './storage.js'
7 |
8 |
9 | const {
10 | apply,
11 | construct,
12 | } = Reflect
13 |
14 |
15 | /**
16 | * Hook 页面和 Worker 相同的 API
17 | *
18 | * @param {Window} global WindowOrWorkerGlobalScope
19 | * @param {string} origin
20 | */
21 | export function init(global, origin) {
22 | // lockNative(win)
23 |
24 | // hook Storage API
25 | createStorage(global, origin)
26 |
27 | // hook Location API
28 | const fakeLoc = createFakeLoc(global)
29 |
30 | // hook Performance API
31 | const perfProto = global['PerformanceEntry'].prototype
32 | hook.prop(perfProto, 'name',
33 | getter => function() {
34 | const val = getter.call(this)
35 | if (/^https?:/.test(val)) {
36 | return urlx.decUrlStrAbs(val)
37 | }
38 | return val
39 | }
40 | )
41 |
42 |
43 | // hook AJAX API
44 | const xhrProto = global['XMLHttpRequest'].prototype
45 | hook.func(xhrProto, 'open', oldFn => function(_0, url) {
46 | if (url) {
47 | arguments[1] = urlx.encUrlStrRel(url, this)
48 | }
49 | return apply(oldFn, this, arguments)
50 | })
51 |
52 | hook.prop(xhrProto, 'responseURL',
53 | getter => function(oldFn) {
54 | const val = getter.call(this)
55 | return urlx.decUrlStrRel(val, this)
56 | }
57 | )
58 |
59 |
60 | hook.func(global, 'fetch', oldFn => function(v) {
61 | if (v) {
62 | if (v.url) {
63 | // v is Request
64 | const newUrl = urlx.encUrlStrAbs(v.url)
65 | arguments[0] = new Request(newUrl, v)
66 | } else {
67 | // v is string
68 | // TODO: 字符串不传引用,无法获取创建时的 constructor
69 | arguments[0] = urlx.encUrlStrRel(v, v)
70 | }
71 | }
72 | return apply(oldFn, this, arguments)
73 | })
74 |
75 |
76 | hook.func(global, 'WebSocket', oldFn => function(url) {
77 | const urlObj = urlx.newUrl(url)
78 | if (urlObj) {
79 | const {ori} = env.get(this)
80 | if (ori) {
81 | const args = {
82 | 'origin': ori.origin,
83 | }
84 | arguments[0] = route.genWsUrl(urlObj, args)
85 | }
86 | }
87 | return construct(oldFn, arguments)
88 | })
89 |
90 | /**
91 | * @param {string} type
92 | */
93 | function hookWorker(type) {
94 | hook.func(global, type, oldFn => function(url) {
95 | if (url) {
96 | console.log('[jsproxy] new %s: %s', type, url)
97 | arguments[0] = urlx.encUrlStrRel(url, this)
98 | }
99 | return construct(oldFn, arguments)
100 | })
101 | }
102 |
103 | hookWorker('Worker')
104 | hookWorker('SharedWorker')
105 |
106 |
107 | hook.func(global, 'importScripts', oldFn => function(...args) {
108 | const urls = args.map(urlx.encUrlStrRel)
109 | console.log('[jsproxy] importScripts:', urls)
110 | return apply(oldFn, this, urls)
111 | })
112 | }
--------------------------------------------------------------------------------
/src/proxy/src/cookie.js:
--------------------------------------------------------------------------------
1 | import {Database} from './database.js'
2 |
3 | /** @type {Set} */
4 | let mDirtySet = new Set()
5 |
6 |
7 | function Cookie() {
8 | this.id = ''
9 | this.name = ''
10 | this.value = ''
11 | this.domain = ''
12 | this.hostOnly = false
13 | this.path = ''
14 | this.expires = NaN
15 | this.isExpired = false
16 | this.secure = false
17 | this.httpOnly = false
18 | this.sameSite = ''
19 | }
20 |
21 | /**
22 | * @param {Cookie} src
23 | * @param {Cookie} dst
24 | */
25 | function copy(dst, src) {
26 | dst.id = src.id
27 | dst.name = src.name
28 | dst.value = src.value
29 | dst.domain = src.domain
30 | dst.hostOnly = src.hostOnly
31 | dst.path = src.path
32 | dst.expires = src.expires
33 | dst.isExpired = src.isExpired
34 | dst.secure = src.secure
35 | dst.httpOnly = src.httpOnly
36 | dst.sameSite = src.sameSite
37 | }
38 |
39 |
40 | /**
41 | * @param {string} cookiePath
42 | * @param {string} urlPath
43 | */
44 | function isSubPath(cookiePath, urlPath) {
45 | if (urlPath === cookiePath) {
46 | return true
47 | }
48 | if (!cookiePath.endsWith('/')) {
49 | cookiePath += '/'
50 | }
51 | return urlPath.startsWith(cookiePath)
52 | }
53 |
54 |
55 | /**
56 | * @param {string} cookieDomain
57 | * @param {string} urlDomain
58 | */
59 | function isSubDomain(cookieDomain, urlDomain) {
60 | return urlDomain === cookieDomain ||
61 | urlDomain.endsWith('.' + cookieDomain)
62 | }
63 |
64 |
65 | /**
66 | * @param {Cookie} item
67 | * @param {number} now
68 | */
69 | function isExpire(item, now) {
70 | const v = item.expires
71 | return !isNaN(v) && v < now
72 | }
73 |
74 |
75 | class CookieDomainNode {
76 | constructor() {
77 | /** @type {Cookie[]} */
78 | this.items = null
79 |
80 | /** @type {Object} */
81 | this.children = {}
82 | }
83 |
84 | /**
85 | * @param {string} name
86 | */
87 | nextChild(name) {
88 | return this.children[name] || (
89 | this.children[name] = new CookieDomainNode
90 | )
91 | }
92 |
93 | /**
94 | * @param {string} name
95 | */
96 | getChild(name) {
97 | return this.children[name]
98 | }
99 |
100 | /**
101 | * @param {Cookie} cookie
102 | */
103 | addCookie(cookie) {
104 | if (this.items) {
105 | this.items.push(cookie)
106 | } else {
107 | this.items = [cookie]
108 | }
109 | }
110 | }
111 |
112 | /** @type {Map} */
113 | const mIdCookieMap = new Map()
114 |
115 | const mCookieNodeRoot = new CookieDomainNode()
116 |
117 |
118 |
119 | export function getNonHttpOnlyItems() {
120 | const ret = []
121 | for (const item of mIdCookieMap.values()) {
122 | if (!item.httpOnly) {
123 | ret.push(item)
124 | }
125 | }
126 | return ret
127 | }
128 |
129 |
130 | /**
131 | * @param {string} str
132 | * @param {URL} urlObj
133 | * @param {number} now
134 | */
135 | export function parse(str, urlObj, now) {
136 | const item = new Cookie()
137 | const arr = str.split(';')
138 |
139 | for (let i = 0; i < arr.length; i++) {
140 | let key, val
141 | const s = arr[i].trim()
142 | const p = s.indexOf('=')
143 |
144 | if (p !== -1) {
145 | key = s.substr(0, p)
146 | val = s.substr(p + 1)
147 | } else {
148 | //
149 | // cookie = 's; secure; httponly'
150 | // 0: { key: '', val: 's' }
151 | // 1: { key: 'secure', val: '' }
152 | // 2: { key: 'httponly', val: '' }
153 | //
154 | key = (i === 0) ? '' : s
155 | val = (i === 0) ? s : ''
156 | }
157 |
158 | if (i === 0) {
159 | item.name = key
160 | item.value = val
161 | continue
162 | }
163 |
164 | switch (key.toLocaleLowerCase()) {
165 | case 'expires':
166 | if (isNaN(item.expires)) {
167 | item.expires = Date.parse(val)
168 | }
169 | break
170 | case 'domain':
171 | if (val[0] === '.') {
172 | val = val.substr(1)
173 | }
174 | item.domain = val
175 | break
176 | case 'path':
177 | item.path = val
178 | break
179 | case 'httponly':
180 | item.httpOnly = true
181 | break
182 | case 'secure':
183 | item.secure = true
184 | break
185 | case 'max-age':
186 | item.expires = now + (+val) * 1000
187 | break
188 | case 'samesite':
189 | item.sameSite = val
190 | break
191 | }
192 | }
193 |
194 | if (isExpire(item, now)) {
195 | item.isExpired = true
196 | }
197 |
198 | // https://developer.mozilla.org/zh-CN/docs/Web/HTTP/Headers/Set-Cookie
199 | if (item.name.startsWith('__Secure-')) {
200 | if (!(
201 | urlObj.protocol === 'https:' &&
202 | item.secure
203 | )) {
204 | return
205 | }
206 | }
207 | if (item.name.startsWith('__Host-')) {
208 | if (!(
209 | urlObj.protocol === 'https:' &&
210 | item.secure &&
211 | item.domain === '' &&
212 | item.path === '/'
213 | )) {
214 | return
215 | }
216 | }
217 |
218 | // https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Set-Cookie#Compatibility_notes
219 | if (item.secure && urlObj.protocol === 'http:') {
220 | return
221 | }
222 |
223 | // check hostname
224 | const domain = urlObj.hostname
225 |
226 | if (item.domain) {
227 | if (!isSubDomain(item.domain, domain)) {
228 | console.warn('[jsproxy] invalid cookie domain! `%s` ⊄ `%s`',
229 | item.domain, domain)
230 | return
231 | }
232 | } else {
233 | item.domain = domain
234 | item.hostOnly = true
235 | }
236 |
237 | // check pathname
238 | const path = urlObj.pathname
239 |
240 | if (item.path) {
241 | if (!isSubPath(item.path, path)) {
242 | console.warn('[jsproxy] invalid cookie path! `%s` ⊄ `%s`',
243 | item.path, path)
244 | return
245 | }
246 | } else {
247 | item.path = path
248 | }
249 |
250 | item.id = (item.secure ? ';' : '') +
251 | item.name + ';' +
252 | item.domain +
253 | item.path
254 |
255 | return item
256 | }
257 |
258 |
259 | /**
260 | * @param {Cookie} item
261 | */
262 | export function set(item) {
263 | // console.log('set:', item)
264 | const id = item.id
265 | const matched = mIdCookieMap.get(id)
266 |
267 | if (matched) {
268 | if (item.isExpired) {
269 | // delete
270 | mIdCookieMap.delete(id)
271 | matched.isExpired = true
272 | // TODO: remove node
273 | } else {
274 | // update
275 | copy(matched, item)
276 | }
277 | mDirtySet.add(matched)
278 | } else {
279 | // create
280 | const labels = item.domain.split('.')
281 | let labelPos = labels.length
282 | let node = mCookieNodeRoot
283 | do {
284 | node = node.nextChild(labels[--labelPos])
285 | } while (labelPos !== 0)
286 |
287 | node.addCookie(item)
288 | mIdCookieMap.set(id, item)
289 |
290 | mDirtySet.add(item)
291 | }
292 | }
293 |
294 |
295 | /**
296 | * @param {URL} urlObj
297 | */
298 | export function query(urlObj) {
299 | const ret = []
300 | const now = Date.now()
301 | const domain = urlObj.hostname
302 | const path = urlObj.pathname
303 | const isHttps = (urlObj.protocol === 'https:')
304 |
305 | const labels = domain.split('.')
306 | let labelPos = labels.length
307 | let node = mCookieNodeRoot
308 |
309 | do {
310 | node = node.getChild(labels[--labelPos])
311 | if (!node) {
312 | break
313 | }
314 | const items = node.items
315 | if (!items) {
316 | continue
317 | }
318 | for (let i = 0; i < items.length; i++) {
319 | const item = items[i]
320 | // https url | secure flag | carry
321 | // ✔ | ✔ | ✔
322 | // ✔ | ✘ | ✔
323 | // ✘ | ✘ | ✔
324 | // ✘ | ✔ | ✘
325 | if (!isHttps && item.secure) {
326 | continue
327 | }
328 | // HostOnly Cookie 需匹配完整域名
329 | if (item.hostOnly && labelPos !== 0) {
330 | continue
331 | }
332 | if (!isSubPath(item.path, path)) {
333 | continue
334 | }
335 | if (item.isExpired) {
336 | continue
337 | }
338 | if (isExpire(item, now)) {
339 | item.isExpired = true
340 | continue
341 | }
342 | // TODO: same site
343 |
344 | let str = item.value
345 | if (item.name) {
346 | str = item.name + '=' + str
347 | }
348 | ret.push(str)
349 | }
350 | } while (labelPos !== 0)
351 |
352 | return ret.join('; ')
353 | }
354 |
355 |
356 | /** @type {Database} */
357 | let mDB
358 |
359 | export async function setDB(db) {
360 | mDB = db
361 |
362 | const now = Date.now()
363 | await mDB.enum('cookie', v => {
364 | if (isExpire(v, now)) {
365 | mDB.delete('cookie', v.id)
366 | } else {
367 | set(v)
368 | }
369 | return true
370 | })
371 |
372 | setInterval(save, 1000 * 3)
373 | }
374 |
375 |
376 | export async function save() {
377 | if (mDirtySet.size === 0) {
378 | return
379 | }
380 |
381 | const tmp = mDirtySet
382 | mDirtySet = new Set()
383 |
384 | for (const item of tmp) {
385 | if (item.isExpired) {
386 | await mDB.delete('cookie', item.id)
387 | } else if (!isNaN(item.expires)) {
388 | // 不保存 session cookie
389 | await mDB.put('cookie', item)
390 | }
391 | }
392 | }
393 |
--------------------------------------------------------------------------------
/src/proxy/src/database.js:
--------------------------------------------------------------------------------
1 | import {Signal} from './signal.js'
2 |
3 |
4 | export class Database {
5 | /**
6 | * @param {string} name
7 | */
8 | constructor(name) {
9 | this._name = name
10 |
11 | /** @type {IDBDatabase} */
12 | this._db = null
13 | }
14 |
15 | /**
16 | * @param {string} table
17 | * @param {IDBTransactionMode} mode
18 | */
19 | _getStore(table, mode) {
20 | return this._db
21 | .transaction(table, mode)
22 | .objectStore(table)
23 | }
24 |
25 | /**
26 | * @param {Object} opts
27 | */
28 | open(opts) {
29 | const s = new Signal()
30 | const req = indexedDB.open(this._name)
31 |
32 | req.onsuccess = (e) => {
33 | const idb = req.result
34 | this._db = idb
35 |
36 | idb.onclose = (e) => {
37 | console.warn('[jsproxy] indexedDB disconnected, reopen...')
38 | this.open(opts)
39 | }
40 | s.notify()
41 | }
42 | req.onerror = (e) => {
43 | console.warn('req.onerror:', e)
44 | s.abort(req.error)
45 | }
46 | req.onupgradeneeded = (e) => {
47 | const idb = req.result
48 | for (const [k, v] of Object.entries(opts)) {
49 | idb.createObjectStore(k, v)
50 | }
51 | }
52 | return s.wait()
53 | }
54 |
55 |
56 | close() {
57 | this._db.close()
58 | }
59 |
60 | /**
61 | * @param {string} table
62 | * @param {any} key
63 | */
64 | get(table, key) {
65 | const s = new Signal()
66 | const obj = this._getStore(table, 'readonly')
67 | const req = obj.get(key)
68 |
69 | req.onsuccess = (e) => {
70 | s.notify(req.result)
71 | }
72 | req.onerror = (e) => {
73 | s.abort(req.error)
74 | }
75 | return s.wait()
76 | }
77 |
78 | /**
79 | * @param {string} table
80 | * @param {any} record
81 | */
82 | put(table, record) {
83 | const s = new Signal()
84 | const obj = this._getStore(table, 'readwrite')
85 | const req = obj.put(record)
86 |
87 | req.onsuccess = (e) => {
88 | s.notify()
89 | }
90 | req.onerror = (e) => {
91 | s.abort(req.error)
92 | }
93 | return s.wait()
94 | }
95 |
96 | /**
97 | * @param {string} table
98 | * @param {any} key
99 | */
100 | delete(table, key) {
101 | const s = new Signal()
102 | const obj = this._getStore(table, 'readwrite')
103 | const req = obj.delete(key)
104 |
105 | req.onsuccess = (e) => {
106 | s.notify()
107 | }
108 | req.onerror = (e) => {
109 | s.abort(req.error)
110 | }
111 | return s.wait()
112 | }
113 |
114 | /**
115 | * @param {string} table
116 | * @param {(any) => boolean} callback
117 | */
118 | enum(table, callback, ...args) {
119 | const s = new Signal()
120 | const obj = this._getStore(table, 'readonly')
121 | const req = obj.openCursor(...args)
122 |
123 | req.onsuccess = (e) => {
124 | const {result} = req
125 | if (result) {
126 | if (callback(result.value) !== false) {
127 | result.continue()
128 | }
129 | } else {
130 | s.notify()
131 | }
132 | }
133 | req.onerror = (e) => {
134 | s.abort(req.error)
135 | }
136 | return s.wait()
137 | }
138 | }
139 |
--------------------------------------------------------------------------------
/src/proxy/src/env.js:
--------------------------------------------------------------------------------
1 | export const ENV_PAGE = 1
2 | export const ENV_WORKER = 2
3 | export const ENV_SW = 3
4 |
5 | let mEnvType = 0
6 |
7 | export function setEnvType(v) {
8 | mEnvType = v
9 | }
10 |
11 | export function isSwEnv() {
12 | return mEnvType === ENV_SW
13 | }
14 |
15 | export function isWorkerEnv() {
16 | return mEnvType === ENV_WORKER
17 | }
18 |
19 |
20 | /**
21 | * @type {WeakMap