Files
certctl/internal/service/acme_test.go
T
shankar0123 44a85d6f85 acme-server: account resource + JWS verifier (Phase 1b/7)
Layers JWS-authenticated POST machinery onto the Phase 1a foundation
(commit ec88a61). After this commit, an ACME client can run

  POST /acme/profile/<id>/new-account

against certctl and successfully register an account. Account update
+ deactivation via POST /acme/profile/<id>/account/<acc-id> work.
Orders + challenges remain Phase 2 / 3.

Background:
  Two prior dispatch attempts at the original Phase 1 ("skeleton +
  directory + new-nonce + new-account" as a single commit) failed on
  go-jose v4 API speculation (jws.GetPayload, sig.Algorithm,
  jose.SHA256, etc. — none of those exist in v4). Splitting Phase 1
  into 1a (foundation, no go-jose) and 1b (this commit, all go-jose
  in one place) concentrated the JWS work where attention pays off.
  The verifier reads the actual go-jose v4 surface — ParseSigned with
  closed alg allow-list, Header struct fields (Algorithm, KeyID,
  JSONWebKey, Nonce, ExtraHeaders[HeaderKey]), JWK.Thumbprint with
  stdlib crypto.SHA256.

What ships:
  - internal/api/acme/jws.go: 487-line verifier + sentinel error
    family. Enforces RFC 8555 §6.2 + §6.4 + §6.5 invariants:
      - alg in {RS256, ES256, EdDSA} (closed allow-list passed to
        jose.ParseSigned — HS256 / none / etc. rejected at parse time)
      - exactly one of `kid` / `jwk` in protected header (per
        endpoint policy — new-account demands jwk, others demand kid)
      - protected `url` matches request URL exactly
      - protected `nonce` consumed against acme_nonces (badNonce on
        miss/replay/expiry per RFC 8555 §6.5.1)
      - kid round-trips against canonical AccountKID(accountID) URL
        (catches cross-profile / cross-host replay)
      - kid path: account exists + status=valid (deactivated /
        revoked accounts cannot authenticate)
      - signature verifies; post-Verify payload bytes equal
        UnsafePayloadWithoutVerification (defense in depth)
    + JWK persistence helpers (JWKToPEM / ParseJWKFromPEM round-
    trip a public-only JWK as a PEM-wrapped JSON envelope; stored
    as TEXT in acme_accounts.jwk_pem for diff-friendliness) +
    JWKThumbprint per RFC 7638.
  - internal/api/acme/jws_test.go: 16 cases covering happy paths
    (RS256 kid, ES256 jwk, EdDSA kid) + every named failure mode
    (alg-not-allowed, bad-sig, missing-nonce, unknown-nonce,
    replay, url-mismatch, mixed kid+jwk, deactivated-account,
    cross-host kid). Uses real keypairs + real go-jose Signer to
    build JWS objects.
  - internal/api/acme/account.go: NewAccountRequest /
    AccountUpdateRequest payload shapes (RFC 8555 §7.3 + §7.3.2 +
    §7.3.6) + AccountResponseJSON wire shape + MarshalAccount
    helper.
  - internal/domain/acme.go: ACMEAccount struct + ACMEAccountStatus
    closed enum (valid / deactivated / revoked).
  - internal/repository/postgres/acme.go: full account CRUD path
    (CreateAccountWithTx with 23505-unique-violation sentinel
    translation, GetAccountByID, GetAccountByThumbprint,
    UpdateAccountContactWithTx, UpdateAccountStatusWithTx) +
    sql.ErrNoRows-wrapped repository.ErrNotFound on lookup misses.
  - internal/service/acme.go: ACMERepo interface extended;
    SetTransactor + SetAuditService wires; NewAccount (idempotent
    re-registration per RFC 8555 §7.3.1 — same JWK returns existing
    row without an update or new audit event); LookupAccount;
    UpdateAccount; DeactivateAccount; VerifyJWS adapter that bridges
    api/acme.VerifierConfig to the service-layer ACMERepo; per-op
    metrics extended (new_account_total + _failures_total +
    _idempotent_total + update_account_total + _failures_total +
    deactivate_account_total).
  - internal/service/acme_test.go: 8 new tests covering
    new-account happy path / idempotent re-registration / only-
    return-existing match + no-match / contact update / deactivate
    / lookup-not-found / requires-transactor.
  - internal/api/handler/acme.go: NewAccount + Account handlers.
    Account dispatches POST-as-GET (RFC 8555 §6.3 — empty body or
    {} payload returns the account row), contact update, and
    deactivation from the same endpoint. Defense-in-depth check
    that the kid path-segment matches the URL path-segment (the
    verifier already round-tripped the kid against canonical URL,
    but the handler re-asserts to catch any future verifier
    refactor).
  - internal/api/handler/acme_handler_test.go: 7 new cases
    covering happy-create, idempotent-200, only-return-existing-
    no-match-400, malformed-JWS-400, kid-URL-mismatch-401,
    deactivate, contact-update, POST-as-GET.
  - internal/api/router/router.go: 4 new Register calls (per-
    profile + shorthand for new-account and account/{acc_id}).
  - internal/api/router/openapi_parity_test.go: SpecParityExceptions
    extended with the 4 new routes (RFC 8555 wire-protocol surface,
    not OpenAPI-shaped — same precedent as Phase 1a).
  - cmd/server/main.go: SetTransactor + SetAuditService on
    acmeService at startup so the WithinTx-based new-account /
    update / deactivate paths run with the same transactor instance
    shared across CertificateService / RevocationSvc / RenewalService.
  - docs/acme-server.md: Phase status updated; endpoints table grows
    new-account + account/<acc_id> rows; new "JWS verification
    (Phase 1b)" section enumerates the 7 invariants the verifier
    enforces; phases-cross-reference table marks 1b live.
  - go.mod / go.sum: github.com/go-jose/go-jose/v4 v4.0.4 added.

Atomicity: every account-state mutation writes its acme_accounts row
+ its audit_events row inside one repository.Transactor.WithinTx
call — the canonical certctl atomicity contract (matches
CertificateService.Create at internal/service/certificate.go:131).
Idempotent re-registration explicitly does NOT write an audit row
(RFC 8555 §7.3.1 returns the existing row unmodified).

Tests: 16 jws_test.go cases + 11 service tests + 11 handler tests
all pass under -short. Bad-signature test uses a real registered
account whose stored JWK is a different keypair from the signer's,
so the JWS parses cleanly but jose.Verify rejects — exercises the
ErrJWSSignatureInvalid path directly.

Engineering history: cowork/WORKSPACE-CHANGELOG.md "ACME-Server-1b".
2026-05-03 13:21:56 +00:00

485 lines
16 KiB
Go

// Copyright (c) certctl
// SPDX-License-Identifier: BSL-1.1
package service
import (
"context"
"crypto/rand"
"crypto/rsa"
"errors"
"testing"
"time"
jose "github.com/go-jose/go-jose/v4"
"github.com/shankar0123/certctl/internal/config"
"github.com/shankar0123/certctl/internal/domain"
"github.com/shankar0123/certctl/internal/repository"
)
// fakeACMERepo is an in-memory ACMERepo for tests. It tracks issued
// nonces in a map; Consume removes the entry to model one-shot use.
// Phase 1b extends with account state.
type fakeACMERepo struct {
issued map[string]time.Time // nonce → expires_at
issueErr error
// Phase 1b — account state.
accounts map[string]*domain.ACMEAccount // account_id → row
thumbToAccount map[string]string // (profile|thumbprint) → account_id
createAccountErr error
}
func newFakeACMERepo() *fakeACMERepo {
return &fakeACMERepo{
issued: make(map[string]time.Time),
accounts: make(map[string]*domain.ACMEAccount),
thumbToAccount: make(map[string]string),
}
}
func (f *fakeACMERepo) IssueNonce(ctx context.Context, nonce string, ttl time.Duration) error {
if f.issueErr != nil {
return f.issueErr
}
f.issued[nonce] = time.Now().Add(ttl)
return nil
}
func (f *fakeACMERepo) ConsumeNonce(ctx context.Context, nonce string) error {
exp, ok := f.issued[nonce]
if !ok {
return errors.New("not found")
}
if time.Now().After(exp) {
return errors.New("expired")
}
delete(f.issued, nonce)
return nil
}
func (f *fakeACMERepo) CreateAccountWithTx(ctx context.Context, q repository.Querier, acct *domain.ACMEAccount) error {
if f.createAccountErr != nil {
return f.createAccountErr
}
key := acct.ProfileID + "|" + acct.JWKThumbprint
if _, exists := f.thumbToAccount[key]; exists {
return errors.New("duplicate")
}
cp := *acct
cp.CreatedAt = time.Now()
cp.UpdatedAt = cp.CreatedAt
f.accounts[acct.AccountID] = &cp
f.thumbToAccount[key] = acct.AccountID
return nil
}
func (f *fakeACMERepo) GetAccountByID(ctx context.Context, accountID string) (*domain.ACMEAccount, error) {
acct, ok := f.accounts[accountID]
if !ok {
return nil, repository.ErrNotFound
}
cp := *acct
return &cp, nil
}
func (f *fakeACMERepo) GetAccountByThumbprint(ctx context.Context, profileID, thumbprint string) (*domain.ACMEAccount, error) {
id, ok := f.thumbToAccount[profileID+"|"+thumbprint]
if !ok {
return nil, repository.ErrNotFound
}
return f.GetAccountByID(ctx, id)
}
func (f *fakeACMERepo) UpdateAccountContactWithTx(ctx context.Context, q repository.Querier, accountID string, contact []string) error {
acct, ok := f.accounts[accountID]
if !ok {
return repository.ErrNotFound
}
acct.Contact = contact
acct.UpdatedAt = time.Now()
return nil
}
func (f *fakeACMERepo) UpdateAccountStatusWithTx(ctx context.Context, q repository.Querier, accountID string, status domain.ACMEAccountStatus) error {
acct, ok := f.accounts[accountID]
if !ok {
return repository.ErrNotFound
}
acct.Status = status
acct.UpdatedAt = time.Now()
return nil
}
// fakeTransactor is the repository.Transactor stand-in: runs fn
// against the supplied querier (we just pass nil — fakes ignore it).
// Mirrors how production transactor works without an actual DB.
type fakeTransactor struct{}
func (f *fakeTransactor) WithinTx(ctx context.Context, fn func(q repository.Querier) error) error {
return fn(nil)
}
// fakeAuditRepo records the audit events fakeAuditService emits so
// tests can assert on the audit row count + shape.
type fakeAuditRepo struct {
events []*domain.AuditEvent
}
func (f *fakeAuditRepo) Create(ctx context.Context, event *domain.AuditEvent) error {
return f.CreateWithTx(ctx, nil, event)
}
func (f *fakeAuditRepo) CreateWithTx(ctx context.Context, q repository.Querier, event *domain.AuditEvent) error {
cp := *event
f.events = append(f.events, &cp)
return nil
}
func (f *fakeAuditRepo) List(ctx context.Context, filter *repository.AuditFilter) ([]*domain.AuditEvent, error) {
return f.events, nil
}
// fakeProfileLookup is an in-memory profileLookup that returns the
// profile by ID. Unknown IDs return repository.ErrNotFound (the
// canonical sentinel ACMEService maps to ErrACMEProfileNotFound).
type fakeProfileLookup struct {
profiles map[string]*domain.CertificateProfile
}
func (f *fakeProfileLookup) Get(ctx context.Context, id string) (*domain.CertificateProfile, error) {
p, ok := f.profiles[id]
if !ok {
return nil, repository.ErrNotFound
}
return p, nil
}
func newSvc(t *testing.T, cfg config.ACMEServerConfig, profiles map[string]*domain.CertificateProfile) (*ACMEService, *fakeACMERepo) {
t.Helper()
repo := newFakeACMERepo()
pl := &fakeProfileLookup{profiles: profiles}
return NewACMEService(repo, pl, cfg), repo
}
// newSvcWithAudit returns a service wired with the transactor + audit
// service required by the JWS-authenticated POST endpoints.
func newSvcWithAudit(t *testing.T, cfg config.ACMEServerConfig, profiles map[string]*domain.CertificateProfile) (*ACMEService, *fakeACMERepo, *fakeAuditRepo) {
t.Helper()
repo := newFakeACMERepo()
pl := &fakeProfileLookup{profiles: profiles}
auditRepo := &fakeAuditRepo{}
auditSvc := NewAuditService(auditRepo)
svc := NewACMEService(repo, pl, cfg)
svc.SetTransactor(&fakeTransactor{})
svc.SetAuditService(auditSvc)
return svc, repo, auditRepo
}
func TestBuildDirectory_HappyPath(t *testing.T) {
cfg := config.ACMEServerConfig{
NonceTTL: 5 * time.Minute,
}
cfg.DirectoryMeta.TermsOfService = "https://example.com/tos"
cfg.DirectoryMeta.Website = "https://example.com"
svc, _ := newSvc(t, cfg, map[string]*domain.CertificateProfile{
"prof-corp": {ID: "prof-corp", Name: "corp"},
})
dir, err := svc.BuildDirectory(context.Background(), "prof-corp", "https://server/acme/profile/prof-corp")
if err != nil {
t.Fatalf("BuildDirectory: %v", err)
}
if dir == nil {
t.Fatal("dir is nil")
}
if dir.NewNonce != "https://server/acme/profile/prof-corp/new-nonce" {
t.Errorf("NewNonce = %q", dir.NewNonce)
}
if dir.Meta == nil || dir.Meta.TermsOfService != "https://example.com/tos" {
t.Errorf("meta tos = %+v", dir.Meta)
}
if got := svc.Metrics().DirectoryTotal.Load(); got != 1 {
t.Errorf("DirectoryTotal = %d, want 1", got)
}
}
func TestBuildDirectory_UnknownProfile(t *testing.T) {
cfg := config.ACMEServerConfig{NonceTTL: 5 * time.Minute}
svc, _ := newSvc(t, cfg, nil)
_, err := svc.BuildDirectory(context.Background(), "prof-missing", "https://server/acme/profile/prof-missing")
if !errors.Is(err, ErrACMEProfileNotFound) {
t.Errorf("err = %v, want ErrACMEProfileNotFound", err)
}
if got := svc.Metrics().DirectoryFailureTotal.Load(); got != 1 {
t.Errorf("DirectoryFailureTotal = %d, want 1", got)
}
}
func TestBuildDirectory_EmptyProfileNoDefault(t *testing.T) {
cfg := config.ACMEServerConfig{NonceTTL: 5 * time.Minute}
svc, _ := newSvc(t, cfg, nil)
_, err := svc.BuildDirectory(context.Background(), "", "https://server/acme")
if !errors.Is(err, ErrACMEUserActionRequired) {
t.Errorf("err = %v, want ErrACMEUserActionRequired", err)
}
}
func TestBuildDirectory_EmptyProfileWithDefault(t *testing.T) {
cfg := config.ACMEServerConfig{
NonceTTL: 5 * time.Minute,
DefaultProfileID: "prof-default",
}
svc, _ := newSvc(t, cfg, map[string]*domain.CertificateProfile{
"prof-default": {ID: "prof-default", Name: "default"},
})
dir, err := svc.BuildDirectory(context.Background(), "", "https://server/acme")
if err != nil {
t.Fatalf("BuildDirectory: %v", err)
}
if dir.NewNonce != "https://server/acme/new-nonce" {
t.Errorf("NewNonce = %q (shorthand path)", dir.NewNonce)
}
}
func TestIssueNonce_HappyPath(t *testing.T) {
cfg := config.ACMEServerConfig{NonceTTL: 5 * time.Minute}
svc, repo := newSvc(t, cfg, nil)
n, err := svc.IssueNonce(context.Background())
if err != nil {
t.Fatalf("IssueNonce: %v", err)
}
if len(n) != 43 {
t.Errorf("nonce length = %d, want 43 (base64url-no-pad of 32 bytes)", len(n))
}
if _, ok := repo.issued[n]; !ok {
t.Errorf("issued nonce was not persisted")
}
if got := svc.Metrics().NewNonceTotal.Load(); got != 1 {
t.Errorf("NewNonceTotal = %d, want 1", got)
}
}
func TestIssueNonce_RepoFailure(t *testing.T) {
cfg := config.ACMEServerConfig{NonceTTL: 5 * time.Minute}
svc, repo := newSvc(t, cfg, nil)
repo.issueErr = errors.New("disk full")
_, err := svc.IssueNonce(context.Background())
if err == nil {
t.Fatal("expected error from IssueNonce when repo fails")
}
if got := svc.Metrics().NewNonceFailureTotal.Load(); got != 1 {
t.Errorf("NewNonceFailureTotal = %d, want 1", got)
}
}
func TestACMEMetrics_Snapshot(t *testing.T) {
m := NewACMEMetrics()
m.DirectoryTotal.Store(7)
m.NewNonceTotal.Store(11)
m.NewNonceFailureTotal.Store(2)
m.NewAccountTotal.Store(3)
m.NewAccountIdempotentTotal.Store(1)
snap := m.Snapshot()
if snap["certctl_acme_directory_total"] != 7 {
t.Errorf("directory_total = %d", snap["certctl_acme_directory_total"])
}
if snap["certctl_acme_new_nonce_total"] != 11 {
t.Errorf("new_nonce_total = %d", snap["certctl_acme_new_nonce_total"])
}
if snap["certctl_acme_new_nonce_failures_total"] != 2 {
t.Errorf("new_nonce_failures_total = %d", snap["certctl_acme_new_nonce_failures_total"])
}
if snap["certctl_acme_new_account_total"] != 3 {
t.Errorf("new_account_total = %d", snap["certctl_acme_new_account_total"])
}
if snap["certctl_acme_new_account_idempotent_total"] != 1 {
t.Errorf("new_account_idempotent_total = %d", snap["certctl_acme_new_account_idempotent_total"])
}
}
// --- Phase 1b — account management -------------------------------------
func mustGenJWK(t *testing.T) *jose.JSONWebKey {
t.Helper()
k, err := rsa.GenerateKey(rand.Reader, 2048)
if err != nil {
t.Fatalf("rsa keygen: %v", err)
}
return &jose.JSONWebKey{Key: &k.PublicKey}
}
func TestNewAccount_HappyPath(t *testing.T) {
cfg := config.ACMEServerConfig{NonceTTL: 5 * time.Minute}
svc, repo, auditRepo := newSvcWithAudit(t, cfg, map[string]*domain.CertificateProfile{
"prof-corp": {ID: "prof-corp", Name: "corp"},
})
jwk := mustGenJWK(t)
acct, isNew, err := svc.NewAccount(context.Background(), "prof-corp", jwk, []string{"mailto:a@example.com"}, false, true)
if err != nil {
t.Fatalf("NewAccount: %v", err)
}
if !isNew {
t.Errorf("isNew = false; want true")
}
if acct == nil || acct.AccountID == "" || acct.JWKThumbprint == "" {
t.Fatalf("account row is malformed: %+v", acct)
}
if got := svc.Metrics().NewAccountTotal.Load(); got != 1 {
t.Errorf("NewAccountTotal = %d, want 1", got)
}
if got := len(auditRepo.events); got != 1 {
t.Errorf("audit events = %d, want 1", got)
}
if got := auditRepo.events[0].Action; got != "acme_account_created" {
t.Errorf("audit action = %q", got)
}
if _, ok := repo.accounts[acct.AccountID]; !ok {
t.Errorf("account row not in repo")
}
}
func TestNewAccount_Idempotent_ExistingJWKReturnsExistingRow(t *testing.T) {
cfg := config.ACMEServerConfig{NonceTTL: 5 * time.Minute}
svc, _, auditRepo := newSvcWithAudit(t, cfg, map[string]*domain.CertificateProfile{
"prof-corp": {ID: "prof-corp", Name: "corp"},
})
jwk := mustGenJWK(t)
first, _, err := svc.NewAccount(context.Background(), "prof-corp", jwk, []string{"mailto:a@example.com"}, false, true)
if err != nil {
t.Fatalf("first NewAccount: %v", err)
}
second, isNew, err := svc.NewAccount(context.Background(), "prof-corp", jwk, []string{"mailto:b@example.com"}, false, true)
if err != nil {
t.Fatalf("second NewAccount: %v", err)
}
if isNew {
t.Errorf("isNew = true on idempotent re-registration; want false")
}
if second.AccountID != first.AccountID {
t.Errorf("second account ID = %q; want first %q", second.AccountID, first.AccountID)
}
// Idempotent re-registration MUST NOT update contact / write a
// second audit row (RFC 8555 §7.3.1 says return the existing row
// unmodified).
if got := len(auditRepo.events); got != 1 {
t.Errorf("audit events = %d after idempotent call; want 1", got)
}
if got := svc.Metrics().NewAccountIdempotentTotal.Load(); got != 1 {
t.Errorf("NewAccountIdempotentTotal = %d, want 1", got)
}
}
func TestNewAccount_OnlyReturnExisting_NoMatch(t *testing.T) {
cfg := config.ACMEServerConfig{NonceTTL: 5 * time.Minute}
svc, _, _ := newSvcWithAudit(t, cfg, map[string]*domain.CertificateProfile{
"prof-corp": {ID: "prof-corp", Name: "corp"},
})
jwk := mustGenJWK(t)
_, _, err := svc.NewAccount(context.Background(), "prof-corp", jwk, nil, true /*onlyReturnExisting*/, false)
if !errors.Is(err, ErrACMEAccountDoesNotExist) {
t.Errorf("err = %v; want ErrACMEAccountDoesNotExist", err)
}
}
func TestNewAccount_OnlyReturnExisting_Match(t *testing.T) {
cfg := config.ACMEServerConfig{NonceTTL: 5 * time.Minute}
svc, _, _ := newSvcWithAudit(t, cfg, map[string]*domain.CertificateProfile{
"prof-corp": {ID: "prof-corp", Name: "corp"},
})
jwk := mustGenJWK(t)
first, _, err := svc.NewAccount(context.Background(), "prof-corp", jwk, nil, false, false)
if err != nil {
t.Fatalf("first: %v", err)
}
second, isNew, err := svc.NewAccount(context.Background(), "prof-corp", jwk, nil, true, false)
if err != nil {
t.Fatalf("second: %v", err)
}
if isNew {
t.Errorf("isNew = true; want false")
}
if second.AccountID != first.AccountID {
t.Errorf("ids differ")
}
}
func TestUpdateAccount_HappyPath(t *testing.T) {
cfg := config.ACMEServerConfig{NonceTTL: 5 * time.Minute}
svc, _, auditRepo := newSvcWithAudit(t, cfg, map[string]*domain.CertificateProfile{
"prof-corp": {ID: "prof-corp", Name: "corp"},
})
jwk := mustGenJWK(t)
acct, _, err := svc.NewAccount(context.Background(), "prof-corp", jwk, []string{"mailto:old@example.com"}, false, false)
if err != nil {
t.Fatalf("seed: %v", err)
}
updated, err := svc.UpdateAccount(context.Background(), acct.AccountID, []string{"mailto:new@example.com"})
if err != nil {
t.Fatalf("UpdateAccount: %v", err)
}
if len(updated.Contact) != 1 || updated.Contact[0] != "mailto:new@example.com" {
t.Errorf("contact = %v", updated.Contact)
}
// Two audit rows: the create + the update.
if got := len(auditRepo.events); got != 2 {
t.Errorf("audit events = %d, want 2", got)
}
if got := auditRepo.events[1].Action; got != "acme_account_updated" {
t.Errorf("update audit action = %q", got)
}
}
func TestDeactivateAccount_HappyPath(t *testing.T) {
cfg := config.ACMEServerConfig{NonceTTL: 5 * time.Minute}
svc, _, auditRepo := newSvcWithAudit(t, cfg, map[string]*domain.CertificateProfile{
"prof-corp": {ID: "prof-corp", Name: "corp"},
})
jwk := mustGenJWK(t)
acct, _, err := svc.NewAccount(context.Background(), "prof-corp", jwk, nil, false, false)
if err != nil {
t.Fatalf("seed: %v", err)
}
deactivated, err := svc.DeactivateAccount(context.Background(), acct.AccountID)
if err != nil {
t.Fatalf("DeactivateAccount: %v", err)
}
if deactivated.Status != domain.ACMEAccountStatusDeactivated {
t.Errorf("status = %q, want %q", deactivated.Status, domain.ACMEAccountStatusDeactivated)
}
if got := svc.Metrics().DeactivateAccountTotal.Load(); got != 1 {
t.Errorf("DeactivateAccountTotal = %d, want 1", got)
}
if got := auditRepo.events[len(auditRepo.events)-1].Action; got != "acme_account_deactivated" {
t.Errorf("last audit action = %q", got)
}
}
func TestLookupAccount_NotFound(t *testing.T) {
cfg := config.ACMEServerConfig{NonceTTL: 5 * time.Minute}
svc, _ := newSvc(t, cfg, nil)
_, err := svc.LookupAccount(context.Background(), "acme-acc-missing")
if !errors.Is(err, ErrACMEAccountNotFound) {
t.Errorf("err = %v; want ErrACMEAccountNotFound", err)
}
}
func TestNewAccount_RequiresTransactor(t *testing.T) {
cfg := config.ACMEServerConfig{NonceTTL: 5 * time.Minute}
// Use newSvc (no transactor wired) — NewAccount should refuse.
svc, _ := newSvc(t, cfg, map[string]*domain.CertificateProfile{
"prof-corp": {ID: "prof-corp", Name: "corp"},
})
jwk := mustGenJWK(t)
_, _, err := svc.NewAccount(context.Background(), "prof-corp", jwk, nil, false, false)
if err == nil {
t.Fatal("expected error when transactor is unset")
}
}