Files
solution-erp/.gitea/workflows/deploy.yml
pqhuy1987 df5988b7a9
All checks were successful
Deploy SOLUTION_ERP / build-deploy (push) Successful in 3m16s
[CLAUDE] Tests Phase 2: Code generator format + sequence tests (SQLite in-memory)
Phase 2 — chống regression code generator. 17 test mới integration với
DB thật (SQLite in-memory) tổng cộng 71 test pass < 3 giây.

Test project:
- tests/SolutionErp.Infrastructure.Tests/ (xUnit + FluentAssertions + EF SQLite 10)
- ProjectReference SolutionErp.Infrastructure (transitively get Application + Domain)
- Added vào SolutionErp.slnx

Test fixtures:
- Common/SqliteDbFixture.cs:
  - SQLite ":memory:" + shared connection + EnsureCreated() từ DbContext model
  - TestApplicationDbContext subclass — override OnModelCreating replace
    'nvarchar(max)' → 'TEXT' (SQLite không support max keyword)
  - FixedDateTime stub IDateTime cho deterministic year boundary test

Test files:
- Services/ContractCodeGeneratorTests.cs (10 test):
  - Format per ContractType (5 type × Project scope) — RG-001 spec
  - Framework HĐ (NguyenTacNCC + NguyenTacDV) → year scope thay vì project
  - Sequence increment per prefix (3 calls → /01, /02, /03)
  - Different prefixes (project / supplier) → independent sequences
  - Year change (2026 → 2027) → reset sequence vì prefix khác
  - PersistsSequenceRow LastSeq verification
- Services/PurchaseEvaluationCodeGeneratorTests.cs (7 test):
  - Format A/B (DuyetNcc → 'A', DuyetNccPhuongAn → 'B')
  - Seq là 3-digit padded (001..012)
  - Type A và B sequence độc lập trong cùng năm
  - Year boundary reset cả A và B

CI gate update (.gitea/workflows/deploy.yml):
- Step "Run integration tests (Infrastructure)" thêm sau Domain tests
- TRX log saved riêng (infra-tests.trx)
- Cả 2 step đều exit non-zero → no deploy

Verify local:
- dotnet test SolutionErp.slnx → Total tests: 71 (54 Domain + 17 Infra) / Passed: 71 / 2.1s
- dotnet build SolutionErp.slnx → 0 error

Phase 3+ pending:
- Application handler tests (CQRS) với EF InMemory hoặc SQLite (~1 ngày)
- API smoke tests qua WebApplicationFactory (~0.5 ngày)
- FE Vitest cho lib utility (~0.5 ngày)

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-04-29 13:29:06 +07:00

149 lines
6.2 KiB
YAML

# Gitea Actions CI/CD - build + deploy SOLUTION_ERP to IIS on same VPS.
# Trigger: push to main, or manual dispatch.
#
# Self-hosted Windows runner on VPS (shared with VIETREPORT). Runner has:
# - git, .NET 10 SDK, Node 20, IIS
# - Can deploy locally (no WinRM needed)
#
# Secrets required in Gitea repo settings:
# - JWT_SECRET (64+ chars random)
# - DB_CONNECTION (full connection string with vrapp password)
name: Deploy SOLUTION_ERP
on:
push:
branches: [main]
workflow_dispatch:
jobs:
build-deploy:
runs-on: windows-latest
steps:
- uses: actions/checkout@v4
- name: Show tool versions
shell: powershell
run: |
& 'C:\Program Files\dotnet\dotnet.exe' --version
& 'C:\Program Files\nodejs\node.exe' --version
& 'C:\Program Files\nodejs\npm.cmd' --version
# ============== TEST GATE ==============
# Run tests TRƯỚC build/publish/deploy. Fail → exit non-zero → no deploy.
# Phase 1: Domain (54 test policy state machine).
# Phase 2: Infrastructure (17 test code generators format/sequence/year scope).
- name: Run unit tests (Domain)
shell: powershell
run: |
& 'C:\Program Files\dotnet\dotnet.exe' test tests/SolutionErp.Domain.Tests/SolutionErp.Domain.Tests.csproj `
--configuration Release `
--logger "trx;LogFileName=domain-tests.trx" `
--results-directory test-results
if ($LASTEXITCODE -ne 0) { exit $LASTEXITCODE }
- name: Run integration tests (Infrastructure - SQLite in-memory)
shell: powershell
run: |
& 'C:\Program Files\dotnet\dotnet.exe' test tests/SolutionErp.Infrastructure.Tests/SolutionErp.Infrastructure.Tests.csproj `
--configuration Release `
--logger "trx;LogFileName=infra-tests.trx" `
--results-directory test-results
if ($LASTEXITCODE -ne 0) { exit $LASTEXITCODE }
- name: Upload test results
if: always()
continue-on-error: true # nếu Gitea runner chưa có upload-artifact action, skip không block deploy
uses: actions/upload-artifact@v4
with:
name: test-results
path: test-results/*.trx
retention-days: 14
# ============== BUILD ==============
- name: Build backend
shell: powershell
run: |
& 'C:\Program Files\dotnet\dotnet.exe' restore SolutionErp.slnx
& 'C:\Program Files\dotnet\dotnet.exe' publish src/Backend/SolutionErp.Api/SolutionErp.Api.csproj `
--configuration Release `
--output out/api `
--runtime win-x64 `
--self-contained false
- name: Build fe-admin
shell: powershell
working-directory: fe-admin
run: |
# Vite 8 rolldown native binding must match platform; fresh resolve on Windows
Remove-Item node_modules, package-lock.json -Recurse -Force -ErrorAction SilentlyContinue
& 'C:\Program Files\nodejs\npm.cmd' install --no-audit --no-fund
& 'C:\Program Files\nodejs\npm.cmd' run build
- name: Build fe-user
shell: powershell
working-directory: fe-user
run: |
Remove-Item node_modules, package-lock.json -Recurse -Force -ErrorAction SilentlyContinue
& 'C:\Program Files\nodejs\npm.cmd' install --no-audit --no-fund
& 'C:\Program Files\nodejs\npm.cmd' run build
- name: Deploy to IIS (local)
if: github.ref == 'refs/heads/main'
shell: powershell
env:
JWT_SECRET: ${{ secrets.JWT_SECRET }}
DB_CONNECTION: ${{ secrets.DB_CONNECTION }}
run: |
Import-Module WebAdministration
# Stop app pool (if running) so DLLs are writable
$poolState = (Get-WebAppPoolState -Name SolutionErp-Api -ErrorAction SilentlyContinue).Value
if ($poolState -eq 'Started') {
Stop-WebAppPool -Name SolutionErp-Api
Start-Sleep -Seconds 3
}
# Deploy API
Remove-Item -Path 'C:\inetpub\solution-erp\api\*' -Recurse -Force -Exclude 'appsettings.Production.json','logs','uploads','wwwroot' -ErrorAction SilentlyContinue
Copy-Item -Path 'out\api\*' -Destination 'C:\inetpub\solution-erp\api\' -Recurse -Force
# Write appsettings.Production.json from source template + secrets.
# Template is in source workspace (not in publish output - dotnet publish
# doesn't copy .example files).
$example = 'src\Backend\SolutionErp.Api\appsettings.Production.json.example'
$prod = 'C:\inetpub\solution-erp\api\appsettings.Production.json'
$settings = Get-Content $example -Raw | ConvertFrom-Json
$settings.ConnectionStrings.Default = $env:DB_CONNECTION
$settings.Jwt.Secret = $env:JWT_SECRET
$settings | ConvertTo-Json -Depth 10 | Set-Content -Path $prod -Encoding UTF8
Write-Host "Wrote appsettings.Production.json"
# Restrict ACL
icacls $prod /inheritance:r | Out-Null
icacls $prod /grant:r 'Administrators:(R,W)' 'IIS AppPool\SolutionErp-Api:(R)' | Out-Null
# Deploy fe-admin
Remove-Item -Path 'C:\inetpub\solution-erp\fe-admin\*' -Recurse -Force -Exclude 'web.config' -ErrorAction SilentlyContinue
Copy-Item -Path 'fe-admin\dist\*' -Destination 'C:\inetpub\solution-erp\fe-admin\' -Recurse -Force
# Deploy fe-user
Remove-Item -Path 'C:\inetpub\solution-erp\fe-user\*' -Recurse -Force -Exclude 'web.config' -ErrorAction SilentlyContinue
Copy-Item -Path 'fe-user\dist\*' -Destination 'C:\inetpub\solution-erp\fe-user\' -Recurse -Force
# Restart app pool
Start-WebAppPool -Name SolutionErp-Api
Write-Host "Deploy done. App pool started."
- name: Smoke test
if: github.ref == 'refs/heads/main'
shell: powershell
run: |
Start-Sleep -Seconds 10
try {
$r = Invoke-WebRequest -Uri 'https://api.solutions.com.vn/health/live' -TimeoutSec 30 -UseBasicParsing
Write-Host "API /health/live -> $($r.StatusCode)"
} catch {
Write-Warning "API smoke test: $_"
}