16

将 db from azure bacpac 文件导入本地 sql server 2016 时,出现以下错误。

Error SQL72014: .Net SqlClient Data Provider: Msg 102, Level 15, State 1, Line 1 Incorrect syntax near 'EXTERNAL'.
Error SQL72045: Script execution error.  The executed script: CREATE EXTERNAL DATA SOURCE [BoxDataSrc]
    WITH (
    TYPE = RDBMS,
    LOCATION = N'MYAZUREServer.database.windows.net',
    DATABASE_NAME = N'MyAzureDb',
    CREDENTIAL = [SQL_Credential]
    );

(Microsoft.SqlServer.Dac)

4

5 回答 5

33

我今天遇到了同样的问题。由于“WITH(TYPE = RDBMS)”仅适用于 Azure SQL DB,因此在尝试将 bacpac 导入本地 SQL Server 2017 时出现错误。由于这篇文章,我确实找到了解决方案:

https://blogs.msdn.microsoft.com/azuresqldbsupport/2017/08/16/editing-a-bacpac-file/

相关步骤在这里重写:

  1. 制作 bacpac 文件的副本(以防出错)。
  2. 将文件扩展名更改为zip,然后将其解压缩到一个文件夹中。令人惊讶的是,bacpac 实际上只是一个 zip 文件,而不是专有且难以进入的东西。
  3. 找到 model.xml 文件并对其进行编辑以删除如下所示的部分:

    <Element Type="SqlExternalDataSource" Name="[BoxDataSrc]">
        <Property Name="DataSourceType" Value="1" />
        <Property Name="Location" Value="MYAZUREServer.database.windows.net" />
        <Property Name="DatabaseName" Value="MyAzureDb" />
        <Relationship Name="Credential">
            <Entry>
                <References Name="[SQL_Credential]" />
            </Entry>
        </Relationship>
    </Element>
    
  4. 如果您有多个这种类型的外部数据源,您可能需要对每个数据源重复第 3 步。我只有一个。

  5. 保存并关闭 model.xml。
  6. 现在您需要重新生成 model.xml 的校验和,以便 bacpac 不会认为它被篡改(因为您只是篡改了它)。创建一个名为 computeHash.ps1 的 PowerShell 文件并将此代码放入其中。

    $modelXmlPath = Read-Host "model.xml file path"
    $hasher = [System.Security.Cryptography.HashAlgorithm]::Create("System.Security.Cryptography.SHA256CryptoServiceProvider")
    $fileStream = new-object System.IO.FileStream ` -ArgumentList @($modelXmlPath, [System.IO.FileMode]::Open)
    $hash = $hasher.ComputeHash($fileStream)
    $hashString = ""
    Foreach ($b in $hash) { $hashString += $b.ToString("X2") }
    $fileStream.Close()
    $hashString
    
  7. 运行 PowerShell 脚本并为其提供解压缩和编辑后的 ​​model.xml 文件的文件路径。它将返回一个校验和值。

  8. 复制校验和值,然后打开 Origin.xml 并替换现有的校验和,朝向如下所示的行底部:

    <Checksum Uri="/model.xml">9EA0F06B282D4F42955C78A98822A31AA0ED0225CB131B8759379055A482D01F</Checksum>
    
  9. 保存并关闭 Origin.xml,然后选择所有文件并将它们放入一个新的 zip 文件中,并将扩展名重命名为 bacpac。

现在您可以使用这个新的 bacpac 导入数据库而不会出现错误。它对我有用,它也对你有用。

于 2018-05-25T14:30:25.343 回答
6

根据@SQLDoug 的回答,如果您的 Azure SQL 数据库具有外部表(即来自其他数据库的链接表),则可能会发生这种情况。您可以在此处的 SSMS 中检查:

SSMS 中的外部表节点

已接受答案的附录

如果您删除那些外部表的数据源,您还需要删除 model.xml 文件中使用这些数据源的 SqlExternalTable 元素,它们看起来像这样:

   <Element Type="SqlExternalTable" Name="[dbo].[DeliveryMethodsRestored]">
        <Property Name="ExternalSchemaName" Value="dbo" />
        <Property Name="ExternalObjectName" Value="DeliveryMethods" />
        <Property Name="IsAnsiNullsOn" Value="True" />
        <Property Name="IsQuotedIdentifierOn" Value="False" />
        <Relationship Name="Columns">
            <Entry>
                <Element Type="SqlSimpleColumn" Name="[dbo].[DeliveryMethodsRestored].[DeliveryMethodId]">
                    <Property Name="IsNullable" Value="False" />
                    <Relationship Name="TypeSpecifier">
                        <Entry>
          SNIP....
    </Element>

如果您在 model.xml 中搜索“SqlExternalTable”,您会很容易找到它们。

解决此问题的替代方法

与其在下载后更正 bacpac,另一种处理方法是在创建 bacpac 之前删除外部表,即:

  1. 将数据库的副本还原到单独的数据库
  2. 删除还原副本中的外部表

SSMS 中的外部表节点

  1. 删除恢复副本中的外部数据源

在此处输入图像描述

  1. 从恢复的副本创建 bacpac
  2. 删除副本数据库

这种方法的优点是您不会从实时数据库创建 bacpac,这显然“可能导致导出的表数据不一致,因为与 SQL Server 的物理备份/恢复不同,导出不保证事务一致性”

如果这是您可能经常做的事情,您可能会编写脚本来自动化上述大部分步骤。

于 2018-11-02T09:02:50.467 回答
0

相同的错误代码具有不同的错误。

无法导入包。警告 SQL72012:对象 [PreProd_Data] 存在于目标中,但即使您选中了“为目标数据库中但不在源数据库中的对象生成删除语句”复选框,它也不会被删除。

警告 SQL72012:对象 [PreProd_Log] 存在于目标中,但即使您选中了“为目标数据库中但不在源中的对象生成删除语句”复选框,它也不会被删除。

错误 SQL72014:.Net SqlClient 数据提供程序:消息 102,级别 15,状态 1,第 5 行“OPTIMIZE_FOR_AD_HOC_WORKLOADS”附近的语法不正确。

错误 SQL72045:脚本执行错误。执行的脚本: IF EXISTS (SELECT 1 FROM [master].[dbo].[sysdatabases] WHERE [name] = N'$(DatabaseName)') BEGIN ALTER DATABASE SCOPED CONFIGURATION SET OPTIMIZE_FOR_AD_HOC_WORKLOADS = ON; 结尾

解决方案

此博客将帮助编辑 model.xml 以删除 SQL Server 2017 实例中不需要的 OPTIMIZE_FOR_AD_HOC_WORKLOADS 的关系命令。

https://blogs.msdn.microsoft.com/azuresqldbsupport/2017/08/16/editing-a-bacpac-file/

制作 bacpac 文件的副本(以防出错)。将文件扩展名更改为zip,然后将其解压缩到一个文件夹中。令人惊讶的是,bacpac 实际上只是一个 zip 文件,而不是专有且难以进入的东西。找到 model.xml 文件并对其进行编辑以删除如下所示的部分:

相关步骤在这里重写:

  1. 制作 bacpac 文件的副本(以防出错)。

  2. 将文件扩展名更改为zip,然后将其解压缩到一个文件夹中。令人惊讶的是,bacpac 实际上只是一个 zip 文件,而不是专有且难以进入的东西。

  3. 找到 model.xml 文件并对其进行编辑以删除如下所示的部分:

    <Relationship Name="GenericDatabaseScopedConfigurationOptions">
        <Entry>
            <References Name="[OPTIMIZE_FOR_AD_HOC_WORKLOADS]" />
        </Entry>
    </Relationship>
    
  4. 从 model.xml 中删除以下块

    <Element Type="SqlGenericDatabaseScopedConfigurationOptions" Name="[OPTIMIZE_FOR_AD_HOC_WORKLOADS]">
        <Property Name="GenericValueType" Value="2" />
        <Property Name="GenericValue" Value="ON" />
    </Element>
    
  5. 保存并关闭 model.xml。

  6. 现在您需要重新生成 model.xml 的校验和,以便 bacpac 不会认为它被篡改(因为您只是篡改了它)。创建一个名为 computeHash.ps1 的 PowerShell 文件并将此代码放入其中。

  7. 运行 PowerShell 脚本并为其提供解压缩和编辑后的 ​​model.xml 文件的文件路径。它将返回一个校验和值。

  8. 复制校验和值,然后打开 Origin.xml 并替换现有的校验和。

  9. 保存并关闭 Origin.xml,然后选择所有文件并将它们放入一个新的 zip 文件中,并将扩展名重命名为 bacpac。

现在 bacpack 文件将准备好导入,它对我有用。

谢谢。

于 2019-08-13T10:26:34.560 回答
0

弹性数据库查询仅在 Azure SQL 数据库 v12 或更高版本上受支持,在本地服务器上不支持。 https://msdn.microsoft.com/en-us/library/dn935022.aspx

于 2017-03-19T01:04:13.707 回答
0

导入 bacpac 时,我得到了相同的错误代码 (SQL72045),即使我们已经删除了 Azure 中用于同步数据的外部数据源。事实证明,参考另一个数据库的 SCOPED CREDENTIAL 留下了一个过程“TransferDo”。在我们删除程序后,导入运行良好。

于 2021-02-18T08:18:52.093 回答