diff --git a/FabWorks.Api/FabWorks.Api.csproj b/FabWorks.Api/FabWorks.Api.csproj
index e4f3a2f..921afe9 100644
--- a/FabWorks.Api/FabWorks.Api.csproj
+++ b/FabWorks.Api/FabWorks.Api.csproj
@@ -10,4 +10,11 @@
+
+
+ runtime; build; native; contentfiles; analyzers; buildtransitive
+ all
+
+
+
diff --git a/FabWorks.Core/Migrations/20260218113525_InitialCreate.Designer.cs b/FabWorks.Core/Migrations/20260218113525_InitialCreate.Designer.cs
new file mode 100644
index 0000000..57ec80a
--- /dev/null
+++ b/FabWorks.Core/Migrations/20260218113525_InitialCreate.Designer.cs
@@ -0,0 +1,258 @@
+//
+using System;
+using FabWorks.Core.Data;
+using Microsoft.EntityFrameworkCore;
+using Microsoft.EntityFrameworkCore.Infrastructure;
+using Microsoft.EntityFrameworkCore.Metadata;
+using Microsoft.EntityFrameworkCore.Migrations;
+using Microsoft.EntityFrameworkCore.Storage.ValueConversion;
+
+#nullable disable
+
+namespace FabWorks.Core.Migrations
+{
+ [DbContext(typeof(FabWorksDbContext))]
+ [Migration("20260218113525_InitialCreate")]
+ partial class InitialCreate
+ {
+ ///
+ protected override void BuildTargetModel(ModelBuilder modelBuilder)
+ {
+#pragma warning disable 612, 618
+ modelBuilder
+ .HasAnnotation("ProductVersion", "8.0.11")
+ .HasAnnotation("Relational:MaxIdentifierLength", 128);
+
+ SqlServerModelBuilderExtensions.UseIdentityColumns(modelBuilder);
+
+ modelBuilder.Entity("FabWorks.Core.Models.BomItem", b =>
+ {
+ b.Property("ID")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("int");
+
+ SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property("ID"));
+
+ b.Property("ConfigurationName")
+ .HasMaxLength(100)
+ .HasColumnType("nvarchar(100)");
+
+ b.Property("Description")
+ .HasMaxLength(500)
+ .HasColumnType("nvarchar(500)");
+
+ b.Property("ExportRecordId")
+ .HasColumnType("int");
+
+ b.Property("ItemNo")
+ .HasMaxLength(50)
+ .HasColumnType("nvarchar(50)");
+
+ b.Property("Material")
+ .HasMaxLength(100)
+ .HasColumnType("nvarchar(100)");
+
+ b.Property("PartName")
+ .HasMaxLength(200)
+ .HasColumnType("nvarchar(200)");
+
+ b.Property("PartNo")
+ .HasMaxLength(100)
+ .HasColumnType("nvarchar(100)");
+
+ b.Property("Qty")
+ .HasColumnType("int");
+
+ b.Property("SortOrder")
+ .HasColumnType("int");
+
+ b.Property("TotalQty")
+ .HasColumnType("int");
+
+ b.HasKey("ID");
+
+ b.HasIndex("ExportRecordId");
+
+ b.ToTable("BomItems");
+ });
+
+ modelBuilder.Entity("FabWorks.Core.Models.CutTemplate", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("int");
+
+ SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property("Id"));
+
+ b.Property("BomItemId")
+ .HasColumnType("int");
+
+ b.Property("ContentHash")
+ .HasMaxLength(64)
+ .HasColumnType("nvarchar(64)");
+
+ b.Property("CutTemplateName")
+ .HasMaxLength(100)
+ .HasColumnType("nvarchar(100)");
+
+ b.Property("DefaultBendRadius")
+ .HasColumnType("float");
+
+ b.Property("DxfFilePath")
+ .HasMaxLength(500)
+ .HasColumnType("nvarchar(500)");
+
+ b.Property("KFactor")
+ .HasColumnType("float");
+
+ b.Property("Thickness")
+ .HasColumnType("float");
+
+ b.HasKey("Id");
+
+ b.HasIndex("BomItemId")
+ .IsUnique();
+
+ b.ToTable("CutTemplates");
+ });
+
+ modelBuilder.Entity("FabWorks.Core.Models.ExportRecord", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("int");
+
+ SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property("Id"));
+
+ b.Property("DrawingNumber")
+ .HasMaxLength(100)
+ .HasColumnType("nvarchar(100)");
+
+ b.Property("ExportedAt")
+ .HasColumnType("datetime2");
+
+ b.Property("ExportedBy")
+ .HasMaxLength(100)
+ .HasColumnType("nvarchar(100)");
+
+ b.Property("OutputFolder")
+ .HasMaxLength(500)
+ .HasColumnType("nvarchar(500)");
+
+ b.Property("PdfContentHash")
+ .HasMaxLength(64)
+ .HasColumnType("nvarchar(64)");
+
+ b.Property("SourceFilePath")
+ .HasMaxLength(500)
+ .HasColumnType("nvarchar(500)");
+
+ b.HasKey("Id");
+
+ b.ToTable("ExportRecords");
+ });
+
+ modelBuilder.Entity("FabWorks.Core.Models.FormProgram", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("int");
+
+ SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property("Id"));
+
+ b.Property("BendCount")
+ .HasColumnType("int");
+
+ b.Property("BomItemId")
+ .HasColumnType("int");
+
+ b.Property("ContentHash")
+ .HasMaxLength(64)
+ .HasColumnType("nvarchar(64)");
+
+ b.Property("KFactor")
+ .HasColumnType("float");
+
+ b.Property("LowerToolNames")
+ .HasMaxLength(500)
+ .HasColumnType("nvarchar(500)");
+
+ b.Property("MaterialType")
+ .HasMaxLength(50)
+ .HasColumnType("nvarchar(50)");
+
+ b.Property("ProgramFilePath")
+ .HasMaxLength(500)
+ .HasColumnType("nvarchar(500)");
+
+ b.Property("ProgramName")
+ .HasMaxLength(200)
+ .HasColumnType("nvarchar(200)");
+
+ b.Property("SetupNotes")
+ .HasMaxLength(2000)
+ .HasColumnType("nvarchar(2000)");
+
+ b.Property("Thickness")
+ .HasColumnType("float");
+
+ b.Property("UpperToolNames")
+ .HasMaxLength(500)
+ .HasColumnType("nvarchar(500)");
+
+ b.HasKey("Id");
+
+ b.HasIndex("BomItemId")
+ .IsUnique();
+
+ b.ToTable("FormPrograms");
+ });
+
+ modelBuilder.Entity("FabWorks.Core.Models.BomItem", b =>
+ {
+ b.HasOne("FabWorks.Core.Models.ExportRecord", "ExportRecord")
+ .WithMany("BomItems")
+ .HasForeignKey("ExportRecordId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired();
+
+ b.Navigation("ExportRecord");
+ });
+
+ modelBuilder.Entity("FabWorks.Core.Models.CutTemplate", b =>
+ {
+ b.HasOne("FabWorks.Core.Models.BomItem", "BomItem")
+ .WithOne("CutTemplate")
+ .HasForeignKey("FabWorks.Core.Models.CutTemplate", "BomItemId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired();
+
+ b.Navigation("BomItem");
+ });
+
+ modelBuilder.Entity("FabWorks.Core.Models.FormProgram", b =>
+ {
+ b.HasOne("FabWorks.Core.Models.BomItem", "BomItem")
+ .WithOne("FormProgram")
+ .HasForeignKey("FabWorks.Core.Models.FormProgram", "BomItemId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired();
+
+ b.Navigation("BomItem");
+ });
+
+ modelBuilder.Entity("FabWorks.Core.Models.BomItem", b =>
+ {
+ b.Navigation("CutTemplate");
+
+ b.Navigation("FormProgram");
+ });
+
+ modelBuilder.Entity("FabWorks.Core.Models.ExportRecord", b =>
+ {
+ b.Navigation("BomItems");
+ });
+#pragma warning restore 612, 618
+ }
+ }
+}
diff --git a/FabWorks.Core/Migrations/20260218113525_InitialCreate.cs b/FabWorks.Core/Migrations/20260218113525_InitialCreate.cs
new file mode 100644
index 0000000..1a41881
--- /dev/null
+++ b/FabWorks.Core/Migrations/20260218113525_InitialCreate.cs
@@ -0,0 +1,61 @@
+using System;
+using Microsoft.EntityFrameworkCore.Migrations;
+
+#nullable disable
+
+namespace FabWorks.Core.Migrations
+{
+ ///
+ public partial class InitialCreate : Migration
+ {
+ ///
+ protected override void Up(MigrationBuilder migrationBuilder)
+ {
+ // NOTE: ExportRecords, BomItems, and CutTemplates tables already exist
+ // in the database (created by ExportDXF's ExportDxfDbContext migrations).
+ // This migration only adds the new FormPrograms table.
+
+ migrationBuilder.CreateTable(
+ name: "FormPrograms",
+ columns: table => new
+ {
+ Id = table.Column(type: "int", nullable: false)
+ .Annotation("SqlServer:Identity", "1, 1"),
+ ProgramFilePath = table.Column(type: "nvarchar(500)", maxLength: 500, nullable: true),
+ ContentHash = table.Column(type: "nvarchar(64)", maxLength: 64, nullable: true),
+ ProgramName = table.Column(type: "nvarchar(200)", maxLength: 200, nullable: true),
+ Thickness = table.Column(type: "float", nullable: true),
+ MaterialType = table.Column(type: "nvarchar(50)", maxLength: 50, nullable: true),
+ KFactor = table.Column(type: "float", nullable: true),
+ BendCount = table.Column(type: "int", nullable: false),
+ UpperToolNames = table.Column(type: "nvarchar(500)", maxLength: 500, nullable: true),
+ LowerToolNames = table.Column(type: "nvarchar(500)", maxLength: 500, nullable: true),
+ SetupNotes = table.Column(type: "nvarchar(2000)", maxLength: 2000, nullable: true),
+ BomItemId = table.Column(type: "int", nullable: false)
+ },
+ constraints: table =>
+ {
+ table.PrimaryKey("PK_FormPrograms", x => x.Id);
+ table.ForeignKey(
+ name: "FK_FormPrograms_BomItems_BomItemId",
+ column: x => x.BomItemId,
+ principalTable: "BomItems",
+ principalColumn: "ID",
+ onDelete: ReferentialAction.Cascade);
+ });
+
+ migrationBuilder.CreateIndex(
+ name: "IX_FormPrograms_BomItemId",
+ table: "FormPrograms",
+ column: "BomItemId",
+ unique: true);
+ }
+
+ ///
+ protected override void Down(MigrationBuilder migrationBuilder)
+ {
+ migrationBuilder.DropTable(
+ name: "FormPrograms");
+ }
+ }
+}
diff --git a/FabWorks.Core/Migrations/FabWorksDbContextModelSnapshot.cs b/FabWorks.Core/Migrations/FabWorksDbContextModelSnapshot.cs
new file mode 100644
index 0000000..1096f92
--- /dev/null
+++ b/FabWorks.Core/Migrations/FabWorksDbContextModelSnapshot.cs
@@ -0,0 +1,255 @@
+//
+using System;
+using FabWorks.Core.Data;
+using Microsoft.EntityFrameworkCore;
+using Microsoft.EntityFrameworkCore.Infrastructure;
+using Microsoft.EntityFrameworkCore.Metadata;
+using Microsoft.EntityFrameworkCore.Storage.ValueConversion;
+
+#nullable disable
+
+namespace FabWorks.Core.Migrations
+{
+ [DbContext(typeof(FabWorksDbContext))]
+ partial class FabWorksDbContextModelSnapshot : ModelSnapshot
+ {
+ protected override void BuildModel(ModelBuilder modelBuilder)
+ {
+#pragma warning disable 612, 618
+ modelBuilder
+ .HasAnnotation("ProductVersion", "8.0.11")
+ .HasAnnotation("Relational:MaxIdentifierLength", 128);
+
+ SqlServerModelBuilderExtensions.UseIdentityColumns(modelBuilder);
+
+ modelBuilder.Entity("FabWorks.Core.Models.BomItem", b =>
+ {
+ b.Property("ID")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("int");
+
+ SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property("ID"));
+
+ b.Property("ConfigurationName")
+ .HasMaxLength(100)
+ .HasColumnType("nvarchar(100)");
+
+ b.Property("Description")
+ .HasMaxLength(500)
+ .HasColumnType("nvarchar(500)");
+
+ b.Property("ExportRecordId")
+ .HasColumnType("int");
+
+ b.Property("ItemNo")
+ .HasMaxLength(50)
+ .HasColumnType("nvarchar(50)");
+
+ b.Property("Material")
+ .HasMaxLength(100)
+ .HasColumnType("nvarchar(100)");
+
+ b.Property("PartName")
+ .HasMaxLength(200)
+ .HasColumnType("nvarchar(200)");
+
+ b.Property("PartNo")
+ .HasMaxLength(100)
+ .HasColumnType("nvarchar(100)");
+
+ b.Property("Qty")
+ .HasColumnType("int");
+
+ b.Property("SortOrder")
+ .HasColumnType("int");
+
+ b.Property("TotalQty")
+ .HasColumnType("int");
+
+ b.HasKey("ID");
+
+ b.HasIndex("ExportRecordId");
+
+ b.ToTable("BomItems");
+ });
+
+ modelBuilder.Entity("FabWorks.Core.Models.CutTemplate", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("int");
+
+ SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property("Id"));
+
+ b.Property("BomItemId")
+ .HasColumnType("int");
+
+ b.Property("ContentHash")
+ .HasMaxLength(64)
+ .HasColumnType("nvarchar(64)");
+
+ b.Property("CutTemplateName")
+ .HasMaxLength(100)
+ .HasColumnType("nvarchar(100)");
+
+ b.Property("DefaultBendRadius")
+ .HasColumnType("float");
+
+ b.Property("DxfFilePath")
+ .HasMaxLength(500)
+ .HasColumnType("nvarchar(500)");
+
+ b.Property("KFactor")
+ .HasColumnType("float");
+
+ b.Property("Thickness")
+ .HasColumnType("float");
+
+ b.HasKey("Id");
+
+ b.HasIndex("BomItemId")
+ .IsUnique();
+
+ b.ToTable("CutTemplates");
+ });
+
+ modelBuilder.Entity("FabWorks.Core.Models.ExportRecord", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("int");
+
+ SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property("Id"));
+
+ b.Property("DrawingNumber")
+ .HasMaxLength(100)
+ .HasColumnType("nvarchar(100)");
+
+ b.Property("ExportedAt")
+ .HasColumnType("datetime2");
+
+ b.Property("ExportedBy")
+ .HasMaxLength(100)
+ .HasColumnType("nvarchar(100)");
+
+ b.Property("OutputFolder")
+ .HasMaxLength(500)
+ .HasColumnType("nvarchar(500)");
+
+ b.Property("PdfContentHash")
+ .HasMaxLength(64)
+ .HasColumnType("nvarchar(64)");
+
+ b.Property("SourceFilePath")
+ .HasMaxLength(500)
+ .HasColumnType("nvarchar(500)");
+
+ b.HasKey("Id");
+
+ b.ToTable("ExportRecords");
+ });
+
+ modelBuilder.Entity("FabWorks.Core.Models.FormProgram", b =>
+ {
+ b.Property("Id")
+ .ValueGeneratedOnAdd()
+ .HasColumnType("int");
+
+ SqlServerPropertyBuilderExtensions.UseIdentityColumn(b.Property("Id"));
+
+ b.Property("BendCount")
+ .HasColumnType("int");
+
+ b.Property("BomItemId")
+ .HasColumnType("int");
+
+ b.Property("ContentHash")
+ .HasMaxLength(64)
+ .HasColumnType("nvarchar(64)");
+
+ b.Property("KFactor")
+ .HasColumnType("float");
+
+ b.Property("LowerToolNames")
+ .HasMaxLength(500)
+ .HasColumnType("nvarchar(500)");
+
+ b.Property("MaterialType")
+ .HasMaxLength(50)
+ .HasColumnType("nvarchar(50)");
+
+ b.Property("ProgramFilePath")
+ .HasMaxLength(500)
+ .HasColumnType("nvarchar(500)");
+
+ b.Property("ProgramName")
+ .HasMaxLength(200)
+ .HasColumnType("nvarchar(200)");
+
+ b.Property("SetupNotes")
+ .HasMaxLength(2000)
+ .HasColumnType("nvarchar(2000)");
+
+ b.Property("Thickness")
+ .HasColumnType("float");
+
+ b.Property("UpperToolNames")
+ .HasMaxLength(500)
+ .HasColumnType("nvarchar(500)");
+
+ b.HasKey("Id");
+
+ b.HasIndex("BomItemId")
+ .IsUnique();
+
+ b.ToTable("FormPrograms");
+ });
+
+ modelBuilder.Entity("FabWorks.Core.Models.BomItem", b =>
+ {
+ b.HasOne("FabWorks.Core.Models.ExportRecord", "ExportRecord")
+ .WithMany("BomItems")
+ .HasForeignKey("ExportRecordId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired();
+
+ b.Navigation("ExportRecord");
+ });
+
+ modelBuilder.Entity("FabWorks.Core.Models.CutTemplate", b =>
+ {
+ b.HasOne("FabWorks.Core.Models.BomItem", "BomItem")
+ .WithOne("CutTemplate")
+ .HasForeignKey("FabWorks.Core.Models.CutTemplate", "BomItemId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired();
+
+ b.Navigation("BomItem");
+ });
+
+ modelBuilder.Entity("FabWorks.Core.Models.FormProgram", b =>
+ {
+ b.HasOne("FabWorks.Core.Models.BomItem", "BomItem")
+ .WithOne("FormProgram")
+ .HasForeignKey("FabWorks.Core.Models.FormProgram", "BomItemId")
+ .OnDelete(DeleteBehavior.Cascade)
+ .IsRequired();
+
+ b.Navigation("BomItem");
+ });
+
+ modelBuilder.Entity("FabWorks.Core.Models.BomItem", b =>
+ {
+ b.Navigation("CutTemplate");
+
+ b.Navigation("FormProgram");
+ });
+
+ modelBuilder.Entity("FabWorks.Core.Models.ExportRecord", b =>
+ {
+ b.Navigation("BomItems");
+ });
+#pragma warning restore 612, 618
+ }
+ }
+}