Inno Setup执行SQL脚本的方法

作为和NSIS并立的、两个最流行的免费Windows应用程序安装包制作工具之一,Inno在学习难度上相对要低一些,非常适合对一些简单的桌面程序打包。但对于较复杂的安装过程,或者Web应用程序来说,我个人觉得不是Inno的强项。当然,既然Inno内嵌了Pascal语言用以扩展功能,理论上不是不可以应付复杂的安装过程,但实现起来要复杂一些。

比如对于在安装过程中连接数据库并执行SQL脚本这样的需求,使用InstallShield应该会简单地多,而Inno却不支持直接操作数据库,并且相关的资料说明少之又少,还不如NSIS丰富,以至于我踏破铁鞋无觅处,最终却在NSIS的资料中找到了思路。

主要的思路是,在安装过程中,调用数据库客户端连接数据库并执行SQL脚本,然后将执行结果或错误信息输出到文件中,最后通过分析这个文件来判断命令执行的结果。但是,既然是调用特定的客户端,那么对不同数据库的操作自然就有所区别,具体情况如下所述。

首先在打包脚本的[Files]段将必需的文件包含进来:

[Files]
Source: "D:/Development/MyDemoApp/code/*"; DestDir: "{app}"; Flags: ignoreversion recursesubdirs createallsubdirs

;osql.exe在SQL Server2000安装目录中
Source: "D:/Development/MyDemoApp/osql.exe"; Flags: dontcopy
Source: "D:/Development/MyDemoApp/MySQL.exe"; Flags: dontcopy
Source: "D:/Development/MyDemoApp/script_mssql.sql"; Flags: dontcopy
Source: "D:/Development/MyDemoApp/script_mysql.sql"; Flags: dontcopy
Source: "D:/Development/MyDemoApp/script_oracle.sql"; Flags: dontcopy

在SQL Server中执行脚本的代码片断:

function ExecScriptInMSSQL(DBHost, DBLogin, DBPass, DBName: String): Boolean;
var
 ConnectExe: String;
 ConnectParam: String;
begin
 {解压临时文件}
 ExtractTemporaryFile(‘osql.exe‘);
 ExtractTemporaryFile(‘script_mssql.sql‘);
 {构造数据库连接字符串}
 ConnectExe := ExpandConstant(‘{tmp}‘) + ‘/osql.exe‘;
 ConnectParam := ‘ -S ‘ + DBHost
  + ‘ -U ‘ + DBLogin
  + ‘ -P ‘ + DBPass
  + ‘ -d ‘ + DBName
  + ‘ -i script_mssql.sql -o ‘
  + ExpandConstant(‘{tmp}‘) + ‘/dbstatus.txt‘;
 {建立数据库连接并执行脚本}
 if Exec(ConnectExe, ConnectParam, ‘‘, SW_HIDE, ewWaitUntilTerminated, ResultCode) then begin
  Result := ResultCode = 0;
  LoadStringFromFile(ExpandConstant(‘{tmp}‘) + ‘/dbstatus.txt‘, StatusString);
  if StatusString <> ‘‘ then begin
   MsgBox(StatusString, mbError, MB_OK);
   Result := False;
  end else begin
   Result := True;
  end;
 end else begin
  MsgBox(‘Database update failed:‘#10#10 + SysErrorMessage(ResultCode), mbError, MB_OK);
  Result := False;
 end;
end;

在MySQL中执行脚本的代码片断:

function ExecScriptInMYSQL(DBHost, DBLogin, DBPass, DBName: String): Boolean;
var
 ConnectExe: String;
 ConnectParam: String;
begin
 {解压临时文件}
 ExtractTemporaryFile(‘mysql.exe‘);
 ExtractTemporaryFile(‘script_mysql.sql‘);
 {构造数据库连接字符串}
 ConnectExe := ExpandConstant(‘cmd‘);
 ConnectParam := ‘ /c "‘ + ExpandConstant(‘{tmp}‘) + ‘/mysql.exe‘
  + ‘ -h‘ + DBHost
  + ‘ -u‘ + DBLogin
  + ‘ -p‘ + DBPass
  + ‘ -D‘ + DBName
  + ‘ -e "source ‘ + ExpandConstant(‘{tmp}‘) + ‘/script_mysql.sql""> ‘ + ExpandConstant(‘{tmp}‘) + ‘/dbstatus.txt 2>&1‘;
 {建立数据库连接并执行脚本}
 if Exec(ConnectExe, ConnectParam, ‘‘, SW_HIDE, ewWaitUntilTerminated, ResultCode) then begin
  Result := ResultCode = 0;
  LoadStringFromFile(ExpandConstant(‘{tmp}‘) + ‘/dbstatus.txt‘, StatusString);
  if StatusString <> ‘‘ then begin
   MsgBox(StatusString, mbError, MB_OK);
   Result := False;
  end else begin
   Result := True;
  end;
 end else begin
  MsgBox(‘Database update failed:‘#10#10 + SysErrorMessage(ResultCode), mbError, MB_OK);
  Result := False;
 end;
end;

由于mysql.exe没有输出结果到文件的参数,故需要使用cmd.exe来运行mysql.exe以便将其输出重定向到文件dbstatus.txt中。此外,在命令的最后加上参数2>&1,将标准错误输出设备也重定向到文件上,否则命令执行的错误信息不会输出到文件中。

在Oracle中执行脚本的代码片断:
function ExecScriptInORACLE(ClientPath, DBInstance, DBLogin, DBPass: String): Boolean;
begin
 {解压临时文件}
 ExtractTemporaryFile(‘script_oracle.sql‘);
 {连接数据库并执行脚本}
 if Exec(ExpandConstant(‘cmd‘), ‘ /c "‘ + ClientPath + ‘ -L -S ‘ + DBLogin
  + ‘/‘ + DBPass
  + ‘@‘ + DBInstance
  + ‘ @‘ + ExpandConstant(‘{tmp}‘) + ‘/script_oracle.sql> ‘ + ExpandConstant(‘{tmp}‘) + ‘/dbstatus.txt 2>&1‘,
  ‘‘,
  SW_HIDE, ewWaitUntilTerminated, ResultCode)
 then begin
  Result := ResultCode = 0;
  LoadStringFromFile(ExpandConstant(‘{tmp}‘) + ‘/dbstatus.txt‘, StatusString);
  if Pos(‘holytail‘, StatusString) <> 0 then begin
   {若输出信息中有“holytail”的子串,则表示脚本成功执行}
   {若执行有误,提示用户打开日志文件}
   if Pos(‘ORA-‘, StatusString) <> 0 then begin
    {提示用户脚本执行出错}
    if MsgBox(‘数据库更新出错,是否打开日志文件?‘, mbConfirmation, MB_YESNO) = IDYES then begin
     {打开日志}
     if not ShellExec(‘‘, ExpandConstant(‘{tmp}‘) + ‘/dbstatus.txt‘, ‘‘, ‘‘, SW_SHOW, ewNoWait, ErrorCode) then begin
      MsgBox(‘日志文件打开错误!‘, mbError, MB_OK);
     end;
    end;
    Result := False;
   {若执行无误,返回True}
   end else begin
    Result := True;
   end;
  end else if StatusString <> ‘‘ then begin
   MsgBox(StatusString, mbError, MB_OK);
   Result := False;
  end else begin
   Result := True;
  end;
 end else begin
  MsgBox(‘Database update failed:‘#10#10 + SysErrorMessage(ResultCode), mbError, MB_OK);
  Result := False;
 end;
end;
Oracle的客户端太大,不能集成到安装包中,应使用一个TInputFileWizardPage由用户选择sqlplus.exe的安装位置。同时,由于sqlplus.exe也没有输出结果到文件的参数,也须使用cmd.exe来运行它并重定向输出到文件。此外,由于sqlplus.exe执行脚本时无论成功还是失败,都会输出信息,故无法像使用sqlcmd.exe和mysql.exe那样简单地判断脚本是否执行成功,需要在脚本的最后通过select语句输出一个特殊的字符串到文件中,然后通过判断dbstatus.txt中是否存在该字符串来判断脚本的执行情况;且由于sqlplus.exe执行完脚本后不会自动退出,还要在脚本最后加上exit语句;故script_oracle.sql的最后必须是如下内容:

SELECT ‘holytail‘ FROM dual;
exit;

时间: 2024-08-07 00:17:46

Inno Setup执行SQL脚本的方法的相关文章

InstallShield在MySQL和Oracle中执行SQL脚本的方法InstallShield在MySQL和Oracle中执行SQL脚本的方法

简述 InstallShield已经内建了对MySQL和Oracle的支持.但是这个功能是通过ODBC实现的,它对SQL脚本的格式要求非常严格,因此已经通过官方客户端测试的脚本在IS中执行时往往就会报错. 一般来说,数据库脚本只保证通过官方客户端测试即可,同时维护一份供IS执行的脚本费时费力.因此,考虑安装程序对两数据库的支持通过官方客户端实现. MySQL   function InstallMySQLComponent(szComponent) NUMBER nResult; STRING

Python中执行sql脚本时GO语句出错问题

sql脚本: --创建链接服务器 (其中db2server是DB2的ip地址或服务器名)exec sp_addlinkedserver 'srv_lnk', '', 'SQLOLEDB','10.138.60.94'exec sp_addlinkedsrvlogin 'srv_lnk','false',null,'sa','123456' Go--数据导入(表B存在) --Insert into srv_lnk.sys.Estamp.EstampConfig(field1,field2)INSE

Inno Setup的常用脚本

Inno Setup的常用脚本 分类: VC++神奇理论 2012-12-06 10:07 3234人阅读 评论(2) 收藏 举报 安装不同的目录: [Files] Source: "我的程序\*"; DestDir: "{app}"; Flags: ignoreversion recursesubdirs createallsubdirs Source: "我的程序\*"; DestDir: {cf}\我的程序; Flags: ignoreve

inno setup教程解释脚本

inno setup教程解释脚本 2007-04-08 21:31:36|  分类: 科技-> Inno Setu |  标签:inno   |举报 |字号 订阅 下载LOFTER客户端 ;Inno Setup 是一个免费的 Windows 安装程序制作软件.;第一次发表是在 1997 年,Inno Setup 今天在功能设置和稳定性上的竞争力可能已经超过一些商业的安装程序制作软件;它是真正免费的软件,即使作为商业应用;还有一个特点,压缩率特别高,特别适合VB软件,我这个安装打包后只有3.7M,

.net(C#)在Access数据库中执行sql脚本

自己写的一个工具类,主要是业务场景的需要. 主要有两个功能: ①执行包含sql语句的字符串 ②执行包含sql语句的文件 调用方式 1 /// <summary> 2 /// 执行sql语句 3 /// </summary> 4 /// <param name="sql">需要执行的sql语句</param> 5 public bool ExecuteSql(string sql, ref string errorMsg) 6 { 7 Se

Java执行SQL脚本文件到数据库

方式一:直接读取SQL脚本文件的内容,然后传递到SQL中. 代码:RunSqlService: @Autowired private RunSqlDao runSqlDao; /** * 读取文件内容到SQL中执行 * @param sqlPath SQL文件的路径:如:D:/TestProject/web/sql/脚本.Sql */ public void runSqlByReadFileContent(String sqlPath) throws Exception { try { Stri

mysql下如何执行sql脚本

1.编写sql脚本,假设内容如下: create database dearabao;  use dearabao;  create table niuzi (name varchar(20)); 保存脚本文件,假设我把它保存在F盘的hello world目录下,于是该文件的路径为:F:\hello world\niuzi.sql 2.执行sql脚本,可以有2种方法:  第一种方法: 在命令行下(未连接数据库),输入 mysql -h localhost -u root -p123456 < F

SQLYog执行SQL脚本提示:错误代码: 1067 - Invalid default value for &#39;数据库表&#39;查询:解决办法

强烈建议:完全卸载当前版本MySQL,重新安装5.6及以上版本 完全卸载方法:https://jingyan.baidu.com/article/3d69c551611290f0ce02d77b.html 卸载完之后记得删除C:\ProgramData下的隐藏文件MySQL 这是我在网上查阅多方资料,尝试无数次,踩了很多坑之后得到的最优解决办法! 至于网上修改sql_mode之类的办法,亲测无效,可能和我代码有关,不做赘述! 出现该错误原因:MySQL版本不同,导致5.6版本之前和之后的语法有很

Java执行SQL脚本导入

今天做了一个用java导入sql脚本的功能,前台上传一个sql文件,后台先保存本地,然后进行导入.主要代码如下: String cmd = "cmd /c sqlplus usr/[email protected]{//服务器IP:1521/}sid @d:\\1.sql>d:\\t.log"; Process p = null; try {     p = Runtime.getRuntime().exec(cmd); } catch (IOException e) {