为解决Windows中路径过长的问题issues, 并且更快的 require
以及简单的隐匿源码, 你可以将代码打包成 asar.
asar类似于 tar
压缩包,它将全部文件打包至一个文件里,Electron可从中读取且无需解压整个文件.
打包成 asar
的两个步骤:
$ npm install -g asar
$ asar pack your-app app.asar
在Electron中有两组API: 由Chromium提供的Node.js和Web API提供的Node API, 这两种API都支持从 asar
中读取.
由于 Electron 中打了特别补丁, Node API 中如 fs.readFile
或者 require
之类的方法可以将 asar
当作虚拟文件夹,读取 asar
里面的文件就和从真实的文件系统中读取一样。
假设我们在 /path/to
文件夹下有个 example.asar
包:
$ asar list /path/to/example.asar
/app.js
/file.txt
/dir/module.js
/static/index.html
/static/main.css
/static/jquery.min.js
从 asar
包读取一个文件:
const fs = require('fs')
fs.readFileSync('/path/to/example.asar/file.txt')
列出 asar
包中根目录下的所有文件:
const fs = require('fs')
fs.readdirSync('/path/to/example.asar')
使用 asar
包中的一个模块:
require('/path/to/example.asar/dir/module.js')
您还可以使用 BrowserWindow
来显示一个 asar
包里的 web 页面:
const {BrowserWindow} = require('electron')
let win = new BrowserWindow({width: 800, height: 600})
win.loadURL('file:///path/to/example.asar/static/index.html')
在 Web 页面里,用 file:
协议可以获取 asar
包中文件。和 Node API 一样,视 asar
包如虚拟文件夹。
例如,用 $.get
获取文件:
<script>
let $ = require('./jquery.min.js')
$.get('file:///path/to/example.asar/file.txt', (data) => {
console.log(data)
})
</script>
有些场景,如:核查 asar
包的校验值,我们需要像读取普通文件那样读取 asar
包的内容(而不是当成虚拟文件夹)。
你可以使用内置的 original-fs
(提供和 fs
一样的 API)模块来读取 asar
包的真实信息:
const originalFs = require('original-fs')
originalFs.readFileSync('/path/to/example.asar')
您也可以将 process.noAsar
设置为 true
,以禁用 fs
模块中 asar
的支持:
const fs = require('fs')
process.noAsar = true
fs.readFileSync('/path/to/example.asar')
尽管我们已经尽了最大努力使得 asar
包在 Node API 下的应用尽可能的趋向于真实的目录结构,但仍有一些底层 Node API 我们无法保证其正常工作。
asar
包中的内容不可更改,所以 Node APIs 里那些可以用来修改文件的方法在对待 asar
包时都无法正常工作。
尽管 asar
包是虚拟文件夹,但其实并没有真实的目录架构对应在文件系统里,所以你不可能将 working Directory 设置成 asar
包里的一个文件夹。将 asar
中的文件夹以 cwd
形式作为参数传入一些 API 中也会报错。
大部分 fs
API 可以无需解压即从 asar
包中读取文件或者文件的信息,但是在处理一些依赖真实文件路径的底层系统方法时,Electron 会将所需文件解压到临时目录下,然后将临时目录下的真实文件路径传给底层系统方法使其正常工作。 对于这类API,耗费会略多一些。
以下是一些需要额外解压的 API:
child_process.execFile
child_process.execFileSync
fs.open
fs.openSync
process.dlopen
-require
native模块时用到
对 asar
包中的文件取 fs.stat
,返回的 Stats
对象不是精确值,因为这些文件不是真实存在于文件系统里。所以除了文件大小和文件类型以外,你不应该依赖 Stats
对象的值。
Node 中有一些可以执行程序的 API,如 child_process.exec
,child_process.spawn
和 child_process.execFile
等,但只有 execFile
可以执行 asar
包中的程序。
因为 exec
和 spawn
允许 command
替代 file
作为输入,而 command
是需要在 shell 下执行的,目前没有可靠的方法来判断 command
中是否在操作一个 asar
包中的文件,而且即便可以判断,我们依旧无法保证可以在无任何副作用的情况下替换 command
中的文件路径。
如上所述,一些 Node API 会在调用时将文件解压到文件系统中,除了效率问题外,也有可能引起杀毒软件的注意!
为解决这个问题,你可以在生成 asar
包时使用 --unpack
选项来排除一些文件,使其不打包到 asar
包中,下面是如何排除一些用作共享用途的 native 模块的方法:
$ asar pack app app.asar --unpack *.node
经过上述命令后,除了生成的 app.asar
包以外,还有一个包含了排除文件的 app.asar.unpacked
文件夹,你需要将这个文件夹一起拷贝,提供给用户。