diff --git a/Common/Migrations/20240327034706_Petrainer998DR.Designer.cs b/Common/Migrations/20240327034706_Petrainer998DR.Designer.cs
new file mode 100644
index 00000000..c8211473
--- /dev/null
+++ b/Common/Migrations/20240327034706_Petrainer998DR.Designer.cs
@@ -0,0 +1,775 @@
+//
+using System;
+using Microsoft.EntityFrameworkCore;
+using Microsoft.EntityFrameworkCore.Infrastructure;
+using Microsoft.EntityFrameworkCore.Migrations;
+using Microsoft.EntityFrameworkCore.Storage.ValueConversion;
+using Npgsql.EntityFrameworkCore.PostgreSQL.Metadata;
+using OpenShock.Common.OpenShockDb;
+
+#nullable disable
+
+namespace OpenShock.Common.Migrations
+{
+ [DbContext(typeof(OpenShockContext))]
+ [Migration("20240327034706_Petrainer998DR")]
+ partial class Petrainer998DR
+ {
+ ///
+ protected override void BuildTargetModel(ModelBuilder modelBuilder)
+ {
+#pragma warning disable 612, 618
+ modelBuilder
+ .HasAnnotation("Npgsql:CollationDefinition:public.ndcoll", "und-u-ks-level2,und-u-ks-level2,icu,False")
+ .HasAnnotation("ProductVersion", "8.0.3")
+ .HasAnnotation("Relational:MaxIdentifierLength", 63);
+
+ NpgsqlModelBuilderExtensions.HasPostgresEnum(modelBuilder, "control_type", new[] { "sound", "vibrate", "shock", "stop" });
+ NpgsqlModelBuilderExtensions.HasPostgresEnum(modelBuilder, "ota_update_status", new[] { "started", "running", "finished", "error", "timeout" });
+ NpgsqlModelBuilderExtensions.HasPostgresEnum(modelBuilder, "password_encryption_type", new[] { "pbkdf2", "bcrypt_enhanced" });
+ NpgsqlModelBuilderExtensions.HasPostgresEnum(modelBuilder, "permission_type", new[] { "shockers.use" });
+ NpgsqlModelBuilderExtensions.HasPostgresEnum(modelBuilder, "rank_type", new[] { "user", "support", "staff", "admin", "system" });
+ NpgsqlModelBuilderExtensions.HasPostgresEnum(modelBuilder, "shocker_model_type", new[] { "caiXianlin", "petTrainer", "petrainer998DR" });
+ NpgsqlModelBuilderExtensions.UseIdentityByDefaultColumns(modelBuilder);
+
+ modelBuilder.Entity("OpenShock.Common.OpenShockDb.ApiToken", b =>
+ {
+ b.Property("Id")
+ .HasColumnType("uuid")
+ .HasColumnName("id");
+
+ b.Property("CreatedByIp")
+ .IsRequired()
+ .HasColumnType("character varying")
+ .HasColumnName("created_by_ip");
+
+ b.Property("CreatedOn")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("created_on")
+ .HasDefaultValueSql("CURRENT_TIMESTAMP");
+
+ b.Property("Name")
+ .IsRequired()
+ .HasMaxLength(64)
+ .HasColumnType("character varying(64)")
+ .HasColumnName("name");
+
+ b.Property("Permissions")
+ .IsRequired()
+ .HasColumnType("permission_type[]")
+ .HasColumnName("permissions");
+
+ b.Property("Token")
+ .IsRequired()
+ .HasMaxLength(256)
+ .HasColumnType("character varying(256)")
+ .HasColumnName("token");
+
+ b.Property("UserId")
+ .HasColumnType("uuid")
+ .HasColumnName("user_id");
+
+ b.Property("ValidUntil")
+ .HasColumnType("date")
+ .HasColumnName("valid_until");
+
+ b.HasKey("Id")
+ .HasName("api_tokens_pkey");
+
+ b.HasIndex("UserId");
+
+ b.ToTable("api_tokens", (string)null);
+ });
+
+ modelBuilder.Entity("OpenShock.Common.OpenShockDb.Device", b =>
+ {
+ b.Property("Id")
+ .HasColumnType("uuid")
+ .HasColumnName("id");
+
+ b.Property("CreatedOn")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("created_on")
+ .HasDefaultValueSql("CURRENT_TIMESTAMP");
+
+ b.Property("Name")
+ .IsRequired()
+ .HasColumnType("character varying")
+ .HasColumnName("name");
+
+ b.Property("Owner")
+ .HasColumnType("uuid")
+ .HasColumnName("owner");
+
+ b.Property("Token")
+ .IsRequired()
+ .HasMaxLength(256)
+ .HasColumnType("character varying(256)")
+ .HasColumnName("token");
+
+ b.HasKey("Id")
+ .HasName("devices_pkey");
+
+ b.HasIndex("Owner");
+
+ b.ToTable("devices", (string)null);
+ });
+
+ modelBuilder.Entity("OpenShock.Common.OpenShockDb.DeviceOtaUpdate", b =>
+ {
+ b.Property("Device")
+ .HasColumnType("uuid")
+ .HasColumnName("device");
+
+ b.Property("UpdateId")
+ .HasColumnType("integer")
+ .HasColumnName("update_id");
+
+ b.Property("CreatedOn")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("created_on")
+ .HasDefaultValueSql("CURRENT_TIMESTAMP");
+
+ b.Property("Message")
+ .HasColumnType("character varying")
+ .HasColumnName("message");
+
+ b.Property("Status")
+ .HasColumnType("ota_update_status")
+ .HasColumnName("status");
+
+ b.Property("Version")
+ .IsRequired()
+ .HasColumnType("character varying")
+ .HasColumnName("version");
+
+ b.HasKey("Device", "UpdateId")
+ .HasName("device_ota_updates_pkey");
+
+ b.HasIndex(new[] { "CreatedOn" }, "device_ota_updates_created_on_idx")
+ .HasAnnotation("Npgsql:StorageParameter:deduplicate_items", "true");
+
+ b.ToTable("device_ota_updates", (string)null);
+ });
+
+ modelBuilder.Entity("OpenShock.Common.OpenShockDb.PasswordReset", b =>
+ {
+ b.Property("Id")
+ .HasColumnType("uuid")
+ .HasColumnName("id");
+
+ b.Property("CreatedOn")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("created_on")
+ .HasDefaultValueSql("CURRENT_TIMESTAMP");
+
+ b.Property("Secret")
+ .IsRequired()
+ .HasColumnType("character varying")
+ .HasColumnName("secret");
+
+ b.Property("UsedOn")
+ .HasColumnType("time with time zone")
+ .HasColumnName("used_on");
+
+ b.Property("UserId")
+ .HasColumnType("uuid")
+ .HasColumnName("user_id");
+
+ b.HasKey("Id")
+ .HasName("password_resets_pkey");
+
+ b.HasIndex("UserId");
+
+ b.ToTable("password_resets", (string)null);
+ });
+
+ modelBuilder.Entity("OpenShock.Common.OpenShockDb.Shocker", b =>
+ {
+ b.Property("Id")
+ .HasColumnType("uuid")
+ .HasColumnName("id");
+
+ b.Property("CreatedOn")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("created_on")
+ .HasDefaultValueSql("CURRENT_TIMESTAMP");
+
+ b.Property("Device")
+ .HasColumnType("uuid")
+ .HasColumnName("device");
+
+ b.Property("Model")
+ .HasColumnType("shocker_model_type")
+ .HasColumnName("model");
+
+ b.Property("Name")
+ .IsRequired()
+ .HasMaxLength(64)
+ .HasColumnType("character varying(64)")
+ .HasColumnName("name");
+
+ b.Property("Paused")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("boolean")
+ .HasDefaultValue(false)
+ .HasColumnName("paused");
+
+ b.Property("RfId")
+ .HasColumnType("integer")
+ .HasColumnName("rf_id");
+
+ b.HasKey("Id")
+ .HasName("shockers_pkey");
+
+ b.HasIndex("Device");
+
+ b.ToTable("shockers", (string)null);
+ });
+
+ modelBuilder.Entity("OpenShock.Common.OpenShockDb.ShockerControlLog", b =>
+ {
+ b.Property("Id")
+ .HasColumnType("uuid")
+ .HasColumnName("id");
+
+ b.Property("ControlledBy")
+ .HasColumnType("uuid")
+ .HasColumnName("controlled_by");
+
+ b.Property("CreatedOn")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("created_on")
+ .HasDefaultValueSql("CURRENT_TIMESTAMP");
+
+ b.Property("CustomName")
+ .HasColumnType("character varying")
+ .HasColumnName("custom_name");
+
+ b.Property("Duration")
+ .HasColumnType("bigint")
+ .HasColumnName("duration");
+
+ b.Property("Intensity")
+ .HasColumnType("smallint")
+ .HasColumnName("intensity");
+
+ b.Property("LiveControl")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("boolean")
+ .HasDefaultValue(false)
+ .HasColumnName("live_control");
+
+ b.Property("ShockerId")
+ .HasColumnType("uuid")
+ .HasColumnName("shocker_id");
+
+ b.Property("Type")
+ .HasColumnType("control_type")
+ .HasColumnName("type");
+
+ b.HasKey("Id")
+ .HasName("shocker_control_logs_pkey");
+
+ b.HasIndex("ControlledBy");
+
+ b.HasIndex("ShockerId");
+
+ b.ToTable("shocker_control_logs", (string)null);
+ });
+
+ modelBuilder.Entity("OpenShock.Common.OpenShockDb.ShockerShare", b =>
+ {
+ b.Property("ShockerId")
+ .HasColumnType("uuid")
+ .HasColumnName("shocker_id");
+
+ b.Property("SharedWith")
+ .HasColumnType("uuid")
+ .HasColumnName("shared_with");
+
+ b.Property("CreatedOn")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("created_on")
+ .HasDefaultValueSql("CURRENT_TIMESTAMP");
+
+ b.Property("LimitDuration")
+ .HasColumnType("integer")
+ .HasColumnName("limit_duration");
+
+ b.Property("LimitIntensity")
+ .HasColumnType("smallint")
+ .HasColumnName("limit_intensity");
+
+ b.Property("Paused")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("boolean")
+ .HasDefaultValue(false)
+ .HasColumnName("paused");
+
+ b.Property("PermLive")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("boolean")
+ .HasDefaultValue(true)
+ .HasColumnName("perm_live");
+
+ b.Property("PermShock")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("boolean")
+ .HasDefaultValue(true)
+ .HasColumnName("perm_shock");
+
+ b.Property("PermSound")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("boolean")
+ .HasDefaultValue(true)
+ .HasColumnName("perm_sound");
+
+ b.Property("PermVibrate")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("boolean")
+ .HasDefaultValue(true)
+ .HasColumnName("perm_vibrate");
+
+ b.HasKey("ShockerId", "SharedWith")
+ .HasName("shocker_shares_pkey");
+
+ b.HasIndex("SharedWith");
+
+ b.ToTable("shocker_shares", (string)null);
+ });
+
+ modelBuilder.Entity("OpenShock.Common.OpenShockDb.ShockerShareCode", b =>
+ {
+ b.Property("Id")
+ .HasColumnType("uuid")
+ .HasColumnName("id");
+
+ b.Property("CreatedOn")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("created_on")
+ .HasDefaultValueSql("CURRENT_TIMESTAMP");
+
+ b.Property("LimitDuration")
+ .HasColumnType("integer")
+ .HasColumnName("limit_duration");
+
+ b.Property("LimitIntensity")
+ .HasColumnType("smallint")
+ .HasColumnName("limit_intensity");
+
+ b.Property("PermShock")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("boolean")
+ .HasDefaultValue(true)
+ .HasColumnName("perm_shock");
+
+ b.Property("PermSound")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("boolean")
+ .HasDefaultValue(true)
+ .HasColumnName("perm_sound");
+
+ b.Property("PermVibrate")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("boolean")
+ .HasDefaultValue(true)
+ .HasColumnName("perm_vibrate");
+
+ b.Property("ShockerId")
+ .HasColumnType("uuid")
+ .HasColumnName("shocker_id");
+
+ b.HasKey("Id")
+ .HasName("shocker_share_codes_pkey");
+
+ b.HasIndex("ShockerId");
+
+ b.ToTable("shocker_share_codes", (string)null);
+ });
+
+ modelBuilder.Entity("OpenShock.Common.OpenShockDb.ShockerSharesLink", b =>
+ {
+ b.Property("Id")
+ .HasColumnType("uuid")
+ .HasColumnName("id");
+
+ b.Property("CreatedOn")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("created_on")
+ .HasDefaultValueSql("CURRENT_TIMESTAMP");
+
+ b.Property("ExpiresOn")
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("expires_on");
+
+ b.Property("Name")
+ .IsRequired()
+ .HasColumnType("character varying")
+ .HasColumnName("name");
+
+ b.Property("OwnerId")
+ .HasColumnType("uuid")
+ .HasColumnName("owner_id");
+
+ b.HasKey("Id")
+ .HasName("shocker_shares_links_pkey");
+
+ b.HasIndex("OwnerId");
+
+ b.ToTable("shocker_shares_links", (string)null);
+ });
+
+ modelBuilder.Entity("OpenShock.Common.OpenShockDb.ShockerSharesLinksShocker", b =>
+ {
+ b.Property("ShareLinkId")
+ .HasColumnType("uuid")
+ .HasColumnName("share_link_id");
+
+ b.Property("ShockerId")
+ .HasColumnType("uuid")
+ .HasColumnName("shocker_id");
+
+ b.Property("Cooldown")
+ .HasColumnType("integer")
+ .HasColumnName("cooldown");
+
+ b.Property("LimitDuration")
+ .HasColumnType("integer")
+ .HasColumnName("limit_duration");
+
+ b.Property("LimitIntensity")
+ .HasColumnType("smallint")
+ .HasColumnName("limit_intensity");
+
+ b.Property("Paused")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("boolean")
+ .HasDefaultValue(false)
+ .HasColumnName("paused");
+
+ b.Property("PermLive")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("boolean")
+ .HasDefaultValue(true)
+ .HasColumnName("perm_live");
+
+ b.Property("PermShock")
+ .HasColumnType("boolean")
+ .HasColumnName("perm_shock");
+
+ b.Property("PermSound")
+ .HasColumnType("boolean")
+ .HasColumnName("perm_sound");
+
+ b.Property("PermVibrate")
+ .HasColumnType("boolean")
+ .HasColumnName("perm_vibrate");
+
+ b.HasKey("ShareLinkId", "ShockerId")
+ .HasName("shocker_shares_links_shockers_pkey");
+
+ b.HasIndex("ShockerId");
+
+ b.ToTable("shocker_shares_links_shockers", (string)null);
+ });
+
+ modelBuilder.Entity("OpenShock.Common.OpenShockDb.User", b =>
+ {
+ b.Property("Id")
+ .HasColumnType("uuid")
+ .HasColumnName("id");
+
+ b.Property("CreatedAt")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("created_at")
+ .HasDefaultValueSql("CURRENT_TIMESTAMP");
+
+ b.Property("Email")
+ .IsRequired()
+ .HasColumnType("character varying")
+ .HasColumnName("email");
+
+ b.Property("EmailActived")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("boolean")
+ .HasDefaultValue(false)
+ .HasColumnName("email_actived");
+
+ b.Property("Name")
+ .IsRequired()
+ .HasColumnType("character varying")
+ .HasColumnName("name")
+ .UseCollation("ndcoll");
+
+ b.Property("PasswordHash")
+ .IsRequired()
+ .HasColumnType("character varying")
+ .HasColumnName("password_hash");
+
+ b.Property("Rank")
+ .HasColumnType("rank_type")
+ .HasColumnName("rank");
+
+ b.HasKey("Id")
+ .HasName("users_pkey");
+
+ b.HasIndex(new[] { "Email" }, "email")
+ .IsUnique();
+
+ b.HasIndex(new[] { "Email" }, "idx_email");
+
+ b.HasIndex(new[] { "Name" }, "idx_name");
+
+ NpgsqlIndexBuilderExtensions.UseCollation(b.HasIndex(new[] { "Name" }, "idx_name"), new[] { "ndcoll" });
+
+ b.HasIndex(new[] { "Name" }, "username")
+ .IsUnique();
+
+ b.ToTable("users", (string)null);
+ });
+
+ modelBuilder.Entity("OpenShock.Common.OpenShockDb.UsersActivation", b =>
+ {
+ b.Property("Id")
+ .HasColumnType("uuid")
+ .HasColumnName("id");
+
+ b.Property("CreatedOn")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("created_on")
+ .HasDefaultValueSql("CURRENT_TIMESTAMP");
+
+ b.Property("Secret")
+ .IsRequired()
+ .HasColumnType("character varying")
+ .HasColumnName("secret");
+
+ b.Property("UsedOn")
+ .HasColumnType("timestamp with time zone")
+ .HasColumnName("used_on");
+
+ b.Property("UserId")
+ .HasColumnType("uuid")
+ .HasColumnName("user_id");
+
+ b.HasKey("Id")
+ .HasName("users_activation_pkey");
+
+ b.HasIndex("UserId");
+
+ b.ToTable("users_activation", (string)null);
+ });
+
+ modelBuilder.Entity("OpenShock.Common.OpenShockDb.ApiToken", b =>
+ {
+ b.HasOne("OpenShock.Common.OpenShockDb.User", "User")
+ .WithMany("ApiTokens")
+ .HasForeignKey("UserId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired()
+ .HasConstraintName("fk_user_id");
+
+ b.Navigation("User");
+ });
+
+ modelBuilder.Entity("OpenShock.Common.OpenShockDb.Device", b =>
+ {
+ b.HasOne("OpenShock.Common.OpenShockDb.User", "OwnerNavigation")
+ .WithMany("Devices")
+ .HasForeignKey("Owner")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired()
+ .HasConstraintName("owner_user_id");
+
+ b.Navigation("OwnerNavigation");
+ });
+
+ modelBuilder.Entity("OpenShock.Common.OpenShockDb.DeviceOtaUpdate", b =>
+ {
+ b.HasOne("OpenShock.Common.OpenShockDb.Device", "DeviceNavigation")
+ .WithMany("DeviceOtaUpdates")
+ .HasForeignKey("Device")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired()
+ .HasConstraintName("device_ota_updates_device");
+
+ b.Navigation("DeviceNavigation");
+ });
+
+ modelBuilder.Entity("OpenShock.Common.OpenShockDb.PasswordReset", b =>
+ {
+ b.HasOne("OpenShock.Common.OpenShockDb.User", "User")
+ .WithMany("PasswordResets")
+ .HasForeignKey("UserId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired()
+ .HasConstraintName("user_id");
+
+ b.Navigation("User");
+ });
+
+ modelBuilder.Entity("OpenShock.Common.OpenShockDb.Shocker", b =>
+ {
+ b.HasOne("OpenShock.Common.OpenShockDb.Device", "DeviceNavigation")
+ .WithMany("Shockers")
+ .HasForeignKey("Device")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired()
+ .HasConstraintName("device_id");
+
+ b.Navigation("DeviceNavigation");
+ });
+
+ modelBuilder.Entity("OpenShock.Common.OpenShockDb.ShockerControlLog", b =>
+ {
+ b.HasOne("OpenShock.Common.OpenShockDb.User", "ControlledByNavigation")
+ .WithMany("ShockerControlLogs")
+ .HasForeignKey("ControlledBy")
+ .OnDelete(DeleteBehavior.Cascade)
+ .HasConstraintName("fk_controlled_by");
+
+ b.HasOne("OpenShock.Common.OpenShockDb.Shocker", "Shocker")
+ .WithMany("ShockerControlLogs")
+ .HasForeignKey("ShockerId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired()
+ .HasConstraintName("fk_shocker_id");
+
+ b.Navigation("ControlledByNavigation");
+
+ b.Navigation("Shocker");
+ });
+
+ modelBuilder.Entity("OpenShock.Common.OpenShockDb.ShockerShare", b =>
+ {
+ b.HasOne("OpenShock.Common.OpenShockDb.User", "SharedWithNavigation")
+ .WithMany("ShockerShares")
+ .HasForeignKey("SharedWith")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired()
+ .HasConstraintName("shared_with_user_id");
+
+ b.HasOne("OpenShock.Common.OpenShockDb.Shocker", "Shocker")
+ .WithMany("ShockerShares")
+ .HasForeignKey("ShockerId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired()
+ .HasConstraintName("ref_shocker_id");
+
+ b.Navigation("SharedWithNavigation");
+
+ b.Navigation("Shocker");
+ });
+
+ modelBuilder.Entity("OpenShock.Common.OpenShockDb.ShockerShareCode", b =>
+ {
+ b.HasOne("OpenShock.Common.OpenShockDb.Shocker", "Shocker")
+ .WithMany("ShockerShareCodes")
+ .HasForeignKey("ShockerId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired()
+ .HasConstraintName("fk_shocker_id");
+
+ b.Navigation("Shocker");
+ });
+
+ modelBuilder.Entity("OpenShock.Common.OpenShockDb.ShockerSharesLink", b =>
+ {
+ b.HasOne("OpenShock.Common.OpenShockDb.User", "Owner")
+ .WithMany("ShockerSharesLinks")
+ .HasForeignKey("OwnerId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired()
+ .HasConstraintName("owner_id");
+
+ b.Navigation("Owner");
+ });
+
+ modelBuilder.Entity("OpenShock.Common.OpenShockDb.ShockerSharesLinksShocker", b =>
+ {
+ b.HasOne("OpenShock.Common.OpenShockDb.ShockerSharesLink", "ShareLink")
+ .WithMany("ShockerSharesLinksShockers")
+ .HasForeignKey("ShareLinkId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired()
+ .HasConstraintName("share_link_id");
+
+ b.HasOne("OpenShock.Common.OpenShockDb.Shocker", "Shocker")
+ .WithMany("ShockerSharesLinksShockers")
+ .HasForeignKey("ShockerId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired()
+ .HasConstraintName("shocker_id");
+
+ b.Navigation("ShareLink");
+
+ b.Navigation("Shocker");
+ });
+
+ modelBuilder.Entity("OpenShock.Common.OpenShockDb.UsersActivation", b =>
+ {
+ b.HasOne("OpenShock.Common.OpenShockDb.User", "User")
+ .WithMany("UsersActivations")
+ .HasForeignKey("UserId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired()
+ .HasConstraintName("user_id");
+
+ b.Navigation("User");
+ });
+
+ modelBuilder.Entity("OpenShock.Common.OpenShockDb.Device", b =>
+ {
+ b.Navigation("DeviceOtaUpdates");
+
+ b.Navigation("Shockers");
+ });
+
+ modelBuilder.Entity("OpenShock.Common.OpenShockDb.Shocker", b =>
+ {
+ b.Navigation("ShockerControlLogs");
+
+ b.Navigation("ShockerShareCodes");
+
+ b.Navigation("ShockerShares");
+
+ b.Navigation("ShockerSharesLinksShockers");
+ });
+
+ modelBuilder.Entity("OpenShock.Common.OpenShockDb.ShockerSharesLink", b =>
+ {
+ b.Navigation("ShockerSharesLinksShockers");
+ });
+
+ modelBuilder.Entity("OpenShock.Common.OpenShockDb.User", b =>
+ {
+ b.Navigation("ApiTokens");
+
+ b.Navigation("Devices");
+
+ b.Navigation("PasswordResets");
+
+ b.Navigation("ShockerControlLogs");
+
+ b.Navigation("ShockerShares");
+
+ b.Navigation("ShockerSharesLinks");
+
+ b.Navigation("UsersActivations");
+ });
+#pragma warning restore 612, 618
+ }
+ }
+}
diff --git a/Common/Migrations/20240327034706_Petrainer998DR.cs b/Common/Migrations/20240327034706_Petrainer998DR.cs
new file mode 100644
index 00000000..e2aca3cf
--- /dev/null
+++ b/Common/Migrations/20240327034706_Petrainer998DR.cs
@@ -0,0 +1,48 @@
+using Microsoft.EntityFrameworkCore.Migrations;
+
+#nullable disable
+
+namespace OpenShock.Common.Migrations
+{
+ ///
+ public partial class Petrainer998DR : Migration
+ {
+ ///
+ protected override void Up(MigrationBuilder migrationBuilder)
+ {
+ migrationBuilder.AlterDatabase()
+ .Annotation("Npgsql:CollationDefinition:public.ndcoll", "und-u-ks-level2,und-u-ks-level2,icu,False")
+ .Annotation("Npgsql:Enum:control_type", "sound,vibrate,shock,stop")
+ .Annotation("Npgsql:Enum:ota_update_status", "started,running,finished,error,timeout")
+ .Annotation("Npgsql:Enum:password_encryption_type", "pbkdf2,bcrypt_enhanced")
+ .Annotation("Npgsql:Enum:permission_type", "shockers.use")
+ .Annotation("Npgsql:Enum:rank_type", "user,support,staff,admin,system")
+ .Annotation("Npgsql:Enum:shocker_model_type", "caiXianlin,petTrainer,petrainer998DR")
+ .OldAnnotation("Npgsql:CollationDefinition:public.ndcoll", "und-u-ks-level2,und-u-ks-level2,icu,False")
+ .OldAnnotation("Npgsql:Enum:control_type", "sound,vibrate,shock,stop")
+ .OldAnnotation("Npgsql:Enum:ota_update_status", "started,running,finished,error,timeout")
+ .OldAnnotation("Npgsql:Enum:permission_type", "shockers.use")
+ .OldAnnotation("Npgsql:Enum:rank_type", "user,support,staff,admin,system")
+ .OldAnnotation("Npgsql:Enum:shocker_model_type", "caiXianlin,petTrainer");
+ }
+
+ ///
+ protected override void Down(MigrationBuilder migrationBuilder)
+ {
+ migrationBuilder.AlterDatabase()
+ .Annotation("Npgsql:CollationDefinition:public.ndcoll", "und-u-ks-level2,und-u-ks-level2,icu,False")
+ .Annotation("Npgsql:Enum:control_type", "sound,vibrate,shock,stop")
+ .Annotation("Npgsql:Enum:ota_update_status", "started,running,finished,error,timeout")
+ .Annotation("Npgsql:Enum:permission_type", "shockers.use")
+ .Annotation("Npgsql:Enum:rank_type", "user,support,staff,admin,system")
+ .Annotation("Npgsql:Enum:shocker_model_type", "caiXianlin,petTrainer")
+ .OldAnnotation("Npgsql:CollationDefinition:public.ndcoll", "und-u-ks-level2,und-u-ks-level2,icu,False")
+ .OldAnnotation("Npgsql:Enum:control_type", "sound,vibrate,shock,stop")
+ .OldAnnotation("Npgsql:Enum:ota_update_status", "started,running,finished,error,timeout")
+ .OldAnnotation("Npgsql:Enum:password_encryption_type", "pbkdf2,bcrypt_enhanced")
+ .OldAnnotation("Npgsql:Enum:permission_type", "shockers.use")
+ .OldAnnotation("Npgsql:Enum:rank_type", "user,support,staff,admin,system")
+ .OldAnnotation("Npgsql:Enum:shocker_model_type", "caiXianlin,petTrainer,petrainer998DR");
+ }
+ }
+}
diff --git a/Common/Migrations/OpenShockContextModelSnapshot.cs b/Common/Migrations/OpenShockContextModelSnapshot.cs
index 0bab4fbb..7feb0493 100644
--- a/Common/Migrations/OpenShockContextModelSnapshot.cs
+++ b/Common/Migrations/OpenShockContextModelSnapshot.cs
@@ -18,14 +18,15 @@ protected override void BuildModel(ModelBuilder modelBuilder)
#pragma warning disable 612, 618
modelBuilder
.HasAnnotation("Npgsql:CollationDefinition:public.ndcoll", "und-u-ks-level2,und-u-ks-level2,icu,False")
- .HasAnnotation("ProductVersion", "8.0.2")
+ .HasAnnotation("ProductVersion", "8.0.3")
.HasAnnotation("Relational:MaxIdentifierLength", 63);
NpgsqlModelBuilderExtensions.HasPostgresEnum(modelBuilder, "control_type", new[] { "sound", "vibrate", "shock", "stop" });
NpgsqlModelBuilderExtensions.HasPostgresEnum(modelBuilder, "ota_update_status", new[] { "started", "running", "finished", "error", "timeout" });
+ NpgsqlModelBuilderExtensions.HasPostgresEnum(modelBuilder, "password_encryption_type", new[] { "pbkdf2", "bcrypt_enhanced" });
NpgsqlModelBuilderExtensions.HasPostgresEnum(modelBuilder, "permission_type", new[] { "shockers.use" });
NpgsqlModelBuilderExtensions.HasPostgresEnum(modelBuilder, "rank_type", new[] { "user", "support", "staff", "admin", "system" });
- NpgsqlModelBuilderExtensions.HasPostgresEnum(modelBuilder, "shocker_model_type", new[] { "caiXianlin", "petTrainer" });
+ NpgsqlModelBuilderExtensions.HasPostgresEnum(modelBuilder, "shocker_model_type", new[] { "caiXianlin", "petTrainer", "petrainer998DR" });
NpgsqlModelBuilderExtensions.UseIdentityByDefaultColumns(modelBuilder);
modelBuilder.Entity("OpenShock.Common.OpenShockDb.ApiToken", b =>
diff --git a/Common/Models/ShockerModelType.cs b/Common/Models/ShockerModelType.cs
index bba47362..6ce4e64a 100644
--- a/Common/Models/ShockerModelType.cs
+++ b/Common/Models/ShockerModelType.cs
@@ -5,5 +5,6 @@ namespace OpenShock.Common.Models;
public enum ShockerModelType
{
[PgName("caiXianlin")] CaiXianlin = 0,
- [PgName("petTrainer")] PetTrainer = 1 // Misspelled, should be "petrainer"
+ [PgName("petTrainer")] PetTrainer = 1, // Misspelled, should be "petrainer",
+ [PgName("petrainer998DR")] Petrainer998DR = 2,
}
\ No newline at end of file
diff --git a/Common/OpenShockDb/OpenShockContext.cs b/Common/OpenShockDb/OpenShockContext.cs
index ce3993d4..b957d981 100644
--- a/Common/OpenShockDb/OpenShockContext.cs
+++ b/Common/OpenShockDb/OpenShockContext.cs
@@ -54,7 +54,7 @@ protected override void OnModelCreating(ModelBuilder modelBuilder)
.HasPostgresEnum("password_encryption_type", new[] { "pbkdf2", "bcrypt_enhanced" })
.HasPostgresEnum("permission_type", new[] { "shockers.use" })
.HasPostgresEnum("rank_type", new[] { "user", "support", "staff", "admin", "system" })
- .HasPostgresEnum("shocker_model_type", new[] { "caiXianlin", "petTrainer" })
+ .HasPostgresEnum("shocker_model_type", new[] { "caiXianlin", "petTrainer", "petrainer998DR" })
.HasAnnotation("Npgsql:CollationDefinition:public.ndcoll", "und-u-ks-level2,und-u-ks-level2,icu,False");
modelBuilder.Entity(entity =>
diff --git a/Common/Serialization/Types/ShockerModelType.fbs b/Common/Serialization/Types/ShockerModelType.fbs
index 06f14657..7433302a 100644
--- a/Common/Serialization/Types/ShockerModelType.fbs
+++ b/Common/Serialization/Types/ShockerModelType.fbs
@@ -2,5 +2,6 @@
enum ShockerModelType : uint8 {
CaiXianlin = 0,
- Petrainer = 1
+ Petrainer = 1,
+ Petrainer998DR = 2
}
\ No newline at end of file