私はEFコードファーストアプローチを使用していて、Id
フィールドをguid
に変更したいのですが、以下のエラーを回避できません。
これは私の最初の移行です:
public partial class CreateDownloadToken : DbMigration
{
public override void Up()
{
CreateTable(
"dbo.DownloadTokens",
c => new
{
Id = c.Int(nullable: false, identity: true),
FileId = c.Int(),
UserId = c.String(nullable: false, maxLength: 128),
ValidUntil = c.DateTime(nullable: false),
})
.PrimaryKey(t => t.Id)
.ForeignKey("dbo.Files", t => t.FileId)
.ForeignKey("dbo.Users", t => t.UserId, cascadeDelete: true)
.Index(t => t.FileId)
.Index(t => t.UserId);
}
public override void Down()
{
DropForeignKey("dbo.DownloadTokens", "UserId", "dbo.Users");
DropForeignKey("dbo.DownloadTokens", "FileId", "dbo.Files");
DropIndex("dbo.DownloadTokens", new[] { "UserId" });
DropIndex("dbo.DownloadTokens", new[] { "FileId" });
DropTable("dbo.DownloadTokens");
}
}
後で、Id
列をGUIDにする必要があることがわかったので、モデルファイルを変更しました。
public class DownloadToken
{
[Key, DatabaseGenerated(DatabaseGeneratedOption.Computed)]
public Guid Id { get; set; }
public int? FileId { get; set; }
[ForeignKey("FileId")]
public virtual File File { get; set; }
[Required]
public string UserId { get; set; }
[ForeignKey("UserId")]
public virtual User User { get; set; }
[Required]
public DateTime ValidUntil { get; set; }
}
Add-Migration ChangeDownloadTokenIdToGuid
を実行すると、次のファイルが生成されます。
public partial class ChangeDownloadTokenIdToGuid : DbMigration
{
public override void Up()
{
DropPrimaryKey("dbo.DownloadTokens");
AlterColumn("dbo.DownloadTokens", "Id", c => c.Guid(nullable: false));
AddPrimaryKey("dbo.DownloadTokens", "Id");
}
public override void Down()
{
DropPrimaryKey("dbo.DownloadTokens");
AlterColumn("dbo.DownloadTokens", "Id", c => c.Int(nullable: false, identity: true));
AddPrimaryKey("dbo.DownloadTokens", "Id");
}
}
このファイルをUpdate-Database
で実行すると、次のエラーが発生します。
Identity column 'Id' must be of data type int, bigint, smallint, tinyint, or decimal or numeric with a scale of 0, and constrained to be nonnullable.
なぜこれが起こっているのでしょうか?
これは、以前のint
タイプのId
列をGuid
タイプに変換することができないために発生しました(まさにAlterColumn
メソッドを実行しようとします)。また、エラーメッセージは、新しいタイプのId
列がセットからのタイプの1つである可能性があることを示唆しています:int、bigint、smallint、tinyint、または小数またはスケール付きの数値それらの0の場合、int
型からの変換を実行することが可能です。
ソリューション-Id
列を削除してから、新しいGuid
タイプで再作成し、マイグレーションをそのように変更します。
public partial class ChangeDownloadTokenIdToGuid : DbMigration
{
public override void Up()
{
DropPrimaryKey("dbo.DownloadTokens");
DropColumn("dbo.DownloadTokens", "Id");
AddColumn("dbo.DownloadTokens", "Id", c => c.Guid(nullable: false, identity: true));
AddPrimaryKey("dbo.DownloadTokens", "Id");
}
public override void Down()
{
DropPrimaryKey("dbo.DownloadTokens");
DropColumn("dbo.DownloadTokens", "Id");
AddColumn("dbo.DownloadTokens", "Id", c => c.Int(nullable: false, identity: true));
AddPrimaryKey("dbo.DownloadTokens", "Id");
}
}
追伸
DatabaseGeneratedOption.Computed
ではなくDatabaseGeneratedOption.Identity
属性を使用する理由
Slava Utesinovは機能しますが、空のテーブル、または変換中のテーブルを参照している他のテーブルがない場合にのみ機能します。したがって、この回答は、このページに到達して、より複雑なデータベースのセットアップを行う人を助けるでしょう。
以下は、Up/Down関数から呼び出す必要がある移行クラスから使用できるユーティリティ関数です。この関数は、IntからGuidに変換しようとしているテーブルを参照するテーブルも処理します。このヘルパー関数は、変換している列が「Id」と呼ばれていることを前提としていますが、それ以外はかなり汎用的である必要があります。
public void Convert(bool toGuid, string parent, params string[] children)
{
if (toGuid)
{
AddColumn($"dbo.{parent}s", "Id2", c => c.Guid(nullable: false, identity: true, defaultValueSql: "newid()"));
}
else
{
AddColumn($"dbo.{parent}s", "Id2", c => c.Int(nullable: false, identity: true));
}
foreach (var child in children)
{
DropForeignKey($"dbo.{child}s", $"{parent}_Id", $"dbo.{parent}s");
DropIndex($"dbo.{child}s", new[] { $"{parent}_Id" });
RenameColumn($"dbo.{child}s", $"{parent}_Id", $"old_{parent}_Id");
if (toGuid)
{
AddColumn($"dbo.{child}s", $"{parent}_Id", c => c.Guid());
}
else
{
AddColumn($"dbo.{child}s", $"{parent}_Id", c => c.Int());
}
Sql($"update c set {parent}_Id=p.Id2 from {child}s c inner join {parent}s p on p.Id=c.old_{parent}_Id");
DropColumn($"dbo.{child}s", $"old_{parent}_Id");
}
DropPrimaryKey($"dbo.{parent}s");
DropColumn($"dbo.{parent}s", "Id");
RenameColumn($"dbo.{parent}s", "Id2", "Id");
AddPrimaryKey($"dbo.{parent}s", "Id");
foreach (var child in children)
{
CreateIndex($"dbo.{child}s", $"{parent}_Id");
AddForeignKey($"dbo.{child}s", $"{parent}_Id", $"dbo.{parent}s", "Id");
}
}
したがって、あなたの場合、アップ/ダウン関数は次のようになります:
public override void Up()
{
Convert(true,"DownloadToken");
}
public override void Down()
{
Convert(false, "DownloadToken");
}