salrashid123 / signer

golang crypto.Signer for Trusted Platform Module (TPM) and Google Cloud KMS
Apache License 2.0
30 stars 10 forks source link
golang hardware-security-module hashicorp-vault key-management-service kms trusted-platform-module

crypto.Signer, implementations for Google Cloud KMS and Trusted Platform Modules

where private keys as embedded inside:

Basically, you will get a crypto.Signer interface where the private keys are saved on those platform.

Use the signer to create a TLS session, sign CA/CSRs, generate signed url or just sign anything.

Some implementations:

other stuff:

see the example/ folder for more information.


this library is not supported by google


Usage Signer

Initialize a signer and directly use .sign() as shown in this sample for GCS SignedURL:

Usage TLS

Sign/Verify PSS

see example/sign_verify* folders

Sign/Verify ECC

The default output signature format for ECC based keys is ASN1 format as described in ecdsa.SignASN1

If you need the raw output format, set ECCRawOutput: true in the config.

See the examples folder for usage

Usage: Generate self-signed certificate

see util/

go run certgen/certgen.go -cn server.domain.com

Usage: Generate CSR

see util/csrgen/

go run certgen/certgen.go -cn server.domain.com

If you just want to issue JWT's, see

TPM Signer Device management

NOTE there will be a breaking change if you are using this library for TPM based signature after v0.8.0. The new structure uses the tpm-direct API. If you would rather use the tpm2/legacy branch, please use the signer at v0.7.2. Library managed device was removed (it seems tpm resource managers work well enough...I'm clearly on the fence here given the recent commits..)

The TPM device is managed externally outside of the signer. You have to instantiate the TPM device ReadWriteCloser and client.Key outside of the library and pass that in.

The advantage of this is you control it opening and closing.

    rwc, err := OpenTPM(*tpmPath)
    rwr := transport.FromReadWriter(rwc)

    pub, err := tpm2.ReadPublic{
        ObjectHandle: tpm2.TPMHandle(*handle),
    }.Execute(rwr)

    r, err := saltpm.NewTPMCrypto(&saltpm.TPM{
        TpmDevice: rwc,
        NamedHandle: &tpm2.NamedHandle{
            Handle: tpm2.TPMHandle(*handle),
            Name:   pub.Name,
        },
    })

    s, err := r.Sign(rand.Reader, digest, crypto.SHA256)

    // close the TPM if you are done signing
    rwc.Close()

    // you need to reinitialize NewTPMCrypto if you 
    // want to sign again after closing

TODO use a backoff retry similar to tpmrand to prevent contention.