mirror of
https://github.com/shankar0123/certctl.git
synced 2026-06-07 19:21:29 +00:00
8b75e0311b
Mechanical sed across the main go.mod's module declaration, the f5-mock-icontrol
sub-module's go.mod, every Go file's import path (361 files), and a rebuild of
the checked-in f5-mock-icontrol binary so its embedded build-info reflects the
new module path. No behavior change.
Choice B from cowork/transfer-certctl-to-org.md, executed 2026-05-04. Choice A
(keep module path declared as github.com/shankar0123/certctl regardless of
repo URL) shipped on the day of the org transfer (2026-05-03) since we had no
external Go consumers; this commit closes that deferral.
Backward-compat: GitHub HTTP redirects continue to forward
github.com/shankar0123/certctl → github.com/certctl-io/certctl at the URL
level, but Go's module proxy uses the path declared in go.mod as the
canonical name. Pre-fix, anyone trying `go get github.com/certctl-io/certctl/...`
hit a "module path mismatch" error because go.mod said
github.com/shankar0123/certctl and the URL they fetched it from said
certctl-io/certctl. Post-fix, the canonical name and the URL agree, so
go get / go install / external Go consumers / Go-tooling integrations
work cleanly via either the new path (preferred) or the old path (which
redirects and Go follows the redirect for source fetch).
Anyone still importing the old path inside their own code keeps working
provided they update their go.mod's `require` line to match — the module
path declared in their consumer's go.sum / go.mod is the authoritative
import name, so a mass sed across their import statements is the migration
on the consumer side. No external consumers exist today.
Diff shape:
361 *.go files — import path replacement only
2 go.mod — module declaration replacement only
1 binary — deploy/test/f5-mock-icontrol/f5-mock-icontrol rebuilt
so embedded build-info reflects the new path (8618965 vs
8618933 bytes; 32-byte diff is the build-info change)
Total: 364 files, 730 insertions / 730 deletions, net-zero size, pure
mechanical substitution.
Verification:
gofmt: 17 files needed re-alignment after sed (the new path is one char
shorter than the old, so column-aligned import groups drifted). Applied
`gofmt -w` to fix.
go mod tidy: clean exit on both modules.
go vet ./...: clean exit.
go build ./...: clean exit.
go test -short -count=1 on representative packages: all green
(internal/domain, internal/validation, internal/crypto, internal/crypto/signer,
cmd/agent). Test output now reads `ok github.com/certctl-io/certctl/...`
confirming the module path resolves correctly.
binary: f5-mock-icontrol rebuilt; `strings | grep shankar0123` returns
nothing; `strings | grep certctl-io/certctl` shows the new module path
embedded in build-info.
Files intentionally NOT touched in this commit:
README.md / CHANGELOG.md / docs/ / etc. — already swept to certctl-io
URLs in commit 0729ee4 (the post-transfer URL refresh). This commit is
purely the Go-tooling layer.
Scarf pixels (`shankar0123.docker.scarf.sh/...`) — Scarf-account
namespace, not a Go import or GitHub repo URL. Stays.
This is a non-blocking, non-customer-impacting change. Operators pulling
container images, running `make verify`, hitting the API, or installing the
agent see no functional difference. Only Go-tooling consumers (none today)
are affected, and they're enabled — not broken — by this commit.
675 lines
19 KiB
Go
675 lines
19 KiB
Go
package service
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/certctl-io/certctl/internal/domain"
|
|
"github.com/certctl-io/certctl/internal/repository"
|
|
)
|
|
|
|
// mockDiscoveryRepo is a test implementation of DiscoveryRepository
|
|
type mockDiscoveryRepo struct {
|
|
Scans map[string]*domain.DiscoveryScan
|
|
Discovered map[string]*domain.DiscoveredCertificate
|
|
CreateScanErr error
|
|
GetScanErr error
|
|
ListScansErr error
|
|
CreateDiscoveredErr error
|
|
GetDiscoveredErr error
|
|
ListDiscoveredErr error
|
|
UpdateStatusErr error
|
|
GetByFingerprintErr error
|
|
CountByStatusErr error
|
|
}
|
|
|
|
func newMockDiscoveryRepository() *mockDiscoveryRepo {
|
|
return &mockDiscoveryRepo{
|
|
Scans: make(map[string]*domain.DiscoveryScan),
|
|
Discovered: make(map[string]*domain.DiscoveredCertificate),
|
|
}
|
|
}
|
|
|
|
func (m *mockDiscoveryRepo) CreateScan(ctx context.Context, scan *domain.DiscoveryScan) error {
|
|
if m.CreateScanErr != nil {
|
|
return m.CreateScanErr
|
|
}
|
|
m.Scans[scan.ID] = scan
|
|
return nil
|
|
}
|
|
|
|
func (m *mockDiscoveryRepo) GetScan(ctx context.Context, id string) (*domain.DiscoveryScan, error) {
|
|
if m.GetScanErr != nil {
|
|
return nil, m.GetScanErr
|
|
}
|
|
scan, ok := m.Scans[id]
|
|
if !ok {
|
|
return nil, errNotFound
|
|
}
|
|
return scan, nil
|
|
}
|
|
|
|
func (m *mockDiscoveryRepo) ListScans(ctx context.Context, agentID string, page, perPage int) ([]*domain.DiscoveryScan, int, error) {
|
|
if m.ListScansErr != nil {
|
|
return nil, 0, m.ListScansErr
|
|
}
|
|
var scans []*domain.DiscoveryScan
|
|
for _, s := range m.Scans {
|
|
if agentID == "" || s.AgentID == agentID {
|
|
scans = append(scans, s)
|
|
}
|
|
}
|
|
return scans, len(scans), nil
|
|
}
|
|
|
|
func (m *mockDiscoveryRepo) CreateDiscovered(ctx context.Context, cert *domain.DiscoveredCertificate) (bool, error) {
|
|
if m.CreateDiscoveredErr != nil {
|
|
return false, m.CreateDiscoveredErr
|
|
}
|
|
_, exists := m.Discovered[cert.ID]
|
|
m.Discovered[cert.ID] = cert
|
|
return !exists, nil // true if new (not existed before)
|
|
}
|
|
|
|
func (m *mockDiscoveryRepo) GetDiscovered(ctx context.Context, id string) (*domain.DiscoveredCertificate, error) {
|
|
if m.GetDiscoveredErr != nil {
|
|
return nil, m.GetDiscoveredErr
|
|
}
|
|
cert, ok := m.Discovered[id]
|
|
if !ok {
|
|
return nil, errNotFound
|
|
}
|
|
return cert, nil
|
|
}
|
|
|
|
func (m *mockDiscoveryRepo) ListDiscovered(ctx context.Context, filter *repository.DiscoveryFilter) ([]*domain.DiscoveredCertificate, int, error) {
|
|
if m.ListDiscoveredErr != nil {
|
|
return nil, 0, m.ListDiscoveredErr
|
|
}
|
|
var certs []*domain.DiscoveredCertificate
|
|
for _, c := range m.Discovered {
|
|
if filter.AgentID != "" && c.AgentID != filter.AgentID {
|
|
continue
|
|
}
|
|
if filter.Status != "" && string(c.Status) != filter.Status {
|
|
continue
|
|
}
|
|
certs = append(certs, c)
|
|
}
|
|
return certs, len(certs), nil
|
|
}
|
|
|
|
func (m *mockDiscoveryRepo) UpdateDiscoveredStatus(ctx context.Context, id string, status domain.DiscoveryStatus, managedCertID string) error {
|
|
if m.UpdateStatusErr != nil {
|
|
return m.UpdateStatusErr
|
|
}
|
|
cert, ok := m.Discovered[id]
|
|
if !ok {
|
|
return errNotFound
|
|
}
|
|
cert.Status = status
|
|
cert.ManagedCertificateID = managedCertID
|
|
now := time.Now()
|
|
if status == domain.DiscoveryStatusDismissed {
|
|
cert.DismissedAt = &now
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (m *mockDiscoveryRepo) GetByFingerprint(ctx context.Context, fingerprint string) ([]*domain.DiscoveredCertificate, error) {
|
|
if m.GetByFingerprintErr != nil {
|
|
return nil, m.GetByFingerprintErr
|
|
}
|
|
var certs []*domain.DiscoveredCertificate
|
|
for _, c := range m.Discovered {
|
|
if c.FingerprintSHA256 == fingerprint {
|
|
certs = append(certs, c)
|
|
}
|
|
}
|
|
return certs, nil
|
|
}
|
|
|
|
func (m *mockDiscoveryRepo) CountByStatus(ctx context.Context) (map[string]int, error) {
|
|
if m.CountByStatusErr != nil {
|
|
return nil, m.CountByStatusErr
|
|
}
|
|
counts := make(map[string]int)
|
|
for _, c := range m.Discovered {
|
|
counts[string(c.Status)]++
|
|
}
|
|
return counts, nil
|
|
}
|
|
|
|
// helper to create a test DiscoveryService wired for discovery tests
|
|
func newDiscoveryTestService() (*DiscoveryService, *mockDiscoveryRepo, *mockCertRepo, *mockAuditRepo) {
|
|
discoveryRepo := newMockDiscoveryRepository()
|
|
certRepo := newMockCertificateRepository()
|
|
auditRepo := newMockAuditRepository()
|
|
|
|
auditService := NewAuditService(auditRepo)
|
|
discoveryService := NewDiscoveryService(discoveryRepo, certRepo, auditService)
|
|
|
|
return discoveryService, discoveryRepo, certRepo, auditRepo
|
|
}
|
|
|
|
func TestProcessDiscoveryReport_Success(t *testing.T) {
|
|
svc, discoveryRepo, _, auditRepo := newDiscoveryTestService()
|
|
|
|
report := &domain.DiscoveryReport{
|
|
AgentID: "agent-1",
|
|
Directories: []string{"/etc/certs", "/opt/certs"},
|
|
ScanDurationMs: 150,
|
|
Certificates: []domain.DiscoveredCertEntry{
|
|
{
|
|
FingerprintSHA256: "abc123",
|
|
CommonName: "example.com",
|
|
SANs: []string{"www.example.com"},
|
|
SerialNumber: "001",
|
|
IssuerDN: "CN=Let's Encrypt",
|
|
SubjectDN: "CN=example.com",
|
|
NotBefore: time.Now().AddDate(-1, 0, 0).Format(time.RFC3339),
|
|
NotAfter: time.Now().AddDate(1, 0, 0).Format(time.RFC3339),
|
|
KeyAlgorithm: "RSA",
|
|
KeySize: 2048,
|
|
IsCA: false,
|
|
PEMData: "-----BEGIN CERTIFICATE-----...",
|
|
SourcePath: "/etc/certs/example.com.crt",
|
|
SourceFormat: "PEM",
|
|
},
|
|
},
|
|
Errors: []string{},
|
|
}
|
|
|
|
scan, err := svc.ProcessDiscoveryReport(context.Background(), report)
|
|
if err != nil {
|
|
t.Fatalf("expected no error, got: %v", err)
|
|
}
|
|
|
|
if scan == nil {
|
|
t.Fatal("expected scan to be returned")
|
|
}
|
|
if scan.AgentID != "agent-1" {
|
|
t.Errorf("expected agent ID agent-1, got %s", scan.AgentID)
|
|
}
|
|
if scan.CertificatesFound != 1 {
|
|
t.Errorf("expected 1 certificate found, got %d", scan.CertificatesFound)
|
|
}
|
|
if scan.CertificatesNew != 1 {
|
|
t.Errorf("expected 1 new certificate, got %d", scan.CertificatesNew)
|
|
}
|
|
|
|
// Verify scan was persisted
|
|
if len(discoveryRepo.Scans) != 1 {
|
|
t.Fatalf("expected 1 scan in repo, got %d", len(discoveryRepo.Scans))
|
|
}
|
|
|
|
// Verify discovered cert was persisted
|
|
if len(discoveryRepo.Discovered) != 1 {
|
|
t.Fatalf("expected 1 discovered cert in repo, got %d", len(discoveryRepo.Discovered))
|
|
}
|
|
|
|
// Verify audit event was recorded
|
|
if len(auditRepo.Events) == 0 {
|
|
t.Error("expected audit event to be recorded")
|
|
}
|
|
foundDiscoveryAudit := false
|
|
for _, e := range auditRepo.Events {
|
|
if e.Action == "discovery_scan_completed" {
|
|
foundDiscoveryAudit = true
|
|
}
|
|
}
|
|
if !foundDiscoveryAudit {
|
|
t.Error("expected discovery_scan_completed audit event")
|
|
}
|
|
}
|
|
|
|
func TestProcessDiscoveryReport_EmptyAgentID(t *testing.T) {
|
|
svc, _, _, _ := newDiscoveryTestService()
|
|
|
|
report := &domain.DiscoveryReport{
|
|
AgentID: "", // empty agent ID
|
|
Certificates: []domain.DiscoveredCertEntry{
|
|
{
|
|
FingerprintSHA256: "abc123",
|
|
CommonName: "example.com",
|
|
},
|
|
},
|
|
}
|
|
|
|
_, err := svc.ProcessDiscoveryReport(context.Background(), report)
|
|
if err == nil {
|
|
t.Fatal("expected error for empty agent_id")
|
|
}
|
|
if !errors.Is(err, err) { // just verify error occurred
|
|
t.Errorf("expected validation error")
|
|
}
|
|
}
|
|
|
|
func TestProcessDiscoveryReport_EmptyReport(t *testing.T) {
|
|
svc, _, _, _ := newDiscoveryTestService()
|
|
|
|
report := &domain.DiscoveryReport{
|
|
AgentID: "agent-1",
|
|
Certificates: []domain.DiscoveredCertEntry{},
|
|
Errors: []string{},
|
|
ScanDurationMs: 100,
|
|
}
|
|
|
|
_, err := svc.ProcessDiscoveryReport(context.Background(), report)
|
|
if err == nil {
|
|
t.Fatal("expected error for empty report")
|
|
}
|
|
}
|
|
|
|
func TestListDiscovered_Success(t *testing.T) {
|
|
svc, discoveryRepo, _, _ := newDiscoveryTestService()
|
|
|
|
now := time.Now()
|
|
cert1 := &domain.DiscoveredCertificate{
|
|
ID: "dcert-1",
|
|
AgentID: "agent-1",
|
|
CommonName: "example.com",
|
|
Status: domain.DiscoveryStatusUnmanaged,
|
|
CreatedAt: now,
|
|
UpdatedAt: now,
|
|
}
|
|
cert2 := &domain.DiscoveredCertificate{
|
|
ID: "dcert-2",
|
|
AgentID: "agent-1",
|
|
CommonName: "api.example.com",
|
|
Status: domain.DiscoveryStatusManaged,
|
|
CreatedAt: now,
|
|
UpdatedAt: now,
|
|
}
|
|
discoveryRepo.Discovered[cert1.ID] = cert1
|
|
discoveryRepo.Discovered[cert2.ID] = cert2
|
|
|
|
certs, total, err := svc.ListDiscovered(context.Background(), "agent-1", "", 1, 50)
|
|
if err != nil {
|
|
t.Fatalf("expected no error, got: %v", err)
|
|
}
|
|
|
|
if len(certs) != 2 {
|
|
t.Errorf("expected 2 certs, got %d", len(certs))
|
|
}
|
|
if total != 2 {
|
|
t.Errorf("expected total 2, got %d", total)
|
|
}
|
|
}
|
|
|
|
func TestListDiscovered_WithStatusFilter(t *testing.T) {
|
|
svc, discoveryRepo, _, _ := newDiscoveryTestService()
|
|
|
|
now := time.Now()
|
|
cert1 := &domain.DiscoveredCertificate{
|
|
ID: "dcert-1",
|
|
AgentID: "agent-1",
|
|
CommonName: "example.com",
|
|
Status: domain.DiscoveryStatusUnmanaged,
|
|
CreatedAt: now,
|
|
UpdatedAt: now,
|
|
}
|
|
cert2 := &domain.DiscoveredCertificate{
|
|
ID: "dcert-2",
|
|
AgentID: "agent-1",
|
|
CommonName: "api.example.com",
|
|
Status: domain.DiscoveryStatusManaged,
|
|
CreatedAt: now,
|
|
UpdatedAt: now,
|
|
}
|
|
discoveryRepo.Discovered[cert1.ID] = cert1
|
|
discoveryRepo.Discovered[cert2.ID] = cert2
|
|
|
|
certs, total, err := svc.ListDiscovered(context.Background(), "agent-1", "Unmanaged", 1, 50)
|
|
if err != nil {
|
|
t.Fatalf("expected no error, got: %v", err)
|
|
}
|
|
|
|
if len(certs) != 1 {
|
|
t.Errorf("expected 1 cert, got %d", len(certs))
|
|
}
|
|
if total != 1 {
|
|
t.Errorf("expected total 1, got %d", total)
|
|
}
|
|
}
|
|
|
|
func TestGetDiscovered_Success(t *testing.T) {
|
|
svc, discoveryRepo, _, _ := newDiscoveryTestService()
|
|
|
|
now := time.Now()
|
|
cert := &domain.DiscoveredCertificate{
|
|
ID: "dcert-1",
|
|
CommonName: "example.com",
|
|
Status: domain.DiscoveryStatusUnmanaged,
|
|
CreatedAt: now,
|
|
UpdatedAt: now,
|
|
}
|
|
discoveryRepo.Discovered[cert.ID] = cert
|
|
|
|
retrieved, err := svc.GetDiscovered(context.Background(), "dcert-1")
|
|
if err != nil {
|
|
t.Fatalf("expected no error, got: %v", err)
|
|
}
|
|
|
|
if retrieved.ID != "dcert-1" {
|
|
t.Errorf("expected ID dcert-1, got %s", retrieved.ID)
|
|
}
|
|
}
|
|
|
|
func TestClaimDiscovered_Success(t *testing.T) {
|
|
svc, discoveryRepo, certRepo, auditRepo := newDiscoveryTestService()
|
|
|
|
now := time.Now()
|
|
discoveredCert := &domain.DiscoveredCertificate{
|
|
ID: "dcert-1",
|
|
CommonName: "example.com",
|
|
FingerprintSHA256: "abc123",
|
|
Status: domain.DiscoveryStatusUnmanaged,
|
|
CreatedAt: now,
|
|
UpdatedAt: now,
|
|
}
|
|
discoveryRepo.Discovered[discoveredCert.ID] = discoveredCert
|
|
|
|
managedCert := &domain.ManagedCertificate{
|
|
ID: "mc-prod-1",
|
|
CommonName: "example.com",
|
|
Status: domain.CertificateStatusActive,
|
|
CreatedAt: now,
|
|
UpdatedAt: now,
|
|
}
|
|
certRepo.AddCert(managedCert)
|
|
|
|
err := svc.ClaimDiscovered(context.Background(), "dcert-1", "mc-prod-1", "alice@corp")
|
|
if err != nil {
|
|
t.Fatalf("expected no error, got: %v", err)
|
|
}
|
|
|
|
// Verify status was updated
|
|
updated := discoveryRepo.Discovered["dcert-1"]
|
|
if updated.Status != domain.DiscoveryStatusManaged {
|
|
t.Errorf("expected status Managed, got %s", updated.Status)
|
|
}
|
|
if updated.ManagedCertificateID != "mc-prod-1" {
|
|
t.Errorf("expected managed cert ID mc-prod-1, got %s", updated.ManagedCertificateID)
|
|
}
|
|
|
|
// Verify audit event was recorded
|
|
if len(auditRepo.Events) == 0 {
|
|
t.Error("expected audit event to be recorded")
|
|
}
|
|
foundClaimAudit := false
|
|
for _, e := range auditRepo.Events {
|
|
if e.Action == "discovery_cert_claimed" {
|
|
foundClaimAudit = true
|
|
}
|
|
}
|
|
if !foundClaimAudit {
|
|
t.Error("expected discovery_cert_claimed audit event")
|
|
}
|
|
}
|
|
|
|
func TestClaimDiscovered_MissingManagedCertID(t *testing.T) {
|
|
svc, discoveryRepo, _, _ := newDiscoveryTestService()
|
|
|
|
now := time.Now()
|
|
cert := &domain.DiscoveredCertificate{
|
|
ID: "dcert-1",
|
|
CommonName: "example.com",
|
|
Status: domain.DiscoveryStatusUnmanaged,
|
|
CreatedAt: now,
|
|
UpdatedAt: now,
|
|
}
|
|
discoveryRepo.Discovered[cert.ID] = cert
|
|
|
|
err := svc.ClaimDiscovered(context.Background(), "dcert-1", "", "test-actor")
|
|
if err == nil {
|
|
t.Fatal("expected error for empty managed_certificate_id")
|
|
}
|
|
}
|
|
|
|
func TestClaimDiscovered_ManagedCertNotFound(t *testing.T) {
|
|
svc, discoveryRepo, _, _ := newDiscoveryTestService()
|
|
|
|
now := time.Now()
|
|
cert := &domain.DiscoveredCertificate{
|
|
ID: "dcert-1",
|
|
CommonName: "example.com",
|
|
Status: domain.DiscoveryStatusUnmanaged,
|
|
CreatedAt: now,
|
|
UpdatedAt: now,
|
|
}
|
|
discoveryRepo.Discovered[cert.ID] = cert
|
|
|
|
err := svc.ClaimDiscovered(context.Background(), "dcert-1", "nonexistent-cert", "test-actor")
|
|
if err == nil {
|
|
t.Fatal("expected error for nonexistent managed certificate")
|
|
}
|
|
if !errors.Is(err, err) { // just verify error occurred
|
|
t.Errorf("expected 'not found' error")
|
|
}
|
|
}
|
|
|
|
func TestDismissDiscovered_Success(t *testing.T) {
|
|
svc, discoveryRepo, _, auditRepo := newDiscoveryTestService()
|
|
|
|
now := time.Now()
|
|
cert := &domain.DiscoveredCertificate{
|
|
ID: "dcert-1",
|
|
CommonName: "example.com",
|
|
Status: domain.DiscoveryStatusUnmanaged,
|
|
CreatedAt: now,
|
|
UpdatedAt: now,
|
|
}
|
|
discoveryRepo.Discovered[cert.ID] = cert
|
|
|
|
err := svc.DismissDiscovered(context.Background(), "dcert-1", "bob@corp")
|
|
if err != nil {
|
|
t.Fatalf("expected no error, got: %v", err)
|
|
}
|
|
|
|
// Verify status was updated
|
|
updated := discoveryRepo.Discovered["dcert-1"]
|
|
if updated.Status != domain.DiscoveryStatusDismissed {
|
|
t.Errorf("expected status Dismissed, got %s", updated.Status)
|
|
}
|
|
if updated.DismissedAt == nil {
|
|
t.Error("expected DismissedAt to be set")
|
|
}
|
|
|
|
// Verify audit event was recorded
|
|
if len(auditRepo.Events) == 0 {
|
|
t.Error("expected audit event to be recorded")
|
|
}
|
|
foundDismissAudit := false
|
|
for _, e := range auditRepo.Events {
|
|
if e.Action == "discovery_cert_dismissed" {
|
|
foundDismissAudit = true
|
|
}
|
|
}
|
|
if !foundDismissAudit {
|
|
t.Error("expected discovery_cert_dismissed audit event")
|
|
}
|
|
}
|
|
|
|
func TestDismissDiscovered_NotFound(t *testing.T) {
|
|
svc, discoveryRepo, _, _ := newDiscoveryTestService()
|
|
|
|
discoveryRepo.UpdateStatusErr = errNotFound
|
|
err := svc.DismissDiscovered(context.Background(), "nonexistent", "test-actor")
|
|
if err == nil {
|
|
t.Fatal("expected error for nonexistent cert")
|
|
}
|
|
}
|
|
|
|
// M-005 regression: caller-supplied actor must propagate onto the
|
|
// discovery_cert_claimed audit event so the trail identifies who performed
|
|
// triage (pre-M-005 the service hardcoded "operator").
|
|
func TestDiscoveryService_ClaimDiscovered_AuditActor(t *testing.T) {
|
|
svc, discoveryRepo, certRepo, auditRepo := newDiscoveryTestService()
|
|
|
|
now := time.Now()
|
|
discoveredCert := &domain.DiscoveredCertificate{
|
|
ID: "dcert-1",
|
|
CommonName: "example.com",
|
|
FingerprintSHA256: "abc123",
|
|
Status: domain.DiscoveryStatusUnmanaged,
|
|
CreatedAt: now,
|
|
UpdatedAt: now,
|
|
}
|
|
discoveryRepo.Discovered[discoveredCert.ID] = discoveredCert
|
|
|
|
managedCert := &domain.ManagedCertificate{
|
|
ID: "mc-prod-1",
|
|
CommonName: "example.com",
|
|
Status: domain.CertificateStatusActive,
|
|
CreatedAt: now,
|
|
UpdatedAt: now,
|
|
}
|
|
certRepo.AddCert(managedCert)
|
|
|
|
if err := svc.ClaimDiscovered(context.Background(), "dcert-1", "mc-prod-1", "alice@corp"); err != nil {
|
|
t.Fatalf("expected no error, got: %v", err)
|
|
}
|
|
|
|
// Locate the discovery_cert_claimed audit event and assert actor propagation.
|
|
var claimEvent *domain.AuditEvent
|
|
for _, e := range auditRepo.Events {
|
|
if e.Action == "discovery_cert_claimed" {
|
|
claimEvent = e
|
|
break
|
|
}
|
|
}
|
|
if claimEvent == nil {
|
|
t.Fatal("expected discovery_cert_claimed audit event to be recorded")
|
|
}
|
|
if claimEvent.Actor != "alice@corp" {
|
|
t.Errorf("expected audit actor to be caller-supplied 'alice@corp', got %q", claimEvent.Actor)
|
|
}
|
|
if claimEvent.Actor == "operator" {
|
|
t.Error("audit actor must not be hardcoded 'operator' (M-005 regression)")
|
|
}
|
|
}
|
|
|
|
// M-005 regression symmetric pair for DismissDiscovered.
|
|
func TestDiscoveryService_DismissDiscovered_AuditActor(t *testing.T) {
|
|
svc, discoveryRepo, _, auditRepo := newDiscoveryTestService()
|
|
|
|
now := time.Now()
|
|
cert := &domain.DiscoveredCertificate{
|
|
ID: "dcert-1",
|
|
CommonName: "example.com",
|
|
Status: domain.DiscoveryStatusUnmanaged,
|
|
CreatedAt: now,
|
|
UpdatedAt: now,
|
|
}
|
|
discoveryRepo.Discovered[cert.ID] = cert
|
|
|
|
if err := svc.DismissDiscovered(context.Background(), "dcert-1", "bob@corp"); err != nil {
|
|
t.Fatalf("expected no error, got: %v", err)
|
|
}
|
|
|
|
var dismissEvent *domain.AuditEvent
|
|
for _, e := range auditRepo.Events {
|
|
if e.Action == "discovery_cert_dismissed" {
|
|
dismissEvent = e
|
|
break
|
|
}
|
|
}
|
|
if dismissEvent == nil {
|
|
t.Fatal("expected discovery_cert_dismissed audit event to be recorded")
|
|
}
|
|
if dismissEvent.Actor != "bob@corp" {
|
|
t.Errorf("expected audit actor to be caller-supplied 'bob@corp', got %q", dismissEvent.Actor)
|
|
}
|
|
if dismissEvent.Actor == "operator" {
|
|
t.Error("audit actor must not be hardcoded 'operator' (M-005 regression)")
|
|
}
|
|
}
|
|
|
|
// M-005 regression: when the caller passes an empty actor (e.g., the handler's
|
|
// resolveActor helper returns "" because no auth context is present), the
|
|
// service must fall back to the safe sentinel "api" — never to the pre-M-005
|
|
// hardcoded "operator".
|
|
func TestDiscoveryService_ClaimDiscovered_EmptyActorFallsBackToAPI(t *testing.T) {
|
|
svc, discoveryRepo, certRepo, auditRepo := newDiscoveryTestService()
|
|
|
|
now := time.Now()
|
|
discoveredCert := &domain.DiscoveredCertificate{
|
|
ID: "dcert-1",
|
|
CommonName: "example.com",
|
|
FingerprintSHA256: "abc123",
|
|
Status: domain.DiscoveryStatusUnmanaged,
|
|
CreatedAt: now,
|
|
UpdatedAt: now,
|
|
}
|
|
discoveryRepo.Discovered[discoveredCert.ID] = discoveredCert
|
|
|
|
managedCert := &domain.ManagedCertificate{
|
|
ID: "mc-prod-1",
|
|
CommonName: "example.com",
|
|
Status: domain.CertificateStatusActive,
|
|
CreatedAt: now,
|
|
UpdatedAt: now,
|
|
}
|
|
certRepo.AddCert(managedCert)
|
|
|
|
if err := svc.ClaimDiscovered(context.Background(), "dcert-1", "mc-prod-1", ""); err != nil {
|
|
t.Fatalf("expected no error, got: %v", err)
|
|
}
|
|
|
|
var claimEvent *domain.AuditEvent
|
|
for _, e := range auditRepo.Events {
|
|
if e.Action == "discovery_cert_claimed" {
|
|
claimEvent = e
|
|
break
|
|
}
|
|
}
|
|
if claimEvent == nil {
|
|
t.Fatal("expected discovery_cert_claimed audit event to be recorded")
|
|
}
|
|
if claimEvent.Actor != "api" {
|
|
t.Errorf("expected empty actor to fall back to 'api', got %q", claimEvent.Actor)
|
|
}
|
|
if claimEvent.Actor == "operator" {
|
|
t.Error("audit actor must not be hardcoded 'operator' (M-005 regression)")
|
|
}
|
|
}
|
|
|
|
// M-005 regression symmetric pair for DismissDiscovered empty-actor fallback.
|
|
func TestDiscoveryService_DismissDiscovered_EmptyActorFallsBackToAPI(t *testing.T) {
|
|
svc, discoveryRepo, _, auditRepo := newDiscoveryTestService()
|
|
|
|
now := time.Now()
|
|
cert := &domain.DiscoveredCertificate{
|
|
ID: "dcert-1",
|
|
CommonName: "example.com",
|
|
Status: domain.DiscoveryStatusUnmanaged,
|
|
CreatedAt: now,
|
|
UpdatedAt: now,
|
|
}
|
|
discoveryRepo.Discovered[cert.ID] = cert
|
|
|
|
if err := svc.DismissDiscovered(context.Background(), "dcert-1", ""); err != nil {
|
|
t.Fatalf("expected no error, got: %v", err)
|
|
}
|
|
|
|
var dismissEvent *domain.AuditEvent
|
|
for _, e := range auditRepo.Events {
|
|
if e.Action == "discovery_cert_dismissed" {
|
|
dismissEvent = e
|
|
break
|
|
}
|
|
}
|
|
if dismissEvent == nil {
|
|
t.Fatal("expected discovery_cert_dismissed audit event to be recorded")
|
|
}
|
|
if dismissEvent.Actor != "api" {
|
|
t.Errorf("expected empty actor to fall back to 'api', got %q", dismissEvent.Actor)
|
|
}
|
|
if dismissEvent.Actor == "operator" {
|
|
t.Error("audit actor must not be hardcoded 'operator' (M-005 regression)")
|
|
}
|
|
}
|