mirror of
https://github.com/azaion/satellite-provider.git
synced 2026-06-21 11:11:14 +00:00
51b572108a
Captures the post-implementation autodev gates for AZ-484 multi-source tile storage: - Step 12 (Test-Spec Sync): added 7 AC rows (AZ-484 AC-1..AC-7) and a PT-07 NFR row to traceability-matrix.md; added PT-07 scenario to performance-tests.md. - Step 13 (Update Docs): refreshed data_model.md (tiles columns + indexes + selection rule + UPSERT contract + migrations 012/013), module-layout.md (Common/Enums section with L-001 guidance, DataAccess imports-from now lists 6 sites), 6 module / component docs to reflect the new repo signatures, source/captured_at fields, and Dapper enum bypass workaround. ripple_log_cycle1.md records zero out-of-scope ripple. - Step 14 (Security Audit): PASS_WITH_WARNINGS - 0 Critical, 0 High, 5 Medium, 5 Low. AZ-484 itself added zero new findings. Hardening items (Postgres default creds, .env in build context, GMaps key rotation, ASP.NET Core 8.0.21 -> 8.0.25, rate limiter) recorded for separate tickets. - Step 15 (Performance Test): all PT-01..PT-07 scenarios Unverified (non-blocking); PT-07 baseline-comparison harness deferred to a leftover for next cycle. - Step 16 (Deploy): cycle deploy report covering migration safety, rollback path, post-deploy verification, security caveats. Co-authored-by: Cursor <cursoragent@cursor.com>
11 KiB
11 KiB
Module Layout
Status: derived-from-code
Language: csharp Layout Convention: custom (per-component .csproj per logical component) Root: ./ Last Updated: 2026-05-11 (post AZ-350 03-code-quality-refactoring run; corrects DataAccess→Common dependency)
Layout Rules
- Each component owns ONE top-level project directory (
.csprojboundary). The previous sharedSatelliteProvider.Servicesproject was split into three per-component csprojs in epic AZ-309. - Shared code lives under
SatelliteProvider.Common/— the foundation layer. - Cross-cutting concerns (DTOs, interfaces, configs, geo-math, common exceptions) all reside in Common.
- Public API surface per component =
publictypes in the namespace root. Everything markedinternalor private is internal. - Tests live in separate projects:
SatelliteProvider.Tests/(unit) andSatelliteProvider.IntegrationTests/(integration). - DI registration per component lives in a
<Component>ServiceCollectionExtensions.csadjacent to the component's classes (e.g.TileDownloaderServiceCollectionExtensions.AddTileDownloader()).
Per-Component Mapping
Component: Common
- Directory:
SatelliteProvider.Common/ - Public API:
SatelliteProvider.Common/Configs/MapConfig.csSatelliteProvider.Common/Configs/StorageConfig.csSatelliteProvider.Common/Configs/ProcessingConfig.csSatelliteProvider.Common/Configs/DatabaseConfig.csSatelliteProvider.Common/DTO/*.cs(all DTOs)SatelliteProvider.Common/Enums/RegionStatus.csSatelliteProvider.Common/Enums/RoutePointType.csSatelliteProvider.Common/Enums/TileSource.cs(added by AZ-484; backed by thetile-storagev1.0.0 contract)SatelliteProvider.Common/Enums/TileSourceConverter.cs(added by AZ-484; convertsTileSourceenum to/from the snake_case wire string used byTileEntity.Source)SatelliteProvider.Common/Exceptions/RateLimitException.csSatelliteProvider.Common/Interfaces/*.cs(all service interfaces)SatelliteProvider.Common/Utils/GeoUtils.cs
- Internal: (none — all types are public, shared across components)
- Owns:
SatelliteProvider.Common/** - Imports from: (none)
- Consumed by: DataAccess, TileDownloader, RegionProcessing, RouteManagement, WebApi
Component: DataAccess
- Directory:
SatelliteProvider.DataAccess/ - Public API:
SatelliteProvider.DataAccess/Models/TileEntity.csSatelliteProvider.DataAccess/Models/RegionEntity.csSatelliteProvider.DataAccess/Models/RouteEntity.csSatelliteProvider.DataAccess/Models/RoutePointEntity.csSatelliteProvider.DataAccess/Repositories/ITileRepository.csSatelliteProvider.DataAccess/Repositories/IRegionRepository.csSatelliteProvider.DataAccess/Repositories/IRouteRepository.csSatelliteProvider.DataAccess/Repositories/TileRepository.csSatelliteProvider.DataAccess/Repositories/RegionRepository.csSatelliteProvider.DataAccess/Repositories/RouteRepository.csSatelliteProvider.DataAccess/DatabaseMigrator.cs
- Internal: (none — all repository types are public for DI registration)
- Owns:
SatelliteProvider.DataAccess/** - ProjectReferences:
SatelliteProvider.Common - Imports from:
SatelliteProvider.Common.Enums(6 sites:RegionRepository,IRegionRepository,Models/RegionEntity,Models/RoutePointEntity,TypeHandlers/EnumStringTypeHandler,Models/TileEntity— referencesTileSourceConverter.GoogleMapsWireValueconst for the AZ-484 default value);SatelliteProvider.Common.Configs(MapConfig.DefaultTileSizePixelsinTileRepository);SatelliteProvider.Common.Utils(GeoUtils.EarthEquatorialCircumferenceMeters,GeoUtils.MetersPerDegreeLatitudeinTileRepository). - Consumed by: TileDownloader, RegionProcessing, RouteManagement, WebApi
Component: TileDownloader
- Directory:
SatelliteProvider.Services.TileDownloader/ - csproj:
SatelliteProvider.Services.TileDownloader/SatelliteProvider.Services.TileDownloader.csproj - Public API:
SatelliteProvider.Services.TileDownloader/GoogleMapsDownloaderV2.cs(implementsISatelliteDownloader)SatelliteProvider.Services.TileDownloader/TileService.cs(implementsITileService)SatelliteProvider.Services.TileDownloader/TileDownloaderServiceCollectionExtensions.cs(DI:AddTileDownloader())
- Internal: (none)
- Owns:
SatelliteProvider.Services.TileDownloader/** - ProjectReferences:
SatelliteProvider.Common,SatelliteProvider.DataAccess - Imports from: Common, DataAccess
- Consumed by: RegionProcessing (via
ITileServicefrom Common; no direct ProjectReference), WebApi
Component: RegionProcessing
- Directory:
SatelliteProvider.Services.RegionProcessing/ - csproj:
SatelliteProvider.Services.RegionProcessing/SatelliteProvider.Services.RegionProcessing.csproj - Public API:
SatelliteProvider.Services.RegionProcessing/RegionService.cs(implementsIRegionService)SatelliteProvider.Services.RegionProcessing/RegionProcessingService.cs(background hosted service)SatelliteProvider.Services.RegionProcessing/RegionRequestQueue.cs(implementsIRegionRequestQueue)SatelliteProvider.Services.RegionProcessing/RegionProcessingServiceCollectionExtensions.cs(DI:AddRegionProcessing())
- Internal: (none)
- Owns:
SatelliteProvider.Services.RegionProcessing/** - ProjectReferences:
SatelliteProvider.Common,SatelliteProvider.DataAccess - Imports from: Common, DataAccess (uses
ITileServicefrom Common — no compile-time dependency on TileDownloader) - Consumed by: RouteManagement (via
IRegionServiceandIRegionRequestQueuefrom Common; no direct ProjectReference), WebApi
Component: RouteManagement
- Directory:
SatelliteProvider.Services.RouteManagement/ - csproj:
SatelliteProvider.Services.RouteManagement/SatelliteProvider.Services.RouteManagement.csproj - Public API:
SatelliteProvider.Services.RouteManagement/RouteService.cs(implementsIRouteService)SatelliteProvider.Services.RouteManagement/RouteProcessingService.cs(background hosted service)SatelliteProvider.Services.RouteManagement/RouteManagementServiceCollectionExtensions.cs(DI:AddRouteManagement())
- Internal: (none)
- Owns:
SatelliteProvider.Services.RouteManagement/** - ProjectReferences:
SatelliteProvider.Common,SatelliteProvider.DataAccess - Imports from: Common, DataAccess (uses
IRegionService/IRegionRequestQueuefrom Common — no compile-time dependency on RegionProcessing) - Consumed by: WebApi
Component: WebApi
- Directory:
SatelliteProvider.Api/ - Public API:
SatelliteProvider.Api/Program.cs(minimal API endpoints, DI setup)
- Internal: (none)
- Owns:
SatelliteProvider.Api/** - Imports from: Common, DataAccess, TileDownloader, RegionProcessing, RouteManagement
- Consumed by: (none — top-level entry point)
Shared / Cross-Cutting
Common/Configs
- Directory:
SatelliteProvider.Common/Configs/ - Purpose: Strongly-typed configuration POCOs bound via
IOptions<T> - Consumed by: all components
Common/DTO
- Directory:
SatelliteProvider.Common/DTO/ - Purpose: Data transfer objects shared across layers (request/response models, value types)
- Consumed by: all components
Common/Interfaces
- Directory:
SatelliteProvider.Common/Interfaces/ - Purpose: Service contracts enabling DI and testability
- Consumed by: all components (services implement, API and consumers depend on)
Common/Utils
- Directory:
SatelliteProvider.Common/Utils/ - Purpose: Stateless geospatial utility functions (coordinate math, distance, bearing)
- Consumed by: TileDownloader, RegionProcessing, RouteManagement
Common/Enums
- Directory:
SatelliteProvider.Common/Enums/ - Purpose: Domain enums shared across layers (
RegionStatus,RoutePointType,TileSource) plus their explicit wire-value converters when persistence requires snake_case strings (TileSourceConverter). Converter classes belong here — not in DataAccess — because they encode a domain-level vocabulary that must be visible to every component. - Consumed by: DataAccess (entity defaults, type handler registration), TileDownloader (sets
TileEntity.SourceviaTileSourceConverter.ToWireValue), Tests - Important constraint: Dapper's
SqlMapper.TypeHandler<TEnum>is bypassed for enum reads (Dapper issue #259 — see_docs/LESSONS.mdL-001). For any new enum that must round-trip through a database column, prefer thestring-on-entity +Enum-at-API-boundary pattern with a converter class in this folder. Do NOT register aTypeHandler<TEnum>and assume it will be honored on reads.
Allowed Dependencies (layering)
| Layer | Components | May import from (compile-time ProjectReferences) |
|---|---|---|
| 4. API / Entry | WebApi | Common, DataAccess, TileDownloader, RegionProcessing, RouteManagement |
| 3. Application | TileDownloader, RegionProcessing, RouteManagement | Common, DataAccess only — siblings communicate through interfaces in Common, never through direct ProjectReferences |
| 1. Foundation | Common (leaf-most), DataAccess | Common: (none); DataAccess: Common only — Common MUST NOT import from DataAccess |
Key constraint enforced by the AZ-309 split: the three Layer-3 components are compile-time siblings. Any cross-sibling call (e.g. RegionProcessing invoking tile download) MUST go through an interface defined in SatelliteProvider.Common.Interfaces and resolved via DI — adding a ProjectReference between siblings is now structurally impossible without re-introducing the coupling the refactor removed.
Verification
- No detected cycles: The dependency graph is a clean DAG.
- No cross-sibling ProjectReferences: TileDownloader, RegionProcessing, and RouteManagement each reference only Common + DataAccess. Verified by inspecting all three csproj files.
- DataAccess layer placement: DataAccess sits at Layer 1 (Foundation) alongside Common because it is consumed uniformly by all service components. It is one half-step above Common because it depends on Common for shared enums and a small number of constants/configs.
- DataAccess→Common ProjectReference: confirmed present in
SatelliteProvider.DataAccess.csprojline 18 and used by 7 source sites (5 enum imports, 1MapConfig.DefaultTileSizePixelssite, 1GeoUtils.*site). The earlier compliance baseline F5 entry that claimed "DataAccess has no Common dependency" was inaccurate — bothmodule-layout.mdandarchitecture_compliance_baseline.mdwere corrected during the 03-code-quality-refactoring run (2026-05-11). The actual constraint that holds is one-way:CommonMUST NOT import fromDataAccess.