赞
踩
一:背景
由于我们以SAAS服务方式,为客户分发软件包,每个软件包中都有客户自定义的一些特定信息,如:公司名称、LOGO、快捷方式名称、安装路径指定等,这些信息是在后台操作界面中指定,如下:
目前我们采用的是在客户下载客户端时,根据后台的设置,使用WINRAR生成一个自解压包,其中将客户定制的这些信息均会被自动打包到此自解压包中;在一般情况下,我们的这个方案还是可行的,并且也使用此方案运营了几年。
但我们逐步发现此方案有如下缺点:
1. 使用WINRAR自解压方式,没有请求管理员权限运行,导致在WIN7等系统上安装不成功的问题
2. 快捷方式创建只能在当前用户桌面
3. 由于使用的是自解压运行方式,并且没有添加数字签名【动态生成的,为了安全,没有在生成后再加签名,并且有些环境下也签不了名】,会引起如金山毒霸的报毒
4. 在WIN7上运行安装后,会提示此软件未正确安装的问题
以上问题,给客户的正常使用带来了非常多的困扰,特别是第3点,让客户误以为我们的安装包有病毒【我们为了客户安装的便利性,一次性做成了自解压运行行为】,在此背景下,我们进行了探索,使用了另外的方式来进行安装包的生成与安装。
二:方案与工具
1. 涉及工具:
Inno setup为打包工具
2. 方案
此方案采用InnoSetup来进行客户端文件的打包,完成后进行签名,放到服务器上后,即不再对此文件包进行任何变更,以免破坏其数字签名信息;
在SAAS服务器上针对不同客户进行信息定制时,以不同的文件名来反映出不同的客户信息。如:ClientSetup_100000001.exe
此文件名用于标识不同的客户,如:1000000为客户标识,后面的01作为对1000000的校验串,如果客户无意修改了文件名,则通过提示信息告知。
启动安装时,ClientSetup_100000001.exe先对文件名进行校验,如果正确,则通过此ID获取其定制的信息【连接的服务器地址已经包含在安装包中】,获取到安装地址、快捷方式名称、LOGO等信息后,控制后续的安装流程进行安装。
这里会涉及到安装过程中的网络访问问题,特别是代理的处理,所以需要处理好在安装时的网络访问,目前我们的处理是:
1) 如果直接能访问网络,下载信息,则下载,如果失败,则进入第2步
2) 如果检测到IE设置了代理,则通过此代理进行访问,如果需要授权则进入第3步
3) 则要设置代理信息等
见下图:
三:具体实现
由于我们对Pascal 脚本不熟悉,所以我更多的是使用C++写成DLL,通过在innosetup的相关事件及函数中调用的方式来实现。
1. 检测文件名是否有应该放在安装包刚启动时进行检测,在事件InitializeSetup中检测,如果文件名校验不正确,则直接退出安装:
- function InitializeSetup(): Boolean;
-
- begin
-
- if StringCheck(ExpandConstant('{srcexe}')) = 0 then
-
- begin
-
- MsgBox('文件名无效,请重新下载安装包!', mbConfirmation, MB_OK);
-
- Result := False;
-
- end
-
- else
-
- begin
-
- Result := True;
-
- end
-
- MyProgChecked := False;
-
- ShotCutName := '';
-
- end;
-
-
2. 在文件检验检测通过,则开始复制文件,当复制了辅助安装的DLL后,即可在其AfterInstall中进行网络访问,获取服务器中定制的信息:
- Source: "..\dependency\RTSClient.ini"; DestDir: "{app}"; Flags: ignoreversion
-
- Source: "..\dependency\ClientLogo.ico"; DestDir: "{app}"; Flags: ignoreversion;
-
- Source: "..\ch\ProxyWrapper.dll"; DestDir: "{app}"; Flags: ignoreversion;
-
- Source: "..\ch\SetupWrapper.dll"; DestDir: "{app}"; Flags: ignoreversion; AfterInstall:DownLoadCustomizeData()
-
3. 对于DLL的引用,需要进行声明后方可调用:
- function StringCheck(lpPath: AnsiString): Integer;
-
- external 'StringCheck@files:SetupWrapper.dll stdcall setuponly';
-
-
-
- function DownCustomizeData(lpSetupPath:AnsiString): Integer;
-
- external 'DownCustomizeData@files:SetupWrapper.dll stdcall setuponly';
-
-
-
- function GetSpecificPath(lpchar:PChar; nFlag:Integer): Integer;
-
- external 'GetSpecificPath@files:SetupWrapper.dll stdcall setuponly';
-
-
-
- function CopyCustomizeData(lpPath: AnsiString): Integer;
-
- external 'CopyCustomizeData@files:SetupWrapper.dll stdcall setuponly';
-
-
4. 在安装的过程中,要对路径选择、快捷方式创建等窗口进行隐藏,如下:
- function ShouldSkipPage(PageID: Integer): Boolean;
-
- begin
-
- Result := False;
-
- if PageID <> wpFinished then
-
- begin
-
- Result := True;
-
- end
-
- end;
-
-
-
- procedure CurPageChanged(CurPageID: Integer);
-
- begin
-
- if CurPageID <> wpFinished then
-
- begin
-
- PostMessage(WizardForm.NextButton.Handle,WM_LBUTTONDOWN,0,0);
-
- PostMessage(WizardForm.NextButton.Handle,WM_LBUTTONUP,0,0);
-
- end
-
- end;
-
5. 在获取到真正的安装路径后,需要将文件复制的路径定位到真正路径下[快捷方式的创建也是同样]:
- Source: "..\ch\RTSKeyGenDll.Dll"; DestDir: "{code:GetRealSetupPath}"; Flags: ignoreversion
-
- Name: "{group}\{code:GetShotCutName}"; Filename: "{code:GetRealSetupPath}\{#MyAppExeName}"; WorkingDir:"{code:GetRealSetupPath}"; IconFilename: "{code:GetRealSetupPath}\ClientLogo.ico"
-
这样才能保证安装的信息与服务器上配置的一致。
6. 对于调用的DLL中的函数,则具体执行文件名检测、服务器数据下载等事务,在此就不多述了
7. 打包的完整代码脚本如下【部分与公司业务相关代码未包含】
- ; 脚本由 Inno Setup 脚本向导 生成!
- ; 有关创建 Inno Setup 脚本文件的详细资料请查阅帮助文档!
-
- #define MyAppName "远程在线系统客户端"
- #define MyAppVersion "3.1"
- #define MyAppPublisher "测试公司名称"
- #define MyAppURL "http://www.test.com/"
- #define MyAppExeName "Test.exe"
-
- [Setup]
- ; 注: AppId的值为单独标识该应用程序。
- ; 不要为其他安装程序使用相同的AppId值。
- ; (生成新的GUID,点击 工具|在IDE中生成GUID。)
- AppId={{2C5393DC-8D35-4D13-B7AB-B389AE5E4111}
- ;AppName={code:testtest|a}
- AppName={#MyAppName}
- AppVersion={#MyAppVersion}
- AppVerName={#MyAppName} {#MyAppVersion}
- AppPublisher={#MyAppPublisher}
- AppPublisherURL={#MyAppURL}
- AppSupportURL={#MyAppURL}
- AppUpdatesURL={#MyAppURL}
- VersionInfoDescription={#MyAppName}
- VersionInfoProductTextVersion={#MyAppVersion}
- VersionInfoVersion={#MyAppVersion}
- DefaultDirName={pf}\Test
- DefaultGroupName={#MyAppName}
- OutputDir=.\..\package
- OutputBaseFilename=Test
- Compression=lzma
- SolidCompression=yes
-
- [Languages]
- Name: "chinesesimp"; MessagesFile: "compiler:Default.isl"
-
- [Files]
- Source: "..\dependency\RTSClient.ini"; DestDir: "{app}"; Flags: ignoreversion
- Source: "..\dependency\ClientLogo.ico"; DestDir: "{app}"; Flags: ignoreversion;
- Source: "..\ch\ProxyWrapper.dll"; DestDir: "{app}"; Flags: ignoreversion;
- Source: "..\ch\SetupWrapper.dll"; DestDir: "{app}"; Flags: ignoreversion; AfterInstall:DownLoadCustomizeData()
- ;完成LOGO后,需要将Rtsclient.ini与logo复制过去
- Source: "..\ch\AESEncryption.dll"; DestDir: "{code:GetRealSetupPath}"; Flags: ignoreversion; AfterInstall:CopyLogo()
- Source: "..\ch\ChatLib.dll"; DestDir: "{code:GetRealSetupPath}"; Flags: ignoreversion
- Source: "..\ch\FileTransferLib.dll"; DestDir: "{code:GetRealSetupPath}"; Flags: ignoreversion
-
- [Icons]
- Name: "{group}\{code:GetShotCutName}"; Filename: "{code:GetRealSetupPath}\{#MyAppExeName}"; WorkingDir:"{code:GetRealSetupPath}"; IconFilename: "{code:GetRealSetupPath}\ClientLogo.ico"
- Name: "{group}\{cm:UninstallProgram,{code:GetShotCutName}}"; Filename: "{uninstallexe}"; WorkingDir:"{code:GetRealSetupPath}"; IconFilename: "{code:GetRealSetupPath}\ClientLogo.ico"
- Name: "{commondesktop}\{code:GetShotCutName}"; Filename: "{code:GetRealSetupPath}\{#MyAppExeName}"; WorkingDir:"{code:GetRealSetupPath}"; IconFilename: "{code:GetRealSetupPath}\ClientLogo.ico"
-
- [Run]
- Filename: "{code:GetRealSetupPath}\{#MyAppExeName}"; Description: "{cm:LaunchProgram,{code:GetShotCutName}}"; Flags: nowait postinstall skipifsilent
-
- [Code]
- const
- WM_LBUTTONDOWN = 513;
- WM_LBUTTONUP = 514;
-
- var
- MyProgChecked: Boolean;
- RealSetupPath: String;
- ShotCutName : String;
- function StringCheck(lpPath: AnsiString): Integer;
- external 'StringCheck@files:SetupWrapper.dll stdcall setuponly';
-
- function DownCustomizeData(lpSetupPath:AnsiString): Integer;
- external 'DownCustomizeData@files:SetupWrapper.dll stdcall setuponly';
-
- function GetSpecificPath(lpchar:PChar; nFlag:Integer): Integer;
- external 'GetSpecificPath@files:SetupWrapper.dll stdcall setuponly';
-
- function CopyCustomizeData(lpPath: AnsiString): Integer;
- external 'CopyCustomizeData@files:SetupWrapper.dll stdcall setuponly';
-
- function InitializeSetup(): Boolean;
- begin
- if StringCheck(ExpandConstant('{srcexe}')) = 0 then
- begin
- MsgBox('文件名无效,请重新下载安装包!', mbConfirmation, MB_OK);
- Result := False;
- end
- else
- begin
- Result := True;
- end
- MyProgChecked := False;
- ShotCutName := '';
- end;
-
- procedure CancelButtonClick ( CurPageID : Integer; var Cancel, Confirm: Boolean);
- begin
- Cancel := True;
- Confirm := False;
- end;
-
- function GetShotCutName(Param: String) : String;
- var
- ShotName: String;
- ReturnLength: Integer;
- begin
- if Length(ShotCutName) = 0 then
- begin
- SetLength(ShotName, 270);
- ReturnLength := GetSpecificPath(ShotName, 2);
- ShotCutName := Copy(ShotName, 0, ReturnLength);
- //MsgBox(ShotCutName, mbConfirmation, MB_OK);
- end
- Result := ShotCutName;
- end;
- procedure CopyLogo();
- begin
- CopyCustomizeData('');
- end;
-
- procedure DownLoadCustomizeData();
-
- var
- RealPath: String;
- PathLength: Integer;
- ReturnValue: Integer;
- begin
- //在此处去获取服务器配置等信息
- ReturnValue := DownCustomizeData(ExpandConstant('{app}'));
- if ReturnValue < 0 then
- begin
- MsgBox(Format('连接服务器失败[错误码:%d],将按默认设置进行安装!', [ReturnValue]), mbConfirmation, MB_OK);
- //PostMessage(WizardForm.CancelButton.Handle,WM_LBUTTONDOWN,0,0);
- //PostMessage(WizardForm.CancelButton.Handle,WM_LBUTTONUP,0,0);
- end
- //else
- //begin
- //设置好真正的安装目录
- //设置好是否换了LOGO
- SetLength(RealPath, 270);
- PathLength := GetSpecificPath(RealPath, 1);
- RealSetupPath := Copy(RealPath, 0, PathLength);
- //MsgBox(RealSetupPath, mbConfirmation, MB_OK);
- //end
- end;
-
- function GetRealSetupPath(Param: String) : String;
- begin
- Result := RealSetupPath;
- end;
-
- function ShouldSkipPage(PageID: Integer): Boolean;
- begin
- Result := False;
- if PageID <> wpFinished then
- begin
- Result := True;
- end
- end;
-
- procedure CurPageChanged(CurPageID: Integer);
- begin
- if CurPageID <> wpFinished then
- begin
- PostMessage(WizardForm.NextButton.Handle,WM_LBUTTONDOWN,0,0);
- PostMessage(WizardForm.NextButton.Handle,WM_LBUTTONUP,0,0);
- end
- end;
-
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。