Fix build and code structure improvements. New but essential UI functionality. CI improvements. Documentation improvements. AI module improvements.

This commit is contained in:
StellaOps Bot
2025-12-26 21:54:17 +02:00
parent 335ff7da16
commit c2b9cd8d1f
3717 changed files with 264714 additions and 48202 deletions

View File

@@ -0,0 +1,379 @@
Microsoft Visual Studio Solution File, Format Version 12.00
# Visual Studio Version 17
VisualStudioVersion = 17.0.31903.59
MinimumVisualStudioVersion = 10.0.40219.1
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "StellaOps.IssuerDirectory", "StellaOps.IssuerDirectory", "{75E942AC-399F-FD3A-327B-F96331A1E421}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "StellaOps.IssuerDirectory.Core", "StellaOps.IssuerDirectory.Core", "{BA238A15-0667-90EF-4042-5796AE165618}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "StellaOps.IssuerDirectory.Core.Tests", "StellaOps.IssuerDirectory.Core.Tests", "{F7673C2F-B609-9794-F1A0-68CF08485B48}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "StellaOps.IssuerDirectory.Infrastructure", "StellaOps.IssuerDirectory.Infrastructure", "{76C4B0C3-0C95-9EAD-D52B-B8EACB9E9F00}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "StellaOps.IssuerDirectory.WebService", "StellaOps.IssuerDirectory.WebService", "{4DF82FED-CD90-ACCE-70F6-48A715296084}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "__External", "__External", "{5B52EF8A-3661-DCFF-797D-BC4D6AC60BDA}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Authority", "Authority", "{C1DCEFBD-12A5-EAAE-632E-8EEB9BE491B6}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "StellaOps.Authority", "StellaOps.Authority", "{A6928CBA-4D4D-AB2B-CA19-FEE6E73ECE70}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "StellaOps.Auth.Abstractions", "StellaOps.Auth.Abstractions", "{F2E6CB0E-DF77-1FAA-582B-62B040DF3848}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "StellaOps.Auth.ServerIntegration", "StellaOps.Auth.ServerIntegration", "{7E890DF9-B715-B6DF-2498-FD74DDA87D71}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "StellaOps.Authority.Plugins.Abstractions", "StellaOps.Authority.Plugins.Abstractions", "{64689413-46D7-8499-68A6-B6367ACBC597}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Router", "Router", "{FC018E5B-1E2F-DE19-1E97-0C845058C469}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "__Libraries", "__Libraries", "{1BE5B76C-B486-560B-6CB2-44C6537249AA}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "StellaOps.Microservice", "StellaOps.Microservice", "{3DE1DCDC-C845-4AC7-7B66-34B0A9E8626B}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "StellaOps.Microservice.AspNetCore", "StellaOps.Microservice.AspNetCore", "{6FA01E92-606B-0CB8-8583-6F693A903CFC}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "StellaOps.Router.AspNet", "StellaOps.Router.AspNet", "{A5994E92-7E0E-89FE-5628-DE1A0176B8BA}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "StellaOps.Router.Common", "StellaOps.Router.Common", "{54C11B29-4C54-7255-AB44-BEB63AF9BD1F}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "__Libraries", "__Libraries", "{1345DD29-BB3A-FB5F-4B3D-E29F6045A27A}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "StellaOps.Canonical.Json", "StellaOps.Canonical.Json", "{79E122F4-2325-3E92-438E-5825A307B594}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "StellaOps.Configuration", "StellaOps.Configuration", "{538E2D98-5325-3F54-BE74-EFE5FC1ECBD8}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "StellaOps.Cryptography", "StellaOps.Cryptography", "{66557252-B5C4-664B-D807-07018C627474}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "StellaOps.Cryptography.DependencyInjection", "StellaOps.Cryptography.DependencyInjection", "{7203223D-FF02-7BEB-2798-D1639ACC01C4}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "StellaOps.Cryptography.Plugin.CryptoPro", "StellaOps.Cryptography.Plugin.CryptoPro", "{3C69853C-90E3-D889-1960-3B9229882590}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "StellaOps.Cryptography.Plugin.OpenSslGost", "StellaOps.Cryptography.Plugin.OpenSslGost", "{643E4D4C-BC96-A37F-E0EC-488127F0B127}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "StellaOps.Cryptography.Plugin.Pkcs11Gost", "StellaOps.Cryptography.Plugin.Pkcs11Gost", "{6F2CA7F5-3E7C-C61B-94E6-E7DD1227B5B1}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "StellaOps.Cryptography.Plugin.PqSoft", "StellaOps.Cryptography.Plugin.PqSoft", "{F04B7DBB-77A5-C978-B2DE-8C189A32AA72}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "StellaOps.Cryptography.Plugin.SimRemote", "StellaOps.Cryptography.Plugin.SimRemote", "{7C72F22A-20FF-DF5B-9191-6DFD0D497DB2}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "StellaOps.Cryptography.Plugin.SmRemote", "StellaOps.Cryptography.Plugin.SmRemote", "{C896CC0A-F5E6-9AA4-C582-E691441F8D32}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "StellaOps.Cryptography.Plugin.SmSoft", "StellaOps.Cryptography.Plugin.SmSoft", "{0AA3A418-AB45-CCA4-46D4-EEBFE011FECA}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "StellaOps.Cryptography.Plugin.WineCsp", "StellaOps.Cryptography.Plugin.WineCsp", "{225D9926-4AE8-E539-70AD-8698E688F271}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "StellaOps.Cryptography.PluginLoader", "StellaOps.Cryptography.PluginLoader", "{D6E8E69C-F721-BBCB-8C39-9716D53D72AD}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "StellaOps.DependencyInjection", "StellaOps.DependencyInjection", "{589A43FD-8213-E9E3-6CFF-9CBA72D53E98}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "StellaOps.Infrastructure.EfCore", "StellaOps.Infrastructure.EfCore", "{FCD529E0-DD17-6587-B29C-12D425C0AD0C}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "StellaOps.Infrastructure.Postgres", "StellaOps.Infrastructure.Postgres", "{61B23570-4F2D-B060-BE1F-37995682E494}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "StellaOps.IssuerDirectory.Client", "StellaOps.IssuerDirectory.Client", "{F4D43AC8-DDB8-E523-449D-D1B438713F12}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "StellaOps.Plugin", "StellaOps.Plugin", "{772B02B5-6280-E1D4-3E2E-248D0455C2FB}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "StellaOps.TestKit", "StellaOps.TestKit", "{8380A20C-A5B8-EE91-1A58-270323688CB9}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "__Tests", "__Tests", "{90659617-4DF7-809A-4E5B-29BB5A98E8E1}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "__Libraries", "__Libraries", "{AB8B269C-5A2A-A4B8-0488-B5F81E55B4D9}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "StellaOps.Infrastructure.Postgres.Testing", "StellaOps.Infrastructure.Postgres.Testing", "{CEDC2447-F717-3C95-7E08-F214D575A7B7}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "__Libraries", "__Libraries", "{A5C98087-E847-D2C4-2143-20869479839D}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "StellaOps.IssuerDirectory.Persistence", "StellaOps.IssuerDirectory.Persistence", "{EF65A356-0E2C-ADEC-6516-E5367F5F675F}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "__Tests", "__Tests", "{BB76B5A5-14BA-E317-828D-110B711D71F5}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "StellaOps.IssuerDirectory.Persistence.Tests", "StellaOps.IssuerDirectory.Persistence.Tests", "{FB6B89EB-69C4-1C97-A590-587BCE5244EB}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Auth.Abstractions", "E:\dev\git.stella-ops.org\src\Authority\StellaOps.Authority\StellaOps.Auth.Abstractions\StellaOps.Auth.Abstractions.csproj", "{55D9B653-FB76-FCE8-1A3C-67B1BEDEC214}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Auth.ServerIntegration", "E:\dev\git.stella-ops.org\src\Authority\StellaOps.Authority\StellaOps.Auth.ServerIntegration\StellaOps.Auth.ServerIntegration.csproj", "{ECA25786-A3A8-92C4-4AA3-D4A73C69FDCA}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Authority.Plugins.Abstractions", "E:\dev\git.stella-ops.org\src\Authority\StellaOps.Authority\StellaOps.Authority.Plugins.Abstractions\StellaOps.Authority.Plugins.Abstractions.csproj", "{97F94029-5419-6187-5A63-5C8FD9232FAE}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Canonical.Json", "E:\dev\git.stella-ops.org\src\__Libraries\StellaOps.Canonical.Json\StellaOps.Canonical.Json.csproj", "{AF9E7F02-25AD-3540-18D7-F6A4F8BA5A60}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Configuration", "E:\dev\git.stella-ops.org\src\__Libraries\StellaOps.Configuration\StellaOps.Configuration.csproj", "{92C62F7B-8028-6EE1-B71B-F45F459B8E97}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Cryptography", "E:\dev\git.stella-ops.org\src\__Libraries\StellaOps.Cryptography\StellaOps.Cryptography.csproj", "{F664A948-E352-5808-E780-77A03F19E93E}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Cryptography.DependencyInjection", "E:\dev\git.stella-ops.org\src\__Libraries\StellaOps.Cryptography.DependencyInjection\StellaOps.Cryptography.DependencyInjection.csproj", "{FA83F778-5252-0B80-5555-E69F790322EA}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Cryptography.Plugin.CryptoPro", "E:\dev\git.stella-ops.org\src\__Libraries\StellaOps.Cryptography.Plugin.CryptoPro\StellaOps.Cryptography.Plugin.CryptoPro.csproj", "{C53E0895-879A-D9E6-0A43-24AD17A2F270}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Cryptography.Plugin.OpenSslGost", "E:\dev\git.stella-ops.org\src\__Libraries\StellaOps.Cryptography.Plugin.OpenSslGost\StellaOps.Cryptography.Plugin.OpenSslGost.csproj", "{0AED303F-69E6-238F-EF80-81985080EDB7}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Cryptography.Plugin.Pkcs11Gost", "E:\dev\git.stella-ops.org\src\__Libraries\StellaOps.Cryptography.Plugin.Pkcs11Gost\StellaOps.Cryptography.Plugin.Pkcs11Gost.csproj", "{2904D288-CE64-A565-2C46-C2E85A96A1EE}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Cryptography.Plugin.PqSoft", "E:\dev\git.stella-ops.org\src\__Libraries\StellaOps.Cryptography.Plugin.PqSoft\StellaOps.Cryptography.Plugin.PqSoft.csproj", "{A6667CC3-B77F-023E-3A67-05F99E9FF46A}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Cryptography.Plugin.SimRemote", "E:\dev\git.stella-ops.org\src\__Libraries\StellaOps.Cryptography.Plugin.SimRemote\StellaOps.Cryptography.Plugin.SimRemote.csproj", "{A26E2816-F787-F76B-1D6C-E086DD3E19CE}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Cryptography.Plugin.SmRemote", "E:\dev\git.stella-ops.org\src\__Libraries\StellaOps.Cryptography.Plugin.SmRemote\StellaOps.Cryptography.Plugin.SmRemote.csproj", "{B3DEC619-67AC-1B5A-4F3E-A1F24C3F6877}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Cryptography.Plugin.SmSoft", "E:\dev\git.stella-ops.org\src\__Libraries\StellaOps.Cryptography.Plugin.SmSoft\StellaOps.Cryptography.Plugin.SmSoft.csproj", "{90DB65B4-8F6E-FB8E-0281-505AD8BC6BA6}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Cryptography.Plugin.WineCsp", "E:\dev\git.stella-ops.org\src\__Libraries\StellaOps.Cryptography.Plugin.WineCsp\StellaOps.Cryptography.Plugin.WineCsp.csproj", "{059FBB86-DEE6-8207-3F23-2A1A3EC00DEA}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Cryptography.PluginLoader", "E:\dev\git.stella-ops.org\src\__Libraries\StellaOps.Cryptography.PluginLoader\StellaOps.Cryptography.PluginLoader.csproj", "{8BBA3159-C4CC-F685-A28C-7FE6CBD3D2A1}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.DependencyInjection", "E:\dev\git.stella-ops.org\src\__Libraries\StellaOps.DependencyInjection\StellaOps.DependencyInjection.csproj", "{632A1F0D-1BA5-C84B-B716-2BE638A92780}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Infrastructure.EfCore", "E:\dev\git.stella-ops.org\src\__Libraries\StellaOps.Infrastructure.EfCore\StellaOps.Infrastructure.EfCore.csproj", "{A63897D9-9531-989B-7309-E384BCFC2BB9}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Infrastructure.Postgres", "E:\dev\git.stella-ops.org\src\__Libraries\StellaOps.Infrastructure.Postgres\StellaOps.Infrastructure.Postgres.csproj", "{8C594D82-3463-3367-4F06-900AC707753D}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Infrastructure.Postgres.Testing", "E:\dev\git.stella-ops.org\src\__Tests\__Libraries\StellaOps.Infrastructure.Postgres.Testing\StellaOps.Infrastructure.Postgres.Testing.csproj", "{52F400CD-D473-7A1F-7986-89011CD2A887}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.IssuerDirectory.Client", "E:\dev\git.stella-ops.org\src\__Libraries\StellaOps.IssuerDirectory.Client\StellaOps.IssuerDirectory.Client.csproj", "{A0F46FA3-7796-5830-56F9-380D60D1AAA3}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.IssuerDirectory.Core", "StellaOps.IssuerDirectory\StellaOps.IssuerDirectory.Core\StellaOps.IssuerDirectory.Core.csproj", "{F98D6028-FAFF-2A7B-C540-EA73C74CF059}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.IssuerDirectory.Core.Tests", "StellaOps.IssuerDirectory\StellaOps.IssuerDirectory.Core.Tests\StellaOps.IssuerDirectory.Core.Tests.csproj", "{8CAEF4CA-4CF8-77B0-7B61-2519E8E35FFA}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.IssuerDirectory.Infrastructure", "StellaOps.IssuerDirectory\StellaOps.IssuerDirectory.Infrastructure\StellaOps.IssuerDirectory.Infrastructure.csproj", "{20C2A7EF-AA5F-79CE-813F-5EFB3D2DAE82}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.IssuerDirectory.Persistence", "__Libraries\StellaOps.IssuerDirectory.Persistence\StellaOps.IssuerDirectory.Persistence.csproj", "{1B4F6879-6791-E78E-3622-7CE094FE34A7}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.IssuerDirectory.Persistence.Tests", "__Tests\StellaOps.IssuerDirectory.Persistence.Tests\StellaOps.IssuerDirectory.Persistence.Tests.csproj", "{F00467DF-5759-9B2F-8A19-B571764F6EAE}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.IssuerDirectory.WebService", "StellaOps.IssuerDirectory\StellaOps.IssuerDirectory.WebService\StellaOps.IssuerDirectory.WebService.csproj", "{FF4E7BB2-C27F-7FF5-EE7C-99A15CB55418}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Microservice", "E:\dev\git.stella-ops.org\src\Router\__Libraries\StellaOps.Microservice\StellaOps.Microservice.csproj", "{BAD08D96-A80A-D27F-5D9C-656AEEB3D568}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Microservice.AspNetCore", "E:\dev\git.stella-ops.org\src\Router\__Libraries\StellaOps.Microservice.AspNetCore\StellaOps.Microservice.AspNetCore.csproj", "{F63694F1-B56D-6E72-3F5D-5D38B1541F0F}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Plugin", "E:\dev\git.stella-ops.org\src\__Libraries\StellaOps.Plugin\StellaOps.Plugin.csproj", "{38A9EE9B-6FC8-93BC-0D43-2A906E678D66}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Router.AspNet", "E:\dev\git.stella-ops.org\src\Router\__Libraries\StellaOps.Router.AspNet\StellaOps.Router.AspNet.csproj", "{79104479-B087-E5D0-5523-F1803282A246}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.Router.Common", "E:\dev\git.stella-ops.org\src\Router\__Libraries\StellaOps.Router.Common\StellaOps.Router.Common.csproj", "{F17A6F0B-3120-2BA9-84D8-5F8BA0B9705D}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.TestKit", "E:\dev\git.stella-ops.org\src\__Libraries\StellaOps.TestKit\StellaOps.TestKit.csproj", "{AF043113-CCE3-59C1-DF71-9804155F26A8}"
EndProject
Global
GlobalSection(SolutionConfigurationPlatforms) = preSolution
Debug|Any CPU = Debug|Any CPU
Release|Any CPU = Release|Any CPU
EndGlobalSection
GlobalSection(ProjectConfigurationPlatforms) = postSolution
{55D9B653-FB76-FCE8-1A3C-67B1BEDEC214}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{55D9B653-FB76-FCE8-1A3C-67B1BEDEC214}.Debug|Any CPU.Build.0 = Debug|Any CPU
{55D9B653-FB76-FCE8-1A3C-67B1BEDEC214}.Release|Any CPU.ActiveCfg = Release|Any CPU
{55D9B653-FB76-FCE8-1A3C-67B1BEDEC214}.Release|Any CPU.Build.0 = Release|Any CPU
{ECA25786-A3A8-92C4-4AA3-D4A73C69FDCA}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{ECA25786-A3A8-92C4-4AA3-D4A73C69FDCA}.Debug|Any CPU.Build.0 = Debug|Any CPU
{ECA25786-A3A8-92C4-4AA3-D4A73C69FDCA}.Release|Any CPU.ActiveCfg = Release|Any CPU
{ECA25786-A3A8-92C4-4AA3-D4A73C69FDCA}.Release|Any CPU.Build.0 = Release|Any CPU
{97F94029-5419-6187-5A63-5C8FD9232FAE}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{97F94029-5419-6187-5A63-5C8FD9232FAE}.Debug|Any CPU.Build.0 = Debug|Any CPU
{97F94029-5419-6187-5A63-5C8FD9232FAE}.Release|Any CPU.ActiveCfg = Release|Any CPU
{97F94029-5419-6187-5A63-5C8FD9232FAE}.Release|Any CPU.Build.0 = Release|Any CPU
{AF9E7F02-25AD-3540-18D7-F6A4F8BA5A60}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{AF9E7F02-25AD-3540-18D7-F6A4F8BA5A60}.Debug|Any CPU.Build.0 = Debug|Any CPU
{AF9E7F02-25AD-3540-18D7-F6A4F8BA5A60}.Release|Any CPU.ActiveCfg = Release|Any CPU
{AF9E7F02-25AD-3540-18D7-F6A4F8BA5A60}.Release|Any CPU.Build.0 = Release|Any CPU
{92C62F7B-8028-6EE1-B71B-F45F459B8E97}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{92C62F7B-8028-6EE1-B71B-F45F459B8E97}.Debug|Any CPU.Build.0 = Debug|Any CPU
{92C62F7B-8028-6EE1-B71B-F45F459B8E97}.Release|Any CPU.ActiveCfg = Release|Any CPU
{92C62F7B-8028-6EE1-B71B-F45F459B8E97}.Release|Any CPU.Build.0 = Release|Any CPU
{F664A948-E352-5808-E780-77A03F19E93E}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{F664A948-E352-5808-E780-77A03F19E93E}.Debug|Any CPU.Build.0 = Debug|Any CPU
{F664A948-E352-5808-E780-77A03F19E93E}.Release|Any CPU.ActiveCfg = Release|Any CPU
{F664A948-E352-5808-E780-77A03F19E93E}.Release|Any CPU.Build.0 = Release|Any CPU
{FA83F778-5252-0B80-5555-E69F790322EA}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{FA83F778-5252-0B80-5555-E69F790322EA}.Debug|Any CPU.Build.0 = Debug|Any CPU
{FA83F778-5252-0B80-5555-E69F790322EA}.Release|Any CPU.ActiveCfg = Release|Any CPU

View File

@@ -1,4 +1,4 @@
using System;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Net;
@@ -93,7 +93,6 @@ public class IssuerDirectoryClientTests
reasonValues!.Should().Equal("rollout");
using var document = JsonDocument.Parse(putRequest.Body ?? string.Empty);
using StellaOps.TestKit;
var root = document.RootElement;
root.GetProperty("weight").GetDecimal().Should().Be(1.5m);
root.GetProperty("reason").GetString().Should().Be("rollout");

View File

@@ -8,11 +8,11 @@
<TreatWarningsAsErrors>false</TreatWarningsAsErrors>
</PropertyGroup>
<ItemGroup>
<PackageReference Include="FluentAssertions" Version="6.12.0" />
<PackageReference Include="FluentAssertions" />
</ItemGroup>
<ItemGroup>
<ProjectReference Include="..\StellaOps.IssuerDirectory.Core\StellaOps.IssuerDirectory.Core.csproj" />
<ProjectReference Include="..\..\..\__Libraries\StellaOps.IssuerDirectory.Client\StellaOps.IssuerDirectory.Client.csproj" />
<ProjectReference Include="../../../__Libraries/StellaOps.TestKit/StellaOps.TestKit.csproj" />
</ItemGroup>
</Project>
</Project>

View File

@@ -7,6 +7,6 @@
<TreatWarningsAsErrors>false</TreatWarningsAsErrors>
</PropertyGroup>
<ItemGroup>
<PackageReference Include="Microsoft.Extensions.Logging.Abstractions" Version="10.0.0" />
<PackageReference Include="Microsoft.Extensions.Logging.Abstractions" />
</ItemGroup>
</Project>

View File

@@ -7,10 +7,10 @@
<TreatWarningsAsErrors>false</TreatWarningsAsErrors>
</PropertyGroup>
<ItemGroup>
<PackageReference Include="Microsoft.Extensions.Configuration.Abstractions" Version="10.0.0" />
<PackageReference Include="Microsoft.Extensions.DependencyInjection.Abstractions" Version="10.0.0" />
<PackageReference Include="Microsoft.Extensions.Logging.Abstractions" Version="10.0.0" />
<PackageReference Include="Microsoft.Extensions.Options.ConfigurationExtensions" Version="10.0.0" />
<PackageReference Include="Microsoft.Extensions.Configuration.Abstractions" />
<PackageReference Include="Microsoft.Extensions.DependencyInjection.Abstractions" />
<PackageReference Include="Microsoft.Extensions.Logging.Abstractions" />
<PackageReference Include="Microsoft.Extensions.Options.ConfigurationExtensions" />
</ItemGroup>
<ItemGroup>
<ProjectReference Include="..\\StellaOps.IssuerDirectory.Core\\StellaOps.IssuerDirectory.Core.csproj" />

View File

@@ -1,31 +0,0 @@
<?xml version="1.0" ?>
<Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup>
<TargetFramework>net10.0</TargetFramework>
<LangVersion>preview</LangVersion>
<Nullable>enable</Nullable>
<ImplicitUsings>enable</ImplicitUsings>
<TreatWarningsAsErrors>false</TreatWarningsAsErrors>
<RootNamespace>StellaOps.IssuerDirectory.Storage.Postgres</RootNamespace>
<AssemblyName>StellaOps.IssuerDirectory.Storage.Postgres</AssemblyName>
<Description>PostgreSQL storage implementation for IssuerDirectory module</Description>
</PropertyGroup>
<ItemGroup>
<PackageReference Include="Microsoft.Extensions.DependencyInjection.Abstractions" Version="10.0.0" />
<PackageReference Include="Microsoft.Extensions.Logging.Abstractions" Version="10.0.0" />
<PackageReference Include="Microsoft.Extensions.Options" Version="10.0.0" />
<PackageReference Include="Npgsql" Version="9.0.2" />
</ItemGroup>
<ItemGroup>
<ProjectReference Include="..\StellaOps.IssuerDirectory.Core\StellaOps.IssuerDirectory.Core.csproj" />
<ProjectReference Include="..\..\..\__Libraries\StellaOps.Infrastructure.Postgres\StellaOps.Infrastructure.Postgres.csproj" />
</ItemGroup>
<ItemGroup>
<EmbeddedResource Include="Migrations\*.sql" LogicalName="%(Filename)%(Extension)" />
</ItemGroup>
</Project>

View File

@@ -16,7 +16,8 @@ using StellaOps.Configuration;
using StellaOps.IssuerDirectory.Core.Services;
using StellaOps.Infrastructure.Postgres.Options;
using StellaOps.IssuerDirectory.Infrastructure;
using StellaOps.IssuerDirectory.Storage.Postgres;
using StellaOps.IssuerDirectory.Persistence.Extensions;
using StellaOps.IssuerDirectory.Persistence.Postgres;
using StellaOps.IssuerDirectory.Infrastructure.Seed;
using StellaOps.IssuerDirectory.WebService.Endpoints;
using StellaOps.IssuerDirectory.WebService.Options;
@@ -138,7 +139,7 @@ static void ConfigurePersistence(
if (provider == "postgres")
{
Log.Information("Using PostgreSQL persistence for IssuerDirectory.");
builder.Services.AddIssuerDirectoryPostgresStorage(new PostgresOptions
builder.Services.AddIssuerDirectoryPersistence(new PostgresOptions
{
ConnectionString = options.Persistence.PostgresConnectionString,
SchemaName = "issuer"

View File

@@ -0,0 +1,12 @@
{
"profiles": {
"StellaOps.IssuerDirectory.WebService": {
"commandName": "Project",
"launchBrowser": true,
"environmentVariables": {
"ASPNETCORE_ENVIRONMENT": "Development"
},
"applicationUrl": "https://localhost:62527;http://localhost:62528"
}
}
}

View File

@@ -7,22 +7,22 @@
<TreatWarningsAsErrors>false</TreatWarningsAsErrors>
</PropertyGroup>
<ItemGroup>
<PackageReference Include="Microsoft.AspNetCore.OpenApi" Version="10.0.0" />
<PackageReference Include="OpenTelemetry.Extensions.Hosting" Version="1.12.0" />
<PackageReference Include="OpenTelemetry.Instrumentation.AspNetCore" Version="1.12.0" />
<PackageReference Include="OpenTelemetry.Instrumentation.Http" Version="1.12.0" />
<PackageReference Include="OpenTelemetry.Instrumentation.Runtime" Version="1.12.0" />
<PackageReference Include="Serilog.AspNetCore" Version="8.0.1" />
<PackageReference Include="Serilog.Sinks.Console" Version="5.0.1" />
<PackageReference Include="Microsoft.AspNetCore.OpenApi" />
<PackageReference Include="OpenTelemetry.Extensions.Hosting" />
<PackageReference Include="OpenTelemetry.Instrumentation.AspNetCore" />
<PackageReference Include="OpenTelemetry.Instrumentation.Http" />
<PackageReference Include="OpenTelemetry.Instrumentation.Runtime" />
<PackageReference Include="Serilog.AspNetCore" />
<PackageReference Include="Serilog.Sinks.Console" />
</ItemGroup>
<ItemGroup>
<ProjectReference Include="..\\StellaOps.IssuerDirectory.Core\\StellaOps.IssuerDirectory.Core.csproj" />
<ProjectReference Include="..\\StellaOps.IssuerDirectory.Infrastructure\\StellaOps.IssuerDirectory.Infrastructure.csproj" />
<ProjectReference Include="..\\StellaOps.IssuerDirectory.Storage.Postgres\\StellaOps.IssuerDirectory.Storage.Postgres.csproj" />
<ProjectReference Include="..\\..\\__Libraries\\StellaOps.IssuerDirectory.Persistence\\StellaOps.IssuerDirectory.Persistence.csproj" />
<ProjectReference Include="..\\..\\..\\Authority\\StellaOps.Authority\\StellaOps.Auth.Abstractions\\StellaOps.Auth.Abstractions.csproj" />
<ProjectReference Include="..\\..\\..\\Authority\\StellaOps.Authority\\StellaOps.Auth.ServerIntegration\\StellaOps.Auth.ServerIntegration.csproj" />
<ProjectReference Include="..\\..\\..\\__Libraries\\StellaOps.Configuration\\StellaOps.Configuration.csproj" />
<ProjectReference Include="..\\..\\..\\__Libraries\\StellaOps.Router.AspNet\\StellaOps.Router.AspNet.csproj" />
<ProjectReference Include="..\..\..\Router\__Libraries\StellaOps.Router.AspNet\StellaOps.Router.AspNet.csproj" />
</ItemGroup>
<ItemGroup>
<Content Include="..\\data\\csaf-publishers.json">

View File

@@ -1,49 +0,0 @@

Microsoft Visual Studio Solution File, Format Version 12.00
# Visual Studio Version 17
VisualStudioVersion = 17.0.31903.59
MinimumVisualStudioVersion = 10.0.40219.1
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.IssuerDirectory.Core", "StellaOps.IssuerDirectory.Core\StellaOps.IssuerDirectory.Core.csproj", "{298FE21A-B406-486C-972C-E8CE6FE81D38}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.IssuerDirectory.Infrastructure", "StellaOps.IssuerDirectory.Infrastructure\StellaOps.IssuerDirectory.Infrastructure.csproj", "{0F76EF16-3194-4127-BC50-15F01E48F2B7}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.IssuerDirectory.WebService", "StellaOps.IssuerDirectory.WebService\StellaOps.IssuerDirectory.WebService.csproj", "{8ECE3570-9BA0-470B-A8E3-C244F6AAEF92}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.IssuerDirectory.Core.Tests", "StellaOps.IssuerDirectory.Core.Tests\StellaOps.IssuerDirectory.Core.Tests.csproj", "{22842BC6-D909-4919-8FB1-B2C3ED7E4DDE}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StellaOps.IssuerDirectory.Storage.Postgres", "StellaOps.IssuerDirectory.Storage.Postgres\StellaOps.IssuerDirectory.Storage.Postgres.csproj", "{A1B2C3D4-E5F6-7890-ABCD-EF1234567890}"
EndProject
Global
GlobalSection(SolutionConfigurationPlatforms) = preSolution
Debug|Any CPU = Debug|Any CPU
Release|Any CPU = Release|Any CPU
EndGlobalSection
GlobalSection(ProjectConfigurationPlatforms) = postSolution
{298FE21A-B406-486C-972C-E8CE6FE81D38}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{298FE21A-B406-486C-972C-E8CE6FE81D38}.Debug|Any CPU.Build.0 = Debug|Any CPU
{298FE21A-B406-486C-972C-E8CE6FE81D38}.Release|Any CPU.ActiveCfg = Release|Any CPU
{298FE21A-B406-486C-972C-E8CE6FE81D38}.Release|Any CPU.Build.0 = Release|Any CPU
{0F76EF16-3194-4127-BC50-15F01E48F2B7}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{0F76EF16-3194-4127-BC50-15F01E48F2B7}.Debug|Any CPU.Build.0 = Debug|Any CPU
{0F76EF16-3194-4127-BC50-15F01E48F2B7}.Release|Any CPU.ActiveCfg = Release|Any CPU
{0F76EF16-3194-4127-BC50-15F01E48F2B7}.Release|Any CPU.Build.0 = Release|Any CPU
{8ECE3570-9BA0-470B-A8E3-C244F6AAEF92}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{8ECE3570-9BA0-470B-A8E3-C244F6AAEF92}.Debug|Any CPU.Build.0 = Debug|Any CPU
{8ECE3570-9BA0-470B-A8E3-C244F6AAEF92}.Release|Any CPU.ActiveCfg = Release|Any CPU
{8ECE3570-9BA0-470B-A8E3-C244F6AAEF92}.Release|Any CPU.Build.0 = Release|Any CPU
{22842BC6-D909-4919-8FB1-B2C3ED7E4DDE}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{22842BC6-D909-4919-8FB1-B2C3ED7E4DDE}.Debug|Any CPU.Build.0 = Debug|Any CPU
{22842BC6-D909-4919-8FB1-B2C3ED7E4DDE}.Release|Any CPU.ActiveCfg = Release|Any CPU
{22842BC6-D909-4919-8FB1-B2C3ED7E4DDE}.Release|Any CPU.Build.0 = Release|Any CPU
{A1B2C3D4-E5F6-7890-ABCD-EF1234567890}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{A1B2C3D4-E5F6-7890-ABCD-EF1234567890}.Debug|Any CPU.Build.0 = Debug|Any CPU
{A1B2C3D4-E5F6-7890-ABCD-EF1234567890}.Release|Any CPU.ActiveCfg = Release|Any CPU
{A1B2C3D4-E5F6-7890-ABCD-EF1234567890}.Release|Any CPU.Build.0 = Release|Any CPU
EndGlobalSection
GlobalSection(SolutionProperties) = preSolution
HideSolutionNode = FALSE
EndGlobalSection
GlobalSection(ExtensibilityGlobals) = postSolution
SolutionGuid = {291CD30E-130B-4349-AD46-80801170D9F5}
EndGlobalSection
EndGlobal

View File

@@ -1,28 +0,0 @@
using System.Reflection;
using StellaOps.IssuerDirectory.Storage.Postgres;
using StellaOps.Infrastructure.Postgres.Testing;
using Xunit;
namespace StellaOps.IssuerDirectory.Storage.Postgres.Tests;
/// <summary>
/// PostgreSQL integration test fixture for the IssuerDirectory module.
/// Runs migrations from embedded resources and provides test isolation.
/// </summary>
public sealed class IssuerDirectoryPostgresFixture : PostgresIntegrationFixture, ICollectionFixture<IssuerDirectoryPostgresFixture>
{
protected override Assembly? GetMigrationAssembly()
=> typeof(IssuerDirectoryDataSource).Assembly;
protected override string GetModuleName() => "IssuerDirectory";
}
/// <summary>
/// Collection definition for IssuerDirectory PostgreSQL integration tests.
/// Tests in this collection share a single PostgreSQL container instance.
/// </summary>
[CollectionDefinition(Name)]
public sealed class IssuerDirectoryPostgresCollection : ICollectionFixture<IssuerDirectoryPostgresFixture>
{
public const string Name = "IssuerDirectoryPostgres";
}

View File

@@ -1,301 +0,0 @@
using FluentAssertions;
using Microsoft.Extensions.Logging.Abstractions;
using StellaOps.Infrastructure.Postgres.Options;
using StellaOps.IssuerDirectory.Core.Domain;
using StellaOps.IssuerDirectory.Storage.Postgres.Repositories;
using Xunit;
using StellaOps.TestKit;
namespace StellaOps.IssuerDirectory.Storage.Postgres.Tests;
[Collection(IssuerDirectoryPostgresCollection.Name)]
public sealed class IssuerKeyRepositoryTests : IAsyncLifetime
{
private readonly IssuerDirectoryPostgresFixture _fixture;
private readonly PostgresIssuerRepository _issuerRepository;
private readonly PostgresIssuerKeyRepository _keyRepository;
private readonly string _tenantId = Guid.NewGuid().ToString();
private string _issuerId = null!;
public IssuerKeyRepositoryTests(IssuerDirectoryPostgresFixture fixture)
{
_fixture = fixture;
var options = new PostgresOptions
{
ConnectionString = fixture.ConnectionString,
SchemaName = fixture.SchemaName
};
var dataSource = new IssuerDirectoryDataSource(options, NullLogger<IssuerDirectoryDataSource>.Instance);
_issuerRepository = new PostgresIssuerRepository(dataSource, NullLogger<PostgresIssuerRepository>.Instance);
_keyRepository = new PostgresIssuerKeyRepository(dataSource, NullLogger<PostgresIssuerKeyRepository>.Instance);
}
public async Task InitializeAsync()
{
await _fixture.TruncateAllTablesAsync();
_issuerId = await SeedIssuerAsync();
}
public Task DisposeAsync() => Task.CompletedTask;
[Trait("Category", TestCategories.Unit)]
[Fact]
public async Task UpsertAsync_CreatesNewKey()
{
var keyRecord = CreateKeyRecord("key-001", IssuerKeyType.Ed25519PublicKey);
await _keyRepository.UpsertAsync(keyRecord, CancellationToken.None);
var fetched = await _keyRepository.GetAsync(_tenantId, _issuerId, "key-001", CancellationToken.None);
fetched.Should().NotBeNull();
fetched!.Id.Should().Be("key-001");
fetched.Type.Should().Be(IssuerKeyType.Ed25519PublicKey);
fetched.Status.Should().Be(IssuerKeyStatus.Active);
}
[Trait("Category", TestCategories.Unit)]
[Fact]
public async Task UpsertAsync_UpdatesExistingKey()
{
var keyRecord = CreateKeyRecord("key-update", IssuerKeyType.Ed25519PublicKey);
await _keyRepository.UpsertAsync(keyRecord, CancellationToken.None);
var updated = keyRecord with
{
Status = IssuerKeyStatus.Retired,
RetiredAtUtc = DateTimeOffset.UtcNow,
UpdatedAtUtc = DateTimeOffset.UtcNow,
UpdatedBy = "admin@test.com"
};
await _keyRepository.UpsertAsync(updated, CancellationToken.None);
var fetched = await _keyRepository.GetAsync(_tenantId, _issuerId, "key-update", CancellationToken.None);
fetched.Should().NotBeNull();
fetched!.Status.Should().Be(IssuerKeyStatus.Retired);
fetched.RetiredAtUtc.Should().NotBeNull();
}
[Trait("Category", TestCategories.Unit)]
[Fact]
public async Task GetAsync_ReturnsNullForNonExistentKey()
{
var result = await _keyRepository.GetAsync(_tenantId, _issuerId, "nonexistent", CancellationToken.None);
result.Should().BeNull();
}
[Trait("Category", TestCategories.Unit)]
[Fact]
public async Task GetByFingerprintAsync_ReturnsKey()
{
var fingerprint = $"fp_{Guid.NewGuid():N}";
var keyRecord = CreateKeyRecord("key-fp", IssuerKeyType.Ed25519PublicKey) with { Fingerprint = fingerprint };
await _keyRepository.UpsertAsync(keyRecord, CancellationToken.None);
var fetched = await _keyRepository.GetByFingerprintAsync(_tenantId, _issuerId, fingerprint, CancellationToken.None);
fetched.Should().NotBeNull();
fetched!.Fingerprint.Should().Be(fingerprint);
}
[Trait("Category", TestCategories.Unit)]
[Fact]
public async Task ListAsync_ReturnsAllKeysForIssuer()
{
var key1 = CreateKeyRecord("key-list-1", IssuerKeyType.Ed25519PublicKey);
var key2 = CreateKeyRecord("key-list-2", IssuerKeyType.X509Certificate);
await _keyRepository.UpsertAsync(key1, CancellationToken.None);
await _keyRepository.UpsertAsync(key2, CancellationToken.None);
var results = await _keyRepository.ListAsync(_tenantId, _issuerId, CancellationToken.None);
results.Should().HaveCount(2);
results.Select(k => k.Id).Should().BeEquivalentTo(["key-list-1", "key-list-2"]);
}
[Trait("Category", TestCategories.Unit)]
[Fact]
public async Task ListGlobalAsync_ReturnsGlobalKeys()
{
var globalIssuerId = await SeedGlobalIssuerAsync();
var globalKey = CreateKeyRecord("global-key", IssuerKeyType.Ed25519PublicKey, globalIssuerId, IssuerTenants.Global);
await _keyRepository.UpsertAsync(globalKey, CancellationToken.None);
var results = await _keyRepository.ListGlobalAsync(globalIssuerId, CancellationToken.None);
results.Should().Contain(k => k.Id == "global-key");
}
[Trait("Category", TestCategories.Unit)]
[Fact]
public async Task UpsertAsync_PersistsKeyTypeEd25519()
{
var keyRecord = CreateKeyRecord("key-ed25519", IssuerKeyType.Ed25519PublicKey);
await _keyRepository.UpsertAsync(keyRecord, CancellationToken.None);
var fetched = await _keyRepository.GetAsync(_tenantId, _issuerId, "key-ed25519", CancellationToken.None);
fetched.Should().NotBeNull();
fetched!.Type.Should().Be(IssuerKeyType.Ed25519PublicKey);
}
[Trait("Category", TestCategories.Unit)]
[Fact]
public async Task UpsertAsync_PersistsKeyTypeX509()
{
var keyRecord = CreateKeyRecord("key-x509", IssuerKeyType.X509Certificate);
await _keyRepository.UpsertAsync(keyRecord, CancellationToken.None);
var fetched = await _keyRepository.GetAsync(_tenantId, _issuerId, "key-x509", CancellationToken.None);
fetched.Should().NotBeNull();
fetched!.Type.Should().Be(IssuerKeyType.X509Certificate);
}
[Trait("Category", TestCategories.Unit)]
[Fact]
public async Task UpsertAsync_PersistsKeyTypeDsse()
{
var keyRecord = CreateKeyRecord("key-dsse", IssuerKeyType.DssePublicKey);
await _keyRepository.UpsertAsync(keyRecord, CancellationToken.None);
var fetched = await _keyRepository.GetAsync(_tenantId, _issuerId, "key-dsse", CancellationToken.None);
fetched.Should().NotBeNull();
fetched!.Type.Should().Be(IssuerKeyType.DssePublicKey);
}
[Trait("Category", TestCategories.Unit)]
[Fact]
public async Task UpsertAsync_PersistsRevokedStatus()
{
var keyRecord = CreateKeyRecord("key-revoked", IssuerKeyType.Ed25519PublicKey) with
{
Status = IssuerKeyStatus.Revoked,
RevokedAtUtc = DateTimeOffset.UtcNow
};
await _keyRepository.UpsertAsync(keyRecord, CancellationToken.None);
var fetched = await _keyRepository.GetAsync(_tenantId, _issuerId, "key-revoked", CancellationToken.None);
fetched.Should().NotBeNull();
fetched!.Status.Should().Be(IssuerKeyStatus.Revoked);
fetched.RevokedAtUtc.Should().NotBeNull();
}
[Trait("Category", TestCategories.Unit)]
[Fact]
public async Task UpsertAsync_PersistsReplacesKeyId()
{
var oldKey = CreateKeyRecord("old-key", IssuerKeyType.Ed25519PublicKey) with
{
Status = IssuerKeyStatus.Retired,
RetiredAtUtc = DateTimeOffset.UtcNow
};
await _keyRepository.UpsertAsync(oldKey, CancellationToken.None);
var newKey = CreateKeyRecord("new-key", IssuerKeyType.Ed25519PublicKey) with
{
ReplacesKeyId = "old-key"
};
await _keyRepository.UpsertAsync(newKey, CancellationToken.None);
var fetched = await _keyRepository.GetAsync(_tenantId, _issuerId, "new-key", CancellationToken.None);
fetched.Should().NotBeNull();
fetched!.ReplacesKeyId.Should().Be("old-key");
}
[Trait("Category", TestCategories.Unit)]
[Fact]
public async Task UpsertAsync_PersistsExpirationDate()
{
var expiresAt = DateTimeOffset.UtcNow.AddYears(1);
var keyRecord = CreateKeyRecord("key-expires", IssuerKeyType.Ed25519PublicKey) with
{
ExpiresAtUtc = expiresAt
};
await _keyRepository.UpsertAsync(keyRecord, CancellationToken.None);
var fetched = await _keyRepository.GetAsync(_tenantId, _issuerId, "key-expires", CancellationToken.None);
fetched.Should().NotBeNull();
fetched!.ExpiresAtUtc.Should().BeCloseTo(expiresAt, TimeSpan.FromSeconds(1));
}
private async Task<string> SeedIssuerAsync()
{
var issuerId = Guid.NewGuid().ToString();
var now = DateTimeOffset.UtcNow;
var issuer = new IssuerRecord
{
Id = issuerId,
TenantId = _tenantId,
Slug = $"test-issuer-{Guid.NewGuid():N}",
DisplayName = "Test Issuer",
Description = "Test issuer for key tests",
Contact = new IssuerContact(null, null, null, null),
Metadata = new IssuerMetadata(null, null, null, null, [], new Dictionary<string, string>()),
Endpoints = [],
Tags = [],
IsSystemSeed = false,
CreatedAtUtc = now,
CreatedBy = "test@test.com",
UpdatedAtUtc = now,
UpdatedBy = "test@test.com"
};
await _issuerRepository.UpsertAsync(issuer, CancellationToken.None);
return issuerId;
}
private async Task<string> SeedGlobalIssuerAsync()
{
var issuerId = Guid.NewGuid().ToString();
var now = DateTimeOffset.UtcNow;
var issuer = new IssuerRecord
{
Id = issuerId,
TenantId = IssuerTenants.Global,
Slug = $"global-issuer-{Guid.NewGuid():N}",
DisplayName = "Global Test Issuer",
Description = "Global test issuer",
Contact = new IssuerContact(null, null, null, null),
Metadata = new IssuerMetadata(null, null, null, null, [], new Dictionary<string, string>()),
Endpoints = [],
Tags = [],
IsSystemSeed = true,
CreatedAtUtc = now,
CreatedBy = "system",
UpdatedAtUtc = now,
UpdatedBy = "system"
};
await _issuerRepository.UpsertAsync(issuer, CancellationToken.None);
return issuerId;
}
private IssuerKeyRecord CreateKeyRecord(string keyId, IssuerKeyType type, string? issuerId = null, string? tenantId = null)
{
var now = DateTimeOffset.UtcNow;
return new IssuerKeyRecord
{
Id = keyId,
IssuerId = issuerId ?? _issuerId,
TenantId = tenantId ?? _tenantId,
Type = type,
Status = IssuerKeyStatus.Active,
Material = new IssuerKeyMaterial("pem", $"-----BEGIN PUBLIC KEY-----\nMFkwE...\n-----END PUBLIC KEY-----"),
Fingerprint = $"fp_{Guid.NewGuid():N}",
CreatedAtUtc = now,
CreatedBy = "test@test.com",
UpdatedAtUtc = now,
UpdatedBy = "test@test.com",
ExpiresAtUtc = null,
RetiredAtUtc = null,
RevokedAtUtc = null,
ReplacesKeyId = null
};
}
}

View File

@@ -1,231 +0,0 @@
using FluentAssertions;
using Microsoft.Extensions.Logging.Abstractions;
using StellaOps.Infrastructure.Postgres.Options;
using StellaOps.IssuerDirectory.Core.Domain;
using StellaOps.IssuerDirectory.Storage.Postgres.Repositories;
using Xunit;
using StellaOps.TestKit;
namespace StellaOps.IssuerDirectory.Storage.Postgres.Tests;
[Collection(IssuerDirectoryPostgresCollection.Name)]
public sealed class IssuerRepositoryTests : IAsyncLifetime
{
private readonly IssuerDirectoryPostgresFixture _fixture;
private readonly PostgresIssuerRepository _repository;
private readonly string _tenantId = Guid.NewGuid().ToString();
public IssuerRepositoryTests(IssuerDirectoryPostgresFixture fixture)
{
_fixture = fixture;
var options = new PostgresOptions
{
ConnectionString = fixture.ConnectionString,
SchemaName = fixture.SchemaName
};
var dataSource = new IssuerDirectoryDataSource(options, NullLogger<IssuerDirectoryDataSource>.Instance);
_repository = new PostgresIssuerRepository(dataSource, NullLogger<PostgresIssuerRepository>.Instance);
}
public async Task InitializeAsync()
{
await _fixture.TruncateAllTablesAsync();
}
public Task DisposeAsync() => Task.CompletedTask;
[Trait("Category", TestCategories.Unit)]
[Fact]
public async Task UpsertAsync_CreatesNewIssuer()
{
var record = CreateIssuerRecord("test-issuer", "Test Issuer");
await _repository.UpsertAsync(record, CancellationToken.None);
var fetched = await _repository.GetAsync(_tenantId, record.Id, CancellationToken.None);
fetched.Should().NotBeNull();
fetched!.Id.Should().Be(record.Id);
fetched.Slug.Should().Be("test-issuer");
fetched.DisplayName.Should().Be("Test Issuer");
fetched.TenantId.Should().Be(_tenantId);
}
[Trait("Category", TestCategories.Unit)]
[Fact]
public async Task UpsertAsync_UpdatesExistingIssuer()
{
var record = CreateIssuerRecord("update-test", "Original Name");
await _repository.UpsertAsync(record, CancellationToken.None);
var updated = record with
{
DisplayName = "Updated Name",
Description = "Updated description",
UpdatedAtUtc = DateTimeOffset.UtcNow,
UpdatedBy = "updater@test.com"
};
await _repository.UpsertAsync(updated, CancellationToken.None);
var fetched = await _repository.GetAsync(_tenantId, record.Id, CancellationToken.None);
fetched.Should().NotBeNull();
fetched!.DisplayName.Should().Be("Updated Name");
fetched.Description.Should().Be("Updated description");
}
[Trait("Category", TestCategories.Unit)]
[Fact]
public async Task GetAsync_ReturnsNullForNonExistentIssuer()
{
var result = await _repository.GetAsync(_tenantId, Guid.NewGuid().ToString(), CancellationToken.None);
result.Should().BeNull();
}
[Trait("Category", TestCategories.Unit)]
[Fact]
public async Task ListAsync_ReturnsAllIssuersForTenant()
{
var issuer1 = CreateIssuerRecord("issuer-a", "Issuer A");
var issuer2 = CreateIssuerRecord("issuer-b", "Issuer B");
await _repository.UpsertAsync(issuer1, CancellationToken.None);
await _repository.UpsertAsync(issuer2, CancellationToken.None);
var results = await _repository.ListAsync(_tenantId, CancellationToken.None);
results.Should().HaveCount(2);
results.Select(i => i.Slug).Should().BeEquivalentTo(["issuer-a", "issuer-b"]);
}
[Trait("Category", TestCategories.Unit)]
[Fact]
public async Task ListGlobalAsync_ReturnsGlobalIssuers()
{
var globalIssuer = CreateIssuerRecord("global-issuer", "Global Issuer", IssuerTenants.Global);
await _repository.UpsertAsync(globalIssuer, CancellationToken.None);
var results = await _repository.ListGlobalAsync(CancellationToken.None);
results.Should().Contain(i => i.Slug == "global-issuer");
}
[Trait("Category", TestCategories.Unit)]
[Fact]
public async Task DeleteAsync_RemovesIssuer()
{
var record = CreateIssuerRecord("to-delete", "To Delete");
await _repository.UpsertAsync(record, CancellationToken.None);
await _repository.DeleteAsync(_tenantId, record.Id, CancellationToken.None);
var fetched = await _repository.GetAsync(_tenantId, record.Id, CancellationToken.None);
fetched.Should().BeNull();
}
[Trait("Category", TestCategories.Unit)]
[Fact]
public async Task UpsertAsync_PersistsContactInformation()
{
var contact = new IssuerContact(
"security@example.com",
"+1-555-0100",
new Uri("https://example.com/security"),
"UTC");
var record = CreateIssuerRecord("contact-test", "Contact Test") with { Contact = contact };
await _repository.UpsertAsync(record, CancellationToken.None);
var fetched = await _repository.GetAsync(_tenantId, record.Id, CancellationToken.None);
fetched.Should().NotBeNull();
fetched!.Contact.Email.Should().Be("security@example.com");
fetched.Contact.Phone.Should().Be("+1-555-0100");
fetched.Contact.Website.Should().Be(new Uri("https://example.com/security"));
fetched.Contact.Timezone.Should().Be("UTC");
}
[Trait("Category", TestCategories.Unit)]
[Fact]
public async Task UpsertAsync_PersistsEndpoints()
{
var endpoints = new List<IssuerEndpoint>
{
new("csaf", new Uri("https://example.com/.well-known/csaf/provider-metadata.json"), "json", false),
new("oidc", new Uri("https://example.com/.well-known/openid-configuration"), "json", true)
};
var record = CreateIssuerRecord("endpoints-test", "Endpoints Test") with { Endpoints = endpoints };
await _repository.UpsertAsync(record, CancellationToken.None);
var fetched = await _repository.GetAsync(_tenantId, record.Id, CancellationToken.None);
fetched.Should().NotBeNull();
fetched!.Endpoints.Should().HaveCount(2);
fetched.Endpoints.Should().Contain(e => e.Kind == "csaf");
fetched.Endpoints.Should().Contain(e => e.Kind == "oidc" && e.RequiresAuthentication);
}
[Trait("Category", TestCategories.Unit)]
[Fact]
public async Task UpsertAsync_PersistsMetadata()
{
var metadata = new IssuerMetadata(
"CVE-2024-0001",
"csaf-pub-123",
new Uri("https://example.com/security-advisories"),
new Uri("https://example.com/catalog"),
["en", "de"],
new Dictionary<string, string> { ["custom"] = "value" });
var record = CreateIssuerRecord("metadata-test", "Metadata Test") with { Metadata = metadata };
await _repository.UpsertAsync(record, CancellationToken.None);
var fetched = await _repository.GetAsync(_tenantId, record.Id, CancellationToken.None);
fetched.Should().NotBeNull();
fetched!.Metadata.CveOrgId.Should().Be("CVE-2024-0001");
fetched.Metadata.CsafPublisherId.Should().Be("csaf-pub-123");
fetched.Metadata.SupportedLanguages.Should().BeEquivalentTo(["en", "de"]);
fetched.Metadata.Attributes.Should().ContainKey("custom");
}
[Trait("Category", TestCategories.Unit)]
[Fact]
public async Task UpsertAsync_PersistsTags()
{
var record = CreateIssuerRecord("tags-test", "Tags Test") with
{
Tags = ["vendor", "upstream", "critical"]
};
await _repository.UpsertAsync(record, CancellationToken.None);
var fetched = await _repository.GetAsync(_tenantId, record.Id, CancellationToken.None);
fetched.Should().NotBeNull();
fetched!.Tags.Should().BeEquivalentTo(["vendor", "upstream", "critical"]);
}
private IssuerRecord CreateIssuerRecord(string slug, string displayName, string? tenantId = null)
{
var now = DateTimeOffset.UtcNow;
return new IssuerRecord
{
Id = Guid.NewGuid().ToString(),
TenantId = tenantId ?? _tenantId,
Slug = slug,
DisplayName = displayName,
Description = $"Test issuer: {displayName}",
Contact = new IssuerContact(null, null, null, null),
Metadata = new IssuerMetadata(null, null, null, null, [], new Dictionary<string, string>()),
Endpoints = [],
Tags = [],
IsSystemSeed = false,
CreatedAtUtc = now,
CreatedBy = "test@test.com",
UpdatedAtUtc = now,
UpdatedBy = "test@test.com"
};
}
}

View File

@@ -1,200 +0,0 @@
using FluentAssertions;
using Microsoft.Extensions.Logging.Abstractions;
using StellaOps.Infrastructure.Postgres.Options;
using StellaOps.IssuerDirectory.Core.Domain;
using StellaOps.IssuerDirectory.Storage.Postgres.Repositories;
using Xunit;
using StellaOps.TestKit;
namespace StellaOps.IssuerDirectory.Storage.Postgres.Tests;
[Collection(IssuerDirectoryPostgresCollection.Name)]
public sealed class IssuerTrustRepositoryTests : IAsyncLifetime
{
private readonly IssuerDirectoryPostgresFixture _fixture;
private readonly PostgresIssuerRepository _issuerRepository;
private readonly PostgresIssuerTrustRepository _trustRepository;
private readonly string _tenantId = Guid.NewGuid().ToString();
private string _issuerId = null!;
public IssuerTrustRepositoryTests(IssuerDirectoryPostgresFixture fixture)
{
_fixture = fixture;
var options = new PostgresOptions
{
ConnectionString = fixture.ConnectionString,
SchemaName = fixture.SchemaName
};
var dataSource = new IssuerDirectoryDataSource(options, NullLogger<IssuerDirectoryDataSource>.Instance);
_issuerRepository = new PostgresIssuerRepository(dataSource, NullLogger<PostgresIssuerRepository>.Instance);
_trustRepository = new PostgresIssuerTrustRepository(dataSource, NullLogger<PostgresIssuerTrustRepository>.Instance);
}
public async Task InitializeAsync()
{
await _fixture.TruncateAllTablesAsync();
_issuerId = await SeedIssuerAsync();
}
public Task DisposeAsync() => Task.CompletedTask;
[Trait("Category", TestCategories.Unit)]
[Fact]
public async Task UpsertAsync_CreatesNewTrustOverride()
{
var record = CreateTrustRecord(5.5m, "Trusted vendor");
await _trustRepository.UpsertAsync(record, CancellationToken.None);
var fetched = await _trustRepository.GetAsync(_tenantId, _issuerId, CancellationToken.None);
fetched.Should().NotBeNull();
fetched!.Weight.Should().Be(5.5m);
fetched.Reason.Should().Be("Trusted vendor");
}
[Trait("Category", TestCategories.Unit)]
[Fact]
public async Task UpsertAsync_UpdatesExistingTrustOverride()
{
var record = CreateTrustRecord(3.0m, "Initial trust");
await _trustRepository.UpsertAsync(record, CancellationToken.None);
var updated = record.WithUpdated(7.5m, "Upgraded trust", DateTimeOffset.UtcNow, "admin@test.com");
await _trustRepository.UpsertAsync(updated, CancellationToken.None);
var fetched = await _trustRepository.GetAsync(_tenantId, _issuerId, CancellationToken.None);
fetched.Should().NotBeNull();
fetched!.Weight.Should().Be(7.5m);
fetched.Reason.Should().Be("Upgraded trust");
fetched.UpdatedBy.Should().Be("admin@test.com");
}
[Trait("Category", TestCategories.Unit)]
[Fact]
public async Task GetAsync_ReturnsNullForNonExistentOverride()
{
var result = await _trustRepository.GetAsync(_tenantId, Guid.NewGuid().ToString(), CancellationToken.None);
result.Should().BeNull();
}
[Trait("Category", TestCategories.Unit)]
[Fact]
public async Task DeleteAsync_RemovesTrustOverride()
{
var record = CreateTrustRecord(2.0m, "To be deleted");
await _trustRepository.UpsertAsync(record, CancellationToken.None);
await _trustRepository.DeleteAsync(_tenantId, _issuerId, CancellationToken.None);
var fetched = await _trustRepository.GetAsync(_tenantId, _issuerId, CancellationToken.None);
fetched.Should().BeNull();
}
[Trait("Category", TestCategories.Unit)]
[Fact]
public async Task UpsertAsync_PersistsPositiveWeight()
{
var record = CreateTrustRecord(10.0m, "Maximum trust");
await _trustRepository.UpsertAsync(record, CancellationToken.None);
var fetched = await _trustRepository.GetAsync(_tenantId, _issuerId, CancellationToken.None);
fetched.Should().NotBeNull();
fetched!.Weight.Should().Be(10.0m);
}
[Trait("Category", TestCategories.Unit)]
[Fact]
public async Task UpsertAsync_PersistsNegativeWeight()
{
var record = CreateTrustRecord(-5.0m, "Distrust override");
await _trustRepository.UpsertAsync(record, CancellationToken.None);
var fetched = await _trustRepository.GetAsync(_tenantId, _issuerId, CancellationToken.None);
fetched.Should().NotBeNull();
fetched!.Weight.Should().Be(-5.0m);
}
[Trait("Category", TestCategories.Unit)]
[Fact]
public async Task UpsertAsync_PersistsZeroWeight()
{
var record = CreateTrustRecord(0m, "Neutral trust");
await _trustRepository.UpsertAsync(record, CancellationToken.None);
var fetched = await _trustRepository.GetAsync(_tenantId, _issuerId, CancellationToken.None);
fetched.Should().NotBeNull();
fetched!.Weight.Should().Be(0m);
}
[Trait("Category", TestCategories.Unit)]
[Fact]
public async Task UpsertAsync_PersistsNullReason()
{
var record = CreateTrustRecord(5.0m, null);
await _trustRepository.UpsertAsync(record, CancellationToken.None);
var fetched = await _trustRepository.GetAsync(_tenantId, _issuerId, CancellationToken.None);
fetched.Should().NotBeNull();
fetched!.Reason.Should().BeNull();
}
[Trait("Category", TestCategories.Unit)]
[Fact]
public async Task UpsertAsync_PersistsTimestamps()
{
var now = DateTimeOffset.UtcNow;
var record = CreateTrustRecord(5.0m, "Time test");
await _trustRepository.UpsertAsync(record, CancellationToken.None);
var fetched = await _trustRepository.GetAsync(_tenantId, _issuerId, CancellationToken.None);
fetched.Should().NotBeNull();
fetched!.CreatedAtUtc.Should().BeCloseTo(now, TimeSpan.FromSeconds(5));
fetched.UpdatedAtUtc.Should().BeCloseTo(now, TimeSpan.FromSeconds(5));
fetched.CreatedBy.Should().Be("test@test.com");
fetched.UpdatedBy.Should().Be("test@test.com");
}
private async Task<string> SeedIssuerAsync()
{
var issuerId = Guid.NewGuid().ToString();
var now = DateTimeOffset.UtcNow;
var issuer = new IssuerRecord
{
Id = issuerId,
TenantId = _tenantId,
Slug = $"test-issuer-{Guid.NewGuid():N}",
DisplayName = "Test Issuer",
Description = "Test issuer for trust tests",
Contact = new IssuerContact(null, null, null, null),
Metadata = new IssuerMetadata(null, null, null, null, [], new Dictionary<string, string>()),
Endpoints = [],
Tags = [],
IsSystemSeed = false,
CreatedAtUtc = now,
CreatedBy = "test@test.com",
UpdatedAtUtc = now,
UpdatedBy = "test@test.com"
};
await _issuerRepository.UpsertAsync(issuer, CancellationToken.None);
return issuerId;
}
private IssuerTrustOverrideRecord CreateTrustRecord(decimal weight, string? reason)
{
return IssuerTrustOverrideRecord.Create(
_issuerId,
_tenantId,
weight,
reason,
DateTimeOffset.UtcNow,
"test@test.com");
}
}

View File

@@ -1,38 +0,0 @@
<?xml version="1.0" ?>
<Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup>
<TargetFramework>net10.0</TargetFramework>
<ImplicitUsings>enable</ImplicitUsings>
<Nullable>enable</Nullable>
<LangVersion>preview</LangVersion>
<IsPackable>false</IsPackable>
<IsTestProject>true</IsTestProject>
<!-- Disable Concelier test infra since we use our own Postgres testing infra -->
<UseConcelierTestInfra>false</UseConcelierTestInfra>
</PropertyGroup>
<ItemGroup>
<PackageReference Include="FluentAssertions" Version="6.12.0" />
<PackageReference Include="Microsoft.NET.Test.Sdk" Version="17.14.0" />
<PackageReference Include="Microsoft.Extensions.Logging.Abstractions" Version="10.0.0" />
<PackageReference Include="Moq" Version="4.20.70" />
<PackageReference Include="xunit" Version="2.9.2" />
<PackageReference Include="xunit.runner.visualstudio" Version="2.8.2">
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
<PrivateAssets>all</PrivateAssets>
</PackageReference>
<PackageReference Include="coverlet.collector" Version="6.0.4">
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
<PrivateAssets>all</PrivateAssets>
</PackageReference>
</ItemGroup>
<ItemGroup>
<ProjectReference Include="..\..\StellaOps.IssuerDirectory.Storage.Postgres\StellaOps.IssuerDirectory.Storage.Postgres.csproj" />
<ProjectReference Include="..\..\StellaOps.IssuerDirectory.Core\StellaOps.IssuerDirectory.Core.csproj" />
<ProjectReference Include="..\..\..\..\__Tests\__Libraries\StellaOps.Infrastructure.Postgres.Testing\StellaOps.Infrastructure.Postgres.Testing.csproj" />
<ProjectReference Include="../../../../__Libraries/StellaOps.TestKit/StellaOps.TestKit.csproj" />
</ItemGroup>
</Project>

View File

@@ -0,0 +1,21 @@
using Microsoft.EntityFrameworkCore;
namespace StellaOps.IssuerDirectory.Persistence.EfCore.Context;
/// <summary>
/// EF Core DbContext for IssuerDirectory module.
/// This is a stub that will be scaffolded from the PostgreSQL database.
/// </summary>
public class IssuerDirectoryDbContext : DbContext
{
public IssuerDirectoryDbContext(DbContextOptions<IssuerDirectoryDbContext> options)
: base(options)
{
}
protected override void OnModelCreating(ModelBuilder modelBuilder)
{
modelBuilder.HasDefaultSchema("issuer");
base.OnModelCreating(modelBuilder);
}
}

View File

@@ -2,14 +2,15 @@ using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Logging;
using StellaOps.Infrastructure.Postgres.Options;
using StellaOps.IssuerDirectory.Core.Abstractions;
using StellaOps.IssuerDirectory.Storage.Postgres.Repositories;
using StellaOps.IssuerDirectory.Persistence.Postgres;
using StellaOps.IssuerDirectory.Persistence.Postgres.Repositories;
namespace StellaOps.IssuerDirectory.Storage.Postgres;
namespace StellaOps.IssuerDirectory.Persistence.Extensions;
/// <summary>
/// Extension methods for registering IssuerDirectory PostgreSQL storage services.
/// Extension methods for registering IssuerDirectory persistence services.
/// </summary>
public static class ServiceCollectionExtensions
public static class IssuerDirectoryPersistenceExtensions
{
/// <summary>
/// Registers the IssuerDirectory PostgreSQL data source.
@@ -17,7 +18,7 @@ public static class ServiceCollectionExtensions
/// <param name="services">Service collection.</param>
/// <param name="configureOptions">Options configuration delegate.</param>
/// <returns>The service collection for chaining.</returns>
public static IServiceCollection AddIssuerDirectoryPostgresStorage(
public static IServiceCollection AddIssuerDirectoryPersistence(
this IServiceCollection services,
Action<PostgresOptions> configureOptions)
{
@@ -47,7 +48,7 @@ public static class ServiceCollectionExtensions
/// <param name="services">Service collection.</param>
/// <param name="options">PostgreSQL options.</param>
/// <returns>The service collection for chaining.</returns>
public static IServiceCollection AddIssuerDirectoryPostgresStorage(
public static IServiceCollection AddIssuerDirectoryPersistence(
this IServiceCollection services,
PostgresOptions options)
{

View File

@@ -2,7 +2,7 @@ using Microsoft.Extensions.Logging;
using StellaOps.Infrastructure.Postgres.Connections;
using StellaOps.Infrastructure.Postgres.Options;
namespace StellaOps.IssuerDirectory.Storage.Postgres;
namespace StellaOps.IssuerDirectory.Persistence.Postgres;
/// <summary>
/// PostgreSQL data source for the IssuerDirectory module.

View File

@@ -5,7 +5,7 @@ using NpgsqlTypes;
using StellaOps.IssuerDirectory.Core.Abstractions;
using StellaOps.IssuerDirectory.Core.Domain;
namespace StellaOps.IssuerDirectory.Storage.Postgres.Repositories;
namespace StellaOps.IssuerDirectory.Persistence.Postgres.Repositories;
/// <summary>
/// PostgreSQL implementation of the issuer audit sink.

View File

@@ -4,7 +4,7 @@ using NpgsqlTypes;
using StellaOps.IssuerDirectory.Core.Abstractions;
using StellaOps.IssuerDirectory.Core.Domain;
namespace StellaOps.IssuerDirectory.Storage.Postgres.Repositories;
namespace StellaOps.IssuerDirectory.Persistence.Postgres.Repositories;
/// <summary>
/// PostgreSQL implementation of the issuer key repository.

View File

@@ -5,7 +5,7 @@ using NpgsqlTypes;
using StellaOps.IssuerDirectory.Core.Abstractions;
using StellaOps.IssuerDirectory.Core.Domain;
namespace StellaOps.IssuerDirectory.Storage.Postgres.Repositories;
namespace StellaOps.IssuerDirectory.Persistence.Postgres.Repositories;
/// <summary>
/// PostgreSQL implementation of the issuer repository.

View File

@@ -4,7 +4,7 @@ using NpgsqlTypes;
using StellaOps.IssuerDirectory.Core.Abstractions;
using StellaOps.IssuerDirectory.Core.Domain;
namespace StellaOps.IssuerDirectory.Storage.Postgres.Repositories;
namespace StellaOps.IssuerDirectory.Persistence.Postgres.Repositories;
/// <summary>
/// PostgreSQL implementation of the issuer trust repository.

View File

@@ -0,0 +1,35 @@
<?xml version="1.0" ?>
<Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup>
<TargetFramework>net10.0</TargetFramework>
<LangVersion>preview</LangVersion>
<Nullable>enable</Nullable>
<ImplicitUsings>enable</ImplicitUsings>
<TreatWarningsAsErrors>false</TreatWarningsAsErrors>
<RootNamespace>StellaOps.IssuerDirectory.Persistence</RootNamespace>
<AssemblyName>StellaOps.IssuerDirectory.Persistence</AssemblyName>
<Description>Consolidated persistence layer for StellaOps IssuerDirectory module</Description>
</PropertyGroup>
<ItemGroup>
<PackageReference Include="Microsoft.EntityFrameworkCore" />
<PackageReference Include="Microsoft.EntityFrameworkCore.Design" PrivateAssets="all" />
<PackageReference Include="Microsoft.Extensions.DependencyInjection.Abstractions" />
<PackageReference Include="Microsoft.Extensions.Logging.Abstractions" />
<PackageReference Include="Microsoft.Extensions.Options" />
<PackageReference Include="Npgsql" />
<PackageReference Include="Npgsql.EntityFrameworkCore.PostgreSQL" />
</ItemGroup>
<ItemGroup>
<ProjectReference Include="..\..\StellaOps.IssuerDirectory\StellaOps.IssuerDirectory.Core\StellaOps.IssuerDirectory.Core.csproj" />
<ProjectReference Include="..\..\..\__Libraries\StellaOps.Infrastructure.Postgres\StellaOps.Infrastructure.Postgres.csproj" />
<ProjectReference Include="..\..\..\__Libraries\StellaOps.Infrastructure.EfCore\StellaOps.Infrastructure.EfCore.csproj" />
</ItemGroup>
<ItemGroup>
<EmbeddedResource Include="Migrations\*.sql" LogicalName="%(Filename)%(Extension)" />
</ItemGroup>
</Project>

View File

@@ -0,0 +1,35 @@
<?xml version="1.0" encoding="utf-8"?>
<Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup>
<TargetFramework>net10.0</TargetFramework>
<LangVersion>preview</LangVersion>
<Nullable>enable</Nullable>
<ImplicitUsings>enable</ImplicitUsings>
<TreatWarningsAsErrors>false</TreatWarningsAsErrors>
<RootNamespace>StellaOps.IssuerDirectory.Persistence</RootNamespace>
<AssemblyName>StellaOps.IssuerDirectory.Persistence</AssemblyName>
<Description>Consolidated persistence layer for StellaOps IssuerDirectory module</Description>
</PropertyGroup>
<ItemGroup>
<PackageReference Include="Microsoft.EntityFrameworkCore" Version="10.0.0" />
<PackageReference Include="Microsoft.EntityFrameworkCore.Design" Version="10.0.0" PrivateAssets="all" />
<PackageReference Include="Microsoft.Extensions.DependencyInjection.Abstractions" Version="10.0.0" />
<PackageReference Include="Microsoft.Extensions.Logging.Abstractions" Version="10.0.1" />
<PackageReference Include="Microsoft.Extensions.Options" Version="10.0.1" />
<PackageReference Include="Npgsql" Version="10.0.0" />
<PackageReference Include="Npgsql.EntityFrameworkCore.PostgreSQL" Version="10.0.0" />
</ItemGroup>
<ItemGroup>
<ProjectReference Include="..\..\StellaOps.IssuerDirectory\StellaOps.IssuerDirectory.Core\StellaOps.IssuerDirectory.Core.csproj" />
<ProjectReference Include="..\..\..\__Libraries\StellaOps.Infrastructure.Postgres\StellaOps.Infrastructure.Postgres.csproj" />
<ProjectReference Include="..\..\..\__Libraries\StellaOps.Infrastructure.EfCore\StellaOps.Infrastructure.EfCore.csproj" />
</ItemGroup>
<ItemGroup>
<EmbeddedResource Include="Migrations\*.sql" LogicalName="%(Filename)%(Extension)" />
</ItemGroup>
</Project>

View File

@@ -4,12 +4,12 @@ using Microsoft.Extensions.Logging.Abstractions;
using Npgsql;
using StellaOps.Infrastructure.Postgres.Options;
using StellaOps.IssuerDirectory.Core.Domain;
using StellaOps.IssuerDirectory.Storage.Postgres.Repositories;
using StellaOps.IssuerDirectory.Persistence.Postgres.Repositories;
using Xunit;
using StellaOps.TestKit;
namespace StellaOps.IssuerDirectory.Storage.Postgres.Tests;
namespace StellaOps.IssuerDirectory.Persistence.Postgres.Tests;
[Collection(IssuerDirectoryPostgresCollection.Name)]
public sealed class IssuerAuditSinkTests : IAsyncLifetime
@@ -241,7 +241,6 @@ public sealed class IssuerAuditSinkTests : IAsyncLifetime
""";
await using var command = new NpgsqlCommand(sql, connection);
using StellaOps.TestKit;
command.Parameters.AddWithValue("tenantId", Guid.Parse(tenantId));
command.Parameters.AddWithValue("issuerId", Guid.Parse(issuerId));

View File

@@ -0,0 +1,9 @@
using Xunit;
namespace StellaOps.IssuerDirectory.Persistence.Postgres.Tests;
[CollectionDefinition(Name)]
public sealed class IssuerDirectoryPostgresCollection : ICollectionFixture<IssuerDirectoryPostgresFixture>
{
public const string Name = "IssuerDirectoryPostgresCollection";
}

View File

@@ -1,14 +1,14 @@
using System.Reflection;
using Microsoft.Extensions.Logging;
using StellaOps.Infrastructure.Postgres.Testing;
using StellaOps.IssuerDirectory.Storage.Postgres;
using StellaOps.IssuerDirectory.Persistence.Postgres;
namespace StellaOps.IssuerDirectory.Storage.Postgres.Tests;
namespace StellaOps.IssuerDirectory.Persistence.Postgres.Tests;
public sealed class IssuerDirectoryPostgresFixture : PostgresIntegrationFixture
{
protected override Assembly? GetMigrationAssembly() => typeof(IssuerDirectoryDataSource).Assembly;
protected override string GetModuleName() => "issuer";
protected override string? GetResourcePrefix() => "IssuerDirectory.Storage.Postgres.Migrations";
protected override string? GetResourcePrefix() => "StellaOps.IssuerDirectory.Persistence.Migrations";
protected override ILogger Logger => Microsoft.Extensions.Logging.Abstractions.NullLogger.Instance;
}

View File

@@ -0,0 +1,70 @@
using FluentAssertions;
using Microsoft.Extensions.Logging.Abstractions;
using StellaOps.IssuerDirectory.Core.Domain;
using StellaOps.IssuerDirectory.Persistence.Postgres;
using StellaOps.IssuerDirectory.Persistence.Postgres.Repositories;
using Xunit;
using StellaOps.TestKit;
namespace StellaOps.IssuerDirectory.Persistence.Postgres.Tests;
public class IssuerKeyRepositoryTests : IClassFixture<IssuerDirectoryPostgresFixture>
{
private readonly IssuerDirectoryPostgresFixture _fixture;
public IssuerKeyRepositoryTests(IssuerDirectoryPostgresFixture fixture)
{
_fixture = fixture;
}
private PostgresIssuerRepository CreateIssuerRepo() =>
new(new IssuerDirectoryDataSource(_fixture.Fixture.CreateOptions(), NullLogger<IssuerDirectoryDataSource>.Instance),
NullLogger<PostgresIssuerRepository>.Instance);
private PostgresIssuerKeyRepository CreateKeyRepo() =>
new(new IssuerDirectoryDataSource(_fixture.Fixture.CreateOptions(), NullLogger<IssuerDirectoryDataSource>.Instance),
NullLogger<PostgresIssuerKeyRepository>.Instance);
[Trait("Category", TestCategories.Unit)]
[Fact]
public async Task AddKey_And_List_Works()
{
var tenant = Guid.NewGuid().ToString();
var issuerId = Guid.NewGuid().ToString();
var issuerRepo = CreateIssuerRepo();
var keyRepo = CreateKeyRepo();
var timestamp = DateTimeOffset.UtcNow;
var issuer = IssuerRecord.Create(
id: issuerId,
tenantId: tenant,
displayName: "Vendor X",
slug: "vendor-x",
description: null,
contact: new IssuerContact(null, null, null, null),
metadata: new IssuerMetadata(null, null, null, null, null, null),
endpoints: null,
tags: null,
timestampUtc: timestamp,
actor: "test",
isSystemSeed: false);
await issuerRepo.UpsertAsync(issuer, CancellationToken.None);
var key = IssuerKeyRecord.Create(
id: Guid.NewGuid().ToString(),
issuerId: issuerId,
tenantId: tenant,
type: IssuerKeyType.Ed25519PublicKey,
material: new IssuerKeyMaterial("pem", "pubkey"),
fingerprint: "fp-1",
createdAtUtc: DateTimeOffset.UtcNow,
createdBy: "test",
expiresAtUtc: null,
replacesKeyId: null);
await keyRepo.UpsertAsync(key, CancellationToken.None);
var keys = await keyRepo.ListAsync(tenant, issuerId, CancellationToken.None);
keys.Should().ContainSingle(k => k.IssuerId == issuerId);
}
}

View File

@@ -1,12 +1,12 @@
using FluentAssertions;
using Microsoft.Extensions.Logging.Abstractions;
using StellaOps.IssuerDirectory.Core.Domain;
using StellaOps.IssuerDirectory.Storage.Postgres;
using StellaOps.IssuerDirectory.Storage.Postgres.Repositories;
using StellaOps.IssuerDirectory.Persistence.Postgres;
using StellaOps.IssuerDirectory.Persistence.Postgres.Repositories;
using Xunit;
using StellaOps.TestKit;
namespace StellaOps.IssuerDirectory.Storage.Postgres.Tests;
namespace StellaOps.IssuerDirectory.Persistence.Postgres.Tests;
public class IssuerRepositoryTests : IClassFixture<IssuerDirectoryPostgresFixture>
{
@@ -20,7 +20,7 @@ public class IssuerRepositoryTests : IClassFixture<IssuerDirectoryPostgresFixtur
private PostgresIssuerRepository CreateRepository()
{
var dataSource = new IssuerDirectoryDataSource(
_fixture.Fixture.Options,
_fixture.Fixture.CreateOptions(),
NullLogger<IssuerDirectoryDataSource>.Instance);
return new PostgresIssuerRepository(dataSource, NullLogger<PostgresIssuerRepository>.Instance);
}
@@ -32,22 +32,20 @@ public class IssuerRepositoryTests : IClassFixture<IssuerDirectoryPostgresFixtur
var repo = CreateRepository();
var tenant = Guid.NewGuid().ToString();
var issuerId = Guid.NewGuid().ToString();
var record = new IssuerRecord(
issuerId,
tenant,
slug: "acme",
var timestamp = DateTimeOffset.UtcNow;
var record = IssuerRecord.Create(
id: issuerId,
tenantId: tenant,
displayName: "Acme Corp",
slug: "acme",
description: "Test issuer",
endpoints: new[] { new IssuerEndpoint("csaf", "https://acme.test/csaf") },
contact: new IssuerContact("security@acme.test", null),
metadata: new IssuerMetadata(Array.Empty<string>(), null),
contact: new IssuerContact("security@acme.test", null, null, null),
metadata: new IssuerMetadata(null, null, null, null, null, null),
endpoints: new[] { new IssuerEndpoint("csaf", new Uri("https://acme.test/csaf"), null, false) },
tags: new[] { "vendor", "csaf" },
status: "active",
isSystemSeed: false,
createdAt: DateTimeOffset.UtcNow,
createdBy: "test",
updatedAt: DateTimeOffset.UtcNow,
updatedBy: "test");
timestampUtc: timestamp,
actor: "test",
isSystemSeed: false);
await repo.UpsertAsync(record, CancellationToken.None);

View File

@@ -0,0 +1,21 @@
<?xml version="1.0" ?>
<Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup>
<TargetFramework>net10.0</TargetFramework>
<ImplicitUsings>enable</ImplicitUsings>
<Nullable>enable</Nullable>
<LangVersion>preview</LangVersion>
<IsPackable>false</IsPackable>
<IsTestProject>true</IsTestProject>
<RootNamespace>StellaOps.IssuerDirectory.Persistence.Tests</RootNamespace>
</PropertyGroup>
<ItemGroup>
<PackageReference Include="FluentAssertions" />
</ItemGroup>
<ItemGroup>
<ProjectReference Include="..\..\__Libraries\StellaOps.IssuerDirectory.Persistence\StellaOps.IssuerDirectory.Persistence.csproj" />
<ProjectReference Include="..\..\..\__Tests\__Libraries\StellaOps.Infrastructure.Postgres.Testing\StellaOps.Infrastructure.Postgres.Testing.csproj" />
<ProjectReference Include="..\..\StellaOps.IssuerDirectory\StellaOps.IssuerDirectory.Core\StellaOps.IssuerDirectory.Core.csproj" />
<ProjectReference Include="..\..\..\__Libraries\StellaOps.TestKit\StellaOps.TestKit.csproj" />
</ItemGroup>
</Project>

View File

@@ -1,12 +1,12 @@
using FluentAssertions;
using Microsoft.Extensions.Logging.Abstractions;
using StellaOps.IssuerDirectory.Core.Domain;
using StellaOps.IssuerDirectory.Storage.Postgres;
using StellaOps.IssuerDirectory.Storage.Postgres.Repositories;
using StellaOps.IssuerDirectory.Persistence.Postgres;
using StellaOps.IssuerDirectory.Persistence.Postgres.Repositories;
using Xunit;
using StellaOps.TestKit;
namespace StellaOps.IssuerDirectory.Storage.Postgres.Tests;
namespace StellaOps.IssuerDirectory.Persistence.Postgres.Tests;
public class TrustRepositoryTests : IClassFixture<IssuerDirectoryPostgresFixture>
{
@@ -18,11 +18,11 @@ public class TrustRepositoryTests : IClassFixture<IssuerDirectoryPostgresFixture
}
private PostgresIssuerRepository CreateIssuerRepo() =>
new(new IssuerDirectoryDataSource(_fixture.Fixture.Options, NullLogger<IssuerDirectoryDataSource>.Instance),
new(new IssuerDirectoryDataSource(_fixture.Fixture.CreateOptions(), NullLogger<IssuerDirectoryDataSource>.Instance),
NullLogger<PostgresIssuerRepository>.Instance);
private PostgresIssuerTrustRepository CreateTrustRepo() =>
new(new IssuerDirectoryDataSource(_fixture.Fixture.Options, NullLogger<IssuerDirectoryDataSource>.Instance),
new(new IssuerDirectoryDataSource(_fixture.Fixture.CreateOptions(), NullLogger<IssuerDirectoryDataSource>.Instance),
NullLogger<PostgresIssuerTrustRepository>.Instance);
[Trait("Category", TestCategories.Unit)]
@@ -34,35 +34,29 @@ public class TrustRepositoryTests : IClassFixture<IssuerDirectoryPostgresFixture
var issuerRepo = CreateIssuerRepo();
var trustRepo = CreateTrustRepo();
var issuer = new IssuerRecord(
issuerId,
tenant,
slug: "trusty",
var timestamp = DateTimeOffset.UtcNow;
var issuer = IssuerRecord.Create(
id: issuerId,
tenantId: tenant,
displayName: "Trusty Issuer",
slug: "trusty",
description: null,
endpoints: Array.Empty<IssuerEndpoint>(),
contact: new IssuerContact(null, null),
metadata: new IssuerMetadata(Array.Empty<string>(), null),
tags: Array.Empty<string>(),
status: "active",
isSystemSeed: false,
createdAt: DateTimeOffset.UtcNow,
createdBy: "test",
updatedAt: DateTimeOffset.UtcNow,
updatedBy: "test");
contact: new IssuerContact(null, null, null, null),
metadata: new IssuerMetadata(null, null, null, null, null, null),
endpoints: null,
tags: null,
timestampUtc: timestamp,
actor: "test",
isSystemSeed: false);
await issuerRepo.UpsertAsync(issuer, CancellationToken.None);
var trust = new IssuerTrustOverrideRecord(
id: Guid.NewGuid().ToString(),
var trust = IssuerTrustOverrideRecord.Create(
issuerId: issuerId,
tenantId: tenant,
weight: 0.75m,
rationale: "vendor override",
expiresAt: null,
createdAt: DateTimeOffset.UtcNow,
createdBy: "test",
updatedAt: DateTimeOffset.UtcNow,
updatedBy: "test");
reason: "vendor override",
timestampUtc: DateTimeOffset.UtcNow,
actor: "test");
await trustRepo.UpsertAsync(trust, CancellationToken.None);

View File

@@ -1,77 +0,0 @@
using FluentAssertions;
using Microsoft.Extensions.Logging.Abstractions;
using StellaOps.IssuerDirectory.Core.Domain;
using StellaOps.IssuerDirectory.Storage.Postgres;
using StellaOps.IssuerDirectory.Storage.Postgres.Repositories;
using Xunit;
using StellaOps.TestKit;
namespace StellaOps.IssuerDirectory.Storage.Postgres.Tests;
public class IssuerKeyRepositoryTests : IClassFixture<IssuerDirectoryPostgresFixture>
{
private readonly IssuerDirectoryPostgresFixture _fixture;
public IssuerKeyRepositoryTests(IssuerDirectoryPostgresFixture fixture)
{
_fixture = fixture;
}
private PostgresIssuerRepository CreateIssuerRepo() =>
new(new IssuerDirectoryDataSource(_fixture.Fixture.Options, NullLogger<IssuerDirectoryDataSource>.Instance),
NullLogger<PostgresIssuerRepository>.Instance);
private PostgresIssuerKeyRepository CreateKeyRepo() =>
new(new IssuerDirectoryDataSource(_fixture.Fixture.Options, NullLogger<IssuerDirectoryDataSource>.Instance),
NullLogger<PostgresIssuerKeyRepository>.Instance);
[Trait("Category", TestCategories.Unit)]
[Fact]
public async Task AddKey_And_List_Works()
{
var tenant = Guid.NewGuid().ToString();
var issuerId = Guid.NewGuid().ToString();
var issuerRepo = CreateIssuerRepo();
var keyRepo = CreateKeyRepo();
var issuer = new IssuerRecord(
issuerId,
tenant,
slug: "vendor-x",
displayName: "Vendor X",
description: null,
endpoints: Array.Empty<IssuerEndpoint>(),
contact: new IssuerContact(null, null),
metadata: new IssuerMetadata(Array.Empty<string>(), null),
tags: Array.Empty<string>(),
status: "active",
isSystemSeed: false,
createdAt: DateTimeOffset.UtcNow,
createdBy: "test",
updatedAt: DateTimeOffset.UtcNow,
updatedBy: "test");
await issuerRepo.UpsertAsync(issuer, CancellationToken.None);
var key = new IssuerKeyRecord(
id: Guid.NewGuid().ToString(),
issuerId: issuerId,
keyId: "kid-1",
keyType: IssuerKeyType.Ed25519,
publicKey: "pubkey",
fingerprint: "fp-1",
notBefore: null,
notAfter: null,
status: IssuerKeyStatus.Active,
createdAt: DateTimeOffset.UtcNow,
createdBy: "test",
revokedAt: null,
revokedBy: null,
revokeReason: null,
metadata: new IssuerKeyMetadata(null, null));
await keyRepo.UpsertAsync(key, CancellationToken.None);
var keys = await keyRepo.ListAsync(tenant, issuerId, CancellationToken.None);
keys.Should().ContainSingle(k => k.KeyId == "kid-1" && k.IssuerId == issuerId);
}
}

View File

@@ -1,29 +0,0 @@
<Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup>
<TargetFramework>net10.0</TargetFramework>
<ImplicitUsings>enable</ImplicitUsings>
<Nullable>enable</Nullable>
<LangVersion>preview</LangVersion>
<IsPackable>false</IsPackable>
<IsTestProject>true</IsTestProject>
</PropertyGroup>
<ItemGroup>
<PackageReference Include="FluentAssertions" Version="6.12.0" />
<PackageReference Include="Microsoft.NET.Test.Sdk" Version="17.11.1" />
<PackageReference Include="xunit" Version="2.9.2" />
<PackageReference Include="xunit.runner.visualstudio" Version="2.8.2">
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
<PrivateAssets>all</PrivateAssets>
</PackageReference>
<PackageReference Include="coverlet.collector" Version="6.0.4">
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
<PrivateAssets>all</PrivateAssets>
</PackageReference>
</ItemGroup>
<ItemGroup>
<ProjectReference Include="..\\..\\StellaOps.IssuerDirectory\\StellaOps.IssuerDirectory.Storage.Postgres\\StellaOps.IssuerDirectory.Storage.Postgres.csproj" />
<ProjectReference Include="..\\..\\..\\__Tests\\__Libraries\\StellaOps.Infrastructure.Postgres.Testing\\StellaOps.Infrastructure.Postgres.Testing.csproj" />
<ProjectReference Include="..\\..\\StellaOps.IssuerDirectory\\StellaOps.IssuerDirectory.Core\\StellaOps.IssuerDirectory.Core.csproj" />
<ProjectReference Include="../../../__Libraries/StellaOps.TestKit/StellaOps.TestKit.csproj" />
</ItemGroup>
</Project>