From a077c65f16eb1cfe0e558cbfd3c2b6ab0bd024a9 Mon Sep 17 00:00:00 2001 From: Daisuke Maki Date: Thu, 24 Aug 2023 11:45:17 +0900 Subject: [PATCH 1/6] Rough cut of exposing ways to integrate new key types better --- examples/jwk_example_test.go | 2 +- internal/jwxtest/jwxtest.go | 10 +- internal/keyconv/keyconv.go | 14 +- jwe/internal/keyenc/keyenc_test.go | 4 +- jwe/jwe.go | 8 +- jwe/jwe_test.go | 8 +- jwk/convert.go | 260 +++++++++++++++++++++++++ jwk/ecdsa.go | 4 +- jwk/es256k.go | 193 +++++++++++++++++++ jwk/es256k_test.go | 16 ++ jwk/jwk.go | 241 +---------------------- jwk/jwk_test.go | 26 +-- jwk/pem.go | 294 +++++++++++++++++++++++++++++ jwk/rsa.go | 6 +- jwk/symmetric.go | 2 +- jws/jws_test.go | 10 +- jwx_test.go | 8 +- 17 files changed, 818 insertions(+), 288 deletions(-) create mode 100644 jwk/convert.go create mode 100644 jwk/pem.go diff --git a/examples/jwk_example_test.go b/examples/jwk_example_test.go index 2eb294c2c..505ed2d6d 100644 --- a/examples/jwk_example_test.go +++ b/examples/jwk_example_test.go @@ -32,7 +32,7 @@ func ExampleJWK_Usage() { key := pair.Value.(jwk.Key) var rawkey interface{} // This is the raw key, like *rsa.PrivateKey or *ecdsa.PrivateKey - if err := key.Raw(&rawkey); err != nil { + if err := jwk.Raw(key, &rawkey); err != nil { log.Printf("failed to create public key: %s", err) return } diff --git a/internal/jwxtest/jwxtest.go b/internal/jwxtest/jwxtest.go index cd9e8af2e..24220acee 100644 --- a/internal/jwxtest/jwxtest.go +++ b/internal/jwxtest/jwxtest.go @@ -270,7 +270,7 @@ func DecryptJweFile(ctx context.Context, file string, alg jwa.KeyEncryptionAlgor } var rawkey interface{} - if err := key.Raw(&rawkey); err != nil { + if err := jwk.Raw(key, &rawkey); err != nil { return nil, fmt.Errorf(`failed to obtain raw key from JWK: %w`, err) } @@ -288,19 +288,19 @@ func EncryptJweFile(ctx context.Context, payload []byte, keyalg jwa.KeyEncryptio switch keyalg { case jwa.RSA1_5, jwa.RSA_OAEP, jwa.RSA_OAEP_256: var rawkey rsa.PrivateKey - if err := key.Raw(&rawkey); err != nil { + if err := jwk.Raw(key, &rawkey); err != nil { return "", nil, fmt.Errorf(`failed to obtain raw key: %w`, err) } keyif = rawkey.PublicKey case jwa.ECDH_ES, jwa.ECDH_ES_A128KW, jwa.ECDH_ES_A192KW, jwa.ECDH_ES_A256KW: var rawkey ecdsa.PrivateKey - if err := key.Raw(&rawkey); err != nil { + if err := jwk.Raw(key, &rawkey); err != nil { return "", nil, fmt.Errorf(`failed to obtain raw key: %w`, err) } keyif = rawkey.PublicKey default: var rawkey []byte - if err := key.Raw(&rawkey); err != nil { + if err := jwk.Raw(key, &rawkey); err != nil { return "", nil, fmt.Errorf(`failed to obtain raw key: %w`, err) } keyif = rawkey @@ -326,7 +326,7 @@ func VerifyJwsFile(ctx context.Context, file string, alg jwa.SignatureAlgorithm, } var rawkey, pubkey interface{} - if err := key.Raw(&rawkey); err != nil { + if err := jwk.Raw(key, &rawkey); err != nil { return nil, fmt.Errorf(`failed to obtain raw key from JWK: %w`, err) } pubkey = rawkey diff --git a/internal/keyconv/keyconv.go b/internal/keyconv/keyconv.go index 807da1dee..0114325ed 100644 --- a/internal/keyconv/keyconv.go +++ b/internal/keyconv/keyconv.go @@ -17,7 +17,7 @@ import ( func RSAPrivateKey(dst, src interface{}) error { if jwkKey, ok := src.(jwk.Key); ok { var raw rsa.PrivateKey - if err := jwkKey.Raw(&raw); err != nil { + if err := jwk.Raw(jwkKey, &raw); err != nil { return fmt.Errorf(`failed to produce rsa.PrivateKey from %T: %w`, src, err) } src = &raw @@ -42,7 +42,7 @@ func RSAPrivateKey(dst, src interface{}) error { func RSAPublicKey(dst, src interface{}) error { if jwkKey, ok := src.(jwk.Key); ok { var raw rsa.PublicKey - if err := jwkKey.Raw(&raw); err != nil { + if err := jwk.Raw(jwkKey, &raw); err != nil { return fmt.Errorf(`failed to produce rsa.PublicKey from %T: %w`, src, err) } src = &raw @@ -66,7 +66,7 @@ func RSAPublicKey(dst, src interface{}) error { func ECDSAPrivateKey(dst, src interface{}) error { if jwkKey, ok := src.(jwk.Key); ok { var raw ecdsa.PrivateKey - if err := jwkKey.Raw(&raw); err != nil { + if err := jwk.Raw(jwkKey, &raw); err != nil { return fmt.Errorf(`failed to produce ecdsa.PrivateKey from %T: %w`, src, err) } src = &raw @@ -89,7 +89,7 @@ func ECDSAPrivateKey(dst, src interface{}) error { func ECDSAPublicKey(dst, src interface{}) error { if jwkKey, ok := src.(jwk.Key); ok { var raw ecdsa.PublicKey - if err := jwkKey.Raw(&raw); err != nil { + if err := jwk.Raw(jwkKey, &raw); err != nil { return fmt.Errorf(`failed to produce ecdsa.PublicKey from %T: %w`, src, err) } src = &raw @@ -110,7 +110,7 @@ func ECDSAPublicKey(dst, src interface{}) error { func ByteSliceKey(dst, src interface{}) error { if jwkKey, ok := src.(jwk.Key); ok { var raw []byte - if err := jwkKey.Raw(&raw); err != nil { + if err := jwk.Raw(jwkKey, &raw); err != nil { return fmt.Errorf(`failed to produce []byte from %T: %w`, src, err) } src = raw @@ -125,7 +125,7 @@ func ByteSliceKey(dst, src interface{}) error { func Ed25519PrivateKey(dst, src interface{}) error { if jwkKey, ok := src.(jwk.Key); ok { var raw ed25519.PrivateKey - if err := jwkKey.Raw(&raw); err != nil { + if err := jwk.Raw(jwkKey, &raw); err != nil { return fmt.Errorf(`failed to produce ed25519.PrivateKey from %T: %w`, src, err) } src = &raw @@ -146,7 +146,7 @@ func Ed25519PrivateKey(dst, src interface{}) error { func Ed25519PublicKey(dst, src interface{}) error { if jwkKey, ok := src.(jwk.Key); ok { var raw ed25519.PublicKey - if err := jwkKey.Raw(&raw); err != nil { + if err := jwk.Raw(jwkKey, &raw); err != nil { return fmt.Errorf(`failed to produce ed25519.PublicKey from %T: %w`, src, err) } src = &raw diff --git a/jwe/internal/keyenc/keyenc_test.go b/jwe/internal/keyenc/keyenc_test.go index f2ed9c994..169e6a24b 100644 --- a/jwe/internal/keyenc/keyenc_test.go +++ b/jwe/internal/keyenc/keyenc_test.go @@ -101,7 +101,7 @@ func TestDeriveECDHES(t *testing.T) { if !assert.NoError(t, err, `jwk.ParseKey should succeed`) { return } - if !assert.NoError(t, aliceWebKey.Raw(&aliceKey), `aliceWebKey.Raw should succeed`) { + if !assert.NoError(t, jwk.Raw(aliceWebKey, &aliceKey), `aliceWebKey.Raw should succeed`) { return } @@ -109,7 +109,7 @@ func TestDeriveECDHES(t *testing.T) { if !assert.NoError(t, err, `jwk.ParseKey should succeed`) { return } - if !assert.NoError(t, bobWebKey.Raw(&bobKey), `bobWebKey.Raw should succeed`) { + if !assert.NoError(t, jwk.Raw(bobWebKey, &bobKey), `bobWebKey.Raw should succeed`) { return } diff --git a/jwe/jwe.go b/jwe/jwe.go index 67b8e97b3..3d2c84a4d 100644 --- a/jwe/jwe.go +++ b/jwe/jwe.go @@ -76,7 +76,7 @@ func (b *recipientBuilder) Build(cek []byte, calg jwa.ContentEncryptionAlgorithm keyID = jwkKey.KeyID() var raw interface{} - if err := jwkKey.Raw(&raw); err != nil { + if err := jwk.Raw(jwkKey, &raw); err != nil { return nil, nil, fmt.Errorf(`failed to retrieve raw key out of %T: %w`, b.key, err) } @@ -573,7 +573,7 @@ func (dctx *decryptCtx) try(ctx context.Context, recipient Recipient, keyUsed in func (dctx *decryptCtx) decryptContent(ctx context.Context, alg jwa.KeyEncryptionAlgorithm, key interface{}, recipient Recipient) ([]byte, error) { if jwkKey, ok := key.(jwk.Key); ok { var raw interface{} - if err := jwkKey.Raw(&raw); err != nil { + if err := jwk.Raw(jwkKey, &raw); err != nil { return nil, fmt.Errorf(`failed to retrieve raw key from %T: %w`, key, err) } key = raw @@ -609,13 +609,13 @@ func (dctx *decryptCtx) decryptContent(ctx context.Context, alg jwa.KeyEncryptio switch epk := epkif.(type) { case jwk.ECDSAPublicKey: var pubkey ecdsa.PublicKey - if err := epk.Raw(&pubkey); err != nil { + if err := jwk.Raw(epk, &pubkey); err != nil { return nil, fmt.Errorf(`failed to get public key: %w`, err) } dec.PublicKey(&pubkey) case jwk.OKPPublicKey: var pubkey interface{} - if err := epk.Raw(&pubkey); err != nil { + if err := jwk.Raw(epk, &pubkey); err != nil { return nil, fmt.Errorf(`failed to get public key: %w`, err) } dec.PublicKey(pubkey) diff --git a/jwe/jwe_test.go b/jwe/jwe_test.go index 64800de9d..b9459b6d7 100644 --- a/jwe/jwe_test.go +++ b/jwe/jwe_test.go @@ -50,7 +50,7 @@ func init() { panic(err) } - if err := privkey.Raw(&rsaPrivKey); err != nil { + if err := jwk.Raw(privkey, &rsaPrivKey); err != nil { panic(err) } } @@ -168,7 +168,7 @@ func TestParse_RSAES_OAEP_AES_GCM(t *testing.T) { } var rawkey rsa.PrivateKey - if !assert.NoError(t, privkey.Raw(&rawkey), `obtaining raw key should succeed`) { + if !assert.NoError(t, jwk.Raw(privkey, &rawkey), `obtaining raw key should succeed`) { return } @@ -503,7 +503,7 @@ func Test_GHIssue207(t *testing.T) { } var key ecdsa.PrivateKey - if !assert.NoError(t, webKey.Raw(&key), `jwk.Raw should succeed`) { + if !assert.NoError(t, jwk.Raw(webKey, &key), `jwk.Raw should succeed`) { return } @@ -630,7 +630,7 @@ func TestDecodePredefined_Direct(t *testing.T) { } var key []byte - if !assert.NoError(t, webKey.Raw(&key), `jwk.Raw should succeed`) { + if !assert.NoError(t, jwk.Raw(webKey, &key), `jwk.Raw should succeed`) { return } diff --git a/jwk/convert.go b/jwk/convert.go new file mode 100644 index 000000000..502f69131 --- /dev/null +++ b/jwk/convert.go @@ -0,0 +1,260 @@ +package jwk + +import ( + "crypto/ecdsa" + "crypto/ed25519" + "crypto/rsa" + "fmt" + "sync" + + "github.com/lestrrat-go/jwx/v2/x25519" +) + +type RawFromKeyer interface { + RawFromKey(Key, interface{}) error +} + +type ChainedRawFromKeyer interface { + Next(RawFromKeyer, Key, interface{}) error +} + +type ChainedRawFromKeyFunc func(RawFromKeyer, Key, interface{}) error + +func (fn ChainedRawFromKeyFunc) Next(n RawFromKeyer, key Key, raw interface{}) error { + return fn(n, key, raw) +} + +type chainedRawFromKey struct { + mu sync.RWMutex + list []ChainedRawFromKeyer +} + +type chainedRawFromKeyCallState struct { + current int + parent *chainedRawFromKey +} + +func (c *chainedRawFromKey) Add(rfk ChainedRawFromKeyer) { + if rfk == nil { + return // no-op + } + c.mu.Lock() + defer c.mu.Unlock() + c.list = append(c.list, rfk) +} + +func (c *chainedRawFromKey) Next(key Key, raw interface{}) error { + c.mu.RLock() + lrfk := len(c.list) + c.mu.RUnlock() + st := &chainedRawFromKeyCallState{parent: c, current: lrfk} + return st.RawFromKey(key, raw) +} + +func (s *chainedRawFromKeyCallState) RawFromKey(key Key, raw interface{}) error { + idx := s.current - 1 + + s.parent.mu.RLock() + defer s.parent.mu.RUnlock() + + llist := len(s.parent.list) + if idx < 0 || idx >= llist { + return fmt.Errorf(`jwk.Raw: invalid raw key type %T`, raw) + } + s.current = idx + + rfk := s.parent.list[idx] + return rfk.Next(s, key, raw) +} + +type chainedKeyFromRaw struct { + mu sync.RWMutex + list []ChainedKeyFromRawer +} + +type chainedKeyFromRawCallState struct { + current int + parent *chainedKeyFromRaw +} + +func (c *chainedKeyFromRaw) Add(kfr ChainedKeyFromRawer) { + if kfr == nil { + return // no-op + } + c.mu.Lock() + defer c.mu.Unlock() + c.list = append(c.list, kfr) +} + +func (c *chainedKeyFromRaw) Next(raw interface{}) (Key, error) { + c.mu.RLock() + lkfr := len(c.list) + c.mu.RUnlock() + st := &chainedKeyFromRawCallState{parent: c, current: lkfr} + return st.KeyFromRaw(raw) +} + +func (s *chainedKeyFromRawCallState) KeyFromRaw(raw interface{}) (Key, error) { + idx := s.current - 1 + + s.parent.mu.RLock() + defer s.parent.mu.RUnlock() + + llist := len(s.parent.list) + if idx < 0 || idx >= llist { + return nil, fmt.Errorf(`jwk.FromRaw: invalid raw key type %T`, raw) + } + s.current = idx + + kfr := s.parent.list[idx] + return kfr.Next(s, raw) +} + +var chainedKFR = &chainedKeyFromRaw{ + list: []ChainedKeyFromRawer{ChainedKeyFromRawFunc(fromRaw)}, +} + +var chainedRFK = &chainedRawFromKey{ + list: []ChainedRawFromKeyer{ChainedRawFromKeyFunc(toRaw)}, +} + +type KeyFromRawer interface { + KeyFromRaw(interface{}) (Key, error) +} + +// ChainedKeyFromRawer describes a type that can build a Key from a raw key +// +// ChainedKeyFromRawer objects are expected to be called in sequence. When a new +// object is added to the list of KeyFromRawer objects, they are called +// from the most recently added all the way up to the default object, +// if you choose to do so by invokind the first argument. +type ChainedKeyFromRawer interface { + // Next calls the handler in the subsequent chain of handlers. + // + // The first argument invokes the _next_ KeyFromRawer that can be called in the + // chain of possible KeyFromRawers that are registered. For example, + // if your KeyFromRawer failed to match any key type that you can handle, + // you can defer to the next KeyFromRawer to see if it can handle it + Next(KeyFromRawer, interface{}) (Key, error) +} + +// ChainedKeyFromRawFunc is an instance of ChainedKeyFromRawer represented by a function +type ChainedKeyFromRawFunc func(KeyFromRawer, interface{}) (Key, error) + +func (fn ChainedKeyFromRawFunc) Next(n KeyFromRawer, raw interface{}) (Key, error) { + return fn(n, raw) +} + +// AddKeyFromRaw adds a new KeyFromRawer object that is used in the FromRaw() function, which +// in turn will handle converting a raw key to a Key. +func AddKeyFromRaw(kfr ChainedKeyFromRawer) { + chainedKFR.Add(kfr) +} + +func fromRaw(_ KeyFromRawer, key interface{}) (Key, error) { + var ptr interface{} + switch v := key.(type) { + case rsa.PrivateKey: + ptr = &v + case rsa.PublicKey: + ptr = &v + case ecdsa.PrivateKey: + ptr = &v + case ecdsa.PublicKey: + ptr = &v + default: + ptr = v + } + + switch rawKey := ptr.(type) { + case *rsa.PrivateKey: + k := newRSAPrivateKey() + if err := k.FromRaw(rawKey); err != nil { + return nil, fmt.Errorf(`failed to initialize %T from %T: %w`, k, rawKey, err) + } + return k, nil + case *rsa.PublicKey: + k := newRSAPublicKey() + if err := k.FromRaw(rawKey); err != nil { + return nil, fmt.Errorf(`failed to initialize %T from %T: %w`, k, rawKey, err) + } + return k, nil + case *ecdsa.PrivateKey: + k := newECDSAPrivateKey() + if err := k.FromRaw(rawKey); err != nil { + return nil, fmt.Errorf(`failed to initialize %T from %T: %w`, k, rawKey, err) + } + return k, nil + case *ecdsa.PublicKey: + k := newECDSAPublicKey() + if err := k.FromRaw(rawKey); err != nil { + return nil, fmt.Errorf(`failed to initialize %T from %T: %w`, k, rawKey, err) + } + return k, nil + case ed25519.PrivateKey: + k := newOKPPrivateKey() + if err := k.FromRaw(rawKey); err != nil { + return nil, fmt.Errorf(`failed to initialize %T from %T: %w`, k, rawKey, err) + } + return k, nil + case ed25519.PublicKey: + k := newOKPPublicKey() + if err := k.FromRaw(rawKey); err != nil { + return nil, fmt.Errorf(`failed to initialize %T from %T: %w`, k, rawKey, err) + } + return k, nil + case x25519.PrivateKey: + k := newOKPPrivateKey() + if err := k.FromRaw(rawKey); err != nil { + return nil, fmt.Errorf(`failed to initialize %T from %T: %w`, k, rawKey, err) + } + return k, nil + case x25519.PublicKey: + k := newOKPPublicKey() + if err := k.FromRaw(rawKey); err != nil { + return nil, fmt.Errorf(`failed to initialize %T from %T: %w`, k, rawKey, err) + } + return k, nil + case []byte: + k := newSymmetricKey() + if err := k.FromRaw(rawKey); err != nil { + return nil, fmt.Errorf(`failed to initialize %T from %T: %w`, k, rawKey, err) + } + return k, nil + default: + return nil, fmt.Errorf(`invalid key type '%T' for jwk.FromRaw`, key) + } +} + +// FromRaw creates a Key from the given key (RSA/ECDSA/symmetric keys). +// +// The constructor auto-detects the type of key to be instantiated +// based on the input type: +// +// - "crypto/rsa".PrivateKey and "crypto/rsa".PublicKey creates an RSA based key +// - "crypto/ecdsa".PrivateKey and "crypto/ecdsa".PublicKey creates an EC based key +// - "crypto/ed25519".PrivateKey and "crypto/ed25519".PublicKey creates an OKP based key +// - []byte creates a symmetric key +func FromRaw(raw interface{}) (Key, error) { + if raw == nil { + return nil, fmt.Errorf(`jwk.FromRaw requires a non-nil key`) + } + + return chainedKFR.Next(raw) +} + +// Raw converts a jwk.Key to its raw form and stores in the `raw` variable. +// `raw` must be a pointer to a compatible object. +// +// As of v2.0.12, it is recommended to use `jwk.Raw()` instead of `keyObject.Raw()`. +func Raw(key Key, raw interface{}) error { + return chainedRFK.Next(key, raw) +} + +func toRaw(_ RawFromKeyer, key Key, raw interface{}) error { + return key.Raw(raw) +} + +func AddRawFromKey(rfk ChainedRawFromKeyer) { + chainedRFK.Add(rfk) +} diff --git a/jwk/ecdsa.go b/jwk/ecdsa.go index 67a14ba63..a8293223d 100644 --- a/jwk/ecdsa.go +++ b/jwk/ecdsa.go @@ -186,7 +186,7 @@ func (k ecdsaPublicKey) Thumbprint(hash crypto.Hash) ([]byte, error) { defer k.mu.RUnlock() var key ecdsa.PublicKey - if err := k.Raw(&key); err != nil { + if err := Raw(&k, &key); err != nil { return nil, fmt.Errorf(`failed to materialize ecdsa.PublicKey for thumbprint generation: %w`, err) } @@ -210,7 +210,7 @@ func (k ecdsaPrivateKey) Thumbprint(hash crypto.Hash) ([]byte, error) { defer k.mu.RUnlock() var key ecdsa.PrivateKey - if err := k.Raw(&key); err != nil { + if err := Raw(&k, &key); err != nil { return nil, fmt.Errorf(`failed to materialize ecdsa.PrivateKey for thumbprint generation: %w`, err) } diff --git a/jwk/es256k.go b/jwk/es256k.go index 1a9d2346a..81ba314a6 100644 --- a/jwk/es256k.go +++ b/jwk/es256k.go @@ -4,11 +4,204 @@ package jwk import ( + "crypto/ecdsa" + "crypto/elliptic" + "crypto/x509/pkix" + "encoding/asn1" + "encoding/pem" + "fmt" + "sync" + "github.com/decred/dcrd/dcrec/secp256k1/v4" + "github.com/lestrrat-go/blackmagic" "github.com/lestrrat-go/jwx/v2/internal/ecutil" "github.com/lestrrat-go/jwx/v2/jwa" ) func init() { ecutil.RegisterCurve(secp256k1.S256(), jwa.Secp256k1) + + AddKeyFromRaw(ChainedKeyFromRawFunc(secp256k1FromRaw)) + AddRawFromKey(ChainedRawFromKeyFunc(secp256k1Raw)) + AddASN1Encoder(ASN1EncodeFunc(secp256k1ASN1Encode)) + AddASN1Decoder(ChainedASN1DecodeFunc(secp256k1ASN1Decode)) +} + +var secp256k1OID = asn1.ObjectIdentifier{1, 3, 132, 0, 10} +var secp256k1PkPool = sync.Pool{ + New: func() interface{} { + return make([]byte, 32+1) // 32 bytes + 1 + }, +} + +func getPkBuf(size int) []byte { + buf := secp256k1PkPool.Get().([]byte) + if cap(buf) < size { + buf = make([]byte, size) + } else { + buf = buf[:size] + } + return buf +} + +func releasePkBuf(buf []byte) { + // XXX Replace this with clear() when we remove support for go < 1.21 + for i := 0; i < len(buf); i++ { + buf[i] = byte(0) + } + + secp256k1PkPool.Put(buf) +} + +type secp256k1ASN1PrivateKey struct { + Version int + PrivateKey []byte + NamedCurveOID asn1.ObjectIdentifier `asn1:"optional,explicit,tag:0"` + PublicKey asn1.BitString `asn1:"optional,explicit,tag:1"` +} + +type secp256k1ASN1PublicKey struct { + Algorithm pkix.AlgorithmIdentifier + BitString asn1.BitString +} + +func secp256k1ASN1Encode(n NextASN1Encoder, key Key) (string, []byte, error) { + switch key := key.(type) { + case ECDSAPrivateKey: + if key.Crv() == jwa.Secp256k1 { + var raw secp256k1.PrivateKey + if err := Raw(key, &raw); err != nil { + return "", nil, fmt.Errorf(`failed to convert jwk.Key into raw key: %w`, err) + } + return secp256k1EncodePrivateKey(&raw) + } + case ECDSAPublicKey: + if key.Crv() == jwa.Secp256k1 { + var raw secp256k1.PublicKey + if err := Raw(key, &raw); err != nil { + return "", nil, fmt.Errorf(`failed to convert jwk.Key into raw key: %w`, err) + } + return secp256k1EncodePublicKey(&raw) + } + } + + return n(key) +} + +func secp256k1ASN1Decode(n ASN1Decoder, buf []byte) (interface{}, []byte, error) { + block, rest := pem.Decode(buf) + if block == nil { + return nil, buf, fmt.Errorf(`jwk: PEM block decoded to nil`) + } + + if block.Type == pmECPrivateKey { + var priv secp256k1ASN1PrivateKey + // for 1-3, we're going to believe that this may have been + // another EC key that can be decoded by the next decoder + if _, err := asn1.Unmarshal(block.Bytes, &priv); err != nil { // (1) + return n.ASN1Decode(buf) + } + + if !priv.NamedCurveOID.Equal(secp256k1OID) { // (2) + return n.ASN1Decode(buf) + } + + if priv.Version != 1 { // (3) + return n.ASN1Decode(buf) + } + + key := secp256k1.PrivKeyFromBytes(priv.PrivateKey) + return key, rest, nil + } + // All other cases including secp256k1 public key can be handled + // by the default handler + return n.ASN1Decode(buf) +} + +func secp256k1EncodePrivateKey(key *secp256k1.PrivateKey) (string, []byte, error) { + asECDSA := key.ToECDSA() + size := (asECDSA.Curve.Params().N.BitLen() + 7) / 8 + pkbuf := getPkBuf(size) + defer releasePkBuf(pkbuf) + + buf, err := asn1.Marshal(secp256k1ASN1PrivateKey{ + Version: 1, + PrivateKey: asECDSA.D.FillBytes(pkbuf), + NamedCurveOID: secp256k1OID, + PublicKey: asn1.BitString{ + Bytes: elliptic.Marshal(asECDSA.Curve, asECDSA.X, asECDSA.Y), + }, + }) + if err != nil { + return "", nil, fmt.Errorf(`failed to marshal secp256k1 private key: %w`, err) + } + + return pmECPrivateKey, buf, nil +} + +func secp256k1EncodePublicKey(key *secp256k1.PublicKey) (string, []byte, error) { + asECDSA := key.ToECDSA() + + pkbuf := elliptic.Marshal(asECDSA.Curve, asECDSA.X, asECDSA.Y) + + oidBuf, err := asn1.Marshal(secp256k1OID) + if err != nil { + return "", nil, fmt.Errorf(`failed to marshal oid in ASN.1 format`) + } + + buf, err := asn1.Marshal(secp256k1ASN1PublicKey{ + Algorithm: pkix.AlgorithmIdentifier{ + Algorithm: secp256k1OID, + Parameters: asn1.RawValue{ + FullBytes: oidBuf, + }, + }, + BitString: asn1.BitString{ + Bytes: pkbuf, + BitLength: 8 * len(pkbuf), + }, + }) + if err != nil { + return "", nil, fmt.Errorf(`failed to marshal secp256k1 public key: %w`, err) + } + + return pmPublicKey, buf, nil +} + +func secp256k1FromRaw(nextKFR KeyFromRawer, key interface{}) (Key, error) { + switch key := key.(type) { + case *secp256k1.PrivateKey: + return nextKFR.KeyFromRaw(key.ToECDSA()) + case *secp256k1.PublicKey: + return nextKFR.KeyFromRaw(key.ToECDSA()) + default: + return nextKFR.KeyFromRaw(key) + } +} + +func secp256k1Raw(nextRFK RawFromKeyer, key Key, raw interface{}) error { + // for secp256k1Raw keys, you can either create a ecdsa.* key or a + // secp256k1.* key. + switch raw := raw.(type) { + case *secp256k1.PrivateKey: + // we first get a ecdsa.PrivateKey, then convert it to secp256k1.PrivateKey + var ecdsaKey ecdsa.PrivateKey + if err := key.Raw(&ecdsaKey); err != nil { + return fmt.Errorf(`failed to convert JWK into raw ecdsa.PrivateKey: %w`, err) + } + return blackmagic.AssignIfCompatible(raw, secp256k1.PrivKeyFromBytes(ecdsaKey.D.Bytes())) + case *secp256k1.PublicKey: + // we first get a ecdsa.PublicKey, then convert it to secp256k1.PublicKey + var ecdsaKey ecdsa.PublicKey + if err := key.Raw(&ecdsaKey); err != nil { + return fmt.Errorf(`failed to convert JWK into raw ecdsa.PublicKey: %w`, err) + } + var x secp256k1.FieldVal + var y secp256k1.FieldVal + x.SetByteSlice(ecdsaKey.X.Bytes()) + y.SetByteSlice(ecdsaKey.Y.Bytes()) + return blackmagic.AssignIfCompatible(raw, secp256k1.NewPublicKey(&x, &y)) + default: + return nextRFK.RawFromKey(key, raw) + } } diff --git a/jwk/es256k_test.go b/jwk/es256k_test.go index 728aba812..b9c224631 100644 --- a/jwk/es256k_test.go +++ b/jwk/es256k_test.go @@ -19,6 +19,22 @@ import ( func TestES256K(t *testing.T) { require.True(t, ecutil.IsAvailable(jwa.Secp256k1), `jwa.Secp256k1 should be available`) + + t.Run("PEM", func(t *testing.T) { + privkey, err := secp256k1.GeneratePrivateKey() + require.NoError(t, err, `secp256k1.GeneratePrivateKey should succeed`) + + key, err := jwk.FromRaw(privkey) + require.NoError(t, err, `jwk.FromRaw should succeed`) + + buf, err := jwk.Pem(key) + require.NoError(t, err, `jwk.Pem should succeed`) + + parsed, err := jwk.ParseKey(buf, jwk.WithPEM(true)) + require.NoError(t, err, `jwk.ParseKey should succeed`) + + require.Equal(t, key, parsed, `jwk.ParseKey should return the same key`) + }) } func BenchmarkKeyInstantiation(b *testing.B) { diff --git a/jwk/jwk.go b/jwk/jwk.go index 8521ba6e9..d511c843c 100644 --- a/jwk/jwk.go +++ b/jwk/jwk.go @@ -11,7 +11,6 @@ import ( "crypto/elliptic" "crypto/rsa" "crypto/x509" - "encoding/pem" "fmt" "io" "math/big" @@ -32,94 +31,6 @@ func bigIntToBytes(n *big.Int) ([]byte, error) { return n.Bytes(), nil } -// FromRaw creates a jwk.Key from the given key (RSA/ECDSA/symmetric keys). -// -// The constructor auto-detects the type of key to be instantiated -// based on the input type: -// -// - "crypto/rsa".PrivateKey and "crypto/rsa".PublicKey creates an RSA based key -// - "crypto/ecdsa".PrivateKey and "crypto/ecdsa".PublicKey creates an EC based key -// - "crypto/ed25519".PrivateKey and "crypto/ed25519".PublicKey creates an OKP based key -// - []byte creates a symmetric key -func FromRaw(key interface{}) (Key, error) { - if key == nil { - return nil, fmt.Errorf(`jwk.FromRaw requires a non-nil key`) - } - - var ptr interface{} - switch v := key.(type) { - case rsa.PrivateKey: - ptr = &v - case rsa.PublicKey: - ptr = &v - case ecdsa.PrivateKey: - ptr = &v - case ecdsa.PublicKey: - ptr = &v - default: - ptr = v - } - - switch rawKey := ptr.(type) { - case *rsa.PrivateKey: - k := newRSAPrivateKey() - if err := k.FromRaw(rawKey); err != nil { - return nil, fmt.Errorf(`failed to initialize %T from %T: %w`, k, rawKey, err) - } - return k, nil - case *rsa.PublicKey: - k := newRSAPublicKey() - if err := k.FromRaw(rawKey); err != nil { - return nil, fmt.Errorf(`failed to initialize %T from %T: %w`, k, rawKey, err) - } - return k, nil - case *ecdsa.PrivateKey: - k := newECDSAPrivateKey() - if err := k.FromRaw(rawKey); err != nil { - return nil, fmt.Errorf(`failed to initialize %T from %T: %w`, k, rawKey, err) - } - return k, nil - case *ecdsa.PublicKey: - k := newECDSAPublicKey() - if err := k.FromRaw(rawKey); err != nil { - return nil, fmt.Errorf(`failed to initialize %T from %T: %w`, k, rawKey, err) - } - return k, nil - case ed25519.PrivateKey: - k := newOKPPrivateKey() - if err := k.FromRaw(rawKey); err != nil { - return nil, fmt.Errorf(`failed to initialize %T from %T: %w`, k, rawKey, err) - } - return k, nil - case ed25519.PublicKey: - k := newOKPPublicKey() - if err := k.FromRaw(rawKey); err != nil { - return nil, fmt.Errorf(`failed to initialize %T from %T: %w`, k, rawKey, err) - } - return k, nil - case x25519.PrivateKey: - k := newOKPPrivateKey() - if err := k.FromRaw(rawKey); err != nil { - return nil, fmt.Errorf(`failed to initialize %T from %T: %w`, k, rawKey, err) - } - return k, nil - case x25519.PublicKey: - k := newOKPPublicKey() - if err := k.FromRaw(rawKey); err != nil { - return nil, fmt.Errorf(`failed to initialize %T from %T: %w`, k, rawKey, err) - } - return k, nil - case []byte: - k := newSymmetricKey() - if err := k.FromRaw(rawKey); err != nil { - return nil, fmt.Errorf(`failed to initialize %T from %T: %w`, k, rawKey, err) - } - return k, nil - default: - return nil, fmt.Errorf(`invalid key type '%T' for jwk.New`, key) - } -} - // PublicSetOf returns a new jwk.Set consisting of // public keys of the keys contained in the set. // @@ -188,7 +99,7 @@ func PublicRawKeyOf(v interface{}) (interface{}, error) { } var raw interface{} - if err := pubk.Raw(&raw); err != nil { + if err := Raw(pubk, &raw); err != nil { return nil, fmt.Errorf(`failed to obtain raw key from %T: %w`, pubk, err) } return raw, nil @@ -233,14 +144,6 @@ func PublicRawKeyOf(v interface{}) (interface{}, error) { } } -const ( - pmPrivateKey = `PRIVATE KEY` - pmPublicKey = `PUBLIC KEY` - pmECPrivateKey = `EC PRIVATE KEY` - pmRSAPublicKey = `RSA PUBLIC KEY` - pmRSAPrivateKey = `RSA PRIVATE KEY` -) - // EncodeX509 encodes the key into a byte sequence in ASN.1 DER format // suitable for to be PEM encoded. The key can be a jwk.Key or a raw key // instance, but it must be one of the types supported by `x509` package. @@ -254,9 +157,9 @@ const ( // The second return value is the encoded byte sequence. func EncodeX509(v interface{}) (string, []byte, error) { // we can't import jwk, so just use the interface - if key, ok := v.(interface{ Raw(interface{}) error }); ok { + if key, ok := v.(Key); ok { var raw interface{} - if err := key.Raw(&raw); err != nil { + if err := Raw(key, &raw); err != nil { return "", nil, fmt.Errorf(`failed to get raw key out of %T: %w`, key, err) } @@ -290,77 +193,6 @@ func EncodeX509(v interface{}) (string, []byte, error) { } } -// EncodePEM encodes the key into a PEM encoded ASN.1 DER format. -// The key can be a jwk.Key or a raw key instance, but it must be one of -// the types supported by `x509` package. -// -// Internally, it uses the same routine as `jwk.EncodeX509()`, and therefore -// the same caveats apply -func EncodePEM(v interface{}) ([]byte, error) { - typ, marshaled, err := EncodeX509(v) - if err != nil { - return nil, fmt.Errorf(`failed to encode key in x509: %w`, err) - } - - block := &pem.Block{ - Type: typ, - Bytes: marshaled, - } - return pem.EncodeToMemory(block), nil -} - -// DecodePEM decodes a key in PEM encoded ASN.1 DER format. -// and returns a raw key -func DecodePEM(src []byte) (interface{}, []byte, error) { - block, rest := pem.Decode(src) - if block == nil { - return nil, nil, fmt.Errorf(`failed to decode PEM data`) - } - - switch block.Type { - // Handle the semi-obvious cases - case pmRSAPrivateKey: - key, err := x509.ParsePKCS1PrivateKey(block.Bytes) - if err != nil { - return nil, nil, fmt.Errorf(`failed to parse PKCS1 private key: %w`, err) - } - return key, rest, nil - case pmRSAPublicKey: - key, err := x509.ParsePKCS1PublicKey(block.Bytes) - if err != nil { - return nil, nil, fmt.Errorf(`failed to parse PKCS1 public key: %w`, err) - } - return key, rest, nil - case pmECPrivateKey: - key, err := x509.ParseECPrivateKey(block.Bytes) - if err != nil { - return nil, nil, fmt.Errorf(`failed to parse EC private key: %w`, err) - } - return key, rest, nil - case pmPublicKey: - // XXX *could* return dsa.PublicKey - key, err := x509.ParsePKIXPublicKey(block.Bytes) - if err != nil { - return nil, nil, fmt.Errorf(`failed to parse PKIX public key: %w`, err) - } - return key, rest, nil - case pmPrivateKey: - key, err := x509.ParsePKCS8PrivateKey(block.Bytes) - if err != nil { - return nil, nil, fmt.Errorf(`failed to parse PKCS8 private key: %w`, err) - } - return key, rest, nil - case "CERTIFICATE": - cert, err := x509.ParseCertificate(block.Bytes) - if err != nil { - return nil, nil, fmt.Errorf(`failed to parse certificate: %w`, err) - } - return cert.PublicKey, rest, nil - default: - return nil, nil, fmt.Errorf(`invalid PEM block type %s`, block.Type) - } -} - // ParseRawKey is a combination of ParseKey and Raw. It parses a single JWK key, // and assigns the "raw" key to the given parameter. The key must either be // a pointer to an empty interface, or a pointer to the actual raw key type @@ -371,7 +203,7 @@ func ParseRawKey(data []byte, rawkey interface{}) error { return fmt.Errorf(`failed to parse key: %w`, err) } - if err := key.Raw(rawkey); err != nil { + if err := Raw(key, rawkey); err != nil { return fmt.Errorf(`failed to assign to raw key variable: %w`, err) } @@ -634,71 +466,6 @@ func cloneKey(src Key) (Key, error) { return dst, nil } -// Pem serializes the given jwk.Key in PEM encoded ASN.1 DER format, -// using either PKCS8 for private keys and PKIX for public keys. -// If you need to encode using PKCS1 or SEC1, you must do it yourself. -// -// # Argument must be of type jwk.Key or jwk.Set -// -// Currently only EC (including Ed25519) and RSA keys (and jwk.Set -// comprised of these key types) are supported. -func Pem(v interface{}) ([]byte, error) { - var set Set - switch v := v.(type) { - case Key: - set = NewSet() - if err := set.AddKey(v); err != nil { - return nil, fmt.Errorf(`failed to add key to set: %w`, err) - } - case Set: - set = v - default: - return nil, fmt.Errorf(`argument to Pem must be either jwk.Key or jwk.Set: %T`, v) - } - - var ret []byte - for i := 0; i < set.Len(); i++ { - key, _ := set.Key(i) - typ, buf, err := asnEncode(key) - if err != nil { - return nil, fmt.Errorf(`failed to encode content for key #%d: %w`, i, err) - } - - var block pem.Block - block.Type = typ - block.Bytes = buf - ret = append(ret, pem.EncodeToMemory(&block)...) - } - return ret, nil -} - -func asnEncode(key Key) (string, []byte, error) { - switch key := key.(type) { - case RSAPrivateKey, ECDSAPrivateKey, OKPPrivateKey: - var rawkey interface{} - if err := key.Raw(&rawkey); err != nil { - return "", nil, fmt.Errorf(`failed to get raw key from jwk.Key: %w`, err) - } - buf, err := x509.MarshalPKCS8PrivateKey(rawkey) - if err != nil { - return "", nil, fmt.Errorf(`failed to marshal PKCS8: %w`, err) - } - return pmPrivateKey, buf, nil - case RSAPublicKey, ECDSAPublicKey, OKPPublicKey: - var rawkey interface{} - if err := key.Raw(&rawkey); err != nil { - return "", nil, fmt.Errorf(`failed to get raw key from jwk.Key: %w`, err) - } - buf, err := x509.MarshalPKIXPublicKey(rawkey) - if err != nil { - return "", nil, fmt.Errorf(`failed to marshal PKIX: %w`, err) - } - return pmPublicKey, buf, nil - default: - return "", nil, fmt.Errorf(`unsupported key type %T`, key) - } -} - // RegisterCustomField allows users to specify that a private field // be decoded as an instance of the specified type. This option has // a global effect. diff --git a/jwk/jwk_test.go b/jwk/jwk_test.go index af7dfd4d4..7d9f8efb2 100644 --- a/jwk/jwk_test.go +++ b/jwk/jwk_test.go @@ -279,7 +279,7 @@ func VerifyKey(t *testing.T, def map[string]keyDef) { typ := expectedRawKeyType(key) var rawkey interface{} - if !assert.NoError(t, key.Raw(&rawkey), `Raw() should succeed`) { + if !assert.NoError(t, jwk.Raw(key, &rawkey), `Raw() should succeed`) { return } if !assert.IsType(t, rawkey, typ, `raw key should be of this type`) { @@ -391,7 +391,7 @@ func TestParse(t *testing.T) { t.Helper() var irawkey interface{} - if !assert.NoError(t, key.Raw(&irawkey), `key.Raw(&interface) should ucceed`) { + if !assert.NoError(t, jwk.Raw(key, &irawkey), `jwk.Raw(key,&interface) should ucceed`) { return } @@ -399,19 +399,19 @@ func TestParse(t *testing.T) { switch k := key.(type) { case jwk.RSAPrivateKey: var rawkey rsa.PrivateKey - if !assert.NoError(t, key.Raw(&rawkey), `key.Raw(&rsa.PrivateKey) should succeed`) { + if !assert.NoError(t, jwk.Raw(key, &rawkey), `jwk.Raw(key,&rsa.PrivateKey) should succeed`) { return } crawkey = &rawkey case jwk.RSAPublicKey: var rawkey rsa.PublicKey - if !assert.NoError(t, key.Raw(&rawkey), `key.Raw(&rsa.PublicKey) should succeed`) { + if !assert.NoError(t, jwk.Raw(key, &rawkey), `jwk.Raw(key,&rsa.PublicKey) should succeed`) { return } crawkey = &rawkey case jwk.ECDSAPrivateKey: var rawkey ecdsa.PrivateKey - if !assert.NoError(t, key.Raw(&rawkey), `key.Raw(&ecdsa.PrivateKey) should succeed`) { + if !assert.NoError(t, jwk.Raw(key, &rawkey), `jwk.Raw(key,&ecdsa.PrivateKey) should succeed`) { return } crawkey = &rawkey @@ -419,13 +419,13 @@ func TestParse(t *testing.T) { switch k.Crv() { case jwa.Ed25519: var rawkey ed25519.PrivateKey - if !assert.NoError(t, key.Raw(&rawkey), `key.Raw(&ed25519.PrivateKey) should succeed`) { + if !assert.NoError(t, jwk.Raw(key, &rawkey), `jwk.Raw(key,&ed25519.PrivateKey) should succeed`) { return } crawkey = rawkey case jwa.X25519: var rawkey x25519.PrivateKey - if !assert.NoError(t, key.Raw(&rawkey), `key.Raw(&x25519.PrivateKey) should succeed`) { + if !assert.NoError(t, jwk.Raw(key, &rawkey), `jwk.Raw(key,&x25519.PrivateKey) should succeed`) { return } crawkey = rawkey @@ -439,13 +439,13 @@ func TestParse(t *testing.T) { switch k.Crv() { case jwa.Ed25519: var rawkey ed25519.PublicKey - if !assert.NoError(t, key.Raw(&rawkey), `key.Raw(&ed25519.PublicKey) should succeed`) { + if !assert.NoError(t, jwk.Raw(key, &rawkey), `jwk.Raw(key,&ed25519.PublicKey) should succeed`) { return } crawkey = rawkey case jwa.X25519: var rawkey x25519.PublicKey - if !assert.NoError(t, key.Raw(&rawkey), `key.Raw(&x25519.PublicKey) should succeed`) { + if !assert.NoError(t, jwk.Raw(key, &rawkey), `jwk.Raw(key,&x25519.PublicKey) should succeed`) { return } crawkey = rawkey @@ -934,7 +934,7 @@ func TestPublicKeyOf(t *testing.T) { // Get the raw key to compare var rawKey interface{} - if !assert.NoError(t, pubJwkKey.Raw(&rawKey), `pubJwkKey.Raw should succeed`) { + if !assert.NoError(t, jwk.Raw(pubJwkKey, &rawKey), `pubJwkKey.Raw should succeed`) { return } @@ -987,7 +987,7 @@ func TestPublicKeyOf(t *testing.T) { // Get the raw key to compare var rawKey interface{} - if !assert.NoError(t, setKey.Raw(&rawKey), `pubJwkKey.Raw should succeed`) { + if !assert.NoError(t, jwk.Raw(setKey, &rawKey), `pubJwkKey.Raw should succeed`) { return } @@ -1453,7 +1453,7 @@ c4wOvhbalcX0FqTM3mXCgMFRbibquhwdxbU= } var pubkey rsa.PublicKey - if !assert.NoError(t, key.Raw(&pubkey), `key.Raw should succeed`) { + if !assert.NoError(t, jwk.Raw(key, &pubkey), `key.Raw should succeed`) { return } @@ -2183,5 +2183,5 @@ func TestGH947(t *testing.T) { k, err := jwk.ParseKey(raw) require.NoError(t, err, `jwk.ParseKey should succeed`) var exported []byte - require.Error(t, k.Raw(&exported), `(okpkey).Raw with 0-length OKP key should fail`) + require.Error(t, jwk.Raw(k, &exported), `(okpkey).Raw with 0-length OKP key should fail`) } diff --git a/jwk/pem.go b/jwk/pem.go new file mode 100644 index 000000000..0b8df6349 --- /dev/null +++ b/jwk/pem.go @@ -0,0 +1,294 @@ +package jwk + +import ( + "crypto/x509" + "encoding/pem" + "fmt" + "sync" +) + +const ( + pmPrivateKey = `PRIVATE KEY` + pmPublicKey = `PUBLIC KEY` + pmECPrivateKey = `EC PRIVATE KEY` + pmRSAPublicKey = `RSA PUBLIC KEY` + pmRSAPrivateKey = `RSA PRIVATE KEY` +) + +// ASN1Decoder decodes a given byte sequence into a key. +type ASN1Decoder interface { + ASN1Decode([]byte) (interface{}, []byte, error) +} + +type ChainedASN1Decoder interface { + Next(ASN1Decoder, []byte) (interface{}, []byte, error) +} + +type ChainedASN1DecodeFunc func(ASN1Decoder, []byte) (interface{}, []byte, error) + +func (fn ChainedASN1DecodeFunc) Next(n ASN1Decoder, src []byte) (interface{}, []byte, error) { + return fn(n, src) +} + +type chainedASN1Decoder struct { + mu sync.RWMutex + list []ChainedASN1Decoder +} + +func (c *chainedASN1Decoder) Add(d ChainedASN1Decoder) { + c.mu.Lock() + defer c.mu.Unlock() + c.list = append(c.list, d) +} + +func (c *chainedASN1Decoder) Next(src []byte) (interface{}, []byte, error) { + c.mu.RLock() + llist := len(c.list) + c.mu.RUnlock() + st := &chainedASN1DecoderCallState{parent: c, current: llist} + return st.ASN1Decode(src) +} + +type chainedASN1DecoderCallState struct { + current int + parent *chainedASN1Decoder +} + +func (st *chainedASN1DecoderCallState) ASN1Decode(src []byte) (interface{}, []byte, error) { + idx := st.current - 1 + + st.parent.mu.RLock() + defer st.parent.mu.RUnlock() + + llist := len(st.parent.list) + if idx < 0 || idx >= llist { + return nil, nil, fmt.Errorf(`failed to decode PEM data`) + } + + st.current = idx + + d := st.parent.list[idx] + return d.Next(st, src) +} + +type NextASN1Decoder func([]byte) (interface{}, []byte, error) + +func AddASN1Decoder(dec ChainedASN1Decoder) { + chainedASN1D.Add(dec) +} + +var chainedASN1D = &chainedASN1Decoder{ + list: []ChainedASN1Decoder{ChainedASN1DecodeFunc(asn1Decode)}, +} + +type chainedASN1Encoder struct { + mu sync.RWMutex + list []ASN1Encoder +} + +func (c *chainedASN1Encoder) Add(e ASN1Encoder) { + c.mu.Lock() + defer c.mu.Unlock() + c.list = append(c.list, e) +} + +func (c *chainedASN1Encoder) Next(key Key) (string, []byte, error) { + c.mu.RLock() + llist := len(c.list) + c.mu.RUnlock() + st := &chainedASN1EncoderCallState{parent: c, current: llist} + return st.Next(key) +} + +type chainedASN1EncoderCallState struct { + current int + parent *chainedASN1Encoder +} + +func (st *chainedASN1EncoderCallState) Next(key Key) (string, []byte, error) { + idx := st.current - 1 + + st.parent.mu.RLock() + defer st.parent.mu.RUnlock() + + llist := len(st.parent.list) + if idx < 0 || idx >= llist { + return "", nil, fmt.Errorf(`failed to encode to jwk.Key %T to PEM`, key) + } + + st.current = idx + + e := st.parent.list[idx] + return e.ASN1Encode(st.Next, key) +} + +type NextASN1Encoder func(Key) (string, []byte, error) + +// ASN1Encoder encodes a given key into ASN.1 format, so that it can be +// further encoded in PEM format. +type ASN1Encoder interface { + // ASN1Encode takes a key, and returns three elements. The first string + // is the name to be used when encoded in PEM format, the second + // is the actual byte sequence encoded in ASN.1 format, and the + // third is an error, if any. + ASN1Encode(NextASN1Encoder, Key) (string, []byte, error) +} +type ASN1EncodeFunc func(NextASN1Encoder, Key) (string, []byte, error) + +func (fn ASN1EncodeFunc) ASN1Encode(n NextASN1Encoder, key Key) (string, []byte, error) { + return fn(n, key) +} + +var chainedASN1E = &chainedASN1Encoder{ + list: []ASN1Encoder{ASN1EncodeFunc(asn1Encode)}, +} + +// AddASN1Encoder allows users +func AddASN1Encoder(enc ASN1Encoder) { + chainedASN1E.Add(enc) +} + +// Encodes a Key in DER ASN.1 format. Can handle RSA, EC, OKP keys. +func EncodeASN1(key Key) (string, []byte, error) { + return chainedASN1E.Next(key) +} + +func asn1Encode(_ NextASN1Encoder, key Key) (string, []byte, error) { + switch key := key.(type) { + case RSAPrivateKey, ECDSAPrivateKey, OKPPrivateKey: + var rawkey interface{} + if err := Raw(key, &rawkey); err != nil { + return "", nil, fmt.Errorf(`failed to get raw key from jwk.Key: %w`, err) + } + buf, err := x509.MarshalPKCS8PrivateKey(rawkey) + if err != nil { + return "", nil, fmt.Errorf(`failed to marshal PKCS8: %w`, err) + } + return pmPrivateKey, buf, nil + case RSAPublicKey, ECDSAPublicKey, OKPPublicKey: + var rawkey interface{} + if err := Raw(key, &rawkey); err != nil { + return "", nil, fmt.Errorf(`failed to get raw key from jwk.Key: %w`, err) + } + buf, err := x509.MarshalPKIXPublicKey(rawkey) + if err != nil { + return "", nil, fmt.Errorf(`failed to marshal PKIX: %w`, err) + } + return pmPublicKey, buf, nil + default: + return "", nil, fmt.Errorf(`encoding key to ASN.1 failed: unsupported key type %T`, key) + } +} + +// Pem serializes the given jwk.Key in PEM encoded ASN.1 DER format, +// using either PKCS8 for private keys and PKIX for public keys. +// If you need to encode using PKCS1 or SEC1, you must do it yourself. +// +// # Argument must be of type jwk.Key or jwk.Set +// +// Currently only EC (including Ed25519) and RSA keys (and jwk.Set +// comprised of these key types) are supported. +func Pem(v interface{}) ([]byte, error) { + var set Set + switch v := v.(type) { + case Key: + set = NewSet() + if err := set.AddKey(v); err != nil { + return nil, fmt.Errorf(`failed to add key to set: %w`, err) + } + case Set: + set = v + default: + return nil, fmt.Errorf(`argument to Pem must be either jwk.Key or jwk.Set: %T`, v) + } + + var ret []byte + for i := 0; i < set.Len(); i++ { + key, _ := set.Key(i) + typ, buf, err := chainedASN1E.Next(key) + if err != nil { + return nil, fmt.Errorf(`failed to encode content for key #%d: %w`, i, err) + } + + var block pem.Block + block.Type = typ + block.Bytes = buf + ret = append(ret, pem.EncodeToMemory(&block)...) + } + return ret, nil +} + +// DecodePEM decodes a key in PEM encoded ASN.1 DER format. +// and returns a raw key +func DecodePEM(src []byte) (interface{}, []byte, error) { + return chainedASN1D.Next(src) +} + +func asn1Decode(_ ASN1Decoder, src []byte) (interface{}, []byte, error) { + block, rest := pem.Decode(src) + if block == nil { + return nil, nil, fmt.Errorf(`failed to decode PEM data`) + } + + switch block.Type { + // Handle the semi-obvious cases + case pmRSAPrivateKey: + key, err := x509.ParsePKCS1PrivateKey(block.Bytes) + if err != nil { + return nil, nil, fmt.Errorf(`failed to parse PKCS1 private key: %w`, err) + } + return key, rest, nil + case pmRSAPublicKey: + key, err := x509.ParsePKCS1PublicKey(block.Bytes) + if err != nil { + return nil, nil, fmt.Errorf(`failed to parse PKCS1 public key: %w`, err) + } + return key, rest, nil + case pmECPrivateKey: + key, err := x509.ParseECPrivateKey(block.Bytes) + if err != nil { + return nil, nil, fmt.Errorf(`failed to parse EC private key: %w`, err) + } + return key, rest, nil + case pmPublicKey: + // XXX *could* return dsa.PublicKey + key, err := x509.ParsePKIXPublicKey(block.Bytes) + if err != nil { + return nil, nil, fmt.Errorf(`failed to parse PKIX public key: %w`, err) + } + return key, rest, nil + case pmPrivateKey: + key, err := x509.ParsePKCS8PrivateKey(block.Bytes) + if err != nil { + return nil, nil, fmt.Errorf(`failed to parse PKCS8 private key: %w`, err) + } + return key, rest, nil + case "CERTIFICATE": + cert, err := x509.ParseCertificate(block.Bytes) + if err != nil { + return nil, nil, fmt.Errorf(`failed to parse certificate: %w`, err) + } + return cert.PublicKey, rest, nil + default: + return nil, nil, fmt.Errorf(`invalid PEM block type %s`, block.Type) + } +} + +// EncodePEM encodes the key into a PEM encoded ASN.1 DER format. +// The key can be a jwk.Key or a raw key instance, but it must be one of +// the types supported by `x509` package. +// +// Internally, it uses the same routine as `jwk.EncodeX509()`, and therefore +// the same caveats apply +func EncodePEM(v interface{}) ([]byte, error) { + typ, marshaled, err := EncodeX509(v) + if err != nil { + return nil, fmt.Errorf(`failed to encode key in x509: %w`, err) + } + + block := &pem.Block{ + Type: typ, + Bytes: marshaled, + } + return pem.EncodeToMemory(block), nil +} diff --git a/jwk/rsa.go b/jwk/rsa.go index 5de6b6358..d37b0fcd6 100644 --- a/jwk/rsa.go +++ b/jwk/rsa.go @@ -129,7 +129,7 @@ func (k *rsaPrivateKey) Raw(v interface{}) error { pubk := newRSAPublicKey() pubk.n = k.n pubk.e = k.e - if err := pubk.Raw(&key.PublicKey); err != nil { + if err := Raw(pubk, &key.PublicKey); err != nil { return fmt.Errorf(`failed to materialize RSA public key: %w`, err) } @@ -208,7 +208,7 @@ func (k rsaPrivateKey) Thumbprint(hash crypto.Hash) ([]byte, error) { defer k.mu.RUnlock() var key rsa.PrivateKey - if err := k.Raw(&key); err != nil { + if err := Raw(&k, &key); err != nil { return nil, fmt.Errorf(`failed to materialize RSA private key: %w`, err) } return rsaThumbprint(hash, &key.PublicKey) @@ -219,7 +219,7 @@ func (k rsaPublicKey) Thumbprint(hash crypto.Hash) ([]byte, error) { defer k.mu.RUnlock() var key rsa.PublicKey - if err := k.Raw(&key); err != nil { + if err := Raw(&k, &key); err != nil { return nil, fmt.Errorf(`failed to materialize RSA public key: %w`, err) } return rsaThumbprint(hash, &key) diff --git a/jwk/symmetric.go b/jwk/symmetric.go index d2498e334..9f86403c2 100644 --- a/jwk/symmetric.go +++ b/jwk/symmetric.go @@ -35,7 +35,7 @@ func (k *symmetricKey) Thumbprint(hash crypto.Hash) ([]byte, error) { k.mu.RLock() defer k.mu.RUnlock() var octets []byte - if err := k.Raw(&octets); err != nil { + if err := Raw(k, &octets); err != nil { return nil, fmt.Errorf(`failed to materialize symmetric key: %w`, err) } diff --git a/jws/jws_test.go b/jws/jws_test.go index 53d1803b1..763095ece 100644 --- a/jws/jws_test.go +++ b/jws/jws_test.go @@ -511,7 +511,7 @@ func TestEncode(t *testing.T) { t.Fatal("Failed to parse JWK") } var key interface{} - if !assert.NoError(t, jwkKey.Raw(&key), `jwk.Raw should succeed`) { + if !assert.NoError(t, jwk.Raw(jwkKey, &key), `jwk.Raw should succeed`) { return } var jwsCompact []byte @@ -583,7 +583,7 @@ func TestEncode(t *testing.T) { } var rawkey rsa.PrivateKey - if !assert.NoError(t, privkey.Raw(&rawkey), `obtaining raw key should succeed`) { + if !assert.NoError(t, jwk.Raw(privkey, &rawkey), `obtaining raw key should succeed`) { return } @@ -660,7 +660,7 @@ func TestEncode(t *testing.T) { } var rawkey ecdsa.PrivateKey - if !assert.NoError(t, privkey.Raw(&rawkey), `obtaining raw key should succeed`) { + if !assert.NoError(t, jwk.Raw(privkey, &rawkey), `obtaining raw key should succeed`) { return } @@ -745,7 +745,7 @@ func TestEncode(t *testing.T) { } var rawkey ed25519.PrivateKey - if !assert.NoError(t, privkey.Raw(&rawkey), `obtaining raw key should succeed`) { + if !assert.NoError(t, jwk.Raw(privkey, &rawkey), `obtaining raw key should succeed`) { return } @@ -1024,7 +1024,7 @@ func TestDecode_ES384Compact_NoSigTrim(t *testing.T) { } var rawkey ecdsa.PublicKey - if !assert.NoError(t, pubkey.Raw(&rawkey), `obtaining raw key should succeed`) { + if !assert.NoError(t, jwk.Raw(pubkey, &rawkey), `obtaining raw key should succeed`) { return } diff --git a/jwx_test.go b/jwx_test.go index b74243404..5111e49ab 100644 --- a/jwx_test.go +++ b/jwx_test.go @@ -167,7 +167,7 @@ func TestJoseCompatibility(t *testing.T) { } } - if !assert.NoError(t, webkey.Raw(&tc.Raw), `jwk.Raw should succeed`) { + if !assert.NoError(t, jwk.Raw(webkey, &tc.Raw), `jwk.Raw should succeed`) { return } }) @@ -298,17 +298,17 @@ func joseInteropTest(ctx context.Context, spec interopTest, t *testing.T) { switch spec.alg { case jwa.RSA1_5, jwa.RSA_OAEP, jwa.RSA_OAEP_256: var rawkey rsa.PrivateKey - if !assert.NoError(t, jwxJwk.Raw(&rawkey), `jwk.Raw should succeed`) { + if !assert.NoError(t, jwk.Raw(jwxJwk, &rawkey), `jwk.Raw should succeed`) { return } case jwa.ECDH_ES, jwa.ECDH_ES_A128KW, jwa.ECDH_ES_A192KW, jwa.ECDH_ES_A256KW: var rawkey ecdsa.PrivateKey - if !assert.NoError(t, jwxJwk.Raw(&rawkey), `jwk.Raw should succeed`) { + if !assert.NoError(t, jwk.Raw(jwxJwk, &rawkey), `jwk.Raw should succeed`) { return } default: var rawkey []byte - if !assert.NoError(t, jwxJwk.Raw(&rawkey), `jwk.Raw should succeed`) { + if !assert.NoError(t, jwk.Raw(jwxJwk, &rawkey), `jwk.Raw should succeed`) { return } } From d966b13ce8e7c4ea78bd87788f8fab59a4e82b07 Mon Sep 17 00:00:00 2001 From: Daisuke Maki Date: Thu, 24 Aug 2023 12:02:25 +0900 Subject: [PATCH 2/6] Update go.mod --- examples/go.mod | 4 +++- examples/go.sum | 13 ++++++++----- 2 files changed, 11 insertions(+), 6 deletions(-) diff --git a/examples/go.mod b/examples/go.mod index 13c5e40b0..867076a18 100644 --- a/examples/go.mod +++ b/examples/go.mod @@ -4,7 +4,9 @@ go 1.16 require ( github.com/cloudflare/circl v1.3.3 - github.com/lestrrat-go/jwx/v2 v2.0.11 + github.com/lestrrat-go/jwx/v2 v2.0.12-0.20230824024517-a077c65f16eb ) replace github.com/cloudflare/circl v1.0.0 => github.com/cloudflare/circl v1.0.1-0.20210104183656-96a0695de3c3 + +replace github.com/lestrrat-go/jwx/v2 v2.0.11 => github.com/lestrrat-go/jwx/v2 v2.0.12-0.20230824024517-a077c65f16eb diff --git a/examples/go.sum b/examples/go.sum index 38291427b..480afbf86 100644 --- a/examples/go.sum +++ b/examples/go.sum @@ -17,8 +17,8 @@ github.com/lestrrat-go/httprc v1.0.4 h1:bAZymwoZQb+Oq8MEbyipag7iSq6YIga8Wj6GOiJG github.com/lestrrat-go/httprc v1.0.4/go.mod h1:mwwz3JMTPBjHUkkDv/IGJ39aALInZLrhBp0X7KGUZlo= github.com/lestrrat-go/iter v1.0.2 h1:gMXo1q4c2pHmC3dn8LzRhJfP1ceCbgSiT9lUydIzltI= github.com/lestrrat-go/iter v1.0.2/go.mod h1:Momfcq3AnRlRjI5b5O8/G5/BvpzrhoFTZcn06fEOPt4= -github.com/lestrrat-go/jwx/v2 v2.0.11 h1:ViHMnaMeaO0qV16RZWBHM7GTrAnX2aFLVKofc7FuKLQ= -github.com/lestrrat-go/jwx/v2 v2.0.11/go.mod h1:ZtPtMFlrfDrH2Y0iwfa3dRFn8VzwBrB+cyrm3IBWdDg= +github.com/lestrrat-go/jwx/v2 v2.0.12-0.20230824024517-a077c65f16eb h1:qPUmVTD6gWn0S8zfmAzjgzF5xdYtJrGhroN+i7u/TrE= +github.com/lestrrat-go/jwx/v2 v2.0.12-0.20230824024517-a077c65f16eb/go.mod h1:Mq4KN1mM7bp+5z/W5HS8aCNs5RKZ911G/0y2qUjAQuQ= github.com/lestrrat-go/option v1.0.0/go.mod h1:5ZHFbivi4xwXxhxY9XHDe2FHo6/Z7WWmtT7T5nBBp3I= github.com/lestrrat-go/option v1.0.1 h1:oAzP2fvZGQKWkvHa1/SAcFolBEca1oN+mQ7eooNBEYU= github.com/lestrrat-go/option v1.0.1/go.mod h1:5ZHFbivi4xwXxhxY9XHDe2FHo6/Z7WWmtT7T5nBBp3I= @@ -38,8 +38,8 @@ github.com/yuin/goldmark v1.4.13/go.mod h1:6yULJ656Px+3vBD8DxQVa3kxgyrAnzto9xy5t golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= golang.org/x/crypto v0.3.1-0.20221117191849-2c476679df9a/go.mod h1:hebNnKkNXi2UzZN1eVRvBB7co0a+JxK6XbPiWVs/3J4= -golang.org/x/crypto v0.9.0 h1:LF6fAI+IutBocDJ2OT0Q1g8plpYljMZ4+lty+dsqw3g= -golang.org/x/crypto v0.9.0/go.mod h1:yrmDGqONDYtNj3tH8X9dzUun2m2lzPa9ngI6/RUPGR0= +golang.org/x/crypto v0.12.0 h1:tFM/ta59kqch6LlvYnPa0yx5a83cL2nHflFhYKvv9Yk= +golang.org/x/crypto v0.12.0/go.mod h1:NF0Gs7EO5K4qLn+Ylc+fih8BSTeIjAP05siRnAh98yw= golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4= golang.org/x/mod v0.8.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs= golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= @@ -60,19 +60,22 @@ golang.org/x/sys v0.0.0-20220722155257-8c9f86f7a55f/go.mod h1:oPkhp1MJrh7nUepCBc golang.org/x/sys v0.2.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.3.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.8.0 h1:EBmGv8NaZBZTWvrbjNoL6HVt+IVy3QDQpJs7VRIw3tU= golang.org/x/sys v0.8.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.11.0 h1:eG7RXZHdqOJ1i+0lgLgCpSXAp6M3LYlAo6osgSi0xOM= +golang.org/x/sys v0.11.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= golang.org/x/term v0.2.0/go.mod h1:TVmDHMZPmdnySmBfhjOoOdhjzdE1h4u1VwSiw2l1Nuc= golang.org/x/term v0.5.0/go.mod h1:jMB1sMXY+tzblOD4FWmEbocvup2/aLOaQEp7JmGp78k= golang.org/x/term v0.8.0/go.mod h1:xPskH00ivmX89bAKVGSKKtLOWNx2+17Eiy94tnKShWo= +golang.org/x/term v0.11.0/go.mod h1:zC9APTIj3jG3FdV/Ons+XE1riIZXG4aZ4GTHiPZJPIU= golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ= golang.org/x/text v0.4.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8= golang.org/x/text v0.7.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8= golang.org/x/text v0.9.0/go.mod h1:e1OnstbJyHTd6l/uOt8jFFHp6TRDWZR/bV3emEE/zU8= +golang.org/x/text v0.12.0/go.mod h1:TvPlkZtksWOMsz7fbANvkp4WM8x/WCo/om8BMLbz+aE= golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc= From f733c9a4fd83135627da6f5e2772774c133f69d5 Mon Sep 17 00:00:00 2001 From: Daisuke Maki Date: Thu, 24 Aug 2023 13:07:57 +0900 Subject: [PATCH 3/6] Fix BUILD.bazel --- jwk/BUILD.bazel | 2 ++ 1 file changed, 2 insertions(+) diff --git a/jwk/BUILD.bazel b/jwk/BUILD.bazel index a61a919f5..0bc749805 100644 --- a/jwk/BUILD.bazel +++ b/jwk/BUILD.bazel @@ -4,6 +4,7 @@ go_library( name = "jwk", srcs = [ "cache.go", + "convert.go", "ecdsa.go", "ecdsa_gen.go", "fetch.go", @@ -16,6 +17,7 @@ go_library( "okp_gen.go", "options.go", "options_gen.go", + "pem.go", "rsa.go", "rsa_gen.go", "set.go", From e704b4f5506c757fef9d96ec2a1770bb4f114666 Mon Sep 17 00:00:00 2001 From: Daisuke Maki Date: Thu, 24 Aug 2023 19:04:48 +0900 Subject: [PATCH 4/6] tweak --- jwk/es256k.go | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/jwk/es256k.go b/jwk/es256k.go index 81ba314a6..4972ab11c 100644 --- a/jwk/es256k.go +++ b/jwk/es256k.go @@ -189,6 +189,10 @@ func secp256k1Raw(nextRFK RawFromKeyer, key Key, raw interface{}) error { if err := key.Raw(&ecdsaKey); err != nil { return fmt.Errorf(`failed to convert JWK into raw ecdsa.PrivateKey: %w`, err) } + // Make sure the curve is secp256k1 + if ecdsaKey.Curve.Params().Name != secp256k1.S256().Params().Name { + return fmt.Errorf(`invalid curve for secp256k1: %s`, ecdsaKey.Curve.Params().Name) + } return blackmagic.AssignIfCompatible(raw, secp256k1.PrivKeyFromBytes(ecdsaKey.D.Bytes())) case *secp256k1.PublicKey: // we first get a ecdsa.PublicKey, then convert it to secp256k1.PublicKey @@ -196,6 +200,10 @@ func secp256k1Raw(nextRFK RawFromKeyer, key Key, raw interface{}) error { if err := key.Raw(&ecdsaKey); err != nil { return fmt.Errorf(`failed to convert JWK into raw ecdsa.PublicKey: %w`, err) } + // Make sure the curve is secp256k1 + if ecdsaKey.Curve.Params().Name != secp256k1.S256().Params().Name { + return fmt.Errorf(`invalid curve for secp256k1: %s`, ecdsaKey.Curve.Params().Name) + } var x secp256k1.FieldVal var y secp256k1.FieldVal x.SetByteSlice(ecdsaKey.X.Bytes()) From 7ec8e06f5652f889668c8fa0f89367563127018d Mon Sep 17 00:00:00 2001 From: Daisuke Maki Date: Fri, 25 Aug 2023 05:51:29 +0900 Subject: [PATCH 5/6] Fix naming --- jwk/es256k.go | 6 +++--- jwk/jwk.go | 14 ++++++++++++++ jwk/pem.go | 49 ++++++++++++++++++++++--------------------------- 3 files changed, 39 insertions(+), 30 deletions(-) diff --git a/jwk/es256k.go b/jwk/es256k.go index 4972ab11c..b87354fb6 100644 --- a/jwk/es256k.go +++ b/jwk/es256k.go @@ -23,7 +23,7 @@ func init() { AddKeyFromRaw(ChainedKeyFromRawFunc(secp256k1FromRaw)) AddRawFromKey(ChainedRawFromKeyFunc(secp256k1Raw)) - AddASN1Encoder(ASN1EncodeFunc(secp256k1ASN1Encode)) + AddASN1Encoder(ChainedASN1EncodeFunc(secp256k1ASN1Encode)) AddASN1Decoder(ChainedASN1DecodeFunc(secp256k1ASN1Decode)) } @@ -65,7 +65,7 @@ type secp256k1ASN1PublicKey struct { BitString asn1.BitString } -func secp256k1ASN1Encode(n NextASN1Encoder, key Key) (string, []byte, error) { +func secp256k1ASN1Encode(n ASN1Encoder, key Key) (string, []byte, error) { switch key := key.(type) { case ECDSAPrivateKey: if key.Crv() == jwa.Secp256k1 { @@ -85,7 +85,7 @@ func secp256k1ASN1Encode(n NextASN1Encoder, key Key) (string, []byte, error) { } } - return n(key) + return n.ASN1Encode(key) } func secp256k1ASN1Decode(n ASN1Decoder, buf []byte) (interface{}, []byte, error) { diff --git a/jwk/jwk.go b/jwk/jwk.go index d511c843c..8bb2910c7 100644 --- a/jwk/jwk.go +++ b/jwk/jwk.go @@ -494,3 +494,17 @@ func AvailableCurves() []elliptic.Curve { func CurveForAlgorithm(alg jwa.EllipticCurveAlgorithm) (elliptic.Curve, bool) { return ecutil.CurveForAlgorithm(alg) } + +// KeySpec is a specification for additional key types +// to be added to the jwk system. +// +// This mechanism should be considered experimental and subject +// to change, even between micro versions. If you are adding a +// new key type, please be ready to update your code when +// a new version of this library is released. +type KeySpec struct { + Curve elliptic.Curve + Algorithm jwa.EllipticCurveAlgorithm + RawFromKey ChainedRawFromKeyer + KeyFromRaw ChainedKeyFromRawer +} diff --git a/jwk/pem.go b/jwk/pem.go index 0b8df6349..9637217a5 100644 --- a/jwk/pem.go +++ b/jwk/pem.go @@ -71,8 +71,6 @@ func (st *chainedASN1DecoderCallState) ASN1Decode(src []byte) (interface{}, []by return d.Next(st, src) } -type NextASN1Decoder func([]byte) (interface{}, []byte, error) - func AddASN1Decoder(dec ChainedASN1Decoder) { chainedASN1D.Add(dec) } @@ -81,12 +79,26 @@ var chainedASN1D = &chainedASN1Decoder{ list: []ChainedASN1Decoder{ChainedASN1DecodeFunc(asn1Decode)}, } +type ASN1Encoder interface { + ASN1Encode(Key) (string, []byte, error) +} + +type ChainedASN1Encoder interface { + Next(ASN1Encoder, Key) (string, []byte, error) +} + +type ChainedASN1EncodeFunc func(ASN1Encoder, Key) (string, []byte, error) + +func (fn ChainedASN1EncodeFunc) Next(n ASN1Encoder, key Key) (string, []byte, error) { + return fn(n, key) +} + type chainedASN1Encoder struct { mu sync.RWMutex - list []ASN1Encoder + list []ChainedASN1Encoder } -func (c *chainedASN1Encoder) Add(e ASN1Encoder) { +func (c *chainedASN1Encoder) Add(e ChainedASN1Encoder) { c.mu.Lock() defer c.mu.Unlock() c.list = append(c.list, e) @@ -97,7 +109,7 @@ func (c *chainedASN1Encoder) Next(key Key) (string, []byte, error) { llist := len(c.list) c.mu.RUnlock() st := &chainedASN1EncoderCallState{parent: c, current: llist} - return st.Next(key) + return st.ASN1Encode(key) } type chainedASN1EncoderCallState struct { @@ -105,7 +117,7 @@ type chainedASN1EncoderCallState struct { parent *chainedASN1Encoder } -func (st *chainedASN1EncoderCallState) Next(key Key) (string, []byte, error) { +func (st *chainedASN1EncoderCallState) ASN1Encode(key Key) (string, []byte, error) { idx := st.current - 1 st.parent.mu.RLock() @@ -119,32 +131,15 @@ func (st *chainedASN1EncoderCallState) Next(key Key) (string, []byte, error) { st.current = idx e := st.parent.list[idx] - return e.ASN1Encode(st.Next, key) -} - -type NextASN1Encoder func(Key) (string, []byte, error) - -// ASN1Encoder encodes a given key into ASN.1 format, so that it can be -// further encoded in PEM format. -type ASN1Encoder interface { - // ASN1Encode takes a key, and returns three elements. The first string - // is the name to be used when encoded in PEM format, the second - // is the actual byte sequence encoded in ASN.1 format, and the - // third is an error, if any. - ASN1Encode(NextASN1Encoder, Key) (string, []byte, error) -} -type ASN1EncodeFunc func(NextASN1Encoder, Key) (string, []byte, error) - -func (fn ASN1EncodeFunc) ASN1Encode(n NextASN1Encoder, key Key) (string, []byte, error) { - return fn(n, key) + return e.Next(st, key) } var chainedASN1E = &chainedASN1Encoder{ - list: []ASN1Encoder{ASN1EncodeFunc(asn1Encode)}, + list: []ChainedASN1Encoder{ChainedASN1EncodeFunc(asn1Encode)}, } // AddASN1Encoder allows users -func AddASN1Encoder(enc ASN1Encoder) { +func AddASN1Encoder(enc ChainedASN1Encoder) { chainedASN1E.Add(enc) } @@ -153,7 +148,7 @@ func EncodeASN1(key Key) (string, []byte, error) { return chainedASN1E.Next(key) } -func asn1Encode(_ NextASN1Encoder, key Key) (string, []byte, error) { +func asn1Encode(_ ASN1Encoder, key Key) (string, []byte, error) { switch key := key.(type) { case RSAPrivateKey, ECDSAPrivateKey, OKPPrivateKey: var rawkey interface{} From 92bd45d29f6df78b907296c30e0463f742f9bf92 Mon Sep 17 00:00:00 2001 From: Daisuke Maki Date: Fri, 25 Aug 2023 05:59:09 +0900 Subject: [PATCH 6/6] tweak doc --- jwk/convert.go | 8 +++++++- 1 file changed, 7 insertions(+), 1 deletion(-) diff --git a/jwk/convert.go b/jwk/convert.go index 502f69131..cb548c137 100644 --- a/jwk/convert.go +++ b/jwk/convert.go @@ -235,6 +235,9 @@ func fromRaw(_ KeyFromRawer, key interface{}) (Key, error) { // - "crypto/ecdsa".PrivateKey and "crypto/ecdsa".PublicKey creates an EC based key // - "crypto/ed25519".PrivateKey and "crypto/ed25519".PublicKey creates an OKP based key // - []byte creates a symmetric key +// +// This function also takes care of additional key types added by external +// libraries such as secp256k1 keys. func FromRaw(raw interface{}) (Key, error) { if raw == nil { return nil, fmt.Errorf(`jwk.FromRaw requires a non-nil key`) @@ -244,9 +247,12 @@ func FromRaw(raw interface{}) (Key, error) { } // Raw converts a jwk.Key to its raw form and stores in the `raw` variable. -// `raw` must be a pointer to a compatible object. +// `raw` must be a pointer to a compatible object, otherwise an error will +// be returned. // // As of v2.0.12, it is recommended to use `jwk.Raw()` instead of `keyObject.Raw()`. +// The latter will NOT take care of converting additional key types added by +// external libraries, such as secp256k1 keys. func Raw(key Key, raw interface{}) error { return chainedRFK.Next(key, raw) }