• 应用打包
    • 生成asar
      • 1. 安装asar
      • 2. 用asar pack打包
    • 使用asar
      • Node API
      • Web API
      • 像“文件”那样处理asar
    • Node API缺陷
      • asar包是只读的
      • Working Directory在asar包中无效
      • API中的额外“开箱”
      • fs.stat获取的stat信息不可靠
      • 执行asar包中的程序
    • 打包时排除文件

    应用打包

    为舒缓Windows下路径名过长的问题issues, 也略对require加速以及简单隐匿你的源代码, 你可以通过极小的源代码改动将你的应用打包成asar.

    生成asar

    asar是一种将多个文件合并成一个文件的类tar风格的归档格式。 Electron可以无需解压,即从其中读取任意文件内容。

    参照如下步骤将你的应用打包成asar

    1. 安装asar

    1. $ npm install -g asar

    2. 用asar pack打包

    1. $ asar pack your-app app.asar

    使用asar

    在Electron中有两类APIs:Node.js提供的Node APIs和Chromium提供的Web APIs。这两种APIs都支持从asar包中读取文件。

    Node API

    由于Electron中打了特别补丁, Node APIs中如fs.readFile或者require之类的方法可以将asar视之为虚拟文件夹,读取asar里面的文件就和从真实的文件系统中读取一样。

    例如,假设我们在/path/to文件夹下有个example.asar包:

    1. $ asar list /path/to/example.asar
    2. /app.js
    3. /file.txt
    4. /dir/module.js
    5. /static/index.html
    6. /static/main.css
    7. /static/jquery.min.js

    asar包读取一个文件:

    1. const fs = require('fs');
    2. fs.readFileSync('/path/to/example.asar/file.txt');

    列出asar包中根目录下的所有文件:

    1. const fs = require('fs');
    2. fs.readdirSync('/path/to/example.asar');

    使用asar包中的一个模块:

    1. require('/path/to/example.asar/dir/module.js');

    你也可以使用BrowserWindow来显示一个asar包里的web页面:

    1. const BrowserWindow = require('electron').BrowserWindow;
    2. var win = new BrowserWindow({width: 800, height: 600});
    3. win.loadURL('file:///path/to/example.asar/static/index.html');

    Web API

    在Web页面里,用file:协议可以获取asar包中文件。和Node API一样,视asar包如虚拟文件夹。

    例如,用$.get获取文件:

    1. <script>
    2. var $ = require('./jquery.min.js');
    3. $.get('file:///path/to/example.asar/file.txt', function(data) {
    4. console.log(data);
    5. });
    6. </script>

    像“文件”那样处理asar

    有些场景,如:核查asar包的校验和,我们需要像读取“文件”那样读取asar包的内容(而不是当成虚拟文件夹)。你可以使用内置的original-fs(提供和fs一样的APIs)模块来读取asar包的真实信息。

    1. var originalFs = require('original-fs');
    2. originalFs.readFileSync('/path/to/example.asar');

    Node API缺陷

    尽管我们已经尽了最大努力使得asar包在Node API下的应用尽可能的趋向于真实的目录结构,但仍有一些底层Node API我们无法保证其正常工作。

    asar包是只读的

    asar包中的内容不可更改,所以Node APIs里那些可以用来修改文件的方法在对待asar包时都无法正常工作。

    Working Directory在asar包中无效

    尽管asar包是虚拟文件夹,但其实并没有真实的目录架构对应在文件系统里,所以你不可能将working Directory设置成asar包里的一个文件夹。将asar中的文件夹以cwd形式作为参数传入一些API中也会报错。

    API中的额外“开箱”

    大部分fsAPI可以无需解压即从asar包中读取文件或者文件的信息,但是在处理一些依赖真实文件路径的底层系统方法时,Electron会将所需文件解压到临时目录下,然后将临时目录下的真实文件路径传给底层系统方法使其正常工作。 对于这类API,耗费会略多一些。

    以下是一些需要额外解压的APIs:

    • child_process.execFile
    • child_process.execFileSync
    • fs.open
    • fs.openSync
    • process.dlopen - requirenative模块时用到

    fs.stat获取的stat信息不可靠

    asar包中的文件取fs.stat,返回的Stats对象不是精确值,因为这些文件不是真实存在于文件系统里。所以除了文件大小和文件类型以外,你不应该依赖Stats对象的值。

    执行asar包中的程序

    Node中有一些可以执行程序的API,如child_process.execchild_process.spawnchild_process.execFile等,但只有execFile可以执行asar包中的程序。

    因为execspawn允许command替代file作为输入,而command是需要在shell下执行的,目前没有可靠的方法来判断command中是否在操作一个asar包中的文件,而且即便可以判断,我们依旧无法保证可以在无任何副作用的情况下替换command中的文件路径。

    打包时排除文件

    如上所述,一些Node API会在调用时将文件解压到文件系统中,除了效率问题外,也有可能引起杀毒软件的注意!

    为解决这个问题,你可以在生成asar包时使用--unpack选项来排除一些文件,使其不打包到asar包中,下面是如何排除一些用作共享用途的native模块的方法:

    1. $ asar pack app app.asar --unpack *.node

    经过上述命令后,除了生成的app.asar包以外,还有一个包含了排除文件的app.asar.unpacked文件夹,你需要将这个文件夹一起拷贝,提供给用户。