InstallAnyWhere学习笔记 ( by quqi99 )
作者:张华 发表于:2009-12-23
版权声明:可以任意转载,转载时请务必以超链接形式标明文章原始出处和作者信息及本版权声明
目前,由于工作需要,需要将程序在windows, hp-ux, aix, solaris-x86, solaris-sparc平台上打包,跨平台的打包工具当然是
InstallAnyWhere了,下面是使用它的过程中需要注意的地方。
由于不同的平台使用的代码不一样,那样我需要依据不同的平台去copy相应的依赖包,所以我必须从InstallAnyWhere里
知道何种操作系统,虽然通过变量 $prop.os.name$ 我可以知道平台名称 (InstallAnyWhere的标准变量可参见,http://helpnet.flexerasoftware.com/robo/projects/installanywhere%207.1%20help%20library/Standard_InstallAnywhere_Variables.htm),但是在window平台里取出的却是"Windows 2003“之类的字眼,显示后面的
2003不是我想的,我需要截取字符串,但是却找不着相应的函数,所以这也是InstallAnyWhere功能不足地方,在这点上,
SetupFactory显然要比它强得多,只不过 SetupFactory做出的安装包无法跨平台。研究了近半天,没有思路,后来我想到了
JAVA,也就是我将所有的安装过程全部在JAVA里做,在JAVA代码中去获取OS名称,去执行maven命令,去复制,删除文件等等。。客户化代码实际上很简单,步骤如下:
1) 将InstallAnyWhere安装目录下的IAClasses.zip加入到classpath中。
2) 继承CustomCodeAction类,实现install方法,如:
public class ModifyString extends CustomCodeAction
{
public void install(InstallerProxy ip) throws com.zerog.ia.api.pub.InstallException
{
//参数的传入以InstallAnyWhere的变量传入
String path = ip.substitute("$MS_STRING_TO_CONVERT$");
//在这里通过JAVA代码你想做什么做什么
//程序结果通过变量传出
ip.setVariable("MS_STRING_TRIM", path.trim());
}
像上面那样做就可以可以通过调用Custom Code功能调用了。如果我们还想做成插件的话,就要在classpath根目录下新建一个customcode.proprties文件,内容为
plugin.main.class=com.zerog.ia.customcode.utils.ModifyString
plugin.name=ModifyString
plugin.type=action
plugin.available=preinstall,install, postinstall,preuninstall, postuninstall
property.MS_STRING_TO_CONVERT=
如果是插件,打成JAR包后,要放到D:\Program Files\InstallAnywhere2009Enterprise\plugins目录下。
3)打成JAR包。
4)使用时,在添加action里有“添加custom code“这一项吧,将定这个JAR包,并指定主类指可。如果如了第三方依赖包,在
dependency中设置。如果是插件,放到 D:\Program Files\InstallAnywhere2009Enterprise\plugins目录下,在action中直接用就行了。
既然可以通过JAVA来做任何安装要做的事情,现在剩下的事情就是需要JDK环境,在http://www.flexerasoftware.com/products/installanywhere/files-utilities.htm里有各个操作系统平台上的VM,下载下来,放到D:\Program Files\InstallAnywhere2009Enterprise\resource\installer_vms目录下去,然后重启InstallAnyWhere,你就会发现在“Build"菜单里看到这些JDK了。当然,这些VM
你也可以自己做,你将后缀改成zip,用winwar打开,你一看就知道这些VM怎么做了,非常非常简单的。
最后需要注意的就是变量了,如想跨平台,文件分隔符就得用 $/$ or $\$ , 环境变量里的分隔符得用
$;$ or $:$ ,更多的标准变量可参见:http://helpnet.flexerasoftware.com/robo/projects/installanywhere%207.1%20help%20library/Standard_InstallAnywhere_Variables.htm
其他注意点:
1)修改图标,有四处:
1、显示“语言选择”处的欢迎页,Install UI --Look & Feel -- General UI Settings -- Startup Spalash Screen
2、安装界面左边的背景,Install UI --Look & Feel -- General UI Settings -- Installer Background Image
3、安装界面右边的图片,Install UI --Billboards
2)添加安装集:
1、Organization -- Install Sets : 如Complete Installation 与Documentation-only Installation两个
2、Organization -- Features与Install Sets关联,如Sever与Doc两个Feature
3、Install面板中将文件夹docs与Feature Server关联,其余文件与Feature Doc 关联
3、在Install时得添加Pre-Panel, Choose Install Sets
在使用InstallAnyWhere之后,我觉得它不怎么样,缺点如下:
1)有时候往往不同的OS有不同的安装文件,它无法根据不同的OS解压相应的文件,而是整个将文件全拷到用户机器上,这点它有点弱。
(补充一下:后来熟练之后,知道了可以通过Rules解决 )
2)虽然它可以通过变量取得操作系统名称,但是取出的却是windows 2003之类的,所以需要进行字符串截取,
而它却没有进行字符串截取的函数,虽然可以通过Custom Code加以解决,却还是显得功能弱,起码也得像SQL语句那样吧
3)小问题不断,如修改文本文件的action无法存在跨平台行换行符的问题,搞得我后来只能通过Custom Code加以解决了
当然 InstallAnyWhere也有很多优点,如下:
1)支持多OS平台,如linux上生成的bin包也可以以GUI形式运行(直接双击,再点运行即可出来JAVA的图形化界面)
2)支持JAVA扩展,Custom Code,这点不错。
下面是我自己做了一个linux下的BIN包,不过只支持Cosole安装,不支持图形化界面:
1) 新建一个脚本install.sh,内容如下:
#!/bin/bash
installDir=${PWD}
if [ X$1 != "X" ];then
installDir=$1;
fi
mkdir $installDir
sed -n -e '1,/^exit 0$/!p' $0 > "${installDir}packages.tar.gz 2>/dev/null
cd $installDir
tar zxf packages.tar.gz
tar zxf vm.tar.Z
JAVA_HOME=./jre
export JAVA_HOME
chmod -R 777 ./jre/bin/*
./jre/bin/jar -xf quqi.jar
chmod 777 ./lib/ant/bin/ant
./lib/ant/bin/ant -f ant.xml
rm -f ant.xml
exit 0
2)存在两个文件quqi.jar与vm.tar.Z,quqi99.jar是我们要安装的文件,vm.tar.Z是JDK
先通过下列命令将上述两个文件打成包packages.tar.gz
tar zcvf packages.tar.gz bes.jar vm.tar.Z
3)执行命令 cat install.sh packages.tar.gz > install.bin
这样将install.sh与packages.tar.gz文件合在一起install.bin,并且install.sh在前
而脚本中的命令 sed -n -e '1,/^exit 0$/!p' $0 > "${installDir}packages.tar.gz 2>/dev/null
用于将packages.jar再从install.bin中提取出来,sed命令代表一行一行的读,
/^exit 0$/代表读到以exit开头,0结尾的一行,有点类似正则
1,/^exit 0$/代表从1到exit 0的这些行
p代表打印,
!代表非
所以 '1,/^exit 0$/!p'代表打印除了1到exit 0的行,也就是packages.tar.gz的内容了。
> "${installDir}packages.tar.gz 代表写入到文件中,
2>/dev/null则代表写文件时不输出信息
4)安装时可以这样 ./install.bin /home/zhanghua/test
补充经验:
1)silent安装模式:
定义一个配置文件installer.properties,内容如下:
INSTALLER_UI=SILENT
USER_INSTALL_DIR=c:\\test
CHOSEN_INSTALL_SET=mySet1
.....
调用方式: install.exe -f installer.properties
2)Console安装模式
调用方式:install.exe -i console
另外,在添加Panel时,得添加Console专用的Panel
注意:并且Console模式下定义输入参数时,多个参数共用一个变量,变量的值类似 "aa","bb" ,其中aa为参数标签名,
并且aa上还有两个逗号,这个在使用过程中是不便的,尤其由于在执行完Pre-Install(主要是Panel)后才执行Install(主要是action)
这样会导致Panel与Action中的使用模式不一致哦。这也是installanywhere大问题没有,小问题不断的一个例子,看来一个软件产品
采用让大家多提意见对改善易用性是很有好处的。
3)关于卸载软件,在安装过程中生成的文件,installanywhere不会给我们 卸载的,那我们可以在我们的Cumstom Code Action
中的uninstall方法中 卸载 :
public static void delDir(File f) throws Exception{
f(f.exists()){
File[] subFiles = f.listFiles();
if(subFiles.length==0){
f.delete();
}else{
for(int i=0;i<subFiles.length;i++){
File subFile = subFiles[i];
if(subFile.isDirectory()){
delDir(subFile);
}
subFile.delete();
if(subFile.getparentFile().list().length==0)
subFile.getParentfile().delete();
}
}
}
}
不过在uninstall中调用它时可要注意一点,不要直接调这个方法直接删除我们安装根目录哦(因为
installanywhere工具自身的 卸载在根目录下,你这样一删,会造成删除不干净的),切记切记。
4)如果要校验数据的话,例如,如何用户输入的IP不对,让安装的下一步不可用怎么做呢?
可以在Pre-Install上使用Show Message Dialog这个内置Panel,注意一定是要在 Pre-Install中使用,不能在
Install中使用,因为在Install中一旦安装正在进行是没法让Panel回到Previous的,在这个 Show Message Dialog上
使用它的Rules中通过Evaluate Custom Rule添加一个自定义的Rule,自定义的Rule如下:
public class DataVerify extends CustomCodeRule{
public boolean evaluateRule(){
}
}
5)locale修改,在工程的目录下有两个文件custom_en, custom_zh_CN,直接修改即可。蛮简单。可以这样:
在做安装文件的时候,直接写英文,做完之后,只修改 custom_zh_CN这一个文件,翻译即可。
6)关于输入框的问题,比如说一个输入Panel上有两个radio,它存储在一个变量INSTLL_TYPE中,那么会出现下列变量:
INSTALL_TYPE=\"",\"Advanced\"
INSTALL_TYPE_1=
INSTALL_TYPE_2=Advanced
INSTALL_TYPE_TYPE_BOOLEAN_1=0
INSTALL_TYPE_TYPE_BOOLEAN_2=1
显然,如果locale化的话,不应该用到label的值,所以得用 INSTALL_TYPE_TYPE_BOOLEAN
installanywhere很多细节地方做的不大好,如下:
1)它最大的败笔就是对console,silent,swing三种安装模式的变量处理的不一致,比如说由于有swing中上述第6)点问题的存在
,在silent模式时,你若直接使用 INSTALL_TYPE_TYPE_BOOLEAN_1是不行的,它会给你重置,它还是得在silent的配置文件中
使用 INSTALL_TYPE,然后 在custom code代码反重置 INSTALL_TYPE_TYPE_BOOLEAN_1变量。
2) 它最大的败笔就是对console,silent,swing三种安装模式的变量处理的不一致。比如说两上checkbox,
它在swing模式时有两个变量与之对应,而在console中却有一个变量与之对应,这样造成很多不便,你得自己去拆解
3)它的rule不支持小括号等复杂的逻辑表示,这方面的功能过弱。
4)它的rule还有一个败笔,它估计是异步执行的,如果执行一个rule时间很长,它可能自己就先返回了。
5)它所有的输入panel无法排版,连超码的对齐都没有,只能通过空格来对齐。
6)在console模式中,它的install set估计有BUG,我有两个feature,一个是应用,一个是文档,当我选文档时
,它实际上安装的是应用,当我选应用时它安装的是文档。首先我能保证我的做法是没有问题的,那么那就只能是installanywhere的BUG了
7)在swing模式中,一个panel中可以定义多个元素,而在console中,由于一个panel只能定义一个变量,所以它也只能定义
一个元素,这样造成panel激增。
怎么说了,installanywhere给我的印象是它的功能挺不错,但是它不怎么注重细节,大问题没有,小问题却不断,并且是它本来
是可以做好的,如输入框没有对齐功能的问题,但它却不去做。真不知道开发这输入框的这个哥们究竟是怎么想的,它难道就没有考虑到
这样做对用户相当不友好吗?