2018-10-09 14:00:01 -07:00
|
|
|
/*
|
2019-04-09 11:39:42 -07:00
|
|
|
* MinIO Cloud Storage, (C) 2018 MinIO, Inc.
|
2018-10-09 14:00:01 -07:00
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2019-10-01 15:07:20 -07:00
|
|
|
package openid
|
2018-10-09 14:00:01 -07:00
|
|
|
|
|
|
|
import (
|
|
|
|
"crypto"
|
|
|
|
"crypto/ecdsa"
|
|
|
|
"crypto/elliptic"
|
|
|
|
"crypto/rsa"
|
|
|
|
"encoding/base64"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"math/big"
|
|
|
|
)
|
|
|
|
|
|
|
|
// JWKS - https://tools.ietf.org/html/rfc7517
|
|
|
|
type JWKS struct {
|
|
|
|
Keys []*JWKS `json:"keys,omitempty"`
|
|
|
|
|
|
|
|
Kty string `json:"kty"`
|
|
|
|
Use string `json:"use,omitempty"`
|
|
|
|
Kid string `json:"kid,omitempty"`
|
|
|
|
Alg string `json:"alg,omitempty"`
|
|
|
|
|
|
|
|
Crv string `json:"crv,omitempty"`
|
|
|
|
X string `json:"x,omitempty"`
|
|
|
|
Y string `json:"y,omitempty"`
|
|
|
|
D string `json:"d,omitempty"`
|
|
|
|
N string `json:"n,omitempty"`
|
|
|
|
E string `json:"e,omitempty"`
|
|
|
|
K string `json:"k,omitempty"`
|
|
|
|
}
|
|
|
|
|
|
|
|
var (
|
|
|
|
errMalformedJWKRSAKey = errors.New("malformed JWK RSA key")
|
|
|
|
errMalformedJWKECKey = errors.New("malformed JWK EC key")
|
|
|
|
)
|
|
|
|
|
|
|
|
// DecodePublicKey - decodes JSON Web Key (JWK) as public key
|
|
|
|
func (key *JWKS) DecodePublicKey() (crypto.PublicKey, error) {
|
|
|
|
switch key.Kty {
|
|
|
|
case "RSA":
|
|
|
|
if key.N == "" || key.E == "" {
|
|
|
|
return nil, errMalformedJWKRSAKey
|
|
|
|
}
|
|
|
|
|
|
|
|
// decode exponent
|
2020-07-25 08:42:41 -07:00
|
|
|
ebuf, err := base64.RawURLEncoding.DecodeString(key.E)
|
2018-10-09 14:00:01 -07:00
|
|
|
if err != nil {
|
|
|
|
return nil, errMalformedJWKRSAKey
|
|
|
|
}
|
|
|
|
|
2020-07-25 08:42:41 -07:00
|
|
|
nbuf, err := base64.RawURLEncoding.DecodeString(key.N)
|
2018-10-09 14:00:01 -07:00
|
|
|
if err != nil {
|
|
|
|
return nil, errMalformedJWKRSAKey
|
|
|
|
}
|
|
|
|
|
2020-07-25 08:42:41 -07:00
|
|
|
var n, e big.Int
|
|
|
|
n.SetBytes(nbuf)
|
|
|
|
e.SetBytes(ebuf)
|
|
|
|
|
|
|
|
return &rsa.PublicKey{
|
|
|
|
E: int(e.Int64()),
|
|
|
|
N: &n,
|
|
|
|
}, nil
|
2018-10-09 14:00:01 -07:00
|
|
|
case "EC":
|
|
|
|
if key.Crv == "" || key.X == "" || key.Y == "" {
|
|
|
|
return nil, errMalformedJWKECKey
|
|
|
|
}
|
|
|
|
|
|
|
|
var curve elliptic.Curve
|
|
|
|
switch key.Crv {
|
|
|
|
case "P-224":
|
|
|
|
curve = elliptic.P224()
|
|
|
|
case "P-256":
|
|
|
|
curve = elliptic.P256()
|
|
|
|
case "P-384":
|
|
|
|
curve = elliptic.P384()
|
|
|
|
case "P-521":
|
|
|
|
curve = elliptic.P521()
|
|
|
|
default:
|
|
|
|
return nil, fmt.Errorf("Unknown curve type: %s", key.Crv)
|
|
|
|
}
|
|
|
|
|
2020-07-25 08:42:41 -07:00
|
|
|
xbuf, err := base64.RawURLEncoding.DecodeString(key.X)
|
2018-10-09 14:00:01 -07:00
|
|
|
if err != nil {
|
|
|
|
return nil, errMalformedJWKECKey
|
|
|
|
}
|
|
|
|
|
2020-07-25 08:42:41 -07:00
|
|
|
ybuf, err := base64.RawURLEncoding.DecodeString(key.Y)
|
2018-10-09 14:00:01 -07:00
|
|
|
if err != nil {
|
|
|
|
return nil, errMalformedJWKECKey
|
|
|
|
}
|
|
|
|
|
2020-07-25 08:42:41 -07:00
|
|
|
var x, y big.Int
|
|
|
|
x.SetBytes(xbuf)
|
|
|
|
y.SetBytes(ybuf)
|
|
|
|
|
|
|
|
return &ecdsa.PublicKey{
|
|
|
|
Curve: curve,
|
|
|
|
X: &x,
|
|
|
|
Y: &y,
|
|
|
|
}, nil
|
2018-10-09 14:00:01 -07:00
|
|
|
default:
|
|
|
|
return nil, fmt.Errorf("Unknown JWK key type %s", key.Kty)
|
|
|
|
}
|
|
|
|
}
|