c# - SQLCMD模式下的ServerConnection.ExecuteNonQuery

标签 c# sql-server sqlcmd data-tier-applications sql-server-data-tools

我正在使用 Microsoft Data-Tier Application framework基于 DacPackage 创建部署脚本目的。我正在尝试使用 Microsoft.SqlServer.Management.Smo.Server执行此脚本的类...

SqlConnection deployConnection = new SqlConnection(connBuilder.ToString());
deployConnection.Open();
Server server = new Server(new ServerConnection(deployConnection));
server.ConnectionContext.ExecuteNonQuery(deployScript);

但是,这个错误...

Unhandled Exception: Microsoft.SqlServer.Management.Common.ExecutionFailureException:
  An exception occurred while executing a Transact-SQL statement or batch. --->
  System.Data.SqlClient.SqlException: Incorrect syntax near ':'.

我知道这个问题的答案是我需要处于 SQLCMD 模式,但我不知道如何告诉我的 ServerConnection在所述模式下执行。

我想我的问题并不像我在标题中所说的那样具体。我真正需要做的是通过 .Net 框架执行从 DacPackage 生成的脚本。谁能帮我解决这个问题?

最佳答案

SQLCMD 模式命令不是 T-SQL 命令;它们仅适用于 SQL Server Management Studio (SSMS)/Visual Studio (VS) 和 SQLCMD.EXE。 SQLCMD 模式本质上是 SQLCMD.EXE 的工作方式,可以在 SSMS/VS 中手动启用;它是这些应用程序的一部分,而不是可以通过提供商完成的事情。

这些应用程序解释 SQLCMD 模式命令,但不会将它们传递给 SQL Server。 SQLCMD 模式命令首先被解析/执行(这是它们能够影响将要提交的 SQL 的方式),然后将 SQL 的最终版本提交给 SQL Server。

因此,SQL Server Data Tools (SSDT)/Visual Studio 生成的部署 SQL 脚本需要通过这三个程序之一运行。

由于您已经有了一个 .dacpac 文件,Microsoft 提供了几种发布您应该检查的文件的方法:

您还可以通过 DacServices.GenerateDeployScript() 创建发布 SQL 脚本,但这不会改变上述情况,因为发布/部署 SQL 脚本,无论是从 Visual Studio“Publish {project_name}”还是 GenerateDeployScript() 生成的,都是同一个脚本。这意味着,它将具有 SQLCMD 模式的冒号命令,例如 :setvar:on error exit 以及 SQLCMD 模式的变量,这至少会是在以下行中使用的 $(DatabaseName):

USE [$(DatabaseName)];

虽然可以通过设置 DacDeployOptions 来注释掉初始的 :setvarCommentOutSetVarDeclarations的属性(property)为 true,这仍然会留下 :on error exit 行以及用于 :setvar __IsSqlCmdEnabled "True" 的行检测是否已启用 SQLCMD 模式。就在这个特定的 :setvar 行之上是一条注释:

/*
Detect SQLCMD mode and disable script execution if SQLCMD mode is not supported.
To re-enable the script after enabling SQLCMD mode, execute the following:
SET NOEXEC OFF; 
*/

所以他们真的打算让这个脚本只通过 SQLCMD 运行,无论是通过 DOS -> SQLCMD.EXE 还是 PowerShell -> Invoke-SqlCMD。

从技术上讲,可以生成部署脚本内容的字符串(而不是)并通过 a) 删除任何冒号命令和 b) 替换 "$ (DatabaseName)”与您打算部署到的任何数据库。但是,我没有尝试过,我不推荐这样做,而且我不确定它是否适用于 SQL Server Data Tools 可以生成哪些部署脚本的所有情况。但这确实是一个选项。

此外,次要相关的是:您不需要 SMO 来运行 SQL 脚本。 SMO 是通过对象而不是直接通过 T-SQL 命令与 SQL Server 交互的方法。

编辑:
其他人尝试过但发现它不起作用的链接:

使生成的发布 SQL 脚本以编程方式工作的可能性:

关于c# - SQLCMD模式下的ServerConnection.ExecuteNonQuery,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26617247/

相关文章:

c# - MS 图表和 NaN

c# - 当 FirstOrDefault 在排序列表中找不到任何内容时,我如何理解?

c# - 使用正则表达式选择一个值

.net - 根据关系从具有一对多和多对一关系的表中查询数据

c# - 从 WPF 保存到 SQL Server 数据库

sql-server - PowerShell 中 Invoke-Sqlcmd 的 Unicode 支持

c# - 字符串到 byte[] 数组,反之亦然

sql - SQL Server 2008 中 "CASE"子句内的 "WHERE"语句

linux - 运行命令,然后通过SQLCMD添加结果

sql-server - 通过 sqlalchemy 和 pyodbc 访问 MS SQL 数据库时出现 "Login timeout expired"错误