Cordova config.xml环境变量

Luc*_*tto 14 cordova ionic3

场景:

我正在构建一个Ionic3应用程序,我config.xml有一些数据,我希望能够根据我的环境进行更改(例如,我希望我的facebook app id具有不同的开发,分期和生产价值).

我实现了这个创建模板config.xml(文件是config.tpl.xml)和一个before_prepare cordova钩子,用正确的值替换模板中的变量并保存生成的内容config.xml.

所述cordova钩使用npmes6-template-strings:

npm install es6-template-strings --save-dev
Run Code Online (Sandbox Code Playgroud)

钩子是:

#!/usr/bin/env node
var fs = require('fs');
var path = require('path');
var compile = require('es6-template-strings/compile');
var resolveToString = require('es6-template-strings/resolve-to-string');

var ROOT_DIR = process.argv[2];
var FILES = {
    SRC: "config.tpl.xml",
    DEST: "config.xml"
};

var env = process.env.NODE_ENV || 'dev';
var envFile = 'src/environments/environment.' + env + '.json';

var srcFileFull = path.join(ROOT_DIR, FILES.SRC);
var destFileFull = path.join(ROOT_DIR, FILES.DEST);
var configFileFull = path.join(ROOT_DIR, envFile);

var templateData = fs.readFileSync(srcFileFull, 'utf8');

var configData = fs.readFileSync(configFileFull, 'utf8');
var config = JSON.parse(configData);

var compiled = compile(templateData);
var content = resolveToString(compiled, config);

fs.writeFileSync(destFileFull, content);
Run Code Online (Sandbox Code Playgroud)

我在src/environments/目录中有不同环境的文件,这些文件是根据NODE_ENV我构建时定义的值选择的cordova.例如,如果NODE_ENV=prod(生产),那么它将使用该文件environment.prod.json:

{
    ...
    "FACEBOOK_APP_ID": "11111111",
    "FACEBOOK_APP_NAME": "My Facebook App Name",
    ...
    "PUSH_SENDER_ID": "22222222",
    ...
}
Run Code Online (Sandbox Code Playgroud)

执行钩子时,这部分在cordova.tpl.xml:

<plugin name="cordova-plugin-facebook4" spec="~1.7.4">
    <variable name="APP_ID" value="${FACEBOOK_APP_ID}" />
    <variable name="APP_NAME" value="${FACEBOOK_APP_NAME}" />
</plugin>
<plugin name="phonegap-plugin-push" spec="~1.9.2">
    <variable name="SENDER_ID" value="${PUSH_SENDER_ID}" />
</plugin>
Run Code Online (Sandbox Code Playgroud)

变得像:

<plugin name="cordova-plugin-facebook4" spec="~1.7.4">
    <variable name="APP_ID" value="11111111" />
    <variable name="APP_NAME" value="My Facebook App Name" />
</plugin>
<plugin name="phonegap-plugin-push" spec="~1.9.2">
    <variable name="SENDER_ID" value="22222222" />
</plugin>
Run Code Online (Sandbox Code Playgroud)

问题:

到现在为止还挺好.问题是,当进行一些自动更改config.xml(如添加插件)时,它没有反映出来cordova.tpl.xml,所以我必须记住手动进行更改.

虽然现在完成的方式仍然比必须像以前一样添加每个环境变量要好得多(这是一个维护噩梦并且容易出错),但我仍然需要每次添加/更改/删除插件时更改模板(不那么频繁,当我忘记它时很容易发现问题,但仍然远离理想).

我的问题:

我想知道是否有办法避免这些手动更改,但继续使用环境变量,就像我现在正在做的那样.

它可以实现让到自动改变config.xml工作要做,以config.tpl.xml代替(如添加当一个插件--save),如果可能的话,但我没有发现任何cordova有关此选项.

或者使用config.xml模板作为模板并将其发送到定义了变量的另一个地方(例如,到www/config.xml),并使用config.xml其他位置来构建应用程序(不是config.xml在根目录中,即模板).我只会更改我的钩子中的srcdest文件(分别为config.xmlwww/config.xml).但我也没有找到实现这个目标的方法.

有什么想法吗?

(它不需要是离子特定的解决方案.)

更新(2017-10-13)

根据Bobby的回答,我在安装和卸载插件时实现了我想要的.我创建了4个钩:after_plugin_add,after_plugin_rm,before_plugin_add,before_plugin_rm.

before挂钩复制的模板(config.tpl.xml)到config.xml文件和after钩子做反.

before_plugin_addbefore_plugin_rm钩如下:

#!/usr/bin/env node
var fs = require('fs');
var path = require('path');

var ROOT_DIR = process.argv[2];
var FILES = {
    SRC: 'config.tpl.xml',
    DEST: 'config.xml'
};

var srcFileFull = path.join(ROOT_DIR, FILES.SRC);
var destFileFull = path.join(ROOT_DIR, FILES.DEST);

var templateData = fs.readFileSync(srcFileFull, 'utf8');
fs.writeFileSync(destFileFull, templateData);
Run Code Online (Sandbox Code Playgroud)

after_plugin_addafter_plugin_rm钩几乎是相同的,只是交换FILES.SRCFILES.DEST值.

小智 5

一种解决方案是为before_plugin_installafter_plugin_install创建钩子。

On before_plugin_install copy the cordova.tpl.xml to cordova.xml.
... Plugin is installed ...
On after_plugin_install copy cordova.xml to cordova.tpl.xml
Run Code Online (Sandbox Code Playgroud)