diff --git a/backend/api/Migrations/20231114141124_AddInspectionTargetToInspection.Designer.cs b/backend/api/Migrations/20231114141124_AddInspectionTargetToInspection.Designer.cs
new file mode 100644
index 000000000..05c054ff4
--- /dev/null
+++ b/backend/api/Migrations/20231114141124_AddInspectionTargetToInspection.Designer.cs
@@ -0,0 +1,1312 @@
+//
+using System;
+using Api.Database.Context;
+using Microsoft.EntityFrameworkCore;
+using Microsoft.EntityFrameworkCore.Infrastructure;
+using Microsoft.EntityFrameworkCore.Migrations;
+using Microsoft.EntityFrameworkCore.Storage.ValueConversion;
+using Npgsql.EntityFrameworkCore.PostgreSQL.Metadata;
+
+#nullable disable
+
+namespace Api.Migrations
+{
+ [DbContext(typeof(FlotillaDbContext))]
+ [Migration("20231114141124_AddInspectionTargetToInspection")]
+ partial class AddInspectionTargetToInspection
+ {
+ ///
+ protected override void BuildTargetModel(ModelBuilder modelBuilder)
+ {
+#pragma warning disable 612, 618
+ modelBuilder
+ .HasAnnotation("ProductVersion", "7.0.11")
+ .HasAnnotation("Relational:MaxIdentifierLength", 63);
+
+ NpgsqlModelBuilderExtensions.UseIdentityByDefaultColumns(modelBuilder);
+
+ modelBuilder.Entity("Api.Database.Models.Area", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("text");
+
+ b.Property("DeckId")
+ .HasColumnType("text");
+
+ b.Property("DefaultLocalizationPoseId")
+ .HasColumnType("text");
+
+ b.Property("InstallationId")
+ .IsRequired()
+ .HasColumnType("text");
+
+ b.Property("Name")
+ .IsRequired()
+ .HasMaxLength(200)
+ .HasColumnType("character varying(200)");
+
+ b.Property("PlantId")
+ .IsRequired()
+ .HasColumnType("text");
+
+ b.HasKey("Id");
+
+ b.HasIndex("DeckId");
+
+ b.HasIndex("DefaultLocalizationPoseId");
+
+ b.HasIndex("InstallationId");
+
+ b.HasIndex("PlantId");
+
+ b.ToTable("Areas");
+ });
+
+ modelBuilder.Entity("Api.Database.Models.Deck", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("text");
+
+ b.Property("DefaultLocalizationPoseId")
+ .HasColumnType("text");
+
+ b.Property("InstallationId")
+ .IsRequired()
+ .HasColumnType("text");
+
+ b.Property("Name")
+ .IsRequired()
+ .HasMaxLength(200)
+ .HasColumnType("character varying(200)");
+
+ b.Property("PlantId")
+ .IsRequired()
+ .HasColumnType("text");
+
+ b.HasKey("Id");
+
+ b.HasIndex("DefaultLocalizationPoseId");
+
+ b.HasIndex("InstallationId");
+
+ b.HasIndex("PlantId");
+
+ b.ToTable("Decks");
+ });
+
+ modelBuilder.Entity("Api.Database.Models.DefaultLocalizationPose", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("text");
+
+ b.HasKey("Id");
+
+ b.ToTable("DefaultLocalizationPoses");
+ });
+
+ modelBuilder.Entity("Api.Database.Models.Inspection", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("text");
+
+ b.Property("AnalysisType")
+ .HasColumnType("text");
+
+ b.Property("EndTime")
+ .HasColumnType("timestamp with time zone");
+
+ b.Property("InspectionType")
+ .IsRequired()
+ .HasColumnType("text");
+
+ b.Property("InspectionUrl")
+ .HasMaxLength(250)
+ .HasColumnType("character varying(250)");
+
+ b.Property("IsarStepId")
+ .HasMaxLength(200)
+ .HasColumnType("character varying(200)");
+
+ b.Property("MissionTaskId")
+ .HasColumnType("text");
+
+ b.Property("StartTime")
+ .HasColumnType("timestamp with time zone");
+
+ b.Property("Status")
+ .IsRequired()
+ .HasColumnType("text");
+
+ b.Property("VideoDuration")
+ .HasColumnType("real");
+
+ b.HasKey("Id");
+
+ b.HasIndex("MissionTaskId");
+
+ b.ToTable("Inspections");
+ });
+
+ modelBuilder.Entity("Api.Database.Models.Installation", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("text");
+
+ b.Property("InstallationCode")
+ .IsRequired()
+ .HasMaxLength(10)
+ .HasColumnType("character varying(10)");
+
+ b.Property("Name")
+ .IsRequired()
+ .HasMaxLength(200)
+ .HasColumnType("character varying(200)");
+
+ b.HasKey("Id");
+
+ b.HasIndex("InstallationCode")
+ .IsUnique();
+
+ b.ToTable("Installations");
+ });
+
+ modelBuilder.Entity("Api.Database.Models.MissionDefinition", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("text");
+
+ b.Property("AreaId")
+ .HasColumnType("text");
+
+ b.Property("Comment")
+ .HasMaxLength(1000)
+ .HasColumnType("character varying(1000)");
+
+ b.Property("InspectionFrequency")
+ .HasColumnType("bigint");
+
+ b.Property("InstallationCode")
+ .IsRequired()
+ .HasColumnType("text");
+
+ b.Property("IsDeprecated")
+ .HasColumnType("boolean");
+
+ b.Property("LastSuccessfulRunId")
+ .HasColumnType("text");
+
+ b.Property("Name")
+ .IsRequired()
+ .HasMaxLength(200)
+ .HasColumnType("character varying(200)");
+
+ b.Property("SourceId")
+ .IsRequired()
+ .HasColumnType("text");
+
+ b.HasKey("Id");
+
+ b.HasIndex("AreaId");
+
+ b.HasIndex("LastSuccessfulRunId");
+
+ b.HasIndex("SourceId");
+
+ b.ToTable("MissionDefinitions");
+ });
+
+ modelBuilder.Entity("Api.Database.Models.MissionRun", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("text");
+
+ b.Property("AreaId")
+ .HasColumnType("text");
+
+ b.Property("Comment")
+ .HasMaxLength(1000)
+ .HasColumnType("character varying(1000)");
+
+ b.Property("Description")
+ .HasMaxLength(450)
+ .HasColumnType("character varying(450)");
+
+ b.Property("DesiredStartTime")
+ .HasColumnType("timestamp with time zone");
+
+ b.Property("EndTime")
+ .HasColumnType("timestamp with time zone");
+
+ b.Property("EstimatedDuration")
+ .HasColumnType("bigint");
+
+ b.Property("InstallationCode")
+ .IsRequired()
+ .HasMaxLength(200)
+ .HasColumnType("character varying(200)");
+
+ b.Property("IsarMissionId")
+ .HasMaxLength(200)
+ .HasColumnType("character varying(200)");
+
+ b.Property("MissionId")
+ .HasColumnType("text");
+
+ b.Property("MissionRunPriority")
+ .IsRequired()
+ .HasColumnType("text");
+
+ b.Property("Name")
+ .IsRequired()
+ .HasMaxLength(200)
+ .HasColumnType("character varying(200)");
+
+ b.Property("RobotId")
+ .IsRequired()
+ .HasColumnType("text");
+
+ b.Property("StartTime")
+ .HasColumnType("timestamp with time zone");
+
+ b.Property("Status")
+ .IsRequired()
+ .HasColumnType("text");
+
+ b.Property("StatusReason")
+ .HasMaxLength(450)
+ .HasColumnType("character varying(450)");
+
+ b.HasKey("Id");
+
+ b.HasIndex("AreaId");
+
+ b.HasIndex("RobotId");
+
+ b.ToTable("MissionRuns");
+ });
+
+ modelBuilder.Entity("Api.Database.Models.MissionTask", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("text");
+
+ b.Property("Description")
+ .HasMaxLength(500)
+ .HasColumnType("character varying(500)");
+
+ b.Property("EchoPoseId")
+ .HasColumnType("integer");
+
+ b.Property("EchoTagLink")
+ .HasMaxLength(200)
+ .HasColumnType("character varying(200)");
+
+ b.Property("EndTime")
+ .HasColumnType("timestamp with time zone");
+
+ b.Property("IsarTaskId")
+ .HasMaxLength(200)
+ .HasColumnType("character varying(200)");
+
+ b.Property("MissionRunId")
+ .HasColumnType("text");
+
+ b.Property("StartTime")
+ .HasColumnType("timestamp with time zone");
+
+ b.Property("Status")
+ .IsRequired()
+ .HasColumnType("text");
+
+ b.Property("TagId")
+ .HasMaxLength(200)
+ .HasColumnType("character varying(200)");
+
+ b.Property("TaskOrder")
+ .HasColumnType("integer");
+
+ b.HasKey("Id");
+
+ b.HasIndex("MissionRunId");
+
+ b.ToTable("MissionTasks");
+ });
+
+ modelBuilder.Entity("Api.Database.Models.Plant", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("text");
+
+ b.Property("InstallationId")
+ .IsRequired()
+ .HasColumnType("text");
+
+ b.Property("Name")
+ .IsRequired()
+ .HasMaxLength(200)
+ .HasColumnType("character varying(200)");
+
+ b.Property("PlantCode")
+ .IsRequired()
+ .HasMaxLength(10)
+ .HasColumnType("character varying(10)");
+
+ b.HasKey("Id");
+
+ b.HasIndex("InstallationId");
+
+ b.HasIndex("PlantCode")
+ .IsUnique();
+
+ b.ToTable("Plants");
+ });
+
+ modelBuilder.Entity("Api.Database.Models.Robot", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("text");
+
+ b.Property("BatteryLevel")
+ .HasColumnType("real");
+
+ b.Property("CurrentAreaId")
+ .HasColumnType("text");
+
+ b.Property("CurrentInstallation")
+ .IsRequired()
+ .HasColumnType("text");
+
+ b.Property("CurrentMissionId")
+ .HasColumnType("text");
+
+ b.Property("Enabled")
+ .HasColumnType("boolean");
+
+ b.Property("Host")
+ .IsRequired()
+ .HasMaxLength(200)
+ .HasColumnType("character varying(200)");
+
+ b.Property("IsarId")
+ .IsRequired()
+ .HasMaxLength(200)
+ .HasColumnType("character varying(200)");
+
+ b.Property("MissionQueueFrozen")
+ .HasColumnType("boolean");
+
+ b.Property("ModelId")
+ .IsRequired()
+ .HasColumnType("text");
+
+ b.Property("Name")
+ .IsRequired()
+ .HasMaxLength(200)
+ .HasColumnType("character varying(200)");
+
+ b.Property("Port")
+ .HasColumnType("integer");
+
+ b.Property("PressureLevel")
+ .HasColumnType("real");
+
+ b.Property("SerialNumber")
+ .IsRequired()
+ .HasMaxLength(200)
+ .HasColumnType("character varying(200)");
+
+ b.Property("Status")
+ .IsRequired()
+ .HasColumnType("text");
+
+ b.HasKey("Id");
+
+ b.HasIndex("CurrentAreaId");
+
+ b.HasIndex("ModelId");
+
+ b.ToTable("Robots");
+ });
+
+ modelBuilder.Entity("Api.Database.Models.RobotBatteryTimeseries", b =>
+ {
+ b.Property("BatteryLevel")
+ .HasColumnType("real");
+
+ b.Property("MissionId")
+ .HasColumnType("text");
+
+ b.Property("RobotId")
+ .IsRequired()
+ .HasColumnType("text");
+
+ b.Property("Time")
+ .HasColumnType("timestamp with time zone");
+
+ b.ToTable("RobotBatteryTimeseries");
+ });
+
+ modelBuilder.Entity("Api.Database.Models.RobotModel", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("text");
+
+ b.Property("AverageDurationPerTag")
+ .HasColumnType("real");
+
+ b.Property("BatteryWarningThreshold")
+ .HasColumnType("real");
+
+ b.Property("LowerPressureWarningThreshold")
+ .HasColumnType("real");
+
+ b.Property("Type")
+ .IsRequired()
+ .HasColumnType("text");
+
+ b.Property("UpperPressureWarningThreshold")
+ .HasColumnType("real");
+
+ b.HasKey("Id");
+
+ b.HasIndex("Type")
+ .IsUnique();
+
+ b.ToTable("RobotModels");
+ });
+
+ modelBuilder.Entity("Api.Database.Models.RobotPoseTimeseries", b =>
+ {
+ b.Property("MissionId")
+ .HasColumnType("text");
+
+ b.Property("OrientationW")
+ .HasColumnType("real");
+
+ b.Property("OrientationX")
+ .HasColumnType("real");
+
+ b.Property("OrientationY")
+ .HasColumnType("real");
+
+ b.Property("OrientationZ")
+ .HasColumnType("real");
+
+ b.Property("PositionX")
+ .HasColumnType("real");
+
+ b.Property("PositionY")
+ .HasColumnType("real");
+
+ b.Property("PositionZ")
+ .HasColumnType("real");
+
+ b.Property("RobotId")
+ .IsRequired()
+ .HasColumnType("text");
+
+ b.Property("Time")
+ .HasColumnType("timestamp with time zone");
+
+ b.ToTable("RobotPoseTimeseries");
+ });
+
+ modelBuilder.Entity("Api.Database.Models.RobotPressureTimeseries", b =>
+ {
+ b.Property("MissionId")
+ .HasColumnType("text");
+
+ b.Property("Pressure")
+ .HasColumnType("real");
+
+ b.Property("RobotId")
+ .IsRequired()
+ .HasColumnType("text");
+
+ b.Property("Time")
+ .HasColumnType("timestamp with time zone");
+
+ b.ToTable("RobotPressureTimeseries");
+ });
+
+ modelBuilder.Entity("Api.Database.Models.SafePosition", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("text");
+
+ b.Property("AreaId")
+ .HasColumnType("text");
+
+ b.HasKey("Id");
+
+ b.HasIndex("AreaId");
+
+ b.ToTable("SafePositions");
+ });
+
+ modelBuilder.Entity("Api.Database.Models.Source", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("text");
+
+ b.Property("SourceId")
+ .IsRequired()
+ .HasColumnType("text");
+
+ b.Property("Type")
+ .IsRequired()
+ .HasColumnType("text");
+
+ b.HasKey("Id");
+
+ b.ToTable("Sources");
+ });
+
+ modelBuilder.Entity("Api.Database.Models.Area", b =>
+ {
+ b.HasOne("Api.Database.Models.Deck", "Deck")
+ .WithMany()
+ .HasForeignKey("DeckId")
+ .OnDelete(DeleteBehavior.Restrict);
+
+ b.HasOne("Api.Database.Models.DefaultLocalizationPose", "DefaultLocalizationPose")
+ .WithMany()
+ .HasForeignKey("DefaultLocalizationPoseId");
+
+ b.HasOne("Api.Database.Models.Installation", "Installation")
+ .WithMany()
+ .HasForeignKey("InstallationId")
+ .OnDelete(DeleteBehavior.Restrict)
+ .IsRequired();
+
+ b.HasOne("Api.Database.Models.Plant", "Plant")
+ .WithMany()
+ .HasForeignKey("PlantId")
+ .OnDelete(DeleteBehavior.Restrict)
+ .IsRequired();
+
+ b.OwnsOne("Api.Database.Models.MapMetadata", "MapMetadata", b1 =>
+ {
+ b1.Property("AreaId")
+ .HasColumnType("text");
+
+ b1.Property("MapName")
+ .IsRequired()
+ .HasMaxLength(200)
+ .HasColumnType("character varying(200)");
+
+ b1.HasKey("AreaId");
+
+ b1.ToTable("Areas");
+
+ b1.WithOwner()
+ .HasForeignKey("AreaId");
+
+ b1.OwnsOne("Api.Database.Models.Boundary", "Boundary", b2 =>
+ {
+ b2.Property("MapMetadataAreaId")
+ .HasColumnType("text");
+
+ b2.Property("X1")
+ .HasColumnType("double precision");
+
+ b2.Property("X2")
+ .HasColumnType("double precision");
+
+ b2.Property("Y1")
+ .HasColumnType("double precision");
+
+ b2.Property("Y2")
+ .HasColumnType("double precision");
+
+ b2.Property("Z1")
+ .HasColumnType("double precision");
+
+ b2.Property("Z2")
+ .HasColumnType("double precision");
+
+ b2.HasKey("MapMetadataAreaId");
+
+ b2.ToTable("Areas");
+
+ b2.WithOwner()
+ .HasForeignKey("MapMetadataAreaId");
+ });
+
+ b1.OwnsOne("Api.Database.Models.TransformationMatrices", "TransformationMatrices", b2 =>
+ {
+ b2.Property("MapMetadataAreaId")
+ .HasColumnType("text");
+
+ b2.Property("C1")
+ .HasColumnType("double precision");
+
+ b2.Property("C2")
+ .HasColumnType("double precision");
+
+ b2.Property("D1")
+ .HasColumnType("double precision");
+
+ b2.Property("D2")
+ .HasColumnType("double precision");
+
+ b2.HasKey("MapMetadataAreaId");
+
+ b2.ToTable("Areas");
+
+ b2.WithOwner()
+ .HasForeignKey("MapMetadataAreaId");
+ });
+
+ b1.Navigation("Boundary")
+ .IsRequired();
+
+ b1.Navigation("TransformationMatrices")
+ .IsRequired();
+ });
+
+ b.Navigation("Deck");
+
+ b.Navigation("DefaultLocalizationPose");
+
+ b.Navigation("Installation");
+
+ b.Navigation("MapMetadata")
+ .IsRequired();
+
+ b.Navigation("Plant");
+ });
+
+ modelBuilder.Entity("Api.Database.Models.Deck", b =>
+ {
+ b.HasOne("Api.Database.Models.DefaultLocalizationPose", "DefaultLocalizationPose")
+ .WithMany()
+ .HasForeignKey("DefaultLocalizationPoseId");
+
+ b.HasOne("Api.Database.Models.Installation", "Installation")
+ .WithMany()
+ .HasForeignKey("InstallationId")
+ .OnDelete(DeleteBehavior.Restrict)
+ .IsRequired();
+
+ b.HasOne("Api.Database.Models.Plant", "Plant")
+ .WithMany()
+ .HasForeignKey("PlantId")
+ .OnDelete(DeleteBehavior.Restrict)
+ .IsRequired();
+
+ b.Navigation("DefaultLocalizationPose");
+
+ b.Navigation("Installation");
+
+ b.Navigation("Plant");
+ });
+
+ modelBuilder.Entity("Api.Database.Models.DefaultLocalizationPose", b =>
+ {
+ b.OwnsOne("Api.Database.Models.Pose", "Pose", b1 =>
+ {
+ b1.Property("DefaultLocalizationPoseId")
+ .HasColumnType("text");
+
+ b1.HasKey("DefaultLocalizationPoseId");
+
+ b1.ToTable("DefaultLocalizationPoses");
+
+ b1.WithOwner()
+ .HasForeignKey("DefaultLocalizationPoseId");
+
+ b1.OwnsOne("Api.Database.Models.Orientation", "Orientation", b2 =>
+ {
+ b2.Property("PoseDefaultLocalizationPoseId")
+ .HasColumnType("text");
+
+ b2.Property("W")
+ .HasColumnType("real");
+
+ b2.Property("X")
+ .HasColumnType("real");
+
+ b2.Property("Y")
+ .HasColumnType("real");
+
+ b2.Property("Z")
+ .HasColumnType("real");
+
+ b2.HasKey("PoseDefaultLocalizationPoseId");
+
+ b2.ToTable("DefaultLocalizationPoses");
+
+ b2.WithOwner()
+ .HasForeignKey("PoseDefaultLocalizationPoseId");
+ });
+
+ b1.OwnsOne("Api.Database.Models.Position", "Position", b2 =>
+ {
+ b2.Property("PoseDefaultLocalizationPoseId")
+ .HasColumnType("text");
+
+ b2.Property("X")
+ .HasColumnType("real");
+
+ b2.Property("Y")
+ .HasColumnType("real");
+
+ b2.Property("Z")
+ .HasColumnType("real");
+
+ b2.HasKey("PoseDefaultLocalizationPoseId");
+
+ b2.ToTable("DefaultLocalizationPoses");
+
+ b2.WithOwner()
+ .HasForeignKey("PoseDefaultLocalizationPoseId");
+ });
+
+ b1.Navigation("Orientation")
+ .IsRequired();
+
+ b1.Navigation("Position")
+ .IsRequired();
+ });
+
+ b.Navigation("Pose")
+ .IsRequired();
+ });
+
+ modelBuilder.Entity("Api.Database.Models.Inspection", b =>
+ {
+ b.HasOne("Api.Database.Models.MissionTask", null)
+ .WithMany("Inspections")
+ .HasForeignKey("MissionTaskId");
+
+ b.OwnsOne("Api.Database.Models.Position", "InspectionTarget", b1 =>
+ {
+ b1.Property("InspectionId")
+ .HasColumnType("text");
+
+ b1.Property("X")
+ .HasColumnType("real");
+
+ b1.Property("Y")
+ .HasColumnType("real");
+
+ b1.Property("Z")
+ .HasColumnType("real");
+
+ b1.HasKey("InspectionId");
+
+ b1.ToTable("Inspections");
+
+ b1.WithOwner()
+ .HasForeignKey("InspectionId");
+ });
+
+ b.OwnsMany("Api.Database.Models.InspectionFindings", "InspectionFindings", b1 =>
+ {
+ b1.Property("InspectionId")
+ .HasColumnType("text");
+
+ b1.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b1.Property("Id"));
+
+ b1.Property("Area")
+ .IsRequired()
+ .HasColumnType("text");
+
+ b1.Property("FindingsTag")
+ .IsRequired()
+ .HasColumnType("text");
+
+ b1.Property("InspectionDate")
+ .IsRequired()
+ .HasColumnType("text");
+
+ b1.Property("RobotName")
+ .IsRequired()
+ .HasColumnType("text");
+
+ b1.HasKey("InspectionId", "Id");
+
+ b1.ToTable("InspectionFindings");
+
+ b1.WithOwner()
+ .HasForeignKey("InspectionId");
+ });
+
+ b.Navigation("InspectionFindings");
+
+ b.Navigation("InspectionTarget")
+ .IsRequired();
+ });
+
+ modelBuilder.Entity("Api.Database.Models.MissionDefinition", b =>
+ {
+ b.HasOne("Api.Database.Models.Area", "Area")
+ .WithMany()
+ .HasForeignKey("AreaId");
+
+ b.HasOne("Api.Database.Models.MissionRun", "LastSuccessfulRun")
+ .WithMany()
+ .HasForeignKey("LastSuccessfulRunId");
+
+ b.HasOne("Api.Database.Models.Source", "Source")
+ .WithMany()
+ .HasForeignKey("SourceId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired();
+
+ b.Navigation("Area");
+
+ b.Navigation("LastSuccessfulRun");
+
+ b.Navigation("Source");
+ });
+
+ modelBuilder.Entity("Api.Database.Models.MissionRun", b =>
+ {
+ b.HasOne("Api.Database.Models.Area", "Area")
+ .WithMany()
+ .HasForeignKey("AreaId");
+
+ b.HasOne("Api.Database.Models.Robot", "Robot")
+ .WithMany()
+ .HasForeignKey("RobotId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired();
+
+ b.OwnsOne("Api.Database.Models.MapMetadata", "Map", b1 =>
+ {
+ b1.Property("MissionRunId")
+ .HasColumnType("text");
+
+ b1.Property("MapName")
+ .IsRequired()
+ .HasMaxLength(200)
+ .HasColumnType("character varying(200)");
+
+ b1.HasKey("MissionRunId");
+
+ b1.ToTable("MissionRuns");
+
+ b1.WithOwner()
+ .HasForeignKey("MissionRunId");
+
+ b1.OwnsOne("Api.Database.Models.Boundary", "Boundary", b2 =>
+ {
+ b2.Property("MapMetadataMissionRunId")
+ .HasColumnType("text");
+
+ b2.Property("X1")
+ .HasColumnType("double precision");
+
+ b2.Property("X2")
+ .HasColumnType("double precision");
+
+ b2.Property("Y1")
+ .HasColumnType("double precision");
+
+ b2.Property("Y2")
+ .HasColumnType("double precision");
+
+ b2.Property("Z1")
+ .HasColumnType("double precision");
+
+ b2.Property("Z2")
+ .HasColumnType("double precision");
+
+ b2.HasKey("MapMetadataMissionRunId");
+
+ b2.ToTable("MissionRuns");
+
+ b2.WithOwner()
+ .HasForeignKey("MapMetadataMissionRunId");
+ });
+
+ b1.OwnsOne("Api.Database.Models.TransformationMatrices", "TransformationMatrices", b2 =>
+ {
+ b2.Property("MapMetadataMissionRunId")
+ .HasColumnType("text");
+
+ b2.Property("C1")
+ .HasColumnType("double precision");
+
+ b2.Property("C2")
+ .HasColumnType("double precision");
+
+ b2.Property("D1")
+ .HasColumnType("double precision");
+
+ b2.Property("D2")
+ .HasColumnType("double precision");
+
+ b2.HasKey("MapMetadataMissionRunId");
+
+ b2.ToTable("MissionRuns");
+
+ b2.WithOwner()
+ .HasForeignKey("MapMetadataMissionRunId");
+ });
+
+ b1.Navigation("Boundary")
+ .IsRequired();
+
+ b1.Navigation("TransformationMatrices")
+ .IsRequired();
+ });
+
+ b.Navigation("Area");
+
+ b.Navigation("Map");
+
+ b.Navigation("Robot");
+ });
+
+ modelBuilder.Entity("Api.Database.Models.MissionTask", b =>
+ {
+ b.HasOne("Api.Database.Models.MissionRun", null)
+ .WithMany("Tasks")
+ .HasForeignKey("MissionRunId");
+
+ b.OwnsOne("Api.Database.Models.Position", "InspectionTarget", b1 =>
+ {
+ b1.Property("MissionTaskId")
+ .HasColumnType("text");
+
+ b1.Property("X")
+ .HasColumnType("real");
+
+ b1.Property("Y")
+ .HasColumnType("real");
+
+ b1.Property("Z")
+ .HasColumnType("real");
+
+ b1.HasKey("MissionTaskId");
+
+ b1.ToTable("MissionTasks");
+
+ b1.WithOwner()
+ .HasForeignKey("MissionTaskId");
+ });
+
+ b.OwnsOne("Api.Database.Models.Pose", "RobotPose", b1 =>
+ {
+ b1.Property("MissionTaskId")
+ .HasColumnType("text");
+
+ b1.HasKey("MissionTaskId");
+
+ b1.ToTable("MissionTasks");
+
+ b1.WithOwner()
+ .HasForeignKey("MissionTaskId");
+
+ b1.OwnsOne("Api.Database.Models.Orientation", "Orientation", b2 =>
+ {
+ b2.Property("PoseMissionTaskId")
+ .HasColumnType("text");
+
+ b2.Property("W")
+ .HasColumnType("real");
+
+ b2.Property("X")
+ .HasColumnType("real");
+
+ b2.Property("Y")
+ .HasColumnType("real");
+
+ b2.Property("Z")
+ .HasColumnType("real");
+
+ b2.HasKey("PoseMissionTaskId");
+
+ b2.ToTable("MissionTasks");
+
+ b2.WithOwner()
+ .HasForeignKey("PoseMissionTaskId");
+ });
+
+ b1.OwnsOne("Api.Database.Models.Position", "Position", b2 =>
+ {
+ b2.Property("PoseMissionTaskId")
+ .HasColumnType("text");
+
+ b2.Property("X")
+ .HasColumnType("real");
+
+ b2.Property("Y")
+ .HasColumnType("real");
+
+ b2.Property("Z")
+ .HasColumnType("real");
+
+ b2.HasKey("PoseMissionTaskId");
+
+ b2.ToTable("MissionTasks");
+
+ b2.WithOwner()
+ .HasForeignKey("PoseMissionTaskId");
+ });
+
+ b1.Navigation("Orientation")
+ .IsRequired();
+
+ b1.Navigation("Position")
+ .IsRequired();
+ });
+
+ b.Navigation("InspectionTarget")
+ .IsRequired();
+
+ b.Navigation("RobotPose")
+ .IsRequired();
+ });
+
+ modelBuilder.Entity("Api.Database.Models.Plant", b =>
+ {
+ b.HasOne("Api.Database.Models.Installation", "Installation")
+ .WithMany()
+ .HasForeignKey("InstallationId")
+ .OnDelete(DeleteBehavior.Restrict)
+ .IsRequired();
+
+ b.Navigation("Installation");
+ });
+
+ modelBuilder.Entity("Api.Database.Models.Robot", b =>
+ {
+ b.HasOne("Api.Database.Models.Area", "CurrentArea")
+ .WithMany()
+ .HasForeignKey("CurrentAreaId");
+
+ b.HasOne("Api.Database.Models.RobotModel", "Model")
+ .WithMany()
+ .HasForeignKey("ModelId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired();
+
+ b.OwnsOne("Api.Database.Models.Pose", "Pose", b1 =>
+ {
+ b1.Property("RobotId")
+ .HasColumnType("text");
+
+ b1.HasKey("RobotId");
+
+ b1.ToTable("Robots");
+
+ b1.WithOwner()
+ .HasForeignKey("RobotId");
+
+ b1.OwnsOne("Api.Database.Models.Orientation", "Orientation", b2 =>
+ {
+ b2.Property("PoseRobotId")
+ .HasColumnType("text");
+
+ b2.Property("W")
+ .HasColumnType("real");
+
+ b2.Property("X")
+ .HasColumnType("real");
+
+ b2.Property("Y")
+ .HasColumnType("real");
+
+ b2.Property("Z")
+ .HasColumnType("real");
+
+ b2.HasKey("PoseRobotId");
+
+ b2.ToTable("Robots");
+
+ b2.WithOwner()
+ .HasForeignKey("PoseRobotId");
+ });
+
+ b1.OwnsOne("Api.Database.Models.Position", "Position", b2 =>
+ {
+ b2.Property("PoseRobotId")
+ .HasColumnType("text");
+
+ b2.Property("X")
+ .HasColumnType("real");
+
+ b2.Property("Y")
+ .HasColumnType("real");
+
+ b2.Property("Z")
+ .HasColumnType("real");
+
+ b2.HasKey("PoseRobotId");
+
+ b2.ToTable("Robots");
+
+ b2.WithOwner()
+ .HasForeignKey("PoseRobotId");
+ });
+
+ b1.Navigation("Orientation")
+ .IsRequired();
+
+ b1.Navigation("Position")
+ .IsRequired();
+ });
+
+ b.OwnsMany("Api.Database.Models.VideoStream", "VideoStreams", b1 =>
+ {
+ b1.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("text");
+
+ b1.Property("Name")
+ .IsRequired()
+ .HasMaxLength(200)
+ .HasColumnType("character varying(200)");
+
+ b1.Property("RobotId")
+ .IsRequired()
+ .HasColumnType("text");
+
+ b1.Property("ShouldRotate270Clockwise")
+ .HasColumnType("boolean");
+
+ b1.Property("Type")
+ .IsRequired()
+ .HasMaxLength(64)
+ .HasColumnType("character varying(64)");
+
+ b1.Property("Url")
+ .IsRequired()
+ .HasMaxLength(200)
+ .HasColumnType("character varying(200)");
+
+ b1.HasKey("Id");
+
+ b1.HasIndex("RobotId");
+
+ b1.ToTable("VideoStream");
+
+ b1.WithOwner()
+ .HasForeignKey("RobotId");
+ });
+
+ b.Navigation("CurrentArea");
+
+ b.Navigation("Model");
+
+ b.Navigation("Pose")
+ .IsRequired();
+
+ b.Navigation("VideoStreams");
+ });
+
+ modelBuilder.Entity("Api.Database.Models.SafePosition", b =>
+ {
+ b.HasOne("Api.Database.Models.Area", null)
+ .WithMany("SafePositions")
+ .HasForeignKey("AreaId");
+
+ b.OwnsOne("Api.Database.Models.Pose", "Pose", b1 =>
+ {
+ b1.Property("SafePositionId")
+ .HasColumnType("text");
+
+ b1.HasKey("SafePositionId");
+
+ b1.ToTable("SafePositions");
+
+ b1.WithOwner()
+ .HasForeignKey("SafePositionId");
+
+ b1.OwnsOne("Api.Database.Models.Orientation", "Orientation", b2 =>
+ {
+ b2.Property("PoseSafePositionId")
+ .HasColumnType("text");
+
+ b2.Property("W")
+ .HasColumnType("real");
+
+ b2.Property("X")
+ .HasColumnType("real");
+
+ b2.Property("Y")
+ .HasColumnType("real");
+
+ b2.Property("Z")
+ .HasColumnType("real");
+
+ b2.HasKey("PoseSafePositionId");
+
+ b2.ToTable("SafePositions");
+
+ b2.WithOwner()
+ .HasForeignKey("PoseSafePositionId");
+ });
+
+ b1.OwnsOne("Api.Database.Models.Position", "Position", b2 =>
+ {
+ b2.Property("PoseSafePositionId")
+ .HasColumnType("text");
+
+ b2.Property("X")
+ .HasColumnType("real");
+
+ b2.Property("Y")
+ .HasColumnType("real");
+
+ b2.Property("Z")
+ .HasColumnType("real");
+
+ b2.HasKey("PoseSafePositionId");
+
+ b2.ToTable("SafePositions");
+
+ b2.WithOwner()
+ .HasForeignKey("PoseSafePositionId");
+ });
+
+ b1.Navigation("Orientation")
+ .IsRequired();
+
+ b1.Navigation("Position")
+ .IsRequired();
+ });
+
+ b.Navigation("Pose")
+ .IsRequired();
+ });
+
+ modelBuilder.Entity("Api.Database.Models.Area", b =>
+ {
+ b.Navigation("SafePositions");
+ });
+
+ modelBuilder.Entity("Api.Database.Models.MissionRun", b =>
+ {
+ b.Navigation("Tasks");
+ });
+
+ modelBuilder.Entity("Api.Database.Models.MissionTask", b =>
+ {
+ b.Navigation("Inspections");
+ });
+#pragma warning restore 612, 618
+ }
+ }
+}
diff --git a/backend/api/Migrations/20231114141124_AddInspectionTargetToInspection.cs b/backend/api/Migrations/20231114141124_AddInspectionTargetToInspection.cs
new file mode 100644
index 000000000..f2761f87e
--- /dev/null
+++ b/backend/api/Migrations/20231114141124_AddInspectionTargetToInspection.cs
@@ -0,0 +1,133 @@
+using Microsoft.EntityFrameworkCore.Migrations;
+
+#nullable disable
+
+namespace Api.Migrations
+{
+ ///
+ public partial class AddInspectionTargetToInspection : Migration
+ {
+ ///
+ protected override void Up(MigrationBuilder migrationBuilder)
+ {
+ migrationBuilder.DropForeignKey(
+ name: "FK_Inspections_MissionTasks_MissionTaskId",
+ table: "Inspections");
+
+ migrationBuilder.DropForeignKey(
+ name: "FK_MissionTasks_MissionRuns_MissionRunId",
+ table: "MissionTasks");
+
+ migrationBuilder.AlterColumn(
+ name: "MissionRunId",
+ table: "MissionTasks",
+ type: "text",
+ nullable: true,
+ oldClrType: typeof(string),
+ oldType: "text");
+
+ migrationBuilder.AlterColumn(
+ name: "MissionTaskId",
+ table: "Inspections",
+ type: "text",
+ nullable: true,
+ oldClrType: typeof(string),
+ oldType: "text");
+
+ migrationBuilder.AddColumn(
+ name: "InspectionTarget_X",
+ table: "Inspections",
+ type: "real",
+ nullable: false,
+ defaultValue: 0f);
+
+ migrationBuilder.AddColumn(
+ name: "InspectionTarget_Y",
+ table: "Inspections",
+ type: "real",
+ nullable: false,
+ defaultValue: 0f);
+
+ migrationBuilder.AddColumn(
+ name: "InspectionTarget_Z",
+ table: "Inspections",
+ type: "real",
+ nullable: false,
+ defaultValue: 0f);
+
+ migrationBuilder.AddForeignKey(
+ name: "FK_Inspections_MissionTasks_MissionTaskId",
+ table: "Inspections",
+ column: "MissionTaskId",
+ principalTable: "MissionTasks",
+ principalColumn: "Id");
+
+ migrationBuilder.AddForeignKey(
+ name: "FK_MissionTasks_MissionRuns_MissionRunId",
+ table: "MissionTasks",
+ column: "MissionRunId",
+ principalTable: "MissionRuns",
+ principalColumn: "Id");
+ }
+
+ ///
+ protected override void Down(MigrationBuilder migrationBuilder)
+ {
+ migrationBuilder.DropForeignKey(
+ name: "FK_Inspections_MissionTasks_MissionTaskId",
+ table: "Inspections");
+
+ migrationBuilder.DropForeignKey(
+ name: "FK_MissionTasks_MissionRuns_MissionRunId",
+ table: "MissionTasks");
+
+ migrationBuilder.DropColumn(
+ name: "InspectionTarget_X",
+ table: "Inspections");
+
+ migrationBuilder.DropColumn(
+ name: "InspectionTarget_Y",
+ table: "Inspections");
+
+ migrationBuilder.DropColumn(
+ name: "InspectionTarget_Z",
+ table: "Inspections");
+
+ migrationBuilder.AlterColumn(
+ name: "MissionRunId",
+ table: "MissionTasks",
+ type: "text",
+ nullable: false,
+ defaultValue: "",
+ oldClrType: typeof(string),
+ oldType: "text",
+ oldNullable: true);
+
+ migrationBuilder.AlterColumn(
+ name: "MissionTaskId",
+ table: "Inspections",
+ type: "text",
+ nullable: false,
+ defaultValue: "",
+ oldClrType: typeof(string),
+ oldType: "text",
+ oldNullable: true);
+
+ migrationBuilder.AddForeignKey(
+ name: "FK_Inspections_MissionTasks_MissionTaskId",
+ table: "Inspections",
+ column: "MissionTaskId",
+ principalTable: "MissionTasks",
+ principalColumn: "Id",
+ onDelete: ReferentialAction.Cascade);
+
+ migrationBuilder.AddForeignKey(
+ name: "FK_MissionTasks_MissionRuns_MissionRunId",
+ table: "MissionTasks",
+ column: "MissionRunId",
+ principalTable: "MissionRuns",
+ principalColumn: "Id",
+ onDelete: ReferentialAction.Cascade);
+ }
+ }
+}
diff --git a/backend/api/Migrations/FlotillaDbContextModelSnapshot.cs b/backend/api/Migrations/FlotillaDbContextModelSnapshot.cs
index 67b9f092d..6d1c2c68c 100644
--- a/backend/api/Migrations/FlotillaDbContextModelSnapshot.cs
+++ b/backend/api/Migrations/FlotillaDbContextModelSnapshot.cs
@@ -104,6 +104,50 @@ protected override void BuildModel(ModelBuilder modelBuilder)
b.ToTable("DefaultLocalizationPoses");
});
+ modelBuilder.Entity("Api.Database.Models.Inspection", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("text");
+
+ b.Property("AnalysisType")
+ .HasColumnType("text");
+
+ b.Property("EndTime")
+ .HasColumnType("timestamp with time zone");
+
+ b.Property("InspectionType")
+ .IsRequired()
+ .HasColumnType("text");
+
+ b.Property("InspectionUrl")
+ .HasMaxLength(250)
+ .HasColumnType("character varying(250)");
+
+ b.Property("IsarStepId")
+ .HasMaxLength(200)
+ .HasColumnType("character varying(200)");
+
+ b.Property("MissionTaskId")
+ .HasColumnType("text");
+
+ b.Property("StartTime")
+ .HasColumnType("timestamp with time zone");
+
+ b.Property("Status")
+ .IsRequired()
+ .HasColumnType("text");
+
+ b.Property("VideoDuration")
+ .HasColumnType("real");
+
+ b.HasKey("Id");
+
+ b.HasIndex("MissionTaskId");
+
+ b.ToTable("Inspections");
+ });
+
modelBuilder.Entity("Api.Database.Models.Installation", b =>
{
b.Property("Id")
@@ -245,6 +289,54 @@ protected override void BuildModel(ModelBuilder modelBuilder)
b.ToTable("MissionRuns");
});
+ modelBuilder.Entity("Api.Database.Models.MissionTask", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("text");
+
+ b.Property("Description")
+ .HasMaxLength(500)
+ .HasColumnType("character varying(500)");
+
+ b.Property("EchoPoseId")
+ .HasColumnType("integer");
+
+ b.Property("EchoTagLink")
+ .HasMaxLength(200)
+ .HasColumnType("character varying(200)");
+
+ b.Property("EndTime")
+ .HasColumnType("timestamp with time zone");
+
+ b.Property("IsarTaskId")
+ .HasMaxLength(200)
+ .HasColumnType("character varying(200)");
+
+ b.Property("MissionRunId")
+ .HasColumnType("text");
+
+ b.Property("StartTime")
+ .HasColumnType("timestamp with time zone");
+
+ b.Property("Status")
+ .IsRequired()
+ .HasColumnType("text");
+
+ b.Property("TagId")
+ .HasMaxLength(200)
+ .HasColumnType("character varying(200)");
+
+ b.Property("TaskOrder")
+ .HasColumnType("integer");
+
+ b.HasKey("Id");
+
+ b.HasIndex("MissionRunId");
+
+ b.ToTable("MissionTasks");
+ });
+
modelBuilder.Entity("Api.Database.Models.Plant", b =>
{
b.Property("Id")
@@ -692,6 +784,75 @@ protected override void BuildModel(ModelBuilder modelBuilder)
.IsRequired();
});
+ modelBuilder.Entity("Api.Database.Models.Inspection", b =>
+ {
+ b.HasOne("Api.Database.Models.MissionTask", null)
+ .WithMany("Inspections")
+ .HasForeignKey("MissionTaskId");
+
+ b.OwnsOne("Api.Database.Models.Position", "InspectionTarget", b1 =>
+ {
+ b1.Property("InspectionId")
+ .HasColumnType("text");
+
+ b1.Property("X")
+ .HasColumnType("real");
+
+ b1.Property("Y")
+ .HasColumnType("real");
+
+ b1.Property("Z")
+ .HasColumnType("real");
+
+ b1.HasKey("InspectionId");
+
+ b1.ToTable("Inspections");
+
+ b1.WithOwner()
+ .HasForeignKey("InspectionId");
+ });
+
+ b.OwnsMany("Api.Database.Models.InspectionFindings", "InspectionFindings", b1 =>
+ {
+ b1.Property("InspectionId")
+ .HasColumnType("text");
+
+ b1.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("integer");
+
+ NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b1.Property("Id"));
+
+ b1.Property("Area")
+ .IsRequired()
+ .HasColumnType("text");
+
+ b1.Property("FindingsTag")
+ .IsRequired()
+ .HasColumnType("text");
+
+ b1.Property("InspectionDate")
+ .IsRequired()
+ .HasColumnType("text");
+
+ b1.Property("RobotName")
+ .IsRequired()
+ .HasColumnType("text");
+
+ b1.HasKey("InspectionId", "Id");
+
+ b1.ToTable("InspectionFindings");
+
+ b1.WithOwner()
+ .HasForeignKey("InspectionId");
+ });
+
+ b.Navigation("InspectionFindings");
+
+ b.Navigation("InspectionTarget")
+ .IsRequired();
+ });
+
modelBuilder.Entity("Api.Database.Models.MissionDefinition", b =>
{
b.HasOne("Api.Database.Models.Area", "Area")
@@ -807,144 +968,81 @@ protected override void BuildModel(ModelBuilder modelBuilder)
.IsRequired();
});
- b.OwnsMany("Api.Database.Models.MissionTask", "Tasks", b1 =>
- {
- b1.Property("Id")
- .ValueGeneratedOnAdd()
- .HasColumnType("text");
-
- b1.Property("Description")
- .HasMaxLength(500)
- .HasColumnType("character varying(500)");
-
- b1.Property("EchoPoseId")
- .HasColumnType("integer");
-
- b1.Property("EchoTagLink")
- .HasMaxLength(200)
- .HasColumnType("character varying(200)");
-
- b1.Property("EndTime")
- .HasColumnType("timestamp with time zone");
+ b.Navigation("Area");
- b1.Property("IsarTaskId")
- .HasMaxLength(200)
- .HasColumnType("character varying(200)");
+ b.Navigation("Map");
- b1.Property("MissionRunId")
- .HasColumnType("text");
+ b.Navigation("Robot");
+ });
- b1.Property("StartTime")
- .HasColumnType("timestamp with time zone");
+ modelBuilder.Entity("Api.Database.Models.MissionTask", b =>
+ {
+ b.HasOne("Api.Database.Models.MissionRun", null)
+ .WithMany("Tasks")
+ .HasForeignKey("MissionRunId");
- b1.Property("Status")
- .IsRequired()
+ b.OwnsOne("Api.Database.Models.Position", "InspectionTarget", b1 =>
+ {
+ b1.Property("MissionTaskId")
.HasColumnType("text");
- b1.Property("TagId")
- .HasMaxLength(200)
- .HasColumnType("character varying(200)");
+ b1.Property("X")
+ .HasColumnType("real");
- b1.Property("TaskOrder")
- .HasColumnType("integer");
+ b1.Property("Y")
+ .HasColumnType("real");
- b1.HasKey("Id");
+ b1.Property("Z")
+ .HasColumnType("real");
- b1.HasIndex("MissionRunId");
+ b1.HasKey("MissionTaskId");
b1.ToTable("MissionTasks");
b1.WithOwner()
- .HasForeignKey("MissionRunId");
-
- b1.OwnsMany("Api.Database.Models.Inspection", "Inspections", b2 =>
- {
- b2.Property