diff --git a/dotnet/AipsCore/Infrastructure/Persistence/Db/Migrations/20260210123114_AddedShapesAndMemberships.Designer.cs b/dotnet/AipsCore/Infrastructure/Persistence/Db/Migrations/20260210123114_AddedShapesAndMemberships.Designer.cs
new file mode 100644
index 0000000..d3069f1
--- /dev/null
+++ b/dotnet/AipsCore/Infrastructure/Persistence/Db/Migrations/20260210123114_AddedShapesAndMemberships.Designer.cs
@@ -0,0 +1,246 @@
+//
+using System;
+using AipsCore.Infrastructure.Persistence.Db;
+using Microsoft.EntityFrameworkCore;
+using Microsoft.EntityFrameworkCore.Infrastructure;
+using Microsoft.EntityFrameworkCore.Migrations;
+using Microsoft.EntityFrameworkCore.Storage.ValueConversion;
+using Npgsql.EntityFrameworkCore.PostgreSQL.Metadata;
+
+#nullable disable
+
+namespace AipsCore.Infrastructure.Persistence.Db.Migrations
+{
+ [DbContext(typeof(AipsDbContext))]
+ [Migration("20260210123114_AddedShapesAndMemberships")]
+ partial class AddedShapesAndMemberships
+ {
+ ///
+ protected override void BuildTargetModel(ModelBuilder modelBuilder)
+ {
+#pragma warning disable 612, 618
+ modelBuilder
+ .HasAnnotation("ProductVersion", "10.0.2")
+ .HasAnnotation("Relational:MaxIdentifierLength", 63);
+
+ NpgsqlModelBuilderExtensions.UseIdentityByDefaultColumns(modelBuilder);
+
+ modelBuilder.Entity("AipsCore.Infrastructure.Persistence.Shape.Shape", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("uuid");
+
+ b.Property("AuthorId")
+ .HasColumnType("uuid");
+
+ b.Property("Color")
+ .IsRequired()
+ .HasMaxLength(10)
+ .HasColumnType("character varying(10)");
+
+ b.Property("EndPositionX")
+ .HasColumnType("integer");
+
+ b.Property("EndPositionY")
+ .HasColumnType("integer");
+
+ b.Property("PositionX")
+ .HasColumnType("integer");
+
+ b.Property("PositionY")
+ .HasColumnType("integer");
+
+ b.Property("TextSize")
+ .HasColumnType("integer");
+
+ b.Property("TextValue")
+ .HasColumnType("text");
+
+ b.Property("Thickness")
+ .HasColumnType("integer");
+
+ b.Property("Type")
+ .HasColumnType("integer");
+
+ b.Property("WhiteboardId")
+ .HasColumnType("uuid");
+
+ b.HasKey("Id");
+
+ b.HasIndex("AuthorId");
+
+ b.HasIndex("WhiteboardId");
+
+ b.ToTable("Shapes");
+ });
+
+ modelBuilder.Entity("AipsCore.Infrastructure.Persistence.User.User", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("uuid");
+
+ b.Property("CreatedAt")
+ .HasColumnType("timestamp with time zone");
+
+ b.Property("DeletedAt")
+ .HasColumnType("timestamp with time zone");
+
+ b.Property("Email")
+ .IsRequired()
+ .HasMaxLength(255)
+ .HasColumnType("character varying(255)");
+
+ b.Property("Username")
+ .IsRequired()
+ .HasMaxLength(255)
+ .HasColumnType("character varying(255)");
+
+ b.HasKey("Id");
+
+ b.ToTable("Users");
+ });
+
+ modelBuilder.Entity("AipsCore.Infrastructure.Persistence.Whiteboard.Whiteboard", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("uuid");
+
+ b.Property("Code")
+ .IsRequired()
+ .HasMaxLength(8)
+ .HasColumnType("character varying(8)");
+
+ b.Property("CreatedAt")
+ .HasColumnType("timestamp with time zone");
+
+ b.Property("DeletedAt")
+ .HasColumnType("timestamp with time zone");
+
+ b.Property("JoinPolicy")
+ .HasColumnType("integer");
+
+ b.Property("MaxParticipants")
+ .HasColumnType("integer");
+
+ b.Property("OwnerId")
+ .HasColumnType("uuid");
+
+ b.Property("State")
+ .HasColumnType("integer");
+
+ b.Property("Title")
+ .IsRequired()
+ .HasMaxLength(32)
+ .HasColumnType("character varying(32)");
+
+ b.HasKey("Id");
+
+ b.HasIndex("OwnerId");
+
+ b.ToTable("Whiteboards");
+ });
+
+ modelBuilder.Entity("AipsCore.Infrastructure.Persistence.WhiteboardMembership.WhiteboardMembership", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("uuid");
+
+ b.Property("CanJoin")
+ .HasColumnType("boolean");
+
+ b.Property("EditingEnabled")
+ .HasColumnType("boolean");
+
+ b.Property("IsBanned")
+ .HasColumnType("boolean");
+
+ b.Property("LastInteractedAt")
+ .HasColumnType("timestamp with time zone");
+
+ b.Property("UserId")
+ .HasColumnType("uuid");
+
+ b.Property("WhiteboardId")
+ .HasColumnType("uuid");
+
+ b.HasKey("Id");
+
+ b.HasIndex("UserId");
+
+ b.HasIndex("WhiteboardId");
+
+ b.ToTable("WhiteboardMemberships");
+ });
+
+ modelBuilder.Entity("AipsCore.Infrastructure.Persistence.Shape.Shape", b =>
+ {
+ b.HasOne("AipsCore.Infrastructure.Persistence.User.User", "Author")
+ .WithMany("Shapes")
+ .HasForeignKey("AuthorId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired();
+
+ b.HasOne("AipsCore.Infrastructure.Persistence.Whiteboard.Whiteboard", "Whiteboard")
+ .WithMany("Shapes")
+ .HasForeignKey("WhiteboardId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired();
+
+ b.Navigation("Author");
+
+ b.Navigation("Whiteboard");
+ });
+
+ modelBuilder.Entity("AipsCore.Infrastructure.Persistence.Whiteboard.Whiteboard", b =>
+ {
+ b.HasOne("AipsCore.Infrastructure.Persistence.User.User", "Owner")
+ .WithMany("Whiteboards")
+ .HasForeignKey("OwnerId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired();
+
+ b.Navigation("Owner");
+ });
+
+ modelBuilder.Entity("AipsCore.Infrastructure.Persistence.WhiteboardMembership.WhiteboardMembership", b =>
+ {
+ b.HasOne("AipsCore.Infrastructure.Persistence.User.User", "User")
+ .WithMany("Memberships")
+ .HasForeignKey("UserId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired();
+
+ b.HasOne("AipsCore.Infrastructure.Persistence.Whiteboard.Whiteboard", "Whiteboard")
+ .WithMany("Memberships")
+ .HasForeignKey("WhiteboardId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired();
+
+ b.Navigation("User");
+
+ b.Navigation("Whiteboard");
+ });
+
+ modelBuilder.Entity("AipsCore.Infrastructure.Persistence.User.User", b =>
+ {
+ b.Navigation("Memberships");
+
+ b.Navigation("Shapes");
+
+ b.Navigation("Whiteboards");
+ });
+
+ modelBuilder.Entity("AipsCore.Infrastructure.Persistence.Whiteboard.Whiteboard", b =>
+ {
+ b.Navigation("Memberships");
+
+ b.Navigation("Shapes");
+ });
+#pragma warning restore 612, 618
+ }
+ }
+}
diff --git a/dotnet/AipsCore/Infrastructure/Persistence/Db/Migrations/20260210123114_AddedShapesAndMemberships.cs b/dotnet/AipsCore/Infrastructure/Persistence/Db/Migrations/20260210123114_AddedShapesAndMemberships.cs
new file mode 100644
index 0000000..91a9dc8
--- /dev/null
+++ b/dotnet/AipsCore/Infrastructure/Persistence/Db/Migrations/20260210123114_AddedShapesAndMemberships.cs
@@ -0,0 +1,214 @@
+using System;
+using Microsoft.EntityFrameworkCore.Migrations;
+
+#nullable disable
+
+namespace AipsCore.Infrastructure.Persistence.Db.Migrations
+{
+ ///
+ public partial class AddedShapesAndMemberships : Migration
+ {
+ ///
+ protected override void Up(MigrationBuilder migrationBuilder)
+ {
+ migrationBuilder.RenameColumn(
+ name: "WhiteboardOwnerId",
+ table: "Whiteboards",
+ newName: "OwnerId");
+
+ migrationBuilder.AddColumn(
+ name: "CreatedAt",
+ table: "Whiteboards",
+ type: "timestamp with time zone",
+ nullable: false,
+ defaultValue: new DateTime(1, 1, 1, 0, 0, 0, 0, DateTimeKind.Unspecified));
+
+ migrationBuilder.AddColumn(
+ name: "DeletedAt",
+ table: "Whiteboards",
+ type: "timestamp with time zone",
+ nullable: true);
+
+ migrationBuilder.AddColumn(
+ name: "JoinPolicy",
+ table: "Whiteboards",
+ type: "integer",
+ nullable: false,
+ defaultValue: 0);
+
+ migrationBuilder.AddColumn(
+ name: "MaxParticipants",
+ table: "Whiteboards",
+ type: "integer",
+ nullable: false,
+ defaultValue: 0);
+
+ migrationBuilder.AddColumn(
+ name: "State",
+ table: "Whiteboards",
+ type: "integer",
+ nullable: false,
+ defaultValue: 0);
+
+ migrationBuilder.AddColumn(
+ name: "CreatedAt",
+ table: "Users",
+ type: "timestamp with time zone",
+ nullable: false,
+ defaultValue: new DateTime(1, 1, 1, 0, 0, 0, 0, DateTimeKind.Unspecified));
+
+ migrationBuilder.AddColumn(
+ name: "DeletedAt",
+ table: "Users",
+ type: "timestamp with time zone",
+ nullable: true);
+
+ migrationBuilder.CreateTable(
+ name: "Shapes",
+ columns: table => new
+ {
+ Id = table.Column(type: "uuid", nullable: false),
+ WhiteboardId = table.Column(type: "uuid", nullable: false),
+ AuthorId = table.Column(type: "uuid", nullable: false),
+ Type = table.Column(type: "integer", nullable: false),
+ PositionX = table.Column(type: "integer", nullable: false),
+ PositionY = table.Column(type: "integer", nullable: false),
+ Color = table.Column(type: "character varying(10)", maxLength: 10, nullable: false),
+ EndPositionX = table.Column(type: "integer", nullable: true),
+ EndPositionY = table.Column(type: "integer", nullable: true),
+ Thickness = table.Column(type: "integer", nullable: true),
+ TextValue = table.Column(type: "text", nullable: true),
+ TextSize = table.Column(type: "integer", nullable: true)
+ },
+ constraints: table =>
+ {
+ table.PrimaryKey("PK_Shapes", x => x.Id);
+ table.ForeignKey(
+ name: "FK_Shapes_Users_AuthorId",
+ column: x => x.AuthorId,
+ principalTable: "Users",
+ principalColumn: "Id",
+ onDelete: ReferentialAction.Cascade);
+ table.ForeignKey(
+ name: "FK_Shapes_Whiteboards_WhiteboardId",
+ column: x => x.WhiteboardId,
+ principalTable: "Whiteboards",
+ principalColumn: "Id",
+ onDelete: ReferentialAction.Cascade);
+ });
+
+ migrationBuilder.CreateTable(
+ name: "WhiteboardMemberships",
+ columns: table => new
+ {
+ Id = table.Column(type: "uuid", nullable: false),
+ WhiteboardId = table.Column(type: "uuid", nullable: false),
+ UserId = table.Column(type: "uuid", nullable: false),
+ IsBanned = table.Column(type: "boolean", nullable: false),
+ EditingEnabled = table.Column(type: "boolean", nullable: false),
+ CanJoin = table.Column(type: "boolean", nullable: false),
+ LastInteractedAt = table.Column(type: "timestamp with time zone", nullable: false)
+ },
+ constraints: table =>
+ {
+ table.PrimaryKey("PK_WhiteboardMemberships", x => x.Id);
+ table.ForeignKey(
+ name: "FK_WhiteboardMemberships_Users_UserId",
+ column: x => x.UserId,
+ principalTable: "Users",
+ principalColumn: "Id",
+ onDelete: ReferentialAction.Cascade);
+ table.ForeignKey(
+ name: "FK_WhiteboardMemberships_Whiteboards_WhiteboardId",
+ column: x => x.WhiteboardId,
+ principalTable: "Whiteboards",
+ principalColumn: "Id",
+ onDelete: ReferentialAction.Cascade);
+ });
+
+ migrationBuilder.CreateIndex(
+ name: "IX_Whiteboards_OwnerId",
+ table: "Whiteboards",
+ column: "OwnerId");
+
+ migrationBuilder.CreateIndex(
+ name: "IX_Shapes_AuthorId",
+ table: "Shapes",
+ column: "AuthorId");
+
+ migrationBuilder.CreateIndex(
+ name: "IX_Shapes_WhiteboardId",
+ table: "Shapes",
+ column: "WhiteboardId");
+
+ migrationBuilder.CreateIndex(
+ name: "IX_WhiteboardMemberships_UserId",
+ table: "WhiteboardMemberships",
+ column: "UserId");
+
+ migrationBuilder.CreateIndex(
+ name: "IX_WhiteboardMemberships_WhiteboardId",
+ table: "WhiteboardMemberships",
+ column: "WhiteboardId");
+
+ migrationBuilder.AddForeignKey(
+ name: "FK_Whiteboards_Users_OwnerId",
+ table: "Whiteboards",
+ column: "OwnerId",
+ principalTable: "Users",
+ principalColumn: "Id",
+ onDelete: ReferentialAction.Cascade);
+ }
+
+ ///
+ protected override void Down(MigrationBuilder migrationBuilder)
+ {
+ migrationBuilder.DropForeignKey(
+ name: "FK_Whiteboards_Users_OwnerId",
+ table: "Whiteboards");
+
+ migrationBuilder.DropTable(
+ name: "Shapes");
+
+ migrationBuilder.DropTable(
+ name: "WhiteboardMemberships");
+
+ migrationBuilder.DropIndex(
+ name: "IX_Whiteboards_OwnerId",
+ table: "Whiteboards");
+
+ migrationBuilder.DropColumn(
+ name: "CreatedAt",
+ table: "Whiteboards");
+
+ migrationBuilder.DropColumn(
+ name: "DeletedAt",
+ table: "Whiteboards");
+
+ migrationBuilder.DropColumn(
+ name: "JoinPolicy",
+ table: "Whiteboards");
+
+ migrationBuilder.DropColumn(
+ name: "MaxParticipants",
+ table: "Whiteboards");
+
+ migrationBuilder.DropColumn(
+ name: "State",
+ table: "Whiteboards");
+
+ migrationBuilder.DropColumn(
+ name: "CreatedAt",
+ table: "Users");
+
+ migrationBuilder.DropColumn(
+ name: "DeletedAt",
+ table: "Users");
+
+ migrationBuilder.RenameColumn(
+ name: "OwnerId",
+ table: "Whiteboards",
+ newName: "WhiteboardOwnerId");
+ }
+ }
+}
diff --git a/dotnet/AipsCore/Infrastructure/Persistence/Db/Migrations/AipsDbContextModelSnapshot.cs b/dotnet/AipsCore/Infrastructure/Persistence/Db/Migrations/AipsDbContextModelSnapshot.cs
index f8c68d1..e9468c5 100644
--- a/dotnet/AipsCore/Infrastructure/Persistence/Db/Migrations/AipsDbContextModelSnapshot.cs
+++ b/dotnet/AipsCore/Infrastructure/Persistence/Db/Migrations/AipsDbContextModelSnapshot.cs
@@ -22,12 +22,68 @@ namespace AipsCore.Infrastructure.Persistence.Db.Migrations
NpgsqlModelBuilderExtensions.UseIdentityByDefaultColumns(modelBuilder);
+ modelBuilder.Entity("AipsCore.Infrastructure.Persistence.Shape.Shape", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("uuid");
+
+ b.Property("AuthorId")
+ .HasColumnType("uuid");
+
+ b.Property("Color")
+ .IsRequired()
+ .HasMaxLength(10)
+ .HasColumnType("character varying(10)");
+
+ b.Property("EndPositionX")
+ .HasColumnType("integer");
+
+ b.Property("EndPositionY")
+ .HasColumnType("integer");
+
+ b.Property("PositionX")
+ .HasColumnType("integer");
+
+ b.Property("PositionY")
+ .HasColumnType("integer");
+
+ b.Property("TextSize")
+ .HasColumnType("integer");
+
+ b.Property("TextValue")
+ .HasColumnType("text");
+
+ b.Property("Thickness")
+ .HasColumnType("integer");
+
+ b.Property("Type")
+ .HasColumnType("integer");
+
+ b.Property("WhiteboardId")
+ .HasColumnType("uuid");
+
+ b.HasKey("Id");
+
+ b.HasIndex("AuthorId");
+
+ b.HasIndex("WhiteboardId");
+
+ b.ToTable("Shapes");
+ });
+
modelBuilder.Entity("AipsCore.Infrastructure.Persistence.User.User", b =>
{
b.Property("Id")
.ValueGeneratedOnAdd()
.HasColumnType("uuid");
+ b.Property("CreatedAt")
+ .HasColumnType("timestamp with time zone");
+
+ b.Property("DeletedAt")
+ .HasColumnType("timestamp with time zone");
+
b.Property("Email")
.IsRequired()
.HasMaxLength(255)
@@ -54,17 +110,132 @@ namespace AipsCore.Infrastructure.Persistence.Db.Migrations
.HasMaxLength(8)
.HasColumnType("character varying(8)");
+ b.Property("CreatedAt")
+ .HasColumnType("timestamp with time zone");
+
+ b.Property("DeletedAt")
+ .HasColumnType("timestamp with time zone");
+
+ b.Property("JoinPolicy")
+ .HasColumnType("integer");
+
+ b.Property("MaxParticipants")
+ .HasColumnType("integer");
+
+ b.Property("OwnerId")
+ .HasColumnType("uuid");
+
+ b.Property("State")
+ .HasColumnType("integer");
+
b.Property("Title")
.IsRequired()
.HasMaxLength(32)
.HasColumnType("character varying(32)");
- b.Property("WhiteboardOwnerId")
+ b.HasKey("Id");
+
+ b.HasIndex("OwnerId");
+
+ b.ToTable("Whiteboards");
+ });
+
+ modelBuilder.Entity("AipsCore.Infrastructure.Persistence.WhiteboardMembership.WhiteboardMembership", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("uuid");
+
+ b.Property("CanJoin")
+ .HasColumnType("boolean");
+
+ b.Property("EditingEnabled")
+ .HasColumnType("boolean");
+
+ b.Property("IsBanned")
+ .HasColumnType("boolean");
+
+ b.Property("LastInteractedAt")
+ .HasColumnType("timestamp with time zone");
+
+ b.Property("UserId")
+ .HasColumnType("uuid");
+
+ b.Property("WhiteboardId")
.HasColumnType("uuid");
b.HasKey("Id");
- b.ToTable("Whiteboards");
+ b.HasIndex("UserId");
+
+ b.HasIndex("WhiteboardId");
+
+ b.ToTable("WhiteboardMemberships");
+ });
+
+ modelBuilder.Entity("AipsCore.Infrastructure.Persistence.Shape.Shape", b =>
+ {
+ b.HasOne("AipsCore.Infrastructure.Persistence.User.User", "Author")
+ .WithMany("Shapes")
+ .HasForeignKey("AuthorId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired();
+
+ b.HasOne("AipsCore.Infrastructure.Persistence.Whiteboard.Whiteboard", "Whiteboard")
+ .WithMany("Shapes")
+ .HasForeignKey("WhiteboardId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired();
+
+ b.Navigation("Author");
+
+ b.Navigation("Whiteboard");
+ });
+
+ modelBuilder.Entity("AipsCore.Infrastructure.Persistence.Whiteboard.Whiteboard", b =>
+ {
+ b.HasOne("AipsCore.Infrastructure.Persistence.User.User", "Owner")
+ .WithMany("Whiteboards")
+ .HasForeignKey("OwnerId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired();
+
+ b.Navigation("Owner");
+ });
+
+ modelBuilder.Entity("AipsCore.Infrastructure.Persistence.WhiteboardMembership.WhiteboardMembership", b =>
+ {
+ b.HasOne("AipsCore.Infrastructure.Persistence.User.User", "User")
+ .WithMany("Memberships")
+ .HasForeignKey("UserId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired();
+
+ b.HasOne("AipsCore.Infrastructure.Persistence.Whiteboard.Whiteboard", "Whiteboard")
+ .WithMany("Memberships")
+ .HasForeignKey("WhiteboardId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired();
+
+ b.Navigation("User");
+
+ b.Navigation("Whiteboard");
+ });
+
+ modelBuilder.Entity("AipsCore.Infrastructure.Persistence.User.User", b =>
+ {
+ b.Navigation("Memberships");
+
+ b.Navigation("Shapes");
+
+ b.Navigation("Whiteboards");
+ });
+
+ modelBuilder.Entity("AipsCore.Infrastructure.Persistence.Whiteboard.Whiteboard", b =>
+ {
+ b.Navigation("Memberships");
+
+ b.Navigation("Shapes");
});
#pragma warning restore 612, 618
}
diff --git a/dotnet/AipsCore/Infrastructure/Persistence/User/User.cs b/dotnet/AipsCore/Infrastructure/Persistence/User/User.cs
index 1e5c9d0..3d6f3a2 100644
--- a/dotnet/AipsCore/Infrastructure/Persistence/User/User.cs
+++ b/dotnet/AipsCore/Infrastructure/Persistence/User/User.cs
@@ -19,4 +19,10 @@ public class User
public DateTime CreatedAt { get; set; }
public DateTime? DeletedAt { get; set; }
+
+ public ICollection Shapes { get; set; } = new List();
+
+ public ICollection Whiteboards { get; set; } = new List();
+
+ public ICollection Memberships { get; set; } = new List();
}
\ No newline at end of file
diff --git a/dotnet/AipsCore/Infrastructure/Persistence/Whiteboard/Whiteboard.cs b/dotnet/AipsCore/Infrastructure/Persistence/Whiteboard/Whiteboard.cs
index d44b0fd..f0f0f5b 100644
--- a/dotnet/AipsCore/Infrastructure/Persistence/Whiteboard/Whiteboard.cs
+++ b/dotnet/AipsCore/Infrastructure/Persistence/Whiteboard/Whiteboard.cs
@@ -32,4 +32,8 @@ public class Whiteboard
public WhiteboardJoinPolicy JoinPolicy { get; set; }
public WhiteboardState State { get; set; }
+
+ public ICollection Shapes { get; set; } = new List();
+
+ public ICollection Memberships { get; set; } = new List();
}
\ No newline at end of file