I'm using EF code-first approach and want to change the Id
field to guid
but can't seem to get past below error.
This is my first migration:
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");
}
}
Later I realized that I need my Id
column to be GUID so I changed my model file:
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; }
}
When running Add-Migration ChangeDownloadTokenIdToGuid
it generates this file:
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");
}
}
Running this file with Update-Database
causes this error:
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.
Any ideas why this might be happening?
It was caused because it is impossible to convert previous
int
type ofId
column toGuid
type(exactly that try to performAlterColumn
method). Also, error message suggest you, that new type ofId
column can be one of type from set: int, bigint, smallint, tinyint, or decimal or numeric with a scale of 0, for them, it is possible to perform conversion fromint
type.Solution - simply drop
Id
column and then recreate it with newGuid
type, change migration that way:Even though Slava Utesinov's works, it only works on empty tables or in cases where no other tables are referring to the table you're converting. So this answer will help those ending up on this page with a more complex database setup.
Below is a utility function you can use from your migration class, which should be invoked from the Up/Down functions. The function also handles tables references the table you're trying to convert from Int to Guid. This helper function assumes the column you're converting is called 'Id', but should otherwise be rather generic.
So in your case your Up/Down functions would be: