使用 DACPAC,当表上有触发器时,如何重命名列?

Posted

技术标签:

【中文标题】使用 DACPAC,当表上有触发器时,如何重命名列?【英文标题】:With a DACPAC, how can I rename a column when there is a trigger on the table? 【发布时间】:2021-11-15 05:50:18 【问题描述】:

我正在尝试重命名表中存在触发器的列。我使用 SQL > Refactor > Rename 来重命名列。并且触发文件也得到更新。但是,当我发布时,我得到了这个:

过程 TR_accrual_Accrual_Update,第 134 行无效的列名 'MinHoursRule'。

这是旧的列名。我假设发布首先更新表,并看到当前/旧触发器仍然具有旧列名。

是否可以重命名列、更新触发器和发布?

我真正能想到的唯一解决方案是这样做:

删除触发器并发布 重命名列 再次添加触发器 发布

【问题讨论】:

【参考方案1】:

这是我作为解决方法所做的:

    添加新列 离开旧列 让触发器使用两组列 尽快发布/部署到产品 删除旧列 稍后发布/部署到产品

因此,我没有重命名,而是创建了新列,然后最终删除了旧列。

哎呀。但它奏效了。

注意:在我们的 C# 域模型中,我只引用了新列。

【讨论】:

【参考方案2】:

我猜您的发布配置文件设置有问题。您可能禁用了某些东西,例如“不要修改触发器”或类似的东西。我刚刚在 VS 2019 中创建了新的 SSDT 项目,结构如下:

CREATE TABLE [dbo].[test]
(
    [Id] INT ,
    b int
)

GO

CREATE TRIGGER [dbo].[Trigger_test]
    ON [dbo].[test]
    FOR DELETE, INSERT, UPDATE
    AS
    BEGIN
        SET NoCount ON
        insert into test2 select b from inserted
    END
GO
CREATE TABLE [dbo].[test2]
(
    a int
)
GO

使用默认设置将项目发布到新数据库,并对dbo.test 表进行单次插入。确保dbo.test2 表中有记录。之后,我将dbo.test.b 列重新分解为dbo.test.a,然后再次发布,一切正常。这是生成的脚本:

/*
Deployment script for trg_test

This code was generated by a tool.
Changes to this file may cause incorrect behavior and will be lost if
the code is regenerated.
*/

GO
SET ANSI_NULLS, ANSI_PADDING, ANSI_WARNINGS, ARITHABORT, CONCAT_NULL_YIELDS_NULL, QUOTED_IDENTIFIER ON;

SET NUMERIC_ROUNDABORT OFF;


GO
:setvar DatabaseName "trg_test"
:setvar DefaultFilePrefix "trg_test"
:setvar DefaultDataPath ""
:setvar DefaultLogPath ""

GO
:on error exit
GO
/*
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; 
*/
:setvar __IsSqlCmdEnabled "True"
GO
IF N'$(__IsSqlCmdEnabled)' NOT LIKE N'True'
    BEGIN
        PRINT N'SQLCMD mode must be enabled to successfully execute this script.';
        SET NOEXEC ON;
    END


GO
USE [$(DatabaseName)];


GO
PRINT N'The following operation was generated from a refactoring log file 80d0e5de-e188-465e-b83c-18f38a1cec98';

PRINT N'Rename [dbo].[test].[b] to a';


GO
EXECUTE sp_rename @objname = N'[dbo].[test].[b]', @newname = N'a', @objtype = N'COLUMN';


GO
PRINT N'Altering Trigger [dbo].[Trigger_test]...';


GO

ALTER TRIGGER [dbo].[Trigger_test]
    ON [dbo].[test]
    FOR DELETE, INSERT, UPDATE
    AS
    BEGIN
        SET NoCount ON
        insert into test2 select a from inserted
    END
GO
-- Refactoring step to update target server with deployed transaction logs

IF OBJECT_ID(N'dbo.__RefactorLog') IS NULL
BEGIN
    CREATE TABLE [dbo].[__RefactorLog] (OperationKey UNIQUEIDENTIFIER NOT NULL PRIMARY KEY)
    EXEC sp_addextendedproperty N'microsoft_database_tools_support', N'refactoring log', N'schema', N'dbo', N'table', N'__RefactorLog'
END
GO
IF NOT EXISTS (SELECT OperationKey FROM [dbo].[__RefactorLog] WHERE OperationKey = '80d0e5de-e188-465e-b83c-18f38a1cec98')
INSERT INTO [dbo].[__RefactorLog] (OperationKey) values ('80d0e5de-e188-465e-b83c-18f38a1cec98')

GO

GO
PRINT N'Update complete.';


GO

【讨论】:

谢谢你。这是一个很好的想法:“我猜你的发布配置文件设置有问题。你可能禁用了某些东西,例如'不要修改触发器'或类似的东西。”

以上是关于使用 DACPAC,当表上有触发器时,如何重命名列?的主要内容,如果未能解决你的问题,请参考以下文章

如何使用 Visual Studio 数据库项目重命名列?

如果触发器失败,插入 TABLE_A 不应该失败。如何?

如何按列索引重命名列?

php 加入两个表并同时重命名列

熊猫按位置重命名列? [复制]

如何使用dplyr重命名列,其中新列名和原始列名都是变量[duplicate]