diff --git a/.github/workflows/release.yml b/.github/workflows/release.yml index 807cfdd6..b4336472 100644 --- a/.github/workflows/release.yml +++ b/.github/workflows/release.yml @@ -12,7 +12,7 @@ jobs: runs-on: ubuntu-20.04 strategy: matrix: - go: [ '1.17', '1.18' ] + go: [ '1.18', '1.19' ] outputs: is_prerelease: ${{ steps.is_prerelease.outputs.IS_PRERELEASE }} steps: @@ -32,26 +32,8 @@ jobs: name: golangci-lint uses: golangci/golangci-lint-action@v2 with: - # Optional: version of golangci-lint to use in form of v1.2 or v1.2.3 or `latest` to use the latest version - version: 'v1.45.2' - - # Optional: working directory, useful for monorepos - # working-directory: somedir - - # Optional: golangci-lint command line arguments. + version: ${{ secrets.GOLANGCI_LINT_VERSION }} args: --timeout=30m - - # Optional: show only new issues if it's a pull request. The default value is `false`. - # only-new-issues: true - - # Optional: if set to true then the action will use pre-installed Go. - # skip-go-installation: true - - # Optional: if set to true then the action don't cache or restore ~/go/pkg. - # skip-pkg-cache: true - - # Optional: if set to true then the action don't cache or restore ~/.cache/go-build. - # skip-build-cache: true - name: Test, Build id: lint_test_build @@ -106,7 +88,7 @@ jobs: name: Set up Go uses: actions/setup-go@v2 with: - go-version: 1.18 + go-version: 1.19 - name: APT Install id: aptInstall @@ -159,7 +141,7 @@ jobs: name: Setup Go uses: actions/setup-go@v2 with: - go-version: '1.18' + go-version: '1.19' - name: Install cosign uses: sigstore/cosign-installer@v1.1.0 diff --git a/.github/workflows/test.yml b/.github/workflows/test.yml index 046589af..8a2f391c 100644 --- a/.github/workflows/test.yml +++ b/.github/workflows/test.yml @@ -14,7 +14,7 @@ jobs: runs-on: ubuntu-20.04 strategy: matrix: - go: [ '1.17', '1.18' ] + go: [ '1.18', '1.19' ] steps: - name: Checkout @@ -32,33 +32,15 @@ jobs: name: golangci-lint uses: golangci/golangci-lint-action@v2 with: - # Optional: version of golangci-lint to use in form of v1.2 or v1.2.3 or `latest` to use the latest version - version: 'v1.45.2' - - # Optional: working directory, useful for monorepos - # working-directory: somedir - - # Optional: golangci-lint command line arguments. + version: ${{ secrets.GOLANGCI_LINT_VERSION }} args: --timeout=30m - - # Optional: show only new issues if it's a pull request. The default value is `false`. - # only-new-issues: true - - # Optional: if set to true then the action will use pre-installed Go. - # skip-go-installation: true - - # Optional: if set to true then the action don't cache or restore ~/go/pkg. - # skip-pkg-cache: true - - # Optional: if set to true then the action don't cache or restore ~/.cache/go-build. - # skip-build-cache: true - name: Test, Build id: lint_test_build run: V=1 make ci - name: Codecov - if: matrix.go == '1.18' + if: matrix.go == '1.19' uses: codecov/codecov-action@v2 with: token: ${{ secrets.CODECOV_TOKEN }} diff --git a/CHANGELOG.md b/CHANGELOG.md index f6134aa5..b2d6ee81 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -16,8 +16,19 @@ and this project adheres to [Semantic Versioning](http://semver.org/spec/v2.0.0. --- ## [Unreleased] +### Added +- Added automatic configuration of Linked RAs. +- Send provisioner configuration on Linked RAs. ### Changed -- Certificates signed by an issuer using an RSA key will be signed using the same algorithm as the issuer certificate was signed with. The signature will no longer default to PKCS #1. For example, if the issuer certificate was signed using RSA-PSS with SHA-256, a new certificate will also be signed using RSA-PSS with SHA-256. +- Certificates signed by an issuer using an RSA key will be signed using the + same algorithm used to sign the issuer certificate. The signature will no + longer default to PKCS #1. For example, if the issuer certificate was signed + using RSA-PSS with SHA-256, a new certificate will also be signed using + RSA-PSS with SHA-256. +- Support two latest versions of Go (1.18, 1.19). +- Validate revocation serial number (either base 10 or prefixed with an + appropriate base). +- Sanitize TLS options. ## [0.20.0] - 2022-05-26 ### Added diff --git a/acme/api/middleware_test.go b/acme/api/middleware_test.go index 193f5347..e43f6f99 100644 --- a/acme/api/middleware_test.go +++ b/acme/api/middleware_test.go @@ -19,6 +19,7 @@ import ( "github.com/smallstep/certificates/acme" "github.com/smallstep/nosql/database" "go.step.sm/crypto/jose" + "go.step.sm/crypto/keyutil" ) var testBody = []byte("foo") @@ -1136,6 +1137,8 @@ func TestHandler_validateJWS(t *testing.T) { } }, "fail/rsa-key-too-small": func(t *testing.T) test { + revert := keyutil.Insecure() + defer revert() jwk, err := jose.GenerateJWK("RSA", "", "", "sig", "", 1024) assert.FatalError(t, err) pub := jwk.Public() diff --git a/acme/api/order.go b/acme/api/order.go index 679fe32f..2927a620 100644 --- a/acme/api/order.go +++ b/acme/api/order.go @@ -13,6 +13,7 @@ import ( "github.com/go-chi/chi" "go.step.sm/crypto/randutil" + "go.step.sm/crypto/x509util" "github.com/smallstep/certificates/acme" "github.com/smallstep/certificates/api/render" @@ -33,12 +34,20 @@ func (n *NewOrderRequest) Validate() error { return acme.NewError(acme.ErrorMalformedType, "identifiers list cannot be empty") } for _, id := range n.Identifiers { - if !(id.Type == acme.DNS || id.Type == acme.IP) { + switch id.Type { + case acme.IP: + if net.ParseIP(id.Value) == nil { + return acme.NewError(acme.ErrorMalformedType, "invalid IP address: %s", id.Value) + } + case acme.DNS: + value, _ := trimIfWildcard(id.Value) + if _, err := x509util.SanitizeName(value); err != nil { + return acme.NewError(acme.ErrorMalformedType, "invalid DNS name: %s", id.Value) + } + default: return acme.NewError(acme.ErrorMalformedType, "identifier type unsupported: %s", id.Type) } - if id.Type == acme.IP && net.ParseIP(id.Value) == nil { - return acme.NewError(acme.ErrorMalformedType, "invalid IP address: %s", id.Value) - } + // TODO(hs): add some validations for DNS domains? // TODO(hs): combine the errors from this with allow/deny policy, like example error in https://datatracker.ietf.org/doc/html/rfc8555#section-6.7.1 } @@ -218,13 +227,19 @@ func newACMEPolicyEngine(eak *acme.ExternalAccountKey) (policy.X509Policy, error return policy.NewX509PolicyEngine(eak.Policy) } +func trimIfWildcard(value string) (string, bool) { + if strings.HasPrefix(value, "*.") { + return strings.TrimPrefix(value, "*."), true + } + return value, false +} + func newAuthorization(ctx context.Context, az *acme.Authorization) error { - if strings.HasPrefix(az.Identifier.Value, "*.") { - az.Wildcard = true - az.Identifier = acme.Identifier{ - Value: strings.TrimPrefix(az.Identifier.Value, "*."), - Type: az.Identifier.Type, - } + value, isWildcard := trimIfWildcard(az.Identifier.Value) + az.Wildcard = isWildcard + az.Identifier = acme.Identifier{ + Value: value, + Type: az.Identifier.Type, } chTypes := challengeTypes(az) diff --git a/acme/api/order_test.go b/acme/api/order_test.go index 7f67c72e..724357d8 100644 --- a/acme/api/order_test.go +++ b/acme/api/order_test.go @@ -49,6 +49,36 @@ func TestNewOrderRequest_Validate(t *testing.T) { err: acme.NewError(acme.ErrorMalformedType, "identifier type unsupported: foo"), } }, + "fail/bad-identifier/bad-dns": func(t *testing.T) test { + return test{ + nor: &NewOrderRequest{ + Identifiers: []acme.Identifier{ + {Type: "dns", Value: "xn--bücher.example.com"}, + }, + }, + err: acme.NewError(acme.ErrorMalformedType, "invalid DNS name: xn--bücher.example.com"), + } + }, + "fail/bad-identifier/dns-port": func(t *testing.T) test { + return test{ + nor: &NewOrderRequest{ + Identifiers: []acme.Identifier{ + {Type: "dns", Value: "example.com:8080"}, + }, + }, + err: acme.NewError(acme.ErrorMalformedType, "invalid DNS name: example.com:8080"), + } + }, + "fail/bad-identifier/dns-wildcard-port": func(t *testing.T) test { + return test{ + nor: &NewOrderRequest{ + Identifiers: []acme.Identifier{ + {Type: "dns", Value: "*.example.com:8080"}, + }, + }, + err: acme.NewError(acme.ErrorMalformedType, "invalid DNS name: *.example.com:8080"), + } + }, "fail/bad-ip": func(t *testing.T) test { nbf := time.Now().UTC().Add(time.Minute) naf := time.Now().UTC().Add(5 * time.Minute) @@ -72,7 +102,7 @@ func TestNewOrderRequest_Validate(t *testing.T) { nor: &NewOrderRequest{ Identifiers: []acme.Identifier{ {Type: "dns", Value: "example.com"}, - {Type: "dns", Value: "bar.com"}, + {Type: "dns", Value: "*.bar.com"}, }, NotAfter: naf, NotBefore: nbf, @@ -2097,3 +2127,32 @@ func TestHandler_challengeTypes(t *testing.T) { }) } } + +func TestTrimIfWildcard(t *testing.T) { + tests := []struct { + name string + arg string + wantValue string + wantBool bool + }{ + { + name: "no trim", + arg: "smallstep.com", + wantValue: "smallstep.com", + wantBool: false, + }, + { + name: "trim", + arg: "*.smallstep.com", + wantValue: "smallstep.com", + wantBool: true, + }, + } + for _, tt := range tests { + t.Run(tt.name, func(t *testing.T) { + v, ok := trimIfWildcard(tt.arg) + assert.Equals(t, v, tt.wantValue) + assert.Equals(t, ok, tt.wantBool) + }) + } +} diff --git a/acme/challenge.go b/acme/challenge.go index 8d8466bd..96637627 100644 --- a/acme/challenge.go +++ b/acme/challenge.go @@ -149,7 +149,7 @@ func tlsalpn01Validate(ctx context.Context, ch *Challenge, db DB, jwk *jose.JSON // [RFC5246] or higher when connecting to clients for validation. MinVersion: tls.VersionTLS12, ServerName: serverName(ch), - InsecureSkipVerify: true, // we expect a self-signed challenge certificate + InsecureSkipVerify: true, // nolint:gosec // we expect a self-signed challenge certificate } hostPort := net.JoinHostPort(ch.Value, "443") diff --git a/acme/client.go b/acme/client.go index 31f4c975..cf5f8c09 100644 --- a/acme/client.go +++ b/acme/client.go @@ -56,7 +56,8 @@ func NewClient() Client { Timeout: 30 * time.Second, Transport: &http.Transport{ TLSClientConfig: &tls.Config{ - InsecureSkipVerify: true, + // nolint:gosec // used on tls-alpn-01 challenge + InsecureSkipVerify: true, // lgtm[go/disabled-certificate-check] }, }, }, diff --git a/api/api_test.go b/api/api_test.go index 1f27ab8c..4c84871a 100644 --- a/api/api_test.go +++ b/api/api_test.go @@ -1437,7 +1437,7 @@ func Test_fmtPublicKey(t *testing.T) { if err != nil { t.Fatal(err) } - rsa1024, err := rsa.GenerateKey(rand.Reader, 1024) + rsa2048, err := rsa.GenerateKey(rand.Reader, 2048) if err != nil { t.Fatal(err) } @@ -1463,7 +1463,7 @@ func Test_fmtPublicKey(t *testing.T) { want string }{ {"p256", args{p256.Public(), p256, nil}, "ECDSA P-256"}, - {"rsa1024", args{rsa1024.Public(), rsa1024, nil}, "RSA 1024"}, + {"rsa2048", args{rsa2048.Public(), rsa2048, nil}, "RSA 2048"}, {"ed25519", args{edPub, edPriv, nil}, "Ed25519"}, {"dsa2048", args{cert: &x509.Certificate{PublicKeyAlgorithm: x509.DSA, PublicKey: &dsa2048.PublicKey}}, "DSA 2048"}, {"unknown", args{cert: &x509.Certificate{PublicKeyAlgorithm: x509.ECDSA, PublicKey: []byte("12345678")}}, "ECDSA unknown"}, diff --git a/api/log/log.go b/api/log/log.go index cb31410b..e5c8c45a 100644 --- a/api/log/log.go +++ b/api/log/log.go @@ -3,7 +3,6 @@ package log import ( "fmt" - "log" "net/http" "os" @@ -28,8 +27,6 @@ type StackTracedError interface { func Error(rw http.ResponseWriter, err error) { rl, ok := rw.(logging.ResponseLogger) if !ok { - log.Println(err) - return } @@ -72,8 +69,6 @@ func EnabledResponse(rw http.ResponseWriter, v interface{}) { rl.WithFields(map[string]interface{}{ "response": out, }) - } else { - log.Println(out) } } } diff --git a/api/revoke.go b/api/revoke.go index aebbb875..4221696a 100644 --- a/api/revoke.go +++ b/api/revoke.go @@ -1,6 +1,7 @@ package api import ( + "math/big" "net/http" "golang.org/x/crypto/ocsp" @@ -33,6 +34,11 @@ func (r *RevokeRequest) Validate() (err error) { if r.Serial == "" { return errs.BadRequest("missing serial") } + sn, ok := new(big.Int).SetString(r.Serial, 0) + if !ok { + return errs.BadRequest("'%s' is not a valid serial number - use a base 10 representation or a base 16 representation with '0x' prefix", r.Serial) + } + r.Serial = sn.String() if r.ReasonCode < ocsp.Unspecified || r.ReasonCode > ocsp.AACompromise { return errs.BadRequest("reasonCode out of bounds") } diff --git a/api/revoke_test.go b/api/revoke_test.go index c3fa6ceb..0955244e 100644 --- a/api/revoke_test.go +++ b/api/revoke_test.go @@ -31,9 +31,13 @@ func TestRevokeRequestValidate(t *testing.T) { rr: &RevokeRequest{}, err: &errs.Error{Err: errors.New("missing serial"), Status: http.StatusBadRequest}, }, + "error/bad sn": { + rr: &RevokeRequest{Serial: "sn"}, + err: &errs.Error{Err: errors.New("'sn' is not a valid serial number - use a base 10 representation or a base 16 representation with '0x' prefix"), Status: http.StatusBadRequest}, + }, "error/bad reasonCode": { rr: &RevokeRequest{ - Serial: "sn", + Serial: "10", ReasonCode: 15, Passive: true, }, @@ -41,7 +45,7 @@ func TestRevokeRequestValidate(t *testing.T) { }, "error/non-passive not implemented": { rr: &RevokeRequest{ - Serial: "sn", + Serial: "10", ReasonCode: 8, Passive: false, }, @@ -49,7 +53,7 @@ func TestRevokeRequestValidate(t *testing.T) { }, "ok": { rr: &RevokeRequest{ - Serial: "sn", + Serial: "10", ReasonCode: 9, Passive: true, }, @@ -97,7 +101,7 @@ func Test_caHandler_Revoke(t *testing.T) { }, "200/ott": func(t *testing.T) test { input, err := json.Marshal(RevokeRequest{ - Serial: "sn", + Serial: "10", ReasonCode: 4, Reason: "foo", OTT: "valid", @@ -114,7 +118,7 @@ func Test_caHandler_Revoke(t *testing.T) { revoke: func(ctx context.Context, opts *authority.RevokeOptions) error { assert.True(t, opts.PassiveOnly) assert.False(t, opts.MTLS) - assert.Equals(t, opts.Serial, "sn") + assert.Equals(t, opts.Serial, "10") assert.Equals(t, opts.ReasonCode, 4) assert.Equals(t, opts.Reason, "foo") return nil @@ -125,7 +129,7 @@ func Test_caHandler_Revoke(t *testing.T) { }, "400/no OTT and no peer certificate": func(t *testing.T) test { input, err := json.Marshal(RevokeRequest{ - Serial: "sn", + Serial: "10", ReasonCode: 4, Passive: true, }) @@ -176,7 +180,7 @@ func Test_caHandler_Revoke(t *testing.T) { }, "500/ott authority.Revoke": func(t *testing.T) test { input, err := json.Marshal(RevokeRequest{ - Serial: "sn", + Serial: "10", ReasonCode: 4, Reason: "foo", OTT: "valid", @@ -198,7 +202,7 @@ func Test_caHandler_Revoke(t *testing.T) { }, "403/ott authority.Revoke": func(t *testing.T) test { input, err := json.Marshal(RevokeRequest{ - Serial: "sn", + Serial: "10", ReasonCode: 4, Reason: "foo", OTT: "valid", diff --git a/authority/admin/api/provisioner.go b/authority/admin/api/provisioner.go index b879b9c7..c584361b 100644 --- a/authority/admin/api/provisioner.go +++ b/authority/admin/api/provisioner.go @@ -202,7 +202,7 @@ func UpdateProvisioner(w http.ResponseWriter, r *http.Request) { } // validateTemplates validates the X.509 and SSH templates and template data if set. -func validateTemplates(x509 *linkedca.Template, ssh *linkedca.Template) error { +func validateTemplates(x509, ssh *linkedca.Template) error { if x509 != nil { if len(x509.Template) > 0 { if err := x509util.ValidateTemplate(x509.Template); err != nil { diff --git a/authority/admin/api/provisioner_test.go b/authority/admin/api/provisioner_test.go index f1267990..86f8a31b 100644 --- a/authority/admin/api/provisioner_test.go +++ b/authority/admin/api/provisioner_test.go @@ -355,7 +355,7 @@ func TestHandler_CreateProvisioner(t *testing.T) { Type: linkedca.Provisioner_OIDC, Name: "provName", X509Template: &linkedca.Template{ - Template: []byte("{!?}"), + Template: []byte(`{ {{missingFunction "foo"}} }`), }, } body, err := protojson.Marshal(prov) @@ -368,7 +368,7 @@ func TestHandler_CreateProvisioner(t *testing.T) { Type: "badRequest", Status: 400, Detail: "bad request", - Message: "invalid template: invalid X.509 template: invalid JSON: invalid character '!' looking for beginning of object key string", + Message: "invalid template: invalid X.509 template: error parsing template: template: template:1: function \"missingFunction\" not defined", }, } }, @@ -973,7 +973,7 @@ func TestHandler_UpdateProvisioner(t *testing.T) { CreatedAt: timestamppb.New(createdAt), DeletedAt: timestamppb.New(deletedAt), X509Template: &linkedca.Template{ - Template: []byte("{!?}"), + Template: []byte("{ {{ missingFunction }} }"), }, } body, err := protojson.Marshal(prov) @@ -1010,7 +1010,7 @@ func TestHandler_UpdateProvisioner(t *testing.T) { Type: "badRequest", Status: 400, Detail: "bad request", - Message: "invalid template: invalid X.509 template: invalid JSON: invalid character '!' looking for beginning of object key string", + Message: "invalid template: invalid X.509 template: error parsing template: template: template:1: function \"missingFunction\" not defined", }, } }, @@ -1220,13 +1220,13 @@ func Test_validateTemplates(t *testing.T) { err: nil, }, { - name: "fail/x509-template-trailing-comma", + name: "fail/x509-template-missing-quote", args: args{ x509: &linkedca.Template{ - Template: []byte(`{"x": 1,}`), + Template: []byte(`{ {{printf "%q" "quoted}} }`), }, }, - err: errors.New("invalid X.509 template: invalid JSON: invalid character '}' looking for beginning of object key string"), + err: errors.New("invalid X.509 template: error parsing template: template: template:1: unterminated quoted string"), }, { name: "fail/x509-template-data", @@ -1235,16 +1235,16 @@ func Test_validateTemplates(t *testing.T) { Data: []byte(`{!?}`), }, }, - err: errors.New("invalid X.509 template data: invalid JSON: invalid character '!' looking for beginning of object key string"), + err: errors.New("invalid X.509 template data: error validating json template data"), }, { - name: "fail/ssh-template-trailing-comma", + name: "fail/ssh-template-unknown-function", args: args{ ssh: &linkedca.Template{ - Template: []byte(`{"x": 1,}`), + Template: []byte(`{ {{unknownFunction "foo"}} }`), }, }, - err: errors.New("invalid SSH template: invalid JSON: invalid character '}' looking for beginning of object key string"), + err: errors.New("invalid SSH template: error parsing template: template: template:1: function \"unknownFunction\" not defined"), }, { name: "fail/ssh-template-data", @@ -1253,7 +1253,7 @@ func Test_validateTemplates(t *testing.T) { Data: []byte(`{!?}`), }, }, - err: errors.New("invalid SSH template data: invalid JSON: invalid character '!' looking for beginning of object key string"), + err: errors.New("invalid SSH template data: error validating json template data"), }, } for _, tt := range tests { diff --git a/authority/authority.go b/authority/authority.go index 933ceb14..73aa9cca 100644 --- a/authority/authority.go +++ b/authority/authority.go @@ -14,6 +14,9 @@ import ( "github.com/pkg/errors" "golang.org/x/crypto/ssh" + "go.step.sm/crypto/kms" + kmsapi "go.step.sm/crypto/kms/apiv1" + "go.step.sm/crypto/kms/sshagentkms" "go.step.sm/crypto/pemutil" "go.step.sm/linkedca" @@ -26,9 +29,6 @@ import ( "github.com/smallstep/certificates/cas" casapi "github.com/smallstep/certificates/cas/apiv1" "github.com/smallstep/certificates/db" - "github.com/smallstep/certificates/kms" - kmsapi "github.com/smallstep/certificates/kms/apiv1" - "github.com/smallstep/certificates/kms/sshagentkms" "github.com/smallstep/certificates/scep" "github.com/smallstep/certificates/templates" "github.com/smallstep/nosql" @@ -312,6 +312,7 @@ func (a *Authority) init() error { if id := a.config.AuthorityConfig.AuthorityID; id != "" && !strings.EqualFold(id, linkedcaClient.authorityID) { return errors.New("error initializing linkedca: token authority and configured authority do not match") } + a.config.AuthorityConfig.AuthorityID = linkedcaClient.authorityID linkedcaClient.Run() } @@ -322,6 +323,9 @@ func (a *Authority) init() error { options = *a.config.AuthorityConfig.Options } + // AuthorityID might be empty. It's always available linked CAs/RAs. + options.AuthorityID = a.config.AuthorityConfig.AuthorityID + // Configure linked RA if linkedcaClient != nil && options.CertificateAuthority == "" { conf, err := linkedcaClient.GetConfiguration(ctx) @@ -335,6 +339,19 @@ func (a *Authority) init() error { Type: conf.RaConfig.Provisioner.Type.String(), Provisioner: conf.RaConfig.Provisioner.Name, } + // Configure the RA authority type if needed + if options.Type == "" { + options.Type = casapi.StepCAS + } + } + // Remote configuration is currently only supported on a linked RA + if sc := conf.ServerConfig; sc != nil { + if a.config.Address == "" { + a.config.Address = sc.Address + } + if len(a.config.DNSNames) == 0 { + a.config.DNSNames = sc.DnsNames + } } } @@ -357,7 +374,6 @@ func (a *Authority) init() error { return err } } - a.x509CAService, err = cas.New(ctx, options) if err != nil { return err @@ -428,7 +444,7 @@ func (a *Authority) init() error { // erroring out with: ssh: unsupported key type *agent.Key switch s := signer.(type) { case *sshagentkms.WrappedSSHSigner: - a.sshCAHostCertSignKey = s.Sshsigner + a.sshCAHostCertSignKey = s.Signer case crypto.Signer: a.sshCAHostCertSignKey, err = ssh.NewSignerFromSigner(s) default: @@ -454,7 +470,7 @@ func (a *Authority) init() error { // erroring out with: ssh: unsupported key type *agent.Key switch s := signer.(type) { case *sshagentkms.WrappedSSHSigner: - a.sshCAUserCertSignKey = s.Sshsigner + a.sshCAUserCertSignKey = s.Signer case crypto.Signer: a.sshCAUserCertSignKey, err = ssh.NewSignerFromSigner(s) default: diff --git a/authority/config/config.go b/authority/config/config.go index c764e8f9..c5e74b39 100644 --- a/authority/config/config.go +++ b/authority/config/config.go @@ -9,13 +9,13 @@ import ( "github.com/pkg/errors" + kms "go.step.sm/crypto/kms/apiv1" "go.step.sm/linkedca" "github.com/smallstep/certificates/authority/policy" "github.com/smallstep/certificates/authority/provisioner" cas "github.com/smallstep/certificates/cas/apiv1" "github.com/smallstep/certificates/db" - kms "github.com/smallstep/certificates/kms/apiv1" "github.com/smallstep/certificates/templates" ) @@ -72,6 +72,7 @@ type Config struct { Password string `json:"password,omitempty"` Templates *templates.Templates `json:"templates,omitempty"` CommonName string `json:"commonName,omitempty"` + SkipValidation bool `json:"-"` } // ASN1DN contains ASN1.DN attributes that are used in Subject and Issuer @@ -102,6 +103,7 @@ type AuthConfig struct { DisableIssuedAtCheck bool `json:"disableIssuedAtCheck,omitempty"` Backdate *provisioner.Duration `json:"backdate,omitempty"` EnableAdmin bool `json:"enableAdmin,omitempty"` + DisableGetSSHHosts bool `json:"disableGetSSHHosts,omitempty"` } // init initializes the required fields in the AuthConfig if they are not @@ -200,6 +202,8 @@ func (c *Config) Save(filename string) error { // Validate validates the configuration. func (c *Config) Validate() error { switch { + case c.SkipValidation: + return nil case c.Address == "": return errors.New("address cannot be empty") case len(c.DNSNames) == 0: diff --git a/authority/config/config_test.go b/authority/config/config_test.go index 5a05b3f6..9b5b26aa 100644 --- a/authority/config/config_test.go +++ b/authority/config/config_test.go @@ -35,9 +35,16 @@ func TestConfigValidate(t *testing.T) { type ConfigValidateTest struct { config *Config err error - tls TLSOptions + tls *TLSOptions } tests := map[string]func(*testing.T) ConfigValidateTest{ + "skip-validation": func(t *testing.T) ConfigValidateTest { + return ConfigValidateTest{ + config: &Config{ + SkipValidation: true, + }, + } + }, "empty-address": func(t *testing.T) ConfigValidateTest { return ConfigValidateTest{ config: &Config{ @@ -128,7 +135,7 @@ func TestConfigValidate(t *testing.T) { Password: "pass", AuthorityConfig: ac, }, - tls: DefaultTLSOptions, + tls: &DefaultTLSOptions, } }, "empty-TLS-values": func(t *testing.T) ConfigValidateTest { @@ -143,7 +150,7 @@ func TestConfigValidate(t *testing.T) { AuthorityConfig: ac, TLS: &TLSOptions{}, }, - tls: DefaultTLSOptions, + tls: &DefaultTLSOptions, } }, "custom-tls-values": func(t *testing.T) ConfigValidateTest { @@ -165,7 +172,7 @@ func TestConfigValidate(t *testing.T) { Renegotiation: true, }, }, - tls: TLSOptions{ + tls: &TLSOptions{ CipherSuites: CipherSuites{ "TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305", }, @@ -209,9 +216,9 @@ func TestConfigValidate(t *testing.T) { } } else { if assert.Nil(t, tc.err) { - fmt.Printf("tc.tls = %+v\n", tc.tls) - fmt.Printf("*tc.config.TLS = %+v\n", *tc.config.TLS) - assert.Equals(t, *tc.config.TLS, tc.tls) + fmt.Printf("tc.tls = %v\n", tc.tls) + fmt.Printf("*tc.config.TLS = %v\n", tc.config.TLS) + assert.Equals(t, tc.config.TLS, tc.tls) } } }) diff --git a/authority/config/tls_options.go b/authority/config/tls_options.go index 0db202e5..01ab3d0a 100644 --- a/authority/config/tls_options.go +++ b/authority/config/tls_options.go @@ -22,18 +22,19 @@ var ( } // ApprovedTLSCipherSuites smallstep approved ciphersuites. ApprovedTLSCipherSuites = CipherSuites{ - "TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA", - "TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA256", + // AEADs w/ ECDHE "TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256", - "TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA", - "TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384", - "TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305_SHA256", - "TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA", - "TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA256", "TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256", - "TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA", + "TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384", "TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384", + "TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305_SHA256", "TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305_SHA256", + + // CBC w/ ECDHE + "TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA", + "TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA", + "TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA", + "TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA", } // DefaultTLSOptions represents the default TLS version as well as the cipher // suites used in the TLS certificates. @@ -117,22 +118,22 @@ func (c CipherSuites) Value() []uint16 { // cipherSuites has the list of supported cipher suites. var cipherSuites = map[string]uint16{ // TLS 1.0 - 1.2 cipher suites. - "TLS_RSA_WITH_RC4_128_SHA": tls.TLS_RSA_WITH_RC4_128_SHA, + "TLS_RSA_WITH_RC4_128_SHA": tls.TLS_RSA_WITH_RC4_128_SHA, // lgtm[go/insecure-tls] "TLS_RSA_WITH_3DES_EDE_CBC_SHA": tls.TLS_RSA_WITH_3DES_EDE_CBC_SHA, "TLS_RSA_WITH_AES_128_CBC_SHA": tls.TLS_RSA_WITH_AES_128_CBC_SHA, "TLS_RSA_WITH_AES_256_CBC_SHA": tls.TLS_RSA_WITH_AES_256_CBC_SHA, - "TLS_RSA_WITH_AES_128_CBC_SHA256": tls.TLS_RSA_WITH_AES_128_CBC_SHA256, + "TLS_RSA_WITH_AES_128_CBC_SHA256": tls.TLS_RSA_WITH_AES_128_CBC_SHA256, // lgtm[go/insecure-tls] "TLS_RSA_WITH_AES_128_GCM_SHA256": tls.TLS_RSA_WITH_AES_128_GCM_SHA256, "TLS_RSA_WITH_AES_256_GCM_SHA384": tls.TLS_RSA_WITH_AES_256_GCM_SHA384, - "TLS_ECDHE_ECDSA_WITH_RC4_128_SHA": tls.TLS_ECDHE_ECDSA_WITH_RC4_128_SHA, + "TLS_ECDHE_ECDSA_WITH_RC4_128_SHA": tls.TLS_ECDHE_ECDSA_WITH_RC4_128_SHA, // lgtm[go/insecure-tls] "TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA": tls.TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA, "TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA": tls.TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA, - "TLS_ECDHE_RSA_WITH_RC4_128_SHA": tls.TLS_ECDHE_RSA_WITH_RC4_128_SHA, + "TLS_ECDHE_RSA_WITH_RC4_128_SHA": tls.TLS_ECDHE_RSA_WITH_RC4_128_SHA, // lgtm[go/insecure-tls] "TLS_ECDHE_RSA_WITH_3DES_EDE_CBC_SHA": tls.TLS_ECDHE_RSA_WITH_3DES_EDE_CBC_SHA, "TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA": tls.TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA, "TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA": tls.TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA, - "TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA256": tls.TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA256, - "TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA256": tls.TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA256, + "TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA256": tls.TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA256, // lgtm[go/insecure-tls] + "TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA256": tls.TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA256, // lgtm[go/insecure-tls] "TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256": tls.TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256, "TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256": tls.TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256, "TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384": tls.TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384, @@ -146,8 +147,8 @@ var cipherSuites = map[string]uint16{ "TLS_CHACHA20_POLY1305_SHA256": tls.TLS_CHACHA20_POLY1305_SHA256, // Legacy names. - "TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305": tls.TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305, - "TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305": tls.TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305, + "TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305": tls.TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305_SHA256, + "TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305": tls.TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305_SHA256, } // TLSOptions represents the TLS options that can be specified on *tls.Config @@ -168,6 +169,7 @@ func (t *TLSOptions) TLSConfig() *tls.Config { rs = tls.RenegotiateNever } + // nolint:gosec // default MinVersion 1.2, if defined but empty 1.3 is used return &tls.Config{ CipherSuites: t.CipherSuites.Value(), MinVersion: t.MinVersion.Value(), diff --git a/authority/export.go b/authority/export.go index 14229823..b8efbbd7 100644 --- a/authority/export.go +++ b/authority/export.go @@ -89,7 +89,7 @@ func (a *Authority) Export() (c *linkedca.Configuration, err error) { if v.Type == "" { typ = int32(linkedca.KMS_SOFTKMS) } else { - typ, ok = linkedca.KMS_Type_value[strings.ToUpper(v.Type)] + typ, ok = linkedca.KMS_Type_value[strings.ToUpper(string(v.Type))] if !ok { return nil, errors.Errorf("unsupported kms type %s", v.Type) } diff --git a/authority/linkedca.go b/authority/linkedca.go index 0b98f877..133ae616 100644 --- a/authority/linkedca.go +++ b/authority/linkedca.go @@ -273,10 +273,13 @@ func (c *linkedCaClient) GetCertificateData(serial string) (*db.CertificateData, func (c *linkedCaClient) StoreCertificateChain(p provisioner.Interface, fullchain ...*x509.Certificate) error { ctx, cancel := context.WithTimeout(context.Background(), 15*time.Second) defer cancel() + raProvisioner, endpointID := createRegistrationAuthorityProvisioner(p) _, err := c.client.PostCertificate(ctx, &linkedca.CertificateRequest{ PemCertificate: serializeCertificateChain(fullchain[0]), PemCertificateChain: serializeCertificateChain(fullchain[1:]...), Provisioner: createProvisionerIdentity(p), + RaProvisioner: raProvisioner, + EndpointId: endpointID, }) return errors.Wrap(err, "error posting certificate") } @@ -392,6 +395,26 @@ func createProvisionerIdentity(p provisioner.Interface) *linkedca.ProvisionerIde } } +type raProvisioner interface { + RAInfo() *provisioner.RAInfo +} + +func createRegistrationAuthorityProvisioner(p provisioner.Interface) (*linkedca.RegistrationAuthorityProvisioner, string) { + if rap, ok := p.(raProvisioner); ok { + info := rap.RAInfo() + typ := linkedca.Provisioner_Type_value[strings.ToUpper(info.ProvisionerType)] + return &linkedca.RegistrationAuthorityProvisioner{ + AuthorityId: info.AuthorityID, + Provisioner: &linkedca.ProvisionerIdentity{ + Id: info.ProvisionerID, + Type: linkedca.Provisioner_Type(typ), + Name: info.ProvisionerName, + }, + }, info.EndpointID + } + return nil, "" +} + func serializeCertificate(crt *x509.Certificate) string { if crt == nil { return "" @@ -438,7 +461,8 @@ func getRootCertificate(endpoint, fingerprint string) (*x509.Certificate, error) defer cancel() conn, err := grpc.DialContext(ctx, endpoint, grpc.WithTransportCredentials(credentials.NewTLS(&tls.Config{ - InsecureSkipVerify: true, + // nolint:gosec // used in bootstrap protocol + InsecureSkipVerify: true, // lgtm[go/disabled-certificate-check] }))) if err != nil { return nil, errors.Wrapf(err, "error connecting %s", endpoint) @@ -491,7 +515,8 @@ func login(authority, token string, csr *x509.CertificateRequest, signer crypto. defer cancel() conn, err := grpc.DialContext(ctx, endpoint, grpc.WithTransportCredentials(credentials.NewTLS(&tls.Config{ - RootCAs: rootCAs, + MinVersion: tls.VersionTLS12, + RootCAs: rootCAs, }))) if err != nil { return nil, nil, errors.Wrapf(err, "error connecting %s", endpoint) @@ -567,6 +592,7 @@ func login(authority, token string, csr *x509.CertificateRequest, signer crypto. rootCAs.AddCert(bundle[last]) return cert, &tls.Config{ - RootCAs: rootCAs, + MinVersion: tls.VersionTLS12, + RootCAs: rootCAs, }, nil } diff --git a/authority/options.go b/authority/options.go index 6e1949f5..9cef89f0 100644 --- a/authority/options.go +++ b/authority/options.go @@ -7,14 +7,16 @@ import ( "encoding/pem" "github.com/pkg/errors" + "golang.org/x/crypto/ssh" + + "go.step.sm/crypto/kms" + "github.com/smallstep/certificates/authority/admin" "github.com/smallstep/certificates/authority/config" "github.com/smallstep/certificates/authority/provisioner" "github.com/smallstep/certificates/cas" casapi "github.com/smallstep/certificates/cas/apiv1" "github.com/smallstep/certificates/db" - "github.com/smallstep/certificates/kms" - "golang.org/x/crypto/ssh" ) // Option sets options to the Authority. diff --git a/authority/provisioner/aws.go b/authority/provisioner/aws.go index a5b403a4..463a4aee 100644 --- a/authority/provisioner/aws.go +++ b/authority/provisioner/aws.go @@ -35,6 +35,7 @@ const awsIdentityURL = "http://169.254.169.254/latest/dynamic/instance-identity/ const awsSignatureURL = "http://169.254.169.254/latest/dynamic/instance-identity/signature" // awsAPITokenURL is the url used to get the IMDSv2 API token +// nolint:gosec // no credentials here const awsAPITokenURL = "http://169.254.169.254/latest/api/token" // awsAPITokenTTL is the default TTL to use when requesting IMDSv2 API tokens @@ -42,9 +43,11 @@ const awsAPITokenURL = "http://169.254.169.254/latest/api/token" const awsAPITokenTTL = "30" // awsMetadataTokenHeader is the header that must be passed with every IMDSv2 request +// nolint:gosec // no credentials here const awsMetadataTokenHeader = "X-aws-ec2-metadata-token" // awsMetadataTokenTTLHeader is the header used to indicate the token TTL requested +// nolint:gosec // no credentials here const awsMetadataTokenTTLHeader = "X-aws-ec2-metadata-token-ttl-seconds" // awsCertificate is the certificate used to validate the instance identity diff --git a/authority/provisioner/aws_test.go b/authority/provisioner/aws_test.go index d12d0626..0660c3f0 100644 --- a/authority/provisioner/aws_test.go +++ b/authority/provisioner/aws_test.go @@ -316,7 +316,7 @@ func TestAWS_authorizeToken(t *testing.T) { } key, err := x509.ParsePKCS1PrivateKey(block.Bytes) assert.FatalError(t, err) - badKey, err := rsa.GenerateKey(rand.Reader, 1024) + badKey, err := rsa.GenerateKey(rand.Reader, 2048) assert.FatalError(t, err) type test struct { @@ -579,7 +579,7 @@ func TestAWS_AuthorizeSign(t *testing.T) { key, err := x509.ParsePKCS1PrivateKey(block.Bytes) assert.FatalError(t, err) - badKey, err := rsa.GenerateKey(rand.Reader, 1024) + badKey, err := rsa.GenerateKey(rand.Reader, 2048) assert.FatalError(t, err) t4, err := generateAWSToken( @@ -748,6 +748,7 @@ func TestAWS_AuthorizeSSHSign(t *testing.T) { pub := key.Public().Key rsa2048, err := rsa.GenerateKey(rand.Reader, 2048) assert.FatalError(t, err) + // nolint:gosec // tests minimum size of the key rsa1024, err := rsa.GenerateKey(rand.Reader, 1024) assert.FatalError(t, err) diff --git a/authority/provisioner/azure.go b/authority/provisioner/azure.go index b6f7ec91..3f714a3e 100644 --- a/authority/provisioner/azure.go +++ b/authority/provisioner/azure.go @@ -25,6 +25,7 @@ import ( const azureOIDCBaseURL = "https://login.microsoftonline.com" // azureIdentityTokenURL is the URL to get the identity token for an instance. +// nolint:gosec // no credentials here const azureIdentityTokenURL = "http://169.254.169.254/metadata/identity/oauth2/token?api-version=2018-02-01&resource=https%3A%2F%2Fmanagement.azure.com%2F" // azureDefaultAudience is the default audience used. diff --git a/authority/provisioner/azure_test.go b/authority/provisioner/azure_test.go index 3e745a5b..7f8b70d0 100644 --- a/authority/provisioner/azure_test.go +++ b/authority/provisioner/azure_test.go @@ -624,6 +624,7 @@ func TestAzure_AuthorizeSSHSign(t *testing.T) { pub := key.Public().Key rsa2048, err := rsa.GenerateKey(rand.Reader, 2048) assert.FatalError(t, err) + // nolint:gosec // tests minimum size of the key rsa1024, err := rsa.GenerateKey(rand.Reader, 1024) assert.FatalError(t, err) diff --git a/authority/provisioner/collection.go b/authority/provisioner/collection.go index 8bbace5f..85b489c1 100644 --- a/authority/provisioner/collection.go +++ b/authority/provisioner/collection.go @@ -1,7 +1,7 @@ package provisioner import ( - "crypto/sha1" + "crypto/sha1" // nolint:gosec // not used for cryptographic security "crypto/x509" "encoding/asn1" "encoding/binary" @@ -319,6 +319,7 @@ func loadProvisioner(m *sync.Map, key string) (Interface, bool) { // provisionerSum returns the SHA1 of the provisioners ID. From this we will // create the unique and sorted id. func provisionerSum(p Interface) []byte { + // nolint:gosec // not used for cryptographic security sum := sha1.Sum([]byte(p.GetID())) return sum[:] } diff --git a/authority/provisioner/gcp_test.go b/authority/provisioner/gcp_test.go index 3c0bf92e..3d6b5d75 100644 --- a/authority/provisioner/gcp_test.go +++ b/authority/provisioner/gcp_test.go @@ -623,6 +623,7 @@ func TestGCP_AuthorizeSSHSign(t *testing.T) { pub := key.Public().Key rsa2048, err := rsa.GenerateKey(rand.Reader, 2048) assert.FatalError(t, err) + // nolint:gosec // tests minimum size of the key rsa1024, err := rsa.GenerateKey(rand.Reader, 1024) assert.FatalError(t, err) diff --git a/authority/provisioner/jwk.go b/authority/provisioner/jwk.go index de592941..5cfb0409 100644 --- a/authority/provisioner/jwk.go +++ b/authority/provisioner/jwk.go @@ -24,6 +24,7 @@ type jwtPayload struct { type stepPayload struct { SSH *SignSSHOptions `json:"ssh,omitempty"` + RA *RAInfo `json:"ra,omitempty"` } // JWK is the default provisioner, an entity that can sign tokens necessary for @@ -172,8 +173,17 @@ func (p *JWK) AuthorizeSign(ctx context.Context, token string) ([]SignOption, er return nil, errs.Wrap(http.StatusInternalServerError, err, "jwk.AuthorizeSign") } + // Wrap provisioner if the token is an RA token. + var self Interface = p + if claims.Step != nil && claims.Step.RA != nil { + self = &raProvisioner{ + Interface: p, + raInfo: claims.Step.RA, + } + } + return []SignOption{ - p, + self, templateOptions, // modifiers / withOptions newProvisionerExtensionOption(TypeJWK, p.Name, p.Key.KeyID), diff --git a/authority/provisioner/jwk_test.go b/authority/provisioner/jwk_test.go index bd8b542b..723ccf56 100644 --- a/authority/provisioner/jwk_test.go +++ b/authority/provisioner/jwk_test.go @@ -411,6 +411,7 @@ func TestJWK_AuthorizeSSHSign(t *testing.T) { pub := key.Public().Key rsa2048, err := rsa.GenerateKey(rand.Reader, 2048) assert.FatalError(t, err) + // nolint:gosec // tests minimum size of the key rsa1024, err := rsa.GenerateKey(rand.Reader, 1024) assert.FatalError(t, err) diff --git a/authority/provisioner/keystore.go b/authority/provisioner/keystore.go index d1811fab..8b276a75 100644 --- a/authority/provisioner/keystore.go +++ b/authority/provisioner/keystore.go @@ -85,14 +85,14 @@ func (ks *keyStore) reload() { // 0 it will randomly rotate between 0-12 hours, but every time we call to Get // it will automatically rotate. func (ks *keyStore) nextReloadDuration(age time.Duration) time.Duration { - n := rand.Int63n(int64(ks.jitter)) + n := rand.Int63n(int64(ks.jitter)) // nolint:gosec // not used for cryptographic security age -= time.Duration(n) return abs(age) } func getKeysFromJWKsURI(uri string) (jose.JSONWebKeySet, time.Duration, error) { var keys jose.JSONWebKeySet - resp, err := http.Get(uri) + resp, err := http.Get(uri) // nolint:gosec // openid-configuration jwks_uri if err != nil { return keys, 0, errors.Wrapf(err, "failed to connect to %s", uri) } diff --git a/authority/provisioner/oidc.go b/authority/provisioner/oidc.go index e64d98d9..c1bcc741 100644 --- a/authority/provisioner/oidc.go +++ b/authority/provisioner/oidc.go @@ -464,7 +464,7 @@ func (o *OIDC) AuthorizeSSHRevoke(ctx context.Context, token string) error { } func getAndDecode(uri string, v interface{}) error { - resp, err := http.Get(uri) + resp, err := http.Get(uri) // nolint:gosec // openid-configuration uri if err != nil { return errors.Wrapf(err, "failed to connect to %s", uri) } diff --git a/authority/provisioner/oidc_test.go b/authority/provisioner/oidc_test.go index 3d039496..62ea3f24 100644 --- a/authority/provisioner/oidc_test.go +++ b/authority/provisioner/oidc_test.go @@ -535,6 +535,7 @@ func TestOIDC_AuthorizeSSHSign(t *testing.T) { pub := key.Public().Key rsa2048, err := rsa.GenerateKey(rand.Reader, 2048) assert.FatalError(t, err) + // nolint:gosec // tests minimum size of the key rsa1024, err := rsa.GenerateKey(rand.Reader, 1024) assert.FatalError(t, err) diff --git a/authority/provisioner/options_test.go b/authority/provisioner/options_test.go index 0bcf9ec3..652fff73 100644 --- a/authority/provisioner/options_test.go +++ b/authority/provisioner/options_test.go @@ -254,6 +254,7 @@ func TestCustomTemplateOptions(t *testing.T) { } func Test_unsafeParseSigned(t *testing.T) { + // nolint:gosec // no credentials here okToken := "eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9.eyJzdWIiOiJqYW5lQGRvZS5jb20iLCJpc3MiOiJodHRwczovL2RvZS5jb20iLCJqdGkiOiI4ZmYzMjQ4MS1mZDVmLTRlMmUtOTZkZi05MDhjMTI3Yzg1ZjciLCJpYXQiOjE1OTUzNjAwMjgsImV4cCI6MTU5NTM2MzYyOH0.aid8UuhFucJOFHXaob9zpNtVvhul9ulTGsA52mU6XIw" type args struct { s string diff --git a/authority/provisioner/provisioner.go b/authority/provisioner/provisioner.go index 0d5cd41a..29d44c1c 100644 --- a/authority/provisioner/provisioner.go +++ b/authority/provisioner/provisioner.go @@ -340,6 +340,27 @@ type Permissions struct { CriticalOptions map[string]string `json:"criticalOptions"` } +// RAInfo is the information about a provisioner present in RA tokens generated +// by StepCAS. +type RAInfo struct { + AuthorityID string `json:"authorityId,omitempty"` + EndpointID string `json:"endpointId,omitempty"` + ProvisionerID string `json:"provisionerId,omitempty"` + ProvisionerType string `json:"provisionerType,omitempty"` + ProvisionerName string `json:"provisionerName,omitempty"` +} + +// raProvisioner wraps a provisioner with RA data. +type raProvisioner struct { + Interface + raInfo *RAInfo +} + +// RAInfo returns the RAInfo in the wrapped provisioner. +func (p *raProvisioner) RAInfo() *RAInfo { + return p.raInfo +} + // MockProvisioner for testing type MockProvisioner struct { Mret1, Mret2, Mret3 interface{} diff --git a/authority/provisioner/utils_test.go b/authority/provisioner/utils_test.go index 0a1d176c..265c7b08 100644 --- a/authority/provisioner/utils_test.go +++ b/authority/provisioner/utils_test.go @@ -449,6 +449,7 @@ func generateAWSWithServer() (*AWS, *httptest.Server, error) { if err != nil { return nil, nil, errors.Wrap(err, "error signing document") } + // nolint:gosec // tests minimum size of the key token := "AQAEAEEO9-7Z88ewKFpboZuDlFYWz9A3AN-wMOVzjEhfAyXW31BvVw==" srv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { switch r.URL.Path { diff --git a/authority/provisioner/x5c.go b/authority/provisioner/x5c.go index b9ae24c5..9f9a0e4e 100644 --- a/authority/provisioner/x5c.go +++ b/authority/provisioner/x5c.go @@ -221,8 +221,17 @@ func (p *X5C) AuthorizeSign(ctx context.Context, token string) ([]SignOption, er return nil, errs.Wrap(http.StatusInternalServerError, err, "jwk.AuthorizeSign") } + // Wrap provisioner if the token is an RA token. + var self Interface = p + if claims.Step != nil && claims.Step.RA != nil { + self = &raProvisioner{ + Interface: p, + raInfo: claims.Step.RA, + } + } + return []SignOption{ - p, + self, templateOptions, // modifiers / withOptions newProvisionerExtensionOption(TypeX5C, p.Name, ""), diff --git a/authority/ssh.go b/authority/ssh.go index 1fd7f2e8..d8d5375c 100644 --- a/authority/ssh.go +++ b/authority/ssh.go @@ -602,6 +602,9 @@ func (a *Authority) CheckSSHHost(ctx context.Context, principal, token string) ( // GetSSHHosts returns a list of valid host principals. func (a *Authority) GetSSHHosts(ctx context.Context, cert *x509.Certificate) ([]config.Host, error) { + if a.GetConfig().AuthorityConfig.DisableGetSSHHosts { + return nil, errs.New(http.StatusNotFound, "ssh hosts list api disabled") + } if a.sshGetHostsFunc != nil { hosts, err := a.sshGetHostsFunc(ctx, cert) return hosts, errs.Wrap(http.StatusInternalServerError, err, "getSSHHosts") diff --git a/authority/tls.go b/authority/tls.go index 4c29ca15..c7e2dd09 100644 --- a/authority/tls.go +++ b/authority/tls.go @@ -93,12 +93,17 @@ func (a *Authority) Sign(csr *x509.CertificateRequest, signOpts provisioner.Sign signOpts.Backdate = a.config.AuthorityConfig.Backdate.Duration var prov provisioner.Interface + var pInfo *casapi.ProvisionerInfo for _, op := range extraOpts { switch k := op.(type) { // Capture current provisioner case provisioner.Interface: prov = k - + pInfo = &casapi.ProvisionerInfo{ + ID: prov.GetID(), + Type: prov.GetType().String(), + Name: prov.GetName(), + } // Adds new options to NewCertificate case provisioner.CertificateOptions: certOptions = append(certOptions, k.Options(signOpts)...) @@ -221,10 +226,11 @@ func (a *Authority) Sign(csr *x509.CertificateRequest, signOpts provisioner.Sign // Sign certificate lifetime := leaf.NotAfter.Sub(leaf.NotBefore.Add(signOpts.Backdate)) resp, err := a.x509CAService.CreateCertificate(&casapi.CreateCertificateRequest{ - Template: leaf, - CSR: csr, - Lifetime: lifetime, - Backdate: signOpts.Backdate, + Template: leaf, + CSR: csr, + Lifetime: lifetime, + Backdate: signOpts.Backdate, + Provisioner: pInfo, }) if err != nil { return nil, errs.Wrap(http.StatusInternalServerError, err, "authority.Sign; error creating certificate", opts...) @@ -609,10 +615,11 @@ func (a *Authority) GetTLSCertificate() (*tls.Certificate, error) { certTpl.NotAfter = now.Add(24 * time.Hour) resp, err := a.x509CAService.CreateCertificate(&casapi.CreateCertificateRequest{ - Template: certTpl, - CSR: cr, - Lifetime: 24 * time.Hour, - Backdate: 1 * time.Minute, + Template: certTpl, + CSR: cr, + Lifetime: 24 * time.Hour, + Backdate: 1 * time.Minute, + IsCAServerCert: true, }) if err != nil { return fatal(err) diff --git a/authority/tls_test.go b/authority/tls_test.go index 23d2f8fa..a8521b51 100644 --- a/authority/tls_test.go +++ b/authority/tls_test.go @@ -6,7 +6,7 @@ import ( "crypto/ecdsa" "crypto/elliptic" "crypto/rand" - "crypto/sha1" + "crypto/sha1" // nolint:gosec // used to create the Subject Key Identifier by RFC 5280 "crypto/x509" "crypto/x509/pkix" "encoding/asn1" @@ -199,6 +199,7 @@ func generateSubjectKeyID(pub crypto.PublicKey) ([]byte, error) { if _, err = asn1.Unmarshal(b, &info); err != nil { return nil, fmt.Errorf("error unmarshaling public key: %w", err) } + // nolint:gosec // used to create the Subject Key Identifier by RFC 5280 hash := sha1.Sum(info.SubjectPublicKey.Bytes) return hash[:], nil } diff --git a/ca/bootstrap_test.go b/ca/bootstrap_test.go index ccbdbc22..2a837a3d 100644 --- a/ca/bootstrap_test.go +++ b/ca/bootstrap_test.go @@ -200,6 +200,7 @@ func TestBootstrap(t *testing.T) { } } +// nolint:gosec // insecure test servers func TestBootstrapServerWithoutMTLS(t *testing.T) { srv := startCABootstrapServer() defer srv.Close() @@ -256,6 +257,7 @@ func TestBootstrapServerWithoutMTLS(t *testing.T) { } } +// nolint:gosec // insecure test servers func TestBootstrapServerWithMTLS(t *testing.T) { srv := startCABootstrapServer() defer srv.Close() @@ -405,6 +407,7 @@ func TestBootstrapClientServerRotation(t *testing.T) { // Create bootstrap server token := generateBootstrapToken(caURL, "127.0.0.1", "ef742f95dc0d8aa82d3cca4017af6dac3fce84290344159891952d18c53eefe7") + // nolint:gosec // insecure test server server, err := BootstrapServer(context.Background(), token, &http.Server{ Addr: ":0", Handler: http.HandlerFunc(func(w http.ResponseWriter, req *http.Request) { @@ -523,6 +526,7 @@ func TestBootstrapClientServerFederation(t *testing.T) { // Create bootstrap server token := generateBootstrapToken(caURL1, "127.0.0.1", "ef742f95dc0d8aa82d3cca4017af6dac3fce84290344159891952d18c53eefe7") + // nolint:gosec // insecure test server server, err := BootstrapServer(context.Background(), token, &http.Server{ Addr: ":0", Handler: http.HandlerFunc(func(w http.ResponseWriter, req *http.Request) { diff --git a/ca/ca.go b/ca/ca.go index 7c00bb6b..bddcab79 100644 --- a/ca/ca.go +++ b/ca/ca.go @@ -1,6 +1,7 @@ package ca import ( + "bytes" "context" "crypto/tls" "crypto/x509" @@ -14,6 +15,7 @@ import ( "sync" "github.com/go-chi/chi" + "github.com/go-chi/chi/middleware" "github.com/pkg/errors" "github.com/smallstep/certificates/acme" acmeAPI "github.com/smallstep/certificates/acme/api" @@ -172,6 +174,10 @@ func (ca *CA) Init(cfg *config.Config) (*CA, error) { insecureMux := chi.NewRouter() insecureHandler := http.Handler(insecureMux) + // Add HEAD middleware + mux.Use(middleware.GetHead) + insecureMux.Use(middleware.GetHead) + // Add regular CA api endpoints in / and /1.0 api.Route(mux) mux.Route("/1.0", func(r chi.Router) { @@ -342,10 +348,10 @@ func (ca *CA) Run() error { log.Printf("X.509 Root Fingerprint: %s", x509util.Fingerprint(crt)) } if authorityInfo.SSHCAHostPublicKey != nil { - log.Printf("SSH Host CA Key: %s\n", authorityInfo.SSHCAHostPublicKey) + log.Printf("SSH Host CA Key: %s\n", bytes.TrimSpace(authorityInfo.SSHCAHostPublicKey)) } if authorityInfo.SSHCAUserPublicKey != nil { - log.Printf("SSH User CA Key: %s\n", authorityInfo.SSHCAUserPublicKey) + log.Printf("SSH User CA Key: %s\n", bytes.TrimSpace(authorityInfo.SSHCAUserPublicKey)) } } diff --git a/ca/ca_test.go b/ca/ca_test.go index 29eac575..e76ca8ff 100644 --- a/ca/ca_test.go +++ b/ca/ca_test.go @@ -5,7 +5,7 @@ import ( "context" "crypto" "crypto/rand" - "crypto/sha1" + "crypto/sha1" // nolint:gosec // used to create the Subject Key Identifier by RFC 5280 "crypto/tls" "crypto/x509" "crypto/x509/pkix" @@ -65,6 +65,8 @@ func generateSubjectKeyID(pub crypto.PublicKey) ([]byte, error) { if _, err = asn1.Unmarshal(b, &info); err != nil { return nil, errors.Wrap(err, "error unmarshaling public key") } + + // nolint:gosec // used to create the Subject Key Identifier by RFC 5280 hash := sha1.Sum(info.SubjectPublicKey.Bytes) return hash[:], nil } diff --git a/ca/client.go b/ca/client.go index 44961357..19fcd0bd 100644 --- a/ca/client.go +++ b/ca/client.go @@ -56,6 +56,7 @@ func newClient(transport http.RoundTripper) *uaClient { } } +// nolint:gosec // used in bootstrap protocol func newInsecureClient() *uaClient { return &uaClient{ Client: &http.Client{ @@ -201,7 +202,9 @@ func (o *clientOptions) getTransport(endpoint string) (tr http.RoundTripper, err switch tr := tr.(type) { case *http.Transport: if tr.TLSClientConfig == nil { - tr.TLSClientConfig = &tls.Config{} + tr.TLSClientConfig = &tls.Config{ + MinVersion: tls.VersionTLS12, + } } if len(tr.TLSClientConfig.Certificates) == 0 && tr.TLSClientConfig.GetClientCertificate == nil { tr.TLSClientConfig.Certificates = []tls.Certificate{o.certificate} @@ -209,7 +212,9 @@ func (o *clientOptions) getTransport(endpoint string) (tr http.RoundTripper, err } case *http2.Transport: if tr.TLSClientConfig == nil { - tr.TLSClientConfig = &tls.Config{} + tr.TLSClientConfig = &tls.Config{ + MinVersion: tls.VersionTLS12, + } } if len(tr.TLSClientConfig.Certificates) == 0 && tr.TLSClientConfig.GetClientCertificate == nil { tr.TLSClientConfig.Certificates = []tls.Certificate{o.certificate} @@ -236,11 +241,15 @@ func WithTransport(tr http.RoundTripper) ClientOption { } // WithInsecure adds a insecure transport that bypasses TLS verification. +// nolint:gosec // insecure option func WithInsecure() ClientOption { return func(o *clientOptions) error { o.transport = &http.Transport{ - Proxy: http.ProxyFromEnvironment, - TLSClientConfig: &tls.Config{InsecureSkipVerify: true}, + Proxy: http.ProxyFromEnvironment, + TLSClientConfig: &tls.Config{ + MinVersion: tls.VersionTLS12, + InsecureSkipVerify: true, + }, } return nil } diff --git a/ca/identity/client.go b/ca/identity/client.go index e38529ca..4b0aee82 100644 --- a/ca/identity/client.go +++ b/ca/identity/client.go @@ -62,6 +62,7 @@ func LoadClient() (*Client, error) { // Prepare transport with information in defaults.json and identity.json tr := http.DefaultTransport.(*http.Transport).Clone() tr.TLSClientConfig = &tls.Config{ + MinVersion: tls.VersionTLS12, GetClientCertificate: identity.GetClientCertificateFunc(), } diff --git a/ca/identity/client_test.go b/ca/identity/client_test.go index 9660a3bd..14e6da6c 100644 --- a/ca/identity/client_test.go +++ b/ca/identity/client_test.go @@ -58,6 +58,7 @@ func TestClient(t *testing.T) { Certificates: []tls.Certificate{crt}, ClientCAs: pool, ClientAuth: tls.VerifyClientCertIfGiven, + MinVersion: tls.VersionTLS12, } okServer.StartTLS() @@ -132,6 +133,7 @@ func TestLoadClient(t *testing.T) { tr.TLSClientConfig = &tls.Config{ Certificates: []tls.Certificate{crt}, RootCAs: pool, + MinVersion: tls.VersionTLS12, } expected := &Client{ CaURL: &url.URL{Scheme: "https", Host: "127.0.0.1"}, diff --git a/ca/identity/identity.go b/ca/identity/identity.go index 8aa4b441..2a6b4c39 100644 --- a/ca/identity/identity.go +++ b/ca/identity/identity.go @@ -296,6 +296,7 @@ func (i *Identity) Renew(client Renewer) error { tr.TLSClientConfig = &tls.Config{ Certificates: []tls.Certificate{cert}, RootCAs: client.GetRootCAs(), + MinVersion: tls.VersionTLS12, PreferServerCipherSuites: true, } diff --git a/ca/identity/identity_test.go b/ca/identity/identity_test.go index 55fc60fd..eb32328a 100644 --- a/ca/identity/identity_test.go +++ b/ca/identity/identity_test.go @@ -5,7 +5,6 @@ import ( "crypto/tls" "crypto/x509" "fmt" - "io/ioutil" "net/http" "os" "path/filepath" @@ -188,7 +187,7 @@ func Test_fileExists(t *testing.T) { } func TestWriteDefaultIdentity(t *testing.T) { - tmpDir, err := ioutil.TempDir(os.TempDir(), "go-tests") + tmpDir, err := os.MkdirTemp(os.TempDir(), "go-tests") if err != nil { t.Fatal(err) } @@ -373,7 +372,7 @@ func (r *renewer) Renew(tr http.RoundTripper) (*api.SignResponse, error) { } func TestIdentity_Renew(t *testing.T) { - tmpDir, err := ioutil.TempDir(os.TempDir(), "go-tests") + tmpDir, err := os.MkdirTemp(os.TempDir(), "go-tests") if err != nil { t.Fatal(err) } diff --git a/ca/renew.go b/ca/renew.go index 27898993..a913e59c 100644 --- a/ca/renew.go +++ b/ca/renew.go @@ -173,7 +173,7 @@ func (r *TLSRenewer) renewCertificate() { cert, err := r.RenewCertificate() if err != nil { next = r.renewJitter / 2 - next += time.Duration(rand.Int63n(int64(next))) + next += time.Duration(mathRandInt63n(int64(next))) } else { r.setCertificate(cert) next = r.nextRenewDuration(cert.Leaf.NotAfter) @@ -185,10 +185,15 @@ func (r *TLSRenewer) renewCertificate() { func (r *TLSRenewer) nextRenewDuration(notAfter time.Time) time.Duration { d := time.Until(notAfter).Truncate(time.Second) - r.renewBefore - n := rand.Int63n(int64(r.renewJitter)) + n := mathRandInt63n(int64(r.renewJitter)) d -= time.Duration(n) if d < 0 { d = 0 } return d } + +// nolint:gosec // not used for cryptographic security +func mathRandInt63n(n int64) int64 { + return rand.Int63n(n) +} diff --git a/ca/tls.go b/ca/tls.go index 57440bad..b4d54952 100644 --- a/ca/tls.go +++ b/ca/tls.go @@ -60,6 +60,7 @@ func init() { d := &tls.Dialer{ NetDialer: getDefaultDialer(), Config: &tls.Config{ + MinVersion: tls.VersionTLS12, RootCAs: pool, GetClientCertificate: id.GetClientCertificateFunc(), }, diff --git a/ca/tls_options_test.go b/ca/tls_options_test.go index ca5f80b8..65086315 100644 --- a/ca/tls_options_test.go +++ b/ca/tls_options_test.go @@ -13,6 +13,7 @@ import ( "github.com/smallstep/certificates/api" ) +// nolint:gosec // test tls config func Test_newTLSOptionCtx(t *testing.T) { client, err := NewClient("https://ca.smallstep.com", WithTransport(http.DefaultTransport)) if err != nil { @@ -40,6 +41,7 @@ func Test_newTLSOptionCtx(t *testing.T) { } } +// nolint:gosec // test tls config func TestTLSOptionCtx_apply(t *testing.T) { fail := func() TLSOption { return func(ctx *TLSOptionCtx) error { @@ -76,6 +78,7 @@ func TestTLSOptionCtx_apply(t *testing.T) { } } +// nolint:gosec // test tls config func TestRequireAndVerifyClientCert(t *testing.T) { tests := []struct { name string @@ -100,6 +103,7 @@ func TestRequireAndVerifyClientCert(t *testing.T) { } } +// nolint:gosec // test tls config func TestVerifyClientCertIfGiven(t *testing.T) { tests := []struct { name string @@ -124,6 +128,7 @@ func TestVerifyClientCertIfGiven(t *testing.T) { } } +// nolint:gosec // test tls config func TestAddRootCA(t *testing.T) { cert := parseCertificate(rootPEM) pool := x509.NewCertPool() @@ -156,6 +161,7 @@ func TestAddRootCA(t *testing.T) { } } +// nolint:gosec // test tls config func TestAddClientCA(t *testing.T) { cert := parseCertificate(rootPEM) pool := x509.NewCertPool() @@ -188,6 +194,7 @@ func TestAddClientCA(t *testing.T) { } } +// nolint:gosec // test tls config func TestAddRootsToRootCAs(t *testing.T) { ca := startCATestServer() defer ca.Close() @@ -242,6 +249,7 @@ func TestAddRootsToRootCAs(t *testing.T) { } } +// nolint:gosec // test tls config func TestAddRootsToClientCAs(t *testing.T) { ca := startCATestServer() defer ca.Close() @@ -296,6 +304,7 @@ func TestAddRootsToClientCAs(t *testing.T) { } } +// nolint:gosec // test tls config func TestAddFederationToRootCAs(t *testing.T) { ca := startCATestServer() defer ca.Close() @@ -360,6 +369,7 @@ func TestAddFederationToRootCAs(t *testing.T) { } } +// nolint:gosec // test tls config func TestAddFederationToClientCAs(t *testing.T) { ca := startCATestServer() defer ca.Close() @@ -424,6 +434,7 @@ func TestAddFederationToClientCAs(t *testing.T) { } } +// nolint:gosec // test tls config func TestAddRootsToCAs(t *testing.T) { ca := startCATestServer() defer ca.Close() @@ -478,6 +489,7 @@ func TestAddRootsToCAs(t *testing.T) { } } +// nolint:gosec // test tls config func TestAddFederationToCAs(t *testing.T) { ca := startCATestServer() defer ca.Close() diff --git a/cas/apiv1/options.go b/cas/apiv1/options.go index f69f933b..b8d07f2f 100644 --- a/cas/apiv1/options.go +++ b/cas/apiv1/options.go @@ -6,12 +6,17 @@ import ( "encoding/json" "github.com/pkg/errors" - "github.com/smallstep/certificates/kms" + + "go.step.sm/crypto/kms" ) // Options represents the configuration options used to select and configure the // CertificateAuthorityService (CAS) to use. type Options struct { + // AuthorityID is the the id oc the current authority. This is used on + // StepCAS to add information about the origin of a certificate. + AuthorityID string `json:"-"` + // The type of the CAS to use. Type string `json:"type"` diff --git a/cas/apiv1/requests.go b/cas/apiv1/requests.go index bf745c17..d93cf38d 100644 --- a/cas/apiv1/requests.go +++ b/cas/apiv1/requests.go @@ -5,7 +5,7 @@ import ( "crypto/x509" "time" - "github.com/smallstep/certificates/kms/apiv1" + "go.step.sm/crypto/kms/apiv1" ) // CertificateAuthorityType indicates the type of Certificate Authority to @@ -52,11 +52,21 @@ const ( // CreateCertificateRequest is the request used to sign a new certificate. type CreateCertificateRequest struct { - Template *x509.Certificate - CSR *x509.CertificateRequest - Lifetime time.Duration - Backdate time.Duration - RequestID string + Template *x509.Certificate + CSR *x509.CertificateRequest + Lifetime time.Duration + Backdate time.Duration + RequestID string + Provisioner *ProvisionerInfo + IsCAServerCert bool +} + +// ProvisionerInfo contains information of the provisioner used to authorize a +// certificate. +type ProvisionerInfo struct { + ID string + Type string + Name string } // CreateCertificateResponse is the response to a create certificate request. diff --git a/cas/cas_test.go b/cas/cas_test.go index 7b7b5976..f971c5a8 100644 --- a/cas/cas_test.go +++ b/cas/cas_test.go @@ -9,10 +9,11 @@ import ( "reflect" "testing" + "go.step.sm/crypto/kms" + kmsapi "go.step.sm/crypto/kms/apiv1" + "github.com/smallstep/certificates/cas/apiv1" "github.com/smallstep/certificates/cas/softcas" - "github.com/smallstep/certificates/kms" - kmsapi "github.com/smallstep/certificates/kms/apiv1" ) type mockCAS struct{} diff --git a/cas/cloudcas/certificate.go b/cas/cloudcas/certificate.go index 6f229702..a78deac4 100644 --- a/cas/cloudcas/certificate.go +++ b/cas/cloudcas/certificate.go @@ -11,7 +11,7 @@ import ( "fmt" "github.com/pkg/errors" - kmsapi "github.com/smallstep/certificates/kms/apiv1" + kmsapi "go.step.sm/crypto/kms/apiv1" pb "google.golang.org/genproto/googleapis/cloud/security/privateca/v1" ) diff --git a/cas/cloudcas/certificate_test.go b/cas/cloudcas/certificate_test.go index 0cabdf5b..4e98fdf6 100644 --- a/cas/cloudcas/certificate_test.go +++ b/cas/cloudcas/certificate_test.go @@ -14,7 +14,7 @@ import ( "reflect" "testing" - kmsapi "github.com/smallstep/certificates/kms/apiv1" + kmsapi "go.step.sm/crypto/kms/apiv1" pb "google.golang.org/genproto/googleapis/cloud/security/privateca/v1" ) diff --git a/cas/cloudcas/cloudcas_test.go b/cas/cloudcas/cloudcas_test.go index 1a16ef1e..eee25956 100644 --- a/cas/cloudcas/cloudcas_test.go +++ b/cas/cloudcas/cloudcas_test.go @@ -25,7 +25,7 @@ import ( gax "github.com/googleapis/gax-go/v2" "github.com/pkg/errors" "github.com/smallstep/certificates/cas/apiv1" - kmsapi "github.com/smallstep/certificates/kms/apiv1" + kmsapi "go.step.sm/crypto/kms/apiv1" "google.golang.org/api/option" pb "google.golang.org/genproto/googleapis/cloud/security/privateca/v1" longrunningpb "google.golang.org/genproto/googleapis/longrunning" diff --git a/cas/softcas/softcas.go b/cas/softcas/softcas.go index 2c61fbae..36c1dd69 100644 --- a/cas/softcas/softcas.go +++ b/cas/softcas/softcas.go @@ -8,10 +8,12 @@ import ( "time" "github.com/pkg/errors" - "github.com/smallstep/certificates/cas/apiv1" - "github.com/smallstep/certificates/kms" - kmsapi "github.com/smallstep/certificates/kms/apiv1" + + "go.step.sm/crypto/kms" + kmsapi "go.step.sm/crypto/kms/apiv1" "go.step.sm/crypto/x509util" + + "github.com/smallstep/certificates/cas/apiv1" ) func init() { @@ -201,7 +203,7 @@ func (c *SoftCAS) CreateCertificateAuthority(req *apiv1.CreateCertificateAuthori func (c *SoftCAS) initializeKeyManager() (err error) { if c.KeyManager == nil { c.KeyManager, err = kms.New(context.Background(), kmsapi.Options{ - Type: string(kmsapi.DefaultKMS), + Type: kmsapi.DefaultKMS, }) } return diff --git a/cas/softcas/softcas_test.go b/cas/softcas/softcas_test.go index 0651ab4d..3caa0561 100644 --- a/cas/softcas/softcas_test.go +++ b/cas/softcas/softcas_test.go @@ -17,8 +17,8 @@ import ( "github.com/pkg/errors" "github.com/smallstep/certificates/cas/apiv1" - "github.com/smallstep/certificates/kms" - kmsapi "github.com/smallstep/certificates/kms/apiv1" + "go.step.sm/crypto/kms" + kmsapi "go.step.sm/crypto/kms/apiv1" "go.step.sm/crypto/pemutil" "go.step.sm/crypto/x509util" ) diff --git a/cas/stepcas/issuer.go b/cas/stepcas/issuer.go index be395e33..07607caa 100644 --- a/cas/stepcas/issuer.go +++ b/cas/stepcas/issuer.go @@ -5,13 +5,33 @@ import ( "strings" "time" + "github.com/google/uuid" "github.com/pkg/errors" "github.com/smallstep/certificates/ca" "github.com/smallstep/certificates/cas/apiv1" ) +// raAuthorityNS is a custom namespace used to generate endpoint ids based on +// the authority id. +var raAuthorityNS = uuid.MustParse("d6f14c1f-2f92-47bf-a04f-7b2c11382edd") + +// newServerEndpointID returns a uuid v5 using raAuthorityNS as the namespace. +// The return uuid will be used as the server endpoint id, it will be unique per +// authority. +func newServerEndpointID(data string) uuid.UUID { + return uuid.NewSHA1(raAuthorityNS, []byte(data)) +} + +type raInfo struct { + AuthorityID string `json:"authorityId,omitempty"` + EndpointID string `json:"endpointId,omitempty"` + ProvisionerID string `json:"provisionerId,omitempty"` + ProvisionerType string `json:"provisionerType,omitempty"` + ProvisionerName string `json:"provisionerName,omitempty"` +} + type stepIssuer interface { - SignToken(subject string, sans []string) (string, error) + SignToken(subject string, sans []string, info *raInfo) (string, error) RevokeToken(subject string) (string, error) Lifetime(d time.Duration) time.Duration } diff --git a/cas/stepcas/issuer_test.go b/cas/stepcas/issuer_test.go index 6fffd729..c968237a 100644 --- a/cas/stepcas/issuer_test.go +++ b/cas/stepcas/issuer_test.go @@ -6,6 +6,7 @@ import ( "testing" "time" + "github.com/google/uuid" "github.com/smallstep/certificates/ca" "github.com/smallstep/certificates/cas/apiv1" "go.step.sm/crypto/jose" @@ -13,7 +14,7 @@ import ( type mockErrIssuer struct{} -func (m mockErrIssuer) SignToken(subject string, sans []string) (string, error) { +func (m mockErrIssuer) SignToken(subject string, sans []string, info *raInfo) (string, error) { return "", apiv1.ErrNotImplemented{} } @@ -35,6 +36,42 @@ func (s *mockErrSigner) Options() jose.SignerOptions { return jose.SignerOptions{} } +func Test_newServerEndpointID(t *testing.T) { + type args struct { + name string + } + tests := []struct { + name string + args args + want []byte + }{ + {"ok", args{"foo"}, []byte{ + 0x8f, 0x63, 0x69, 0x20, 0x8a, 0x7a, 0x57, 0x0c, 0xbe, 0x4c, 0x46, 0x66, 0x77, 0xf8, 0x54, 0xe7, + }}, + {"ok uuid", args{"e4fa6d2d-fa9c-4fdc-913e-7484cc9516e4"}, []byte{ + 0x8d, 0x8d, 0x7f, 0x04, 0x73, 0xd4, 0x5f, 0x2f, 0xa8, 0xe1, 0x28, 0x9a, 0xd1, 0xa8, 0xcf, 0x7e, + }}, + } + for _, tt := range tests { + t.Run(tt.name, func(t *testing.T) { + var want uuid.UUID + copy(want[:], tt.want) + got := newServerEndpointID(tt.args.name) + if !reflect.DeepEqual(got, want) { + t.Errorf("newServerEndpointID() = %v, want %v", got, tt.want) + } + // Check version + if v := (got[6] & 0xf0) >> 4; v != 5 { + t.Errorf("newServerEndpointID() version = %d, want 5", v) + } + // Check variant + if v := (got[8] & 0x80) >> 6; v != 2 { + t.Errorf("newServerEndpointID() variant = %d, want 2", v) + } + }) + } +} + func Test_newStepIssuer(t *testing.T) { caURL, client := testCAHelper(t) signer, err := newJWKSignerFromEncryptedKey(testKeyID, testEncryptedJWKKey, testPassword) diff --git a/cas/stepcas/jwk_issuer.go b/cas/stepcas/jwk_issuer.go index db45ef48..4ef4f541 100644 --- a/cas/stepcas/jwk_issuer.go +++ b/cas/stepcas/jwk_issuer.go @@ -53,25 +53,25 @@ func newJWKIssuer(caURL *url.URL, client *ca.Client, cfg *apiv1.CertificateIssue }, nil } -func (i *jwkIssuer) SignToken(subject string, sans []string) (string, error) { +func (i *jwkIssuer) SignToken(subject string, sans []string, info *raInfo) (string, error) { aud := i.caURL.ResolveReference(&url.URL{ Path: "/1.0/sign", }).String() - return i.createToken(aud, subject, sans) + return i.createToken(aud, subject, sans, info) } func (i *jwkIssuer) RevokeToken(subject string) (string, error) { aud := i.caURL.ResolveReference(&url.URL{ Path: "/1.0/revoke", }).String() - return i.createToken(aud, subject, nil) + return i.createToken(aud, subject, nil, nil) } func (i *jwkIssuer) Lifetime(d time.Duration) time.Duration { return d } -func (i *jwkIssuer) createToken(aud, sub string, sans []string) (string, error) { +func (i *jwkIssuer) createToken(aud, sub string, sans []string, info *raInfo) (string, error) { id, err := randutil.Hex(64) // 256 bits if err != nil { return "", err @@ -84,6 +84,13 @@ func (i *jwkIssuer) createToken(aud, sub string, sans []string) (string, error) "sans": sans, }) } + if info != nil { + builder = builder.Claims(map[string]interface{}{ + "step": map[string]interface{}{ + "ra": info, + }, + }) + } tok, err := builder.CompactSerialize() if err != nil { diff --git a/cas/stepcas/jwk_issuer_test.go b/cas/stepcas/jwk_issuer_test.go index 7ebfcb3f..0924414b 100644 --- a/cas/stepcas/jwk_issuer_test.go +++ b/cas/stepcas/jwk_issuer_test.go @@ -27,11 +27,16 @@ func Test_jwkIssuer_SignToken(t *testing.T) { type args struct { subject string sans []string + info *raInfo + } + type stepClaims struct { + RA *raInfo `json:"ra"` } type claims struct { - Aud []string `json:"aud"` - Sub string `json:"sub"` - Sans []string `json:"sans"` + Aud []string `json:"aud"` + Sub string `json:"sub"` + Sans []string `json:"sans"` + Step stepClaims `json:"step"` } tests := []struct { name string @@ -39,8 +44,14 @@ func Test_jwkIssuer_SignToken(t *testing.T) { args args wantErr bool }{ - {"ok", fields{caURL, "ra@doe.org", signer}, args{"doe", []string{"doe.org"}}, false}, - {"fail", fields{caURL, "ra@doe.org", &mockErrSigner{}}, args{"doe", []string{"doe.org"}}, true}, + {"ok", fields{caURL, "ra@doe.org", signer}, args{"doe", []string{"doe.org"}, nil}, false}, + {"ok ra", fields{caURL, "ra@doe.org", signer}, args{"doe", []string{"doe.org"}, &raInfo{ + AuthorityID: "authority-id", ProvisionerID: "provisioner-id", ProvisionerType: "provisioner-type", + }}, false}, + {"ok ra endpoint id", fields{caURL, "ra@doe.org", signer}, args{"doe", []string{"doe.org"}, &raInfo{ + AuthorityID: "authority-id", EndpointID: "endpoint-id", + }}, false}, + {"fail", fields{caURL, "ra@doe.org", &mockErrSigner{}}, args{"doe", []string{"doe.org"}, nil}, true}, } for _, tt := range tests { t.Run(tt.name, func(t *testing.T) { @@ -49,7 +60,7 @@ func Test_jwkIssuer_SignToken(t *testing.T) { issuer: tt.fields.issuer, signer: tt.fields.signer, } - got, err := i.SignToken(tt.args.subject, tt.args.sans) + got, err := i.SignToken(tt.args.subject, tt.args.sans, tt.args.info) if (err != nil) != tt.wantErr { t.Errorf("jwkIssuer.SignToken() error = %v, wantErr %v", err, tt.wantErr) return @@ -65,6 +76,9 @@ func Test_jwkIssuer_SignToken(t *testing.T) { Sub: tt.args.subject, Sans: tt.args.sans, } + if tt.args.info != nil { + want.Step.RA = tt.args.info + } if err := jwt.Claims(testX5CKey.Public(), &c); err != nil { t.Errorf("jwt.Claims() error = %v", err) } diff --git a/cas/stepcas/stepcas.go b/cas/stepcas/stepcas.go index 9fcbd36c..f8770923 100644 --- a/cas/stepcas/stepcas.go +++ b/cas/stepcas/stepcas.go @@ -23,6 +23,7 @@ func init() { type StepCAS struct { iss stepIssuer client *ca.Client + authorityID string fingerprint string } @@ -59,6 +60,7 @@ func New(ctx context.Context, opts apiv1.Options) (*StepCAS, error) { return &StepCAS{ iss: iss, client: client, + authorityID: opts.AuthorityID, fingerprint: opts.CertificateAuthorityFingerprint, }, nil } @@ -73,7 +75,19 @@ func (s *StepCAS) CreateCertificate(req *apiv1.CreateCertificateRequest) (*apiv1 return nil, errors.New("createCertificateRequest `lifetime` cannot be 0") } - cert, chain, err := s.createCertificate(req.CSR, req.Lifetime) + info := &raInfo{ + AuthorityID: s.authorityID, + } + if req.IsCAServerCert { + info.EndpointID = newServerEndpointID(s.authorityID).String() + } + if p := req.Provisioner; p != nil { + info.ProvisionerID = p.ID + info.ProvisionerType = p.Type + info.ProvisionerName = p.Name + } + + cert, chain, err := s.createCertificate(req.CSR, req.Lifetime, info) if err != nil { return nil, err } @@ -135,7 +149,7 @@ func (s *StepCAS) GetCertificateAuthority(req *apiv1.GetCertificateAuthorityRequ }, nil } -func (s *StepCAS) createCertificate(cr *x509.CertificateRequest, lifetime time.Duration) (*x509.Certificate, []*x509.Certificate, error) { +func (s *StepCAS) createCertificate(cr *x509.CertificateRequest, lifetime time.Duration, raInfo *raInfo) (*x509.Certificate, []*x509.Certificate, error) { sans := make([]string, 0, len(cr.DNSNames)+len(cr.EmailAddresses)+len(cr.IPAddresses)+len(cr.URIs)) sans = append(sans, cr.DNSNames...) sans = append(sans, cr.EmailAddresses...) @@ -151,7 +165,7 @@ func (s *StepCAS) createCertificate(cr *x509.CertificateRequest, lifetime time.D commonName = sans[0] } - token, err := s.iss.SignToken(commonName, sans) + token, err := s.iss.SignToken(commonName, sans, raInfo) if err != nil { return nil, nil, err } diff --git a/cas/stepcas/stepcas_test.go b/cas/stepcas/stepcas_test.go index ad7851bf..cc8ea72e 100644 --- a/cas/stepcas/stepcas_test.go +++ b/cas/stepcas/stepcas_test.go @@ -10,7 +10,6 @@ import ( "encoding/json" "encoding/pem" "fmt" - "io/ioutil" "net/http" "net/http/httptest" "net/url" @@ -291,7 +290,7 @@ func TestMain(m *testing.M) { } // Create test files. - path, err := ioutil.TempDir(os.TempDir(), "stepcas") + path, err := os.MkdirTemp(os.TempDir(), "stepcas") if err != nil { panic(err) } @@ -665,6 +664,22 @@ func TestStepCAS_CreateCertificate(t *testing.T) { Certificate: testCrt, CertificateChain: []*x509.Certificate{testIssCrt}, }, false}, + {"ok with provisioner", fields{jwk, client, testRootFingerprint}, args{&apiv1.CreateCertificateRequest{ + CSR: testCR, + Lifetime: time.Hour, + Provisioner: &apiv1.ProvisionerInfo{ID: "provisioner-id", Type: "ACME"}, + }}, &apiv1.CreateCertificateResponse{ + Certificate: testCrt, + CertificateChain: []*x509.Certificate{testIssCrt}, + }, false}, + {"ok with server cert", fields{jwk, client, testRootFingerprint}, args{&apiv1.CreateCertificateRequest{ + CSR: testCR, + Lifetime: time.Hour, + IsCAServerCert: true, + }}, &apiv1.CreateCertificateResponse{ + Certificate: testCrt, + CertificateChain: []*x509.Certificate{testIssCrt}, + }, false}, {"fail CSR", fields{x5c, client, testRootFingerprint}, args{&apiv1.CreateCertificateRequest{ CSR: nil, Lifetime: time.Hour, @@ -691,6 +706,7 @@ func TestStepCAS_CreateCertificate(t *testing.T) { s := &StepCAS{ iss: tt.fields.iss, client: tt.fields.client, + authorityID: "authority-id", fingerprint: tt.fields.fingerprint, } got, err := s.CreateCertificate(tt.args.req) diff --git a/cas/stepcas/x5c_issuer.go b/cas/stepcas/x5c_issuer.go index 76ed9c3c..a005e501 100644 --- a/cas/stepcas/x5c_issuer.go +++ b/cas/stepcas/x5c_issuer.go @@ -46,13 +46,13 @@ func newX5CIssuer(caURL *url.URL, cfg *apiv1.CertificateIssuer) (*x5cIssuer, err }, nil } -func (i *x5cIssuer) SignToken(subject string, sans []string) (string, error) { +func (i *x5cIssuer) SignToken(subject string, sans []string, info *raInfo) (string, error) { aud := i.caURL.ResolveReference(&url.URL{ Path: "/1.0/sign", Fragment: "x5c/" + i.issuer, }).String() - return i.createToken(aud, subject, sans) + return i.createToken(aud, subject, sans, info) } func (i *x5cIssuer) RevokeToken(subject string) (string, error) { @@ -61,7 +61,7 @@ func (i *x5cIssuer) RevokeToken(subject string) (string, error) { Fragment: "x5c/" + i.issuer, }).String() - return i.createToken(aud, subject, nil) + return i.createToken(aud, subject, nil, nil) } func (i *x5cIssuer) Lifetime(d time.Duration) time.Duration { @@ -76,7 +76,7 @@ func (i *x5cIssuer) Lifetime(d time.Duration) time.Duration { return d } -func (i *x5cIssuer) createToken(aud, sub string, sans []string) (string, error) { +func (i *x5cIssuer) createToken(aud, sub string, sans []string, info *raInfo) (string, error) { signer, err := newX5CSigner(i.certFile, i.keyFile, i.password) if err != nil { return "", err @@ -94,6 +94,13 @@ func (i *x5cIssuer) createToken(aud, sub string, sans []string) (string, error) "sans": sans, }) } + if info != nil { + builder = builder.Claims(map[string]interface{}{ + "step": map[string]interface{}{ + "ra": info, + }, + }) + } tok, err := builder.CompactSerialize() if err != nil { diff --git a/cas/stepcas/x5c_issuer_test.go b/cas/stepcas/x5c_issuer_test.go index b1bc653d..3f7f372f 100644 --- a/cas/stepcas/x5c_issuer_test.go +++ b/cas/stepcas/x5c_issuer_test.go @@ -51,11 +51,16 @@ func Test_x5cIssuer_SignToken(t *testing.T) { type args struct { subject string sans []string + info *raInfo + } + type stepClaims struct { + RA *raInfo `json:"ra"` } type claims struct { - Aud []string `json:"aud"` - Sub string `json:"sub"` - Sans []string `json:"sans"` + Aud []string `json:"aud"` + Sub string `json:"sub"` + Sans []string `json:"sans"` + Step stepClaims `json:"step"` } tests := []struct { name string @@ -63,10 +68,16 @@ func Test_x5cIssuer_SignToken(t *testing.T) { args args wantErr bool }{ - {"ok", fields{caURL, testX5CPath, testX5CKeyPath, "X5C"}, args{"doe", []string{"doe.org"}}, false}, - {"fail crt", fields{caURL, "", testX5CKeyPath, "X5C"}, args{"doe", []string{"doe.org"}}, true}, - {"fail key", fields{caURL, testX5CPath, "", "X5C"}, args{"doe", []string{"doe.org"}}, true}, - {"fail no signer", fields{caURL, testIssKeyPath, testIssPath, "X5C"}, args{"doe", []string{"doe.org"}}, true}, + {"ok", fields{caURL, testX5CPath, testX5CKeyPath, "X5C"}, args{"doe", []string{"doe.org"}, nil}, false}, + {"ok ra", fields{caURL, testX5CPath, testX5CKeyPath, "X5C"}, args{"doe", []string{"doe.org"}, &raInfo{ + AuthorityID: "authority-id", ProvisionerID: "provisioner-id", ProvisionerType: "provisioner-type", + }}, false}, + {"ok ra endpoint id", fields{caURL, testX5CPath, testX5CKeyPath, "X5C"}, args{"doe", []string{"doe.org"}, &raInfo{ + AuthorityID: "authority-id", EndpointID: "endpoint-id", + }}, false}, + {"fail crt", fields{caURL, "", testX5CKeyPath, "X5C"}, args{"doe", []string{"doe.org"}, nil}, true}, + {"fail key", fields{caURL, testX5CPath, "", "X5C"}, args{"doe", []string{"doe.org"}, nil}, true}, + {"fail no signer", fields{caURL, testIssKeyPath, testIssPath, "X5C"}, args{"doe", []string{"doe.org"}, nil}, true}, } for _, tt := range tests { t.Run(tt.name, func(t *testing.T) { @@ -76,7 +87,7 @@ func Test_x5cIssuer_SignToken(t *testing.T) { keyFile: tt.fields.keyFile, issuer: tt.fields.issuer, } - got, err := i.SignToken(tt.args.subject, tt.args.sans) + got, err := i.SignToken(tt.args.subject, tt.args.sans, tt.args.info) if (err != nil) != tt.wantErr { t.Errorf("x5cIssuer.SignToken() error = %v, wantErr %v", err, tt.wantErr) } @@ -91,6 +102,9 @@ func Test_x5cIssuer_SignToken(t *testing.T) { Sub: tt.args.subject, Sans: tt.args.sans, } + if tt.args.info != nil { + want.Step.RA = tt.args.info + } if err := jwt.Claims(testX5CKey.Public(), &c); err != nil { t.Errorf("jwt.Claims() error = %v", err) } diff --git a/cmd/step-awskms-init/main.go b/cmd/step-awskms-init/main.go index 5e56f045..48e2aa01 100644 --- a/cmd/step-awskms-init/main.go +++ b/cmd/step-awskms-init/main.go @@ -4,7 +4,7 @@ import ( "context" "crypto" "crypto/rand" - "crypto/sha1" + "crypto/sha1" // nolint:gosec // used to create the Subject Key Identifier by RFC 5280 "crypto/x509" "crypto/x509/pkix" "encoding/pem" @@ -14,10 +14,10 @@ import ( "os" "time" - "github.com/smallstep/certificates/kms/apiv1" - "github.com/smallstep/certificates/kms/awskms" "go.step.sm/cli-utils/fileutil" "go.step.sm/cli-utils/ui" + "go.step.sm/crypto/kms/apiv1" + "go.step.sm/crypto/kms/awskms" "go.step.sm/crypto/pemutil" "golang.org/x/crypto/ssh" ) @@ -35,7 +35,7 @@ func main() { ui.Init() c, err := awskms.New(context.Background(), apiv1.Options{ - Type: string(apiv1.AmazonKMS), + Type: apiv1.AmazonKMS, Region: region, CredentialsFile: credentialsFile, }) @@ -239,6 +239,7 @@ func mustSubjectKeyID(key crypto.PublicKey) []byte { if err != nil { panic(err) } + // nolint:gosec // used to create the Subject Key Identifier by RFC 5280 hash := sha1.Sum(b) return hash[:] } diff --git a/cmd/step-ca/main.go b/cmd/step-ca/main.go index bc7bf2e3..d070b6cf 100644 --- a/cmd/step-ca/main.go +++ b/cmd/step-ca/main.go @@ -14,6 +14,7 @@ import ( "time" // Server profiler + // nolint:gosec // profile server, if enabled runs on a different port _ "net/http/pprof" "github.com/smallstep/certificates/authority" @@ -26,15 +27,13 @@ import ( "go.step.sm/cli-utils/usage" // Enabled kms interfaces. - _ "github.com/smallstep/certificates/kms/awskms" - _ "github.com/smallstep/certificates/kms/azurekms" - _ "github.com/smallstep/certificates/kms/cloudkms" - _ "github.com/smallstep/certificates/kms/softkms" - _ "github.com/smallstep/certificates/kms/sshagentkms" - - // Experimental kms interfaces. - _ "github.com/smallstep/certificates/kms/pkcs11" - _ "github.com/smallstep/certificates/kms/yubikey" + _ "go.step.sm/crypto/kms/awskms" + _ "go.step.sm/crypto/kms/azurekms" + _ "go.step.sm/crypto/kms/cloudkms" + _ "go.step.sm/crypto/kms/pkcs11" + _ "go.step.sm/crypto/kms/softkms" + _ "go.step.sm/crypto/kms/sshagentkms" + _ "go.step.sm/crypto/kms/yubikey" // Enabled cas interfaces. _ "github.com/smallstep/certificates/cas/cloudcas" diff --git a/cmd/step-cloudkms-init/main.go b/cmd/step-cloudkms-init/main.go index 78c1fcc5..aa483fc7 100644 --- a/cmd/step-cloudkms-init/main.go +++ b/cmd/step-cloudkms-init/main.go @@ -4,7 +4,7 @@ import ( "context" "crypto" "crypto/rand" - "crypto/sha1" + "crypto/sha1" // nolint:gosec // used to create the Subject Key Identifier by RFC 5280 "crypto/x509" "crypto/x509/pkix" "encoding/pem" @@ -15,10 +15,10 @@ import ( "strings" "time" - "github.com/smallstep/certificates/kms/apiv1" - "github.com/smallstep/certificates/kms/cloudkms" "go.step.sm/cli-utils/fileutil" "go.step.sm/cli-utils/ui" + "go.step.sm/crypto/kms/apiv1" + "go.step.sm/crypto/kms/cloudkms" "go.step.sm/crypto/pemutil" "golang.org/x/crypto/ssh" ) @@ -66,7 +66,7 @@ func main() { ui.Init() c, err := cloudkms.New(context.Background(), apiv1.Options{ - Type: string(apiv1.CloudKMS), + Type: apiv1.CloudKMS, CredentialsFile: credentialsFile, }) if err != nil { @@ -277,6 +277,7 @@ func mustSubjectKeyID(key crypto.PublicKey) []byte { if err != nil { panic(err) } + // nolint:gosec // used to create the Subject Key Identifier by RFC 5280 hash := sha1.Sum(b) return hash[:] } diff --git a/cmd/step-pkcs11-init/main.go b/cmd/step-pkcs11-init/main.go index 7a23c664..ed64e285 100644 --- a/cmd/step-pkcs11-init/main.go +++ b/cmd/step-pkcs11-init/main.go @@ -6,7 +6,7 @@ import ( "crypto/ecdsa" "crypto/elliptic" "crypto/rand" - "crypto/sha1" + "crypto/sha1" // nolint:gosec // used to create the Subject Key Identifier by RFC 5280 "crypto/x509" "crypto/x509/pkix" "encoding/pem" @@ -18,15 +18,15 @@ import ( "time" "github.com/pkg/errors" - "github.com/smallstep/certificates/kms" - "github.com/smallstep/certificates/kms/apiv1" - "github.com/smallstep/certificates/kms/uri" "go.step.sm/cli-utils/fileutil" "go.step.sm/cli-utils/ui" + "go.step.sm/crypto/kms" + "go.step.sm/crypto/kms/apiv1" + "go.step.sm/crypto/kms/uri" "go.step.sm/crypto/pemutil" // Enable pkcs11. - _ "github.com/smallstep/certificates/kms/pkcs11" + _ "go.step.sm/crypto/kms/pkcs11" ) // Config is a mapping of the cli flags. @@ -171,7 +171,7 @@ func main() { } k, err := kms.New(context.Background(), apiv1.Options{ - Type: string(apiv1.PKCS11), + Type: apiv1.PKCS11, URI: c.KMS, Pin: c.Pin, }) @@ -544,6 +544,7 @@ func mustSubjectKeyID(key crypto.PublicKey) []byte { if err != nil { panic(err) } + // nolint:gosec // used to create the Subject Key Identifier by RFC 5280 hash := sha1.Sum(b) return hash[:] } diff --git a/cmd/step-yubikey-init/main.go b/cmd/step-yubikey-init/main.go index 9f755388..62b18848 100644 --- a/cmd/step-yubikey-init/main.go +++ b/cmd/step-yubikey-init/main.go @@ -6,7 +6,7 @@ import ( "crypto/ecdsa" "crypto/elliptic" "crypto/rand" - "crypto/sha1" + "crypto/sha1" // nolint:gosec // used to create the Subject Key Identifier by RFC 5280 "crypto/x509" "crypto/x509/pkix" "encoding/hex" @@ -18,14 +18,14 @@ import ( "time" "github.com/pkg/errors" - "github.com/smallstep/certificates/kms" - "github.com/smallstep/certificates/kms/apiv1" "go.step.sm/cli-utils/fileutil" "go.step.sm/cli-utils/ui" + "go.step.sm/crypto/kms" + "go.step.sm/crypto/kms/apiv1" "go.step.sm/crypto/pemutil" // Enable yubikey. - _ "github.com/smallstep/certificates/kms/yubikey" + _ "go.step.sm/crypto/kms/yubikey" ) // Config is a mapping of the cli flags. @@ -97,7 +97,7 @@ func main() { c.Pin = string(pin) k, err := kms.New(context.Background(), apiv1.Options{ - Type: string(apiv1.YubiKey), + Type: apiv1.YubiKey, Pin: c.Pin, ManagementKey: c.ManagementKey, }) @@ -346,6 +346,7 @@ func mustSubjectKeyID(key crypto.PublicKey) []byte { if err != nil { panic(err) } + // nolint:gosec // used to create the Subject Key Identifier by RFC 5280 hash := sha1.Sum(b) return hash[:] } diff --git a/commands/app.go b/commands/app.go index 265610f2..7545f1df 100644 --- a/commands/app.go +++ b/commands/app.go @@ -7,12 +7,15 @@ import ( "net" "net/http" "os" + "path/filepath" "strings" "unicode" "github.com/pkg/errors" "github.com/smallstep/certificates/authority/config" + "github.com/smallstep/certificates/authority/provisioner" "github.com/smallstep/certificates/ca" + "github.com/smallstep/certificates/db" "github.com/smallstep/certificates/pki" "github.com/urfave/cli" "go.step.sm/cli-utils/errs" @@ -99,10 +102,35 @@ func appAction(ctx *cli.Context) error { } cfg, err := config.LoadConfiguration(configFile) - if err != nil { + if err != nil && token == "" { fatal(err) } + // Initialize a basic configuration to be used with an automatically + // configured linked RA. Default configuration includes: + // * badgerv2 on $(step path)/db + // * JSON logger + // * Default TLS options + if cfg == nil { + cfg = &config.Config{ + SkipValidation: true, + Logger: []byte(`{"format":"json"}`), + DB: &db.Config{ + Type: "badgerv2", + DataSource: filepath.Join(step.Path(), "db"), + }, + AuthorityConfig: &config.AuthConfig{ + DeploymentType: pki.LinkedDeployment.String(), + Provisioners: provisioner.List{}, + Template: &config.ASN1DN{}, + Backdate: &provisioner.Duration{ + Duration: config.DefaultBackdate, + }, + }, + TLS: &config.DefaultTLSOptions, + } + } + if cfg.AuthorityConfig != nil { if token == "" && strings.EqualFold(cfg.AuthorityConfig.DeploymentType, pki.LinkedDeployment.String()) { return errors.New(`'step-ca' requires the '--token' flag for linked deploy type. diff --git a/commands/onboard.go b/commands/onboard.go index afecba9d..bb704fd4 100644 --- a/commands/onboard.go +++ b/commands/onboard.go @@ -92,6 +92,7 @@ func onboardAction(ctx *cli.Context) error { token := ctx.Args().Get(0) onboardingURL := u.ResolveReference(&url.URL{Path: token}).String() + // nolint:gosec // onboarding url res, err := http.Get(onboardingURL) if err != nil { return errors.Wrap(err, "error connecting onboarding guide") @@ -132,6 +133,7 @@ func onboardAction(ctx *cli.Context) error { return errors.Wrap(err, "error marshaling payload") } + // nolint:gosec // onboarding url resp, err := http.Post(onboardingURL, "application/json", bytes.NewBuffer(payload)) if err != nil { return errors.Wrap(err, "error connecting onboarding guide") diff --git a/go.mod b/go.mod index 53306432..29cbc66d 100644 --- a/go.mod +++ b/go.mod @@ -1,23 +1,22 @@ module github.com/smallstep/certificates -go 1.16 +go 1.18 require ( cloud.google.com/go v0.100.2 - cloud.google.com/go/kms v1.4.0 cloud.google.com/go/security v1.3.0 - github.com/Azure/azure-sdk-for-go v65.0.0+incompatible - github.com/Azure/go-autorest/autorest v0.11.27 - github.com/Azure/go-autorest/autorest/azure/auth v0.5.11 - github.com/Azure/go-autorest/autorest/date v0.3.0 + github.com/Azure/azure-sdk-for-go v65.0.0+incompatible // indirect + github.com/Azure/go-autorest/autorest v0.11.27 // indirect + github.com/Azure/go-autorest/autorest/azure/auth v0.5.11 // indirect + github.com/Azure/go-autorest/autorest/date v0.3.0 // indirect github.com/Masterminds/sprig/v3 v3.2.2 - github.com/ThalesIgnite/crypto11 v1.2.5 - github.com/aws/aws-sdk-go v1.44.37 + github.com/ThalesIgnite/crypto11 v1.2.5 // indirect + github.com/aws/aws-sdk-go v1.44.37 // indirect github.com/dgraph-io/ristretto v0.0.4-0.20200906165740-41ebdbffecfd // indirect github.com/fatih/color v1.9.0 // indirect - github.com/go-chi/chi v4.0.2+incompatible + github.com/go-chi/chi v4.1.2+incompatible github.com/go-kit/kit v0.10.0 // indirect - github.com/go-piv/piv-go v1.9.0 + github.com/go-piv/piv-go v1.10.0 // indirect github.com/go-sql-driver/mysql v1.6.0 // indirect github.com/golang/mock v1.6.0 github.com/google/go-cmp v0.5.8 @@ -31,7 +30,7 @@ require ( github.com/mattn/go-colorable v0.1.8 // indirect github.com/mattn/go-isatty v0.0.13 // indirect github.com/micromdm/scep/v2 v2.1.0 - github.com/newrelic/go-agent v2.15.0+incompatible + github.com/newrelic/go-agent/v3 v3.18.0 github.com/pkg/errors v0.9.1 github.com/rs/xid v1.2.1 github.com/sirupsen/logrus v1.8.1 @@ -40,11 +39,10 @@ require ( github.com/smallstep/nosql v0.4.0 github.com/stretchr/testify v1.7.1 github.com/urfave/cli v1.22.4 - go.etcd.io/bbolt v1.3.6 // indirect go.mozilla.org/pkcs7 v0.0.0-20210826202110-33d05740a352 go.step.sm/cli-utils v0.7.3 go.step.sm/crypto v0.18.0 - go.step.sm/linkedca v0.16.1 + go.step.sm/linkedca v0.18.0 golang.org/x/crypto v0.0.0-20211215153901-e495a2d5b3d3 golang.org/x/net v0.0.0-20220607020251-c690dde0001d golang.org/x/time v0.0.0-20210220033141-f8bda1e9f3ba // indirect @@ -55,6 +53,96 @@ require ( gopkg.in/square/go-jose.v2 v2.6.0 ) +require ( + cloud.google.com/go/compute v1.6.1 // indirect + cloud.google.com/go/iam v0.1.0 // indirect + cloud.google.com/go/kms v1.4.0 // indirect + filippo.io/edwards25519 v1.0.0-rc.1 // indirect + github.com/AndreasBriese/bbloom v0.0.0-20190825152654-46b345b51c96 // indirect + github.com/Azure/go-autorest v14.2.0+incompatible // indirect + github.com/Azure/go-autorest/autorest/adal v0.9.18 // indirect + github.com/Azure/go-autorest/autorest/azure/cli v0.4.5 // indirect + github.com/Azure/go-autorest/autorest/to v0.4.0 // indirect + github.com/Azure/go-autorest/autorest/validation v0.3.1 // indirect + github.com/Azure/go-autorest/logger v0.2.1 // indirect + github.com/Azure/go-autorest/tracing v0.6.0 // indirect + github.com/Masterminds/goutils v1.1.1 // indirect + github.com/Masterminds/semver/v3 v3.1.1 // indirect + github.com/armon/go-metrics v0.3.9 // indirect + github.com/armon/go-radix v1.0.0 // indirect + github.com/cenkalti/backoff/v3 v3.0.0 // indirect + github.com/cespare/xxhash v1.1.0 // indirect + github.com/chzyer/readline v0.0.0-20180603132655-2972be24d48e // indirect + github.com/cpuguy83/go-md2man/v2 v2.0.0 // indirect + github.com/davecgh/go-spew v1.1.1 // indirect + github.com/dgraph-io/badger v1.6.2 // indirect + github.com/dgraph-io/badger/v2 v2.2007.4 // indirect + github.com/dgryski/go-farm v0.0.0-20200201041132-a6ae2369ad13 // indirect + github.com/dimchansky/utfbom v1.1.1 // indirect + github.com/dustin/go-humanize v1.0.0 // indirect + github.com/go-logfmt/logfmt v0.5.0 // indirect + github.com/golang-jwt/jwt/v4 v4.2.0 // indirect + github.com/golang/groupcache v0.0.0-20210331224755-41bb18bfe9da // indirect + github.com/golang/protobuf v1.5.2 // indirect + github.com/golang/snappy v0.0.4 // indirect + github.com/googleapis/enterprise-certificate-proxy v0.0.0-20220520183353-fd19c99a87aa // indirect + github.com/hashicorp/errwrap v1.1.0 // indirect + github.com/hashicorp/go-cleanhttp v0.5.2 // indirect + github.com/hashicorp/go-hclog v0.16.2 // indirect + github.com/hashicorp/go-immutable-radix v1.3.1 // indirect + github.com/hashicorp/go-multierror v1.1.1 // indirect + github.com/hashicorp/go-plugin v1.4.3 // indirect + github.com/hashicorp/go-retryablehttp v0.6.6 // indirect + github.com/hashicorp/go-rootcerts v1.0.2 // indirect + github.com/hashicorp/go-secure-stdlib/mlock v0.1.1 // indirect + github.com/hashicorp/go-secure-stdlib/parseutil v0.1.1 // indirect + github.com/hashicorp/go-secure-stdlib/strutil v0.1.1 // indirect + github.com/hashicorp/go-sockaddr v1.0.2 // indirect + github.com/hashicorp/go-uuid v1.0.2 // indirect + github.com/hashicorp/go-version v1.2.0 // indirect + github.com/hashicorp/golang-lru v0.5.4 // indirect + github.com/hashicorp/hcl v1.0.0 // indirect + github.com/hashicorp/vault/sdk v0.3.0 // indirect + github.com/hashicorp/yamux v0.0.0-20180604194846-3520598351bb // indirect + github.com/huandu/xstrings v1.3.2 // indirect + github.com/imdario/mergo v0.3.12 // indirect + github.com/jackc/chunkreader/v2 v2.0.1 // indirect + github.com/jackc/pgconn v1.10.1 // indirect + github.com/jackc/pgio v1.0.0 // indirect + github.com/jackc/pgpassfile v1.0.0 // indirect + github.com/jackc/pgproto3/v2 v2.2.0 // indirect + github.com/jackc/pgservicefile v0.0.0-20200714003250-2b9c44734f2b // indirect + github.com/jackc/pgtype v1.9.0 // indirect + github.com/jackc/pgx/v4 v4.14.0 // indirect + github.com/jmespath/go-jmespath v0.4.0 // indirect + github.com/klauspost/compress v1.12.3 // indirect + github.com/manifoldco/promptui v0.9.0 // indirect + github.com/mgutz/ansi v0.0.0-20200706080929-d51e80ef957d // indirect + github.com/miekg/pkcs11 v1.1.1 // indirect + github.com/mitchellh/copystructure v1.2.0 // indirect + github.com/mitchellh/go-homedir v1.1.0 // indirect + github.com/mitchellh/go-testing-interface v1.0.0 // indirect + github.com/mitchellh/mapstructure v1.4.2 // indirect + github.com/mitchellh/reflectwalk v1.0.2 // indirect + github.com/oklog/run v1.0.0 // indirect + github.com/pierrec/lz4 v2.5.2+incompatible // indirect + github.com/pmezard/go-difflib v1.0.0 // indirect + github.com/russross/blackfriday/v2 v2.0.1 // indirect + github.com/ryanuber/go-glob v1.0.0 // indirect + github.com/shopspring/decimal v1.2.0 // indirect + github.com/shurcooL/sanitized_anchor_name v1.0.0 // indirect + github.com/spf13/cast v1.4.1 // indirect + github.com/thales-e-security/pool v0.0.2 // indirect + go.etcd.io/bbolt v1.3.6 // indirect + go.opencensus.io v0.23.0 // indirect + go.uber.org/atomic v1.9.0 // indirect + golang.org/x/oauth2 v0.0.0-20220608161450-d0670ef3b1eb // indirect + golang.org/x/sys v0.0.0-20220610221304-9f5ed59c137d // indirect + golang.org/x/text v0.3.8-0.20211004125949-5bd84dd9b33b // indirect + google.golang.org/appengine v1.6.7 // indirect + gopkg.in/yaml.v3 v3.0.0 // indirect +) + // replace github.com/smallstep/nosql => ../nosql // replace go.step.sm/crypto => ../crypto // replace go.step.sm/cli-utils => ../cli-utils @@ -62,5 +150,3 @@ require ( // use github.com/smallstep/pkcs7 fork with patches applied replace go.mozilla.org/pkcs7 => github.com/smallstep/pkcs7 v0.0.0-20211016004704-52592125d6f6 - -replace go.step.sm/crypto => ../crypto diff --git a/go.sum b/go.sum index 1dc18c6e..b4fc5262 100644 --- a/go.sum +++ b/go.sum @@ -93,10 +93,13 @@ github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03 github.com/BurntSushi/xgb v0.0.0-20160522181843-27f122750802/go.mod h1:IVnqGOEym/WlBOVXweHU+Q+/VP0lqqI8lqeDx9IjBqo= github.com/DataDog/datadog-go v3.2.0+incompatible/go.mod h1:LButxg5PwREeZtORoXG3tL4fMGNddJ+vMq1mwgfaqoQ= github.com/Knetic/govaluate v3.0.1-0.20171022003610-9aa49832a739+incompatible/go.mod h1:r7JcOSlj0wfOMncg0iLm8Leh48TZaKVeNIfJntJ2wa0= +github.com/Masterminds/goutils v1.1.0/go.mod h1:8cTjp+g8YejhMuvIA5y2vz3BpJxksy863GQaJW2MFNU= github.com/Masterminds/goutils v1.1.1 h1:5nUrii3FMTL5diU80unEVvNevw1nH4+ZV4DSLVJLSYI= github.com/Masterminds/goutils v1.1.1/go.mod h1:8cTjp+g8YejhMuvIA5y2vz3BpJxksy863GQaJW2MFNU= +github.com/Masterminds/semver/v3 v3.1.0/go.mod h1:VPu/7SZ7ePZ3QOrcuXROw5FAcLl4a0cBrbBpGY/8hQs= github.com/Masterminds/semver/v3 v3.1.1 h1:hLg3sBzpNErnxhQtUy/mmLR2I9foDujNK030IGemrRc= github.com/Masterminds/semver/v3 v3.1.1/go.mod h1:VPu/7SZ7ePZ3QOrcuXROw5FAcLl4a0cBrbBpGY/8hQs= +github.com/Masterminds/sprig/v3 v3.1.0/go.mod h1:ONGMf7UfYGAbMXCZmQLy8x3lCDIPrEZE/rU8pmrbihA= github.com/Masterminds/sprig/v3 v3.2.2 h1:17jRggJu518dr3QaafizSXOjKYp94wKfABxUmyxvxX8= github.com/Masterminds/sprig/v3 v3.2.2/go.mod h1:UoaO7Yp8KlPnJIYWTFkMaqPUYKTfGFPhxNuwnnxkKlk= github.com/OneOfOne/xxhash v1.2.2 h1:KMrpdQIwFcEqXDklaen+P1axHaj9BSKzvpUUfnHldSE= @@ -111,8 +114,6 @@ github.com/alecthomas/template v0.0.0-20160405071501-a0175ee3bccc/go.mod h1:LOuy github.com/alecthomas/template v0.0.0-20190718012654-fb15b899a751/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc= github.com/alecthomas/units v0.0.0-20151022065526-2efee857e7cf/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0= github.com/alecthomas/units v0.0.0-20190717042225-c3de453c63f4/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0= -github.com/alecthomas/units v0.0.0-20190924025748-f65c72e2690d/go.mod h1:rBZYJk541a8SKzHPHnH3zbiI+7dagKZ0cgpgrD7Fyho= -github.com/anmitsu/go-shlex v0.0.0-20200514113438-38f4b401e2be/go.mod h1:ySMOLuWl6zY27l47sB3qLNK6tF2fkHG55UZxx8oIVo4= github.com/antihax/optional v1.0.0/go.mod h1:uupD/76wgC+ih3iEmQUL+0Ugr19nfwCT1kdvxnR2qWY= github.com/apache/thrift v0.12.0/go.mod h1:cp2SuWMxlEZw2r+iP2GNCdIi4C1qmUzdZFSVb+bacwQ= github.com/apache/thrift v0.13.0/go.mod h1:cp2SuWMxlEZw2r+iP2GNCdIi4C1qmUzdZFSVb+bacwQ= @@ -136,7 +137,6 @@ github.com/beorn7/perks v1.0.1/go.mod h1:G2ZrVWU2WbWT9wwq4/hrbKbnv/1ERSJQ0ibhJ6r github.com/bgentry/speakeasy v0.1.0/go.mod h1:+zsyZBPWlz7T6j88CTgSN5bM796AkVf0kBD4zp0CCIs= github.com/boltdb/bolt v1.3.1/go.mod h1:clJnj/oiGkjum5o1McbSZDSLxVThjynRyGBgiAx27Ps= github.com/casbin/casbin/v2 v2.1.2/go.mod h1:YcPU1XXisHhLzuxH9coDNf2FbKpjGlbCg3n9yuLkIJQ= -github.com/cenkalti/backoff v2.2.1+incompatible h1:tNowT99t7UNflLxfYYSlKYsBpXdEet03Pg2g16Swow4= github.com/cenkalti/backoff v2.2.1+incompatible/go.mod h1:90ReRw6GdpyfrHakVjL/QHaoyV4aDUVVkXQJJJ3NXXM= github.com/cenkalti/backoff/v3 v3.0.0 h1:ske+9nBpD9qZsTBoF41nW5L+AIuFBKMeze18XQ3eG1c= github.com/cenkalti/backoff/v3 v3.0.0/go.mod h1:cIeZDE3IrqwwJl6VUwCN6trj1oXrTS4rc0ij+ULvLYs= @@ -144,7 +144,6 @@ github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA github.com/cespare/xxhash v1.1.0 h1:a6HrQnmkObjyL+Gs60czilIUGqrzKutQD6XZog3p+ko= github.com/cespare/xxhash v1.1.0/go.mod h1:XrSqR1VqqWfGrhpAt58auRo0WTKS1nRRg3ghfAqPWnc= github.com/cespare/xxhash/v2 v2.1.1/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= -github.com/cespare/xxhash/v2 v2.1.2/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= github.com/chzyer/logex v1.1.10 h1:Swpa1K6QvQznwJRcfTfQJmTE72DqScAa40E+fbHEXEE= github.com/chzyer/logex v1.1.10/go.mod h1:+Ywpsq7O8HXn0nuIou7OrIPyXbp3wmkHB+jjWRnGsAI= github.com/chzyer/readline v0.0.0-20180603132655-2972be24d48e h1:fY5BOSpyZCqRo5OhCuC+XN+r/bBCmeuuJtjz+bCNIf8= @@ -175,14 +174,12 @@ github.com/coreos/go-systemd v0.0.0-20180511133405-39ca1b05acc7/go.mod h1:F5haX7 github.com/coreos/go-systemd v0.0.0-20190321100706-95778dfbb74e/go.mod h1:F5haX7vjVVG0kc13fIWeqUViNPyEJxv/OmvnBo0Yme4= github.com/coreos/go-systemd v0.0.0-20190719114852-fd7a80b32e1f/go.mod h1:F5haX7vjVVG0kc13fIWeqUViNPyEJxv/OmvnBo0Yme4= github.com/coreos/pkg v0.0.0-20160727233714-3ac0863d7acf/go.mod h1:E3G3o1h8I7cfcXa63jLwjI0eiQQMgzzUDFVpN/nH/eA= -github.com/cpuguy83/go-md2man v1.0.10 h1:BSKMNlYxDvnunlTymqtgONjNnaRV1sTpcovwwjF22jk= github.com/cpuguy83/go-md2man v1.0.10/go.mod h1:SmD6nW6nTyfqj6ABTjUi3V3JVMnlJmwcJI5acqYI6dE= github.com/cpuguy83/go-md2man/v2 v2.0.0-20190314233015-f79a8a8ca69d/go.mod h1:maD7wRr/U5Z6m/iR4s+kqSMx2CaBsrgA7czyZG/E6dU= github.com/cpuguy83/go-md2man/v2 v2.0.0 h1:EoUDS0afbrsXAZ9YQ9jdu/mZ2sXgT1/2yyNng4PGlyM= github.com/cpuguy83/go-md2man/v2 v2.0.0/go.mod h1:maD7wRr/U5Z6m/iR4s+kqSMx2CaBsrgA7czyZG/E6dU= github.com/creack/pty v1.1.7/go.mod h1:lj5s0c3V2DBrqTV7llrYr5NG6My20zk30Fl46Y7DoTY= github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E= -github.com/cyberdelia/go-metrics-graphite v0.0.0-20161219230853-39f87cc3b432/go.mod h1:xwIwAxMvYnVrGJPe2FKx5prTrnAjGOD8zvDOnxnrrkM= github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= @@ -224,7 +221,6 @@ github.com/fatih/color v1.9.0 h1:8xPHl4/q1VyqGIPif1F+1V3Y3lSmrq01EabUW3CoW5s= github.com/fatih/color v1.9.0/go.mod h1:eQcE1qtQxscV5RaZvpXrrb8Drkc3/DdQ+uUYCNjL+zU= github.com/fatih/structs v1.1.0 h1:Q7juDM0QtcnhCpeyLGQKyg4TOIghuNXrkL32pHAUMxo= github.com/fatih/structs v1.1.0/go.mod h1:9NiDSp5zOcgEDl+j00MP/WkGVPOlPRLejGD8Ga6PJ7M= -github.com/flynn/noise v1.0.0/go.mod h1:xbMo+0i6+IGbYdJhF31t2eR1BIU0CYc12+BNAKwUTag= github.com/franela/goblin v0.0.0-20200105215937-c9ffbefa60db/go.mod h1:7dvUGVsVBjqR7JHJk0brhHOZYGmfBYOrK0ZhYMEtBr4= github.com/franela/goreq v0.0.0-20171204163338-bcd34c9993f8/go.mod h1:ZhphrRTfi2rbfLwlschooIH4+wKKDR4Pdxhh+TRoA20= github.com/frankban/quicktest v1.10.0/go.mod h1:ui7WezCLWMWxVWr1GETZY3smRy0G4KWq9vcPtJmFl7Y= @@ -233,8 +229,8 @@ github.com/frankban/quicktest v1.13.0/go.mod h1:qLE0fzW0VuyUAJgPU19zByoIr0HtCHN/ github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo= github.com/ghodss/yaml v1.0.0/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeMEF04= github.com/go-asn1-ber/asn1-ber v1.3.1/go.mod h1:hEBeB/ic+5LoWskz+yKT7vGhhPYkProFKoKdwZRWMe0= -github.com/go-chi/chi v4.0.2+incompatible h1:maB6vn6FqCxrpz4FqWdh4+lwpyZIQS7YEAUcHlgXVRs= -github.com/go-chi/chi v4.0.2+incompatible/go.mod h1:eB3wogJHnLi3x/kFX2A+IbTBlXxmMeXJVKy9tTv1XzQ= +github.com/go-chi/chi v4.1.2+incompatible h1:fGFk2Gmi/YKXk0OmGfBh0WgmN3XB8lVnEyNz34tQRec= +github.com/go-chi/chi v4.1.2+incompatible/go.mod h1:eB3wogJHnLi3x/kFX2A+IbTBlXxmMeXJVKy9tTv1XzQ= github.com/go-gl/glfw v0.0.0-20190409004039-e6da0acd62b1/go.mod h1:vR7hzQXu2zJy9AVAgeJqvqgH9Q5CA+iKCZ2gyEVpxRU= github.com/go-gl/glfw/v3.3/glfw v0.0.0-20191125211704-12ad95a8df72/go.mod h1:tQ2UAYgL5IevRw8kRxooKSPJfGvJ9fJQFa0TUsXzTg8= github.com/go-gl/glfw/v3.3/glfw v0.0.0-20200222043503-6f7a984d4dc4/go.mod h1:tQ2UAYgL5IevRw8kRxooKSPJfGvJ9fJQFa0TUsXzTg8= @@ -249,8 +245,8 @@ github.com/go-logfmt/logfmt v0.3.0/go.mod h1:Qt1PoO58o5twSAckw1HlFXLmHsOX5/0LbT9 github.com/go-logfmt/logfmt v0.4.0/go.mod h1:3RMwSq7FuexP4Kalkev3ejPJsZTpXXBr9+V4qmtdjCk= github.com/go-logfmt/logfmt v0.5.0 h1:TrB8swr/68K7m9CcGut2g3UOihhbcbiMAYiuTXdEih4= github.com/go-logfmt/logfmt v0.5.0/go.mod h1:wCYkCAKZfumFQihp8CzCvQ3paCTfi41vtzG1KdI/P7A= -github.com/go-piv/piv-go v1.9.0 h1:P6j2gjfP7zO7T3nCk/jwCgsvFRwB8shEqAJ4q85jgXc= -github.com/go-piv/piv-go v1.9.0/go.mod h1:NZ2zmjVkfFaL/CF8cVQ/pXdXtuj110zEKGdJM6fJZZM= +github.com/go-piv/piv-go v1.10.0 h1:P1Y1VjBI5DnXW0+YkKmTuh5opWnMIrKriUaIOblee9Q= +github.com/go-piv/piv-go v1.10.0/go.mod h1:NZ2zmjVkfFaL/CF8cVQ/pXdXtuj110zEKGdJM6fJZZM= github.com/go-sql-driver/mysql v1.4.0/go.mod h1:zAC/RDZ24gD3HViQzih4MyKcchzm+sOG5ZlKdlhCg5w= github.com/go-sql-driver/mysql v1.5.0/go.mod h1:DCzpHaOWr8IXmIStZouvnhqoel9Qv2LBy8hT2VhHyBg= github.com/go-sql-driver/mysql v1.6.0 h1:BCTh4TKNUYmOmMUcQ3IipzF5prigylS7XXjEkfCHuOE= @@ -266,7 +262,6 @@ github.com/gogo/googleapis v1.1.0/go.mod h1:gf4bu3Q80BeJ6H1S1vYPm8/ELATdvryBaNFG github.com/gogo/protobuf v1.1.1/go.mod h1:r8qH/GZQm5c6nD/R0oafs1akxWv10x8SbQlK7atdtwQ= github.com/gogo/protobuf v1.2.0/go.mod h1:r8qH/GZQm5c6nD/R0oafs1akxWv10x8SbQlK7atdtwQ= github.com/gogo/protobuf v1.2.1/go.mod h1:hp+jE20tsWTFYpLwKvXlhS1hjn+gTNwPg2I6zVXpSg4= -github.com/gogo/protobuf v1.3.2/go.mod h1:P1XiOD3dCwIKUDQYPy72D8LYyHL2YPYrpS2s69NZV8Q= github.com/golang-jwt/jwt/v4 v4.0.0/go.mod h1:/xlHOz8bRuivTWchD4jCa+NbatV+wEUSzwAxVc6locg= github.com/golang-jwt/jwt/v4 v4.2.0 h1:besgBTC8w8HjP6NzQdxwKH9Z5oQMZ24ThTrHp3cZ8eU= github.com/golang-jwt/jwt/v4 v4.2.0/go.mod h1:/xlHOz8bRuivTWchD4jCa+NbatV+wEUSzwAxVc6locg= @@ -327,7 +322,6 @@ github.com/google/go-cmp v0.5.7/go.mod h1:n+brtR0CgQNWTVd5ZUFpTBC8YFBDLK/h/bpaJ8 github.com/google/go-cmp v0.5.8 h1:e6P7q2lk1O+qJJb4BtCQXlK8vWEO8V1ZeuEdJNOqZyg= github.com/google/go-cmp v0.5.8/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg= -github.com/google/gopacket v1.1.19/go.mod h1:iJ8V8n6KS+z2U1A8pUwu8bW5SyEMkXJB8Yo/Vo+TKTo= github.com/google/martian v2.1.0+incompatible/go.mod h1:9I4somxYTbIHy5NJKHRl3wXiIaQGbYVAs8BPL6v8lEs= github.com/google/martian/v3 v3.0.0/go.mod h1:y5Zk1BBys9G+gd6Jrk0W3cC1+ELVxBWuIGO+w/tUAp0= github.com/google/martian/v3 v3.1.0/go.mod h1:y5Zk1BBys9G+gd6Jrk0W3cC1+ELVxBWuIGO+w/tUAp0= @@ -458,7 +452,6 @@ github.com/imdario/mergo v0.3.12 h1:b6R2BslTbIEToALKP7LxUvijTsNI9TAe80pLWN2g/HU= github.com/imdario/mergo v0.3.12/go.mod h1:jmQim1M+e3UYxmgPu/WyfjB3N3VflVyUjjjwH0dnCYA= github.com/inconshreveable/mousetrap v1.0.0/go.mod h1:PxqpIevigyE2G7u3NXJIT2ANytuPF1OarO4DADm73n8= github.com/influxdata/influxdb1-client v0.0.0-20191209144304-8bf82d3c094d/go.mod h1:qj24IKcXYK6Iy9ceXlo3Tc+vtHo9lIhSX5JddghvEPo= -github.com/jackc/chunkreader v1.0.0 h1:4s39bBR8ByfqH+DKm8rQA3E1LHZWB9XWcrz8fqaZbe0= github.com/jackc/chunkreader v1.0.0/go.mod h1:RT6O25fNZIuasFJRyZ4R/Y2BbhasbmZXF9QQ7T3kePo= github.com/jackc/chunkreader/v2 v2.0.0/go.mod h1:odVSm741yZoC3dpHEUXIqA9tQRhFrgOHwnPIn9lDKlk= github.com/jackc/chunkreader/v2 v2.0.1 h1:i+RDz65UE+mmpjTfyz0MoVTnzeYxroil2G82ki7MGG8= @@ -479,7 +472,6 @@ github.com/jackc/pgmock v0.0.0-20210724152146-4ad1a8207f65 h1:DadwsjnMwFjfWc9y5W github.com/jackc/pgmock v0.0.0-20210724152146-4ad1a8207f65/go.mod h1:5R2h2EEX+qri8jOWMbJCtaPWkrrNc7OHwsp2TCqp7ak= github.com/jackc/pgpassfile v1.0.0 h1:/6Hmqy13Ss2zCq62VdNG8tM1wchn8zjSGOBJ6icpsIM= github.com/jackc/pgpassfile v1.0.0/go.mod h1:CEx0iS5ambNFdcRtxPj5JhEz+xB6uRky5eyVu/W2HEg= -github.com/jackc/pgproto3 v1.1.0 h1:FYYE4yRw+AgI8wXIinMlNjBbp/UitDJwfj5LqqewP1A= github.com/jackc/pgproto3 v1.1.0/go.mod h1:eR5FA3leWg7p9aeAqi37XOTgTIbkABlvcPB3E5rlc78= github.com/jackc/pgproto3/v2 v2.0.0-alpha1.0.20190420180111-c116219b62db/go.mod h1:bhq50y+xrl9n5mRYyCBFKkpRVTLYJVWeCc+mEAI3yXA= github.com/jackc/pgproto3/v2 v2.0.0-alpha1.0.20190609003834-432c2951c711/go.mod h1:uH0AWtUmuShn0bcesswc4aBTWGvw0cAxIJp+6OB//Wg= @@ -517,27 +509,20 @@ github.com/jmespath/go-jmespath v0.4.0/go.mod h1:T8mJZnbsbmF+m6zOOFylbeCJqk5+pHW github.com/jmespath/go-jmespath/internal/testify v1.5.1 h1:shLQSRRSCCPj3f2gpwzGwWFoC7ycTf1rcQZHOlsJ6N8= github.com/jmespath/go-jmespath/internal/testify v1.5.1/go.mod h1:L3OGu8Wl2/fWfCI6z80xFu9LTZmf1ZRjMHUOPmWr69U= github.com/jonboulle/clockwork v0.1.0/go.mod h1:Ii8DK3G1RaLaWxj9trq07+26W01tbo22gdxWY5EU2bo= -github.com/jpillora/backoff v1.0.0/go.mod h1:J/6gKK9jxlEcS3zixgDgUAsiuZ7yrSoa/FX5e0EB2j4= github.com/json-iterator/go v1.1.6/go.mod h1:+SdeFBvtyEkXs7REEP0seUULqWtbJapLOCVDaaPEHmU= github.com/json-iterator/go v1.1.7/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= github.com/json-iterator/go v1.1.8/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= github.com/json-iterator/go v1.1.9/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= -github.com/json-iterator/go v1.1.10/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= -github.com/json-iterator/go v1.1.11/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= github.com/jstemmer/go-junit-report v0.0.0-20190106144839-af01ea7f8024/go.mod h1:6v2b51hI/fHJwM22ozAgKL4VKDeJcHhJFhtBdhmNjmU= github.com/jstemmer/go-junit-report v0.9.1/go.mod h1:Brl9GWCQeLvo8nXZwPNNblvFj/XSXhF0NWZEnDohbsk= github.com/jtolds/gls v4.20.0+incompatible/go.mod h1:QJZ7F/aHp+rZTRtaJ1ow/lLfFfVYBRgL+9YlvaHOwJU= github.com/julienschmidt/httprouter v1.2.0/go.mod h1:SYymIcj16QtmaHHD7aYtjjsJG7VTCxuUUipMqKk8s4w= -github.com/julienschmidt/httprouter v1.3.0/go.mod h1:JR6WtHb+2LUe8TCKY3cZOxFyyO8IZAc4RVcycCCAKdM= -github.com/kardianos/service v1.2.0/go.mod h1:CIMRFEJVL+0DS1a3Nx06NaMn4Dz63Ng6O7dl0qH0zVM= github.com/kisielk/errcheck v1.1.0/go.mod h1:EZBBE59ingxPouuu3KfxchcWSUPOHkagtvWXihfKN4Q= -github.com/kisielk/errcheck v1.5.0/go.mod h1:pFxgyoBC7bSaBwPgfKdkLd5X25qrDl4LWUI2bnpBCr8= github.com/kisielk/gotool v1.0.0/go.mod h1:XhKaO+MFFWcvkIS/tQcRk01m1F5IRFswLeQ+oQHNcck= github.com/klauspost/compress v1.12.3 h1:G5AfA94pHPysR56qqrkO2pxEexdDzrpFJ6yt/VqWxVU= github.com/klauspost/compress v1.12.3/go.mod h1:8dP1Hq4DHOhN9w426knH3Rhby4rFm6D8eO+e+Dq5Gzg= github.com/konsorten/go-windows-terminal-sequences v1.0.1/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= github.com/konsorten/go-windows-terminal-sequences v1.0.2/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= -github.com/konsorten/go-windows-terminal-sequences v1.0.3/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= github.com/kr/logfmt v0.0.0-20140226030751-b84e30acd515/go.mod h1:+0opPa2QZZtGFBFZlji/RkVcI2GknAs/DXo4wKdlNEc= github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= github.com/kr/pretty v0.2.0/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI= @@ -556,8 +541,6 @@ github.com/lib/pq v1.10.2 h1:AqzbZs4ZoCBp+GtejcpCpcxM3zlSMx29dXbUSeVtJb8= github.com/lib/pq v1.10.2/go.mod h1:AlVN5x4E4T544tWzH6hKfbfQvm3HdbOxrmggDNAPY9o= github.com/lightstep/lightstep-tracer-common/golang/gogo v0.0.0-20190605223551-bc2310a04743/go.mod h1:qklhhLq1aX+mtWk9cPHPzaBjWImj5ULL6C7HFJtXQMM= github.com/lightstep/lightstep-tracer-go v0.18.1/go.mod h1:jlF1pusYV4pidLvZ+XD0UBX0ZE6WURAspgAczcDHrL4= -github.com/lxn/walk v0.0.0-20210112085537-c389da54e794/go.mod h1:E23UucZGqpuUANJooIbHWCufXvOcT6E7Stq81gU+CSQ= -github.com/lxn/win v0.0.0-20210218163916-a377121e959e/go.mod h1:KxxjdtRkfNoYDCUP5ryK7XJJNTnpC8atvtmTheChOtk= github.com/lyft/protoc-gen-validate v0.0.13/go.mod h1:XbGvPuh87YZc5TdIa2/I4pLk0QoUACkjt2znoq26NVQ= github.com/magiconair/properties v1.8.0/go.mod h1:PppfXfuXeibc/6YijjN8zIbojt8czPbwD3XqdrwzmxQ= github.com/manifoldco/promptui v0.9.0 h1:3V4HzJk1TtXW1MTZMP7mdlwbBpIinw3HztaIlYthEiA= @@ -585,10 +568,9 @@ github.com/mgutz/ansi v0.0.0-20200706080929-d51e80ef957d/go.mod h1:01TrycV0kFyex github.com/micromdm/scep/v2 v2.1.0 h1:2fS9Rla7qRR266hvUoEauBJ7J6FhgssEiq2OkSKXmaU= github.com/micromdm/scep/v2 v2.1.0/go.mod h1:BkF7TkPPhmgJAMtHfP+sFTKXmgzNJgLQlvvGoOExBcc= github.com/miekg/dns v1.0.14/go.mod h1:W1PPwlIAgtquWBMBEV9nkV9Cazfe8ScdGz/Lj7v3Nrg= -github.com/miekg/dns v1.1.43/go.mod h1:+evo5L0630/F6ca/Z9+GAqzhjGyn8/c+TBaOyfEl0V4= github.com/miekg/pkcs11 v1.0.3-0.20190429190417-a667d056470f/go.mod h1:XsNlhZGX73bx86s2hdc/FuaLm2CPZJemRLMA+WTFxgs= -github.com/miekg/pkcs11 v1.0.3 h1:iMwmD7I5225wv84WxIG/bmxz9AXjWvTWIbM/TYHvWtw= -github.com/miekg/pkcs11 v1.0.3/go.mod h1:XsNlhZGX73bx86s2hdc/FuaLm2CPZJemRLMA+WTFxgs= +github.com/miekg/pkcs11 v1.1.1 h1:Ugu9pdy6vAYku5DEpVWVFPYnzV+bxB+iRdbuFSu7TvU= +github.com/miekg/pkcs11 v1.1.1/go.mod h1:XsNlhZGX73bx86s2hdc/FuaLm2CPZJemRLMA+WTFxgs= github.com/mitchellh/cli v1.0.0/go.mod h1:hNIlj7HEI86fIcpObd7a0FcrxTWetlwJDGcceTlRvqc= github.com/mitchellh/copystructure v1.0.0/go.mod h1:SNtv71yrdKgLRyLFxmLdkAbkKEFWgYaq1OVrnRcwhnw= github.com/mitchellh/copystructure v1.2.0 h1:vpKXTN4ewci03Vljg/q9QvCGUDttBOGBIa15WveJJGw= @@ -615,7 +597,6 @@ github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJ github.com/modern-go/reflect2 v0.0.0-20180701023420-4b7aa43c6742/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0= github.com/modern-go/reflect2 v1.0.1/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0= github.com/mwitkow/go-conntrack v0.0.0-20161129095857-cc309e4a2223/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U= -github.com/mwitkow/go-conntrack v0.0.0-20190716064945-2f068394615f/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U= github.com/nats-io/jwt v0.3.0/go.mod h1:fRYCDE99xlTsqUzISS1Bi75UBJ6ljOJQOAAu5VglpSg= github.com/nats-io/jwt v0.3.2/go.mod h1:/euKqTS1ZD+zzjYrY7pseZrTtWQSjujC7xjPc8wL6eU= github.com/nats-io/nats-server/v2 v2.1.2/go.mod h1:Afk+wRZqkMQs/p45uXdrVLuab3gwv3Z8C4HTBu8GD/k= @@ -623,9 +604,8 @@ github.com/nats-io/nats.go v1.9.1/go.mod h1:ZjDU1L/7fJ09jvUSRVBR2e7+RnLiiIQyqyzE github.com/nats-io/nkeys v0.1.0/go.mod h1:xpnFELMwJABBLVhffcfd1MZx6VsNRFpEugbxziKVo7w= github.com/nats-io/nkeys v0.1.3/go.mod h1:xpnFELMwJABBLVhffcfd1MZx6VsNRFpEugbxziKVo7w= github.com/nats-io/nuid v1.0.1/go.mod h1:19wcPz3Ph3q0Jbyiqsd0kePYG7A95tJPxeL+1OSON2c= -github.com/nbrownus/go-metrics-prometheus v0.0.0-20210712211119-974a6260965f/go.mod h1:nwPd6pDNId/Xi16qtKrFHrauSwMNuvk+zcjk89wrnlA= -github.com/newrelic/go-agent v2.15.0+incompatible h1:IB0Fy+dClpBq9aEoIrLyQXzU34JyI1xVTanPLB/+jvU= -github.com/newrelic/go-agent v2.15.0+incompatible/go.mod h1:a8Fv1b/fYhFSReoTU6HDkTYIMZeSVNffmoS726Y0LzQ= +github.com/newrelic/go-agent/v3 v3.18.0 h1:AOR3hhF2ZVE0yfvNPuOaEhEvNMYyIfEBY8EizQpnt7g= +github.com/newrelic/go-agent/v3 v3.18.0/go.mod h1:BFJOlbZWRlPTXKYIC1TTTtQKTnYntEJaU0VU507hDc0= github.com/nishanths/predeclared v0.0.0-20200524104333-86fad755b4d3/go.mod h1:nt3d53pc1VYcphSCIaYAJtnPYnr3Zyn8fMq2wvPGPso= github.com/oklog/oklog v0.3.2/go.mod h1:FCV+B7mhrz4o+ueLpx+KqkyXRGMWOYEvfiXtdGtbWGs= github.com/oklog/run v1.0.0 h1:Ru7dDtJNOyC66gQ5dQmaCa0qIsAUFY3sFpK1Xk8igrw= @@ -667,8 +647,6 @@ github.com/prometheus/client_golang v0.9.3-0.20190127221311-3c4408c8b829/go.mod github.com/prometheus/client_golang v1.0.0/go.mod h1:db9x61etRT2tGnBNRi70OPL5FsnadC4Ky3P0J6CfImo= github.com/prometheus/client_golang v1.3.0/go.mod h1:hJaj2vgQTGQmVCsAACORcieXFeDPbaTKGT+JTgUa3og= github.com/prometheus/client_golang v1.4.0/go.mod h1:e9GMxYsXl05ICDXkRhurwBS4Q3OK1iX/F2sw+iXX5zU= -github.com/prometheus/client_golang v1.7.1/go.mod h1:PY5Wy2awLA44sXw4AOSfFBetzPP4j5+D6mVACh+pe2M= -github.com/prometheus/client_golang v1.11.0/go.mod h1:Z6t4BnS23TR94PD6BsDNk8yVqroYurpAkEiz0P2BEV0= github.com/prometheus/client_model v0.0.0-20180712105110-5c3871d89910/go.mod h1:MbSGuTsp3dbXC40dX6PRTWyKYBIrTGTE9sqQNg2J8bo= github.com/prometheus/client_model v0.0.0-20190115171406-56726106282f/go.mod h1:MbSGuTsp3dbXC40dX6PRTWyKYBIrTGTE9sqQNg2J8bo= github.com/prometheus/client_model v0.0.0-20190129233127-fd36f4220a90/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= @@ -679,18 +657,11 @@ github.com/prometheus/common v0.2.0/go.mod h1:TNfzLD0ON7rHzMJeJkieUDPYmFC7Snx/y8 github.com/prometheus/common v0.4.1/go.mod h1:TNfzLD0ON7rHzMJeJkieUDPYmFC7Snx/y86RQel1bk4= github.com/prometheus/common v0.7.0/go.mod h1:DjGbpBbp5NYNiECxcL/VnbXCCaQpKd3tt26CguLLsqA= github.com/prometheus/common v0.9.1/go.mod h1:yhUN8i9wzaXS3w1O07YhxHEBxD+W35wd8bs7vj7HSQ4= -github.com/prometheus/common v0.10.0/go.mod h1:Tlit/dnDKsSWFlCLTWaA1cyBgKHSMdTB80sz/V91rCo= -github.com/prometheus/common v0.26.0/go.mod h1:M7rCNAaPfAosfx8veZJCuw84e35h3Cfd9VFqTh1DIvc= -github.com/prometheus/common v0.32.1/go.mod h1:vu+V0TpY+O6vW9J44gczi3Ap/oXXR10b+M/gUGO4Hls= github.com/prometheus/procfs v0.0.0-20181005140218-185b4288413d/go.mod h1:c3At6R/oaqEKCNdg8wHV1ftS6bRYblBhIjjI8uT2IGk= github.com/prometheus/procfs v0.0.0-20190117184657-bf6a532e95b1/go.mod h1:c3At6R/oaqEKCNdg8wHV1ftS6bRYblBhIjjI8uT2IGk= github.com/prometheus/procfs v0.0.2/go.mod h1:TjEm7ze935MbeOT/UhFTIMYKhuLP4wbCsTZCD3I8kEA= github.com/prometheus/procfs v0.0.8/go.mod h1:7Qr8sr6344vo1JqZ6HhLceV9o3AJ1Ff+GxbHq6oeK9A= -github.com/prometheus/procfs v0.1.3/go.mod h1:lV6e/gmhEcM9IjHGsFOCxxuZ+z1YqCvr4OA4YeYWdaU= -github.com/prometheus/procfs v0.6.0/go.mod h1:cz+aTbrPOrUb4q7XlbU9ygM+/jj0fzG6c1xBZuNvfVA= -github.com/prometheus/procfs v0.7.3/go.mod h1:cz+aTbrPOrUb4q7XlbU9ygM+/jj0fzG6c1xBZuNvfVA= github.com/rcrowley/go-metrics v0.0.0-20181016184325-3113b8401b8a/go.mod h1:bCqnVzQkZxMG4s8nGwiZ5l3QUCyqpo9Y+/ZMZ9VjZe4= -github.com/rcrowley/go-metrics v0.0.0-20201227073835-cf1acfcdf475/go.mod h1:bCqnVzQkZxMG4s8nGwiZ5l3QUCyqpo9Y+/ZMZ9VjZe4= github.com/rogpeppe/fastuuid v0.0.0-20150106093220-6724a57986af/go.mod h1:XWv6SoW27p1b0cqNHllgS5HIMJraePCO15w5zCzIWYg= github.com/rogpeppe/fastuuid v1.2.0/go.mod h1:jVj6XXZzXRy/MSR5jhDC/2q6DgLz+nrA6LYCDYWNEvQ= github.com/rogpeppe/go-internal v1.3.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4= @@ -700,7 +671,6 @@ github.com/rs/xid v1.2.1 h1:mhH9Nq+C1fY2l1XIpgxIiUOfNpRBYH1kKcr+qfKgjRc= github.com/rs/xid v1.2.1/go.mod h1:+uKXf+4Djp6Md1KODXJxgGQPKngRmWyn10oCKFzNHOQ= github.com/rs/zerolog v1.13.0/go.mod h1:YbFCdg8HfsridGWAh22vktObvhZbQsZXe4/zB0OKkWU= github.com/rs/zerolog v1.15.0/go.mod h1:xYTKnLHcpfU2225ny5qZjxnj9NvkumZYjJHlAThCjNc= -github.com/russross/blackfriday v1.5.2 h1:HyvC0ARfnZBqnXwABFeSZHpKvJHJJfPz81GNueLj0oo= github.com/russross/blackfriday v1.5.2/go.mod h1:JO/DiYxRf+HjHt06OyowR9PTA263kcR/rfWxYHBV53g= github.com/russross/blackfriday/v2 v2.0.1 h1:lPqVAte+HuHNfhJ/0LC98ESWRz8afy9tM/0RK8m9o+Q= github.com/russross/blackfriday/v2 v2.0.1/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM= @@ -719,10 +689,8 @@ github.com/shurcooL/sanitized_anchor_name v1.0.0/go.mod h1:1NzhyTcUVG4SuEtjjoZeV github.com/sirupsen/logrus v1.2.0/go.mod h1:LxeOpSwHxABJmUn/MG1IvRgCAasNZTLOkJPxbbu5VWo= github.com/sirupsen/logrus v1.4.1/go.mod h1:ni0Sbl8bgC9z8RoU9G6nDWqqs/fq4eDPysMBDgk/93Q= github.com/sirupsen/logrus v1.4.2/go.mod h1:tLMulIdttU9McNUspp0xgXVQah82FyeX6MwdIuYE2rE= -github.com/sirupsen/logrus v1.6.0/go.mod h1:7uNnSEd1DgxDLC74fIahvMZmmYsHGZGEOFrfsX/uA88= github.com/sirupsen/logrus v1.8.1 h1:dJKuHgqk1NNQlqoA6BTlM1Wf9DOH3NBjQyu0h9+AZZE= github.com/sirupsen/logrus v1.8.1/go.mod h1:yWOB1SBYBC5VeMP7gHvWumXLIWorT60ONWic61uBYv0= -github.com/skip2/go-qrcode v0.0.0-20200617195104-da1b6568686e/go.mod h1:XV66xRDqSt+GTGFMVlhk3ULuV0y9ZmzeVGR4mloJI3M= github.com/slackhq/nebula v1.5.2 h1:wuIOHsOnrNw3rQx8yPxXiGu8wAtAxxtUI/K8W7Vj7EI= github.com/slackhq/nebula v1.5.2/go.mod h1:xaCM6wqbFk/NRmmUe1bv88fWBm3a1UioXJVIpR52WlE= github.com/smallstep/assert v0.0.0-20180720014142-de77670473b5/go.mod h1:TC9A4+RjIOS+HyTH7wG17/gSqVv95uDw2J64dQZx7RE= @@ -735,7 +703,6 @@ github.com/smallstep/pkcs7 v0.0.0-20211016004704-52592125d6f6/go.mod h1:SNgMg+Eg github.com/smartystreets/assertions v0.0.0-20180927180507-b2de0cb4f26d/go.mod h1:OnSkiWE9lh6wB0YB77sQom3nweQdgAjqCqsofrRNTgc= github.com/smartystreets/goconvey v1.6.4/go.mod h1:syvi0/a8iFYH4r/RixwvyeAJjdLS9QV7WQ/tjFTllLA= github.com/soheilhy/cmux v0.1.4/go.mod h1:IM3LyeVVIOuxMH7sFAkER9+bJ4dT7Ms6E4xg4kGIyLM= -github.com/songgao/water v0.0.0-20200317203138-2b4b6d7c09d8/go.mod h1:P5HUIBuIWKbyjl083/loAegFkfbFNx5i2qEP4CNbm7E= github.com/sony/gobreaker v0.4.1/go.mod h1:ZKptC7FHNvhBz7dN2LGjPVBz2sZJmc0/PkyDJOjmxWY= github.com/spaolacci/murmur3 v0.0.0-20180118202830-f09979ecbc72/go.mod h1:JwIasOWyU6f++ZhiEuf87xNszmSA2myDM2Kzu9HwQUA= github.com/spaolacci/murmur3 v1.1.0 h1:7c1g84S4BPRrfL5Xrdp6fOJ206sU9y293DDHaoy0bLI= @@ -776,9 +743,6 @@ github.com/urfave/cli v1.22.1/go.mod h1:Gos4lmkARVdJ6EkW0WaNv/tZAAMe9V7XWyB60NtX github.com/urfave/cli v1.22.2/go.mod h1:Gos4lmkARVdJ6EkW0WaNv/tZAAMe9V7XWyB60NtXRu0= github.com/urfave/cli v1.22.4 h1:u7tSpNPPswAFymm8IehJhy4uJMlUuU/GmqSkvJ1InXA= github.com/urfave/cli v1.22.4/go.mod h1:Gos4lmkARVdJ6EkW0WaNv/tZAAMe9V7XWyB60NtXRu0= -github.com/vishvananda/netlink v1.1.0/go.mod h1:cTgwzPIzzgDAYoQrMm0EdrjRUBkTqKYppBueQtXaqoE= -github.com/vishvananda/netns v0.0.0-20191106174202-0a2b9b5464df/go.mod h1:JP3t17pCcGlemwknint6hfoeCVQrEMVwxRLRjXpq+BU= -github.com/vishvananda/netns v0.0.0-20211101163701-50045581ed74/go.mod h1:DD4vA1DwXk04H54A1oHXtwZmA0grkVMdPxx/VGLCah0= github.com/xiang90/probing v0.0.0-20190116061207-43a291ad63a2/go.mod h1:UETIi67q53MR2AWcXfiuqkDkRtnGDLqkBTpCHuJHxtU= github.com/xordataexchange/crypt v0.0.3-0.20170626215501-b2862e3d0a77/go.mod h1:aYKd//L2LvnjZzWKhF00oedf4jCCReLcmhLdhm1A27Q= github.com/yuin/goldmark v1.1.25/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= @@ -786,7 +750,6 @@ github.com/yuin/goldmark v1.1.27/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9de github.com/yuin/goldmark v1.1.32/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= github.com/yuin/goldmark v1.2.1/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= github.com/yuin/goldmark v1.3.5/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k= -github.com/yuin/goldmark v1.4.0/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k= github.com/zenazn/goji v0.9.0/go.mod h1:7S9M489iMyHBNxwZnk9/EHS098H4/F6TATF2mIxtB1Q= go.etcd.io/bbolt v1.3.3/go.mod h1:IbVyRI1SCnLcuJnV2u8VeU0CEYM7e686BmAb1XKL+uU= go.etcd.io/bbolt v1.3.5/go.mod h1:G5EMThwa9y8QZGBClrRx5EY+Yw9kAhnjy3bSjsnlVTQ= @@ -806,8 +769,11 @@ go.opencensus.io v0.23.0/go.mod h1:XItmlyltB5F7CS4xOC1DcqMoFqwtC6OG2xF7mCv7P7E= go.opentelemetry.io/proto/otlp v0.7.0/go.mod h1:PqfVotwruBrMGOCsRd/89rSnXhoiJIqeYNgFYFoEGnI= go.step.sm/cli-utils v0.7.3 h1:IA12IaiXVCI18yOFVQuvMpyvjL8wuwUn1yO+KhAVAr0= go.step.sm/cli-utils v0.7.3/go.mod h1:RJRwbBLqzs5nrepQLAV9FuT3fVpWz66tKzLIB7Izpfk= -go.step.sm/linkedca v0.16.1 h1:CdbMV5SjnlRsgeYTXaaZmQCkYIgJq8BOzpewri57M2k= -go.step.sm/linkedca v0.16.1/go.mod h1:W59ucS4vFpuR0g4PtkGbbtXAwxbDEnNCg+ovkej1ANM= +go.step.sm/crypto v0.9.0/go.mod h1:+CYG05Mek1YDqi5WK0ERc6cOpKly2i/a5aZmU1sfGj0= +go.step.sm/crypto v0.18.0 h1:saD/tMG7uKJmUIPyOyudidVTHPnozTU02CDd+oqwKn0= +go.step.sm/crypto v0.18.0/go.mod h1:qZ+pNU1nV+THwP7TPTNCRMRr9xrRURhETTAK7U5psfw= +go.step.sm/linkedca v0.18.0 h1:uxRBd2WDvJNZ2i0nJm/QmG4lkRxWoebYKJinchX7T7o= +go.step.sm/linkedca v0.18.0/go.mod h1:qSuYlIIhvPmA2+DSSS03E2IXhbXWTLW61Xh9zDQJ3VM= go.uber.org/atomic v1.3.2/go.mod h1:gD2HeocX3+yG+ygLZcrzQJaqmWj9AIm7n08wl/qW/PE= go.uber.org/atomic v1.4.0/go.mod h1:gD2HeocX3+yG+ygLZcrzQJaqmWj9AIm7n08wl/qW/PE= go.uber.org/atomic v1.5.0/go.mod h1:sABNBOSYdrvTF6hTgEIbc7YasKWGhgEQZyfxyTvoXHQ= @@ -833,12 +799,11 @@ golang.org/x/crypto v0.0.0-20190820162420-60c769a6c586/go.mod h1:yigFU9vqHzYiE8U golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= golang.org/x/crypto v0.0.0-20200414173820-0848c9571904/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= +golang.org/x/crypto v0.0.0-20200728195943-123391ffb6de/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= golang.org/x/crypto v0.0.0-20201203163018-be400aefbc4c/go.mod h1:jdWPYTVW3xRLrWPugEBEK3UY2ZEsg3UU495nc5E+M+I= -golang.org/x/crypto v0.0.0-20210322153248-0c34fe9e7dc2/go.mod h1:T9bdIzuCu7OtxOm1hfPfRQxPLYneinmdGuTeoZ9dtd4= golang.org/x/crypto v0.0.0-20210616213533-5ff15b29337e/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= golang.org/x/crypto v0.0.0-20210711020723-a769d52b0f97/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= -golang.org/x/crypto v0.0.0-20211202192323-5770296d904e/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4= golang.org/x/crypto v0.0.0-20211215153901-e495a2d5b3d3 h1:0es+/5331RGQPcXlMfP+WrnIIS6dNnNRe0WB02W0F4M= golang.org/x/crypto v0.0.0-20211215153901-e495a2d5b3d3/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4= golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= @@ -921,9 +886,6 @@ golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v golang.org/x/net v0.0.0-20210316092652-d523dce5a7f4/go.mod h1:RBQZq4jEuRlivfhVLdyRGr576XBO4/greRjx4P4O3yc= golang.org/x/net v0.0.0-20210405180319-a5a99cb37ef4/go.mod h1:p54w0d4576C0XHj96bSt6lcn1PtDYWL6XObtHCRCNQM= golang.org/x/net v0.0.0-20210503060351-7fd8e65b6420/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= -golang.org/x/net v0.0.0-20210525063256-abc453219eb5/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= -golang.org/x/net v0.0.0-20210805182204-aaa1db679c0d/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= -golang.org/x/net v0.0.0-20211020060615-d418f374d309/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= golang.org/x/net v0.0.0-20211112202133-69e39bad7dc2/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= golang.org/x/net v0.0.0-20220127200216-cd36cc0744dd/go.mod h1:CfG3xpIq0wQ8r1q4Su4UZFWDARRcnwPjda9FqA0JpMk= golang.org/x/net v0.0.0-20220225172249-27dd8689420f/go.mod h1:CfG3xpIq0wQ8r1q4Su4UZFWDARRcnwPjda9FqA0JpMk= @@ -984,7 +946,6 @@ golang.org/x/sys v0.0.0-20190422165155-953cdadca894/go.mod h1:h1NjWce9XRLGQEsW7w golang.org/x/sys v0.0.0-20190502145724-3ef323f4f1fd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190507160741-ecd444e8653b/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190606165138-5da285871e9c/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20190606203320-7fc4e5ec1444/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190624142023-c5567b49c5d0/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190626221950-04f50cda93cb/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190726091711-fc99dfbffb4e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= @@ -996,13 +957,11 @@ golang.org/x/sys v0.0.0-20191026070338-33540a1f6037/go.mod h1:h1NjWce9XRLGQEsW7w golang.org/x/sys v0.0.0-20191204072324-ce4227a45e2e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20191220142924-d4481acd189f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20191228213918-04cbcbbfeed8/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20200106162015-b016eb3dc98e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200113162924-86b910548bc1/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200116001909-b77594299b42/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200122134326-e047566fdf82/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200202164722-d101bd2416d5/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200212091648-12a6c2dcc1e4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20200217220822-9197077df867/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200223170610-d5e6a3e2c0ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200302150141-5c8b2ff67527/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200323222414-85ca7c5b95cd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= @@ -1011,21 +970,15 @@ golang.org/x/sys v0.0.0-20200501052902-10377860bb8e/go.mod h1:h1NjWce9XRLGQEsW7w golang.org/x/sys v0.0.0-20200511232937-7e40ca221e25/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200515095857-1151b9dac4a9/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200523222454-059865788121/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20200615200032-f1bc736245b1/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20200625212154-ddb9806d33ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200803210538-64077c9b5642/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200905004654-be1d3432aa8f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200923182605-d9f96fdee20d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200930185726-fdedc70b468f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20201015000850-e3ed0017c211/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20201018230417-eeed37f84f13/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20201201145000-ef89a241ccb3/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210104204734-6f8348627aad/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210119212857-b64e53b001e4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20210124154548-22da62e12c0c/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210220050731-9a76102bfb43/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20210303074136-134d130e1a04/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210305230114-8fe3ee5dd75b/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210315160823-c6e025ad8005/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210320140829-1e4c9ba3b0c4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= @@ -1033,18 +986,14 @@ golang.org/x/sys v0.0.0-20210330210617-4fbd30eecc44/go.mod h1:h1NjWce9XRLGQEsW7w golang.org/x/sys v0.0.0-20210423082822-04245dca01da/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210510120138-977fb7262007/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20210514084401-e8d321eab015/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.0.0-20210603081109-ebe580a85c40/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20210603125802-9665404d3644/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20210616094352-59db8d763f22/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20210630005230-0f9fa26af87c/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20210806184541-e5e7981a1069/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.0.0-20210809222454-d867a43fc93e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20210823070655-63515b42dcdf/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20210908233432-aa78b53d3365/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.0.0-20211025201205-69cdffdb9359/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20211031064116-611d5d643895/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.0.0-20211103235746-7861aae1554b/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20211124211545-fe61309f8881/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20211210111614-af8b64212486/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20211216021012-1d35b9e2eb4e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= @@ -1125,7 +1074,6 @@ golang.org/x/tools v0.0.0-20200512131952-2bc93b1c0c88/go.mod h1:EkVYQZoAsY45+roY golang.org/x/tools v0.0.0-20200515010526-7d3b6ebf133d/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= golang.org/x/tools v0.0.0-20200522201501-cb1345f3a375/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= golang.org/x/tools v0.0.0-20200618134242-20370b0cb4b2/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= -golang.org/x/tools v0.0.0-20200619180055-7c47624df98f/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= golang.org/x/tools v0.0.0-20200717024301-6ddee64345a6/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA= golang.org/x/tools v0.0.0-20200729194436-6467de6f59a7/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA= golang.org/x/tools v0.0.0-20200804011535-6c149bb5ef0d/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA= @@ -1135,14 +1083,12 @@ golang.org/x/tools v0.0.0-20201110124207-079ba7bd75cd/go.mod h1:emZCQorbCU4vsT4f golang.org/x/tools v0.0.0-20201201161351-ac6f37ff4c2a/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= golang.org/x/tools v0.0.0-20201208233053-a543418bbed2/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= golang.org/x/tools v0.0.0-20210105154028-b0ab187a4818/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= -golang.org/x/tools v0.0.0-20210106214847-113979e3529a/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= golang.org/x/tools v0.1.0/go.mod h1:xkSsbof2nBLbhDlRMhhhyNLN/zl3eTqcnHD5viDpcZ0= golang.org/x/tools v0.1.1/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= golang.org/x/tools v0.1.2/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= golang.org/x/tools v0.1.3/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= golang.org/x/tools v0.1.4/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= golang.org/x/tools v0.1.5/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= -golang.org/x/tools v0.1.7/go.mod h1:LGqMHiF4EqQNHR1JncWGqT5BVaXmza+X+BDGol+dOxo= golang.org/x/xerrors v0.0.0-20190410155217-1f06c39b4373/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= golang.org/x/xerrors v0.0.0-20190513163551-3ee3066db522/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= @@ -1151,8 +1097,6 @@ golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8T golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= golang.org/x/xerrors v0.0.0-20220411194840-2f41105eb62f/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= golang.org/x/xerrors v0.0.0-20220609144429-65e65417b02f/go.mod h1:K8+ghG5WaK9qNqU5K3HdILfMLy1f3aNYFI/wnl100a8= -golang.zx2c4.com/wintun v0.0.0-20211104114900-415007cec224/go.mod h1:deeaetjYA+DHMHg+sMSMI58GrEteJUUzzw7en6TJQcI= -golang.zx2c4.com/wireguard/windows v0.5.1/go.mod h1:EApyTk/ZNrkbZjurHL1nleDYnsPpJYBO7LZEBCyDAHk= google.golang.org/api v0.3.1/go.mod h1:6wY9I6uQWHQ8EM57III9mq/AjF+i8G65rmVagqKMtkk= google.golang.org/api v0.4.0/go.mod h1:8k5glujaEP+g9n7WNsDg8QP6cUVNI86fCNMcbazEtwE= google.golang.org/api v0.7.0/go.mod h1:WtwebWUNSVBH/HAw79HIFXZNqEvBhG+Ra+ax0hx3E3M= @@ -1338,9 +1282,8 @@ google.golang.org/protobuf v1.28.0/go.mod h1:HV8QOd/L58Z+nl8r43ehVNZIU/HEI6OcFqw gopkg.in/alecthomas/kingpin.v2 v2.2.6/go.mod h1:FMv+mEhP44yOT+4EoQTLFTRgOQ1FBLkstjWtayDeSgw= gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= +gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15 h1:YR8cESwS4TdDjEe65xsg0ogRM/Nc3DYOhEAlW+xobZo= gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= -gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk= -gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c/go.mod h1:JHkPIbrfpd72SG/EVd6muEfDQjcINNoR0C8j2r3qZ4Q= gopkg.in/cheggaaa/pb.v1 v1.0.25/go.mod h1:V/YB90LKu/1FcN3WVnfiiE5oMCibMjukxqG/qStrOgw= gopkg.in/errgo.v2 v2.1.0/go.mod h1:hNsd1EY+bozCKY1Ytp96fpM3vjJbqLJn88ws8XvfDNI= gopkg.in/fsnotify.v1 v1.4.7/go.mod h1:Tz8NjZHkW78fSQdbUxIjBTcgA1z1m8ZHf0WmKUhAMys= @@ -1362,7 +1305,6 @@ gopkg.in/yaml.v2 v2.2.7/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.3.0/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY= -gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ= gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= gopkg.in/yaml.v3 v3.0.0 h1:hjy8E9ON/egN1tAYqKb61G10WtihqetD4sz2H+8nIeA= gopkg.in/yaml.v3 v3.0.0/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= diff --git a/kms/apiv1/options.go b/kms/apiv1/options.go deleted file mode 100644 index 79b07a60..00000000 --- a/kms/apiv1/options.go +++ /dev/null @@ -1,137 +0,0 @@ -package apiv1 - -import ( - "crypto" - "crypto/x509" - "strings" - - "github.com/pkg/errors" -) - -// KeyManager is the interface implemented by all the KMS. -type KeyManager interface { - GetPublicKey(req *GetPublicKeyRequest) (crypto.PublicKey, error) - CreateKey(req *CreateKeyRequest) (*CreateKeyResponse, error) - CreateSigner(req *CreateSignerRequest) (crypto.Signer, error) - Close() error -} - -// Decrypter is an interface implemented by KMSes that are used -// in operations that require decryption -type Decrypter interface { - CreateDecrypter(req *CreateDecrypterRequest) (crypto.Decrypter, error) -} - -// CertificateManager is the interface implemented by the KMS that can load and -// store x509.Certificates. -type CertificateManager interface { - LoadCertificate(req *LoadCertificateRequest) (*x509.Certificate, error) - StoreCertificate(req *StoreCertificateRequest) error -} - -// ValidateName is an interface that KeyManager can implement to validate a -// given name or URI. -type NameValidator interface { - ValidateName(s string) error -} - -// ErrNotImplemented is the type of error returned if an operation is not -// implemented. -type ErrNotImplemented struct { - Message string -} - -func (e ErrNotImplemented) Error() string { - if e.Message != "" { - return e.Message - } - return "not implemented" -} - -// ErrAlreadyExists is the type of error returned if a key already exists. This -// is currently only implmented on pkcs11. -type ErrAlreadyExists struct { - Message string -} - -func (e ErrAlreadyExists) Error() string { - if e.Message != "" { - return e.Message - } - return "key already exists" -} - -// Type represents the KMS type used. -type Type string - -const ( - // DefaultKMS is a KMS implementation using software. - DefaultKMS Type = "" - // SoftKMS is a KMS implementation using software. - SoftKMS Type = "softkms" - // CloudKMS is a KMS implementation using Google's Cloud KMS. - CloudKMS Type = "cloudkms" - // AmazonKMS is a KMS implementation using Amazon AWS KMS. - AmazonKMS Type = "awskms" - // PKCS11 is a KMS implementation using the PKCS11 standard. - PKCS11 Type = "pkcs11" - // YubiKey is a KMS implementation using a YubiKey PIV. - YubiKey Type = "yubikey" - // SSHAgentKMS is a KMS implementation using ssh-agent to access keys. - SSHAgentKMS Type = "sshagentkms" - // AzureKMS is a KMS implementation using Azure Key Vault. - AzureKMS Type = "azurekms" -) - -// Options are the KMS options. They represent the kms object in the ca.json. -type Options struct { - // The type of the KMS to use. - Type string `json:"type"` - - // Path to the credentials file used in CloudKMS and AmazonKMS. - CredentialsFile string `json:"credentialsFile,omitempty"` - - // URI is based on the PKCS #11 URI Scheme defined in - // https://tools.ietf.org/html/rfc7512 and represents the configuration used - // to connect to the KMS. - // - // Used by: pkcs11 - URI string `json:"uri,omitempty"` - - // Pin used to access the PKCS11 module. It can be defined in the URI using - // the pin-value or pin-source properties. - Pin string `json:"pin,omitempty"` - - // ManagementKey used in YubiKeys. Default management key is the hexadecimal - // string 010203040506070801020304050607080102030405060708: - // []byte{ - // 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, - // 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, - // 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, - // } - ManagementKey string `json:"managementKey,omitempty"` - - // Region to use in AmazonKMS. - Region string `json:"region,omitempty"` - - // Profile to use in AmazonKMS. - Profile string `json:"profile,omitempty"` -} - -// Validate checks the fields in Options. -func (o *Options) Validate() error { - if o == nil { - return nil - } - - switch Type(strings.ToLower(o.Type)) { - case DefaultKMS, SoftKMS: // Go crypto based kms. - case CloudKMS, AmazonKMS, AzureKMS: // Cloud based kms. - case YubiKey, PKCS11: // Hardware based kms. - case SSHAgentKMS: // Others - default: - return errors.Errorf("unsupported kms type %s", o.Type) - } - - return nil -} diff --git a/kms/apiv1/options_test.go b/kms/apiv1/options_test.go deleted file mode 100644 index 5405954f..00000000 --- a/kms/apiv1/options_test.go +++ /dev/null @@ -1,76 +0,0 @@ -package apiv1 - -import ( - "testing" -) - -func TestOptions_Validate(t *testing.T) { - tests := []struct { - name string - options *Options - wantErr bool - }{ - {"nil", nil, false}, - {"softkms", &Options{Type: "softkms"}, false}, - {"cloudkms", &Options{Type: "cloudkms"}, false}, - {"awskms", &Options{Type: "awskms"}, false}, - {"sshagentkms", &Options{Type: "sshagentkms"}, false}, - {"pkcs11", &Options{Type: "pkcs11"}, false}, - {"unsupported", &Options{Type: "unsupported"}, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - if err := tt.options.Validate(); (err != nil) != tt.wantErr { - t.Errorf("Options.Validate() error = %v, wantErr %v", err, tt.wantErr) - } - }) - } -} - -func TestErrNotImplemented_Error(t *testing.T) { - type fields struct { - msg string - } - tests := []struct { - name string - fields fields - want string - }{ - {"default", fields{}, "not implemented"}, - {"custom", fields{"custom message: not implemented"}, "custom message: not implemented"}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - e := ErrNotImplemented{ - Message: tt.fields.msg, - } - if got := e.Error(); got != tt.want { - t.Errorf("ErrNotImplemented.Error() = %v, want %v", got, tt.want) - } - }) - } -} - -func TestErrAlreadyExists_Error(t *testing.T) { - type fields struct { - msg string - } - tests := []struct { - name string - fields fields - want string - }{ - {"default", fields{}, "key already exists"}, - {"custom", fields{"custom message: key already exists"}, "custom message: key already exists"}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - e := ErrAlreadyExists{ - Message: tt.fields.msg, - } - if got := e.Error(); got != tt.want { - t.Errorf("ErrAlreadyExists.Error() = %v, want %v", got, tt.want) - } - }) - } -} diff --git a/kms/apiv1/registry.go b/kms/apiv1/registry.go deleted file mode 100644 index 5a8cf4db..00000000 --- a/kms/apiv1/registry.go +++ /dev/null @@ -1,27 +0,0 @@ -package apiv1 - -import ( - "context" - "sync" -) - -var registry = new(sync.Map) - -// KeyManagerNewFunc is the type that represents the method to initialize a new -// KeyManager. -type KeyManagerNewFunc func(ctx context.Context, opts Options) (KeyManager, error) - -// Register adds to the registry a method to create a KeyManager of type t. -func Register(t Type, fn KeyManagerNewFunc) { - registry.Store(t, fn) -} - -// LoadKeyManagerNewFunc returns the function initialize a KayManager. -func LoadKeyManagerNewFunc(t Type) (KeyManagerNewFunc, bool) { - v, ok := registry.Load(t) - if !ok { - return nil, false - } - fn, ok := v.(KeyManagerNewFunc) - return fn, ok -} diff --git a/kms/apiv1/requests.go b/kms/apiv1/requests.go deleted file mode 100644 index 762d9dd8..00000000 --- a/kms/apiv1/requests.go +++ /dev/null @@ -1,167 +0,0 @@ -package apiv1 - -import ( - "crypto" - "crypto/x509" - "fmt" -) - -// ProtectionLevel specifies on some KMS how cryptographic operations are -// performed. -type ProtectionLevel int - -const ( - // Protection level not specified. - UnspecifiedProtectionLevel ProtectionLevel = iota - // Crypto operations are performed in software. - Software - // Crypto operations are performed in a Hardware Security Module. - HSM -) - -// String returns a string representation of p. -func (p ProtectionLevel) String() string { - switch p { - case UnspecifiedProtectionLevel: - return "unspecified" - case Software: - return "software" - case HSM: - return "hsm" - default: - return fmt.Sprintf("unknown(%d)", p) - } -} - -// SignatureAlgorithm used for cryptographic signing. -type SignatureAlgorithm int - -const ( - // Not specified. - UnspecifiedSignAlgorithm SignatureAlgorithm = iota - // RSASSA-PKCS1-v1_5 key and a SHA256 digest. - SHA256WithRSA - // RSASSA-PKCS1-v1_5 key and a SHA384 digest. - SHA384WithRSA - // RSASSA-PKCS1-v1_5 key and a SHA512 digest. - SHA512WithRSA - // RSASSA-PSS key with a SHA256 digest. - SHA256WithRSAPSS - // RSASSA-PSS key with a SHA384 digest. - SHA384WithRSAPSS - // RSASSA-PSS key with a SHA512 digest. - SHA512WithRSAPSS - // ECDSA on the NIST P-256 curve with a SHA256 digest. - ECDSAWithSHA256 - // ECDSA on the NIST P-384 curve with a SHA384 digest. - ECDSAWithSHA384 - // ECDSA on the NIST P-521 curve with a SHA512 digest. - ECDSAWithSHA512 - // EdDSA on Curve25519 with a SHA512 digest. - PureEd25519 -) - -// String returns a string representation of s. -func (s SignatureAlgorithm) String() string { - switch s { - case UnspecifiedSignAlgorithm: - return "unspecified" - case SHA256WithRSA: - return "SHA256-RSA" - case SHA384WithRSA: - return "SHA384-RSA" - case SHA512WithRSA: - return "SHA512-RSA" - case SHA256WithRSAPSS: - return "SHA256-RSAPSS" - case SHA384WithRSAPSS: - return "SHA384-RSAPSS" - case SHA512WithRSAPSS: - return "SHA512-RSAPSS" - case ECDSAWithSHA256: - return "ECDSA-SHA256" - case ECDSAWithSHA384: - return "ECDSA-SHA384" - case ECDSAWithSHA512: - return "ECDSA-SHA512" - case PureEd25519: - return "Ed25519" - default: - return fmt.Sprintf("unknown(%d)", s) - } -} - -// GetPublicKeyRequest is the parameter used in the kms.GetPublicKey method. -type GetPublicKeyRequest struct { - Name string -} - -// CreateKeyRequest is the parameter used in the kms.CreateKey method. -type CreateKeyRequest struct { - // Name represents the key name or label used to identify a key. - // - // Used by: awskms, cloudkms, azurekms, pkcs11, yubikey. - Name string - - // SignatureAlgorithm represents the type of key to create. - SignatureAlgorithm SignatureAlgorithm - - // Bits is the number of bits on RSA keys. - Bits int - - // ProtectionLevel specifies how cryptographic operations are performed. - // Used by: cloudkms, azurekms. - ProtectionLevel ProtectionLevel - - // Extractable defines if the new key may be exported from the HSM under a - // wrap key. On pkcs11 sets the CKA_EXTRACTABLE bit. - // - // Used by: pkcs11 - Extractable bool -} - -// CreateKeyResponse is the response value of the kms.CreateKey method. -type CreateKeyResponse struct { - Name string - PublicKey crypto.PublicKey - PrivateKey crypto.PrivateKey - CreateSignerRequest CreateSignerRequest -} - -// CreateSignerRequest is the parameter used in the kms.CreateSigner method. -type CreateSignerRequest struct { - Signer crypto.Signer - SigningKey string - SigningKeyPEM []byte - TokenLabel string - PublicKey string - PublicKeyPEM []byte - Password []byte -} - -// CreateDecrypterRequest is the parameter used in the kms.Decrypt method. -type CreateDecrypterRequest struct { - Decrypter crypto.Decrypter - DecryptionKey string - DecryptionKeyPEM []byte - Password []byte -} - -// LoadCertificateRequest is the parameter used in the LoadCertificate method of -// a CertificateManager. -type LoadCertificateRequest struct { - Name string -} - -// StoreCertificateRequest is the parameter used in the StoreCertificate method -// of a CertificateManager. -type StoreCertificateRequest struct { - Name string - Certificate *x509.Certificate - - // Extractable defines if the new certificate may be exported from the HSM - // under a wrap key. On pkcs11 sets the CKA_EXTRACTABLE bit. - // - // Used by: pkcs11 - Extractable bool -} diff --git a/kms/apiv1/requests_test.go b/kms/apiv1/requests_test.go deleted file mode 100644 index b378e631..00000000 --- a/kms/apiv1/requests_test.go +++ /dev/null @@ -1,51 +0,0 @@ -package apiv1 - -import "testing" - -func TestProtectionLevel_String(t *testing.T) { - tests := []struct { - name string - p ProtectionLevel - want string - }{ - {"unspecified", UnspecifiedProtectionLevel, "unspecified"}, - {"software", Software, "software"}, - {"hsm", HSM, "hsm"}, - {"unknown", ProtectionLevel(100), "unknown(100)"}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - if got := tt.p.String(); got != tt.want { - t.Errorf("ProtectionLevel.String() = %v, want %v", got, tt.want) - } - }) - } -} - -func TestSignatureAlgorithm_String(t *testing.T) { - tests := []struct { - name string - s SignatureAlgorithm - want string - }{ - {"UnspecifiedSignAlgorithm", UnspecifiedSignAlgorithm, "unspecified"}, - {"SHA256WithRSA", SHA256WithRSA, "SHA256-RSA"}, - {"SHA384WithRSA", SHA384WithRSA, "SHA384-RSA"}, - {"SHA512WithRSA", SHA512WithRSA, "SHA512-RSA"}, - {"SHA256WithRSAPSS", SHA256WithRSAPSS, "SHA256-RSAPSS"}, - {"SHA384WithRSAPSS", SHA384WithRSAPSS, "SHA384-RSAPSS"}, - {"SHA512WithRSAPSS", SHA512WithRSAPSS, "SHA512-RSAPSS"}, - {"ECDSAWithSHA256", ECDSAWithSHA256, "ECDSA-SHA256"}, - {"ECDSAWithSHA384", ECDSAWithSHA384, "ECDSA-SHA384"}, - {"ECDSAWithSHA512", ECDSAWithSHA512, "ECDSA-SHA512"}, - {"PureEd25519", PureEd25519, "Ed25519"}, - {"unknown", SignatureAlgorithm(100), "unknown(100)"}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - if got := tt.s.String(); got != tt.want { - t.Errorf("SignatureAlgorithm.String() = %v, want %v", got, tt.want) - } - }) - } -} diff --git a/kms/awskms/awskms.go b/kms/awskms/awskms.go deleted file mode 100644 index 1706d188..00000000 --- a/kms/awskms/awskms.go +++ /dev/null @@ -1,267 +0,0 @@ -package awskms - -import ( - "context" - "crypto" - "net/url" - "strings" - "time" - - "github.com/aws/aws-sdk-go/aws" - "github.com/aws/aws-sdk-go/aws/request" - "github.com/aws/aws-sdk-go/aws/session" - "github.com/aws/aws-sdk-go/service/kms" - "github.com/pkg/errors" - "github.com/smallstep/certificates/kms/apiv1" - "github.com/smallstep/certificates/kms/uri" - "go.step.sm/crypto/pemutil" -) - -// Scheme is the scheme used in uris. -const Scheme = "awskms" - -// KMS implements a KMS using AWS Key Management Service. -type KMS struct { - session *session.Session - service KeyManagementClient -} - -// KeyManagementClient defines the methods on KeyManagementClient that this -// package will use. This interface will be used for unit testing. -type KeyManagementClient interface { - GetPublicKeyWithContext(ctx aws.Context, input *kms.GetPublicKeyInput, opts ...request.Option) (*kms.GetPublicKeyOutput, error) - CreateKeyWithContext(ctx aws.Context, input *kms.CreateKeyInput, opts ...request.Option) (*kms.CreateKeyOutput, error) - CreateAliasWithContext(ctx aws.Context, input *kms.CreateAliasInput, opts ...request.Option) (*kms.CreateAliasOutput, error) - SignWithContext(ctx aws.Context, input *kms.SignInput, opts ...request.Option) (*kms.SignOutput, error) -} - -// customerMasterKeySpecMapping is a mapping between the step signature algorithm, -// and bits for RSA keys, with awskms CustomerMasterKeySpec. -var customerMasterKeySpecMapping = map[apiv1.SignatureAlgorithm]interface{}{ - apiv1.UnspecifiedSignAlgorithm: kms.CustomerMasterKeySpecEccNistP256, - apiv1.SHA256WithRSA: map[int]string{ - 0: kms.CustomerMasterKeySpecRsa3072, - 2048: kms.CustomerMasterKeySpecRsa2048, - 3072: kms.CustomerMasterKeySpecRsa3072, - 4096: kms.CustomerMasterKeySpecRsa4096, - }, - apiv1.SHA512WithRSA: map[int]string{ - 0: kms.CustomerMasterKeySpecRsa4096, - 4096: kms.CustomerMasterKeySpecRsa4096, - }, - apiv1.SHA256WithRSAPSS: map[int]string{ - 0: kms.CustomerMasterKeySpecRsa3072, - 2048: kms.CustomerMasterKeySpecRsa2048, - 3072: kms.CustomerMasterKeySpecRsa3072, - 4096: kms.CustomerMasterKeySpecRsa4096, - }, - apiv1.SHA512WithRSAPSS: map[int]string{ - 0: kms.CustomerMasterKeySpecRsa4096, - 4096: kms.CustomerMasterKeySpecRsa4096, - }, - apiv1.ECDSAWithSHA256: kms.CustomerMasterKeySpecEccNistP256, - apiv1.ECDSAWithSHA384: kms.CustomerMasterKeySpecEccNistP384, - apiv1.ECDSAWithSHA512: kms.CustomerMasterKeySpecEccNistP521, -} - -// New creates a new AWSKMS. By default, sessions will be created using the -// credentials in `~/.aws/credentials`, but this can be overridden using the -// CredentialsFile option, the Region and Profile can also be configured as -// options. -// -// AWS sessions can also be configured with environment variables, see docs at -// https://docs.aws.amazon.com/sdk-for-go/api/aws/session/ for all the options. -func New(ctx context.Context, opts apiv1.Options) (*KMS, error) { - var o session.Options - - if opts.URI != "" { - u, err := uri.ParseWithScheme(Scheme, opts.URI) - if err != nil { - return nil, err - } - o.Profile = u.Get("profile") - if v := u.Get("region"); v != "" { - o.Config.Region = new(string) - *o.Config.Region = v - } - if f := u.Get("credentials-file"); f != "" { - o.SharedConfigFiles = []string{f} - } - } - - // Deprecated way to set configuration parameters. - if opts.Region != "" { - o.Config.Region = &opts.Region - } - if opts.Profile != "" { - o.Profile = opts.Profile - } - if opts.CredentialsFile != "" { - o.SharedConfigFiles = []string{opts.CredentialsFile} - } - - sess, err := session.NewSessionWithOptions(o) - if err != nil { - return nil, errors.Wrap(err, "error creating AWS session") - } - - return &KMS{ - session: sess, - service: kms.New(sess), - }, nil -} - -func init() { - apiv1.Register(apiv1.AmazonKMS, func(ctx context.Context, opts apiv1.Options) (apiv1.KeyManager, error) { - return New(ctx, opts) - }) -} - -// GetPublicKey returns a public key from KMS. -func (k *KMS) GetPublicKey(req *apiv1.GetPublicKeyRequest) (crypto.PublicKey, error) { - if req.Name == "" { - return nil, errors.New("getPublicKey 'name' cannot be empty") - } - keyID, err := parseKeyID(req.Name) - if err != nil { - return nil, err - } - - ctx, cancel := defaultContext() - defer cancel() - - resp, err := k.service.GetPublicKeyWithContext(ctx, &kms.GetPublicKeyInput{ - KeyId: &keyID, - }) - if err != nil { - return nil, errors.Wrap(err, "awskms GetPublicKeyWithContext failed") - } - - return pemutil.ParseDER(resp.PublicKey) -} - -// CreateKey generates a new key in KMS and returns the public key version -// of it. -func (k *KMS) CreateKey(req *apiv1.CreateKeyRequest) (*apiv1.CreateKeyResponse, error) { - if req.Name == "" { - return nil, errors.New("createKeyRequest 'name' cannot be empty") - } - - keySpec, err := getCustomerMasterKeySpecMapping(req.SignatureAlgorithm, req.Bits) - if err != nil { - return nil, err - } - - tag := new(kms.Tag) - tag.SetTagKey("name") - tag.SetTagValue(req.Name) - - input := &kms.CreateKeyInput{ - Description: &req.Name, - CustomerMasterKeySpec: &keySpec, - Tags: []*kms.Tag{tag}, - } - input.SetKeyUsage(kms.KeyUsageTypeSignVerify) - - ctx, cancel := defaultContext() - defer cancel() - - resp, err := k.service.CreateKeyWithContext(ctx, input) - if err != nil { - return nil, errors.Wrap(err, "awskms CreateKeyWithContext failed") - } - if err := k.createKeyAlias(*resp.KeyMetadata.KeyId, req.Name); err != nil { - return nil, err - } - - // Create uri for key - name := uri.New("awskms", url.Values{ - "key-id": []string{*resp.KeyMetadata.KeyId}, - }).String() - - publicKey, err := k.GetPublicKey(&apiv1.GetPublicKeyRequest{ - Name: name, - }) - if err != nil { - return nil, err - } - - // Names uses Amazon Resource Name - // https://docs.aws.amazon.com/general/latest/gr/aws-arns-and-namespaces.html - return &apiv1.CreateKeyResponse{ - Name: name, - PublicKey: publicKey, - CreateSignerRequest: apiv1.CreateSignerRequest{ - SigningKey: name, - }, - }, nil -} - -func (k *KMS) createKeyAlias(keyID, alias string) error { - alias = "alias/" + alias + "-" + keyID[:8] - - ctx, cancel := defaultContext() - defer cancel() - - _, err := k.service.CreateAliasWithContext(ctx, &kms.CreateAliasInput{ - AliasName: &alias, - TargetKeyId: &keyID, - }) - if err != nil { - return errors.Wrap(err, "awskms CreateAliasWithContext failed") - } - return nil -} - -// CreateSigner creates a new crypto.Signer with a previously configured key. -func (k *KMS) CreateSigner(req *apiv1.CreateSignerRequest) (crypto.Signer, error) { - if req.SigningKey == "" { - return nil, errors.New("createSigner 'signingKey' cannot be empty") - } - return NewSigner(k.service, req.SigningKey) -} - -// Close closes the connection of the KMS client. -func (k *KMS) Close() error { - return nil -} - -func defaultContext() (context.Context, context.CancelFunc) { - return context.WithTimeout(context.Background(), 15*time.Second) -} - -// parseKeyID extracts the key-id from an uri. -func parseKeyID(name string) (string, error) { - name = strings.ToLower(name) - if strings.HasPrefix(name, "awskms:") || strings.HasPrefix(name, "aws:") { - u, err := uri.Parse(name) - if err != nil { - return "", err - } - if k := u.Get("key-id"); k != "" { - return k, nil - } - return "", errors.Errorf("failed to get key-id from %s", name) - } - return name, nil -} - -func getCustomerMasterKeySpecMapping(alg apiv1.SignatureAlgorithm, bits int) (string, error) { - v, ok := customerMasterKeySpecMapping[alg] - if !ok { - return "", errors.Errorf("awskms does not support signature algorithm '%s'", alg) - } - - switch v := v.(type) { - case string: - return v, nil - case map[int]string: - s, ok := v[bits] - if !ok { - return "", errors.Errorf("awskms does not support signature algorithm '%s' with '%d' bits", alg, bits) - } - return s, nil - default: - return "", errors.Errorf("unexpected error: this should not happen") - } -} diff --git a/kms/awskms/awskms_test.go b/kms/awskms/awskms_test.go deleted file mode 100644 index 3c99fc4c..00000000 --- a/kms/awskms/awskms_test.go +++ /dev/null @@ -1,364 +0,0 @@ -package awskms - -import ( - "context" - "crypto" - "fmt" - "os" - "path/filepath" - "reflect" - "testing" - - "github.com/aws/aws-sdk-go/aws" - "github.com/aws/aws-sdk-go/aws/request" - "github.com/aws/aws-sdk-go/aws/session" - "github.com/aws/aws-sdk-go/service/kms" - "github.com/smallstep/certificates/kms/apiv1" - "go.step.sm/crypto/pemutil" -) - -func TestNew(t *testing.T) { - ctx := context.Background() - - sess, err := session.NewSessionWithOptions(session.Options{}) - if err != nil { - t.Fatal(err) - } - expected := &KMS{ - session: sess, - service: kms.New(sess), - } - - // This will force an error in the session creation. - // It does not fail with missing credentials. - forceError := func(t *testing.T) { - key := "AWS_CA_BUNDLE" - value := os.Getenv(key) - os.Setenv(key, filepath.Join(os.TempDir(), "missing-ca.crt")) - t.Cleanup(func() { - if value == "" { - os.Unsetenv(key) - } else { - os.Setenv(key, value) - } - }) - } - - type args struct { - ctx context.Context - opts apiv1.Options - } - tests := []struct { - name string - args args - want *KMS - wantErr bool - }{ - {"ok", args{ctx, apiv1.Options{}}, expected, false}, - {"ok with options", args{ctx, apiv1.Options{ - Region: "us-east-1", - Profile: "smallstep", - CredentialsFile: "~/aws/credentials", - }}, expected, false}, - {"ok with uri", args{ctx, apiv1.Options{ - URI: "awskms:region=us-east-1;profile=smallstep;credentials-file=/var/run/aws/credentials", - }}, expected, false}, - {"fail", args{ctx, apiv1.Options{}}, nil, true}, - {"fail uri", args{ctx, apiv1.Options{ - URI: "pkcs11:region=us-east-1;profile=smallstep;credentials-file=/var/run/aws/credentials", - }}, nil, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - // Force an error in the session loading - if tt.wantErr { - forceError(t) - } - - got, err := New(tt.args.ctx, tt.args.opts) - if (err != nil) != tt.wantErr { - t.Errorf("New() error = %v, wantErr %v", err, tt.wantErr) - return - } - if err != nil { - if !reflect.DeepEqual(got, tt.want) { - t.Errorf("New() = %#v, want %#v", got, tt.want) - } - } else { - if got.session == nil || got.service == nil { - t.Errorf("New() = %#v, want %#v", got, tt.want) - } - } - }) - } -} - -func TestKMS_GetPublicKey(t *testing.T) { - okClient := getOKClient() - key, err := pemutil.ParseKey([]byte(publicKey)) - if err != nil { - t.Fatal(err) - } - - type fields struct { - session *session.Session - service KeyManagementClient - } - type args struct { - req *apiv1.GetPublicKeyRequest - } - tests := []struct { - name string - fields fields - args args - want crypto.PublicKey - wantErr bool - }{ - {"ok", fields{nil, okClient}, args{&apiv1.GetPublicKeyRequest{ - Name: "awskms:key-id=be468355-ca7a-40d9-a28b-8ae1c4c7f936", - }}, key, false}, - {"fail empty", fields{nil, okClient}, args{&apiv1.GetPublicKeyRequest{}}, nil, true}, - {"fail name", fields{nil, okClient}, args{&apiv1.GetPublicKeyRequest{ - Name: "awskms:key-id=", - }}, nil, true}, - {"fail getPublicKey", fields{nil, &MockClient{ - getPublicKeyWithContext: func(ctx aws.Context, input *kms.GetPublicKeyInput, opts ...request.Option) (*kms.GetPublicKeyOutput, error) { - return nil, fmt.Errorf("an error") - }, - }}, args{&apiv1.GetPublicKeyRequest{ - Name: "awskms:key-id=be468355-ca7a-40d9-a28b-8ae1c4c7f936", - }}, nil, true}, - {"fail not der", fields{nil, &MockClient{ - getPublicKeyWithContext: func(ctx aws.Context, input *kms.GetPublicKeyInput, opts ...request.Option) (*kms.GetPublicKeyOutput, error) { - return &kms.GetPublicKeyOutput{ - KeyId: input.KeyId, - PublicKey: []byte(publicKey), - }, nil - }, - }}, args{&apiv1.GetPublicKeyRequest{ - Name: "awskms:key-id=be468355-ca7a-40d9-a28b-8ae1c4c7f936", - }}, nil, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - k := &KMS{ - session: tt.fields.session, - service: tt.fields.service, - } - got, err := k.GetPublicKey(tt.args.req) - if (err != nil) != tt.wantErr { - t.Errorf("KMS.GetPublicKey() error = %v, wantErr %v", err, tt.wantErr) - return - } - if !reflect.DeepEqual(got, tt.want) { - t.Errorf("KMS.GetPublicKey() = %v, want %v", got, tt.want) - } - }) - } -} - -func TestKMS_CreateKey(t *testing.T) { - okClient := getOKClient() - key, err := pemutil.ParseKey([]byte(publicKey)) - if err != nil { - t.Fatal(err) - } - - type fields struct { - session *session.Session - service KeyManagementClient - } - type args struct { - req *apiv1.CreateKeyRequest - } - tests := []struct { - name string - fields fields - args args - want *apiv1.CreateKeyResponse - wantErr bool - }{ - {"ok", fields{nil, okClient}, args{&apiv1.CreateKeyRequest{ - Name: "root", - SignatureAlgorithm: apiv1.ECDSAWithSHA256, - }}, &apiv1.CreateKeyResponse{ - Name: "awskms:key-id=be468355-ca7a-40d9-a28b-8ae1c4c7f936", - PublicKey: key, - CreateSignerRequest: apiv1.CreateSignerRequest{ - SigningKey: "awskms:key-id=be468355-ca7a-40d9-a28b-8ae1c4c7f936", - }, - }, false}, - {"ok rsa", fields{nil, okClient}, args{&apiv1.CreateKeyRequest{ - Name: "root", - SignatureAlgorithm: apiv1.SHA256WithRSA, - Bits: 2048, - }}, &apiv1.CreateKeyResponse{ - Name: "awskms:key-id=be468355-ca7a-40d9-a28b-8ae1c4c7f936", - PublicKey: key, - CreateSignerRequest: apiv1.CreateSignerRequest{ - SigningKey: "awskms:key-id=be468355-ca7a-40d9-a28b-8ae1c4c7f936", - }, - }, false}, - {"fail empty", fields{nil, okClient}, args{&apiv1.CreateKeyRequest{}}, nil, true}, - {"fail unsupported alg", fields{nil, okClient}, args{&apiv1.CreateKeyRequest{ - Name: "root", - SignatureAlgorithm: apiv1.PureEd25519, - }}, nil, true}, - {"fail unsupported bits", fields{nil, okClient}, args{&apiv1.CreateKeyRequest{ - Name: "root", - SignatureAlgorithm: apiv1.SHA256WithRSA, - Bits: 1234, - }}, nil, true}, - {"fail createKey", fields{nil, &MockClient{ - createKeyWithContext: func(ctx aws.Context, input *kms.CreateKeyInput, opts ...request.Option) (*kms.CreateKeyOutput, error) { - return nil, fmt.Errorf("an error") - }, - createAliasWithContext: okClient.createAliasWithContext, - getPublicKeyWithContext: okClient.getPublicKeyWithContext, - }}, args{&apiv1.CreateKeyRequest{ - Name: "root", - SignatureAlgorithm: apiv1.ECDSAWithSHA256, - }}, nil, true}, - {"fail createAlias", fields{nil, &MockClient{ - createKeyWithContext: okClient.createKeyWithContext, - createAliasWithContext: func(ctx aws.Context, input *kms.CreateAliasInput, opts ...request.Option) (*kms.CreateAliasOutput, error) { - return nil, fmt.Errorf("an error") - }, - getPublicKeyWithContext: okClient.getPublicKeyWithContext, - }}, args{&apiv1.CreateKeyRequest{ - Name: "root", - SignatureAlgorithm: apiv1.ECDSAWithSHA256, - }}, nil, true}, - {"fail getPublicKey", fields{nil, &MockClient{ - createKeyWithContext: okClient.createKeyWithContext, - createAliasWithContext: okClient.createAliasWithContext, - getPublicKeyWithContext: func(ctx aws.Context, input *kms.GetPublicKeyInput, opts ...request.Option) (*kms.GetPublicKeyOutput, error) { - return nil, fmt.Errorf("an error") - }, - }}, args{&apiv1.CreateKeyRequest{ - Name: "root", - SignatureAlgorithm: apiv1.ECDSAWithSHA256, - }}, nil, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - k := &KMS{ - session: tt.fields.session, - service: tt.fields.service, - } - got, err := k.CreateKey(tt.args.req) - if (err != nil) != tt.wantErr { - t.Errorf("KMS.CreateKey() error = %v, wantErr %v", err, tt.wantErr) - return - } - if !reflect.DeepEqual(got, tt.want) { - t.Errorf("KMS.CreateKey() = %v, want %v", got, tt.want) - } - }) - } -} - -func TestKMS_CreateSigner(t *testing.T) { - client := getOKClient() - key, err := pemutil.ParseKey([]byte(publicKey)) - if err != nil { - t.Fatal(err) - } - - type fields struct { - session *session.Session - service KeyManagementClient - } - type args struct { - req *apiv1.CreateSignerRequest - } - tests := []struct { - name string - fields fields - args args - want crypto.Signer - wantErr bool - }{ - {"ok", fields{nil, client}, args{&apiv1.CreateSignerRequest{ - SigningKey: "awskms:key-id=be468355-ca7a-40d9-a28b-8ae1c4c7f936", - }}, &Signer{ - service: client, - keyID: "be468355-ca7a-40d9-a28b-8ae1c4c7f936", - publicKey: key, - }, false}, - {"fail empty", fields{nil, client}, args{&apiv1.CreateSignerRequest{}}, nil, true}, - {"fail preload", fields{nil, client}, args{&apiv1.CreateSignerRequest{}}, nil, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - k := &KMS{ - session: tt.fields.session, - service: tt.fields.service, - } - got, err := k.CreateSigner(tt.args.req) - if (err != nil) != tt.wantErr { - t.Errorf("KMS.CreateSigner() error = %v, wantErr %v", err, tt.wantErr) - return - } - if !reflect.DeepEqual(got, tt.want) { - t.Errorf("KMS.CreateSigner() = %v, want %v", got, tt.want) - } - }) - } -} - -func TestKMS_Close(t *testing.T) { - type fields struct { - session *session.Session - service KeyManagementClient - } - tests := []struct { - name string - fields fields - wantErr bool - }{ - {"ok", fields{nil, getOKClient()}, false}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - k := &KMS{ - session: tt.fields.session, - service: tt.fields.service, - } - if err := k.Close(); (err != nil) != tt.wantErr { - t.Errorf("KMS.Close() error = %v, wantErr %v", err, tt.wantErr) - } - }) - } -} - -func Test_parseKeyID(t *testing.T) { - type args struct { - name string - } - tests := []struct { - name string - args args - want string - wantErr bool - }{ - {"ok uri", args{"awskms:key-id=be468355-ca7a-40d9-a28b-8ae1c4c7f936"}, "be468355-ca7a-40d9-a28b-8ae1c4c7f936", false}, - {"ok key id", args{"be468355-ca7a-40d9-a28b-8ae1c4c7f936"}, "be468355-ca7a-40d9-a28b-8ae1c4c7f936", false}, - {"ok arn", args{"arn:aws:kms:us-east-1:123456789:key/be468355-ca7a-40d9-a28b-8ae1c4c7f936"}, "arn:aws:kms:us-east-1:123456789:key/be468355-ca7a-40d9-a28b-8ae1c4c7f936", false}, - {"fail parse", args{"awskms:key-id=%ZZ"}, "", true}, - {"fail empty key", args{"awskms:key-id="}, "", true}, - {"fail missing", args{"awskms:foo=bar"}, "", true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - got, err := parseKeyID(tt.args.name) - if (err != nil) != tt.wantErr { - t.Errorf("parseKeyID() error = %v, wantErr %v", err, tt.wantErr) - return - } - if got != tt.want { - t.Errorf("parseKeyID() = %v, want %v", got, tt.want) - } - }) - } -} diff --git a/kms/awskms/mock_test.go b/kms/awskms/mock_test.go deleted file mode 100644 index 5a7d5bd4..00000000 --- a/kms/awskms/mock_test.go +++ /dev/null @@ -1,72 +0,0 @@ -package awskms - -import ( - "encoding/pem" - - "github.com/aws/aws-sdk-go/aws" - "github.com/aws/aws-sdk-go/aws/request" - "github.com/aws/aws-sdk-go/service/kms" -) - -type MockClient struct { - getPublicKeyWithContext func(ctx aws.Context, input *kms.GetPublicKeyInput, opts ...request.Option) (*kms.GetPublicKeyOutput, error) - createKeyWithContext func(ctx aws.Context, input *kms.CreateKeyInput, opts ...request.Option) (*kms.CreateKeyOutput, error) - createAliasWithContext func(ctx aws.Context, input *kms.CreateAliasInput, opts ...request.Option) (*kms.CreateAliasOutput, error) - signWithContext func(ctx aws.Context, input *kms.SignInput, opts ...request.Option) (*kms.SignOutput, error) -} - -func (m *MockClient) GetPublicKeyWithContext(ctx aws.Context, input *kms.GetPublicKeyInput, opts ...request.Option) (*kms.GetPublicKeyOutput, error) { - return m.getPublicKeyWithContext(ctx, input, opts...) -} - -func (m *MockClient) CreateKeyWithContext(ctx aws.Context, input *kms.CreateKeyInput, opts ...request.Option) (*kms.CreateKeyOutput, error) { - return m.createKeyWithContext(ctx, input, opts...) -} - -func (m *MockClient) CreateAliasWithContext(ctx aws.Context, input *kms.CreateAliasInput, opts ...request.Option) (*kms.CreateAliasOutput, error) { - return m.createAliasWithContext(ctx, input, opts...) -} - -func (m *MockClient) SignWithContext(ctx aws.Context, input *kms.SignInput, opts ...request.Option) (*kms.SignOutput, error) { - return m.signWithContext(ctx, input, opts...) -} - -const ( - publicKey = `-----BEGIN PUBLIC KEY----- -MFkwEwYHKoZIzj0CAQYIKoZIzj0DAQcDQgAE8XWlIWkOThxNjGbZLYUgRHmsvCrW -KF+HLktPfPTIK3lGd1k4849WQs59XIN+LXZQ6b2eRBEBKAHEyQus8UU7gw== ------END PUBLIC KEY-----` - keyID = "be468355-ca7a-40d9-a28b-8ae1c4c7f936" -) - -var signature = []byte{ - 0xe3, 0xb0, 0xc4, 0x42, 0x98, 0xfc, 0x1c, 0x14, 0x9a, 0xfb, 0xf4, 0xc8, 0x99, 0x6f, 0xb9, 0x24, - 0x27, 0xae, 0x41, 0xe4, 0x64, 0x9b, 0x93, 0x4c, 0xa4, 0x95, 0x99, 0x1b, 0x78, 0x52, 0xb8, 0x55, -} - -func getOKClient() *MockClient { - return &MockClient{ - getPublicKeyWithContext: func(ctx aws.Context, input *kms.GetPublicKeyInput, opts ...request.Option) (*kms.GetPublicKeyOutput, error) { - block, _ := pem.Decode([]byte(publicKey)) - return &kms.GetPublicKeyOutput{ - KeyId: input.KeyId, - PublicKey: block.Bytes, - }, nil - }, - createKeyWithContext: func(ctx aws.Context, input *kms.CreateKeyInput, opts ...request.Option) (*kms.CreateKeyOutput, error) { - md := new(kms.KeyMetadata) - md.SetKeyId(keyID) - return &kms.CreateKeyOutput{ - KeyMetadata: md, - }, nil - }, - createAliasWithContext: func(ctx aws.Context, input *kms.CreateAliasInput, opts ...request.Option) (*kms.CreateAliasOutput, error) { - return &kms.CreateAliasOutput{}, nil - }, - signWithContext: func(ctx aws.Context, input *kms.SignInput, opts ...request.Option) (*kms.SignOutput, error) { - return &kms.SignOutput{ - Signature: signature, - }, nil - }, - } -} diff --git a/kms/awskms/signer.go b/kms/awskms/signer.go deleted file mode 100644 index 0eec10c3..00000000 --- a/kms/awskms/signer.go +++ /dev/null @@ -1,122 +0,0 @@ -package awskms - -import ( - "crypto" - "crypto/ecdsa" - "crypto/rsa" - "io" - - "github.com/aws/aws-sdk-go/service/kms" - "github.com/pkg/errors" - "go.step.sm/crypto/pemutil" -) - -// Signer implements a crypto.Signer using the AWS KMS. -type Signer struct { - service KeyManagementClient - keyID string - publicKey crypto.PublicKey -} - -// NewSigner creates a new signer using a key in the AWS KMS. -func NewSigner(svc KeyManagementClient, signingKey string) (*Signer, error) { - keyID, err := parseKeyID(signingKey) - if err != nil { - return nil, err - } - - // Make sure that the key exists. - signer := &Signer{ - service: svc, - keyID: keyID, - } - if err := signer.preloadKey(keyID); err != nil { - return nil, err - } - - return signer, nil -} - -func (s *Signer) preloadKey(keyID string) error { - ctx, cancel := defaultContext() - defer cancel() - - resp, err := s.service.GetPublicKeyWithContext(ctx, &kms.GetPublicKeyInput{ - KeyId: &keyID, - }) - if err != nil { - return errors.Wrap(err, "awskms GetPublicKeyWithContext failed") - } - - s.publicKey, err = pemutil.ParseDER(resp.PublicKey) - return err -} - -// Public returns the public key of this signer or an error. -func (s *Signer) Public() crypto.PublicKey { - return s.publicKey -} - -// Sign signs digest with the private key stored in the AWS KMS. -func (s *Signer) Sign(rand io.Reader, digest []byte, opts crypto.SignerOpts) ([]byte, error) { - alg, err := getSigningAlgorithm(s.Public(), opts) - if err != nil { - return nil, err - } - - req := &kms.SignInput{ - KeyId: &s.keyID, - SigningAlgorithm: &alg, - Message: digest, - } - req.SetMessageType("DIGEST") - - ctx, cancel := defaultContext() - defer cancel() - - resp, err := s.service.SignWithContext(ctx, req) - if err != nil { - return nil, errors.Wrap(err, "awsKMS SignWithContext failed") - } - - return resp.Signature, nil -} - -func getSigningAlgorithm(key crypto.PublicKey, opts crypto.SignerOpts) (string, error) { - switch key.(type) { - case *rsa.PublicKey: - _, isPSS := opts.(*rsa.PSSOptions) - switch h := opts.HashFunc(); h { - case crypto.SHA256: - if isPSS { - return kms.SigningAlgorithmSpecRsassaPssSha256, nil - } - return kms.SigningAlgorithmSpecRsassaPkcs1V15Sha256, nil - case crypto.SHA384: - if isPSS { - return kms.SigningAlgorithmSpecRsassaPssSha384, nil - } - return kms.SigningAlgorithmSpecRsassaPkcs1V15Sha384, nil - case crypto.SHA512: - if isPSS { - return kms.SigningAlgorithmSpecRsassaPssSha512, nil - } - return kms.SigningAlgorithmSpecRsassaPkcs1V15Sha512, nil - default: - return "", errors.Errorf("unsupported hash function %v", h) - } - case *ecdsa.PublicKey: - switch h := opts.HashFunc(); h { - case crypto.SHA256: - return kms.SigningAlgorithmSpecEcdsaSha256, nil - case crypto.SHA384: - return kms.SigningAlgorithmSpecEcdsaSha384, nil - case crypto.SHA512: - return kms.SigningAlgorithmSpecEcdsaSha512, nil - default: - return "", errors.Errorf("unsupported hash function %v", h) - } - default: - return "", errors.Errorf("unsupported key type %T", key) - } -} diff --git a/kms/awskms/signer_test.go b/kms/awskms/signer_test.go deleted file mode 100644 index 9694c62a..00000000 --- a/kms/awskms/signer_test.go +++ /dev/null @@ -1,191 +0,0 @@ -package awskms - -import ( - "crypto" - "crypto/ecdsa" - "crypto/rand" - "crypto/rsa" - "fmt" - "io" - "reflect" - "testing" - - "github.com/aws/aws-sdk-go/aws" - "github.com/aws/aws-sdk-go/aws/request" - "github.com/aws/aws-sdk-go/service/kms" - "go.step.sm/crypto/pemutil" -) - -func TestNewSigner(t *testing.T) { - okClient := getOKClient() - key, err := pemutil.ParseKey([]byte(publicKey)) - if err != nil { - t.Fatal(err) - } - - type args struct { - svc KeyManagementClient - signingKey string - } - tests := []struct { - name string - args args - want *Signer - wantErr bool - }{ - {"ok", args{okClient, "awskms:key-id=be468355-ca7a-40d9-a28b-8ae1c4c7f936"}, &Signer{ - service: okClient, - keyID: "be468355-ca7a-40d9-a28b-8ae1c4c7f936", - publicKey: key, - }, false}, - {"fail parse", args{okClient, "awskms:key-id="}, nil, true}, - {"fail preload", args{&MockClient{ - getPublicKeyWithContext: func(ctx aws.Context, input *kms.GetPublicKeyInput, opts ...request.Option) (*kms.GetPublicKeyOutput, error) { - return nil, fmt.Errorf("an error") - }, - }, "awskms:key-id=be468355-ca7a-40d9-a28b-8ae1c4c7f936"}, nil, true}, - {"fail preload not der", args{&MockClient{ - getPublicKeyWithContext: func(ctx aws.Context, input *kms.GetPublicKeyInput, opts ...request.Option) (*kms.GetPublicKeyOutput, error) { - return &kms.GetPublicKeyOutput{ - KeyId: input.KeyId, - PublicKey: []byte(publicKey), - }, nil - }, - }, "awskms:key-id=be468355-ca7a-40d9-a28b-8ae1c4c7f936"}, nil, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - got, err := NewSigner(tt.args.svc, tt.args.signingKey) - if (err != nil) != tt.wantErr { - t.Errorf("NewSigner() error = %v, wantErr %v", err, tt.wantErr) - return - } - if !reflect.DeepEqual(got, tt.want) { - t.Errorf("NewSigner() = %v, want %v", got, tt.want) - } - }) - } -} - -func TestSigner_Public(t *testing.T) { - okClient := getOKClient() - key, err := pemutil.ParseKey([]byte(publicKey)) - if err != nil { - t.Fatal(err) - } - - type fields struct { - service KeyManagementClient - keyID string - publicKey crypto.PublicKey - } - tests := []struct { - name string - fields fields - want crypto.PublicKey - }{ - {"ok", fields{okClient, "be468355-ca7a-40d9-a28b-8ae1c4c7f936", key}, key}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - s := &Signer{ - service: tt.fields.service, - keyID: tt.fields.keyID, - publicKey: tt.fields.publicKey, - } - if got := s.Public(); !reflect.DeepEqual(got, tt.want) { - t.Errorf("Signer.Public() = %v, want %v", got, tt.want) - } - }) - } -} - -func TestSigner_Sign(t *testing.T) { - okClient := getOKClient() - key, err := pemutil.ParseKey([]byte(publicKey)) - if err != nil { - t.Fatal(err) - } - - type fields struct { - service KeyManagementClient - keyID string - publicKey crypto.PublicKey - } - type args struct { - rand io.Reader - digest []byte - opts crypto.SignerOpts - } - tests := []struct { - name string - fields fields - args args - want []byte - wantErr bool - }{ - {"ok", fields{okClient, "be468355-ca7a-40d9-a28b-8ae1c4c7f936", key}, args{rand.Reader, []byte("digest"), crypto.SHA256}, signature, false}, - {"fail alg", fields{okClient, "be468355-ca7a-40d9-a28b-8ae1c4c7f936", key}, args{rand.Reader, []byte("digest"), crypto.MD5}, nil, true}, - {"fail key", fields{okClient, "be468355-ca7a-40d9-a28b-8ae1c4c7f936", []byte("key")}, args{rand.Reader, []byte("digest"), crypto.SHA256}, nil, true}, - {"fail sign", fields{&MockClient{ - signWithContext: func(ctx aws.Context, input *kms.SignInput, opts ...request.Option) (*kms.SignOutput, error) { - return nil, fmt.Errorf("an error") - }, - }, "be468355-ca7a-40d9-a28b-8ae1c4c7f936", key}, args{rand.Reader, []byte("digest"), crypto.SHA256}, nil, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - s := &Signer{ - service: tt.fields.service, - keyID: tt.fields.keyID, - publicKey: tt.fields.publicKey, - } - got, err := s.Sign(tt.args.rand, tt.args.digest, tt.args.opts) - if (err != nil) != tt.wantErr { - t.Errorf("Signer.Sign() error = %v, wantErr %v", err, tt.wantErr) - return - } - if !reflect.DeepEqual(got, tt.want) { - t.Errorf("Signer.Sign() = %v, want %v", got, tt.want) - } - }) - } -} - -func Test_getSigningAlgorithm(t *testing.T) { - type args struct { - key crypto.PublicKey - opts crypto.SignerOpts - } - tests := []struct { - name string - args args - want string - wantErr bool - }{ - {"rsa+sha256", args{&rsa.PublicKey{}, crypto.SHA256}, "RSASSA_PKCS1_V1_5_SHA_256", false}, - {"rsa+sha384", args{&rsa.PublicKey{}, crypto.SHA384}, "RSASSA_PKCS1_V1_5_SHA_384", false}, - {"rsa+sha512", args{&rsa.PublicKey{}, crypto.SHA512}, "RSASSA_PKCS1_V1_5_SHA_512", false}, - {"pssrsa+sha256", args{&rsa.PublicKey{}, &rsa.PSSOptions{Hash: crypto.SHA256.HashFunc()}}, "RSASSA_PSS_SHA_256", false}, - {"pssrsa+sha384", args{&rsa.PublicKey{}, &rsa.PSSOptions{Hash: crypto.SHA384.HashFunc()}}, "RSASSA_PSS_SHA_384", false}, - {"pssrsa+sha512", args{&rsa.PublicKey{}, &rsa.PSSOptions{Hash: crypto.SHA512.HashFunc()}}, "RSASSA_PSS_SHA_512", false}, - {"P256", args{&ecdsa.PublicKey{}, crypto.SHA256}, "ECDSA_SHA_256", false}, - {"P384", args{&ecdsa.PublicKey{}, crypto.SHA384}, "ECDSA_SHA_384", false}, - {"P521", args{&ecdsa.PublicKey{}, crypto.SHA512}, "ECDSA_SHA_512", false}, - {"fail type", args{[]byte("key"), crypto.SHA256}, "", true}, - {"fail rsa alg", args{&rsa.PublicKey{}, crypto.MD5}, "", true}, - {"fail ecdsa alg", args{&ecdsa.PublicKey{}, crypto.MD5}, "", true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - got, err := getSigningAlgorithm(tt.args.key, tt.args.opts) - if (err != nil) != tt.wantErr { - t.Errorf("getSigningAlgorithm() error = %v, wantErr %v", err, tt.wantErr) - return - } - if got != tt.want { - t.Errorf("getSigningAlgorithm() = %v, want %v", got, tt.want) - } - }) - } -} diff --git a/kms/azurekms/internal/mock/key_vault_client.go b/kms/azurekms/internal/mock/key_vault_client.go deleted file mode 100644 index 37858854..00000000 --- a/kms/azurekms/internal/mock/key_vault_client.go +++ /dev/null @@ -1,81 +0,0 @@ -// Code generated by MockGen. DO NOT EDIT. -// Source: github.com/smallstep/certificates/kms/azurekms (interfaces: KeyVaultClient) - -// Package mock is a generated GoMock package. -package mock - -import ( - context "context" - reflect "reflect" - - keyvault "github.com/Azure/azure-sdk-for-go/services/keyvault/v7.1/keyvault" - gomock "github.com/golang/mock/gomock" -) - -// KeyVaultClient is a mock of KeyVaultClient interface -type KeyVaultClient struct { - ctrl *gomock.Controller - recorder *KeyVaultClientMockRecorder -} - -// KeyVaultClientMockRecorder is the mock recorder for KeyVaultClient -type KeyVaultClientMockRecorder struct { - mock *KeyVaultClient -} - -// NewKeyVaultClient creates a new mock instance -func NewKeyVaultClient(ctrl *gomock.Controller) *KeyVaultClient { - mock := &KeyVaultClient{ctrl: ctrl} - mock.recorder = &KeyVaultClientMockRecorder{mock} - return mock -} - -// EXPECT returns an object that allows the caller to indicate expected use -func (m *KeyVaultClient) EXPECT() *KeyVaultClientMockRecorder { - return m.recorder -} - -// CreateKey mocks base method -func (m *KeyVaultClient) CreateKey(arg0 context.Context, arg1, arg2 string, arg3 keyvault.KeyCreateParameters) (keyvault.KeyBundle, error) { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "CreateKey", arg0, arg1, arg2, arg3) - ret0, _ := ret[0].(keyvault.KeyBundle) - ret1, _ := ret[1].(error) - return ret0, ret1 -} - -// CreateKey indicates an expected call of CreateKey -func (mr *KeyVaultClientMockRecorder) CreateKey(arg0, arg1, arg2, arg3 interface{}) *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "CreateKey", reflect.TypeOf((*KeyVaultClient)(nil).CreateKey), arg0, arg1, arg2, arg3) -} - -// GetKey mocks base method -func (m *KeyVaultClient) GetKey(arg0 context.Context, arg1, arg2, arg3 string) (keyvault.KeyBundle, error) { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "GetKey", arg0, arg1, arg2, arg3) - ret0, _ := ret[0].(keyvault.KeyBundle) - ret1, _ := ret[1].(error) - return ret0, ret1 -} - -// GetKey indicates an expected call of GetKey -func (mr *KeyVaultClientMockRecorder) GetKey(arg0, arg1, arg2, arg3 interface{}) *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "GetKey", reflect.TypeOf((*KeyVaultClient)(nil).GetKey), arg0, arg1, arg2, arg3) -} - -// Sign mocks base method -func (m *KeyVaultClient) Sign(arg0 context.Context, arg1, arg2, arg3 string, arg4 keyvault.KeySignParameters) (keyvault.KeyOperationResult, error) { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "Sign", arg0, arg1, arg2, arg3, arg4) - ret0, _ := ret[0].(keyvault.KeyOperationResult) - ret1, _ := ret[1].(error) - return ret0, ret1 -} - -// Sign indicates an expected call of Sign -func (mr *KeyVaultClientMockRecorder) Sign(arg0, arg1, arg2, arg3, arg4 interface{}) *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "Sign", reflect.TypeOf((*KeyVaultClient)(nil).Sign), arg0, arg1, arg2, arg3, arg4) -} diff --git a/kms/azurekms/key_vault.go b/kms/azurekms/key_vault.go deleted file mode 100644 index 34d9c3f1..00000000 --- a/kms/azurekms/key_vault.go +++ /dev/null @@ -1,342 +0,0 @@ -package azurekms - -import ( - "context" - "crypto" - "regexp" - "time" - - "github.com/Azure/azure-sdk-for-go/services/keyvault/v7.1/keyvault" - "github.com/Azure/go-autorest/autorest/azure" - "github.com/Azure/go-autorest/autorest/azure/auth" - "github.com/Azure/go-autorest/autorest/date" - "github.com/pkg/errors" - "github.com/smallstep/certificates/kms/apiv1" - "github.com/smallstep/certificates/kms/uri" -) - -func init() { - apiv1.Register(apiv1.AzureKMS, func(ctx context.Context, opts apiv1.Options) (apiv1.KeyManager, error) { - return New(ctx, opts) - }) -} - -// Scheme is the scheme used for the Azure Key Vault uris. -const Scheme = "azurekms" - -// keyIDRegexp is the regular expression that Key Vault uses on the kid. We can -// extract the vault, name and version of the key. -var keyIDRegexp = regexp.MustCompile(`^https://([0-9a-zA-Z-]+)\.vault\.azure\.net/keys/([0-9a-zA-Z-]+)/([0-9a-zA-Z-]+)$`) - -var ( - valueTrue = true - value2048 int32 = 2048 - value3072 int32 = 3072 - value4096 int32 = 4096 -) - -var now = func() time.Time { - return time.Now().UTC() -} - -type keyType struct { - Kty keyvault.JSONWebKeyType - Curve keyvault.JSONWebKeyCurveName -} - -func (k keyType) KeyType(pl apiv1.ProtectionLevel) keyvault.JSONWebKeyType { - switch k.Kty { - case keyvault.EC: - if pl == apiv1.HSM { - return keyvault.ECHSM - } - return k.Kty - case keyvault.RSA: - if pl == apiv1.HSM { - return keyvault.RSAHSM - } - return k.Kty - default: - return "" - } -} - -var signatureAlgorithmMapping = map[apiv1.SignatureAlgorithm]keyType{ - apiv1.UnspecifiedSignAlgorithm: { - Kty: keyvault.EC, - Curve: keyvault.P256, - }, - apiv1.SHA256WithRSA: { - Kty: keyvault.RSA, - }, - apiv1.SHA384WithRSA: { - Kty: keyvault.RSA, - }, - apiv1.SHA512WithRSA: { - Kty: keyvault.RSA, - }, - apiv1.SHA256WithRSAPSS: { - Kty: keyvault.RSA, - }, - apiv1.SHA384WithRSAPSS: { - Kty: keyvault.RSA, - }, - apiv1.SHA512WithRSAPSS: { - Kty: keyvault.RSA, - }, - apiv1.ECDSAWithSHA256: { - Kty: keyvault.EC, - Curve: keyvault.P256, - }, - apiv1.ECDSAWithSHA384: { - Kty: keyvault.EC, - Curve: keyvault.P384, - }, - apiv1.ECDSAWithSHA512: { - Kty: keyvault.EC, - Curve: keyvault.P521, - }, -} - -// vaultResource is the value the client will use as audience. -const vaultResource = "https://vault.azure.net" - -// KeyVaultClient is the interface implemented by keyvault.BaseClient. It will -// be used for testing purposes. -type KeyVaultClient interface { - GetKey(ctx context.Context, vaultBaseURL string, keyName string, keyVersion string) (keyvault.KeyBundle, error) - CreateKey(ctx context.Context, vaultBaseURL string, keyName string, parameters keyvault.KeyCreateParameters) (keyvault.KeyBundle, error) - Sign(ctx context.Context, vaultBaseURL string, keyName string, keyVersion string, parameters keyvault.KeySignParameters) (keyvault.KeyOperationResult, error) -} - -// KeyVault implements a KMS using Azure Key Vault. -// -// The URI format used in Azure Key Vault is the following: -// -// - azurekms:name=key-name;vault=vault-name -// - azurekms:name=key-name;vault=vault-name?version=key-version -// - azurekms:name=key-name;vault=vault-name?hsm=true -// -// The scheme is "azurekms"; "name" is the key name; "vault" is the key vault -// name where the key is located; "version" is an optional parameter that -// defines the version of they key, if version is not given, the latest one will -// be used; "hsm" defines if an HSM want to be used for this key, this is -// specially useful when this is used from `step`. -// -// TODO(mariano): The implementation is using /services/keyvault/v7.1/keyvault -// package, at some point Azure might create a keyvault client with all the -// functionality in /sdk/keyvault, we should migrate to that once available. -type KeyVault struct { - baseClient KeyVaultClient - defaults DefaultOptions -} - -// DefaultOptions are custom options that can be passed as defaults using the -// URI in apiv1.Options. -type DefaultOptions struct { - Vault string - ProtectionLevel apiv1.ProtectionLevel -} - -var createClient = func(ctx context.Context, opts apiv1.Options) (KeyVaultClient, error) { - baseClient := keyvault.New() - - // With an URI, try to log in only using client credentials in the URI. - // Client credentials requires: - // - client-id - // - client-secret - // - tenant-id - // And optionally the aad-endpoint to support custom clouds: - // - aad-endpoint (defaults to https://login.microsoftonline.com/) - if opts.URI != "" { - u, err := uri.ParseWithScheme(Scheme, opts.URI) - if err != nil { - return nil, err - } - - // Required options - clientID := u.Get("client-id") - clientSecret := u.Get("client-secret") - tenantID := u.Get("tenant-id") - // optional - aadEndpoint := u.Get("aad-endpoint") - - if clientID != "" && clientSecret != "" && tenantID != "" { - s := auth.EnvironmentSettings{ - Values: map[string]string{ - auth.ClientID: clientID, - auth.ClientSecret: clientSecret, - auth.TenantID: tenantID, - auth.Resource: vaultResource, - }, - Environment: azure.PublicCloud, - } - if aadEndpoint != "" { - s.Environment.ActiveDirectoryEndpoint = aadEndpoint - } - baseClient.Authorizer, err = s.GetAuthorizer() - if err != nil { - return nil, err - } - return baseClient, nil - } - } - - // Attempt to authorize with the following methods: - // 1. Environment variables. - // - Client credentials - // - Client certificate - // - Username and password - // - MSI - // 2. Using Azure CLI 2.0 on local development. - authorizer, err := auth.NewAuthorizerFromEnvironmentWithResource(vaultResource) - if err != nil { - authorizer, err = auth.NewAuthorizerFromCLIWithResource(vaultResource) - if err != nil { - return nil, errors.Wrap(err, "error getting authorizer for key vault") - } - } - baseClient.Authorizer = authorizer - return &baseClient, nil -} - -// New initializes a new KMS implemented using Azure Key Vault. -func New(ctx context.Context, opts apiv1.Options) (*KeyVault, error) { - baseClient, err := createClient(ctx, opts) - if err != nil { - return nil, err - } - - // step and step-ca do not need and URI, but having a default vault and - // protection level is useful if this package is used as an api - var defaults DefaultOptions - if opts.URI != "" { - u, err := uri.ParseWithScheme(Scheme, opts.URI) - if err != nil { - return nil, err - } - defaults.Vault = u.Get("vault") - if u.GetBool("hsm") { - defaults.ProtectionLevel = apiv1.HSM - } - } - - return &KeyVault{ - baseClient: baseClient, - defaults: defaults, - }, nil -} - -// GetPublicKey loads a public key from Azure Key Vault by its resource name. -func (k *KeyVault) GetPublicKey(req *apiv1.GetPublicKeyRequest) (crypto.PublicKey, error) { - if req.Name == "" { - return nil, errors.New("getPublicKeyRequest 'name' cannot be empty") - } - - vault, name, version, _, err := parseKeyName(req.Name, k.defaults) - if err != nil { - return nil, err - } - - ctx, cancel := defaultContext() - defer cancel() - - resp, err := k.baseClient.GetKey(ctx, vaultBaseURL(vault), name, version) - if err != nil { - return nil, errors.Wrap(err, "keyVault GetKey failed") - } - - return convertKey(resp.Key) -} - -// CreateKey creates a asymmetric key in Azure Key Vault. -func (k *KeyVault) CreateKey(req *apiv1.CreateKeyRequest) (*apiv1.CreateKeyResponse, error) { - if req.Name == "" { - return nil, errors.New("createKeyRequest 'name' cannot be empty") - } - - vault, name, _, hsm, err := parseKeyName(req.Name, k.defaults) - if err != nil { - return nil, err - } - - // Override protection level to HSM only if it's not specified, and is given - // in the uri. - protectionLevel := req.ProtectionLevel - if protectionLevel == apiv1.UnspecifiedProtectionLevel && hsm { - protectionLevel = apiv1.HSM - } - - kt, ok := signatureAlgorithmMapping[req.SignatureAlgorithm] - if !ok { - return nil, errors.Errorf("keyVault does not support signature algorithm '%s'", req.SignatureAlgorithm) - } - var keySize *int32 - if kt.Kty == keyvault.RSA || kt.Kty == keyvault.RSAHSM { - switch req.Bits { - case 2048: - keySize = &value2048 - case 0, 3072: - keySize = &value3072 - case 4096: - keySize = &value4096 - default: - return nil, errors.Errorf("keyVault does not support key size %d", req.Bits) - } - } - - created := date.UnixTime(now()) - - ctx, cancel := defaultContext() - defer cancel() - - resp, err := k.baseClient.CreateKey(ctx, vaultBaseURL(vault), name, keyvault.KeyCreateParameters{ - Kty: kt.KeyType(protectionLevel), - KeySize: keySize, - Curve: kt.Curve, - KeyOps: &[]keyvault.JSONWebKeyOperation{ - keyvault.Sign, keyvault.Verify, - }, - KeyAttributes: &keyvault.KeyAttributes{ - Enabled: &valueTrue, - Created: &created, - NotBefore: &created, - }, - }) - if err != nil { - return nil, errors.Wrap(err, "keyVault CreateKey failed") - } - - publicKey, err := convertKey(resp.Key) - if err != nil { - return nil, err - } - - keyURI := getKeyName(vault, name, resp) - return &apiv1.CreateKeyResponse{ - Name: keyURI, - PublicKey: publicKey, - CreateSignerRequest: apiv1.CreateSignerRequest{ - SigningKey: keyURI, - }, - }, nil -} - -// CreateSigner returns a crypto.Signer from a previously created asymmetric key. -func (k *KeyVault) CreateSigner(req *apiv1.CreateSignerRequest) (crypto.Signer, error) { - if req.SigningKey == "" { - return nil, errors.New("createSignerRequest 'signingKey' cannot be empty") - } - return NewSigner(k.baseClient, req.SigningKey, k.defaults) -} - -// Close closes the client connection to the Azure Key Vault. This is a noop. -func (k *KeyVault) Close() error { - return nil -} - -// ValidateName validates that the given string is a valid URI. -func (k *KeyVault) ValidateName(s string) error { - _, _, _, _, err := parseKeyName(s, k.defaults) - return err -} diff --git a/kms/azurekms/key_vault_test.go b/kms/azurekms/key_vault_test.go deleted file mode 100644 index 8f968189..00000000 --- a/kms/azurekms/key_vault_test.go +++ /dev/null @@ -1,653 +0,0 @@ -//go:generate mockgen -package mock -mock_names=KeyVaultClient=KeyVaultClient -destination internal/mock/key_vault_client.go github.com/smallstep/certificates/kms/azurekms KeyVaultClient -package azurekms - -import ( - "context" - "crypto" - "encoding/json" - "fmt" - "reflect" - "testing" - "time" - - "github.com/Azure/azure-sdk-for-go/services/keyvault/v7.1/keyvault" - "github.com/Azure/go-autorest/autorest/date" - "github.com/golang/mock/gomock" - "github.com/smallstep/certificates/kms/apiv1" - "github.com/smallstep/certificates/kms/azurekms/internal/mock" - "go.step.sm/crypto/keyutil" - "gopkg.in/square/go-jose.v2" -) - -var errTest = fmt.Errorf("test error") - -func mockNow(t *testing.T) time.Time { - old := now - t0 := time.Unix(1234567890, 123).UTC() - now = func() time.Time { - return t0 - } - t.Cleanup(func() { - now = old - }) - return t0 -} - -func mockClient(t *testing.T) *mock.KeyVaultClient { - t.Helper() - ctrl := gomock.NewController(t) - t.Cleanup(func() { - ctrl.Finish() - }) - return mock.NewKeyVaultClient(ctrl) -} - -func createJWK(t *testing.T, pub crypto.PublicKey) *keyvault.JSONWebKey { - t.Helper() - b, err := json.Marshal(&jose.JSONWebKey{ - Key: pub, - }) - if err != nil { - t.Fatal(err) - } - key := new(keyvault.JSONWebKey) - if err := json.Unmarshal(b, key); err != nil { - t.Fatal(err) - } - return key -} - -func Test_now(t *testing.T) { - t0 := now() - if loc := t0.Location(); loc != time.UTC { - t.Errorf("now() Location = %v, want %v", loc, time.UTC) - } -} - -func TestNew(t *testing.T) { - client := mockClient(t) - old := createClient - t.Cleanup(func() { - createClient = old - }) - - type args struct { - ctx context.Context - opts apiv1.Options - } - tests := []struct { - name string - setup func() - args args - want *KeyVault - wantErr bool - }{ - {"ok", func() { - createClient = func(ctx context.Context, opts apiv1.Options) (KeyVaultClient, error) { - return client, nil - } - }, args{context.Background(), apiv1.Options{}}, &KeyVault{ - baseClient: client, - }, false}, - {"ok with vault", func() { - createClient = func(ctx context.Context, opts apiv1.Options) (KeyVaultClient, error) { - return client, nil - } - }, args{context.Background(), apiv1.Options{ - URI: "azurekms:vault=my-vault", - }}, &KeyVault{ - baseClient: client, - defaults: DefaultOptions{ - Vault: "my-vault", - ProtectionLevel: apiv1.UnspecifiedProtectionLevel, - }, - }, false}, - {"ok with vault + hsm", func() { - createClient = func(ctx context.Context, opts apiv1.Options) (KeyVaultClient, error) { - return client, nil - } - }, args{context.Background(), apiv1.Options{ - URI: "azurekms:vault=my-vault;hsm=true", - }}, &KeyVault{ - baseClient: client, - defaults: DefaultOptions{ - Vault: "my-vault", - ProtectionLevel: apiv1.HSM, - }, - }, false}, - {"fail", func() { - createClient = func(ctx context.Context, opts apiv1.Options) (KeyVaultClient, error) { - return nil, errTest - } - }, args{context.Background(), apiv1.Options{}}, nil, true}, - {"fail uri", func() { - createClient = func(ctx context.Context, opts apiv1.Options) (KeyVaultClient, error) { - return client, nil - } - }, args{context.Background(), apiv1.Options{ - URI: "kms:vault=my-vault;hsm=true", - }}, nil, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - tt.setup() - got, err := New(tt.args.ctx, tt.args.opts) - if (err != nil) != tt.wantErr { - t.Errorf("New() error = %v, wantErr %v", err, tt.wantErr) - return - } - if !reflect.DeepEqual(got, tt.want) { - t.Errorf("New() = %v, want %v", got, tt.want) - } - }) - } -} - -func TestKeyVault_createClient(t *testing.T) { - type args struct { - ctx context.Context - opts apiv1.Options - } - tests := []struct { - name string - args args - skip bool - wantErr bool - }{ - {"ok", args{context.Background(), apiv1.Options{}}, true, false}, - {"ok with uri", args{context.Background(), apiv1.Options{ - URI: "azurekms:client-id=id;client-secret=secret;tenant-id=id", - }}, false, false}, - {"ok with uri+aad", args{context.Background(), apiv1.Options{ - URI: "azurekms:client-id=id;client-secret=secret;tenant-id=id;aad-enpoint=https%3A%2F%2Flogin.microsoftonline.us%2F", - }}, false, false}, - {"ok with uri no config", args{context.Background(), apiv1.Options{ - URI: "azurekms:", - }}, true, false}, - {"fail uri", args{context.Background(), apiv1.Options{ - URI: "kms:client-id=id;client-secret=secret;tenant-id=id", - }}, false, true}, - } - - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - if tt.skip { - t.SkipNow() - } - _, err := createClient(tt.args.ctx, tt.args.opts) - if (err != nil) != tt.wantErr { - t.Errorf("New() error = %v, wantErr %v", err, tt.wantErr) - } - }) - } -} - -func TestKeyVault_GetPublicKey(t *testing.T) { - key, err := keyutil.GenerateDefaultSigner() - if err != nil { - t.Fatal(err) - } - pub := key.Public() - jwk := createJWK(t, pub) - - client := mockClient(t) - client.EXPECT().GetKey(gomock.Any(), "https://my-vault.vault.azure.net/", "my-key", "").Return(keyvault.KeyBundle{ - Key: jwk, - }, nil) - client.EXPECT().GetKey(gomock.Any(), "https://my-vault.vault.azure.net/", "my-key", "my-version").Return(keyvault.KeyBundle{ - Key: jwk, - }, nil) - client.EXPECT().GetKey(gomock.Any(), "https://my-vault.vault.azure.net/", "not-found", "my-version").Return(keyvault.KeyBundle{}, errTest) - - type fields struct { - baseClient KeyVaultClient - } - type args struct { - req *apiv1.GetPublicKeyRequest - } - tests := []struct { - name string - fields fields - args args - want crypto.PublicKey - wantErr bool - }{ - {"ok", fields{client}, args{&apiv1.GetPublicKeyRequest{ - Name: "azurekms:vault=my-vault;name=my-key", - }}, pub, false}, - {"ok with version", fields{client}, args{&apiv1.GetPublicKeyRequest{ - Name: "azurekms:vault=my-vault;name=my-key?version=my-version", - }}, pub, false}, - {"fail GetKey", fields{client}, args{&apiv1.GetPublicKeyRequest{ - Name: "azurekms:vault=my-vault;name=not-found?version=my-version", - }}, nil, true}, - {"fail empty", fields{client}, args{&apiv1.GetPublicKeyRequest{ - Name: "", - }}, nil, true}, - {"fail vault", fields{client}, args{&apiv1.GetPublicKeyRequest{ - Name: "azurekms:vault=;name=not-found?version=my-version", - }}, nil, true}, - {"fail id", fields{client}, args{&apiv1.GetPublicKeyRequest{ - Name: "azurekms:vault=;name=?version=my-version", - }}, nil, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - k := &KeyVault{ - baseClient: tt.fields.baseClient, - } - got, err := k.GetPublicKey(tt.args.req) - if (err != nil) != tt.wantErr { - t.Errorf("KeyVault.GetPublicKey() error = %v, wantErr %v", err, tt.wantErr) - return - } - if !reflect.DeepEqual(got, tt.want) { - t.Errorf("KeyVault.GetPublicKey() = %v, want %v", got, tt.want) - } - }) - } -} - -func TestKeyVault_CreateKey(t *testing.T) { - ecKey, err := keyutil.GenerateDefaultSigner() - if err != nil { - t.Fatal(err) - } - rsaKey, err := keyutil.GenerateSigner("RSA", "", 2048) - if err != nil { - t.Fatal(err) - } - ecPub := ecKey.Public() - rsaPub := rsaKey.Public() - ecJWK := createJWK(t, ecPub) - rsaJWK := createJWK(t, rsaPub) - - t0 := date.UnixTime(mockNow(t)) - client := mockClient(t) - - expects := []struct { - Name string - Kty keyvault.JSONWebKeyType - KeySize *int32 - Curve keyvault.JSONWebKeyCurveName - Key *keyvault.JSONWebKey - }{ - {"P-256", keyvault.EC, nil, keyvault.P256, ecJWK}, - {"P-256 HSM", keyvault.ECHSM, nil, keyvault.P256, ecJWK}, - {"P-256 HSM (uri)", keyvault.ECHSM, nil, keyvault.P256, ecJWK}, - {"P-256 Default", keyvault.EC, nil, keyvault.P256, ecJWK}, - {"P-384", keyvault.EC, nil, keyvault.P384, ecJWK}, - {"P-521", keyvault.EC, nil, keyvault.P521, ecJWK}, - {"RSA 0", keyvault.RSA, &value3072, "", rsaJWK}, - {"RSA 0 HSM", keyvault.RSAHSM, &value3072, "", rsaJWK}, - {"RSA 0 HSM (uri)", keyvault.RSAHSM, &value3072, "", rsaJWK}, - {"RSA 2048", keyvault.RSA, &value2048, "", rsaJWK}, - {"RSA 3072", keyvault.RSA, &value3072, "", rsaJWK}, - {"RSA 4096", keyvault.RSA, &value4096, "", rsaJWK}, - } - - for _, e := range expects { - client.EXPECT().CreateKey(gomock.Any(), "https://my-vault.vault.azure.net/", "my-key", keyvault.KeyCreateParameters{ - Kty: e.Kty, - KeySize: e.KeySize, - Curve: e.Curve, - KeyOps: &[]keyvault.JSONWebKeyOperation{ - keyvault.Sign, keyvault.Verify, - }, - KeyAttributes: &keyvault.KeyAttributes{ - Enabled: &valueTrue, - Created: &t0, - NotBefore: &t0, - }, - }).Return(keyvault.KeyBundle{ - Key: e.Key, - }, nil) - } - client.EXPECT().CreateKey(gomock.Any(), "https://my-vault.vault.azure.net/", "not-found", gomock.Any()).Return(keyvault.KeyBundle{}, errTest) - client.EXPECT().CreateKey(gomock.Any(), "https://my-vault.vault.azure.net/", "not-found", gomock.Any()).Return(keyvault.KeyBundle{ - Key: nil, - }, nil) - - type fields struct { - baseClient KeyVaultClient - } - type args struct { - req *apiv1.CreateKeyRequest - } - tests := []struct { - name string - fields fields - args args - want *apiv1.CreateKeyResponse - wantErr bool - }{ - {"ok P-256", fields{client}, args{&apiv1.CreateKeyRequest{ - Name: "azurekms:vault=my-vault;name=my-key", - SignatureAlgorithm: apiv1.ECDSAWithSHA256, - ProtectionLevel: apiv1.Software, - }}, &apiv1.CreateKeyResponse{ - Name: "azurekms:name=my-key;vault=my-vault", - PublicKey: ecPub, - CreateSignerRequest: apiv1.CreateSignerRequest{ - SigningKey: "azurekms:name=my-key;vault=my-vault", - }, - }, false}, - {"ok P-256 HSM", fields{client}, args{&apiv1.CreateKeyRequest{ - Name: "azurekms:vault=my-vault;name=my-key", - SignatureAlgorithm: apiv1.ECDSAWithSHA256, - ProtectionLevel: apiv1.HSM, - }}, &apiv1.CreateKeyResponse{ - Name: "azurekms:name=my-key;vault=my-vault", - PublicKey: ecPub, - CreateSignerRequest: apiv1.CreateSignerRequest{ - SigningKey: "azurekms:name=my-key;vault=my-vault", - }, - }, false}, - {"ok P-256 HSM (uri)", fields{client}, args{&apiv1.CreateKeyRequest{ - Name: "azurekms:vault=my-vault;name=my-key?hsm=true", - SignatureAlgorithm: apiv1.ECDSAWithSHA256, - }}, &apiv1.CreateKeyResponse{ - Name: "azurekms:name=my-key;vault=my-vault", - PublicKey: ecPub, - CreateSignerRequest: apiv1.CreateSignerRequest{ - SigningKey: "azurekms:name=my-key;vault=my-vault", - }, - }, false}, - {"ok P-256 Default", fields{client}, args{&apiv1.CreateKeyRequest{ - Name: "azurekms:vault=my-vault;name=my-key", - }}, &apiv1.CreateKeyResponse{ - Name: "azurekms:name=my-key;vault=my-vault", - PublicKey: ecPub, - CreateSignerRequest: apiv1.CreateSignerRequest{ - SigningKey: "azurekms:name=my-key;vault=my-vault", - }, - }, false}, - {"ok P-384", fields{client}, args{&apiv1.CreateKeyRequest{ - Name: "azurekms:vault=my-vault;name=my-key", - SignatureAlgorithm: apiv1.ECDSAWithSHA384, - }}, &apiv1.CreateKeyResponse{ - Name: "azurekms:name=my-key;vault=my-vault", - PublicKey: ecPub, - CreateSignerRequest: apiv1.CreateSignerRequest{ - SigningKey: "azurekms:name=my-key;vault=my-vault", - }, - }, false}, - {"ok P-521", fields{client}, args{&apiv1.CreateKeyRequest{ - Name: "azurekms:vault=my-vault;name=my-key", - SignatureAlgorithm: apiv1.ECDSAWithSHA512, - }}, &apiv1.CreateKeyResponse{ - Name: "azurekms:name=my-key;vault=my-vault", - PublicKey: ecPub, - CreateSignerRequest: apiv1.CreateSignerRequest{ - SigningKey: "azurekms:name=my-key;vault=my-vault", - }, - }, false}, - {"ok RSA 0", fields{client}, args{&apiv1.CreateKeyRequest{ - Name: "azurekms:vault=my-vault;name=my-key", - Bits: 0, - SignatureAlgorithm: apiv1.SHA256WithRSA, - ProtectionLevel: apiv1.Software, - }}, &apiv1.CreateKeyResponse{ - Name: "azurekms:name=my-key;vault=my-vault", - PublicKey: rsaPub, - CreateSignerRequest: apiv1.CreateSignerRequest{ - SigningKey: "azurekms:name=my-key;vault=my-vault", - }, - }, false}, - {"ok RSA 0 HSM", fields{client}, args{&apiv1.CreateKeyRequest{ - Name: "azurekms:vault=my-vault;name=my-key", - Bits: 0, - SignatureAlgorithm: apiv1.SHA256WithRSAPSS, - ProtectionLevel: apiv1.HSM, - }}, &apiv1.CreateKeyResponse{ - Name: "azurekms:name=my-key;vault=my-vault", - PublicKey: rsaPub, - CreateSignerRequest: apiv1.CreateSignerRequest{ - SigningKey: "azurekms:name=my-key;vault=my-vault", - }, - }, false}, - {"ok RSA 0 HSM (uri)", fields{client}, args{&apiv1.CreateKeyRequest{ - Name: "azurekms:vault=my-vault;name=my-key;hsm=true", - Bits: 0, - SignatureAlgorithm: apiv1.SHA256WithRSAPSS, - }}, &apiv1.CreateKeyResponse{ - Name: "azurekms:name=my-key;vault=my-vault", - PublicKey: rsaPub, - CreateSignerRequest: apiv1.CreateSignerRequest{ - SigningKey: "azurekms:name=my-key;vault=my-vault", - }, - }, false}, - {"ok RSA 2048", fields{client}, args{&apiv1.CreateKeyRequest{ - Name: "azurekms:vault=my-vault;name=my-key", - Bits: 2048, - SignatureAlgorithm: apiv1.SHA384WithRSA, - }}, &apiv1.CreateKeyResponse{ - Name: "azurekms:name=my-key;vault=my-vault", - PublicKey: rsaPub, - CreateSignerRequest: apiv1.CreateSignerRequest{ - SigningKey: "azurekms:name=my-key;vault=my-vault", - }, - }, false}, - {"ok RSA 3072", fields{client}, args{&apiv1.CreateKeyRequest{ - Name: "azurekms:vault=my-vault;name=my-key", - Bits: 3072, - SignatureAlgorithm: apiv1.SHA512WithRSA, - }}, &apiv1.CreateKeyResponse{ - Name: "azurekms:name=my-key;vault=my-vault", - PublicKey: rsaPub, - CreateSignerRequest: apiv1.CreateSignerRequest{ - SigningKey: "azurekms:name=my-key;vault=my-vault", - }, - }, false}, - {"ok RSA 4096", fields{client}, args{&apiv1.CreateKeyRequest{ - Name: "azurekms:vault=my-vault;name=my-key", - Bits: 4096, - SignatureAlgorithm: apiv1.SHA512WithRSAPSS, - }}, &apiv1.CreateKeyResponse{ - Name: "azurekms:name=my-key;vault=my-vault", - PublicKey: rsaPub, - CreateSignerRequest: apiv1.CreateSignerRequest{ - SigningKey: "azurekms:name=my-key;vault=my-vault", - }, - }, false}, - {"fail createKey", fields{client}, args{&apiv1.CreateKeyRequest{ - Name: "azurekms:vault=my-vault;name=not-found", - SignatureAlgorithm: apiv1.ECDSAWithSHA256, - }}, nil, true}, - {"fail convertKey", fields{client}, args{&apiv1.CreateKeyRequest{ - Name: "azurekms:vault=my-vault;name=not-found", - SignatureAlgorithm: apiv1.ECDSAWithSHA256, - }}, nil, true}, - {"fail name", fields{client}, args{&apiv1.CreateKeyRequest{ - Name: "", - }}, nil, true}, - {"fail vault", fields{client}, args{&apiv1.CreateKeyRequest{ - Name: "azurekms:vault=;name=not-found?version=my-version", - }}, nil, true}, - {"fail id", fields{client}, args{&apiv1.CreateKeyRequest{ - Name: "azurekms:vault=my-vault;name=?version=my-version", - }}, nil, true}, - {"fail SignatureAlgorithm", fields{client}, args{&apiv1.CreateKeyRequest{ - Name: "azurekms:vault=my-vault;name=not-found", - SignatureAlgorithm: apiv1.PureEd25519, - }}, nil, true}, - {"fail bit size", fields{client}, args{&apiv1.CreateKeyRequest{ - Name: "azurekms:vault=my-vault;name=not-found", - SignatureAlgorithm: apiv1.SHA384WithRSAPSS, - Bits: 1024, - }}, nil, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - k := &KeyVault{ - baseClient: tt.fields.baseClient, - } - got, err := k.CreateKey(tt.args.req) - if (err != nil) != tt.wantErr { - t.Errorf("KeyVault.CreateKey() error = %v, wantErr %v", err, tt.wantErr) - return - } - if !reflect.DeepEqual(got, tt.want) { - t.Errorf("KeyVault.CreateKey() = %v, want %v", got, tt.want) - } - }) - } -} - -func TestKeyVault_CreateSigner(t *testing.T) { - key, err := keyutil.GenerateDefaultSigner() - if err != nil { - t.Fatal(err) - } - pub := key.Public() - jwk := createJWK(t, pub) - - client := mockClient(t) - client.EXPECT().GetKey(gomock.Any(), "https://my-vault.vault.azure.net/", "my-key", "").Return(keyvault.KeyBundle{ - Key: jwk, - }, nil) - client.EXPECT().GetKey(gomock.Any(), "https://my-vault.vault.azure.net/", "my-key", "my-version").Return(keyvault.KeyBundle{ - Key: jwk, - }, nil) - client.EXPECT().GetKey(gomock.Any(), "https://my-vault.vault.azure.net/", "not-found", "my-version").Return(keyvault.KeyBundle{}, errTest) - - type fields struct { - baseClient KeyVaultClient - } - type args struct { - req *apiv1.CreateSignerRequest - } - tests := []struct { - name string - fields fields - args args - want crypto.Signer - wantErr bool - }{ - {"ok", fields{client}, args{&apiv1.CreateSignerRequest{ - SigningKey: "azurekms:vault=my-vault;name=my-key", - }}, &Signer{ - client: client, - vaultBaseURL: "https://my-vault.vault.azure.net/", - name: "my-key", - version: "", - publicKey: pub, - }, false}, - {"ok with version", fields{client}, args{&apiv1.CreateSignerRequest{ - SigningKey: "azurekms:vault=my-vault;name=my-key;version=my-version", - }}, &Signer{ - client: client, - vaultBaseURL: "https://my-vault.vault.azure.net/", - name: "my-key", - version: "my-version", - publicKey: pub, - }, false}, - {"fail GetKey", fields{client}, args{&apiv1.CreateSignerRequest{ - SigningKey: "azurekms:vault=my-vault;name=not-found;version=my-version", - }}, nil, true}, - {"fail SigningKey", fields{client}, args{&apiv1.CreateSignerRequest{ - SigningKey: "", - }}, nil, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - k := &KeyVault{ - baseClient: tt.fields.baseClient, - } - got, err := k.CreateSigner(tt.args.req) - if (err != nil) != tt.wantErr { - t.Errorf("KeyVault.CreateSigner() error = %v, wantErr %v", err, tt.wantErr) - return - } - if !reflect.DeepEqual(got, tt.want) { - t.Errorf("KeyVault.CreateSigner() = %v, want %v", got, tt.want) - } - }) - } -} - -func TestKeyVault_Close(t *testing.T) { - client := mockClient(t) - type fields struct { - baseClient KeyVaultClient - } - tests := []struct { - name string - fields fields - wantErr bool - }{ - {"ok", fields{client}, false}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - k := &KeyVault{ - baseClient: tt.fields.baseClient, - } - if err := k.Close(); (err != nil) != tt.wantErr { - t.Errorf("KeyVault.Close() error = %v, wantErr %v", err, tt.wantErr) - } - }) - } -} - -func Test_keyType_KeyType(t *testing.T) { - type fields struct { - Kty keyvault.JSONWebKeyType - Curve keyvault.JSONWebKeyCurveName - } - type args struct { - pl apiv1.ProtectionLevel - } - tests := []struct { - name string - fields fields - args args - want keyvault.JSONWebKeyType - }{ - {"ec", fields{keyvault.EC, keyvault.P256}, args{apiv1.UnspecifiedProtectionLevel}, keyvault.EC}, - {"ec software", fields{keyvault.EC, keyvault.P384}, args{apiv1.Software}, keyvault.EC}, - {"ec hsm", fields{keyvault.EC, keyvault.P521}, args{apiv1.HSM}, keyvault.ECHSM}, - {"rsa", fields{keyvault.RSA, keyvault.P256}, args{apiv1.UnspecifiedProtectionLevel}, keyvault.RSA}, - {"rsa software", fields{keyvault.RSA, ""}, args{apiv1.Software}, keyvault.RSA}, - {"rsa hsm", fields{keyvault.RSA, ""}, args{apiv1.HSM}, keyvault.RSAHSM}, - {"empty", fields{"FOO", ""}, args{apiv1.UnspecifiedProtectionLevel}, ""}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - k := keyType{ - Kty: tt.fields.Kty, - Curve: tt.fields.Curve, - } - if got := k.KeyType(tt.args.pl); !reflect.DeepEqual(got, tt.want) { - t.Errorf("keyType.KeyType() = %v, want %v", got, tt.want) - } - }) - } -} - -func TestKeyVault_ValidateName(t *testing.T) { - type args struct { - s string - } - tests := []struct { - name string - args args - wantErr bool - }{ - {"ok", args{"azurekms:name=my-key;vault=my-vault"}, false}, - {"ok hsm", args{"azurekms:name=my-key;vault=my-vault?hsm=true"}, false}, - {"fail scheme", args{"azure:name=my-key;vault=my-vault"}, true}, - {"fail parse uri", args{"azurekms:name=%ZZ;vault=my-vault"}, true}, - {"fail no name", args{"azurekms:vault=my-vault"}, true}, - {"fail no vault", args{"azurekms:name=my-key"}, true}, - {"fail empty", args{""}, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - k := &KeyVault{} - if err := k.ValidateName(tt.args.s); (err != nil) != tt.wantErr { - t.Errorf("KeyVault.ValidateName() error = %v, wantErr %v", err, tt.wantErr) - } - }) - } -} diff --git a/kms/azurekms/signer.go b/kms/azurekms/signer.go deleted file mode 100644 index b0349108..00000000 --- a/kms/azurekms/signer.go +++ /dev/null @@ -1,182 +0,0 @@ -package azurekms - -import ( - "crypto" - "crypto/ecdsa" - "crypto/rsa" - "encoding/base64" - "io" - "math/big" - "time" - - "github.com/Azure/azure-sdk-for-go/services/keyvault/v7.1/keyvault" - "github.com/Azure/go-autorest/autorest/azure" - "github.com/pkg/errors" - "golang.org/x/crypto/cryptobyte" - "golang.org/x/crypto/cryptobyte/asn1" -) - -// Signer implements a crypto.Signer using the AWS KMS. -type Signer struct { - client KeyVaultClient - vaultBaseURL string - name string - version string - publicKey crypto.PublicKey -} - -// NewSigner creates a new signer using a key in the AWS KMS. -func NewSigner(client KeyVaultClient, signingKey string, defaults DefaultOptions) (crypto.Signer, error) { - vault, name, version, _, err := parseKeyName(signingKey, defaults) - if err != nil { - return nil, err - } - - // Make sure that the key exists. - signer := &Signer{ - client: client, - vaultBaseURL: vaultBaseURL(vault), - name: name, - version: version, - } - if err := signer.preloadKey(); err != nil { - return nil, err - } - - return signer, nil -} - -func (s *Signer) preloadKey() error { - ctx, cancel := defaultContext() - defer cancel() - - resp, err := s.client.GetKey(ctx, s.vaultBaseURL, s.name, s.version) - if err != nil { - return errors.Wrap(err, "keyVault GetKey failed") - } - - s.publicKey, err = convertKey(resp.Key) - return err -} - -// Public returns the public key of this signer or an error. -func (s *Signer) Public() crypto.PublicKey { - return s.publicKey -} - -// Sign signs digest with the private key stored in the AWS KMS. -func (s *Signer) Sign(rand io.Reader, digest []byte, opts crypto.SignerOpts) ([]byte, error) { - alg, err := getSigningAlgorithm(s.Public(), opts) - if err != nil { - return nil, err - } - - b64 := base64.RawURLEncoding.EncodeToString(digest) - - // Sign with retry if the key is not ready - resp, err := s.signWithRetry(alg, b64, 3) - if err != nil { - return nil, errors.Wrap(err, "keyVault Sign failed") - } - - sig, err := base64.RawURLEncoding.DecodeString(*resp.Result) - if err != nil { - return nil, errors.Wrap(err, "error decoding keyVault Sign result") - } - - var octetSize int - switch alg { - case keyvault.ES256: - octetSize = 32 // 256-bit, concat(R,S) = 64 bytes - case keyvault.ES384: - octetSize = 48 // 384-bit, concat(R,S) = 96 bytes - case keyvault.ES512: - octetSize = 66 // 528-bit, concat(R,S) = 132 bytes - default: - return sig, nil - } - - // Convert to asn1 - if len(sig) != octetSize*2 { - return nil, errors.Errorf("keyVault Sign failed: unexpected signature length") - } - var b cryptobyte.Builder - b.AddASN1(asn1.SEQUENCE, func(b *cryptobyte.Builder) { - b.AddASN1BigInt(new(big.Int).SetBytes(sig[:octetSize])) // R - b.AddASN1BigInt(new(big.Int).SetBytes(sig[octetSize:])) // S - }) - return b.Bytes() -} - -func (s *Signer) signWithRetry(alg keyvault.JSONWebKeySignatureAlgorithm, b64 string, retryAttempts int) (keyvault.KeyOperationResult, error) { -retry: - ctx, cancel := defaultContext() - defer cancel() - - resp, err := s.client.Sign(ctx, s.vaultBaseURL, s.name, s.version, keyvault.KeySignParameters{ - Algorithm: alg, - Value: &b64, - }) - if err != nil && retryAttempts > 0 { - var requestError *azure.RequestError - if errors.As(err, &requestError) { - if se := requestError.ServiceError; se != nil && se.InnerError != nil { - code, ok := se.InnerError["code"].(string) - if ok && code == "KeyNotYetValid" { - time.Sleep(time.Second / time.Duration(retryAttempts)) - retryAttempts-- - goto retry - } - } - } - } - return resp, err -} - -func getSigningAlgorithm(key crypto.PublicKey, opts crypto.SignerOpts) (keyvault.JSONWebKeySignatureAlgorithm, error) { - switch key.(type) { - case *rsa.PublicKey: - hashFunc := opts.HashFunc() - pss, isPSS := opts.(*rsa.PSSOptions) - // Random salt lengths are not supported - if isPSS && - pss.SaltLength != rsa.PSSSaltLengthAuto && - pss.SaltLength != rsa.PSSSaltLengthEqualsHash && - pss.SaltLength != hashFunc.Size() { - return "", errors.Errorf("unsupported RSA-PSS salt length %d", pss.SaltLength) - } - - switch h := hashFunc; h { - case crypto.SHA256: - if isPSS { - return keyvault.PS256, nil - } - return keyvault.RS256, nil - case crypto.SHA384: - if isPSS { - return keyvault.PS384, nil - } - return keyvault.RS384, nil - case crypto.SHA512: - if isPSS { - return keyvault.PS512, nil - } - return keyvault.RS512, nil - default: - return "", errors.Errorf("unsupported hash function %v", h) - } - case *ecdsa.PublicKey: - switch h := opts.HashFunc(); h { - case crypto.SHA256: - return keyvault.ES256, nil - case crypto.SHA384: - return keyvault.ES384, nil - case crypto.SHA512: - return keyvault.ES512, nil - default: - return "", errors.Errorf("unsupported hash function %v", h) - } - default: - return "", errors.Errorf("unsupported key type %T", key) - } -} diff --git a/kms/azurekms/signer_test.go b/kms/azurekms/signer_test.go deleted file mode 100644 index bd072b25..00000000 --- a/kms/azurekms/signer_test.go +++ /dev/null @@ -1,493 +0,0 @@ -package azurekms - -import ( - "crypto" - "crypto/ecdsa" - "crypto/rand" - "crypto/rsa" - "encoding/base64" - "io" - "reflect" - "testing" - - "github.com/Azure/azure-sdk-for-go/services/keyvault/v7.1/keyvault" - "github.com/Azure/go-autorest/autorest" - "github.com/Azure/go-autorest/autorest/azure" - "github.com/golang/mock/gomock" - "github.com/smallstep/certificates/kms/apiv1" - "go.step.sm/crypto/keyutil" - "golang.org/x/crypto/cryptobyte" - "golang.org/x/crypto/cryptobyte/asn1" -) - -func TestNewSigner(t *testing.T) { - key, err := keyutil.GenerateDefaultSigner() - if err != nil { - t.Fatal(err) - } - pub := key.Public() - jwk := createJWK(t, pub) - - client := mockClient(t) - client.EXPECT().GetKey(gomock.Any(), "https://my-vault.vault.azure.net/", "my-key", "").Return(keyvault.KeyBundle{ - Key: jwk, - }, nil) - client.EXPECT().GetKey(gomock.Any(), "https://my-vault.vault.azure.net/", "my-key", "my-version").Return(keyvault.KeyBundle{ - Key: jwk, - }, nil) - client.EXPECT().GetKey(gomock.Any(), "https://my-vault.vault.azure.net/", "my-key", "my-version").Return(keyvault.KeyBundle{ - Key: jwk, - }, nil) - client.EXPECT().GetKey(gomock.Any(), "https://my-vault.vault.azure.net/", "not-found", "my-version").Return(keyvault.KeyBundle{}, errTest) - - var noOptions DefaultOptions - type args struct { - client KeyVaultClient - signingKey string - defaults DefaultOptions - } - tests := []struct { - name string - args args - want crypto.Signer - wantErr bool - }{ - {"ok", args{client, "azurekms:vault=my-vault;name=my-key", noOptions}, &Signer{ - client: client, - vaultBaseURL: "https://my-vault.vault.azure.net/", - name: "my-key", - version: "", - publicKey: pub, - }, false}, - {"ok with version", args{client, "azurekms:name=my-key;vault=my-vault?version=my-version", noOptions}, &Signer{ - client: client, - vaultBaseURL: "https://my-vault.vault.azure.net/", - name: "my-key", - version: "my-version", - publicKey: pub, - }, false}, - {"ok with options", args{client, "azurekms:name=my-key?version=my-version", DefaultOptions{Vault: "my-vault", ProtectionLevel: apiv1.HSM}}, &Signer{ - client: client, - vaultBaseURL: "https://my-vault.vault.azure.net/", - name: "my-key", - version: "my-version", - publicKey: pub, - }, false}, - {"fail GetKey", args{client, "azurekms:name=not-found;vault=my-vault?version=my-version", noOptions}, nil, true}, - {"fail vault", args{client, "azurekms:name=not-found;vault=", noOptions}, nil, true}, - {"fail id", args{client, "azurekms:name=;vault=my-vault?version=my-version", noOptions}, nil, true}, - {"fail scheme", args{client, "kms:name=not-found;vault=my-vault?version=my-version", noOptions}, nil, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - got, err := NewSigner(tt.args.client, tt.args.signingKey, tt.args.defaults) - if (err != nil) != tt.wantErr { - t.Errorf("NewSigner() error = %v, wantErr %v", err, tt.wantErr) - return - } - if !reflect.DeepEqual(got, tt.want) { - t.Errorf("NewSigner() = %v, want %v", got, tt.want) - } - }) - } -} - -func TestSigner_Public(t *testing.T) { - key, err := keyutil.GenerateDefaultSigner() - if err != nil { - t.Fatal(err) - } - pub := key.Public() - - type fields struct { - publicKey crypto.PublicKey - } - tests := []struct { - name string - fields fields - want crypto.PublicKey - }{ - {"ok", fields{pub}, pub}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - s := &Signer{ - publicKey: tt.fields.publicKey, - } - if got := s.Public(); !reflect.DeepEqual(got, tt.want) { - t.Errorf("Signer.Public() = %v, want %v", got, tt.want) - } - }) - } -} - -func TestSigner_Sign(t *testing.T) { - sign := func(kty, crv string, bits int, opts crypto.SignerOpts) (crypto.PublicKey, []byte, string, []byte) { - key, err := keyutil.GenerateSigner(kty, crv, bits) - if err != nil { - t.Fatal(err) - } - h := opts.HashFunc().New() - h.Write([]byte("random-data")) - sum := h.Sum(nil) - - var sig, resultSig []byte - if priv, ok := key.(*ecdsa.PrivateKey); ok { - r, s, err := ecdsa.Sign(rand.Reader, priv, sum) - if err != nil { - t.Fatal(err) - } - curveBits := priv.Params().BitSize - keyBytes := curveBits / 8 - if curveBits%8 > 0 { - keyBytes++ - } - rBytes := r.Bytes() - rBytesPadded := make([]byte, keyBytes) - copy(rBytesPadded[keyBytes-len(rBytes):], rBytes) - - sBytes := s.Bytes() - sBytesPadded := make([]byte, keyBytes) - copy(sBytesPadded[keyBytes-len(sBytes):], sBytes) - // nolint:gocritic - resultSig = append(rBytesPadded, sBytesPadded...) - - var b cryptobyte.Builder - b.AddASN1(asn1.SEQUENCE, func(b *cryptobyte.Builder) { - b.AddASN1BigInt(r) - b.AddASN1BigInt(s) - }) - sig, err = b.Bytes() - if err != nil { - t.Fatal(err) - } - } else { - sig, err = key.Sign(rand.Reader, sum, opts) - if err != nil { - t.Fatal(err) - } - resultSig = sig - } - - return key.Public(), h.Sum(nil), base64.RawURLEncoding.EncodeToString(resultSig), sig - } - - p256, p256Digest, p256ResultSig, p256Sig := sign("EC", "P-256", 0, crypto.SHA256) - p384, p384Digest, p386ResultSig, p384Sig := sign("EC", "P-384", 0, crypto.SHA384) - p521, p521Digest, p521ResultSig, p521Sig := sign("EC", "P-521", 0, crypto.SHA512) - rsaSHA256, rsaSHA256Digest, rsaSHA256ResultSig, rsaSHA256Sig := sign("RSA", "", 2048, crypto.SHA256) - rsaSHA384, rsaSHA384Digest, rsaSHA384ResultSig, rsaSHA384Sig := sign("RSA", "", 2048, crypto.SHA384) - rsaSHA512, rsaSHA512Digest, rsaSHA512ResultSig, rsaSHA512Sig := sign("RSA", "", 2048, crypto.SHA512) - rsaPSSSHA256, rsaPSSSHA256Digest, rsaPSSSHA256ResultSig, rsaPSSSHA256Sig := sign("RSA", "", 2048, &rsa.PSSOptions{ - SaltLength: rsa.PSSSaltLengthAuto, - Hash: crypto.SHA256, - }) - rsaPSSSHA384, rsaPSSSHA384Digest, rsaPSSSHA384ResultSig, rsaPSSSHA384Sig := sign("RSA", "", 2048, &rsa.PSSOptions{ - SaltLength: rsa.PSSSaltLengthAuto, - Hash: crypto.SHA512, - }) - rsaPSSSHA512, rsaPSSSHA512Digest, rsaPSSSHA512ResultSig, rsaPSSSHA512Sig := sign("RSA", "", 2048, &rsa.PSSOptions{ - SaltLength: rsa.PSSSaltLengthAuto, - Hash: crypto.SHA512, - }) - - ed25519Key, err := keyutil.GenerateSigner("OKP", "Ed25519", 0) - if err != nil { - t.Fatal(err) - } - - client := mockClient(t) - expects := []struct { - name string - keyVersion string - alg keyvault.JSONWebKeySignatureAlgorithm - digest []byte - result keyvault.KeyOperationResult - err error - }{ - {"P-256", "", keyvault.ES256, p256Digest, keyvault.KeyOperationResult{ - Result: &p256ResultSig, - }, nil}, - {"P-384", "my-version", keyvault.ES384, p384Digest, keyvault.KeyOperationResult{ - Result: &p386ResultSig, - }, nil}, - {"P-521", "my-version", keyvault.ES512, p521Digest, keyvault.KeyOperationResult{ - Result: &p521ResultSig, - }, nil}, - {"RSA SHA256", "", keyvault.RS256, rsaSHA256Digest, keyvault.KeyOperationResult{ - Result: &rsaSHA256ResultSig, - }, nil}, - {"RSA SHA384", "", keyvault.RS384, rsaSHA384Digest, keyvault.KeyOperationResult{ - Result: &rsaSHA384ResultSig, - }, nil}, - {"RSA SHA512", "", keyvault.RS512, rsaSHA512Digest, keyvault.KeyOperationResult{ - Result: &rsaSHA512ResultSig, - }, nil}, - {"RSA-PSS SHA256", "", keyvault.PS256, rsaPSSSHA256Digest, keyvault.KeyOperationResult{ - Result: &rsaPSSSHA256ResultSig, - }, nil}, - {"RSA-PSS SHA384", "", keyvault.PS384, rsaPSSSHA384Digest, keyvault.KeyOperationResult{ - Result: &rsaPSSSHA384ResultSig, - }, nil}, - {"RSA-PSS SHA512", "", keyvault.PS512, rsaPSSSHA512Digest, keyvault.KeyOperationResult{ - Result: &rsaPSSSHA512ResultSig, - }, nil}, - // Errors - {"fail Sign", "", keyvault.RS256, rsaSHA256Digest, keyvault.KeyOperationResult{}, errTest}, - {"fail sign length", "", keyvault.ES256, p256Digest, keyvault.KeyOperationResult{ - Result: &rsaSHA256ResultSig, - }, nil}, - {"fail base64", "", keyvault.ES256, p256Digest, keyvault.KeyOperationResult{ - Result: func() *string { - v := "😎" - return &v - }(), - }, nil}, - } - for _, e := range expects { - value := base64.RawURLEncoding.EncodeToString(e.digest) - client.EXPECT().Sign(gomock.Any(), "https://my-vault.vault.azure.net/", "my-key", e.keyVersion, keyvault.KeySignParameters{ - Algorithm: e.alg, - Value: &value, - }).Return(e.result, e.err) - } - - type fields struct { - client KeyVaultClient - vaultBaseURL string - name string - version string - publicKey crypto.PublicKey - } - type args struct { - rand io.Reader - digest []byte - opts crypto.SignerOpts - } - tests := []struct { - name string - fields fields - args args - want []byte - wantErr bool - }{ - {"ok P-256", fields{client, "https://my-vault.vault.azure.net/", "my-key", "", p256}, args{ - rand.Reader, p256Digest, crypto.SHA256, - }, p256Sig, false}, - {"ok P-384", fields{client, "https://my-vault.vault.azure.net/", "my-key", "my-version", p384}, args{ - rand.Reader, p384Digest, crypto.SHA384, - }, p384Sig, false}, - {"ok P-521", fields{client, "https://my-vault.vault.azure.net/", "my-key", "my-version", p521}, args{ - rand.Reader, p521Digest, crypto.SHA512, - }, p521Sig, false}, - {"ok RSA SHA256", fields{client, "https://my-vault.vault.azure.net/", "my-key", "", rsaSHA256}, args{ - rand.Reader, rsaSHA256Digest, crypto.SHA256, - }, rsaSHA256Sig, false}, - {"ok RSA SHA384", fields{client, "https://my-vault.vault.azure.net/", "my-key", "", rsaSHA384}, args{ - rand.Reader, rsaSHA384Digest, crypto.SHA384, - }, rsaSHA384Sig, false}, - {"ok RSA SHA512", fields{client, "https://my-vault.vault.azure.net/", "my-key", "", rsaSHA512}, args{ - rand.Reader, rsaSHA512Digest, crypto.SHA512, - }, rsaSHA512Sig, false}, - {"ok RSA-PSS SHA256", fields{client, "https://my-vault.vault.azure.net/", "my-key", "", rsaPSSSHA256}, args{ - rand.Reader, rsaPSSSHA256Digest, &rsa.PSSOptions{ - SaltLength: rsa.PSSSaltLengthAuto, - Hash: crypto.SHA256, - }, - }, rsaPSSSHA256Sig, false}, - {"ok RSA-PSS SHA384", fields{client, "https://my-vault.vault.azure.net/", "my-key", "", rsaPSSSHA384}, args{ - rand.Reader, rsaPSSSHA384Digest, &rsa.PSSOptions{ - SaltLength: rsa.PSSSaltLengthEqualsHash, - Hash: crypto.SHA384, - }, - }, rsaPSSSHA384Sig, false}, - {"ok RSA-PSS SHA512", fields{client, "https://my-vault.vault.azure.net/", "my-key", "", rsaPSSSHA512}, args{ - rand.Reader, rsaPSSSHA512Digest, &rsa.PSSOptions{ - SaltLength: 64, - Hash: crypto.SHA512, - }, - }, rsaPSSSHA512Sig, false}, - {"fail Sign", fields{client, "https://my-vault.vault.azure.net/", "my-key", "", rsaSHA256}, args{ - rand.Reader, rsaSHA256Digest, crypto.SHA256, - }, nil, true}, - {"fail sign length", fields{client, "https://my-vault.vault.azure.net/", "my-key", "", p256}, args{ - rand.Reader, p256Digest, crypto.SHA256, - }, nil, true}, - {"fail base64", fields{client, "https://my-vault.vault.azure.net/", "my-key", "", p256}, args{ - rand.Reader, p256Digest, crypto.SHA256, - }, nil, true}, - {"fail RSA-PSS salt length", fields{client, "https://my-vault.vault.azure.net/", "my-key", "", rsaPSSSHA256}, args{ - rand.Reader, rsaPSSSHA256Digest, &rsa.PSSOptions{ - SaltLength: 64, - Hash: crypto.SHA256, - }, - }, nil, true}, - {"fail RSA Hash", fields{client, "https://my-vault.vault.azure.net/", "my-key", "", rsaSHA256}, args{ - rand.Reader, rsaSHA256Digest, crypto.SHA1, - }, nil, true}, - {"fail ECDSA Hash", fields{client, "https://my-vault.vault.azure.net/", "my-key", "", p256}, args{ - rand.Reader, p256Digest, crypto.MD5, - }, nil, true}, - {"fail Ed25519", fields{client, "https://my-vault.vault.azure.net/", "my-key", "", ed25519Key}, args{ - rand.Reader, []byte("message"), crypto.Hash(0), - }, nil, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - s := &Signer{ - client: tt.fields.client, - vaultBaseURL: tt.fields.vaultBaseURL, - name: tt.fields.name, - version: tt.fields.version, - publicKey: tt.fields.publicKey, - } - got, err := s.Sign(tt.args.rand, tt.args.digest, tt.args.opts) - if (err != nil) != tt.wantErr { - t.Errorf("Signer.Sign() error = %v, wantErr %v", err, tt.wantErr) - return - } - if !reflect.DeepEqual(got, tt.want) { - t.Errorf("Signer.Sign() = %v, want %v", got, tt.want) - } - }) - } -} - -func TestSigner_Sign_signWithRetry(t *testing.T) { - sign := func(kty, crv string, bits int, opts crypto.SignerOpts) (crypto.PublicKey, []byte, string, []byte) { - key, err := keyutil.GenerateSigner(kty, crv, bits) - if err != nil { - t.Fatal(err) - } - h := opts.HashFunc().New() - h.Write([]byte("random-data")) - sum := h.Sum(nil) - - var sig, resultSig []byte - if priv, ok := key.(*ecdsa.PrivateKey); ok { - r, s, err := ecdsa.Sign(rand.Reader, priv, sum) - if err != nil { - t.Fatal(err) - } - curveBits := priv.Params().BitSize - keyBytes := curveBits / 8 - if curveBits%8 > 0 { - keyBytes++ - } - rBytes := r.Bytes() - rBytesPadded := make([]byte, keyBytes) - copy(rBytesPadded[keyBytes-len(rBytes):], rBytes) - - sBytes := s.Bytes() - sBytesPadded := make([]byte, keyBytes) - copy(sBytesPadded[keyBytes-len(sBytes):], sBytes) - // nolint:gocritic - resultSig = append(rBytesPadded, sBytesPadded...) - - var b cryptobyte.Builder - b.AddASN1(asn1.SEQUENCE, func(b *cryptobyte.Builder) { - b.AddASN1BigInt(r) - b.AddASN1BigInt(s) - }) - sig, err = b.Bytes() - if err != nil { - t.Fatal(err) - } - } else { - sig, err = key.Sign(rand.Reader, sum, opts) - if err != nil { - t.Fatal(err) - } - resultSig = sig - } - - return key.Public(), h.Sum(nil), base64.RawURLEncoding.EncodeToString(resultSig), sig - } - - p256, p256Digest, p256ResultSig, p256Sig := sign("EC", "P-256", 0, crypto.SHA256) - okResult := keyvault.KeyOperationResult{ - Result: &p256ResultSig, - } - failResult := keyvault.KeyOperationResult{} - retryError := autorest.DetailedError{ - Original: &azure.RequestError{ - ServiceError: &azure.ServiceError{ - InnerError: map[string]interface{}{ - "code": "KeyNotYetValid", - }, - }, - }, - } - - client := mockClient(t) - expects := []struct { - name string - keyVersion string - alg keyvault.JSONWebKeySignatureAlgorithm - digest []byte - result keyvault.KeyOperationResult - err error - }{ - {"ok 1", "", keyvault.ES256, p256Digest, failResult, retryError}, - {"ok 2", "", keyvault.ES256, p256Digest, failResult, retryError}, - {"ok 3", "", keyvault.ES256, p256Digest, failResult, retryError}, - {"ok 4", "", keyvault.ES256, p256Digest, okResult, nil}, - {"fail", "fail-version", keyvault.ES256, p256Digest, failResult, retryError}, - {"fail", "fail-version", keyvault.ES256, p256Digest, failResult, retryError}, - {"fail", "fail-version", keyvault.ES256, p256Digest, failResult, retryError}, - {"fail", "fail-version", keyvault.ES256, p256Digest, failResult, retryError}, - } - for _, e := range expects { - value := base64.RawURLEncoding.EncodeToString(e.digest) - client.EXPECT().Sign(gomock.Any(), "https://my-vault.vault.azure.net/", "my-key", e.keyVersion, keyvault.KeySignParameters{ - Algorithm: e.alg, - Value: &value, - }).Return(e.result, e.err) - } - - type fields struct { - client KeyVaultClient - vaultBaseURL string - name string - version string - publicKey crypto.PublicKey - } - type args struct { - rand io.Reader - digest []byte - opts crypto.SignerOpts - } - tests := []struct { - name string - fields fields - args args - want []byte - wantErr bool - }{ - {"ok", fields{client, "https://my-vault.vault.azure.net/", "my-key", "", p256}, args{ - rand.Reader, p256Digest, crypto.SHA256, - }, p256Sig, false}, - {"fail", fields{client, "https://my-vault.vault.azure.net/", "my-key", "fail-version", p256}, args{ - rand.Reader, p256Digest, crypto.SHA256, - }, nil, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - s := &Signer{ - client: tt.fields.client, - vaultBaseURL: tt.fields.vaultBaseURL, - name: tt.fields.name, - version: tt.fields.version, - publicKey: tt.fields.publicKey, - } - got, err := s.Sign(tt.args.rand, tt.args.digest, tt.args.opts) - if (err != nil) != tt.wantErr { - t.Errorf("Signer.Sign() error = %v, wantErr %v", err, tt.wantErr) - return - } - if !reflect.DeepEqual(got, tt.want) { - t.Errorf("Signer.Sign() = %v, want %v", got, tt.want) - } - }) - } -} diff --git a/kms/azurekms/utils.go b/kms/azurekms/utils.go deleted file mode 100644 index d4201907..00000000 --- a/kms/azurekms/utils.go +++ /dev/null @@ -1,98 +0,0 @@ -package azurekms - -import ( - "context" - "crypto" - "encoding/json" - "net/url" - "time" - - "github.com/Azure/azure-sdk-for-go/services/keyvault/v7.1/keyvault" - "github.com/pkg/errors" - "github.com/smallstep/certificates/kms/apiv1" - "github.com/smallstep/certificates/kms/uri" - "go.step.sm/crypto/jose" -) - -// defaultContext returns the default context used in requests to azure. -func defaultContext() (context.Context, context.CancelFunc) { - return context.WithTimeout(context.Background(), 15*time.Second) -} - -// getKeyName returns the uri of the key vault key. -func getKeyName(vault, name string, bundle keyvault.KeyBundle) string { - if bundle.Key != nil && bundle.Key.Kid != nil { - sm := keyIDRegexp.FindAllStringSubmatch(*bundle.Key.Kid, 1) - if len(sm) == 1 && len(sm[0]) == 4 { - m := sm[0] - u := uri.New(Scheme, url.Values{ - "vault": []string{m[1]}, - "name": []string{m[2]}, - }) - u.RawQuery = url.Values{"version": []string{m[3]}}.Encode() - return u.String() - } - } - // Fallback to URI without id. - return uri.New(Scheme, url.Values{ - "vault": []string{vault}, - "name": []string{name}, - }).String() -} - -// parseKeyName returns the key vault, name and version from URIs like: -// -// - azurekms:vault=key-vault;name=key-name -// - azurekms:vault=key-vault;name=key-name?version=key-id -// - azurekms:vault=key-vault;name=key-name?version=key-id&hsm=true -// -// The key-id defines the version of the key, if it is not passed the latest -// version will be used. -// -// HSM can also be passed to define the protection level if this is not given in -// CreateQuery. -func parseKeyName(rawURI string, defaults DefaultOptions) (vault, name, version string, hsm bool, err error) { - var u *uri.URI - - u, err = uri.ParseWithScheme(Scheme, rawURI) - if err != nil { - return - } - if name = u.Get("name"); name == "" { - err = errors.Errorf("key uri %s is not valid: name is missing", rawURI) - return - } - if vault = u.Get("vault"); vault == "" { - if defaults.Vault == "" { - name = "" - err = errors.Errorf("key uri %s is not valid: vault is missing", rawURI) - return - } - vault = defaults.Vault - } - if u.Get("hsm") == "" { - hsm = (defaults.ProtectionLevel == apiv1.HSM) - } else { - hsm = u.GetBool("hsm") - } - - version = u.Get("version") - - return -} - -func vaultBaseURL(vault string) string { - return "https://" + vault + ".vault.azure.net/" -} - -func convertKey(key *keyvault.JSONWebKey) (crypto.PublicKey, error) { - b, err := json.Marshal(key) - if err != nil { - return nil, errors.Wrap(err, "error marshaling key") - } - var jwk jose.JSONWebKey - if err := jwk.UnmarshalJSON(b); err != nil { - return nil, errors.Wrap(err, "error unmarshaling key") - } - return jwk.Key, nil -} diff --git a/kms/azurekms/utils_test.go b/kms/azurekms/utils_test.go deleted file mode 100644 index cded50ea..00000000 --- a/kms/azurekms/utils_test.go +++ /dev/null @@ -1,96 +0,0 @@ -package azurekms - -import ( - "testing" - - "github.com/Azure/azure-sdk-for-go/services/keyvault/v7.1/keyvault" - "github.com/smallstep/certificates/kms/apiv1" -) - -func Test_getKeyName(t *testing.T) { - getBundle := func(kid string) keyvault.KeyBundle { - return keyvault.KeyBundle{ - Key: &keyvault.JSONWebKey{ - Kid: &kid, - }, - } - } - - type args struct { - vault string - name string - bundle keyvault.KeyBundle - } - tests := []struct { - name string - args args - want string - }{ - {"ok", args{"my-vault", "my-key", getBundle("https://my-vault.vault.azure.net/keys/my-key/my-version")}, "azurekms:name=my-key;vault=my-vault?version=my-version"}, - {"ok default", args{"my-vault", "my-key", getBundle("https://my-vault.foo.net/keys/my-key/my-version")}, "azurekms:name=my-key;vault=my-vault"}, - {"ok too short", args{"my-vault", "my-key", getBundle("https://my-vault.vault.azure.net/keys/my-version")}, "azurekms:name=my-key;vault=my-vault"}, - {"ok too long", args{"my-vault", "my-key", getBundle("https://my-vault.vault.azure.net/keys/my-key/my-version/sign")}, "azurekms:name=my-key;vault=my-vault"}, - {"ok nil key", args{"my-vault", "my-key", keyvault.KeyBundle{}}, "azurekms:name=my-key;vault=my-vault"}, - {"ok nil kid", args{"my-vault", "my-key", keyvault.KeyBundle{Key: &keyvault.JSONWebKey{}}}, "azurekms:name=my-key;vault=my-vault"}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - if got := getKeyName(tt.args.vault, tt.args.name, tt.args.bundle); got != tt.want { - t.Errorf("getKeyName() = %v, want %v", got, tt.want) - } - }) - } -} - -func Test_parseKeyName(t *testing.T) { - var noOptions DefaultOptions - type args struct { - rawURI string - defaults DefaultOptions - } - tests := []struct { - name string - args args - wantVault string - wantName string - wantVersion string - wantHsm bool - wantErr bool - }{ - {"ok", args{"azurekms:name=my-key;vault=my-vault?version=my-version", noOptions}, "my-vault", "my-key", "my-version", false, false}, - {"ok opaque version", args{"azurekms:name=my-key;vault=my-vault;version=my-version", noOptions}, "my-vault", "my-key", "my-version", false, false}, - {"ok no version", args{"azurekms:name=my-key;vault=my-vault", noOptions}, "my-vault", "my-key", "", false, false}, - {"ok hsm", args{"azurekms:name=my-key;vault=my-vault?hsm=true", noOptions}, "my-vault", "my-key", "", true, false}, - {"ok hsm false", args{"azurekms:name=my-key;vault=my-vault?hsm=false", noOptions}, "my-vault", "my-key", "", false, false}, - {"ok default vault", args{"azurekms:name=my-key?version=my-version", DefaultOptions{Vault: "my-vault"}}, "my-vault", "my-key", "my-version", false, false}, - {"ok default hsm", args{"azurekms:name=my-key;vault=my-vault?version=my-version", DefaultOptions{Vault: "other-vault", ProtectionLevel: apiv1.HSM}}, "my-vault", "my-key", "my-version", true, false}, - {"fail scheme", args{"azure:name=my-key;vault=my-vault", noOptions}, "", "", "", false, true}, - {"fail parse uri", args{"azurekms:name=%ZZ;vault=my-vault", noOptions}, "", "", "", false, true}, - {"fail no name", args{"azurekms:vault=my-vault", noOptions}, "", "", "", false, true}, - {"fail empty name", args{"azurekms:name=;vault=my-vault", noOptions}, "", "", "", false, true}, - {"fail no vault", args{"azurekms:name=my-key", noOptions}, "", "", "", false, true}, - {"fail empty vault", args{"azurekms:name=my-key;vault=", noOptions}, "", "", "", false, true}, - {"fail empty", args{"", noOptions}, "", "", "", false, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - gotVault, gotName, gotVersion, gotHsm, err := parseKeyName(tt.args.rawURI, tt.args.defaults) - if (err != nil) != tt.wantErr { - t.Errorf("parseKeyName() error = %v, wantErr %v", err, tt.wantErr) - return - } - if gotVault != tt.wantVault { - t.Errorf("parseKeyName() gotVault = %v, want %v", gotVault, tt.wantVault) - } - if gotName != tt.wantName { - t.Errorf("parseKeyName() gotName = %v, want %v", gotName, tt.wantName) - } - if gotVersion != tt.wantVersion { - t.Errorf("parseKeyName() gotVersion = %v, want %v", gotVersion, tt.wantVersion) - } - if gotHsm != tt.wantHsm { - t.Errorf("parseKeyName() gotHsm = %v, want %v", gotHsm, tt.wantHsm) - } - }) - } -} diff --git a/kms/cloudkms/cloudkms.go b/kms/cloudkms/cloudkms.go deleted file mode 100644 index 2f74f1ad..00000000 --- a/kms/cloudkms/cloudkms.go +++ /dev/null @@ -1,348 +0,0 @@ -package cloudkms - -import ( - "context" - "crypto" - "crypto/x509" - "log" - "strings" - "time" - - "google.golang.org/grpc/codes" - "google.golang.org/grpc/status" - - cloudkms "cloud.google.com/go/kms/apiv1" - gax "github.com/googleapis/gax-go/v2" - "github.com/pkg/errors" - "github.com/smallstep/certificates/kms/apiv1" - "github.com/smallstep/certificates/kms/uri" - "go.step.sm/crypto/pemutil" - "google.golang.org/api/option" - kmspb "google.golang.org/genproto/googleapis/cloud/kms/v1" -) - -// Scheme is the scheme used in uris. -const Scheme = "cloudkms" - -const pendingGenerationRetries = 10 - -// protectionLevelMapping maps step protection levels with cloud kms ones. -var protectionLevelMapping = map[apiv1.ProtectionLevel]kmspb.ProtectionLevel{ - apiv1.UnspecifiedProtectionLevel: kmspb.ProtectionLevel_PROTECTION_LEVEL_UNSPECIFIED, - apiv1.Software: kmspb.ProtectionLevel_SOFTWARE, - apiv1.HSM: kmspb.ProtectionLevel_HSM, -} - -// signatureAlgorithmMapping is a mapping between the step signature algorithm, -// and bits for RSA keys, with cloud kms one. -// -// Cloud KMS does not support SHA384WithRSA, SHA384WithRSAPSS, SHA384WithRSAPSS, -// ECDSAWithSHA512, and PureEd25519. -var signatureAlgorithmMapping = map[apiv1.SignatureAlgorithm]interface{}{ - apiv1.UnspecifiedSignAlgorithm: kmspb.CryptoKeyVersion_CRYPTO_KEY_VERSION_ALGORITHM_UNSPECIFIED, - apiv1.SHA256WithRSA: map[int]kmspb.CryptoKeyVersion_CryptoKeyVersionAlgorithm{ - 0: kmspb.CryptoKeyVersion_RSA_SIGN_PKCS1_3072_SHA256, - 2048: kmspb.CryptoKeyVersion_RSA_SIGN_PKCS1_2048_SHA256, - 3072: kmspb.CryptoKeyVersion_RSA_SIGN_PKCS1_3072_SHA256, - 4096: kmspb.CryptoKeyVersion_RSA_SIGN_PKCS1_4096_SHA256, - }, - apiv1.SHA512WithRSA: map[int]kmspb.CryptoKeyVersion_CryptoKeyVersionAlgorithm{ - 0: kmspb.CryptoKeyVersion_RSA_SIGN_PKCS1_4096_SHA512, - 4096: kmspb.CryptoKeyVersion_RSA_SIGN_PKCS1_4096_SHA512, - }, - apiv1.SHA256WithRSAPSS: map[int]kmspb.CryptoKeyVersion_CryptoKeyVersionAlgorithm{ - 0: kmspb.CryptoKeyVersion_RSA_SIGN_PSS_3072_SHA256, - 2048: kmspb.CryptoKeyVersion_RSA_SIGN_PSS_2048_SHA256, - 3072: kmspb.CryptoKeyVersion_RSA_SIGN_PSS_3072_SHA256, - 4096: kmspb.CryptoKeyVersion_RSA_SIGN_PSS_4096_SHA256, - }, - apiv1.SHA512WithRSAPSS: map[int]kmspb.CryptoKeyVersion_CryptoKeyVersionAlgorithm{ - 0: kmspb.CryptoKeyVersion_RSA_SIGN_PSS_4096_SHA512, - 4096: kmspb.CryptoKeyVersion_RSA_SIGN_PSS_4096_SHA512, - }, - apiv1.ECDSAWithSHA256: kmspb.CryptoKeyVersion_EC_SIGN_P256_SHA256, - apiv1.ECDSAWithSHA384: kmspb.CryptoKeyVersion_EC_SIGN_P384_SHA384, -} - -var cryptoKeyVersionMapping = map[kmspb.CryptoKeyVersion_CryptoKeyVersionAlgorithm]x509.SignatureAlgorithm{ - kmspb.CryptoKeyVersion_EC_SIGN_P256_SHA256: x509.ECDSAWithSHA256, - kmspb.CryptoKeyVersion_EC_SIGN_P384_SHA384: x509.ECDSAWithSHA384, - kmspb.CryptoKeyVersion_RSA_SIGN_PKCS1_2048_SHA256: x509.SHA256WithRSA, - kmspb.CryptoKeyVersion_RSA_SIGN_PKCS1_3072_SHA256: x509.SHA256WithRSA, - kmspb.CryptoKeyVersion_RSA_SIGN_PKCS1_4096_SHA256: x509.SHA256WithRSA, - kmspb.CryptoKeyVersion_RSA_SIGN_PKCS1_4096_SHA512: x509.SHA512WithRSA, - kmspb.CryptoKeyVersion_RSA_SIGN_PSS_2048_SHA256: x509.SHA256WithRSAPSS, - kmspb.CryptoKeyVersion_RSA_SIGN_PSS_3072_SHA256: x509.SHA256WithRSAPSS, - kmspb.CryptoKeyVersion_RSA_SIGN_PSS_4096_SHA256: x509.SHA256WithRSAPSS, - kmspb.CryptoKeyVersion_RSA_SIGN_PSS_4096_SHA512: x509.SHA512WithRSAPSS, -} - -// KeyManagementClient defines the methods on KeyManagementClient that this -// package will use. This interface will be used for unit testing. -type KeyManagementClient interface { - Close() error - GetPublicKey(context.Context, *kmspb.GetPublicKeyRequest, ...gax.CallOption) (*kmspb.PublicKey, error) - AsymmetricSign(context.Context, *kmspb.AsymmetricSignRequest, ...gax.CallOption) (*kmspb.AsymmetricSignResponse, error) - CreateCryptoKey(context.Context, *kmspb.CreateCryptoKeyRequest, ...gax.CallOption) (*kmspb.CryptoKey, error) - GetKeyRing(context.Context, *kmspb.GetKeyRingRequest, ...gax.CallOption) (*kmspb.KeyRing, error) - CreateKeyRing(context.Context, *kmspb.CreateKeyRingRequest, ...gax.CallOption) (*kmspb.KeyRing, error) - CreateCryptoKeyVersion(ctx context.Context, req *kmspb.CreateCryptoKeyVersionRequest, opts ...gax.CallOption) (*kmspb.CryptoKeyVersion, error) -} - -var newKeyManagementClient = func(ctx context.Context, opts ...option.ClientOption) (KeyManagementClient, error) { - return cloudkms.NewKeyManagementClient(ctx, opts...) -} - -// CloudKMS implements a KMS using Google's Cloud apiv1. -type CloudKMS struct { - client KeyManagementClient -} - -// New creates a new CloudKMS configured with a new client. -func New(ctx context.Context, opts apiv1.Options) (*CloudKMS, error) { - var cloudOpts []option.ClientOption - - if opts.URI != "" { - u, err := uri.ParseWithScheme(Scheme, opts.URI) - if err != nil { - return nil, err - } - if f := u.Get("credentials-file"); f != "" { - cloudOpts = append(cloudOpts, option.WithCredentialsFile(f)) - } - } - - // Deprecated way to set configuration parameters. - if opts.CredentialsFile != "" { - cloudOpts = append(cloudOpts, option.WithCredentialsFile(opts.CredentialsFile)) - } - - client, err := newKeyManagementClient(ctx, cloudOpts...) - if err != nil { - return nil, err - } - - return &CloudKMS{ - client: client, - }, nil -} - -func init() { - apiv1.Register(apiv1.CloudKMS, func(ctx context.Context, opts apiv1.Options) (apiv1.KeyManager, error) { - return New(ctx, opts) - }) -} - -// NewCloudKMS creates a CloudKMS with a given client. -func NewCloudKMS(client KeyManagementClient) *CloudKMS { - return &CloudKMS{ - client: client, - } -} - -// Close closes the connection of the Cloud KMS client. -func (k *CloudKMS) Close() error { - if err := k.client.Close(); err != nil { - return errors.Wrap(err, "cloudKMS Close failed") - } - return nil -} - -// CreateSigner returns a new cloudkms signer configured with the given signing -// key name. -func (k *CloudKMS) CreateSigner(req *apiv1.CreateSignerRequest) (crypto.Signer, error) { - if req.SigningKey == "" { - return nil, errors.New("signing key cannot be empty") - } - return NewSigner(k.client, req.SigningKey) -} - -// CreateKey creates in Google's Cloud KMS a new asymmetric key for signing. -func (k *CloudKMS) CreateKey(req *apiv1.CreateKeyRequest) (*apiv1.CreateKeyResponse, error) { - if req.Name == "" { - return nil, errors.New("createKeyRequest 'name' cannot be empty") - } - - protectionLevel, ok := protectionLevelMapping[req.ProtectionLevel] - if !ok { - return nil, errors.Errorf("cloudKMS does not support protection level '%s'", req.ProtectionLevel) - } - - var signatureAlgorithm kmspb.CryptoKeyVersion_CryptoKeyVersionAlgorithm - v, ok := signatureAlgorithmMapping[req.SignatureAlgorithm] - if !ok { - return nil, errors.Errorf("cloudKMS does not support signature algorithm '%s'", req.SignatureAlgorithm) - } - switch v := v.(type) { - case kmspb.CryptoKeyVersion_CryptoKeyVersionAlgorithm: - signatureAlgorithm = v - case map[int]kmspb.CryptoKeyVersion_CryptoKeyVersionAlgorithm: - if signatureAlgorithm, ok = v[req.Bits]; !ok { - return nil, errors.Errorf("cloudKMS does not support signature algorithm '%s' with '%d' bits", req.SignatureAlgorithm, req.Bits) - } - default: - return nil, errors.Errorf("unexpected error: this should not happen") - } - - var crytoKeyName string - - // Split `projects/PROJECT_ID/locations/global/keyRings/RING_ID/cryptoKeys/KEY_ID` - // to `projects/PROJECT_ID/locations/global/keyRings/RING_ID` and `KEY_ID`. - keyRing, keyID := Parent(req.Name) - if err := k.createKeyRingIfNeeded(keyRing); err != nil { - return nil, err - } - - ctx, cancel := defaultContext() - defer cancel() - - // Create private key in CloudKMS. - response, err := k.client.CreateCryptoKey(ctx, &kmspb.CreateCryptoKeyRequest{ - Parent: keyRing, - CryptoKeyId: keyID, - CryptoKey: &kmspb.CryptoKey{ - Purpose: kmspb.CryptoKey_ASYMMETRIC_SIGN, - VersionTemplate: &kmspb.CryptoKeyVersionTemplate{ - ProtectionLevel: protectionLevel, - Algorithm: signatureAlgorithm, - }, - }, - }) - if err != nil { - if status.Code(err) != codes.AlreadyExists { - return nil, errors.Wrap(err, "cloudKMS CreateCryptoKey failed") - } - // Create a new version if the key already exists. - // - // Note that it will have the same purpose, protection level and - // algorithm than as previous one. - req := &kmspb.CreateCryptoKeyVersionRequest{ - Parent: req.Name, - CryptoKeyVersion: &kmspb.CryptoKeyVersion{ - State: kmspb.CryptoKeyVersion_ENABLED, - }, - } - response, err := k.client.CreateCryptoKeyVersion(ctx, req) - if err != nil { - return nil, errors.Wrap(err, "cloudKMS CreateCryptoKeyVersion failed") - } - crytoKeyName = response.Name - } else { - crytoKeyName = response.Name + "/cryptoKeyVersions/1" - } - - // Sleep deterministically to avoid retries because of PENDING_GENERATING. - // One second is often enough. - if protectionLevel == kmspb.ProtectionLevel_HSM { - time.Sleep(1 * time.Second) - } - - // Retrieve public key to add it to the response. - pk, err := k.GetPublicKey(&apiv1.GetPublicKeyRequest{ - Name: crytoKeyName, - }) - if err != nil { - return nil, errors.Wrap(err, "cloudKMS GetPublicKey failed") - } - - return &apiv1.CreateKeyResponse{ - Name: crytoKeyName, - PublicKey: pk, - CreateSignerRequest: apiv1.CreateSignerRequest{ - SigningKey: crytoKeyName, - }, - }, nil -} - -func (k *CloudKMS) createKeyRingIfNeeded(name string) error { - ctx, cancel := defaultContext() - defer cancel() - - _, err := k.client.GetKeyRing(ctx, &kmspb.GetKeyRingRequest{ - Name: name, - }) - if err == nil { - return nil - } - - parent, child := Parent(name) - _, err = k.client.CreateKeyRing(ctx, &kmspb.CreateKeyRingRequest{ - Parent: parent, - KeyRingId: child, - }) - if err != nil && status.Code(err) != codes.AlreadyExists { - return errors.Wrap(err, "cloudKMS CreateKeyRing failed") - } - - return nil -} - -// GetPublicKey gets from Google's Cloud KMS a public key by name. Key names -// follow the pattern: -// -// projects/([^/]+)/locations/([a-zA-Z0-9_-]{1,63})/keyRings/([a-zA-Z0-9_-]{1,63})/cryptoKeys/([a-zA-Z0-9_-]{1,63})/cryptoKeyVersions/([a-zA-Z0-9_-]{1,63}) -func (k *CloudKMS) GetPublicKey(req *apiv1.GetPublicKeyRequest) (crypto.PublicKey, error) { - if req.Name == "" { - return nil, errors.New("createKeyRequest 'name' cannot be empty") - } - - response, err := k.getPublicKeyWithRetries(req.Name, pendingGenerationRetries) - if err != nil { - return nil, errors.Wrap(err, "cloudKMS GetPublicKey failed") - } - - pk, err := pemutil.ParseKey([]byte(response.Pem)) - if err != nil { - return nil, err - } - - return pk, nil -} - -// getPublicKeyWithRetries retries the request if the error is -// FailedPrecondition, caused because the key is in the PENDING_GENERATION -// status. -func (k *CloudKMS) getPublicKeyWithRetries(name string, retries int) (response *kmspb.PublicKey, err error) { - workFn := func() (*kmspb.PublicKey, error) { - ctx, cancel := defaultContext() - defer cancel() - return k.client.GetPublicKey(ctx, &kmspb.GetPublicKeyRequest{ - Name: name, - }) - } - for i := 0; i < retries; i++ { - if response, err = workFn(); err == nil { - return - } - if status.Code(err) == codes.FailedPrecondition { - log.Println("Waiting for key generation ...") - time.Sleep(time.Duration(i+1) * time.Second) - continue - } - } - return -} - -func defaultContext() (context.Context, context.CancelFunc) { - return context.WithTimeout(context.Background(), 15*time.Second) -} - -// Parent splits a string in the format `key/value/key2/value2` in a parent and -// child, for the previous string it will return `key/value` and `value2`. -func Parent(name string) (string, string) { - a, b := parent(name) - a, _ = parent(a) - return a, b -} - -func parent(name string) (string, string) { - i := strings.LastIndex(name, "/") - switch i { - case -1: - return "", name - case 0: - return "", name[i+1:] - default: - return name[:i], name[i+1:] - } -} diff --git a/kms/cloudkms/cloudkms_test.go b/kms/cloudkms/cloudkms_test.go deleted file mode 100644 index 814e3638..00000000 --- a/kms/cloudkms/cloudkms_test.go +++ /dev/null @@ -1,464 +0,0 @@ -package cloudkms - -import ( - "context" - "crypto" - "fmt" - "os" - "reflect" - "testing" - - gax "github.com/googleapis/gax-go/v2" - "github.com/smallstep/certificates/kms/apiv1" - "go.step.sm/crypto/pemutil" - "google.golang.org/api/option" - kmspb "google.golang.org/genproto/googleapis/cloud/kms/v1" - "google.golang.org/grpc/codes" - "google.golang.org/grpc/status" -) - -func TestParent(t *testing.T) { - type args struct { - name string - } - tests := []struct { - name string - args args - want string - want1 string - }{ - {"zero", args{"child"}, "", "child"}, - {"one", args{"parent/child"}, "", "child"}, - {"two", args{"grandparent/parent/child"}, "grandparent", "child"}, - {"three", args{"great-grandparent/grandparent/parent/child"}, "great-grandparent/grandparent", "child"}, - {"empty", args{""}, "", ""}, - {"root", args{"/"}, "", ""}, - {"child", args{"/child"}, "", "child"}, - {"parent", args{"parent/"}, "", ""}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - got, got1 := Parent(tt.args.name) - if got != tt.want { - t.Errorf("Parent() got = %v, want %v", got, tt.want) - } - if got1 != tt.want1 { - t.Errorf("Parent() got1 = %v, want %v", got1, tt.want1) - } - }) - } -} - -func TestNew(t *testing.T) { - tmp := newKeyManagementClient - t.Cleanup(func() { - newKeyManagementClient = tmp - }) - newKeyManagementClient = func(ctx context.Context, opts ...option.ClientOption) (KeyManagementClient, error) { - if len(opts) > 0 { - return nil, fmt.Errorf("test error") - } - return &MockClient{}, nil - } - - type args struct { - ctx context.Context - opts apiv1.Options - } - tests := []struct { - name string - args args - want *CloudKMS - wantErr bool - }{ - {"ok", args{context.Background(), apiv1.Options{}}, &CloudKMS{client: &MockClient{}}, false}, - {"ok with uri", args{context.Background(), apiv1.Options{URI: "cloudkms:"}}, &CloudKMS{client: &MockClient{}}, false}, - {"fail credentials", args{context.Background(), apiv1.Options{CredentialsFile: "testdata/missing"}}, nil, true}, - {"fail with uri", args{context.Background(), apiv1.Options{URI: "cloudkms:credentials-file=testdata/missing"}}, nil, true}, - {"fail schema", args{context.Background(), apiv1.Options{URI: "pkcs11:"}}, nil, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - got, err := New(tt.args.ctx, tt.args.opts) - if (err != nil) != tt.wantErr { - t.Errorf("New() error = %v, wantErr %v", err, tt.wantErr) - return - } - if !reflect.DeepEqual(got, tt.want) { - t.Errorf("New() = %v, want %v", got, tt.want) - } - }) - } -} - -func TestNew_real(t *testing.T) { - type args struct { - ctx context.Context - opts apiv1.Options - } - tests := []struct { - name string - args args - want *CloudKMS - wantErr bool - }{ - {"fail credentials", args{context.Background(), apiv1.Options{CredentialsFile: "testdata/missing"}}, nil, true}, - {"fail with uri", args{context.Background(), apiv1.Options{URI: "cloudkms:credentials-file=testdata/missing"}}, nil, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - got, err := New(tt.args.ctx, tt.args.opts) - if (err != nil) != tt.wantErr { - t.Errorf("New() error = %v, wantErr %v", err, tt.wantErr) - return - } - if !reflect.DeepEqual(got, tt.want) { - t.Errorf("New() = %v, want %v", got, tt.want) - } - }) - } -} - -func TestNewCloudKMS(t *testing.T) { - type args struct { - client KeyManagementClient - } - tests := []struct { - name string - args args - want *CloudKMS - }{ - {"ok", args{&MockClient{}}, &CloudKMS{&MockClient{}}}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - if got := NewCloudKMS(tt.args.client); !reflect.DeepEqual(got, tt.want) { - t.Errorf("NewCloudKMS() = %v, want %v", got, tt.want) - } - }) - } -} - -func TestCloudKMS_Close(t *testing.T) { - type fields struct { - client KeyManagementClient - } - tests := []struct { - name string - fields fields - wantErr bool - }{ - {"ok", fields{&MockClient{close: func() error { return nil }}}, false}, - {"fail", fields{&MockClient{close: func() error { return fmt.Errorf("an error") }}}, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - k := &CloudKMS{ - client: tt.fields.client, - } - if err := k.Close(); (err != nil) != tt.wantErr { - t.Errorf("CloudKMS.Close() error = %v, wantErr %v", err, tt.wantErr) - } - }) - } -} - -func TestCloudKMS_CreateSigner(t *testing.T) { - keyName := "projects/p/locations/l/keyRings/k/cryptoKeys/c/cryptoKeyVersions/1" - pemBytes, err := os.ReadFile("testdata/pub.pem") - if err != nil { - t.Fatal(err) - } - pk, err := pemutil.ParseKey(pemBytes) - if err != nil { - t.Fatal(err) - } - - type fields struct { - client KeyManagementClient - } - type args struct { - req *apiv1.CreateSignerRequest - } - tests := []struct { - name string - fields fields - args args - want crypto.Signer - wantErr bool - }{ - {"ok", fields{&MockClient{ - getPublicKey: func(_ context.Context, _ *kmspb.GetPublicKeyRequest, _ ...gax.CallOption) (*kmspb.PublicKey, error) { - return &kmspb.PublicKey{Pem: string(pemBytes)}, nil - }, - }}, args{&apiv1.CreateSignerRequest{SigningKey: keyName}}, &Signer{client: &MockClient{}, signingKey: keyName, publicKey: pk}, false}, - {"fail", fields{&MockClient{ - getPublicKey: func(_ context.Context, _ *kmspb.GetPublicKeyRequest, _ ...gax.CallOption) (*kmspb.PublicKey, error) { - return nil, fmt.Errorf("test error") - }, - }}, args{&apiv1.CreateSignerRequest{SigningKey: ""}}, nil, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - k := &CloudKMS{ - client: tt.fields.client, - } - got, err := k.CreateSigner(tt.args.req) - if (err != nil) != tt.wantErr { - t.Errorf("CloudKMS.CreateSigner() error = %v, wantErr %v", err, tt.wantErr) - return - } - if signer, ok := got.(*Signer); ok { - signer.client = &MockClient{} - } - if !reflect.DeepEqual(got, tt.want) { - t.Errorf("CloudKMS.CreateSigner() = %v, want %v", got, tt.want) - } - }) - } -} - -func TestCloudKMS_CreateKey(t *testing.T) { - keyName := "projects/p/locations/l/keyRings/k/cryptoKeys/c" - testError := fmt.Errorf("an error") - alreadyExists := status.Error(codes.AlreadyExists, "already exists") - - pemBytes, err := os.ReadFile("testdata/pub.pem") - if err != nil { - t.Fatal(err) - } - pk, err := pemutil.ParseKey(pemBytes) - if err != nil { - t.Fatal(err) - } - - var retries int - type fields struct { - client KeyManagementClient - } - type args struct { - req *apiv1.CreateKeyRequest - } - tests := []struct { - name string - fields fields - args args - want *apiv1.CreateKeyResponse - wantErr bool - }{ - {"ok", fields{ - &MockClient{ - getKeyRing: func(_ context.Context, _ *kmspb.GetKeyRingRequest, _ ...gax.CallOption) (*kmspb.KeyRing, error) { - return &kmspb.KeyRing{}, nil - }, - createCryptoKey: func(_ context.Context, _ *kmspb.CreateCryptoKeyRequest, _ ...gax.CallOption) (*kmspb.CryptoKey, error) { - return &kmspb.CryptoKey{Name: keyName}, nil - }, - getPublicKey: func(_ context.Context, _ *kmspb.GetPublicKeyRequest, _ ...gax.CallOption) (*kmspb.PublicKey, error) { - return &kmspb.PublicKey{Pem: string(pemBytes)}, nil - }, - }}, - args{&apiv1.CreateKeyRequest{Name: keyName, ProtectionLevel: apiv1.HSM, SignatureAlgorithm: apiv1.ECDSAWithSHA256}}, - &apiv1.CreateKeyResponse{Name: keyName + "/cryptoKeyVersions/1", PublicKey: pk, CreateSignerRequest: apiv1.CreateSignerRequest{SigningKey: keyName + "/cryptoKeyVersions/1"}}, false}, - {"ok new key ring", fields{ - &MockClient{ - getKeyRing: func(_ context.Context, _ *kmspb.GetKeyRingRequest, _ ...gax.CallOption) (*kmspb.KeyRing, error) { - return nil, testError - }, - createKeyRing: func(_ context.Context, _ *kmspb.CreateKeyRingRequest, _ ...gax.CallOption) (*kmspb.KeyRing, error) { - return nil, alreadyExists - }, - createCryptoKey: func(_ context.Context, _ *kmspb.CreateCryptoKeyRequest, _ ...gax.CallOption) (*kmspb.CryptoKey, error) { - return &kmspb.CryptoKey{Name: keyName}, nil - }, - getPublicKey: func(_ context.Context, _ *kmspb.GetPublicKeyRequest, _ ...gax.CallOption) (*kmspb.PublicKey, error) { - return &kmspb.PublicKey{Pem: string(pemBytes)}, nil - }, - }}, - args{&apiv1.CreateKeyRequest{Name: keyName, ProtectionLevel: apiv1.Software, SignatureAlgorithm: apiv1.SHA256WithRSA, Bits: 3072}}, - &apiv1.CreateKeyResponse{Name: keyName + "/cryptoKeyVersions/1", PublicKey: pk, CreateSignerRequest: apiv1.CreateSignerRequest{SigningKey: keyName + "/cryptoKeyVersions/1"}}, false}, - {"ok new key version", fields{ - &MockClient{ - getKeyRing: func(_ context.Context, _ *kmspb.GetKeyRingRequest, _ ...gax.CallOption) (*kmspb.KeyRing, error) { - return &kmspb.KeyRing{}, nil - }, - createCryptoKey: func(_ context.Context, _ *kmspb.CreateCryptoKeyRequest, _ ...gax.CallOption) (*kmspb.CryptoKey, error) { - return nil, alreadyExists - }, - createCryptoKeyVersion: func(_ context.Context, _ *kmspb.CreateCryptoKeyVersionRequest, _ ...gax.CallOption) (*kmspb.CryptoKeyVersion, error) { - return &kmspb.CryptoKeyVersion{Name: keyName + "/cryptoKeyVersions/2"}, nil - }, - getPublicKey: func(_ context.Context, _ *kmspb.GetPublicKeyRequest, _ ...gax.CallOption) (*kmspb.PublicKey, error) { - return &kmspb.PublicKey{Pem: string(pemBytes)}, nil - }, - }}, - args{&apiv1.CreateKeyRequest{Name: keyName, ProtectionLevel: apiv1.HSM, SignatureAlgorithm: apiv1.ECDSAWithSHA256}}, - &apiv1.CreateKeyResponse{Name: keyName + "/cryptoKeyVersions/2", PublicKey: pk, CreateSignerRequest: apiv1.CreateSignerRequest{SigningKey: keyName + "/cryptoKeyVersions/2"}}, false}, - {"ok with retries", fields{ - &MockClient{ - getKeyRing: func(_ context.Context, _ *kmspb.GetKeyRingRequest, _ ...gax.CallOption) (*kmspb.KeyRing, error) { - return &kmspb.KeyRing{}, nil - }, - createCryptoKey: func(_ context.Context, _ *kmspb.CreateCryptoKeyRequest, _ ...gax.CallOption) (*kmspb.CryptoKey, error) { - return &kmspb.CryptoKey{Name: keyName}, nil - }, - getPublicKey: func(_ context.Context, _ *kmspb.GetPublicKeyRequest, _ ...gax.CallOption) (*kmspb.PublicKey, error) { - if retries != 2 { - retries++ - return nil, status.Error(codes.FailedPrecondition, "key is not enabled, current state is: PENDING_GENERATION") - } - return &kmspb.PublicKey{Pem: string(pemBytes)}, nil - }, - }}, - args{&apiv1.CreateKeyRequest{Name: keyName, ProtectionLevel: apiv1.HSM, SignatureAlgorithm: apiv1.ECDSAWithSHA256}}, - &apiv1.CreateKeyResponse{Name: keyName + "/cryptoKeyVersions/1", PublicKey: pk, CreateSignerRequest: apiv1.CreateSignerRequest{SigningKey: keyName + "/cryptoKeyVersions/1"}}, false}, - {"fail name", fields{&MockClient{}}, args{&apiv1.CreateKeyRequest{}}, nil, true}, - {"fail protection level", fields{&MockClient{}}, args{&apiv1.CreateKeyRequest{Name: keyName, ProtectionLevel: apiv1.ProtectionLevel(100)}}, nil, true}, - {"fail signature algorithm", fields{&MockClient{}}, args{&apiv1.CreateKeyRequest{Name: keyName, ProtectionLevel: apiv1.Software, SignatureAlgorithm: apiv1.SignatureAlgorithm(100)}}, nil, true}, - {"fail number of bits", fields{&MockClient{}}, args{&apiv1.CreateKeyRequest{Name: keyName, ProtectionLevel: apiv1.Software, SignatureAlgorithm: apiv1.SHA256WithRSA, Bits: 1024}}, - nil, true}, - {"fail create key ring", fields{ - &MockClient{ - getKeyRing: func(_ context.Context, _ *kmspb.GetKeyRingRequest, _ ...gax.CallOption) (*kmspb.KeyRing, error) { - return nil, testError - }, - createKeyRing: func(_ context.Context, _ *kmspb.CreateKeyRingRequest, _ ...gax.CallOption) (*kmspb.KeyRing, error) { - return nil, testError - }, - }}, - args{&apiv1.CreateKeyRequest{Name: keyName, ProtectionLevel: apiv1.HSM, SignatureAlgorithm: apiv1.ECDSAWithSHA256}}, - nil, true}, - {"fail create key", fields{ - &MockClient{ - getKeyRing: func(_ context.Context, _ *kmspb.GetKeyRingRequest, _ ...gax.CallOption) (*kmspb.KeyRing, error) { - return &kmspb.KeyRing{}, nil - }, - createCryptoKey: func(_ context.Context, _ *kmspb.CreateCryptoKeyRequest, _ ...gax.CallOption) (*kmspb.CryptoKey, error) { - return nil, testError - }, - }}, - args{&apiv1.CreateKeyRequest{Name: keyName, ProtectionLevel: apiv1.HSM, SignatureAlgorithm: apiv1.ECDSAWithSHA256}}, - nil, true}, - {"fail create key version", fields{ - &MockClient{ - getKeyRing: func(_ context.Context, _ *kmspb.GetKeyRingRequest, _ ...gax.CallOption) (*kmspb.KeyRing, error) { - return &kmspb.KeyRing{}, nil - }, - createCryptoKey: func(_ context.Context, _ *kmspb.CreateCryptoKeyRequest, _ ...gax.CallOption) (*kmspb.CryptoKey, error) { - return nil, alreadyExists - }, - createCryptoKeyVersion: func(_ context.Context, _ *kmspb.CreateCryptoKeyVersionRequest, _ ...gax.CallOption) (*kmspb.CryptoKeyVersion, error) { - return nil, testError - }, - }}, - args{&apiv1.CreateKeyRequest{Name: keyName, ProtectionLevel: apiv1.HSM, SignatureAlgorithm: apiv1.ECDSAWithSHA256}}, - nil, true}, - {"fail get public key", fields{ - &MockClient{ - getKeyRing: func(_ context.Context, _ *kmspb.GetKeyRingRequest, _ ...gax.CallOption) (*kmspb.KeyRing, error) { - return &kmspb.KeyRing{}, nil - }, - createCryptoKey: func(_ context.Context, _ *kmspb.CreateCryptoKeyRequest, _ ...gax.CallOption) (*kmspb.CryptoKey, error) { - return &kmspb.CryptoKey{Name: keyName}, nil - }, - getPublicKey: func(_ context.Context, _ *kmspb.GetPublicKeyRequest, _ ...gax.CallOption) (*kmspb.PublicKey, error) { - return nil, testError - }, - }}, - args{&apiv1.CreateKeyRequest{Name: keyName, ProtectionLevel: apiv1.HSM, SignatureAlgorithm: apiv1.ECDSAWithSHA256}}, - nil, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - k := &CloudKMS{ - client: tt.fields.client, - } - got, err := k.CreateKey(tt.args.req) - if (err != nil) != tt.wantErr { - t.Errorf("CloudKMS.CreateKey() error = %v, wantErr %v", err, tt.wantErr) - return - } - if !reflect.DeepEqual(got, tt.want) { - t.Errorf("CloudKMS.CreateKey() = %v, want %v", got, tt.want) - } - }) - } -} - -func TestCloudKMS_GetPublicKey(t *testing.T) { - keyName := "projects/p/locations/l/keyRings/k/cryptoKeys/c/cryptoKeyVersions/1" - testError := fmt.Errorf("an error") - - pemBytes, err := os.ReadFile("testdata/pub.pem") - if err != nil { - t.Fatal(err) - } - pk, err := pemutil.ParseKey(pemBytes) - if err != nil { - t.Fatal(err) - } - - var retries int - type fields struct { - client KeyManagementClient - } - type args struct { - req *apiv1.GetPublicKeyRequest - } - tests := []struct { - name string - fields fields - args args - want crypto.PublicKey - wantErr bool - }{ - {"ok", fields{ - &MockClient{ - getPublicKey: func(_ context.Context, _ *kmspb.GetPublicKeyRequest, _ ...gax.CallOption) (*kmspb.PublicKey, error) { - return &kmspb.PublicKey{Pem: string(pemBytes)}, nil - }, - }}, - args{&apiv1.GetPublicKeyRequest{Name: keyName}}, pk, false}, - {"ok with retries", fields{ - &MockClient{ - getPublicKey: func(_ context.Context, _ *kmspb.GetPublicKeyRequest, _ ...gax.CallOption) (*kmspb.PublicKey, error) { - if retries != 2 { - retries++ - return nil, status.Error(codes.FailedPrecondition, "key is not enabled, current state is: PENDING_GENERATION") - } - return &kmspb.PublicKey{Pem: string(pemBytes)}, nil - }, - }}, - args{&apiv1.GetPublicKeyRequest{Name: keyName}}, pk, false}, - {"fail name", fields{&MockClient{}}, args{&apiv1.GetPublicKeyRequest{}}, nil, true}, - {"fail get public key", fields{ - &MockClient{ - getPublicKey: func(_ context.Context, _ *kmspb.GetPublicKeyRequest, _ ...gax.CallOption) (*kmspb.PublicKey, error) { - return nil, testError - }, - }}, - args{&apiv1.GetPublicKeyRequest{Name: keyName}}, nil, true}, - {"fail parse pem", fields{ - &MockClient{ - getPublicKey: func(_ context.Context, _ *kmspb.GetPublicKeyRequest, _ ...gax.CallOption) (*kmspb.PublicKey, error) { - return &kmspb.PublicKey{Pem: string("bad pem")}, nil - }, - }}, - args{&apiv1.GetPublicKeyRequest{Name: keyName}}, nil, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - k := &CloudKMS{ - client: tt.fields.client, - } - got, err := k.GetPublicKey(tt.args.req) - if (err != nil) != tt.wantErr { - t.Errorf("CloudKMS.GetPublicKey() error = %v, wantErr %v", err, tt.wantErr) - return - } - if !reflect.DeepEqual(got, tt.want) { - t.Errorf("CloudKMS.GetPublicKey() = %v, want %v", got, tt.want) - } - }) - } -} diff --git a/kms/cloudkms/mock_test.go b/kms/cloudkms/mock_test.go deleted file mode 100644 index 7617bd85..00000000 --- a/kms/cloudkms/mock_test.go +++ /dev/null @@ -1,46 +0,0 @@ -package cloudkms - -import ( - "context" - - gax "github.com/googleapis/gax-go/v2" - kmspb "google.golang.org/genproto/googleapis/cloud/kms/v1" -) - -type MockClient struct { - close func() error - getPublicKey func(context.Context, *kmspb.GetPublicKeyRequest, ...gax.CallOption) (*kmspb.PublicKey, error) - asymmetricSign func(context.Context, *kmspb.AsymmetricSignRequest, ...gax.CallOption) (*kmspb.AsymmetricSignResponse, error) - createCryptoKey func(context.Context, *kmspb.CreateCryptoKeyRequest, ...gax.CallOption) (*kmspb.CryptoKey, error) - getKeyRing func(context.Context, *kmspb.GetKeyRingRequest, ...gax.CallOption) (*kmspb.KeyRing, error) - createKeyRing func(context.Context, *kmspb.CreateKeyRingRequest, ...gax.CallOption) (*kmspb.KeyRing, error) - createCryptoKeyVersion func(context.Context, *kmspb.CreateCryptoKeyVersionRequest, ...gax.CallOption) (*kmspb.CryptoKeyVersion, error) -} - -func (m *MockClient) Close() error { - return m.close() -} - -func (m *MockClient) GetPublicKey(ctx context.Context, req *kmspb.GetPublicKeyRequest, opts ...gax.CallOption) (*kmspb.PublicKey, error) { - return m.getPublicKey(ctx, req, opts...) -} - -func (m *MockClient) AsymmetricSign(ctx context.Context, req *kmspb.AsymmetricSignRequest, opts ...gax.CallOption) (*kmspb.AsymmetricSignResponse, error) { - return m.asymmetricSign(ctx, req, opts...) -} - -func (m *MockClient) CreateCryptoKey(ctx context.Context, req *kmspb.CreateCryptoKeyRequest, opts ...gax.CallOption) (*kmspb.CryptoKey, error) { - return m.createCryptoKey(ctx, req, opts...) -} - -func (m *MockClient) GetKeyRing(ctx context.Context, req *kmspb.GetKeyRingRequest, opts ...gax.CallOption) (*kmspb.KeyRing, error) { - return m.getKeyRing(ctx, req, opts...) -} - -func (m *MockClient) CreateKeyRing(ctx context.Context, req *kmspb.CreateKeyRingRequest, opts ...gax.CallOption) (*kmspb.KeyRing, error) { - return m.createKeyRing(ctx, req, opts...) -} - -func (m *MockClient) CreateCryptoKeyVersion(ctx context.Context, req *kmspb.CreateCryptoKeyVersionRequest, opts ...gax.CallOption) (*kmspb.CryptoKeyVersion, error) { - return m.createCryptoKeyVersion(ctx, req, opts...) -} diff --git a/kms/cloudkms/signer.go b/kms/cloudkms/signer.go deleted file mode 100644 index 5a5443cf..00000000 --- a/kms/cloudkms/signer.go +++ /dev/null @@ -1,95 +0,0 @@ -package cloudkms - -import ( - "crypto" - "crypto/x509" - "io" - - "github.com/pkg/errors" - "go.step.sm/crypto/pemutil" - kmspb "google.golang.org/genproto/googleapis/cloud/kms/v1" -) - -// Signer implements a crypto.Signer using Google's Cloud KMS. -type Signer struct { - client KeyManagementClient - signingKey string - algorithm x509.SignatureAlgorithm - publicKey crypto.PublicKey -} - -// NewSigner creates a new crypto.Signer the given CloudKMS signing key. -func NewSigner(c KeyManagementClient, signingKey string) (*Signer, error) { - // Make sure that the key exists. - signer := &Signer{ - client: c, - signingKey: signingKey, - } - if err := signer.preloadKey(signingKey); err != nil { - return nil, err - } - - return signer, nil -} - -func (s *Signer) preloadKey(signingKey string) error { - ctx, cancel := defaultContext() - defer cancel() - - response, err := s.client.GetPublicKey(ctx, &kmspb.GetPublicKeyRequest{ - Name: signingKey, - }) - if err != nil { - return errors.Wrap(err, "cloudKMS GetPublicKey failed") - } - s.algorithm = cryptoKeyVersionMapping[response.Algorithm] - s.publicKey, err = pemutil.ParseKey([]byte(response.Pem)) - return err -} - -// Public returns the public key of this signer or an error. -func (s *Signer) Public() crypto.PublicKey { - return s.publicKey -} - -// Sign signs digest with the private key stored in Google's Cloud KMS. -func (s *Signer) Sign(rand io.Reader, digest []byte, opts crypto.SignerOpts) ([]byte, error) { - req := &kmspb.AsymmetricSignRequest{ - Name: s.signingKey, - Digest: &kmspb.Digest{}, - } - - switch h := opts.HashFunc(); h { - case crypto.SHA256: - req.Digest.Digest = &kmspb.Digest_Sha256{ - Sha256: digest, - } - case crypto.SHA384: - req.Digest.Digest = &kmspb.Digest_Sha384{ - Sha384: digest, - } - case crypto.SHA512: - req.Digest.Digest = &kmspb.Digest_Sha512{ - Sha512: digest, - } - default: - return nil, errors.Errorf("unsupported hash function %v", h) - } - - ctx, cancel := defaultContext() - defer cancel() - - response, err := s.client.AsymmetricSign(ctx, req) - if err != nil { - return nil, errors.Wrap(err, "cloudKMS AsymmetricSign failed") - } - - return response.Signature, nil -} - -// SignatureAlgorithm returns the algorithm that must be specified in a -// certificate to sign. This is specially important to distinguish RSA and -// RSAPSS schemas. -func (s *Signer) SignatureAlgorithm() x509.SignatureAlgorithm { - return s.algorithm -} diff --git a/kms/cloudkms/signer_test.go b/kms/cloudkms/signer_test.go deleted file mode 100644 index 22d1fe19..00000000 --- a/kms/cloudkms/signer_test.go +++ /dev/null @@ -1,235 +0,0 @@ -package cloudkms - -import ( - "context" - "crypto" - "crypto/rand" - "crypto/x509" - "fmt" - "io" - "os" - "reflect" - "testing" - - gax "github.com/googleapis/gax-go/v2" - "go.step.sm/crypto/pemutil" - kmspb "google.golang.org/genproto/googleapis/cloud/kms/v1" -) - -func Test_newSigner(t *testing.T) { - pemBytes, err := os.ReadFile("testdata/pub.pem") - if err != nil { - t.Fatal(err) - } - pk, err := pemutil.ParseKey(pemBytes) - if err != nil { - t.Fatal(err) - } - - type args struct { - c KeyManagementClient - signingKey string - } - tests := []struct { - name string - args args - want *Signer - wantErr bool - }{ - {"ok", args{&MockClient{ - getPublicKey: func(_ context.Context, _ *kmspb.GetPublicKeyRequest, _ ...gax.CallOption) (*kmspb.PublicKey, error) { - return &kmspb.PublicKey{Pem: string(pemBytes)}, nil - }, - }, "signingKey"}, &Signer{client: &MockClient{}, signingKey: "signingKey", publicKey: pk}, false}, - {"fail get public key", args{&MockClient{ - getPublicKey: func(_ context.Context, _ *kmspb.GetPublicKeyRequest, _ ...gax.CallOption) (*kmspb.PublicKey, error) { - return nil, fmt.Errorf("an error") - }, - }, "signingKey"}, nil, true}, - {"fail parse pem", args{&MockClient{ - getPublicKey: func(_ context.Context, _ *kmspb.GetPublicKeyRequest, _ ...gax.CallOption) (*kmspb.PublicKey, error) { - return &kmspb.PublicKey{Pem: string("bad pem")}, nil - }, - }, "signingKey"}, nil, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - got, err := NewSigner(tt.args.c, tt.args.signingKey) - if (err != nil) != tt.wantErr { - t.Errorf("NewSigner() error = %v, wantErr %v", err, tt.wantErr) - return - } - if got != nil { - got.client = &MockClient{} - } - if !reflect.DeepEqual(got, tt.want) { - t.Errorf("NewSigner() = %v, want %v", got, tt.want) - } - }) - } -} - -func Test_signer_Public(t *testing.T) { - pemBytes, err := os.ReadFile("testdata/pub.pem") - if err != nil { - t.Fatal(err) - } - pk, err := pemutil.ParseKey(pemBytes) - if err != nil { - t.Fatal(err) - } - - type fields struct { - client KeyManagementClient - signingKey string - publicKey crypto.PublicKey - } - tests := []struct { - name string - fields fields - want crypto.PublicKey - }{ - {"ok", fields{&MockClient{}, "signingKey", pk}, pk}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - s := &Signer{ - client: tt.fields.client, - signingKey: tt.fields.signingKey, - publicKey: tt.fields.publicKey, - } - if got := s.Public(); !reflect.DeepEqual(got, tt.want) { - t.Errorf("signer.Public() = %v, want %v", got, tt.want) - } - }) - } -} - -func Test_signer_Sign(t *testing.T) { - keyName := "projects/p/locations/l/keyRings/k/cryptoKeys/c/cryptoKeyVersions/1" - okClient := &MockClient{ - asymmetricSign: func(_ context.Context, _ *kmspb.AsymmetricSignRequest, _ ...gax.CallOption) (*kmspb.AsymmetricSignResponse, error) { - return &kmspb.AsymmetricSignResponse{Signature: []byte("ok signature")}, nil - }, - } - failClient := &MockClient{ - asymmetricSign: func(_ context.Context, _ *kmspb.AsymmetricSignRequest, _ ...gax.CallOption) (*kmspb.AsymmetricSignResponse, error) { - return nil, fmt.Errorf("an error") - }, - } - - type fields struct { - client KeyManagementClient - signingKey string - } - type args struct { - rand io.Reader - digest []byte - opts crypto.SignerOpts - } - tests := []struct { - name string - fields fields - args args - want []byte - wantErr bool - }{ - {"ok sha256", fields{okClient, keyName}, args{rand.Reader, []byte("digest"), crypto.SHA256}, []byte("ok signature"), false}, - {"ok sha384", fields{okClient, keyName}, args{rand.Reader, []byte("digest"), crypto.SHA384}, []byte("ok signature"), false}, - {"ok sha512", fields{okClient, keyName}, args{rand.Reader, []byte("digest"), crypto.SHA512}, []byte("ok signature"), false}, - {"fail MD5", fields{okClient, keyName}, args{rand.Reader, []byte("digest"), crypto.MD5}, nil, true}, - {"fail asymmetric sign", fields{failClient, keyName}, args{rand.Reader, []byte("digest"), crypto.SHA256}, nil, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - s := &Signer{ - client: tt.fields.client, - signingKey: tt.fields.signingKey, - } - got, err := s.Sign(tt.args.rand, tt.args.digest, tt.args.opts) - if (err != nil) != tt.wantErr { - t.Errorf("signer.Sign() error = %v, wantErr %v", err, tt.wantErr) - return - } - if !reflect.DeepEqual(got, tt.want) { - t.Errorf("signer.Sign() = %v, want %v", got, tt.want) - } - }) - } -} - -func TestSigner_SignatureAlgorithm(t *testing.T) { - pemBytes, err := os.ReadFile("testdata/pub.pem") - if err != nil { - t.Fatal(err) - } - - client := &MockClient{ - getPublicKey: func(_ context.Context, req *kmspb.GetPublicKeyRequest, _ ...gax.CallOption) (*kmspb.PublicKey, error) { - var algorithm kmspb.CryptoKeyVersion_CryptoKeyVersionAlgorithm - switch req.Name { - case "ECDSA-SHA256": - algorithm = kmspb.CryptoKeyVersion_EC_SIGN_P256_SHA256 - case "ECDSA-SHA384": - algorithm = kmspb.CryptoKeyVersion_EC_SIGN_P384_SHA384 - case "SHA256-RSA-2048": - algorithm = kmspb.CryptoKeyVersion_RSA_SIGN_PKCS1_2048_SHA256 - case "SHA256-RSA-3072": - algorithm = kmspb.CryptoKeyVersion_RSA_SIGN_PKCS1_3072_SHA256 - case "SHA256-RSA-4096": - algorithm = kmspb.CryptoKeyVersion_RSA_SIGN_PKCS1_4096_SHA256 - case "SHA512-RSA-4096": - algorithm = kmspb.CryptoKeyVersion_RSA_SIGN_PKCS1_4096_SHA512 - case "SHA256-RSAPSS-2048": - algorithm = kmspb.CryptoKeyVersion_RSA_SIGN_PSS_2048_SHA256 - case "SHA256-RSAPSS-3072": - algorithm = kmspb.CryptoKeyVersion_RSA_SIGN_PSS_3072_SHA256 - case "SHA256-RSAPSS-4096": - algorithm = kmspb.CryptoKeyVersion_RSA_SIGN_PSS_4096_SHA256 - case "SHA512-RSAPSS-4096": - algorithm = kmspb.CryptoKeyVersion_RSA_SIGN_PSS_4096_SHA512 - } - return &kmspb.PublicKey{ - Pem: string(pemBytes), - Algorithm: algorithm, - }, nil - }, - } - - if err != nil { - t.Fatal(err) - } - - type fields struct { - client KeyManagementClient - signingKey string - } - tests := []struct { - name string - fields fields - want x509.SignatureAlgorithm - }{ - {"ECDSA-SHA256", fields{client, "ECDSA-SHA256"}, x509.ECDSAWithSHA256}, - {"ECDSA-SHA384", fields{client, "ECDSA-SHA384"}, x509.ECDSAWithSHA384}, - {"SHA256-RSA-2048", fields{client, "SHA256-RSA-2048"}, x509.SHA256WithRSA}, - {"SHA256-RSA-3072", fields{client, "SHA256-RSA-3072"}, x509.SHA256WithRSA}, - {"SHA256-RSA-4096", fields{client, "SHA256-RSA-4096"}, x509.SHA256WithRSA}, - {"SHA512-RSA-4096", fields{client, "SHA512-RSA-4096"}, x509.SHA512WithRSA}, - {"SHA256-RSAPSS-2048", fields{client, "SHA256-RSAPSS-2048"}, x509.SHA256WithRSAPSS}, - {"SHA256-RSAPSS-3072", fields{client, "SHA256-RSAPSS-3072"}, x509.SHA256WithRSAPSS}, - {"SHA256-RSAPSS-4096", fields{client, "SHA256-RSAPSS-4096"}, x509.SHA256WithRSAPSS}, - {"SHA512-RSAPSS-4096", fields{client, "SHA512-RSAPSS-4096"}, x509.SHA512WithRSAPSS}, - {"unknown", fields{client, "UNKNOWN"}, x509.UnknownSignatureAlgorithm}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - signer, err := NewSigner(tt.fields.client, tt.fields.signingKey) - if err != nil { - t.Errorf("NewSigner() error = %v", err) - } - if got := signer.SignatureAlgorithm(); !reflect.DeepEqual(got, tt.want) { - t.Errorf("Signer.SignatureAlgorithm() = %v, want %v", got, tt.want) - } - }) - } -} diff --git a/kms/cloudkms/testdata/pub.pem b/kms/cloudkms/testdata/pub.pem deleted file mode 100644 index e31e583e..00000000 --- a/kms/cloudkms/testdata/pub.pem +++ /dev/null @@ -1,4 +0,0 @@ ------BEGIN PUBLIC KEY----- -MFkwEwYHKoZIzj0CAQYIKoZIzj0DAQcDQgAE5VPD/W5RXn0lrs2MdoNteTSZ+sh1 -veT13hakPZF9YzaNVZgujqK3d1nt+4jPECU+ED/WQ1GgFZiVGUo3flvB/w== ------END PUBLIC KEY----- diff --git a/kms/kms.go b/kms/kms.go deleted file mode 100644 index 92b544df..00000000 --- a/kms/kms.go +++ /dev/null @@ -1,43 +0,0 @@ -package kms - -import ( - "context" - "strings" - - "github.com/pkg/errors" - "github.com/smallstep/certificates/kms/apiv1" - - // Enable default implementation - "github.com/smallstep/certificates/kms/softkms" -) - -// KeyManager is the interface implemented by all the KMS. -type KeyManager = apiv1.KeyManager - -// CertificateManager is the interface implemented by the KMS that can load and -// store x509.Certificates. -type CertificateManager = apiv1.CertificateManager - -// Options are the KMS options. They represent the kms object in the ca.json. -type Options = apiv1.Options - -// Default is the implementation of the default KMS. -var Default = &softkms.SoftKMS{} - -// New initializes a new KMS from the given type. -func New(ctx context.Context, opts apiv1.Options) (KeyManager, error) { - if err := opts.Validate(); err != nil { - return nil, err - } - - t := apiv1.Type(strings.ToLower(opts.Type)) - if t == apiv1.DefaultKMS { - t = apiv1.SoftKMS - } - - fn, ok := apiv1.LoadKeyManagerNewFunc(t) - if !ok { - return nil, errors.Errorf("unsupported kms type '%s'", t) - } - return fn(ctx, opts) -} diff --git a/kms/kms_test.go b/kms/kms_test.go deleted file mode 100644 index f3d2f61f..00000000 --- a/kms/kms_test.go +++ /dev/null @@ -1,52 +0,0 @@ -package kms - -import ( - "context" - "os" - "reflect" - "testing" - - "github.com/smallstep/certificates/kms/apiv1" - "github.com/smallstep/certificates/kms/awskms" - "github.com/smallstep/certificates/kms/cloudkms" - "github.com/smallstep/certificates/kms/softkms" -) - -func TestNew(t *testing.T) { - ctx := context.Background() - - type args struct { - ctx context.Context - opts apiv1.Options - } - tests := []struct { - name string - skipOnCI bool - args args - want KeyManager - wantErr bool - }{ - {"softkms", false, args{ctx, apiv1.Options{Type: "softkms"}}, &softkms.SoftKMS{}, false}, - {"default", false, args{ctx, apiv1.Options{}}, &softkms.SoftKMS{}, false}, - {"awskms", false, args{ctx, apiv1.Options{Type: "awskms"}}, &awskms.KMS{}, false}, - {"cloudkms", true, args{ctx, apiv1.Options{Type: "cloudkms"}}, &cloudkms.CloudKMS{}, true}, // fails because not credentials - {"pkcs11", false, args{ctx, apiv1.Options{Type: "pkcs11"}}, nil, true}, // not yet supported - {"fail validation", false, args{ctx, apiv1.Options{Type: "foobar"}}, nil, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - if tt.skipOnCI && os.Getenv("CI") == "true" { - t.SkipNow() - } - - got, err := New(tt.args.ctx, tt.args.opts) - if (err != nil) != tt.wantErr { - t.Errorf("New() error = %v, wantErr %v", err, tt.wantErr) - return - } - if reflect.TypeOf(got) != reflect.TypeOf(tt.want) { - t.Errorf("New() = %T, want %T", got, tt.want) - } - }) - } -} diff --git a/kms/pkcs11/benchmark_test.go b/kms/pkcs11/benchmark_test.go deleted file mode 100644 index c567872f..00000000 --- a/kms/pkcs11/benchmark_test.go +++ /dev/null @@ -1,83 +0,0 @@ -//go:build cgo -// +build cgo - -package pkcs11 - -import ( - "crypto" - "crypto/rand" - "crypto/rsa" - "testing" - - "github.com/smallstep/certificates/kms/apiv1" -) - -func benchmarkSign(b *testing.B, signer crypto.Signer, opts crypto.SignerOpts) { - hash := opts.HashFunc() - h := hash.New() - h.Write([]byte("buggy-coheir-RUBRIC-rabbet-liberal-eaglet-khartoum-stagger")) - digest := h.Sum(nil) - b.ResetTimer() - for i := 0; i < b.N; i++ { - signer.Sign(rand.Reader, digest, opts) - } - b.StopTimer() -} - -func BenchmarkSignRSA(b *testing.B) { - k := setupPKCS11(b) - signer, err := k.CreateSigner(&apiv1.CreateSignerRequest{ - SigningKey: "pkcs11:id=7371;object=rsa-key", - }) - if err != nil { - b.Fatalf("PKCS11.CreateSigner() error = %v", err) - } - benchmarkSign(b, signer, crypto.SHA256) -} - -func BenchmarkSignRSAPSS(b *testing.B) { - k := setupPKCS11(b) - signer, err := k.CreateSigner(&apiv1.CreateSignerRequest{ - SigningKey: "pkcs11:id=7372;object=rsa-pss-key", - }) - if err != nil { - b.Fatalf("PKCS11.CreateSigner() error = %v", err) - } - benchmarkSign(b, signer, &rsa.PSSOptions{ - SaltLength: rsa.PSSSaltLengthEqualsHash, - Hash: crypto.SHA256, - }) -} - -func BenchmarkSignP256(b *testing.B) { - k := setupPKCS11(b) - signer, err := k.CreateSigner(&apiv1.CreateSignerRequest{ - SigningKey: "pkcs11:id=7373;object=ecdsa-p256-key", - }) - if err != nil { - b.Fatalf("PKCS11.CreateSigner() error = %v", err) - } - benchmarkSign(b, signer, crypto.SHA256) -} - -func BenchmarkSignP384(b *testing.B) { - k := setupPKCS11(b) - signer, err := k.CreateSigner(&apiv1.CreateSignerRequest{ - SigningKey: "pkcs11:id=7374;object=ecdsa-p384-key", - }) - if err != nil { - b.Fatalf("PKCS11.CreateSigner() error = %v", err) - } - benchmarkSign(b, signer, crypto.SHA384) -} - -func BenchmarkSignP521(b *testing.B) { - k := setupPKCS11(b) - signer, err := k.CreateSigner(&apiv1.CreateSignerRequest{ - SigningKey: "pkcs11:id=7375;object=ecdsa-p521-key", - }) - if err != nil { - b.Fatalf("PKCS11.CreateSigner() error = %v", err) - } - benchmarkSign(b, signer, crypto.SHA512) -} diff --git a/kms/pkcs11/opensc_test.go b/kms/pkcs11/opensc_test.go deleted file mode 100644 index 365c075c..00000000 --- a/kms/pkcs11/opensc_test.go +++ /dev/null @@ -1,64 +0,0 @@ -//go:build opensc -// +build opensc - -package pkcs11 - -import ( - "runtime" - "sync" - - "github.com/ThalesIgnite/crypto11" -) - -var softHSM2Once sync.Once - -// mustPKCS11 configures a *PKCS11 KMS to be used with OpenSC, using for example -// a Nitrokey HSM. To initialize these tests we should run: -// -// sc-hsm-tool --initialize --so-pin 3537363231383830 --pin 123456 -// -// Or: -// -// pkcs11-tool --module /usr/local/lib/opensc-pkcs11.so \ -// --init-token --init-pin \ -// --so-pin=3537363231383830 --new-pin=123456 --pin=123456 \ -// --label="pkcs11-test" -func mustPKCS11(t TBTesting) *PKCS11 { - t.Helper() - testModule = "OpenSC" - if runtime.GOARCH != "amd64" { - t.Fatalf("opensc test skipped on %s:%s", runtime.GOOS, runtime.GOARCH) - } - - var path string - switch runtime.GOOS { - case "darwin": - path = "/usr/local/lib/opensc-pkcs11.so" - case "linux": - path = "/usr/local/lib/opensc-pkcs11.so" - default: - t.Skipf("opensc test skipped on %s", runtime.GOOS) - return nil - } - var zero int - p11, err := crypto11.Configure(&crypto11.Config{ - Path: path, - SlotNumber: &zero, - Pin: "123456", - }) - if err != nil { - t.Fatalf("failed to configure opensc on %s: %v", runtime.GOOS, err) - } - - k := &PKCS11{ - p11: p11, - } - - // Setup - softHSM2Once.Do(func() { - teardown(t, k) - setup(t, k) - }) - - return k -} diff --git a/kms/pkcs11/other_test.go b/kms/pkcs11/other_test.go deleted file mode 100644 index 9f4ab4a8..00000000 --- a/kms/pkcs11/other_test.go +++ /dev/null @@ -1,210 +0,0 @@ -//go:build cgo && !softhsm2 && !yubihsm2 && !opensc -// +build cgo,!softhsm2,!yubihsm2,!opensc - -package pkcs11 - -import ( - "crypto" - "crypto/ecdsa" - "crypto/elliptic" - "crypto/rand" - "crypto/rsa" - "crypto/x509" - "io" - "math/big" - - "github.com/ThalesIgnite/crypto11" - "github.com/pkg/errors" -) - -func mustPKCS11(t TBTesting) *PKCS11 { - t.Helper() - testModule = "Golang crypto" - k := &PKCS11{ - p11: &stubPKCS11{ - signerIndex: make(map[keyType]int), - certIndex: make(map[keyType]int), - }, - } - for i := range testCerts { - testCerts[i].Certificates = nil - } - teardown(t, k) - setup(t, k) - return k -} - -type keyType struct { - id string - label string - serial string -} - -func newKey(id, label []byte, serial *big.Int) keyType { - var serialString string - if serial != nil { - serialString = serial.String() - } - return keyType{ - id: string(id), - label: string(label), - serial: serialString, - } -} - -type stubPKCS11 struct { - signers []crypto11.Signer - certs []*x509.Certificate - signerIndex map[keyType]int - certIndex map[keyType]int -} - -func (s *stubPKCS11) FindKeyPair(id, label []byte) (crypto11.Signer, error) { - if id == nil && label == nil { - return nil, errors.New("id and label cannot both be nil") - } - if i, ok := s.signerIndex[newKey(id, label, nil)]; ok { - return s.signers[i], nil - } - return nil, nil -} - -func (s *stubPKCS11) FindCertificate(id, label []byte, serial *big.Int) (*x509.Certificate, error) { - if id == nil && label == nil && serial == nil { - return nil, errors.New("id, label and serial cannot both be nil") - } - if i, ok := s.certIndex[newKey(id, label, serial)]; ok { - return s.certs[i], nil - } - return nil, nil - -} - -func (s *stubPKCS11) ImportCertificateWithAttributes(template crypto11.AttributeSet, cert *x509.Certificate) error { - var id, label []byte - if v := template[crypto11.CkaId]; v != nil { - id = v.Value - } - if v := template[crypto11.CkaLabel]; v != nil { - label = v.Value - } - return s.ImportCertificateWithLabel(id, label, cert) -} - -func (s *stubPKCS11) ImportCertificateWithLabel(id, label []byte, cert *x509.Certificate) error { - switch { - case id == nil: - return errors.New("id cannot both be nil") - case label == nil: - return errors.New("label cannot both be nil") - case cert == nil: - return errors.New("certificate cannot be nil") - } - - i := len(s.certs) - s.certs = append(s.certs, cert) - s.certIndex[newKey(id, label, cert.SerialNumber)] = i - s.certIndex[newKey(id, nil, nil)] = i - s.certIndex[newKey(nil, label, nil)] = i - s.certIndex[newKey(nil, nil, cert.SerialNumber)] = i - s.certIndex[newKey(id, label, nil)] = i - s.certIndex[newKey(id, nil, cert.SerialNumber)] = i - s.certIndex[newKey(nil, label, cert.SerialNumber)] = i - - return nil -} - -func (s *stubPKCS11) DeleteCertificate(id, label []byte, serial *big.Int) error { - if id == nil && label == nil && serial == nil { - return errors.New("id, label and serial cannot both be nil") - } - if i, ok := s.certIndex[newKey(id, label, serial)]; ok { - s.certs[i] = nil - } - return nil -} - -func (s *stubPKCS11) GenerateRSAKeyPairWithAttributes(public, private crypto11.AttributeSet, bits int) (crypto11.SignerDecrypter, error) { - var id, label []byte - if v := public[crypto11.CkaId]; v != nil { - id = v.Value - } - if v := public[crypto11.CkaLabel]; v != nil { - label = v.Value - } - return s.GenerateRSAKeyPairWithLabel(id, label, bits) -} - -func (s *stubPKCS11) GenerateRSAKeyPairWithLabel(id, label []byte, bits int) (crypto11.SignerDecrypter, error) { - if id == nil && label == nil { - return nil, errors.New("id and label cannot both be nil") - } - p, err := rsa.GenerateKey(rand.Reader, bits) - if err != nil { - return nil, err - } - k := &privateKey{ - Signer: p, - index: len(s.signers), - stub: s, - } - s.signers = append(s.signers, k) - s.signerIndex[newKey(id, label, nil)] = k.index - s.signerIndex[newKey(id, nil, nil)] = k.index - s.signerIndex[newKey(nil, label, nil)] = k.index - return k, nil -} - -func (s *stubPKCS11) GenerateECDSAKeyPairWithAttributes(public, private crypto11.AttributeSet, curve elliptic.Curve) (crypto11.Signer, error) { - var id, label []byte - if v := public[crypto11.CkaId]; v != nil { - id = v.Value - } - if v := public[crypto11.CkaLabel]; v != nil { - label = v.Value - } - return s.GenerateECDSAKeyPairWithLabel(id, label, curve) -} - -func (s *stubPKCS11) GenerateECDSAKeyPairWithLabel(id, label []byte, curve elliptic.Curve) (crypto11.Signer, error) { - if id == nil && label == nil { - return nil, errors.New("id and label cannot both be nil") - } - p, err := ecdsa.GenerateKey(curve, rand.Reader) - if err != nil { - return nil, err - } - k := &privateKey{ - Signer: p, - index: len(s.signers), - stub: s, - } - s.signers = append(s.signers, k) - s.signerIndex[newKey(id, label, nil)] = k.index - s.signerIndex[newKey(id, nil, nil)] = k.index - s.signerIndex[newKey(nil, label, nil)] = k.index - return k, nil -} - -func (s *stubPKCS11) Close() error { - return nil -} - -type privateKey struct { - crypto.Signer - index int - stub *stubPKCS11 -} - -func (s *privateKey) Delete() error { - s.stub.signers[s.index] = nil - return nil -} - -func (s *privateKey) Decrypt(rnd io.Reader, msg []byte, opts crypto.DecrypterOpts) (plaintext []byte, err error) { - k, ok := s.Signer.(*rsa.PrivateKey) - if !ok { - return nil, errors.New("key is not an rsa key") - } - return k.Decrypt(rnd, msg, opts) -} diff --git a/kms/pkcs11/pkcs11.go b/kms/pkcs11/pkcs11.go deleted file mode 100644 index c0e06408..00000000 --- a/kms/pkcs11/pkcs11.go +++ /dev/null @@ -1,399 +0,0 @@ -//go:build cgo -// +build cgo - -package pkcs11 - -import ( - "context" - "crypto" - "crypto/elliptic" - "crypto/rsa" - "crypto/x509" - "encoding/hex" - "fmt" - "math/big" - "strconv" - "sync" - - "github.com/ThalesIgnite/crypto11" - "github.com/pkg/errors" - "github.com/smallstep/certificates/kms/apiv1" - "github.com/smallstep/certificates/kms/uri" -) - -// Scheme is the scheme used in uris. -const Scheme = "pkcs11" - -// DefaultRSASize is the number of bits of a new RSA key if no size has been -// specified. -const DefaultRSASize = 3072 - -// P11 defines the methods on crypto11.Context that this package will use. This -// interface will be used for unit testing. -type P11 interface { - FindKeyPair(id, label []byte) (crypto11.Signer, error) - FindCertificate(id, label []byte, serial *big.Int) (*x509.Certificate, error) - ImportCertificateWithAttributes(template crypto11.AttributeSet, certificate *x509.Certificate) error - DeleteCertificate(id, label []byte, serial *big.Int) error - GenerateRSAKeyPairWithAttributes(public, private crypto11.AttributeSet, bits int) (crypto11.SignerDecrypter, error) - GenerateECDSAKeyPairWithAttributes(public, private crypto11.AttributeSet, curve elliptic.Curve) (crypto11.Signer, error) - Close() error -} - -var p11Configure = func(config *crypto11.Config) (P11, error) { - return crypto11.Configure(config) -} - -// PKCS11 is the implementation of a KMS using the PKCS #11 standard. -type PKCS11 struct { - p11 P11 - closed sync.Once -} - -// New returns a new PKCS11 KMS. -func New(ctx context.Context, opts apiv1.Options) (*PKCS11, error) { - var config crypto11.Config - if opts.URI != "" { - u, err := uri.ParseWithScheme(Scheme, opts.URI) - if err != nil { - return nil, err - } - - config.Pin = u.Pin() - config.Path = u.Get("module-path") - config.TokenLabel = u.Get("token") - config.TokenSerial = u.Get("serial") - if v := u.Get("slot-id"); v != "" { - n, err := strconv.Atoi(v) - if err != nil { - return nil, errors.Wrap(err, "kms uri 'slot-id' is not valid") - } - config.SlotNumber = &n - } - } - if config.Pin == "" && opts.Pin != "" { - config.Pin = opts.Pin - } - - switch { - case config.Path == "": - return nil, errors.New("kms uri 'module-path' are required") - case config.TokenLabel == "" && config.TokenSerial == "" && config.SlotNumber == nil: - return nil, errors.New("kms uri 'token', 'serial' or 'slot-id' are required") - case config.Pin == "": - return nil, errors.New("kms 'pin' cannot be empty") - case config.TokenLabel != "" && config.TokenSerial != "": - return nil, errors.New("kms uri 'token' and 'serial' are mutually exclusive") - case config.TokenLabel != "" && config.SlotNumber != nil: - return nil, errors.New("kms uri 'token' and 'slot-id' are mutually exclusive") - case config.TokenSerial != "" && config.SlotNumber != nil: - return nil, errors.New("kms uri 'serial' and 'slot-id' are mutually exclusive") - } - - p11, err := p11Configure(&config) - if err != nil { - return nil, errors.Wrap(err, "error initializing PKCS#11") - } - - return &PKCS11{ - p11: p11, - }, nil -} - -func init() { - apiv1.Register(apiv1.PKCS11, func(ctx context.Context, opts apiv1.Options) (apiv1.KeyManager, error) { - return New(ctx, opts) - }) -} - -// GetPublicKey returns the public key .... -func (k *PKCS11) GetPublicKey(req *apiv1.GetPublicKeyRequest) (crypto.PublicKey, error) { - if req.Name == "" { - return nil, errors.New("getPublicKeyRequest 'name' cannot be empty") - } - - signer, err := findSigner(k.p11, req.Name) - if err != nil { - return nil, errors.Wrap(err, "getPublicKey failed") - } - - return signer.Public(), nil -} - -// CreateKey generates a new key in the PKCS#11 module and returns the public key. -func (k *PKCS11) CreateKey(req *apiv1.CreateKeyRequest) (*apiv1.CreateKeyResponse, error) { - switch { - case req.Name == "": - return nil, errors.New("createKeyRequest 'name' cannot be empty") - case req.Bits < 0: - return nil, errors.New("createKeyRequest 'bits' cannot be negative") - } - - signer, err := generateKey(k.p11, req) - if err != nil { - return nil, errors.Wrap(err, "createKey failed") - } - - return &apiv1.CreateKeyResponse{ - Name: req.Name, - PublicKey: signer.Public(), - CreateSignerRequest: apiv1.CreateSignerRequest{ - SigningKey: req.Name, - }, - }, nil -} - -// CreateSigner creates a signer using a key present in the PKCS#11 module. -func (k *PKCS11) CreateSigner(req *apiv1.CreateSignerRequest) (crypto.Signer, error) { - if req.SigningKey == "" { - return nil, errors.New("createSignerRequest 'signingKey' cannot be empty") - } - - signer, err := findSigner(k.p11, req.SigningKey) - if err != nil { - return nil, errors.Wrap(err, "createSigner failed") - } - - return signer, nil -} - -// CreateDecrypter creates a decrypter using a key present in the PKCS#11 -// module. -func (k *PKCS11) CreateDecrypter(req *apiv1.CreateDecrypterRequest) (crypto.Decrypter, error) { - if req.DecryptionKey == "" { - return nil, errors.New("createDecrypterRequest 'decryptionKey' cannot be empty") - } - - signer, err := findSigner(k.p11, req.DecryptionKey) - if err != nil { - return nil, errors.Wrap(err, "createDecrypterRequest failed") - } - - // Only RSA keys will implement the Decrypter interface. - if _, ok := signer.Public().(*rsa.PublicKey); ok { - if dec, ok := signer.(crypto.Decrypter); ok { - return dec, nil - } - } - return nil, errors.New("createDecrypterRequest failed: signer does not implement crypto.Decrypter") -} - -// LoadCertificate implements kms.CertificateManager and loads a certificate -// from the YubiKey. -func (k *PKCS11) LoadCertificate(req *apiv1.LoadCertificateRequest) (*x509.Certificate, error) { - if req.Name == "" { - return nil, errors.New("loadCertificateRequest 'name' cannot be nil") - } - cert, err := findCertificate(k.p11, req.Name) - if err != nil { - return nil, errors.Wrap(err, "loadCertificate failed") - } - return cert, nil -} - -// StoreCertificate implements kms.CertificateManager and stores a certificate -// in the YubiKey. -func (k *PKCS11) StoreCertificate(req *apiv1.StoreCertificateRequest) error { - switch { - case req.Name == "": - return errors.New("storeCertificateRequest 'name' cannot be empty") - case req.Certificate == nil: - return errors.New("storeCertificateRequest 'Certificate' cannot be nil") - } - - id, object, err := parseObject(req.Name) - if err != nil { - return errors.Wrap(err, "storeCertificate failed") - } - - // Enforce the use of both id and labels. This is not strictly necessary in - // PKCS #11, but it's a good practice. - if len(id) == 0 || len(object) == 0 { - return errors.Errorf("key with uri %s is not valid, id and object are required", req.Name) - } - - cert, err := k.p11.FindCertificate(id, object, nil) - if err != nil { - return errors.Wrap(err, "storeCertificate failed") - } - if cert != nil { - return errors.Wrap(apiv1.ErrAlreadyExists{ - Message: req.Name + " already exists", - }, "storeCertificate failed") - } - - // Import certificate with the necessary attributes. - template, err := crypto11.NewAttributeSetWithIDAndLabel(id, object) - if err != nil { - return errors.Wrap(err, "storeCertificate failed") - } - if req.Extractable { - template.Set(crypto11.CkaExtractable, true) - } - if err := k.p11.ImportCertificateWithAttributes(template, req.Certificate); err != nil { - return errors.Wrap(err, "storeCertificate failed") - } - - return nil -} - -// DeleteKey is a utility function to delete a key given an uri. -func (k *PKCS11) DeleteKey(u string) error { - id, object, err := parseObject(u) - if err != nil { - return errors.Wrap(err, "deleteKey failed") - } - signer, err := k.p11.FindKeyPair(id, object) - if err != nil { - return errors.Wrap(err, "deleteKey failed") - } - if signer == nil { - return nil - } - if err := signer.Delete(); err != nil { - return errors.Wrap(err, "deleteKey failed") - } - return nil -} - -// DeleteCertificate is a utility function to delete a certificate given an uri. -func (k *PKCS11) DeleteCertificate(u string) error { - id, object, err := parseObject(u) - if err != nil { - return errors.Wrap(err, "deleteCertificate failed") - } - if err := k.p11.DeleteCertificate(id, object, nil); err != nil { - return errors.Wrap(err, "deleteCertificate failed") - } - return nil -} - -// Close releases the connection to the PKCS#11 module. -func (k *PKCS11) Close() (err error) { - k.closed.Do(func() { - err = errors.Wrap(k.p11.Close(), "error closing pkcs#11 context") - }) - return -} - -func toByte(s string) []byte { - if s == "" { - return nil - } - return []byte(s) -} - -func parseObject(rawuri string) ([]byte, []byte, error) { - u, err := uri.ParseWithScheme(Scheme, rawuri) - if err != nil { - return nil, nil, err - } - id := u.GetEncoded("id") - object := u.Get("object") - if len(id) == 0 && object == "" { - return nil, nil, errors.Errorf("key with uri %s is not valid, id or object are required", rawuri) - } - - return id, toByte(object), nil -} - -func generateKey(ctx P11, req *apiv1.CreateKeyRequest) (crypto11.Signer, error) { - id, object, err := parseObject(req.Name) - if err != nil { - return nil, err - } - - signer, err := ctx.FindKeyPair(id, object) - if err != nil { - return nil, err - } - if signer != nil { - return nil, apiv1.ErrAlreadyExists{ - Message: req.Name + " already exists", - } - } - - // Enforce the use of both id and labels. This is not strictly necessary in - // PKCS #11, but it's a good practice. - if len(id) == 0 || len(object) == 0 { - return nil, errors.Errorf("key with uri %s is not valid, id and object are required", req.Name) - } - - // Create template for public and private keys - public, err := crypto11.NewAttributeSetWithIDAndLabel(id, object) - if err != nil { - return nil, err - } - private := public.Copy() - if req.Extractable { - private.Set(crypto11.CkaExtractable, true) - } - - bits := req.Bits - if bits == 0 { - bits = DefaultRSASize - } - - switch req.SignatureAlgorithm { - case apiv1.UnspecifiedSignAlgorithm: - return ctx.GenerateECDSAKeyPairWithAttributes(public, private, elliptic.P256()) - case apiv1.SHA256WithRSA, apiv1.SHA384WithRSA, apiv1.SHA512WithRSA: - return ctx.GenerateRSAKeyPairWithAttributes(public, private, bits) - case apiv1.SHA256WithRSAPSS, apiv1.SHA384WithRSAPSS, apiv1.SHA512WithRSAPSS: - return ctx.GenerateRSAKeyPairWithAttributes(public, private, bits) - case apiv1.ECDSAWithSHA256: - return ctx.GenerateECDSAKeyPairWithAttributes(public, private, elliptic.P256()) - case apiv1.ECDSAWithSHA384: - return ctx.GenerateECDSAKeyPairWithAttributes(public, private, elliptic.P384()) - case apiv1.ECDSAWithSHA512: - return ctx.GenerateECDSAKeyPairWithAttributes(public, private, elliptic.P521()) - case apiv1.PureEd25519: - return nil, fmt.Errorf("signature algorithm %s is not supported", req.SignatureAlgorithm) - default: - return nil, fmt.Errorf("signature algorithm %s is not supported", req.SignatureAlgorithm) - } -} - -func findSigner(ctx P11, rawuri string) (crypto11.Signer, error) { - id, object, err := parseObject(rawuri) - if err != nil { - return nil, err - } - signer, err := ctx.FindKeyPair(id, object) - if err != nil { - return nil, errors.Wrapf(err, "error finding key with uri %s", rawuri) - } - if signer == nil { - return nil, errors.Errorf("key with uri %s not found", rawuri) - } - return signer, nil -} - -func findCertificate(ctx P11, rawuri string) (*x509.Certificate, error) { - u, err := uri.ParseWithScheme(Scheme, rawuri) - if err != nil { - return nil, err - } - id, object, serial := u.GetEncoded("id"), u.Get("object"), u.Get("serial") - if len(id) == 0 && object == "" && serial == "" { - return nil, errors.Errorf("key with uri %s is not valid, id, object or serial are required", rawuri) - } - - var serialNumber *big.Int - if serial != "" { - b, err := hex.DecodeString(serial) - if err != nil { - return nil, errors.Errorf("key with uri %s is not valid, failed to decode serial", rawuri) - } - serialNumber = new(big.Int).SetBytes(b) - } - - cert, err := ctx.FindCertificate(id, toByte(object), serialNumber) - if err != nil { - return nil, errors.Wrapf(err, "error finding certificate with uri %s", rawuri) - } - if cert == nil { - return nil, errors.Errorf("certificate with uri %s not found", rawuri) - } - return cert, nil -} diff --git a/kms/pkcs11/pkcs11_no_cgo.go b/kms/pkcs11/pkcs11_no_cgo.go deleted file mode 100644 index 6fa51dff..00000000 --- a/kms/pkcs11/pkcs11_no_cgo.go +++ /dev/null @@ -1,58 +0,0 @@ -//go:build !cgo -// +build !cgo - -package pkcs11 - -import ( - "context" - "crypto" - "os" - "path/filepath" - - "github.com/pkg/errors" - "github.com/smallstep/certificates/kms/apiv1" -) - -var errUnsupported error - -func init() { - name := filepath.Base(os.Args[0]) - errUnsupported = errors.Errorf("unsupported kms type 'pkcs11': %s is compiled without cgo support", name) - - apiv1.Register(apiv1.PKCS11, func(ctx context.Context, opts apiv1.Options) (apiv1.KeyManager, error) { - return nil, errUnsupported - }) -} - -// PKCS11 is the implementation of a KMS using the PKCS #11 standard. -type PKCS11 struct{} - -// New implements the kms.KeyManager interface and without CGO will always -// return an error. -func New(ctx context.Context, opts apiv1.Options) (*PKCS11, error) { - return nil, errUnsupported -} - -// GetPublicKey implements the kms.KeyManager interface and without CGO will always -// return an error. -func (*PKCS11) GetPublicKey(req *apiv1.GetPublicKeyRequest) (crypto.PublicKey, error) { - return nil, errUnsupported -} - -// CreateKey implements the kms.KeyManager interface and without CGO will always -// return an error. -func (*PKCS11) CreateKey(req *apiv1.CreateKeyRequest) (*apiv1.CreateKeyResponse, error) { - return nil, errUnsupported -} - -// CreateSigner implements the kms.KeyManager interface and without CGO will always -// return an error. -func (*PKCS11) CreateSigner(req *apiv1.CreateSignerRequest) (crypto.Signer, error) { - return nil, errUnsupported -} - -// Close implements the kms.KeyManager interface and without CGO will always -// return an error. -func (*PKCS11) Close() error { - return errUnsupported -} diff --git a/kms/pkcs11/pkcs11_test.go b/kms/pkcs11/pkcs11_test.go deleted file mode 100644 index 06edd048..00000000 --- a/kms/pkcs11/pkcs11_test.go +++ /dev/null @@ -1,836 +0,0 @@ -//go:build cgo -// +build cgo - -package pkcs11 - -import ( - "bytes" - "context" - "crypto" - "crypto/ecdsa" - "crypto/ed25519" - "crypto/rand" - "crypto/rsa" - "crypto/x509" - "math/big" - "reflect" - "strings" - "testing" - - "github.com/ThalesIgnite/crypto11" - "github.com/pkg/errors" - "github.com/smallstep/certificates/kms/apiv1" - "golang.org/x/crypto/cryptobyte" - "golang.org/x/crypto/cryptobyte/asn1" -) - -func TestNew(t *testing.T) { - tmp := p11Configure - t.Cleanup(func() { - p11Configure = tmp - }) - - k := mustPKCS11(t) - t.Cleanup(func() { - k.Close() - }) - p11Configure = func(config *crypto11.Config) (P11, error) { - if strings.Contains(config.Path, "fail") { - return nil, errors.New("an error") - } - return k.p11, nil - } - - type args struct { - ctx context.Context - opts apiv1.Options - } - tests := []struct { - name string - args args - want *PKCS11 - wantErr bool - }{ - {"ok", args{context.Background(), apiv1.Options{ - Type: "pkcs11", - URI: "pkcs11:module-path=/usr/local/lib/softhsm/libsofthsm2.so;token=pkcs11-test?pin-value=password", - }}, k, false}, - {"ok with serial", args{context.Background(), apiv1.Options{ - Type: "pkcs11", - URI: "pkcs11:module-path=/usr/local/lib/softhsm/libsofthsm2.so;serial=0123456789?pin-value=password", - }}, k, false}, - {"ok with slot-id", args{context.Background(), apiv1.Options{ - Type: "pkcs11", - URI: "pkcs11:module-path=/usr/local/lib/softhsm/libsofthsm2.so;slot-id=0?pin-value=password", - }}, k, false}, - {"ok with pin", args{context.Background(), apiv1.Options{ - Type: "pkcs11", - URI: "pkcs11:module-path=/usr/local/lib/softhsm/libsofthsm2.so;token=pkcs11-test", - Pin: "passowrd", - }}, k, false}, - {"fail missing module", args{context.Background(), apiv1.Options{ - Type: "pkcs11", - URI: "pkcs11:token=pkcs11-test", - Pin: "passowrd", - }}, nil, true}, - {"fail missing pin", args{context.Background(), apiv1.Options{ - Type: "pkcs11", - URI: "pkcs11:module-path=/usr/local/lib/softhsm/libsofthsm2.so;token=pkcs11-test", - }}, nil, true}, - {"fail missing token/serial/slot-id", args{context.Background(), apiv1.Options{ - Type: "pkcs11", - URI: "pkcs11:module-path=/usr/local/lib/softhsm/libsofthsm2.so", - Pin: "passowrd", - }}, nil, true}, - {"fail token+serial+slot-id", args{context.Background(), apiv1.Options{ - Type: "pkcs11", - URI: "pkcs11:module-path=/usr/local/lib/softhsm/libsofthsm2.so;token=pkcs11-test;serial=0123456789;slot-id=0", - Pin: "passowrd", - }}, nil, true}, - {"fail token+serial", args{context.Background(), apiv1.Options{ - Type: "pkcs11", - URI: "pkcs11:module-path=/usr/local/lib/softhsm/libsofthsm2.so;token=pkcs11-test;serial=0123456789", - Pin: "passowrd", - }}, nil, true}, - {"fail token+slot-id", args{context.Background(), apiv1.Options{ - Type: "pkcs11", - URI: "pkcs11:module-path=/usr/local/lib/softhsm/libsofthsm2.so;token=pkcs11-test;slot-id=0", - Pin: "passowrd", - }}, nil, true}, - {"fail serial+slot-id", args{context.Background(), apiv1.Options{ - Type: "pkcs11", - URI: "pkcs11:module-path=/usr/local/lib/softhsm/libsofthsm2.so;serial=0123456789;slot-id=0", - Pin: "passowrd", - }}, nil, true}, - {"fail slot-id", args{context.Background(), apiv1.Options{ - Type: "pkcs11", - URI: "pkcs11:module-path=/usr/local/lib/softhsm/libsofthsm2.so;slot-id=x?pin-value=password", - }}, nil, true}, - {"fail scheme", args{context.Background(), apiv1.Options{ - Type: "pkcs11", - URI: "foo:module-path=/usr/local/lib/softhsm/libsofthsm2.so;token=pkcs11-test?pin-value=password", - }}, nil, true}, - {"fail configure", args{context.Background(), apiv1.Options{ - Type: "pkcs11", - URI: "pkcs11:module-path=/usr/local/lib/fail.so;token=pkcs11-test?pin-value=password", - }}, nil, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - got, err := New(tt.args.ctx, tt.args.opts) - if (err != nil) != tt.wantErr { - t.Errorf("New() error = %v, wantErr %v", err, tt.wantErr) - return - } - if !reflect.DeepEqual(got, tt.want) { - t.Errorf("New() = %v, want %v", got, tt.want) - } - }) - } -} - -func TestPKCS11_GetPublicKey(t *testing.T) { - k := setupPKCS11(t) - type args struct { - req *apiv1.GetPublicKeyRequest - } - tests := []struct { - name string - args args - want crypto.PublicKey - wantErr bool - }{ - {"RSA", args{&apiv1.GetPublicKeyRequest{ - Name: "pkcs11:id=7371;object=rsa-key", - }}, &rsa.PublicKey{}, false}, - {"RSA by id", args{&apiv1.GetPublicKeyRequest{ - Name: "pkcs11:id=7371", - }}, &rsa.PublicKey{}, false}, - {"RSA by label", args{&apiv1.GetPublicKeyRequest{ - Name: "pkcs11:object=rsa-key", - }}, &rsa.PublicKey{}, false}, - {"ECDSA", args{&apiv1.GetPublicKeyRequest{ - Name: "pkcs11:id=7373;object=ecdsa-p256-key", - }}, &ecdsa.PublicKey{}, false}, - {"ECDSA by id", args{&apiv1.GetPublicKeyRequest{ - Name: "pkcs11:id=7373", - }}, &ecdsa.PublicKey{}, false}, - {"ECDSA by label", args{&apiv1.GetPublicKeyRequest{ - Name: "pkcs11:object=ecdsa-p256-key", - }}, &ecdsa.PublicKey{}, false}, - {"fail name", args{&apiv1.GetPublicKeyRequest{ - Name: "", - }}, nil, true}, - {"fail uri", args{&apiv1.GetPublicKeyRequest{ - Name: "https:id=9999;object=https", - }}, nil, true}, - {"fail missing", args{&apiv1.GetPublicKeyRequest{ - Name: "pkcs11:id=9999;object=rsa-key", - }}, nil, true}, - {"fail FindKeyPair", args{&apiv1.GetPublicKeyRequest{ - Name: "pkcs11:foo=bar", - }}, nil, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - got, err := k.GetPublicKey(tt.args.req) - if (err != nil) != tt.wantErr { - t.Errorf("PKCS11.GetPublicKey() error = %v, wantErr %v", err, tt.wantErr) - return - } - if reflect.TypeOf(got) != reflect.TypeOf(tt.want) { - t.Errorf("PKCS11.GetPublicKey() = %T, want %T", got, tt.want) - } - }) - } -} - -func TestPKCS11_CreateKey(t *testing.T) { - k := setupPKCS11(t) - - // Make sure to delete the created key - k.DeleteKey(testObject) - - type args struct { - req *apiv1.CreateKeyRequest - } - tests := []struct { - name string - args args - want *apiv1.CreateKeyResponse - wantErr bool - }{ - {"default", args{&apiv1.CreateKeyRequest{ - Name: testObject, - }}, &apiv1.CreateKeyResponse{ - Name: testObject, - PublicKey: &ecdsa.PublicKey{}, - CreateSignerRequest: apiv1.CreateSignerRequest{ - SigningKey: testObject, - }, - }, false}, - {"default extractable", args{&apiv1.CreateKeyRequest{ - Name: testObject, - Extractable: true, - }}, &apiv1.CreateKeyResponse{ - Name: testObject, - PublicKey: &ecdsa.PublicKey{}, - CreateSignerRequest: apiv1.CreateSignerRequest{ - SigningKey: testObject, - }, - }, false}, - {"RSA SHA256WithRSA", args{&apiv1.CreateKeyRequest{ - Name: testObject, - SignatureAlgorithm: apiv1.SHA256WithRSA, - }}, &apiv1.CreateKeyResponse{ - Name: testObject, - PublicKey: &rsa.PublicKey{}, - CreateSignerRequest: apiv1.CreateSignerRequest{ - SigningKey: testObject, - }, - }, false}, - {"RSA SHA384WithRSA", args{&apiv1.CreateKeyRequest{ - Name: testObject, - SignatureAlgorithm: apiv1.SHA384WithRSA, - }}, &apiv1.CreateKeyResponse{ - Name: testObject, - PublicKey: &rsa.PublicKey{}, - CreateSignerRequest: apiv1.CreateSignerRequest{ - SigningKey: testObject, - }, - }, false}, - {"RSA SHA512WithRSA", args{&apiv1.CreateKeyRequest{ - Name: testObject, - SignatureAlgorithm: apiv1.SHA512WithRSA, - }}, &apiv1.CreateKeyResponse{ - Name: testObject, - PublicKey: &rsa.PublicKey{}, - CreateSignerRequest: apiv1.CreateSignerRequest{ - SigningKey: testObject, - }, - }, false}, - {"RSA SHA256WithRSAPSS", args{&apiv1.CreateKeyRequest{ - Name: testObject, - SignatureAlgorithm: apiv1.SHA256WithRSAPSS, - }}, &apiv1.CreateKeyResponse{ - Name: testObject, - PublicKey: &rsa.PublicKey{}, - CreateSignerRequest: apiv1.CreateSignerRequest{ - SigningKey: testObject, - }, - }, false}, - {"RSA SHA384WithRSAPSS", args{&apiv1.CreateKeyRequest{ - Name: testObject, - SignatureAlgorithm: apiv1.SHA384WithRSAPSS, - }}, &apiv1.CreateKeyResponse{ - Name: testObject, - PublicKey: &rsa.PublicKey{}, - CreateSignerRequest: apiv1.CreateSignerRequest{ - SigningKey: testObject, - }, - }, false}, - {"RSA SHA512WithRSAPSS", args{&apiv1.CreateKeyRequest{ - Name: testObject, - SignatureAlgorithm: apiv1.SHA512WithRSAPSS, - }}, &apiv1.CreateKeyResponse{ - Name: testObject, - PublicKey: &rsa.PublicKey{}, - CreateSignerRequest: apiv1.CreateSignerRequest{ - SigningKey: testObject, - }, - }, false}, - {"RSA 2048", args{&apiv1.CreateKeyRequest{ - Name: testObject, - SignatureAlgorithm: apiv1.SHA256WithRSA, - Bits: 2048, - }}, &apiv1.CreateKeyResponse{ - Name: testObject, - PublicKey: &rsa.PublicKey{}, - CreateSignerRequest: apiv1.CreateSignerRequest{ - SigningKey: testObject, - }, - }, false}, - {"RSA 4096", args{&apiv1.CreateKeyRequest{ - Name: testObject, - SignatureAlgorithm: apiv1.SHA256WithRSA, - Bits: 4096, - }}, &apiv1.CreateKeyResponse{ - Name: testObject, - PublicKey: &rsa.PublicKey{}, - CreateSignerRequest: apiv1.CreateSignerRequest{ - SigningKey: testObject, - }, - }, false}, - {"ECDSA P256", args{&apiv1.CreateKeyRequest{ - Name: testObject, - SignatureAlgorithm: apiv1.ECDSAWithSHA256, - }}, &apiv1.CreateKeyResponse{ - Name: testObject, - PublicKey: &ecdsa.PublicKey{}, - CreateSignerRequest: apiv1.CreateSignerRequest{ - SigningKey: testObject, - }, - }, false}, - {"ECDSA P384", args{&apiv1.CreateKeyRequest{ - Name: testObject, - SignatureAlgorithm: apiv1.ECDSAWithSHA384, - }}, &apiv1.CreateKeyResponse{ - Name: testObject, - PublicKey: &ecdsa.PublicKey{}, - CreateSignerRequest: apiv1.CreateSignerRequest{ - SigningKey: testObject, - }, - }, false}, - {"ECDSA P521", args{&apiv1.CreateKeyRequest{ - Name: testObject, - SignatureAlgorithm: apiv1.ECDSAWithSHA512, - }}, &apiv1.CreateKeyResponse{ - Name: testObject, - PublicKey: &ecdsa.PublicKey{}, - CreateSignerRequest: apiv1.CreateSignerRequest{ - SigningKey: testObject, - }, - }, false}, - {"fail name", args{&apiv1.CreateKeyRequest{ - Name: "", - }}, nil, true}, - {"fail no id", args{&apiv1.CreateKeyRequest{ - Name: "pkcs11:object=create-key", - }}, nil, true}, - {"fail no object", args{&apiv1.CreateKeyRequest{ - Name: "pkcs11:id=9999", - }}, nil, true}, - {"fail schema", args{&apiv1.CreateKeyRequest{ - Name: "pkcs12:id=9999;object=create-key", - }}, nil, true}, - {"fail bits", args{&apiv1.CreateKeyRequest{ - Name: "pkcs11:id=9999;object=create-key", - Bits: -1, - SignatureAlgorithm: apiv1.SHA256WithRSAPSS, - }}, nil, true}, - {"fail ed25519", args{&apiv1.CreateKeyRequest{ - Name: "pkcs11:id=9999;object=create-key", - SignatureAlgorithm: apiv1.PureEd25519, - }}, nil, true}, - {"fail unknown", args{&apiv1.CreateKeyRequest{ - Name: "pkcs11:id=9999;object=create-key", - SignatureAlgorithm: apiv1.SignatureAlgorithm(100), - }}, nil, true}, - {"fail FindKeyPair", args{&apiv1.CreateKeyRequest{ - Name: "pkcs11:foo=bar", - SignatureAlgorithm: apiv1.SHA256WithRSAPSS, - }}, nil, true}, - {"fail already exists", args{&apiv1.CreateKeyRequest{ - Name: "pkcs11:id=7373;object=ecdsa-p256-key", - SignatureAlgorithm: apiv1.ECDSAWithSHA256, - }}, nil, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - got, err := k.CreateKey(tt.args.req) - if (err != nil) != tt.wantErr { - t.Errorf("PKCS11.CreateKey() error = %v, wantErr %v", err, tt.wantErr) - return - } - if got != nil { - got.PublicKey = tt.want.PublicKey - } - if !reflect.DeepEqual(got, tt.want) { - t.Errorf("PKCS11.CreateKey() = %v, want %v", got, tt.want) - } - if got != nil { - if err := k.DeleteKey(got.Name); err != nil { - t.Errorf("PKCS11.DeleteKey() error = %v", err) - } - } - }) - } -} - -func TestPKCS11_CreateSigner(t *testing.T) { - k := setupPKCS11(t) - data := []byte("buggy-coheir-RUBRIC-rabbet-liberal-eaglet-khartoum-stagger") - - // VerifyASN1 verifies the ASN.1 encoded signature, sig, of hash using the - // public key, pub. Its return value records whether the signature is valid. - verifyASN1 := func(pub *ecdsa.PublicKey, hash, sig []byte) bool { - var ( - r, s = &big.Int{}, &big.Int{} - inner cryptobyte.String - ) - input := cryptobyte.String(sig) - if !input.ReadASN1(&inner, asn1.SEQUENCE) || - !input.Empty() || - !inner.ReadASN1Integer(r) || - !inner.ReadASN1Integer(s) || - !inner.Empty() { - return false - } - return ecdsa.Verify(pub, hash, r, s) - } - - type args struct { - req *apiv1.CreateSignerRequest - } - tests := []struct { - name string - args args - algorithm apiv1.SignatureAlgorithm - signerOpts crypto.SignerOpts - wantErr bool - }{ - // SoftHSM2 - {"RSA", args{&apiv1.CreateSignerRequest{ - SigningKey: "pkcs11:id=7371;object=rsa-key", - }}, apiv1.SHA256WithRSA, crypto.SHA256, false}, - {"RSA PSS", args{&apiv1.CreateSignerRequest{ - SigningKey: "pkcs11:id=7372;object=rsa-pss-key", - }}, apiv1.SHA256WithRSAPSS, &rsa.PSSOptions{ - SaltLength: rsa.PSSSaltLengthEqualsHash, - Hash: crypto.SHA256, - }, false}, - {"ECDSA P256", args{&apiv1.CreateSignerRequest{ - SigningKey: "pkcs11:id=7373;object=ecdsa-p256-key", - }}, apiv1.ECDSAWithSHA256, crypto.SHA256, false}, - {"ECDSA P384", args{&apiv1.CreateSignerRequest{ - SigningKey: "pkcs11:id=7374;object=ecdsa-p384-key", - }}, apiv1.ECDSAWithSHA384, crypto.SHA384, false}, - {"ECDSA P521", args{&apiv1.CreateSignerRequest{ - SigningKey: "pkcs11:id=7375;object=ecdsa-p521-key", - }}, apiv1.ECDSAWithSHA512, crypto.SHA512, false}, - {"fail SigningKey", args{&apiv1.CreateSignerRequest{ - SigningKey: "", - }}, 0, nil, true}, - {"fail uri", args{&apiv1.CreateSignerRequest{ - SigningKey: "https:id=7375;object=ecdsa-p521-key", - }}, 0, nil, true}, - {"fail FindKeyPair", args{&apiv1.CreateSignerRequest{ - SigningKey: "pkcs11:foo=bar", - }}, 0, nil, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - got, err := k.CreateSigner(tt.args.req) - if (err != nil) != tt.wantErr { - t.Errorf("PKCS11.CreateSigner() error = %v, wantErr %v", err, tt.wantErr) - return - } - - if got != nil { - hash := tt.signerOpts.HashFunc() - h := hash.New() - h.Write(data) - digest := h.Sum(nil) - sig, err := got.Sign(rand.Reader, digest, tt.signerOpts) - if err != nil { - t.Errorf("cyrpto.Signer.Sign() error = %v", err) - } - - switch tt.algorithm { - case apiv1.SHA256WithRSA, apiv1.SHA384WithRSA, apiv1.SHA512WithRSA: - pub := got.Public().(*rsa.PublicKey) - if err := rsa.VerifyPKCS1v15(pub, hash, digest, sig); err != nil { - t.Errorf("rsa.VerifyPKCS1v15() error = %v", err) - } - case apiv1.UnspecifiedSignAlgorithm, apiv1.SHA256WithRSAPSS, apiv1.SHA384WithRSAPSS, apiv1.SHA512WithRSAPSS: - pub := got.Public().(*rsa.PublicKey) - if err := rsa.VerifyPSS(pub, hash, digest, sig, tt.signerOpts.(*rsa.PSSOptions)); err != nil { - t.Errorf("rsa.VerifyPSS() error = %v", err) - } - case apiv1.ECDSAWithSHA256, apiv1.ECDSAWithSHA384, apiv1.ECDSAWithSHA512: - pub := got.Public().(*ecdsa.PublicKey) - if !verifyASN1(pub, digest, sig) { - t.Error("ecdsa.VerifyASN1() failed") - } - default: - t.Errorf("signature algorithm %s is not supported", tt.algorithm) - } - - } - - }) - } -} - -func TestPKCS11_CreateDecrypter(t *testing.T) { - k := setupPKCS11(t) - data := []byte("buggy-coheir-RUBRIC-rabbet-liberal-eaglet-khartoum-stagger") - - type args struct { - req *apiv1.CreateDecrypterRequest - } - tests := []struct { - name string - args args - wantErr bool - }{ - {"RSA", args{&apiv1.CreateDecrypterRequest{ - DecryptionKey: "pkcs11:id=7371;object=rsa-key", - }}, false}, - {"RSA PSS", args{&apiv1.CreateDecrypterRequest{ - DecryptionKey: "pkcs11:id=7372;object=rsa-pss-key", - }}, false}, - {"ECDSA P256", args{&apiv1.CreateDecrypterRequest{ - DecryptionKey: "pkcs11:id=7373;object=ecdsa-p256-key", - }}, true}, - {"ECDSA P384", args{&apiv1.CreateDecrypterRequest{ - DecryptionKey: "pkcs11:id=7374;object=ecdsa-p384-key", - }}, true}, - {"ECDSA P521", args{&apiv1.CreateDecrypterRequest{ - DecryptionKey: "pkcs11:id=7375;object=ecdsa-p521-key", - }}, true}, - {"fail DecryptionKey", args{&apiv1.CreateDecrypterRequest{ - DecryptionKey: "", - }}, true}, - {"fail uri", args{&apiv1.CreateDecrypterRequest{ - DecryptionKey: "https:id=7375;object=ecdsa-p521-key", - }}, true}, - {"fail FindKeyPair", args{&apiv1.CreateDecrypterRequest{ - DecryptionKey: "pkcs11:foo=bar", - }}, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - got, err := k.CreateDecrypter(tt.args.req) - if (err != nil) != tt.wantErr { - t.Errorf("PKCS11.CreateDecrypter() error = %v, wantErr %v", err, tt.wantErr) - return - } - - if got != nil { - pub := got.Public().(*rsa.PublicKey) - // PKCS#1 v1.5 - enc, err := rsa.EncryptPKCS1v15(rand.Reader, pub, data) - if err != nil { - t.Errorf("rsa.EncryptPKCS1v15() error = %v", err) - return - } - dec, err := got.Decrypt(rand.Reader, enc, nil) - if err != nil { - t.Errorf("PKCS1v15.Decrypt() error = %v", err) - } else if !bytes.Equal(dec, data) { - t.Errorf("PKCS1v15.Decrypt() failed got = %s, want = %s", dec, data) - } - - // RSA-OAEP - enc, err = rsa.EncryptOAEP(crypto.SHA256.New(), rand.Reader, pub, data, []byte("label")) - if err != nil { - t.Errorf("rsa.EncryptOAEP() error = %v", err) - return - } - dec, err = got.Decrypt(rand.Reader, enc, &rsa.OAEPOptions{ - Hash: crypto.SHA256, - Label: []byte("label"), - }) - if err != nil { - t.Errorf("RSA-OAEP.Decrypt() error = %v", err) - } else if !bytes.Equal(dec, data) { - t.Errorf("RSA-OAEP.Decrypt() RSA-OAEP failed got = %s, want = %s", dec, data) - } - } - }) - } -} - -func TestPKCS11_LoadCertificate(t *testing.T) { - k := setupPKCS11(t) - - getCertFn := func(i, j int) func() *x509.Certificate { - return func() *x509.Certificate { - return testCerts[i].Certificates[j] - } - } - - type args struct { - req *apiv1.LoadCertificateRequest - } - tests := []struct { - name string - args args - wantFn func() *x509.Certificate - wantErr bool - }{ - {"load", args{&apiv1.LoadCertificateRequest{ - Name: "pkcs11:id=7376;object=test-root", - }}, getCertFn(0, 0), false}, - {"load by id", args{&apiv1.LoadCertificateRequest{ - Name: "pkcs11:id=7376", - }}, getCertFn(0, 0), false}, - {"load by label", args{&apiv1.LoadCertificateRequest{ - Name: "pkcs11:object=test-root", - }}, getCertFn(0, 0), false}, - {"load by serial", args{&apiv1.LoadCertificateRequest{ - Name: "pkcs11:serial=64", - }}, getCertFn(0, 0), false}, - {"fail missing", args{&apiv1.LoadCertificateRequest{ - Name: "pkcs11:id=9999;object=test-root", - }}, nil, true}, - {"fail name", args{&apiv1.LoadCertificateRequest{ - Name: "", - }}, nil, true}, - {"fail scheme", args{&apiv1.LoadCertificateRequest{ - Name: "foo:id=7376;object=test-root", - }}, nil, true}, - {"fail serial", args{&apiv1.LoadCertificateRequest{ - Name: "pkcs11:serial=foo", - }}, nil, true}, - {"fail FindCertificate", args{&apiv1.LoadCertificateRequest{ - Name: "pkcs11:foo=bar", - }}, nil, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - got, err := k.LoadCertificate(tt.args.req) - if (err != nil) != tt.wantErr { - t.Errorf("PKCS11.LoadCertificate() error = %v, wantErr %v", err, tt.wantErr) - return - } - var want *x509.Certificate - if tt.wantFn != nil { - want = tt.wantFn() - got.Raw, got.RawIssuer, got.RawSubject, got.RawTBSCertificate, got.RawSubjectPublicKeyInfo = nil, nil, nil, nil, nil - want.Raw, want.RawIssuer, want.RawSubject, want.RawTBSCertificate, want.RawSubjectPublicKeyInfo = nil, nil, nil, nil, nil - } - if !reflect.DeepEqual(got, want) { - t.Errorf("PKCS11.LoadCertificate() = %v, want %v", got, want) - } - }) - } -} - -func TestPKCS11_StoreCertificate(t *testing.T) { - k := setupPKCS11(t) - - pub, priv, err := ed25519.GenerateKey(rand.Reader) - if err != nil { - t.Fatalf("ed25519.GenerateKey() error = %v", err) - } - - cert, err := generateCertificate(pub, priv) - if err != nil { - t.Fatalf("x509.CreateCertificate() error = %v", err) - } - - // Make sure to delete the created certificate - t.Cleanup(func() { - k.DeleteCertificate(testObject) - k.DeleteCertificate(testObjectAlt) - }) - - type args struct { - req *apiv1.StoreCertificateRequest - } - tests := []struct { - name string - args args - wantErr bool - }{ - {"ok", args{&apiv1.StoreCertificateRequest{ - Name: testObject, - Certificate: cert, - }}, false}, - {"ok extractable", args{&apiv1.StoreCertificateRequest{ - Name: testObjectAlt, - Certificate: cert, - Extractable: true, - }}, false}, - {"fail already exists", args{&apiv1.StoreCertificateRequest{ - Name: testObject, - Certificate: cert, - }}, true}, - {"fail name", args{&apiv1.StoreCertificateRequest{ - Name: "", - Certificate: cert, - }}, true}, - {"fail certificate", args{&apiv1.StoreCertificateRequest{ - Name: testObject, - Certificate: nil, - }}, true}, - {"fail uri", args{&apiv1.StoreCertificateRequest{ - Name: "http:id=7770;object=create-cert", - Certificate: cert, - }}, true}, - {"fail missing id", args{&apiv1.StoreCertificateRequest{ - Name: "pkcs11:object=create-cert", - Certificate: cert, - }}, true}, - {"fail missing object", args{&apiv1.StoreCertificateRequest{ - Name: "pkcs11:id=7770;object=", - Certificate: cert, - }}, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - if tt.args.req.Extractable { - if testModule == "SoftHSM2" { - t.Skip("Extractable certificates are not supported on SoftHSM2") - } - } - if err := k.StoreCertificate(tt.args.req); (err != nil) != tt.wantErr { - t.Errorf("PKCS11.StoreCertificate() error = %v, wantErr %v", err, tt.wantErr) - } - if !tt.wantErr { - got, err := k.LoadCertificate(&apiv1.LoadCertificateRequest{ - Name: tt.args.req.Name, - }) - if err != nil { - t.Errorf("PKCS11.LoadCertificate() error = %v", err) - } - if !reflect.DeepEqual(got, cert) { - t.Errorf("PKCS11.LoadCertificate() = %v, want %v", got, cert) - } - } - }) - } -} - -func TestPKCS11_DeleteKey(t *testing.T) { - k := setupPKCS11(t) - - type args struct { - uri string - } - tests := []struct { - name string - args args - wantErr bool - }{ - {"delete", args{testObject}, false}, - {"delete by id", args{testObjectByID}, false}, - {"delete by label", args{testObjectByLabel}, false}, - {"delete missing", args{"pkcs11:id=9999;object=missing-key"}, false}, - {"fail name", args{""}, true}, - {"fail FindKeyPair", args{"pkcs11:foo=bar"}, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - if _, err := k.CreateKey(&apiv1.CreateKeyRequest{ - Name: testObject, - }); err != nil { - t.Fatalf("PKCS1.CreateKey() error = %v", err) - } - if err := k.DeleteKey(tt.args.uri); (err != nil) != tt.wantErr { - t.Errorf("PKCS11.DeleteKey() error = %v, wantErr %v", err, tt.wantErr) - } - if _, err := k.GetPublicKey(&apiv1.GetPublicKeyRequest{ - Name: tt.args.uri, - }); err == nil { - t.Error("PKCS11.GetPublicKey() public key found and not expected") - } - // Make sure to delete the created one. - if err := k.DeleteKey(testObject); err != nil { - t.Errorf("PKCS11.DeleteKey() error = %v", err) - } - }) - } -} - -func TestPKCS11_DeleteCertificate(t *testing.T) { - k := setupPKCS11(t) - - pub, priv, err := ed25519.GenerateKey(rand.Reader) - if err != nil { - t.Fatalf("ed25519.GenerateKey() error = %v", err) - } - - cert, err := generateCertificate(pub, priv) - if err != nil { - t.Fatalf("x509.CreateCertificate() error = %v", err) - } - - type args struct { - uri string - } - tests := []struct { - name string - args args - wantErr bool - }{ - {"delete", args{testObject}, false}, - {"delete by id", args{testObjectByID}, false}, - {"delete by label", args{testObjectByLabel}, false}, - {"delete missing", args{"pkcs11:id=9999;object=missing-key"}, false}, - {"fail name", args{""}, true}, - {"fail DeleteCertificate", args{"pkcs11:foo=bar"}, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - if err := k.StoreCertificate(&apiv1.StoreCertificateRequest{ - Name: testObject, - Certificate: cert, - }); err != nil { - t.Fatalf("PKCS11.StoreCertificate() error = %v", err) - } - if err := k.DeleteCertificate(tt.args.uri); (err != nil) != tt.wantErr { - t.Errorf("PKCS11.DeleteCertificate() error = %v, wantErr %v", err, tt.wantErr) - } - if _, err := k.LoadCertificate(&apiv1.LoadCertificateRequest{ - Name: tt.args.uri, - }); err == nil { - t.Error("PKCS11.LoadCertificate() certificate found and not expected") - } - // Make sure to delete the created one. - if err := k.DeleteCertificate(testObject); err != nil { - t.Errorf("PKCS11.DeleteCertificate() error = %v", err) - } - }) - } -} - -func TestPKCS11_Close(t *testing.T) { - k := mustPKCS11(t) - tests := []struct { - name string - wantErr bool - }{ - {"ok", false}, - {"second", false}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - if err := k.Close(); (err != nil) != tt.wantErr { - t.Errorf("PKCS11.Close() error = %v, wantErr %v", err, tt.wantErr) - } - }) - } -} diff --git a/kms/pkcs11/setup_test.go b/kms/pkcs11/setup_test.go deleted file mode 100644 index 902d89ac..00000000 --- a/kms/pkcs11/setup_test.go +++ /dev/null @@ -1,145 +0,0 @@ -//go:build cgo -// +build cgo - -package pkcs11 - -import ( - "crypto" - "crypto/rand" - "crypto/x509" - "crypto/x509/pkix" - "math/big" - "time" - - "github.com/pkg/errors" - "github.com/smallstep/certificates/kms/apiv1" -) - -var ( - testModule = "" - testObject = "pkcs11:id=7370;object=test-name" - testObjectAlt = "pkcs11:id=7377;object=alt-test-name" - testObjectByID = "pkcs11:id=7370" - testObjectByLabel = "pkcs11:object=test-name" - testKeys = []struct { - Name string - SignatureAlgorithm apiv1.SignatureAlgorithm - Bits int - }{ - {"pkcs11:id=7371;object=rsa-key", apiv1.SHA256WithRSA, 2048}, - {"pkcs11:id=7372;object=rsa-pss-key", apiv1.SHA256WithRSAPSS, DefaultRSASize}, - {"pkcs11:id=7373;object=ecdsa-p256-key", apiv1.ECDSAWithSHA256, 0}, - {"pkcs11:id=7374;object=ecdsa-p384-key", apiv1.ECDSAWithSHA384, 0}, - {"pkcs11:id=7375;object=ecdsa-p521-key", apiv1.ECDSAWithSHA512, 0}, - } - - testCerts = []struct { - Name string - Key string - Certificates []*x509.Certificate - }{ - {"pkcs11:id=7376;object=test-root", "pkcs11:id=7373;object=ecdsa-p256-key", nil}, - } -) - -type TBTesting interface { - Helper() - Cleanup(f func()) - Log(args ...interface{}) - Errorf(format string, args ...interface{}) - Fatalf(format string, args ...interface{}) - Skipf(format string, args ...interface{}) -} - -func generateCertificate(pub crypto.PublicKey, signer crypto.Signer) (*x509.Certificate, error) { - now := time.Now() - template := &x509.Certificate{ - Subject: pkix.Name{CommonName: "Test Root Certificate"}, - Issuer: pkix.Name{CommonName: "Test Root Certificate"}, - IsCA: true, - MaxPathLen: 1, - KeyUsage: x509.KeyUsageCertSign | x509.KeyUsageCRLSign, - NotBefore: now, - NotAfter: now.Add(time.Hour), - SerialNumber: big.NewInt(100), - } - - b, err := x509.CreateCertificate(rand.Reader, template, template, pub, signer) - if err != nil { - return nil, err - } - - return x509.ParseCertificate(b) -} - -func setup(t TBTesting, k *PKCS11) { - t.Log("Running using", testModule) - for _, tk := range testKeys { - _, err := k.CreateKey(&apiv1.CreateKeyRequest{ - Name: tk.Name, - SignatureAlgorithm: tk.SignatureAlgorithm, - Bits: tk.Bits, - }) - if err != nil && !errors.Is(errors.Cause(err), apiv1.ErrAlreadyExists{ - Message: tk.Name + " already exists", - }) { - t.Errorf("PKCS11.GetPublicKey() error = %v", err) - } - } - - for i, c := range testCerts { - signer, err := k.CreateSigner(&apiv1.CreateSignerRequest{ - SigningKey: c.Key, - }) - if err != nil { - t.Errorf("PKCS11.CreateSigner() error = %v", err) - continue - } - cert, err := generateCertificate(signer.Public(), signer) - if err != nil { - t.Errorf("x509.CreateCertificate() error = %v", err) - continue - } - if err := k.StoreCertificate(&apiv1.StoreCertificateRequest{ - Name: c.Name, - Certificate: cert, - }); err != nil && !errors.Is(errors.Cause(err), apiv1.ErrAlreadyExists{ - Message: c.Name + " already exists", - }) { - t.Errorf("PKCS1.StoreCertificate() error = %v", err) - continue - } - testCerts[i].Certificates = append(testCerts[i].Certificates, cert) - } -} - -func teardown(t TBTesting, k *PKCS11) { - testObjects := []string{testObject, testObjectByID, testObjectByLabel} - for _, name := range testObjects { - if err := k.DeleteKey(name); err != nil { - t.Errorf("PKCS11.DeleteKey() error = %v", err) - } - if err := k.DeleteCertificate(name); err != nil { - t.Errorf("PKCS11.DeleteCertificate() error = %v", err) - } - } - for _, tk := range testKeys { - if err := k.DeleteKey(tk.Name); err != nil { - t.Errorf("PKCS11.DeleteKey() error = %v", err) - } - } - for _, tc := range testCerts { - if err := k.DeleteCertificate(tc.Name); err != nil { - t.Errorf("PKCS11.DeleteCertificate() error = %v", err) - } - } -} - -func setupPKCS11(t TBTesting) *PKCS11 { - t.Helper() - k := mustPKCS11(t) - t.Cleanup(func() { - k.Close() - }) - return k -} diff --git a/kms/pkcs11/softhsm2_test.go b/kms/pkcs11/softhsm2_test.go deleted file mode 100644 index 6fc0c248..00000000 --- a/kms/pkcs11/softhsm2_test.go +++ /dev/null @@ -1,62 +0,0 @@ -//go:build cgo && softhsm2 -// +build cgo,softhsm2 - -package pkcs11 - -import ( - "runtime" - "sync" - - "github.com/ThalesIgnite/crypto11" -) - -var softHSM2Once sync.Once - -// mustPKCS11 configures a *PKCS11 KMS to be used with SoftHSM2. To initialize -// these tests, we should run: -// -// softhsm2-util --init-token --free \ -// --token pkcs11-test --label pkcs11-test \ -// --so-pin password --pin password -// -// To delete we should run: -// -// softhsm2-util --delete-token --token pkcs11-test -func mustPKCS11(t TBTesting) *PKCS11 { - t.Helper() - testModule = "SoftHSM2" - if runtime.GOARCH != "amd64" { - t.Fatalf("softHSM2 test skipped on %s:%s", runtime.GOOS, runtime.GOARCH) - } - - var path string - switch runtime.GOOS { - case "darwin": - path = "/usr/local/lib/softhsm/libsofthsm2.so" - case "linux": - path = "/usr/lib/softhsm/libsofthsm2.so" - default: - t.Skipf("softHSM2 test skipped on %s", runtime.GOOS) - return nil - } - p11, err := crypto11.Configure(&crypto11.Config{ - Path: path, - TokenLabel: "pkcs11-test", - Pin: "password", - }) - if err != nil { - t.Fatalf("failed to configure softHSM2 on %s: %v", runtime.GOOS, err) - } - - k := &PKCS11{ - p11: p11, - } - - // Setup - softHSM2Once.Do(func() { - teardown(t, k) - setup(t, k) - }) - - return k -} diff --git a/kms/pkcs11/yubihsm2_test.go b/kms/pkcs11/yubihsm2_test.go deleted file mode 100644 index 49eb13d1..00000000 --- a/kms/pkcs11/yubihsm2_test.go +++ /dev/null @@ -1,56 +0,0 @@ -//go:build cgo && yubihsm2 -// +build cgo,yubihsm2 - -package pkcs11 - -import ( - "runtime" - "sync" - - "github.com/ThalesIgnite/crypto11" -) - -var yubiHSM2Once sync.Once - -// mustPKCS11 configures a *PKCS11 KMS to be used with YubiHSM2. To initialize -// these tests, we should run: -// -// yubihsm-connector -d -func mustPKCS11(t TBTesting) *PKCS11 { - t.Helper() - testModule = "YubiHSM2" - if runtime.GOARCH != "amd64" { - t.Skipf("yubiHSM2 test skipped on %s:%s", runtime.GOOS, runtime.GOARCH) - } - - var path string - switch runtime.GOOS { - case "darwin": - path = "/usr/local/lib/pkcs11/yubihsm_pkcs11.dylib" - case "linux": - path = "/usr/lib/x86_64-linux-gnu/pkcs11/yubihsm_pkcs11.so" - default: - t.Skipf("yubiHSM2 test skipped on %s", runtime.GOOS) - return nil - } - p11, err := crypto11.Configure(&crypto11.Config{ - Path: path, - TokenLabel: "YubiHSM", - Pin: "0001password", - }) - if err != nil { - t.Fatalf("failed to configure YubiHSM2 on %s: %v", runtime.GOOS, err) - } - - k := &PKCS11{ - p11: p11, - } - - // Setup - yubiHSM2Once.Do(func() { - teardown(t, k) - setup(t, k) - }) - - return k -} diff --git a/kms/softkms/softkms.go b/kms/softkms/softkms.go deleted file mode 100644 index a2f43c31..00000000 --- a/kms/softkms/softkms.go +++ /dev/null @@ -1,183 +0,0 @@ -package softkms - -import ( - "context" - "crypto" - "crypto/ecdsa" - "crypto/ed25519" - "crypto/rsa" - "crypto/x509" - - "github.com/pkg/errors" - "github.com/smallstep/certificates/kms/apiv1" - "go.step.sm/cli-utils/ui" - "go.step.sm/crypto/keyutil" - "go.step.sm/crypto/pemutil" -) - -type algorithmAttributes struct { - Type string - Curve string -} - -// DefaultRSAKeySize is the default size for RSA keys. -const DefaultRSAKeySize = 3072 - -var signatureAlgorithmMapping = map[apiv1.SignatureAlgorithm]algorithmAttributes{ - apiv1.UnspecifiedSignAlgorithm: {"EC", "P-256"}, - apiv1.SHA256WithRSA: {"RSA", ""}, - apiv1.SHA384WithRSA: {"RSA", ""}, - apiv1.SHA512WithRSA: {"RSA", ""}, - apiv1.SHA256WithRSAPSS: {"RSA", ""}, - apiv1.SHA384WithRSAPSS: {"RSA", ""}, - apiv1.SHA512WithRSAPSS: {"RSA", ""}, - apiv1.ECDSAWithSHA256: {"EC", "P-256"}, - apiv1.ECDSAWithSHA384: {"EC", "P-384"}, - apiv1.ECDSAWithSHA512: {"EC", "P-521"}, - apiv1.PureEd25519: {"OKP", "Ed25519"}, -} - -// generateKey is used for testing purposes. -var generateKey = func(kty, crv string, size int) (interface{}, interface{}, error) { - if kty == "RSA" && size == 0 { - size = DefaultRSAKeySize - } - return keyutil.GenerateKeyPair(kty, crv, size) -} - -// SoftKMS is a key manager that uses keys stored in disk. -type SoftKMS struct{} - -// New returns a new SoftKMS. -func New(ctx context.Context, opts apiv1.Options) (*SoftKMS, error) { - return &SoftKMS{}, nil -} - -func init() { - pemutil.PromptPassword = func(msg string) ([]byte, error) { - return ui.PromptPassword(msg) - } - apiv1.Register(apiv1.SoftKMS, func(ctx context.Context, opts apiv1.Options) (apiv1.KeyManager, error) { - return New(ctx, opts) - }) -} - -// Close is a noop that just returns nil. -func (k *SoftKMS) Close() error { - return nil -} - -// CreateSigner returns a new signer configured with the given signing key. -func (k *SoftKMS) CreateSigner(req *apiv1.CreateSignerRequest) (crypto.Signer, error) { - var opts []pemutil.Options - if req.Password != nil { - opts = append(opts, pemutil.WithPassword(req.Password)) - } - - switch { - case req.Signer != nil: - return req.Signer, nil - case len(req.SigningKeyPEM) != 0: - v, err := pemutil.ParseKey(req.SigningKeyPEM, opts...) - if err != nil { - return nil, err - } - sig, ok := v.(crypto.Signer) - if !ok { - return nil, errors.New("signingKeyPEM is not a crypto.Signer") - } - return sig, nil - case req.SigningKey != "": - v, err := pemutil.Read(req.SigningKey, opts...) - if err != nil { - return nil, err - } - sig, ok := v.(crypto.Signer) - if !ok { - return nil, errors.New("signingKey is not a crypto.Signer") - } - return sig, nil - default: - return nil, errors.New("failed to load softKMS: please define signingKeyPEM or signingKey") - } -} - -// CreateKey generates a new key using Golang crypto and returns both public and -// private key. -func (k *SoftKMS) CreateKey(req *apiv1.CreateKeyRequest) (*apiv1.CreateKeyResponse, error) { - v, ok := signatureAlgorithmMapping[req.SignatureAlgorithm] - if !ok { - return nil, errors.Errorf("softKMS does not support signature algorithm '%s'", req.SignatureAlgorithm) - } - - pub, priv, err := generateKey(v.Type, v.Curve, req.Bits) - if err != nil { - return nil, err - } - signer, ok := priv.(crypto.Signer) - if !ok { - return nil, errors.Errorf("softKMS createKey result is not a crypto.Signer: type %T", priv) - } - - return &apiv1.CreateKeyResponse{ - Name: req.Name, - PublicKey: pub, - PrivateKey: priv, - CreateSignerRequest: apiv1.CreateSignerRequest{ - Signer: signer, - }, - }, nil -} - -// GetPublicKey returns the public key from the file passed in the request name. -func (k *SoftKMS) GetPublicKey(req *apiv1.GetPublicKeyRequest) (crypto.PublicKey, error) { - v, err := pemutil.Read(req.Name) - if err != nil { - return nil, err - } - - switch vv := v.(type) { - case *x509.Certificate: - return vv.PublicKey, nil - case *rsa.PublicKey, *ecdsa.PublicKey, ed25519.PublicKey: - return vv, nil - default: - return nil, errors.Errorf("unsupported public key type %T", v) - } -} - -// CreateDecrypter creates a new crypto.Decrypter backed by disk/software -func (k *SoftKMS) CreateDecrypter(req *apiv1.CreateDecrypterRequest) (crypto.Decrypter, error) { - - var opts []pemutil.Options - if req.Password != nil { - opts = append(opts, pemutil.WithPassword(req.Password)) - } - - switch { - case req.Decrypter != nil: - return req.Decrypter, nil - case len(req.DecryptionKeyPEM) != 0: - v, err := pemutil.ParseKey(req.DecryptionKeyPEM, opts...) - if err != nil { - return nil, err - } - decrypter, ok := v.(crypto.Decrypter) - if !ok { - return nil, errors.New("decryptorKeyPEM is not a crypto.Decrypter") - } - return decrypter, nil - case req.DecryptionKey != "": - v, err := pemutil.Read(req.DecryptionKey, opts...) - if err != nil { - return nil, err - } - decrypter, ok := v.(crypto.Decrypter) - if !ok { - return nil, errors.New("decryptionKey is not a crypto.Decrypter") - } - return decrypter, nil - default: - return nil, errors.New("failed to load softKMS: please define decryptionKeyPEM or decryptionKey") - } -} diff --git a/kms/softkms/softkms_test.go b/kms/softkms/softkms_test.go deleted file mode 100644 index 907a7efe..00000000 --- a/kms/softkms/softkms_test.go +++ /dev/null @@ -1,381 +0,0 @@ -package softkms - -import ( - "context" - "crypto" - "crypto/ecdsa" - "crypto/ed25519" - "crypto/elliptic" - "crypto/rand" - "crypto/rsa" - "crypto/x509" - "encoding/pem" - "fmt" - "os" - "reflect" - "testing" - - "github.com/smallstep/certificates/kms/apiv1" - "go.step.sm/crypto/pemutil" -) - -func TestNew(t *testing.T) { - type args struct { - ctx context.Context - opts apiv1.Options - } - tests := []struct { - name string - args args - want *SoftKMS - wantErr bool - }{ - {"ok", args{context.Background(), apiv1.Options{}}, &SoftKMS{}, false}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - got, err := New(tt.args.ctx, tt.args.opts) - if (err != nil) != tt.wantErr { - t.Errorf("New() error = %v, wantErr %v", err, tt.wantErr) - return - } - if !reflect.DeepEqual(got, tt.want) { - t.Errorf("New() = %v, want %v", got, tt.want) - } - }) - } -} - -func TestSoftKMS_Close(t *testing.T) { - tests := []struct { - name string - wantErr bool - }{ - {"ok", false}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - k := &SoftKMS{} - if err := k.Close(); (err != nil) != tt.wantErr { - t.Errorf("SoftKMS.Close() error = %v, wantErr %v", err, tt.wantErr) - } - }) - } -} - -func TestSoftKMS_CreateSigner(t *testing.T) { - pk, err := ecdsa.GenerateKey(elliptic.P256(), rand.Reader) - if err != nil { - t.Fatal(err) - } - pemBlock, err := pemutil.Serialize(pk) - if err != nil { - t.Fatal(err) - } - pemBlockPassword, err := pemutil.Serialize(pk, pemutil.WithPassword([]byte("pass"))) - if err != nil { - t.Fatal(err) - } - - // Read and decode file using standard packages - b, err := os.ReadFile("testdata/priv.pem") - if err != nil { - t.Fatal(err) - } - block, _ := pem.Decode(b) - block.Bytes, err = x509.DecryptPEMBlock(block, []byte("pass")) //nolint - if err != nil { - t.Fatal(err) - } - pk2, err := x509.ParseECPrivateKey(block.Bytes) - if err != nil { - t.Fatal(err) - } - - // Create a public PEM - b, err = x509.MarshalPKIXPublicKey(pk.Public()) - if err != nil { - t.Fatal(err) - } - pub := pem.EncodeToMemory(&pem.Block{ - Type: "PUBLIC KEY", - Bytes: b, - }) - - type args struct { - req *apiv1.CreateSignerRequest - } - tests := []struct { - name string - args args - want crypto.Signer - wantErr bool - }{ - {"signer", args{&apiv1.CreateSignerRequest{Signer: pk}}, pk, false}, - {"pem", args{&apiv1.CreateSignerRequest{SigningKeyPEM: pem.EncodeToMemory(pemBlock)}}, pk, false}, - {"pem password", args{&apiv1.CreateSignerRequest{SigningKeyPEM: pem.EncodeToMemory(pemBlockPassword), Password: []byte("pass")}}, pk, false}, - {"file", args{&apiv1.CreateSignerRequest{SigningKey: "testdata/priv.pem", Password: []byte("pass")}}, pk2, false}, - {"fail", args{&apiv1.CreateSignerRequest{}}, nil, true}, - {"fail bad pem", args{&apiv1.CreateSignerRequest{SigningKeyPEM: []byte("bad pem")}}, nil, true}, - {"fail bad password", args{&apiv1.CreateSignerRequest{SigningKey: "testdata/priv.pem", Password: []byte("bad-pass")}}, nil, true}, - {"fail not a signer", args{&apiv1.CreateSignerRequest{SigningKeyPEM: pub}}, nil, true}, - {"fail not a signer from file", args{&apiv1.CreateSignerRequest{SigningKey: "testdata/pub.pem"}}, nil, true}, - {"fail missing", args{&apiv1.CreateSignerRequest{SigningKey: "testdata/missing"}}, nil, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - k := &SoftKMS{} - got, err := k.CreateSigner(tt.args.req) - if (err != nil) != tt.wantErr { - t.Errorf("SoftKMS.CreateSigner() error = %v, wantErr %v", err, tt.wantErr) - return - } - if !reflect.DeepEqual(got, tt.want) { - t.Errorf("SoftKMS.CreateSigner() = %v, want %v", got, tt.want) - } - }) - } -} - -func restoreGenerateKey() func() { - oldGenerateKey := generateKey - return func() { - generateKey = oldGenerateKey - } -} - -func TestSoftKMS_CreateKey(t *testing.T) { - fn := restoreGenerateKey() - defer fn() - - p256, err := ecdsa.GenerateKey(elliptic.P256(), rand.Reader) - if err != nil { - t.Fatal(err) - } - rsa2048, err := rsa.GenerateKey(rand.Reader, 2048) - if err != nil { - t.Fatal(err) - } - edpub, edpriv, err := ed25519.GenerateKey(rand.Reader) - if err != nil { - t.Fatal(err) - } - - type args struct { - req *apiv1.CreateKeyRequest - } - type params struct { - kty string - crv string - size int - } - tests := []struct { - name string - args args - generateKey func() (interface{}, interface{}, error) - want *apiv1.CreateKeyResponse - wantParams params - wantErr bool - }{ - {"p256", args{&apiv1.CreateKeyRequest{Name: "p256", SignatureAlgorithm: apiv1.ECDSAWithSHA256}}, func() (interface{}, interface{}, error) { - return p256.Public(), p256, nil - }, &apiv1.CreateKeyResponse{Name: "p256", PublicKey: p256.Public(), PrivateKey: p256, CreateSignerRequest: apiv1.CreateSignerRequest{Signer: p256}}, params{"EC", "P-256", 0}, false}, - {"rsa", args{&apiv1.CreateKeyRequest{Name: "rsa3072", SignatureAlgorithm: apiv1.SHA256WithRSA}}, func() (interface{}, interface{}, error) { - return rsa2048.Public(), rsa2048, nil - }, &apiv1.CreateKeyResponse{Name: "rsa3072", PublicKey: rsa2048.Public(), PrivateKey: rsa2048, CreateSignerRequest: apiv1.CreateSignerRequest{Signer: rsa2048}}, params{"RSA", "", 0}, false}, - {"rsa2048", args{&apiv1.CreateKeyRequest{Name: "rsa2048", SignatureAlgorithm: apiv1.SHA256WithRSA, Bits: 2048}}, func() (interface{}, interface{}, error) { - return rsa2048.Public(), rsa2048, nil - }, &apiv1.CreateKeyResponse{Name: "rsa2048", PublicKey: rsa2048.Public(), PrivateKey: rsa2048, CreateSignerRequest: apiv1.CreateSignerRequest{Signer: rsa2048}}, params{"RSA", "", 2048}, false}, - {"rsaPSS2048", args{&apiv1.CreateKeyRequest{Name: "rsa2048", SignatureAlgorithm: apiv1.SHA256WithRSAPSS, Bits: 2048}}, func() (interface{}, interface{}, error) { - return rsa2048.Public(), rsa2048, nil - }, &apiv1.CreateKeyResponse{Name: "rsa2048", PublicKey: rsa2048.Public(), PrivateKey: rsa2048, CreateSignerRequest: apiv1.CreateSignerRequest{Signer: rsa2048}}, params{"RSA", "", 2048}, false}, - {"ed25519", args{&apiv1.CreateKeyRequest{Name: "ed25519", SignatureAlgorithm: apiv1.PureEd25519}}, func() (interface{}, interface{}, error) { - return edpub, edpriv, nil - }, &apiv1.CreateKeyResponse{Name: "ed25519", PublicKey: edpub, PrivateKey: edpriv, CreateSignerRequest: apiv1.CreateSignerRequest{Signer: edpriv}}, params{"OKP", "Ed25519", 0}, false}, - {"default", args{&apiv1.CreateKeyRequest{Name: "default"}}, func() (interface{}, interface{}, error) { - return p256.Public(), p256, nil - }, &apiv1.CreateKeyResponse{Name: "default", PublicKey: p256.Public(), PrivateKey: p256, CreateSignerRequest: apiv1.CreateSignerRequest{Signer: p256}}, params{"EC", "P-256", 0}, false}, - {"fail algorithm", args{&apiv1.CreateKeyRequest{Name: "fail", SignatureAlgorithm: apiv1.SignatureAlgorithm(100)}}, func() (interface{}, interface{}, error) { - return p256.Public(), p256, nil - }, nil, params{}, true}, - {"fail generate key", args{&apiv1.CreateKeyRequest{Name: "fail", SignatureAlgorithm: apiv1.ECDSAWithSHA256}}, func() (interface{}, interface{}, error) { - return nil, nil, fmt.Errorf("an error") - }, nil, params{"EC", "P-256", 0}, true}, - {"fail no signer", args{&apiv1.CreateKeyRequest{Name: "fail", SignatureAlgorithm: apiv1.ECDSAWithSHA256}}, func() (interface{}, interface{}, error) { - return 1, 2, nil - }, nil, params{"EC", "P-256", 0}, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - k := &SoftKMS{} - generateKey = func(kty, crv string, size int) (interface{}, interface{}, error) { - if tt.wantParams.kty != kty { - t.Errorf("GenerateKey() kty = %s, want %s", kty, tt.wantParams.kty) - } - if tt.wantParams.crv != crv { - t.Errorf("GenerateKey() crv = %s, want %s", crv, tt.wantParams.crv) - } - if tt.wantParams.size != size { - t.Errorf("GenerateKey() size = %d, want %d", size, tt.wantParams.size) - } - return tt.generateKey() - } - - got, err := k.CreateKey(tt.args.req) - if (err != nil) != tt.wantErr { - t.Errorf("SoftKMS.CreateKey() error = %v, wantErr %v", err, tt.wantErr) - return - } - if !reflect.DeepEqual(got, tt.want) { - t.Errorf("SoftKMS.CreateKey() = %v, want %v", got, tt.want) - } - }) - } -} - -func TestSoftKMS_GetPublicKey(t *testing.T) { - b, err := os.ReadFile("testdata/pub.pem") - if err != nil { - t.Fatal(err) - } - block, _ := pem.Decode(b) - pub, err := x509.ParsePKIXPublicKey(block.Bytes) - if err != nil { - t.Fatal(err) - } - - type args struct { - req *apiv1.GetPublicKeyRequest - } - tests := []struct { - name string - args args - want crypto.PublicKey - wantErr bool - }{ - {"key", args{&apiv1.GetPublicKeyRequest{Name: "testdata/pub.pem"}}, pub, false}, - {"cert", args{&apiv1.GetPublicKeyRequest{Name: "testdata/cert.crt"}}, pub, false}, - {"fail not exists", args{&apiv1.GetPublicKeyRequest{Name: "testdata/missing"}}, nil, true}, - {"fail type", args{&apiv1.GetPublicKeyRequest{Name: "testdata/cert.key"}}, nil, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - k := &SoftKMS{} - got, err := k.GetPublicKey(tt.args.req) - if (err != nil) != tt.wantErr { - t.Errorf("SoftKMS.GetPublicKey() error = %v, wantErr %v", err, tt.wantErr) - return - } - if !reflect.DeepEqual(got, tt.want) { - t.Errorf("SoftKMS.GetPublicKey() = %v, want %v", got, tt.want) - } - }) - } -} - -func Test_generateKey(t *testing.T) { - type args struct { - kty string - crv string - size int - } - tests := []struct { - name string - args args - wantType interface{} - wantType1 interface{} - wantErr bool - }{ - {"rsa2048", args{"RSA", "", 0}, &rsa.PublicKey{}, &rsa.PrivateKey{}, false}, - {"rsa2048", args{"RSA", "", 2048}, &rsa.PublicKey{}, &rsa.PrivateKey{}, false}, - {"p256", args{"EC", "P-256", 0}, &ecdsa.PublicKey{}, &ecdsa.PrivateKey{}, false}, - {"ed25519", args{"OKP", "Ed25519", 0}, ed25519.PublicKey{}, ed25519.PrivateKey{}, false}, - {"fail kty", args{"FOO", "", 0}, nil, nil, true}, - {"fail crv", args{"EC", "P-123", 0}, nil, nil, true}, - {"fail size", args{"RSA", "", 1}, nil, nil, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - got, got1, err := generateKey(tt.args.kty, tt.args.crv, tt.args.size) - if (err != nil) != tt.wantErr { - t.Errorf("generateKey() error = %v, wantErr %v", err, tt.wantErr) - return - } - if reflect.TypeOf(got) != reflect.TypeOf(tt.wantType) { - t.Errorf("generateKey() got = %T, want %T", got, tt.wantType) - } - if reflect.TypeOf(got1) != reflect.TypeOf(tt.wantType1) { - t.Errorf("generateKey() got1 = %T, want %T", got1, tt.wantType1) - } - }) - } -} - -func TestSoftKMS_CreateDecrypter(t *testing.T) { - privateKey, err := rsa.GenerateKey(rand.Reader, 2048) - if err != nil { - t.Fatal(err) - } - pemBlock, err := pemutil.Serialize(privateKey) - if err != nil { - t.Fatal(err) - } - pemBlockPassword, err := pemutil.Serialize(privateKey, pemutil.WithPassword([]byte("pass"))) - if err != nil { - t.Fatal(err) - } - ecdsaPK, err := ecdsa.GenerateKey(elliptic.P256(), rand.Reader) - if err != nil { - t.Fatal(err) - } - ecdsaPemBlock, err := pemutil.Serialize(ecdsaPK) - if err != nil { - t.Fatal(err) - } - b, err := os.ReadFile("testdata/rsa.priv.pem") - if err != nil { - t.Fatal(err) - } - block, _ := pem.Decode(b) - block.Bytes, err = x509.DecryptPEMBlock(block, []byte("pass")) //nolint - if err != nil { - t.Fatal(err) - } - keyFromFile, err := x509.ParsePKCS1PrivateKey(block.Bytes) - if err != nil { - t.Fatal(err) - } - type args struct { - req *apiv1.CreateDecrypterRequest - } - tests := []struct { - name string - args args - want crypto.Decrypter - wantErr bool - }{ - {"decrypter", args{&apiv1.CreateDecrypterRequest{Decrypter: privateKey}}, privateKey, false}, - {"file", args{&apiv1.CreateDecrypterRequest{DecryptionKey: "testdata/rsa.priv.pem", Password: []byte("pass")}}, keyFromFile, false}, - {"pem", args{&apiv1.CreateDecrypterRequest{DecryptionKeyPEM: pem.EncodeToMemory(pemBlock)}}, privateKey, false}, - {"pem password", args{&apiv1.CreateDecrypterRequest{DecryptionKeyPEM: pem.EncodeToMemory(pemBlockPassword), Password: []byte("pass")}}, privateKey, false}, - {"fail none", args{&apiv1.CreateDecrypterRequest{}}, nil, true}, - {"fail missing", args{&apiv1.CreateDecrypterRequest{DecryptionKey: "testdata/missing"}}, nil, true}, - {"fail bad pem", args{&apiv1.CreateDecrypterRequest{DecryptionKeyPEM: []byte("bad pem")}}, nil, true}, - {"fail bad password", args{&apiv1.CreateDecrypterRequest{DecryptionKeyPEM: pem.EncodeToMemory(pemBlockPassword), Password: []byte("bad-pass")}}, nil, true}, - {"fail not a decrypter (ecdsa key)", args{&apiv1.CreateDecrypterRequest{DecryptionKeyPEM: pem.EncodeToMemory(ecdsaPemBlock)}}, nil, true}, - {"fail not a decrypter from file", args{&apiv1.CreateDecrypterRequest{DecryptionKey: "testdata/rsa.pub.pem"}}, nil, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - k := &SoftKMS{} - got, err := k.CreateDecrypter(tt.args.req) - if (err != nil) != tt.wantErr { - t.Errorf("SoftKMS.CreateDecrypter(), error = %v, wantErr %v", err, tt.wantErr) - return - } - if !reflect.DeepEqual(got, tt.want) { - t.Errorf("SoftKMS.CreateDecrypter() = %v, want %v", got, tt.want) - } - }) - } -} diff --git a/kms/softkms/testdata/cert.crt b/kms/softkms/testdata/cert.crt deleted file mode 100644 index d6f02b21..00000000 --- a/kms/softkms/testdata/cert.crt +++ /dev/null @@ -1,11 +0,0 @@ ------BEGIN CERTIFICATE----- -MIIBpzCCAU2gAwIBAgIQWaY8KIDAfak8aYljelf8eTAKBggqhkjOPQQDAjAdMRsw -GQYDVQQDExJ0ZXN0LnNtYWxsc3RlcC5jb20wHhcNMjAwMTE2MDAwNDU4WhcNMjAw -MTE3MDAwNDU4WjAdMRswGQYDVQQDExJ0ZXN0LnNtYWxsc3RlcC5jb20wWTATBgcq -hkjOPQIBBggqhkjOPQMBBwNCAATlU8P9blFefSWuzYx2g215NJn6yHW95PXeFqQ9 -kX1jNo1VmC6Oord3We37iM8QJT4QP9ZDUaAVmJUZSjd+W8H/o28wbTAOBgNVHQ8B -Af8EBAMCBaAwHQYDVR0lBBYwFAYIKwYBBQUHAwEGCCsGAQUFBwMCMB0GA1UdDgQW -BBTn0wonKkm2lLRNYZrKhUukiynvqzAdBgNVHREEFjAUghJ0ZXN0LnNtYWxsc3Rl -cC5jb20wCgYIKoZIzj0EAwIDSAAwRQIhAJ5XqryBIY1X4fl/9l0isV69eQfA0Qo5 -1mjervUcEnOWAiBsmN4frz5YVw7i4UXChVBeZLZfJOKvn5eyh2gEzoq1+w== ------END CERTIFICATE----- diff --git a/kms/softkms/testdata/cert.key b/kms/softkms/testdata/cert.key deleted file mode 100644 index 187713cd..00000000 --- a/kms/softkms/testdata/cert.key +++ /dev/null @@ -1,5 +0,0 @@ ------BEGIN EC PRIVATE KEY----- -MHcCAQEEICB6lIrMa9fVQJtdAYS4qmdYQ1BHJsEQDx8zxL38gA8toAoGCCqGSM49 -AwEHoUQDQgAE5VPD/W5RXn0lrs2MdoNteTSZ+sh1veT13hakPZF9YzaNVZgujqK3 -d1nt+4jPECU+ED/WQ1GgFZiVGUo3flvB/w== ------END EC PRIVATE KEY----- diff --git a/kms/softkms/testdata/priv.pem b/kms/softkms/testdata/priv.pem deleted file mode 100644 index 81116ce7..00000000 --- a/kms/softkms/testdata/priv.pem +++ /dev/null @@ -1,8 +0,0 @@ ------BEGIN EC PRIVATE KEY----- -Proc-Type: 4,ENCRYPTED -DEK-Info: AES-256-CBC,1fcec5dfbf3327f61bfe5ab6ae8a0626 - -V39b/pNHMbP80TXSHLsUY6UOTCzf3KwIxvj1e7S9brNMJJc9b3UiloMBJIYBkl00 -NKI8JU4jSlcerR58DqsTHIELiX6a+RJLe3/iR2/5Gru+CmmWJ68jQu872WCgh6Ms -o8TzhyGx74ETmdKn5CdtylsnKMa9heW3tBLFAbNCgKc= ------END EC PRIVATE KEY----- diff --git a/kms/softkms/testdata/pub.pem b/kms/softkms/testdata/pub.pem deleted file mode 100644 index e31e583e..00000000 --- a/kms/softkms/testdata/pub.pem +++ /dev/null @@ -1,4 +0,0 @@ ------BEGIN PUBLIC KEY----- -MFkwEwYHKoZIzj0CAQYIKoZIzj0DAQcDQgAE5VPD/W5RXn0lrs2MdoNteTSZ+sh1 -veT13hakPZF9YzaNVZgujqK3d1nt+4jPECU+ED/WQ1GgFZiVGUo3flvB/w== ------END PUBLIC KEY----- diff --git a/kms/softkms/testdata/rsa.priv.pem b/kms/softkms/testdata/rsa.priv.pem deleted file mode 100644 index 497ec8d2..00000000 --- a/kms/softkms/testdata/rsa.priv.pem +++ /dev/null @@ -1,30 +0,0 @@ ------BEGIN RSA PRIVATE KEY----- -Proc-Type: 4,ENCRYPTED -DEK-Info: AES-256-CBC,dff7bfd0e0163a4cd7ade8f68b966699 - -jtmOhr2zo244Oq2fVsShZAUoQZ1gi6Iwc4i0sReU66XP9CFkdvJasAfkjQGrbCEy -m2+r7W6aH+L3j/4sXcJe8h4UVnnC4DHCozmtqqFCq7cFS4TiVpco26wEVH5WLm7Y -3Ew/pL0k24E+Ycf+yV5c1tQXRlmsKubjwzrZtGZP2yn3Dxsu97mzOXAfx7r+DIKI -5a4S3m1/yXw76tt6Iho9h4huA25UUDHKUQvOGd5gmOKqJRV9djoyu85ODbmz5nt0 -pB2EzdHOrefgd0rcQQPI1uFBWqASJxTn+uS7ZBP4rlCcs932lI1mPerMh1ujo51F -3aibrwhKE6kaJyOOnUbvyBnaiTb5i4WwTqx/jfsOsggXQb3UlxgDph48VXw8O2jF -CQmle+TR8yr1A14/Dno5Dd4cqPv6AmWWU2zolvLxKQixFcvjsyQYCDajWWRPkOgj -RTKXDqL1mpjrlDqcSXzemCWk6FzqdUQhimhFgARDRfRwwDeWQN5ua4a3gnem/cpA -ZS8J45H0ZC/CxGPfp+qx75n5a875+n4VMmCZerXPzEIj1CzS7D6BVAXTHJaNIB6S -0WNfQnftp09O2l6iXBE+MHt5bVxqt46+vgcceSu7Gsb3ZfD79vnQ7tR+wb+xmHKk -8rVcMrB+kDRXVguH/a3zUGYAEnb6hPkIJywJVD4G65oM+D9D67Mdka8wIMK48doV -my8a0MfT/9AidR6XJVxIkHlPsPzlxirm/NKF7oSlzurcvYcPAYnHYLW2uB8dyidq -1zB+3rxbSYCVqrhqzN4prydGvkIE3/+AJyIGn7uGSTSSyF6BC9APXQaHplRGKwLz -efOIMoEwXJ1DIcKmk9GB65xxrZxMu3Cclcbc4PgY4370G0PfCHuUQNQL2RUWCQn0 -aax+qDiFg1LsLRaI75OaLJ+uKs6rRfytQMmFGqK/b6iVbktiYWMtrDJDo4OUTtZ6 -LBBySH7sAFgI3IIxct2Fwg8X1J4kfHr9jWTLjMEIE2o8cyqvSQ8rdwA25MxRcn75 -DGqSlGE6Sx0XhWCVUiZidVRSYGKmOmH9yw8cjKm17qL23t8Gwns4Xunl7V6YlTCG -BPw5f1jWCQ94TwvUSuHMPYoXlYwRoe+jfDAzp2AQwXqvWX5Qno5PKz9gQ5iYacZ/ -k82fyPbk2XLDkPnaNJKnyiIc252O0WffUlX6Rlv3aF8ZgVvWfZbuHEK6g1W+IKSA -pXAQ+iZBl+fjs/wT0yZSNTB0P1InD9Ve536L94gxXoeMr6F0Eouk3J2R9qdFp0Av -31xylRKSmzUf87/sRxjy3FzSTjIal77y1euJoAEU/nShmNrAZ6B8wnlvHfVwbgmt -xWqxYIi/j/C8Led9uhEhX2WjPsO7ckGA41Tw6hZk/5hr4jmPoZQKHf9OauJFujMh -ybPRQ6SGZJaYQAgpEGHSHFm8lwf5/DcezdSMdzqAKBWJBv6MediMuS60wcJ0Tebk -rdLkNE4bsxfc889BkXBrSqfd+Auu5RcF/kF44gLL7oj4ojQyV44vLZbC4+liGThT -bhayYGV64hsY+zL03u5wVfF1Y+33/uc8o/0JjbfuW5AIdikVES/jnKKFXSTMNL69 ------END RSA PRIVATE KEY----- diff --git a/kms/softkms/testdata/rsa.pub.pem b/kms/softkms/testdata/rsa.pub.pem deleted file mode 100644 index 4bf5de9b..00000000 --- a/kms/softkms/testdata/rsa.pub.pem +++ /dev/null @@ -1,9 +0,0 @@ ------BEGIN PUBLIC KEY----- -MIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAn2Oh7/uWB5RH40la1a43 -IRaLZ8EnJVw5DCKE3BUre8xflVY2wTIS7XHcY0fEGprtq7hzFKors9AIGGn2yGrf -bZX2I+1g+RtQ6cLL6koeLuhRDqCuae0lZPulWc5ixBmM9mpl4ARRcpQFldxFRhis -xUaHMx8VqdZjFSDc5CJHYYK1n2G5DyuzJCk6yOfyMpwxizZJF4IUyqV7zKmZv1z9 -/Xd8X0ag7jRdaTBpupJ1WLaq7LlvyB4nr47JXXkLFbRIL1F/gTcPtg0tdEZiKnxs -VLKwOs3VjhEorUwhmVxr4NnNX/0tuOY1FJ0mx5jKLAevqLVwK2JIg/f3h7JcNxDy -tQIDAQAB ------END PUBLIC KEY----- diff --git a/kms/sshagentkms/sshagentkms.go b/kms/sshagentkms/sshagentkms.go deleted file mode 100644 index b3627a08..00000000 --- a/kms/sshagentkms/sshagentkms.go +++ /dev/null @@ -1,206 +0,0 @@ -package sshagentkms - -import ( - "context" - "crypto" - "crypto/ecdsa" - "crypto/ed25519" - "crypto/rsa" - "crypto/x509" - "io" - "net" - "os" - "strings" - - "golang.org/x/crypto/ssh" - "golang.org/x/crypto/ssh/agent" - - "github.com/pkg/errors" - "github.com/smallstep/certificates/kms/apiv1" - - "go.step.sm/crypto/pemutil" -) - -// SSHAgentKMS is a key manager that uses keys provided by ssh-agent -type SSHAgentKMS struct { - agentClient agent.Agent -} - -// New returns a new SSHAgentKMS. -func New(ctx context.Context, opts apiv1.Options) (*SSHAgentKMS, error) { - socket := os.Getenv("SSH_AUTH_SOCK") - conn, err := net.Dial("unix", socket) - if err != nil { - return nil, errors.Wrap(err, "failed to open SSH_AUTH_SOCK") - } - - agentClient := agent.NewClient(conn) - - return &SSHAgentKMS{ - agentClient: agentClient, - }, nil -} - -// NewFromAgent initializes an SSHAgentKMS from a given agent, this method is -// used for testing purposes. -func NewFromAgent(ctx context.Context, opts apiv1.Options, agentClient agent.Agent) (*SSHAgentKMS, error) { - return &SSHAgentKMS{ - agentClient: agentClient, - }, nil -} - -func init() { - apiv1.Register(apiv1.SSHAgentKMS, func(ctx context.Context, opts apiv1.Options) (apiv1.KeyManager, error) { - return New(ctx, opts) - }) -} - -// Close closes the agent. This is a noop for the SSHAgentKMS. -func (k *SSHAgentKMS) Close() error { - return nil -} - -// WrappedSSHSigner is a utility type to wrap a ssh.Signer as a crypto.Signer -type WrappedSSHSigner struct { - Sshsigner ssh.Signer -} - -// Public returns the agent public key. The type of this public key is -// *agent.Key. -func (s *WrappedSSHSigner) Public() crypto.PublicKey { - return s.Sshsigner.PublicKey() -} - -// Sign signs the given digest using the ssh agent and returns the signature. -func (s *WrappedSSHSigner) Sign(rand io.Reader, digest []byte, opts crypto.SignerOpts) (signature []byte, err error) { - sig, err := s.Sshsigner.Sign(rand, digest) - if err != nil { - return nil, err - } - return sig.Blob, nil -} - -// NewWrappedSignerFromSSHSigner returns a new crypto signer wrapping the given -// one. -func NewWrappedSignerFromSSHSigner(signer ssh.Signer) crypto.Signer { - return &WrappedSSHSigner{signer} -} - -func (k *SSHAgentKMS) findKey(signingKey string) (target int, err error) { - if strings.HasPrefix(signingKey, "sshagentkms:") { - var key = strings.TrimPrefix(signingKey, "sshagentkms:") - - l, err := k.agentClient.List() - if err != nil { - return -1, err - } - for i, s := range l { - if s.Comment == key { - return i, nil - } - } - } - - return -1, errors.Errorf("SSHAgentKMS couldn't find %s", signingKey) -} - -// CreateSigner returns a new signer configured with the given signing key. -func (k *SSHAgentKMS) CreateSigner(req *apiv1.CreateSignerRequest) (crypto.Signer, error) { - if req.Signer != nil { - return req.Signer, nil - } - if strings.HasPrefix(req.SigningKey, "sshagentkms:") { - target, err := k.findKey(req.SigningKey) - - if err != nil { - return nil, err - } - s, err := k.agentClient.Signers() - if err != nil { - return nil, err - } - return NewWrappedSignerFromSSHSigner(s[target]), nil - } - // OK: We don't actually care about non-ssh certificates, - // but we can't disable it in step-ca so this code is copy-pasted from - // softkms just to keep step-ca happy. - var opts []pemutil.Options - if req.Password != nil { - opts = append(opts, pemutil.WithPassword(req.Password)) - } - switch { - case len(req.SigningKeyPEM) != 0: - v, err := pemutil.ParseKey(req.SigningKeyPEM, opts...) - if err != nil { - return nil, err - } - sig, ok := v.(crypto.Signer) - if !ok { - return nil, errors.New("signingKeyPEM is not a crypto.Signer") - } - return sig, nil - case req.SigningKey != "": - v, err := pemutil.Read(req.SigningKey, opts...) - if err != nil { - return nil, err - } - sig, ok := v.(crypto.Signer) - if !ok { - return nil, errors.New("signingKey is not a crypto.Signer") - } - return sig, nil - default: - return nil, errors.New("failed to load softKMS: please define signingKeyPEM or signingKey") - } -} - -// CreateKey generates a new key and returns both public and private key. -func (k *SSHAgentKMS) CreateKey(req *apiv1.CreateKeyRequest) (*apiv1.CreateKeyResponse, error) { - return nil, errors.Errorf("SSHAgentKMS doesn't support generating keys") -} - -// GetPublicKey returns the public key from the file passed in the request name. -func (k *SSHAgentKMS) GetPublicKey(req *apiv1.GetPublicKeyRequest) (crypto.PublicKey, error) { - var v crypto.PublicKey - if strings.HasPrefix(req.Name, "sshagentkms:") { - target, err := k.findKey(req.Name) - - if err != nil { - return nil, err - } - - s, err := k.agentClient.Signers() - if err != nil { - return nil, err - } - - sshPub := s[target].PublicKey() - - sshPubBytes := sshPub.Marshal() - - parsed, err := ssh.ParsePublicKey(sshPubBytes) - if err != nil { - return nil, err - } - - parsedCryptoKey := parsed.(ssh.CryptoPublicKey) - - // Then, we can call CryptoPublicKey() to get the actual crypto.PublicKey - v = parsedCryptoKey.CryptoPublicKey() - } else { - var err error - v, err = pemutil.Read(req.Name) - if err != nil { - return nil, err - } - } - - switch vv := v.(type) { - case *x509.Certificate: - return vv.PublicKey, nil - case *rsa.PublicKey, *ecdsa.PublicKey, ed25519.PublicKey: - return vv, nil - default: - return nil, errors.Errorf("unsupported public key type %T", v) - } -} diff --git a/kms/sshagentkms/sshagentkms_test.go b/kms/sshagentkms/sshagentkms_test.go deleted file mode 100644 index 2c0a8aba..00000000 --- a/kms/sshagentkms/sshagentkms_test.go +++ /dev/null @@ -1,609 +0,0 @@ -package sshagentkms - -import ( - "bytes" - "context" - "crypto" - "crypto/ecdsa" - "crypto/elliptic" - "crypto/rand" - "crypto/x509" - "encoding/pem" - "net" - "os" - "os/exec" - "path/filepath" - "reflect" - "strconv" - "strings" - "testing" - - "github.com/smallstep/certificates/kms/apiv1" - - "golang.org/x/crypto/ssh" - "golang.org/x/crypto/ssh/agent" - - "go.step.sm/crypto/pemutil" -) - -// Some helpers with inspiration from crypto/ssh/agent/client_test.go - -// startOpenSSHAgent executes ssh-agent, and returns an Agent interface to it. -func startOpenSSHAgent(t *testing.T) (client agent.Agent, socket string, cleanup func()) { - /* Always test with OpenSSHAgent - if testing.Short() { - // ssh-agent is not always available, and the key - // types supported vary by platform. - t.Skip("skipping test due to -short") - } - */ - - bin, err := exec.LookPath("ssh-agent") - if err != nil { - t.Skip("could not find ssh-agent") - } - - cmd := exec.Command(bin, "-s") - cmd.Env = []string{} // Do not let the user's environment influence ssh-agent behavior. - cmd.Stderr = new(bytes.Buffer) - out, err := cmd.Output() - if err != nil { - t.Fatalf("%s failed: %v\n%s", strings.Join(cmd.Args, " "), err, cmd.Stderr) - } - - // Output looks like: - // - // SSH_AUTH_SOCK=/tmp/ssh-P65gpcqArqvH/agent.15541; export SSH_AUTH_SOCK; - // SSH_AGENT_PID=15542; export SSH_AGENT_PID; - // echo Agent pid 15542; - - fields := bytes.Split(out, []byte(";")) - line := bytes.SplitN(fields[0], []byte("="), 2) - line[0] = bytes.TrimLeft(line[0], "\n") - if string(line[0]) != "SSH_AUTH_SOCK" { - t.Fatalf("could not find key SSH_AUTH_SOCK in %q", fields[0]) - } - socket = string(line[1]) - - line = bytes.SplitN(fields[2], []byte("="), 2) - line[0] = bytes.TrimLeft(line[0], "\n") - if string(line[0]) != "SSH_AGENT_PID" { - t.Fatalf("could not find key SSH_AGENT_PID in %q", fields[2]) - } - pidStr := line[1] - pid, err := strconv.Atoi(string(pidStr)) - if err != nil { - t.Fatalf("Atoi(%q): %v", pidStr, err) - } - - conn, err := net.Dial("unix", string(socket)) - if err != nil { - t.Fatalf("net.Dial: %v", err) - } - - ac := agent.NewClient(conn) - return ac, socket, func() { - proc, _ := os.FindProcess(pid) - if proc != nil { - proc.Kill() - } - conn.Close() - os.RemoveAll(filepath.Dir(socket)) - } -} - -func startAgent(t *testing.T, sshagent agent.Agent) (client agent.Agent, cleanup func()) { - c1, c2, err := netPipe() - if err != nil { - t.Fatalf("netPipe: %v", err) - } - go agent.ServeAgent(sshagent, c2) - - return agent.NewClient(c1), func() { - c1.Close() - c2.Close() - } -} - -// startKeyringAgent uses Keyring to simulate a ssh-agent Server and returns a client. -func startKeyringAgent(t *testing.T) (client agent.Agent, cleanup func()) { - return startAgent(t, agent.NewKeyring()) -} - -type startTestAgentFunc func(t *testing.T, keysToAdd ...agent.AddedKey) (sshagent agent.Agent) - -func startTestOpenSSHAgent(t *testing.T, keysToAdd ...agent.AddedKey) (sshagent agent.Agent) { - sshagent, _, cleanup := startOpenSSHAgent(t) - for _, keyToAdd := range keysToAdd { - err := sshagent.Add(keyToAdd) - if err != nil { - t.Fatalf("sshagent.add: %v", err) - } - } - t.Cleanup(cleanup) - - //testAgentInterface(t, sshagent, key, cert, lifetimeSecs) - return sshagent -} - -func startTestKeyringAgent(t *testing.T, keysToAdd ...agent.AddedKey) (sshagent agent.Agent) { - sshagent, cleanup := startKeyringAgent(t) - for _, keyToAdd := range keysToAdd { - err := sshagent.Add(keyToAdd) - if err != nil { - t.Fatalf("sshagent.add: %v", err) - } - } - t.Cleanup(cleanup) - - //testAgentInterface(t, agent, key, cert, lifetimeSecs) - return sshagent -} - -// netPipe is analogous to net.Pipe, but it uses a real net.Conn, and -// therefore is buffered (net.Pipe deadlocks if both sides start with -// a write.) -func netPipe() (net.Conn, net.Conn, error) { - listener, err := netListener() - if err != nil { - return nil, nil, err - } - defer listener.Close() - c1, err := net.Dial("tcp", listener.Addr().String()) - if err != nil { - return nil, nil, err - } - - c2, err := listener.Accept() - if err != nil { - c1.Close() - return nil, nil, err - } - - return c1, c2, nil -} - -// netListener creates a localhost network listener. -func netListener() (net.Listener, error) { - listener, err := net.Listen("tcp", "127.0.0.1:0") - if err != nil { - listener, err = net.Listen("tcp", "[::1]:0") - if err != nil { - return nil, err - } - } - return listener, nil -} - -func TestNew(t *testing.T) { - comment := "Key from OpenSSHAgent" - // Ensure we don't "inherit" any SSH_AUTH_SOCK - os.Unsetenv("SSH_AUTH_SOCK") - - sshagent, socket, cleanup := startOpenSSHAgent(t) - - os.Setenv("SSH_AUTH_SOCK", socket) - t.Cleanup(func() { - os.Unsetenv("SSH_AUTH_SOCK") - cleanup() - }) - - // Test that we can't find any signers in the agent before we have loaded them - t.Run("No keys with OpenSSHAgent", func(t *testing.T) { - kms, err := New(context.Background(), apiv1.Options{}) - if kms == nil || err != nil { - t.Errorf("New() = %v, %v", kms, err) - } - signer, err := kms.CreateSigner(&apiv1.CreateSignerRequest{SigningKey: "sshagentkms:" + comment}) - if err == nil || signer != nil { - t.Errorf("SSHAgentKMS.CreateSigner() error = \"%v\", signer = \"%v\"", err, signer) - } - }) - - // Load ssh test fixtures - b, err := os.ReadFile("testdata/ssh") - if err != nil { - t.Fatal(err) - } - privateKey, err := ssh.ParseRawPrivateKey(b) - if err != nil { - t.Fatal(err) - } - - // And add that key to the agent - err = sshagent.Add(agent.AddedKey{PrivateKey: privateKey, Comment: comment}) - if err != nil { - t.Fatalf("sshagent.add: %v", err) - } - - // And test that we can find it when it's loaded - t.Run("Keys with OpenSSHAgent", func(t *testing.T) { - kms, err := New(context.Background(), apiv1.Options{}) - if kms == nil || err != nil { - t.Errorf("New() = %v, %v", kms, err) - } - signer, err := kms.CreateSigner(&apiv1.CreateSignerRequest{SigningKey: "sshagentkms:" + comment}) - if err != nil || signer == nil { - t.Errorf("SSHAgentKMS.CreateSigner() error = \"%v\", signer = \"%v\"", err, signer) - } - }) -} - -func TestNewFromAgent(t *testing.T) { - type args struct { - ctx context.Context - opts apiv1.Options - } - tests := []struct { - name string - args args - sshagentstarter startTestAgentFunc - wantErr bool - }{ - {"ok OpenSSHAgent", args{context.Background(), apiv1.Options{}}, startTestOpenSSHAgent, false}, - {"ok KeyringAgent", args{context.Background(), apiv1.Options{}}, startTestKeyringAgent, false}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - got, err := NewFromAgent(tt.args.ctx, tt.args.opts, tt.sshagentstarter(t)) - if (err != nil) != tt.wantErr { - t.Errorf("NewFromAgent() error = %v, wantErr %v", err, tt.wantErr) - return - } - if got == nil { - t.Errorf("NewFromAgent() = %v", got) - } - }) - } -} - -func TestSSHAgentKMS_Close(t *testing.T) { - tests := []struct { - name string - wantErr bool - }{ - {"ok", false}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - k := &SSHAgentKMS{} - if err := k.Close(); (err != nil) != tt.wantErr { - t.Errorf("SSHAgentKMS.Close() error = %v, wantErr %v", err, tt.wantErr) - } - }) - } -} - -func TestSSHAgentKMS_CreateSigner(t *testing.T) { - pk, err := ecdsa.GenerateKey(elliptic.P256(), rand.Reader) - if err != nil { - t.Fatal(err) - } - pemBlock, err := pemutil.Serialize(pk) - if err != nil { - t.Fatal(err) - } - pemBlockPassword, err := pemutil.Serialize(pk, pemutil.WithPassword([]byte("pass"))) - if err != nil { - t.Fatal(err) - } - - // Read and decode file using standard packages - b, err := os.ReadFile("testdata/priv.pem") - if err != nil { - t.Fatal(err) - } - block, _ := pem.Decode(b) - block.Bytes, err = x509.DecryptPEMBlock(block, []byte("pass")) //nolint - if err != nil { - t.Fatal(err) - } - pk2, err := x509.ParseECPrivateKey(block.Bytes) - if err != nil { - t.Fatal(err) - } - - // Create a public PEM - b, err = x509.MarshalPKIXPublicKey(pk.Public()) - if err != nil { - t.Fatal(err) - } - pub := pem.EncodeToMemory(&pem.Block{ - Type: "PUBLIC KEY", - Bytes: b, - }) - - // Load ssh test fixtures - sshPubKeyStr, err := os.ReadFile("testdata/ssh.pub") - if err != nil { - t.Fatal(err) - } - _, comment, _, _, err := ssh.ParseAuthorizedKey(sshPubKeyStr) - if err != nil { - t.Fatal(err) - } - b, err = os.ReadFile("testdata/ssh") - if err != nil { - t.Fatal(err) - } - privateKey, err := ssh.ParseRawPrivateKey(b) - if err != nil { - t.Fatal(err) - } - sshPrivateKey, err := ssh.NewSignerFromKey(privateKey) - if err != nil { - t.Fatal(err) - } - wrappedSSHPrivateKey := NewWrappedSignerFromSSHSigner(sshPrivateKey) - - type args struct { - req *apiv1.CreateSignerRequest - } - tests := []struct { - name string - args args - want crypto.Signer - wantErr bool - }{ - {"signer", args{&apiv1.CreateSignerRequest{Signer: pk}}, pk, false}, - {"pem", args{&apiv1.CreateSignerRequest{SigningKeyPEM: pem.EncodeToMemory(pemBlock)}}, pk, false}, - {"pem password", args{&apiv1.CreateSignerRequest{SigningKeyPEM: pem.EncodeToMemory(pemBlockPassword), Password: []byte("pass")}}, pk, false}, - {"file", args{&apiv1.CreateSignerRequest{SigningKey: "testdata/priv.pem", Password: []byte("pass")}}, pk2, false}, - {"sshagent", args{&apiv1.CreateSignerRequest{SigningKey: "sshagentkms:" + comment}}, wrappedSSHPrivateKey, false}, - {"sshagent Nonexistant", args{&apiv1.CreateSignerRequest{SigningKey: "sshagentkms:Nonexistant"}}, nil, true}, - {"fail", args{&apiv1.CreateSignerRequest{}}, nil, true}, - {"fail bad pem", args{&apiv1.CreateSignerRequest{SigningKeyPEM: []byte("bad pem")}}, nil, true}, - {"fail bad password", args{&apiv1.CreateSignerRequest{SigningKey: "testdata/priv.pem", Password: []byte("bad-pass")}}, nil, true}, - {"fail not a signer", args{&apiv1.CreateSignerRequest{SigningKeyPEM: pub}}, nil, true}, - {"fail not a signer from file", args{&apiv1.CreateSignerRequest{SigningKey: "testdata/pub.pem"}}, nil, true}, - {"fail missing", args{&apiv1.CreateSignerRequest{SigningKey: "testdata/missing"}}, nil, true}, - } - starters := []struct { - name string - starter startTestAgentFunc - }{ - {"startTestOpenSSHAgent", startTestOpenSSHAgent}, - {"startTestKeyringAgent", startTestKeyringAgent}, - } - for _, starter := range starters { - k, err := NewFromAgent(context.Background(), apiv1.Options{}, starter.starter(t, agent.AddedKey{PrivateKey: privateKey, Comment: comment})) - if err != nil { - t.Fatal(err) - } - for _, tt := range tests { - t.Run(starter.name+"/"+tt.name, func(t *testing.T) { - got, err := k.CreateSigner(tt.args.req) - if (err != nil) != tt.wantErr { - t.Errorf("SSHAgentKMS.CreateSigner() error = %v, wantErr %v", err, tt.wantErr) - return - } - // nolint:gocritic - switch s := got.(type) { - case *WrappedSSHSigner: - gotPkS := s.Sshsigner.PublicKey().(*agent.Key).String() + "\n" - wantPkS := string(sshPubKeyStr) - if !reflect.DeepEqual(gotPkS, wantPkS) { - t.Errorf("SSHAgentKMS.CreateSigner() = %T, want %T", gotPkS, wantPkS) - t.Errorf("SSHAgentKMS.CreateSigner() = %v, want %v", gotPkS, wantPkS) - } - default: - if !reflect.DeepEqual(got, tt.want) { - t.Errorf("SSHAgentKMS.CreateSigner() = %T, want %T", got, tt.want) - t.Errorf("SSHAgentKMS.CreateSigner() = %v, want %v", got, tt.want) - } - } - }) - } - } -} - -/* -func restoreGenerateKey() func() { - oldGenerateKey := generateKey - return func() { - generateKey = oldGenerateKey - } -} -*/ - -/* -func TestSSHAgentKMS_CreateKey(t *testing.T) { - fn := restoreGenerateKey() - defer fn() - - p256, err := ecdsa.GenerateKey(elliptic.P256(), rand.Reader) - if err != nil { - t.Fatal(err) - } - rsa2048, err := rsa.GenerateKey(rand.Reader, 2048) - if err != nil { - t.Fatal(err) - } - edpub, edpriv, err := ed25519.GenerateKey(rand.Reader) - if err != nil { - t.Fatal(err) - } - - type args struct { - req *apiv1.CreateKeyRequest - } - type params struct { - kty string - crv string - size int - } - tests := []struct { - name string - args args - generateKey func() (interface{}, interface{}, error) - want *apiv1.CreateKeyResponse - wantParams params - wantErr bool - }{ - {"p256", args{&apiv1.CreateKeyRequest{Name: "p256", SignatureAlgorithm: apiv1.ECDSAWithSHA256}}, func() (interface{}, interface{}, error) { - return p256.Public(), p256, nil - }, &apiv1.CreateKeyResponse{Name: "p256", PublicKey: p256.Public(), PrivateKey: p256, CreateSignerRequest: apiv1.CreateSignerRequest{Signer: p256}}, params{"EC", "P-256", 0}, false}, - {"rsa", args{&apiv1.CreateKeyRequest{Name: "rsa3072", SignatureAlgorithm: apiv1.SHA256WithRSA}}, func() (interface{}, interface{}, error) { - return rsa2048.Public(), rsa2048, nil - }, &apiv1.CreateKeyResponse{Name: "rsa3072", PublicKey: rsa2048.Public(), PrivateKey: rsa2048, CreateSignerRequest: apiv1.CreateSignerRequest{Signer: rsa2048}}, params{"RSA", "", 0}, false}, - {"rsa2048", args{&apiv1.CreateKeyRequest{Name: "rsa2048", SignatureAlgorithm: apiv1.SHA256WithRSA, Bits: 2048}}, func() (interface{}, interface{}, error) { - return rsa2048.Public(), rsa2048, nil - }, &apiv1.CreateKeyResponse{Name: "rsa2048", PublicKey: rsa2048.Public(), PrivateKey: rsa2048, CreateSignerRequest: apiv1.CreateSignerRequest{Signer: rsa2048}}, params{"RSA", "", 2048}, false}, - {"rsaPSS2048", args{&apiv1.CreateKeyRequest{Name: "rsa2048", SignatureAlgorithm: apiv1.SHA256WithRSAPSS, Bits: 2048}}, func() (interface{}, interface{}, error) { - return rsa2048.Public(), rsa2048, nil - }, &apiv1.CreateKeyResponse{Name: "rsa2048", PublicKey: rsa2048.Public(), PrivateKey: rsa2048, CreateSignerRequest: apiv1.CreateSignerRequest{Signer: rsa2048}}, params{"RSA", "", 2048}, false}, - {"ed25519", args{&apiv1.CreateKeyRequest{Name: "ed25519", SignatureAlgorithm: apiv1.PureEd25519}}, func() (interface{}, interface{}, error) { - return edpub, edpriv, nil - }, &apiv1.CreateKeyResponse{Name: "ed25519", PublicKey: edpub, PrivateKey: edpriv, CreateSignerRequest: apiv1.CreateSignerRequest{Signer: edpriv}}, params{"OKP", "Ed25519", 0}, false}, - {"default", args{&apiv1.CreateKeyRequest{Name: "default"}}, func() (interface{}, interface{}, error) { - return p256.Public(), p256, nil - }, &apiv1.CreateKeyResponse{Name: "default", PublicKey: p256.Public(), PrivateKey: p256, CreateSignerRequest: apiv1.CreateSignerRequest{Signer: p256}}, params{"EC", "P-256", 0}, false}, - {"fail algorithm", args{&apiv1.CreateKeyRequest{Name: "fail", SignatureAlgorithm: apiv1.SignatureAlgorithm(100)}}, func() (interface{}, interface{}, error) { - return p256.Public(), p256, nil - }, nil, params{}, true}, - {"fail generate key", args{&apiv1.CreateKeyRequest{Name: "fail", SignatureAlgorithm: apiv1.ECDSAWithSHA256}}, func() (interface{}, interface{}, error) { - return nil, nil, fmt.Errorf("an error") - }, nil, params{"EC", "P-256", 0}, true}, - {"fail no signer", args{&apiv1.CreateKeyRequest{Name: "fail", SignatureAlgorithm: apiv1.ECDSAWithSHA256}}, func() (interface{}, interface{}, error) { - return 1, 2, nil - }, nil, params{"EC", "P-256", 0}, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - k := &SSHAgentKMS{} - generateKey = func(kty, crv string, size int) (interface{}, interface{}, error) { - if tt.wantParams.kty != kty { - t.Errorf("GenerateKey() kty = %s, want %s", kty, tt.wantParams.kty) - } - if tt.wantParams.crv != crv { - t.Errorf("GenerateKey() crv = %s, want %s", crv, tt.wantParams.crv) - } - if tt.wantParams.size != size { - t.Errorf("GenerateKey() size = %d, want %d", size, tt.wantParams.size) - } - return tt.generateKey() - } - - got, err := k.CreateKey(tt.args.req) - if (err != nil) != tt.wantErr { - t.Errorf("SSHAgentKMS.CreateKey() error = %v, wantErr %v", err, tt.wantErr) - return - } - if !reflect.DeepEqual(got, tt.want) { - t.Errorf("SSHAgentKMS.CreateKey() = %v, want %v", got, tt.want) - } - }) - } -} -*/ - -func TestSSHAgentKMS_GetPublicKey(t *testing.T) { - b, err := os.ReadFile("testdata/pub.pem") - if err != nil { - t.Fatal(err) - } - block, _ := pem.Decode(b) - pub, err := x509.ParsePKIXPublicKey(block.Bytes) - if err != nil { - t.Fatal(err) - } - - // Load ssh test fixtures - b, err = os.ReadFile("testdata/ssh.pub") - if err != nil { - t.Fatal(err) - } - sshPubKey, comment, _, _, err := ssh.ParseAuthorizedKey(b) - if err != nil { - t.Fatal(err) - } - b, err = os.ReadFile("testdata/ssh") - if err != nil { - t.Fatal(err) - } - // crypto.PrivateKey - sshPrivateKey, err := ssh.ParseRawPrivateKey(b) - if err != nil { - t.Fatal(err) - } - - type args struct { - req *apiv1.GetPublicKeyRequest - } - tests := []struct { - name string - args args - want crypto.PublicKey - wantErr bool - }{ - {"key", args{&apiv1.GetPublicKeyRequest{Name: "testdata/pub.pem"}}, pub, false}, - {"cert", args{&apiv1.GetPublicKeyRequest{Name: "testdata/cert.crt"}}, pub, false}, - {"sshagent", args{&apiv1.GetPublicKeyRequest{Name: "sshagentkms:" + comment}}, sshPubKey, false}, - {"sshagent Nonexistant", args{&apiv1.GetPublicKeyRequest{Name: "sshagentkms:Nonexistant"}}, nil, true}, - {"fail not exists", args{&apiv1.GetPublicKeyRequest{Name: "testdata/missing"}}, nil, true}, - {"fail type", args{&apiv1.GetPublicKeyRequest{Name: "testdata/cert.key"}}, nil, true}, - } - starters := []struct { - name string - starter startTestAgentFunc - }{ - {"startTestOpenSSHAgent", startTestOpenSSHAgent}, - {"startTestKeyringAgent", startTestKeyringAgent}, - } - for _, starter := range starters { - k, err := NewFromAgent(context.Background(), apiv1.Options{}, starter.starter(t, agent.AddedKey{PrivateKey: sshPrivateKey, Comment: comment})) - if err != nil { - t.Fatal(err) - } - for _, tt := range tests { - t.Run(starter.name+"/"+tt.name, func(t *testing.T) { - got, err := k.GetPublicKey(tt.args.req) - if (err != nil) != tt.wantErr { - t.Errorf("SSHAgentKMS.GetPublicKey() error = %v, wantErr %v", err, tt.wantErr) - return - } - // nolint:gocritic - switch tt.want.(type) { - case ssh.PublicKey: - // If we want a ssh.PublicKey, protote got to a - got, err = ssh.NewPublicKey(got) - if err != nil { - t.Fatal(err) - } - } - if !reflect.DeepEqual(got, tt.want) { - t.Errorf("SSHAgentKMS.GetPublicKey() = %T, want %T", got, tt.want) - t.Errorf("SSHAgentKMS.GetPublicKey() = %v, want %v", got, tt.want) - } - }) - } - } -} - -func TestSSHAgentKMS_CreateKey(t *testing.T) { - starters := []struct { - name string - starter startTestAgentFunc - }{ - {"startTestOpenSSHAgent", startTestOpenSSHAgent}, - {"startTestKeyringAgent", startTestKeyringAgent}, - } - for _, starter := range starters { - k, err := NewFromAgent(context.Background(), apiv1.Options{}, starter.starter(t)) - if err != nil { - t.Fatal(err) - } - t.Run(starter.name+"/CreateKey", func(t *testing.T) { - got, err := k.CreateKey(&apiv1.CreateKeyRequest{ - Name: "sshagentkms:0", - SignatureAlgorithm: apiv1.ECDSAWithSHA256, - }) - if got != nil { - t.Error("SSHAgentKMS.CreateKey() shoudn't return a value") - } - if err == nil { - t.Error("SSHAgentKMS.CreateKey() didn't return a value") - } - }) - } -} diff --git a/kms/sshagentkms/testdata/cert.crt b/kms/sshagentkms/testdata/cert.crt deleted file mode 100644 index d6f02b21..00000000 --- a/kms/sshagentkms/testdata/cert.crt +++ /dev/null @@ -1,11 +0,0 @@ ------BEGIN CERTIFICATE----- -MIIBpzCCAU2gAwIBAgIQWaY8KIDAfak8aYljelf8eTAKBggqhkjOPQQDAjAdMRsw -GQYDVQQDExJ0ZXN0LnNtYWxsc3RlcC5jb20wHhcNMjAwMTE2MDAwNDU4WhcNMjAw -MTE3MDAwNDU4WjAdMRswGQYDVQQDExJ0ZXN0LnNtYWxsc3RlcC5jb20wWTATBgcq -hkjOPQIBBggqhkjOPQMBBwNCAATlU8P9blFefSWuzYx2g215NJn6yHW95PXeFqQ9 -kX1jNo1VmC6Oord3We37iM8QJT4QP9ZDUaAVmJUZSjd+W8H/o28wbTAOBgNVHQ8B -Af8EBAMCBaAwHQYDVR0lBBYwFAYIKwYBBQUHAwEGCCsGAQUFBwMCMB0GA1UdDgQW -BBTn0wonKkm2lLRNYZrKhUukiynvqzAdBgNVHREEFjAUghJ0ZXN0LnNtYWxsc3Rl -cC5jb20wCgYIKoZIzj0EAwIDSAAwRQIhAJ5XqryBIY1X4fl/9l0isV69eQfA0Qo5 -1mjervUcEnOWAiBsmN4frz5YVw7i4UXChVBeZLZfJOKvn5eyh2gEzoq1+w== ------END CERTIFICATE----- diff --git a/kms/sshagentkms/testdata/cert.key b/kms/sshagentkms/testdata/cert.key deleted file mode 100644 index 187713cd..00000000 --- a/kms/sshagentkms/testdata/cert.key +++ /dev/null @@ -1,5 +0,0 @@ ------BEGIN EC PRIVATE KEY----- -MHcCAQEEICB6lIrMa9fVQJtdAYS4qmdYQ1BHJsEQDx8zxL38gA8toAoGCCqGSM49 -AwEHoUQDQgAE5VPD/W5RXn0lrs2MdoNteTSZ+sh1veT13hakPZF9YzaNVZgujqK3 -d1nt+4jPECU+ED/WQ1GgFZiVGUo3flvB/w== ------END EC PRIVATE KEY----- diff --git a/kms/sshagentkms/testdata/priv.pem b/kms/sshagentkms/testdata/priv.pem deleted file mode 100644 index 81116ce7..00000000 --- a/kms/sshagentkms/testdata/priv.pem +++ /dev/null @@ -1,8 +0,0 @@ ------BEGIN EC PRIVATE KEY----- -Proc-Type: 4,ENCRYPTED -DEK-Info: AES-256-CBC,1fcec5dfbf3327f61bfe5ab6ae8a0626 - -V39b/pNHMbP80TXSHLsUY6UOTCzf3KwIxvj1e7S9brNMJJc9b3UiloMBJIYBkl00 -NKI8JU4jSlcerR58DqsTHIELiX6a+RJLe3/iR2/5Gru+CmmWJ68jQu872WCgh6Ms -o8TzhyGx74ETmdKn5CdtylsnKMa9heW3tBLFAbNCgKc= ------END EC PRIVATE KEY----- diff --git a/kms/sshagentkms/testdata/pub.pem b/kms/sshagentkms/testdata/pub.pem deleted file mode 100644 index e31e583e..00000000 --- a/kms/sshagentkms/testdata/pub.pem +++ /dev/null @@ -1,4 +0,0 @@ ------BEGIN PUBLIC KEY----- -MFkwEwYHKoZIzj0CAQYIKoZIzj0DAQcDQgAE5VPD/W5RXn0lrs2MdoNteTSZ+sh1 -veT13hakPZF9YzaNVZgujqK3d1nt+4jPECU+ED/WQ1GgFZiVGUo3flvB/w== ------END PUBLIC KEY----- diff --git a/kms/sshagentkms/testdata/ssh b/kms/sshagentkms/testdata/ssh deleted file mode 100644 index 3a2ac73d..00000000 --- a/kms/sshagentkms/testdata/ssh +++ /dev/null @@ -1,49 +0,0 @@ ------BEGIN OPENSSH PRIVATE KEY----- -b3BlbnNzaC1rZXktdjEAAAAABG5vbmUAAAAEbm9uZQAAAAAAAAABAAACFwAAAAdzc2gtcn -NhAAAAAwEAAQAAAgEAth/d7zRDbv567o46KT6YYqC/EVdDpZ8m0rzIdroJL+RHVDXNQ1pU -3lrC9IWfkyjX+YwO9jHGbraJ+CgonAkl36mtLzNC4645QGS2/WdFqRR6mQCz7v4G6nOaFN -SCeErMhg0fn4f7jdqXpd0hYozIpktRVNYcpi2RMmr8e/Kadr5EVQfbYZgdKIl1O6Ws9O3Q -1BhLGi9GipEstUTvjqxZzF7oUgWKH54j5eHNXdbFqKqnK8NNQmypNLYGDsTBQHG9zRs+o0 -7C2foO9ddIO2OCarcBWZfGlY05k/ZhEmrEOONh2rSLhJwqw+EJgQeU0Poe/IqjFy7jnTRk -i+tee2elBYVvHYPSofZaBmX7i21s8eBRl/ZiFx3ip6E3M54mXvKZ7SuA2qq/YW0IeKyJ5D -SuL0+sRAyiSQ2Icsyb3YKv6LXojuJTmJ9Hg9v4+aOPxOQhvNfh3b7sIh/cmz1dq/babLyO -ORrbHKDxIJME7VPMspmddV9wJgB4Gu1eWOiR/Cuv6jqYWTfiWJDIoqZRD5nF1tFqKtZ5iA -qkflv4Kbo10tv6nTlXR6TWuPu2Z/pZpx+NN+7QxVUSlRgxb7RTVcHRvpgd0TNEXGduR8ar -WVDlNewOmf5KFroW1IX/yR1OvE5RsDixxcX7Ne+uSlq9hooy9V/Ip0ffcF/Kg0NJoPwrnI -MAAAdQrAxluqwMZboAAAAHc3NoLXJzYQAAAgEAth/d7zRDbv567o46KT6YYqC/EVdDpZ8m -0rzIdroJL+RHVDXNQ1pU3lrC9IWfkyjX+YwO9jHGbraJ+CgonAkl36mtLzNC4645QGS2/W -dFqRR6mQCz7v4G6nOaFNSCeErMhg0fn4f7jdqXpd0hYozIpktRVNYcpi2RMmr8e/Kadr5E -VQfbYZgdKIl1O6Ws9O3Q1BhLGi9GipEstUTvjqxZzF7oUgWKH54j5eHNXdbFqKqnK8NNQm -ypNLYGDsTBQHG9zRs+o07C2foO9ddIO2OCarcBWZfGlY05k/ZhEmrEOONh2rSLhJwqw+EJ -gQeU0Poe/IqjFy7jnTRki+tee2elBYVvHYPSofZaBmX7i21s8eBRl/ZiFx3ip6E3M54mXv -KZ7SuA2qq/YW0IeKyJ5DSuL0+sRAyiSQ2Icsyb3YKv6LXojuJTmJ9Hg9v4+aOPxOQhvNfh -3b7sIh/cmz1dq/babLyOORrbHKDxIJME7VPMspmddV9wJgB4Gu1eWOiR/Cuv6jqYWTfiWJ -DIoqZRD5nF1tFqKtZ5iAqkflv4Kbo10tv6nTlXR6TWuPu2Z/pZpx+NN+7QxVUSlRgxb7RT -VcHRvpgd0TNEXGduR8arWVDlNewOmf5KFroW1IX/yR1OvE5RsDixxcX7Ne+uSlq9hooy9V -/Ip0ffcF/Kg0NJoPwrnIMAAAADAQABAAACADQ4KONYQemGT+ssnqKKzxigbIhlVAEeA/yy -omvgZZf0xTrw/jzMnr7umS2RTrLcKCjmLrgKh5HhBug/Y31x5gkeVojNEuXDY6kB97HqtX -+IXqqWGAFzlroMkWZdlFc3YzMgeiu8yrTes1Kcd+EQ6ss7l0NS7P383L/vCxvi8MURQvh6 -ez2dZubjmtiSZWgI9DKMEKSeX4SFoaML9AAdjNXbdJNoATWVm0djmgXI+f2liK80nWdpTo -7NjikX4y0+L6SqpigfAiGL4FQ++PgGTTOZ62or6YWh65twLl8ge8iv8bPKxqIsQNrPIHF9 -of7VaKMSgTa5fAvsJNQ1lW6exiK1szJ+g+zrkHuOjDaEWyIZi24/xy6iDaT1sdcjTGPJAo -WqgC9hlZQKjOOZJgwqu/kxgcsOGaGb2MD/E4xJVMvPsWYLQ5WGdiakQkVhclpcr3e0d8nw -xvqCqLsasCSECKJK+k3ReqtOe6GlTSzIpFiOgFAuYp+ejRkX6bJ2DRaYkjoWWza2VCpIJC -uyK7B3r1cV+g5KzvT6B+7TxVqYERisjWNvdppF87Vtx7C0p8mDzpJYpPY+yao3vEcq104+ -yXuaPGEDTkTWOUB2uUS+AD9CBjkrGYFab1DBJob+L/7jNgVgWmMw1Yj9SDwXO6YBfbkhCf -Irfmf9Ne5i1+2SpFWBAAABAQCud97O9xI2bMGVGfbDFiaPTYGaGZ0qurLtHPpCX/YFkdBh -Z3LG7psJ/4JhkmMI3RFGhMxpUR9K22T3P/UmUt01PrDwDUpcw1JRPVIGs9AV3+GsAyyE6X -MzYo+8LNcxaPjh6ECXAQLcd9g0NOCbiqrKURBEuIBkxTy8jsmmeUlDsLcs8QKCsObJ2ozO -ACuFG5Z/SUeB7nhHnRUnozE8KsEWAgpys37AnJc1cQR6ALloh23L46rsWbSN5UGRgZdaUo -tklsDRun3qtYkDC8dDbW2Iy5A7GUXBRIA3mDYf4GDEUQvuu5Q/A2Dsr0hVi2wNVWd5O5M0 -NVhuCHJU355wbbUUAAABAQDuet4GZQImmqfj2xAMoHUfSK0WagtzynP2fOSIRtOKQ9UXJN -J1CrSeu93dNACYjXt10X5ZCdZ9x/75ltyZHSUBbT1eQzPD4Jq23EcJ9ECCc4tJMpdNpJyv -8ixfeTCX0m6XP7nDDLgkuYuNTj/NTqIWotHt8/R8BA9FfTchZE+ekqj3TTIac3buU294mO -/0KKGHtt+GPHSD+ES+W28KETiFcz5nSD7oUQPXEbvsJg5bOWt9kY6JBGiizJSsEuLIjcva -H3UQMx6U805NjoGwIiKJyKgcmDMWVbeH87XxV6sllE8UaLUxbcOBdhmF/uJlazQsbqmF7B -CJB/X7SXredw9BAAABAQDDgRzgXsvBH72PMetQpWGswXp6UVsdHUUEyDiJXc5xjiVOxAIw -+pwaBRQ/6WMMJvhpZ/IFN+pAYEW5e0q2eGMpc1or4kf5eTukwJSF6VZf1Hhti6TfiStPCf -KSz07jUFROahMC88BOSwHuCc66emWlsZDrXS+pht1O7yU96epTM/hT/e8Bfi+ZFCJnQoQ5 -dZuONhOYUT32rFKGBwPhsi6pjMB54vqrW1xFJbwj4i4dHFzA7UUa79j7ToAs2g2q8odTCR -CLUxGJ+YOkti67taOuRbzlL9wlxLGT+G2Dai9Ymbt18rmXR+2vazE0xFigYHPZb2QXeLAS -u104cC7ouX7DAAAAFnNzaC50ZXN0LnNtYWxsc3RlcC5jb20BAgME ------END OPENSSH PRIVATE KEY----- diff --git a/kms/sshagentkms/testdata/ssh.pub b/kms/sshagentkms/testdata/ssh.pub deleted file mode 100644 index 35673a99..00000000 --- a/kms/sshagentkms/testdata/ssh.pub +++ /dev/null @@ -1 +0,0 @@ -ssh-rsa AAAAB3NzaC1yc2EAAAADAQABAAACAQC2H93vNENu/nrujjopPphioL8RV0OlnybSvMh2ugkv5EdUNc1DWlTeWsL0hZ+TKNf5jA72McZuton4KCicCSXfqa0vM0LjrjlAZLb9Z0WpFHqZALPu/gbqc5oU1IJ4SsyGDR+fh/uN2pel3SFijMimS1FU1hymLZEyavx78pp2vkRVB9thmB0oiXU7paz07dDUGEsaL0aKkSy1RO+OrFnMXuhSBYofniPl4c1d1sWoqqcrw01CbKk0tgYOxMFAcb3NGz6jTsLZ+g7110g7Y4JqtwFZl8aVjTmT9mESasQ442HatIuEnCrD4QmBB5TQ+h78iqMXLuOdNGSL6157Z6UFhW8dg9Kh9loGZfuLbWzx4FGX9mIXHeKnoTczniZe8pntK4Daqr9hbQh4rInkNK4vT6xEDKJJDYhyzJvdgq/oteiO4lOYn0eD2/j5o4/E5CG81+HdvuwiH9ybPV2r9tpsvI45GtscoPEgkwTtU8yymZ11X3AmAHga7V5Y6JH8K6/qOphZN+JYkMiiplEPmcXW0Woq1nmICqR+W/gpujXS2/qdOVdHpNa4+7Zn+lmnH4037tDFVRKVGDFvtFNVwdG+mB3RM0RcZ25HxqtZUOU17A6Z/koWuhbUhf/JHU68TlGwOLHFxfs1765KWr2GijL1X8inR99wX8qDQ0mg/Cucgw== ssh.test.smallstep.com diff --git a/kms/uri/testdata/pin.txt b/kms/uri/testdata/pin.txt deleted file mode 100644 index 2fca70ea..00000000 --- a/kms/uri/testdata/pin.txt +++ /dev/null @@ -1 +0,0 @@ -trim-this-pin diff --git a/kms/uri/uri.go b/kms/uri/uri.go deleted file mode 100644 index a812f80b..00000000 --- a/kms/uri/uri.go +++ /dev/null @@ -1,148 +0,0 @@ -package uri - -import ( - "bytes" - "encoding/hex" - "net/url" - "os" - "strings" - "unicode" - - "github.com/pkg/errors" -) - -// URI implements a parser for a URI format based on the the PKCS #11 URI Scheme -// defined in https://tools.ietf.org/html/rfc7512 -// -// These URIs will be used to define the key names in a KMS. -type URI struct { - *url.URL - Values url.Values -} - -// New creates a new URI from a scheme and key-value pairs. -func New(scheme string, values url.Values) *URI { - return &URI{ - URL: &url.URL{ - Scheme: scheme, - Opaque: strings.ReplaceAll(values.Encode(), "&", ";"), - }, - Values: values, - } -} - -// NewFile creates an uri for a file. -func NewFile(path string) *URI { - return &URI{ - URL: &url.URL{ - Scheme: "file", - Path: path, - }, - } -} - -// HasScheme returns true if the given uri has the given scheme, false otherwise. -func HasScheme(scheme, rawuri string) bool { - u, err := url.Parse(rawuri) - if err != nil { - return false - } - return strings.EqualFold(u.Scheme, scheme) -} - -// Parse returns the URI for the given string or an error. -func Parse(rawuri string) (*URI, error) { - u, err := url.Parse(rawuri) - if err != nil { - return nil, errors.Wrapf(err, "error parsing %s", rawuri) - } - if u.Scheme == "" { - return nil, errors.Errorf("error parsing %s: scheme is missing", rawuri) - } - // Starting with Go 1.17 url.ParseQuery returns an error using semicolon as - // separator. - v, err := url.ParseQuery(strings.ReplaceAll(u.Opaque, ";", "&")) - if err != nil { - return nil, errors.Wrapf(err, "error parsing %s", rawuri) - } - - return &URI{ - URL: u, - Values: v, - }, nil -} - -// ParseWithScheme returns the URI for the given string only if it has the given -// scheme. -func ParseWithScheme(scheme, rawuri string) (*URI, error) { - u, err := Parse(rawuri) - if err != nil { - return nil, err - } - if !strings.EqualFold(u.Scheme, scheme) { - return nil, errors.Errorf("error parsing %s: scheme not expected", rawuri) - } - return u, nil -} - -// Get returns the first value in the uri with the given key, it will return -// empty string if that field is not present. -func (u *URI) Get(key string) string { - v := u.Values.Get(key) - if v == "" { - v = u.URL.Query().Get(key) - } - return v -} - -// GetBool returns true if a given key has the value "true". It returns false -// otherwise. -func (u *URI) GetBool(key string) bool { - v := u.Values.Get(key) - if v == "" { - v = u.URL.Query().Get(key) - } - return strings.EqualFold(v, "true") -} - -// GetEncoded returns the first value in the uri with the given key, it will -// return empty nil if that field is not present or is empty. If the return -// value is hex encoded it will decode it and return it. -func (u *URI) GetEncoded(key string) []byte { - v := u.Get(key) - if v == "" { - return nil - } - if len(v)%2 == 0 { - if b, err := hex.DecodeString(v); err == nil { - return b - } - } - return []byte(v) -} - -// Pin returns the pin encoded in the url. It will read the pin from the -// pin-value or the pin-source attributes. -func (u *URI) Pin() string { - if value := u.Get("pin-value"); value != "" { - return value - } - if path := u.Get("pin-source"); path != "" { - if b, err := readFile(path); err == nil { - return string(bytes.TrimRightFunc(b, unicode.IsSpace)) - } - } - return "" -} - -func readFile(path string) ([]byte, error) { - u, err := url.Parse(path) - if err == nil && (u.Scheme == "" || u.Scheme == "file") && u.Path != "" { - path = u.Path - } - b, err := os.ReadFile(path) - if err != nil { - return nil, errors.Wrapf(err, "error reading %s", path) - } - return b, nil -} diff --git a/kms/uri/uri_119_test.go b/kms/uri/uri_119_test.go deleted file mode 100644 index af8f9939..00000000 --- a/kms/uri/uri_119_test.go +++ /dev/null @@ -1,62 +0,0 @@ -//go:build go1.19 - -package uri - -import ( - "net/url" - "reflect" - "testing" -) - -func TestParse(t *testing.T) { - type args struct { - rawuri string - } - tests := []struct { - name string - args args - want *URI - wantErr bool - }{ - {"ok", args{"yubikey:slot-id=9a"}, &URI{ - URL: &url.URL{Scheme: "yubikey", Opaque: "slot-id=9a"}, - Values: url.Values{"slot-id": []string{"9a"}}, - }, false}, - {"ok schema", args{"cloudkms:"}, &URI{ - URL: &url.URL{Scheme: "cloudkms"}, - Values: url.Values{}, - }, false}, - {"ok query", args{"yubikey:slot-id=9a;foo=bar?pin=123456&foo=bar"}, &URI{ - URL: &url.URL{Scheme: "yubikey", Opaque: "slot-id=9a;foo=bar", RawQuery: "pin=123456&foo=bar"}, - Values: url.Values{"slot-id": []string{"9a"}, "foo": []string{"bar"}}, - }, false}, - {"ok file", args{"file:///tmp/ca.cert"}, &URI{ - URL: &url.URL{Scheme: "file", Path: "/tmp/ca.cert"}, - Values: url.Values{}, - }, false}, - {"ok file simple", args{"file:/tmp/ca.cert"}, &URI{ - URL: &url.URL{Scheme: "file", Path: "/tmp/ca.cert", OmitHost: true}, - Values: url.Values{}, - }, false}, - {"ok file host", args{"file://tmp/ca.cert"}, &URI{ - URL: &url.URL{Scheme: "file", Host: "tmp", Path: "/ca.cert"}, - Values: url.Values{}, - }, false}, - {"fail schema", args{"cloudkms"}, nil, true}, - {"fail parse", args{"yubi%key:slot-id=9a"}, nil, true}, - {"fail scheme", args{"yubikey"}, nil, true}, - {"fail parse opaque", args{"yubikey:slot-id=%ZZ"}, nil, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - got, err := Parse(tt.args.rawuri) - if (err != nil) != tt.wantErr { - t.Errorf("Parse() error = %v, wantErr %v", err, tt.wantErr) - return - } - if !reflect.DeepEqual(got, tt.want) { - t.Errorf("Parse() = %#v, want %#v", got.URL, tt.want.URL) - } - }) - } -} diff --git a/kms/uri/uri_other_test.go b/kms/uri/uri_other_test.go deleted file mode 100644 index dec50f55..00000000 --- a/kms/uri/uri_other_test.go +++ /dev/null @@ -1,62 +0,0 @@ -//go:build !go1.19 - -package uri - -import ( - "net/url" - "reflect" - "testing" -) - -func TestParse(t *testing.T) { - type args struct { - rawuri string - } - tests := []struct { - name string - args args - want *URI - wantErr bool - }{ - {"ok", args{"yubikey:slot-id=9a"}, &URI{ - URL: &url.URL{Scheme: "yubikey", Opaque: "slot-id=9a"}, - Values: url.Values{"slot-id": []string{"9a"}}, - }, false}, - {"ok schema", args{"cloudkms:"}, &URI{ - URL: &url.URL{Scheme: "cloudkms"}, - Values: url.Values{}, - }, false}, - {"ok query", args{"yubikey:slot-id=9a;foo=bar?pin=123456&foo=bar"}, &URI{ - URL: &url.URL{Scheme: "yubikey", Opaque: "slot-id=9a;foo=bar", RawQuery: "pin=123456&foo=bar"}, - Values: url.Values{"slot-id": []string{"9a"}, "foo": []string{"bar"}}, - }, false}, - {"ok file", args{"file:///tmp/ca.cert"}, &URI{ - URL: &url.URL{Scheme: "file", Path: "/tmp/ca.cert"}, - Values: url.Values{}, - }, false}, - {"ok file simple", args{"file:/tmp/ca.cert"}, &URI{ - URL: &url.URL{Scheme: "file", Path: "/tmp/ca.cert"}, - Values: url.Values{}, - }, false}, - {"ok file host", args{"file://tmp/ca.cert"}, &URI{ - URL: &url.URL{Scheme: "file", Host: "tmp", Path: "/ca.cert"}, - Values: url.Values{}, - }, false}, - {"fail schema", args{"cloudkms"}, nil, true}, - {"fail parse", args{"yubi%key:slot-id=9a"}, nil, true}, - {"fail scheme", args{"yubikey"}, nil, true}, - {"fail parse opaque", args{"yubikey:slot-id=%ZZ"}, nil, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - got, err := Parse(tt.args.rawuri) - if (err != nil) != tt.wantErr { - t.Errorf("Parse() error = %v, wantErr %v", err, tt.wantErr) - return - } - if !reflect.DeepEqual(got, tt.want) { - t.Errorf("Parse() = %#v, want %#v", got.URL, tt.want.URL) - } - }) - } -} diff --git a/kms/uri/uri_test.go b/kms/uri/uri_test.go deleted file mode 100644 index 2ffb5f3d..00000000 --- a/kms/uri/uri_test.go +++ /dev/null @@ -1,282 +0,0 @@ -package uri - -import ( - "net/url" - "reflect" - "testing" -) - -func TestNew(t *testing.T) { - type args struct { - scheme string - values url.Values - } - tests := []struct { - name string - args args - want *URI - }{ - {"ok", args{"yubikey", url.Values{"slot-id": []string{"9a"}}}, &URI{ - URL: &url.URL{Scheme: "yubikey", Opaque: "slot-id=9a"}, - Values: url.Values{"slot-id": []string{"9a"}}, - }}, - {"ok multiple", args{"yubikey", url.Values{"slot-id": []string{"9a"}, "foo": []string{"bar"}}}, &URI{ - URL: &url.URL{Scheme: "yubikey", Opaque: "foo=bar;slot-id=9a"}, - Values: url.Values{ - "slot-id": []string{"9a"}, - "foo": []string{"bar"}, - }, - }}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - if got := New(tt.args.scheme, tt.args.values); !reflect.DeepEqual(got, tt.want) { - t.Errorf("New() = %v, want %v", got, tt.want) - } - }) - } -} - -func TestNewFile(t *testing.T) { - type args struct { - path string - } - tests := []struct { - name string - args args - want *URI - }{ - {"ok", args{"/tmp/ca.crt"}, &URI{ - URL: &url.URL{Scheme: "file", Path: "/tmp/ca.crt"}, - Values: url.Values(nil), - }}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - if got := NewFile(tt.args.path); !reflect.DeepEqual(got, tt.want) { - t.Errorf("NewFile() = %v, want %v", got, tt.want) - } - }) - } -} - -func TestHasScheme(t *testing.T) { - type args struct { - scheme string - rawuri string - } - tests := []struct { - name string - args args - want bool - }{ - {"ok", args{"yubikey", "yubikey:slot-id=9a"}, true}, - {"ok empty", args{"yubikey", "yubikey:"}, true}, - {"ok letter case", args{"awsKMS", "AWSkms:key-id=abcdefg?foo=bar"}, true}, - {"fail", args{"yubikey", "awskms:key-id=abcdefg"}, false}, - {"fail parse", args{"yubikey", "yubi%key:slot-id=9a"}, false}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - if got := HasScheme(tt.args.scheme, tt.args.rawuri); got != tt.want { - t.Errorf("HasScheme() = %v, want %v", got, tt.want) - } - }) - } -} - -func TestParseWithScheme(t *testing.T) { - type args struct { - scheme string - rawuri string - } - tests := []struct { - name string - args args - want *URI - wantErr bool - }{ - {"ok", args{"yubikey", "yubikey:slot-id=9a"}, &URI{ - URL: &url.URL{Scheme: "yubikey", Opaque: "slot-id=9a"}, - Values: url.Values{"slot-id": []string{"9a"}}, - }, false}, - {"ok schema", args{"cloudkms", "cloudkms:"}, &URI{ - URL: &url.URL{Scheme: "cloudkms"}, - Values: url.Values{}, - }, false}, - {"ok file", args{"file", "file:///tmp/ca.cert"}, &URI{ - URL: &url.URL{Scheme: "file", Path: "/tmp/ca.cert"}, - Values: url.Values{}, - }, false}, - {"fail parse", args{"yubikey", "yubikey"}, nil, true}, - {"fail scheme", args{"yubikey", "awskms:slot-id=9a"}, nil, true}, - {"fail schema", args{"cloudkms", "cloudkms"}, nil, true}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - got, err := ParseWithScheme(tt.args.scheme, tt.args.rawuri) - if (err != nil) != tt.wantErr { - t.Errorf("ParseWithScheme() error = %v, wantErr %v", err, tt.wantErr) - return - } - if !reflect.DeepEqual(got, tt.want) { - t.Errorf("ParseWithScheme() = %v, want %v", got, tt.want) - } - }) - } -} - -func TestURI_Get(t *testing.T) { - mustParse := func(s string) *URI { - u, err := Parse(s) - if err != nil { - t.Fatal(err) - } - return u - } - type args struct { - key string - } - tests := []struct { - name string - uri *URI - args args - want string - }{ - {"ok", mustParse("yubikey:slot-id=9a"), args{"slot-id"}, "9a"}, - {"ok first", mustParse("yubikey:slot-id=9a;slot-id=9b"), args{"slot-id"}, "9a"}, - {"ok multiple", mustParse("yubikey:slot-id=9a;foo=bar"), args{"foo"}, "bar"}, - {"ok in query", mustParse("yubikey:slot-id=9a?foo=bar"), args{"foo"}, "bar"}, - {"fail missing", mustParse("yubikey:slot-id=9a"), args{"foo"}, ""}, - {"fail missing query", mustParse("yubikey:slot-id=9a?bar=zar"), args{"foo"}, ""}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - if got := tt.uri.Get(tt.args.key); got != tt.want { - t.Errorf("URI.Get() = %v, want %v", got, tt.want) - } - }) - } -} - -func TestURI_GetBool(t *testing.T) { - mustParse := func(s string) *URI { - u, err := Parse(s) - if err != nil { - t.Fatal(err) - } - return u - } - type args struct { - key string - } - tests := []struct { - name string - uri *URI - args args - want bool - }{ - {"true", mustParse("azurekms:name=foo;vault=bar;hsm=true"), args{"hsm"}, true}, - {"TRUE", mustParse("azurekms:name=foo;vault=bar;hsm=TRUE"), args{"hsm"}, true}, - {"tRUe query", mustParse("azurekms:name=foo;vault=bar?hsm=tRUe"), args{"hsm"}, true}, - {"false", mustParse("azurekms:name=foo;vault=bar;hsm=false"), args{"hsm"}, false}, - {"false query", mustParse("azurekms:name=foo;vault=bar?hsm=false"), args{"hsm"}, false}, - {"empty", mustParse("azurekms:name=foo;vault=bar;hsm=?bar=true"), args{"hsm"}, false}, - {"missing", mustParse("azurekms:name=foo;vault=bar"), args{"hsm"}, false}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - if got := tt.uri.GetBool(tt.args.key); got != tt.want { - t.Errorf("URI.GetBool() = %v, want %v", got, tt.want) - } - }) - } -} - -func TestURI_GetEncoded(t *testing.T) { - mustParse := func(s string) *URI { - u, err := Parse(s) - if err != nil { - t.Fatal(err) - } - return u - } - type args struct { - key string - } - tests := []struct { - name string - uri *URI - args args - want []byte - }{ - {"ok", mustParse("yubikey:slot-id=9a"), args{"slot-id"}, []byte{0x9a}}, - {"ok first", mustParse("yubikey:slot-id=9a9b;slot-id=9b"), args{"slot-id"}, []byte{0x9a, 0x9b}}, - {"ok percent", mustParse("yubikey:slot-id=9a;foo=%9a%9b%9c"), args{"foo"}, []byte{0x9a, 0x9b, 0x9c}}, - {"ok in query", mustParse("yubikey:slot-id=9a?foo=9a"), args{"foo"}, []byte{0x9a}}, - {"ok in query percent", mustParse("yubikey:slot-id=9a?foo=%9a"), args{"foo"}, []byte{0x9a}}, - {"ok missing", mustParse("yubikey:slot-id=9a"), args{"foo"}, nil}, - {"ok missing query", mustParse("yubikey:slot-id=9a?bar=zar"), args{"foo"}, nil}, - {"ok no hex", mustParse("yubikey:slot-id=09a?bar=zar"), args{"slot-id"}, []byte{'0', '9', 'a'}}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - got := tt.uri.GetEncoded(tt.args.key) - if !reflect.DeepEqual(got, tt.want) { - t.Errorf("URI.GetEncoded() = %v, want %v", got, tt.want) - } - }) - } -} - -func TestURI_Pin(t *testing.T) { - mustParse := func(s string) *URI { - u, err := Parse(s) - if err != nil { - t.Fatal(err) - } - return u - } - tests := []struct { - name string - uri *URI - want string - }{ - {"from value", mustParse("pkcs11:id=%72%73?pin-value=0123456789"), "0123456789"}, - {"from source", mustParse("pkcs11:id=%72%73?pin-source=testdata/pin.txt"), "trim-this-pin"}, - {"from missing", mustParse("pkcs11:id=%72%73"), ""}, - {"from source missing", mustParse("pkcs11:id=%72%73?pin-source=testdata/foo.txt"), ""}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - if got := tt.uri.Pin(); got != tt.want { - t.Errorf("URI.Pin() = %v, want %v", got, tt.want) - } - }) - } -} - -func TestURI_String(t *testing.T) { - mustParse := func(s string) *URI { - u, err := Parse(s) - if err != nil { - t.Fatal(err) - } - return u - } - tests := []struct { - name string - uri *URI - want string - }{ - {"ok new", New("yubikey", url.Values{"slot-id": []string{"9a"}, "foo": []string{"bar"}}), "yubikey:foo=bar;slot-id=9a"}, - {"ok parse", mustParse("yubikey:slot-id=9a;foo=bar?bar=zar"), "yubikey:slot-id=9a;foo=bar?bar=zar"}, - } - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - if got := tt.uri.String(); got != tt.want { - t.Errorf("URI.String() = %v, want %v", got, tt.want) - } - }) - } -} diff --git a/kms/yubikey/yubikey.go b/kms/yubikey/yubikey.go deleted file mode 100644 index b1d5f7e3..00000000 --- a/kms/yubikey/yubikey.go +++ /dev/null @@ -1,322 +0,0 @@ -//go:build cgo -// +build cgo - -package yubikey - -import ( - "context" - "crypto" - "crypto/x509" - "encoding/hex" - "net/url" - "strings" - - "github.com/go-piv/piv-go/piv" - "github.com/pkg/errors" - "github.com/smallstep/certificates/kms/apiv1" - "github.com/smallstep/certificates/kms/uri" -) - -// Scheme is the scheme used in uris. -const Scheme = "yubikey" - -// YubiKey implements the KMS interface on a YubiKey. -type YubiKey struct { - yk *piv.YubiKey - pin string - managementKey [24]byte -} - -// New initializes a new YubiKey. -// TODO(mariano): only one card is currently supported. -func New(ctx context.Context, opts apiv1.Options) (*YubiKey, error) { - managementKey := piv.DefaultManagementKey - - if opts.URI != "" { - u, err := uri.ParseWithScheme(Scheme, opts.URI) - if err != nil { - return nil, err - } - if v := u.Pin(); v != "" { - opts.Pin = v - } - if v := u.Get("management-key"); v != "" { - opts.ManagementKey = v - } - } - - // Deprecated way to set configuration parameters. - if opts.ManagementKey != "" { - b, err := hex.DecodeString(opts.ManagementKey) - if err != nil { - return nil, errors.Wrap(err, "error decoding managementKey") - } - if len(b) != 24 { - return nil, errors.New("invalid managementKey: length is not 24 bytes") - } - copy(managementKey[:], b[:24]) - } - - cards, err := piv.Cards() - if err != nil { - return nil, err - } - if len(cards) == 0 { - return nil, errors.New("error detecting yubikey: try removing and reconnecting the device") - } - - yk, err := piv.Open(cards[0]) - if err != nil { - return nil, errors.Wrap(err, "error opening yubikey") - } - - return &YubiKey{ - yk: yk, - pin: opts.Pin, - managementKey: managementKey, - }, nil -} - -func init() { - apiv1.Register(apiv1.YubiKey, func(ctx context.Context, opts apiv1.Options) (apiv1.KeyManager, error) { - return New(ctx, opts) - }) -} - -// LoadCertificate implements kms.CertificateManager and loads a certificate -// from the YubiKey. -func (k *YubiKey) LoadCertificate(req *apiv1.LoadCertificateRequest) (*x509.Certificate, error) { - slot, err := getSlot(req.Name) - if err != nil { - return nil, err - } - - cert, err := k.yk.Certificate(slot) - if err != nil { - return nil, errors.Wrap(err, "error retrieving certificate") - } - - return cert, nil -} - -// StoreCertificate implements kms.CertificateManager and stores a certificate -// in the YubiKey. -func (k *YubiKey) StoreCertificate(req *apiv1.StoreCertificateRequest) error { - if req.Certificate == nil { - return errors.New("storeCertificateRequest 'Certificate' cannot be nil") - } - - slot, err := getSlot(req.Name) - if err != nil { - return err - } - - err = k.yk.SetCertificate(k.managementKey, slot, req.Certificate) - if err != nil { - return errors.Wrap(err, "error storing certificate") - } - - return nil -} - -// GetPublicKey returns the public key present in the YubiKey signature slot. -func (k *YubiKey) GetPublicKey(req *apiv1.GetPublicKeyRequest) (crypto.PublicKey, error) { - slot, err := getSlot(req.Name) - if err != nil { - return nil, err - } - - pub, err := k.getPublicKey(slot) - if err != nil { - return nil, err - } - - return pub, nil -} - -// CreateKey generates a new key in the YubiKey and returns the public key. -func (k *YubiKey) CreateKey(req *apiv1.CreateKeyRequest) (*apiv1.CreateKeyResponse, error) { - alg, err := getSignatureAlgorithm(req.SignatureAlgorithm, req.Bits) - if err != nil { - return nil, err - } - slot, name, err := getSlotAndName(req.Name) - if err != nil { - return nil, err - } - - pub, err := k.yk.GenerateKey(k.managementKey, slot, piv.Key{ - Algorithm: alg, - PINPolicy: piv.PINPolicyAlways, - TouchPolicy: piv.TouchPolicyNever, - }) - if err != nil { - return nil, errors.Wrap(err, "error generating key") - } - return &apiv1.CreateKeyResponse{ - Name: name, - PublicKey: pub, - CreateSignerRequest: apiv1.CreateSignerRequest{ - SigningKey: name, - }, - }, nil -} - -// CreateSigner creates a signer using the key present in the YubiKey signature -// slot. -func (k *YubiKey) CreateSigner(req *apiv1.CreateSignerRequest) (crypto.Signer, error) { - slot, err := getSlot(req.SigningKey) - if err != nil { - return nil, err - } - - pub, err := k.getPublicKey(slot) - if err != nil { - return nil, err - } - - priv, err := k.yk.PrivateKey(slot, pub, piv.KeyAuth{ - PIN: k.pin, - PINPolicy: piv.PINPolicyAlways, - }) - if err != nil { - return nil, errors.Wrap(err, "error retrieving private key") - } - - signer, ok := priv.(crypto.Signer) - if !ok { - return nil, errors.New("private key is not a crypto.Signer") - } - return signer, nil -} - -// Close releases the connection to the YubiKey. -func (k *YubiKey) Close() error { - return errors.Wrap(k.yk.Close(), "error closing yubikey") -} - -// getPublicKey returns the public key on a slot. First it attempts to do -// attestation to get a certificate with the public key in it, if this succeeds -// means that the key was generated in the device. If not we'll try to get the -// key from a stored certificate in the same slot. -func (k *YubiKey) getPublicKey(slot piv.Slot) (crypto.PublicKey, error) { - cert, err := k.yk.Attest(slot) - if err != nil { - if cert, err = k.yk.Certificate(slot); err != nil { - return nil, errors.Wrap(err, "error retrieving public key") - } - } - return cert.PublicKey, nil -} - -// signatureAlgorithmMapping is a mapping between the step signature algorithm, -// and bits for RSA keys, with yubikey ones. -var signatureAlgorithmMapping = map[apiv1.SignatureAlgorithm]interface{}{ - apiv1.UnspecifiedSignAlgorithm: piv.AlgorithmEC256, - apiv1.SHA256WithRSA: map[int]piv.Algorithm{ - 0: piv.AlgorithmRSA2048, - 1024: piv.AlgorithmRSA1024, - 2048: piv.AlgorithmRSA2048, - }, - apiv1.SHA512WithRSA: map[int]piv.Algorithm{ - 0: piv.AlgorithmRSA2048, - 1024: piv.AlgorithmRSA1024, - 2048: piv.AlgorithmRSA2048, - }, - apiv1.SHA256WithRSAPSS: map[int]piv.Algorithm{ - 0: piv.AlgorithmRSA2048, - 1024: piv.AlgorithmRSA1024, - 2048: piv.AlgorithmRSA2048, - }, - apiv1.SHA512WithRSAPSS: map[int]piv.Algorithm{ - 0: piv.AlgorithmRSA2048, - 1024: piv.AlgorithmRSA1024, - 2048: piv.AlgorithmRSA2048, - }, - apiv1.ECDSAWithSHA256: piv.AlgorithmEC256, - apiv1.ECDSAWithSHA384: piv.AlgorithmEC384, -} - -func getSignatureAlgorithm(alg apiv1.SignatureAlgorithm, bits int) (piv.Algorithm, error) { - v, ok := signatureAlgorithmMapping[alg] - if !ok { - return 0, errors.Errorf("YubiKey does not support signature algorithm '%s'", alg) - } - - switch v := v.(type) { - case piv.Algorithm: - return v, nil - case map[int]piv.Algorithm: - signatureAlgorithm, ok := v[bits] - if !ok { - return 0, errors.Errorf("YubiKey does not support signature algorithm '%s' with '%d' bits", alg, bits) - } - return signatureAlgorithm, nil - default: - return 0, errors.Errorf("unexpected error: this should not happen") - } -} - -var slotMapping = map[string]piv.Slot{ - "9a": piv.SlotAuthentication, - "9c": piv.SlotSignature, - "9e": piv.SlotCardAuthentication, - "9d": piv.SlotKeyManagement, - "82": {Key: 0x82, Object: 0x5FC10D}, - "83": {Key: 0x83, Object: 0x5FC10E}, - "84": {Key: 0x84, Object: 0x5FC10F}, - "85": {Key: 0x85, Object: 0x5FC110}, - "86": {Key: 0x86, Object: 0x5FC111}, - "87": {Key: 0x87, Object: 0x5FC112}, - "88": {Key: 0x88, Object: 0x5FC113}, - "89": {Key: 0x89, Object: 0x5FC114}, - "8a": {Key: 0x8a, Object: 0x5FC115}, - "8b": {Key: 0x8b, Object: 0x5FC116}, - "8c": {Key: 0x8c, Object: 0x5FC117}, - "8d": {Key: 0x8d, Object: 0x5FC118}, - "8e": {Key: 0x8e, Object: 0x5FC119}, - "8f": {Key: 0x8f, Object: 0x5FC11A}, - "90": {Key: 0x90, Object: 0x5FC11B}, - "91": {Key: 0x91, Object: 0x5FC11C}, - "92": {Key: 0x92, Object: 0x5FC11D}, - "93": {Key: 0x93, Object: 0x5FC11E}, - "94": {Key: 0x94, Object: 0x5FC11F}, - "95": {Key: 0x95, Object: 0x5FC120}, -} - -func getSlot(name string) (piv.Slot, error) { - slot, _, err := getSlotAndName(name) - return slot, err -} - -func getSlotAndName(name string) (piv.Slot, string, error) { - if name == "" { - return piv.SlotSignature, "yubikey:slot-id=9c", nil - } - - var slotID string - name = strings.ToLower(name) - if strings.HasPrefix(name, "yubikey:") { - u, err := url.Parse(name) - if err != nil { - return piv.Slot{}, "", errors.Wrapf(err, "error parsing '%s'", name) - } - v, err := url.ParseQuery(u.Opaque) - if err != nil { - return piv.Slot{}, "", errors.Wrapf(err, "error parsing '%s'", name) - } - if slotID = v.Get("slot-id"); slotID == "" { - return piv.Slot{}, "", errors.Wrapf(err, "error parsing '%s': slot-id is missing", name) - } - } else { - slotID = name - } - - s, ok := slotMapping[slotID] - if !ok { - return piv.Slot{}, "", errors.Errorf("unsupported slot-id '%s'", name) - } - - name = "yubikey:slot-id=" + url.QueryEscape(slotID) - return s, name, nil -} diff --git a/kms/yubikey/yubikey_no_cgo.go b/kms/yubikey/yubikey_no_cgo.go deleted file mode 100644 index 24a76174..00000000 --- a/kms/yubikey/yubikey_no_cgo.go +++ /dev/null @@ -1,20 +0,0 @@ -//go:build !cgo -// +build !cgo - -package yubikey - -import ( - "context" - "os" - "path/filepath" - - "github.com/pkg/errors" - "github.com/smallstep/certificates/kms/apiv1" -) - -func init() { - apiv1.Register(apiv1.YubiKey, func(ctx context.Context, opts apiv1.Options) (apiv1.KeyManager, error) { - name := filepath.Base(os.Args[0]) - return nil, errors.Errorf("unsupported kms type 'yubikey': %s is compiled without cgo support", name) - }) -} diff --git a/logging/handler.go b/logging/handler.go index dd969f89..a8b77d60 100644 --- a/logging/handler.go +++ b/logging/handler.go @@ -5,6 +5,7 @@ import ( "net/http" "os" "strconv" + "strings" "time" "github.com/sirupsen/logrus" @@ -78,7 +79,7 @@ func (l *LoggerHandler) writeEntry(w ResponseLogger, r *http.Request, t time.Tim uri = r.Host } if uri == "" { - uri = r.URL.RequestURI() + uri = sanitizeLogEntry(r.URL.RequestURI()) } status := w.StatusCode() @@ -96,8 +97,8 @@ func (l *LoggerHandler) writeEntry(w ResponseLogger, r *http.Request, t time.Tim "protocol": r.Proto, "status": status, "size": w.Size(), - "referer": r.Referer(), - "user-agent": r.UserAgent(), + "referer": sanitizeLogEntry(r.Referer()), + "user-agent": sanitizeLogEntry(r.UserAgent()), } for k, v := range w.Fields() { @@ -117,3 +118,8 @@ func (l *LoggerHandler) writeEntry(w ResponseLogger, r *http.Request, t time.Tim l.logger.WithFields(fields).Error() } } + +func sanitizeLogEntry(s string) string { + escaped := strings.ReplaceAll(s, "\n", "") + return strings.ReplaceAll(escaped, "\r", "") +} diff --git a/monitoring/monitoring.go b/monitoring/monitoring.go index 2dda4840..a0d0886b 100644 --- a/monitoring/monitoring.go +++ b/monitoring/monitoring.go @@ -7,7 +7,7 @@ import ( "strconv" "strings" - newrelic "github.com/newrelic/go-agent" + "github.com/newrelic/go-agent/v3/newrelic" "github.com/pkg/errors" "github.com/smallstep/certificates/logging" ) @@ -41,7 +41,10 @@ func New(raw json.RawMessage) (*Monitoring, error) { m := new(Monitoring) switch strings.ToLower(config.Type) { case "", "newrelic": - app, err := newrelic.NewApplication(newrelic.NewConfig(config.Name, config.Key)) + app, err := newrelic.NewApplication( + newrelic.ConfigAppName(config.Name), + newrelic.ConfigLicense(config.Key), + ) if err != nil { return nil, errors.Wrap(err, "error loading New Relic application") } @@ -58,13 +61,16 @@ func (m *Monitoring) Middleware(next http.Handler) http.Handler { return m.middleware(next) } -func newRelicMiddleware(app newrelic.Application) Middleware { +func newRelicMiddleware(app *newrelic.Application) Middleware { return func(next http.Handler) http.Handler { return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { // Start transaction - txn := app.StartTransaction(transactionName(r), w, r) + txn := app.StartTransaction(transactionName(r)) defer txn.End() + w = txn.SetWebResponse(w) + txn.SetWebRequestHTTP(r) + // Wrap request writer if necessary rw := logging.NewResponseLogger(w) diff --git a/pki/pki.go b/pki/pki.go index 998afb74..4f3b2127 100644 --- a/pki/pki.go +++ b/pki/pki.go @@ -26,14 +26,14 @@ import ( "github.com/smallstep/certificates/cas" "github.com/smallstep/certificates/cas/apiv1" "github.com/smallstep/certificates/db" - "github.com/smallstep/certificates/kms" - kmsapi "github.com/smallstep/certificates/kms/apiv1" "github.com/smallstep/nosql" "go.step.sm/cli-utils/errs" "go.step.sm/cli-utils/fileutil" "go.step.sm/cli-utils/step" "go.step.sm/cli-utils/ui" "go.step.sm/crypto/jose" + "go.step.sm/crypto/kms" + kmsapi "go.step.sm/crypto/kms/apiv1" "go.step.sm/crypto/pemutil" "go.step.sm/linkedca" "golang.org/x/crypto/ssh" @@ -807,8 +807,9 @@ func (p *PKI) GenerateConfig(opt ...ConfigOption) (*authconfig.Config, error) { // Enable KMS if necessary if p.Kms != nil { + typ := strings.ToLower(p.Kms.Type.String()) cfg.KMS = &kmsapi.Options{ - Type: strings.ToLower(p.Kms.Type.String()), + Type: kmsapi.Type(typ), } } diff --git a/server/server.go b/server/server.go index 2b864148..e12c792c 100644 --- a/server/server.go +++ b/server/server.go @@ -39,13 +39,14 @@ func New(addr string, handler http.Handler, tlsConfig *tls.Config) *Server { // tls.Config. func newHTTPServer(addr string, handler http.Handler, tlsConfig *tls.Config) *http.Server { return &http.Server{ - Addr: addr, - Handler: handler, - TLSConfig: tlsConfig, - WriteTimeout: 15 * time.Second, - ReadTimeout: 15 * time.Second, - IdleTimeout: 15 * time.Second, - ErrorLog: log.New(os.Stderr, "", log.Ldate|log.Ltime|log.Llongfile), + Addr: addr, + Handler: handler, + TLSConfig: tlsConfig, + WriteTimeout: 15 * time.Second, + ReadTimeout: 15 * time.Second, + ReadHeaderTimeout: 15 * time.Second, + IdleTimeout: 15 * time.Second, + ErrorLog: log.New(os.Stderr, "", log.Ldate|log.Ltime|log.Llongfile), } } diff --git a/templates/templates_test.go b/templates/templates_test.go index 2f169dac..0093d60f 100644 --- a/templates/templates_test.go +++ b/templates/templates_test.go @@ -6,7 +6,6 @@ import ( "crypto/rand" "encoding/base64" "fmt" - "io/ioutil" "os" "path/filepath" "reflect" @@ -369,7 +368,7 @@ func TestTemplate_Output(t *testing.T) { } func TestOutput_Write(t *testing.T) { - dir, err := ioutil.TempDir("", "test-output-write") + dir, err := os.MkdirTemp("", "test-output-write") assert.FatalError(t, err) defer os.RemoveAll(dir)