Skip to content

Secure Releases β€” Checksum Verification

GTB's self-update flow verifies every downloaded binary against a GoReleaser-produced checksums.txt manifest before installing it. A tampered or truncated binary is rejected; a passing check is logged at INFO ("checksum verified") and the update proceeds.

This is Phase 1 of the release-integrity work from 2026-04-02-remote-update-checksum-verification. Phase 2 adds a GPG signature over the manifest, closing the same-origin trust gap (an attacker who can replace the binary on the release platform can also replace checksums.txt β€” only a signature from an off-platform key defeats that).

How it fits together

Update() β†’  findReleaseAsset()           = target binary
         β†’  fetchChecksumsManifest()     = checksums.txt (via ChecksumProvider or asset list)
         β†’  VerifyChecksumFromManifest() = binary SHA-256 vs manifest entry
         β†’  extract()                    = only reached when verify succeeds

checksums.txt is GoReleaser's default manifest β€” one <hex-sha256> <filename> entry per line. If your .goreleaser.yaml uses the defaults, no changes are needed; the file is already attached to every release.

Producing verifiable releases

The default GoReleaser checksum block generates checksums.txt and attaches it to the release. No configuration change is required. Verify locally with:

just snapshot
ls dist/checksums.txt

Manual / CI pipelines

If you don't use GoReleaser, produce the manifest with standard sha256sum output and upload it alongside the binaries. The file format is:

<64-hex-hash>  <filename>
<64-hex-hash>  <filename>

Blank lines at end-of-file are tolerated; every other line must match or the whole manifest is rejected (a truncated manifest never produces a false pass).

Bitbucket

Upload checksums.txt to the repository's Downloads alongside the binaries (same upload flow as your release assets). The Bitbucket provider looks it up by exact filename β€” not via the asset-name regex that the binary uses.

Direct HTTP releases

Set checksum_url_template in your ReleaseSource.Params to a URL template that expands to the manifest location:

props.Tool.ReleaseSource = props.ReleaseSource{
    Type: "direct",
    Params: map[string]string{
        "url_template":          "https://releases.example.com/{tool}/{version}/{tool}_{os}_{arch}.{ext}",
        "checksum_url_template": "https://releases.example.com/{tool}/{version}/checksums.txt",
    },
}

The same placeholders ({version}, {version_bare}, {os}, {arch}, {tool}, {ext}) are available.

Consuming (tool author)

Pick a failure mode

By default, a release without checksums.txt logs a warning and the update proceeds. This preserves backward compatibility for tools whose existing releases predate this feature. Once your tool has shipped at least one release with a manifest, flip the default to fail-closed:

package main

import "gitlab.com/phpboyscout/go-tool-base/pkg/setup"

func main() {
    setup.DefaultRequireChecksum = true
    // ...
}

Overriding at runtime

End users can override via config file:

update:
  require_checksum: true          # abort if manifest missing or verification fails
  checksum_asset_name: ""         # override default "checksums.txt" filename

…or via env var (respects the tool's configured env prefix):

export MYTOOL_UPDATE_REQUIRE_CHECKSUM=true

Config wins over env var; env var wins over setup.DefaultRequireChecksum.

GTB itself ships with setup.DefaultRequireChecksum = true β€” every gtb update verifies. Override with GTB_UPDATE_REQUIRE_CHECKSUM=false or update.require_checksum: false in config only if you need to update across a legacy release that predates the manifest (all GoReleaser-built releases have it, so this should rarely apply).

Size bounds

The manifest download is capped at setup.MaxChecksumsSize (1 MiB); the binary download at setup.MaxBinaryDownloadSize (512 MiB). A hostile server streaming beyond those bounds aborts with ErrChecksumTooLarge. Tool authors who legitimately ship larger artefacts can reassign either variable before calling Update():

setup.MaxBinaryDownloadSize = 2 << 30 // 2 GiB

Phase 2: GPG-signed manifests (planned)

Phase 1 defends against accidental corruption and single-asset tampering, but a full VCS compromise can replace both the binary and checksums.txt on the release. Phase 2 closes that gap by signing the manifest with a project-controlled GPG key β€” an attacker who replaces the files on the VCS still cannot produce a valid checksums.txt.sig without access to the private key.

Status: designed, not yet implemented. See the spec for the full design and the Phase 2 Signing Prep checklist for the ordered operational decisions (KMS choice, WKD endpoint, rollout sequence) that gate implementation. This section documents the intended shape of the public API and configuration so tool authors can plan their signing pipeline.

Trust model at a glance

A signature is only as trustworthy as the key used to verify it. Phase 2 uses a composite trust set: the verifier loads public keys from two independent sources and requires their fingerprints to agree before accepting a signature.

β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”      β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”
β”‚  embedded in binary β”‚      β”‚   external: Web Key Directory β”‚
β”‚  (//go:embed)       β”‚      β”‚   or custom HTTPS endpoint    β”‚
β””β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”¬β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”˜      β””β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”¬β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”˜
           β”‚                                β”‚
           └──────────►  CompositeResolver β—„β”˜
                             fingerprints must match
                                     β”‚
                                     β–Ό
                              TrustSet ──► verify(checksums.txt.sig)
  • Embedded key β€” baked into each binary at build time via //go:embed. Works offline and in air-gapped environments. Rotates only when a new binary is shipped.
  • External key (third-party source) β€” fetched from an HTTPS endpoint under a domain you control. For a VCS compromise to produce a valid signature, the attacker must also control your DNS and TLS termination; the two trust anchors are administered independently. The canonical implementation is Web Key Directory (WKD), an OpenPGP RFC-draft serving public keys from a well-known path. Other HTTPS endpoints (self-hosted, Vault, a static S3 bucket) are supported via a custom KeyResolver.

Resolver implementations

// Interface β€” implement this to plug in any key source.
type KeyResolver interface {
    Resolve(ctx context.Context) (*TrustSet, error)
}

Three ship with GTB:

Resolver Source Offline? Primary use
setup.EmbeddedResolver //go:embed of *.asc files in internal/version/trustkeys/ βœ… Yes Always available; the fallback that keeps air-gapped updates working.
setup.WKDResolver https://openpgpkey.<domain>/.well-known/openpgpkey/<domain>/hu/<z-base-32>?l=<email> ❌ No The project's public key published via the GPG WKD standard; cross-checks the embedded copy.
setup.CompositeResolver{Embedded, WKD} Both, with fingerprint-equality enforcement ⚠️ Partial The production default. Offline builds still work via update.key_source=embedded.

Configuration surface (planned)

update:
  require_signature: false               # library default; flip on via DefaultRequireSignature
  key_source: both                       # "embedded" | "external" | "both"
  external_key_email: [email protected]  # drives the WKD URL
  require_external_crosscheck: false     # true β†’ WKD failure aborts update
  signature_asset_name: ""               # override default "checksums.txt.sig"

Compile-time overrides (tool authors in main):

setup.DefaultRequireSignature = true
setup.DefaultKeySource = "both"
setup.DefaultExternalKeyEmail = "[email protected]"
setup.DefaultRequireExternalCrosscheck = true

Publishing a public key

  1. Generate an Ed25519 signing keypair (RSA-4096 is acceptable if your KMS doesn't support Ed25519). DSA, 1024-bit RSA, and weak curves are refused at load time.
  2. Embed the public half. Drop the ASCII-armored file at internal/version/trustkeys/signing-key-v1.asc in your repo β€” go:embed picks it up at build time. Tests gate a CI check that refuses any accidentally committed private key.
  3. Publish the same key via your chosen external source:
  4. WKD β€” serve the ASCII-armored key at the WKD path under openpgpkey.<yourdomain>. DNS and TLS cert are your trust anchors, administered independently from your VCS.
  5. Custom HTTPS β€” implement KeyResolver with your own endpoint (Vault, static S3, internal CA-served HTTPS). Register it via setup.WithKeyResolver on SelfUpdater.
  6. Store the private half in a KMS (AWS/GCP/Azure), Vault Transit, or a hardware token. GitHub encrypted secrets are a last resort β€” see the spec's Key Management section.

Custom resolvers (third-party key source)

import "gitlab.com/phpboyscout/go-tool-base/pkg/setup"

type VaultResolver struct { /* ... */ }

func (r *VaultResolver) Resolve(ctx context.Context) (*setup.TrustSet, error) {
    // Fetch ASCII-armored key from Vault KV, call setup.LoadTrustSet,
    // return the resulting TrustSet (enforces the minimum-strength policy).
}

func main() {
    resolver := setup.CompositeResolver{
        setup.EmbeddedResolver{},
        &VaultResolver{ /* ... */ },
    }
    // Wire it in at SelfUpdater construction:
    //   setup.NewUpdater(ctx, props, version, force, setup.WithKeyResolver(resolver))
}

Any implementation must:

  • Return a *TrustSet containing only keys that passed the minimum-strength policy.
  • Honour the context's deadline and cancellation.
  • Cap response bodies at setup.MaxWKDResponseSize (64 KiB) or an equivalent bound.
  • Not leak private material anywhere β€” log.Fatal if it ever sees a secret key at load time.

Key rotation

The trust set is a set, not a single key. During a rotation window, ship releases signed by both the old and new key; the verifier accepts either. Once all supported versions of the tool include the new key in their trust set, drop the old key from both the embedded trustkeys directory and the WKD endpoint.

For emergency rotation (compromise of the primary signing key), the design reserves a second "rotation-authority" key whose private half is stored offline. A release signed by the rotation-authority carries a rotate-keys.json manifest; the next update rewrites the embedded trust set from that manifest. This is documented in the spec and deferred to Phase 4.


Testing

Run the Phase 1 tests:

go test ./pkg/setup/ -run "TestVerifyChecksum|TestVerifyAssetChecksum|TestFindChecksumsAsset"

…and the manifest fuzzer:

go test ./pkg/setup/ -run "^$" -fuzz=FuzzParseChecksumManifest -fuzztime=30s