2023-09-27 17:36:23 +08:00
|
|
|
package idimpl
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"context"
|
|
|
|
|
|
|
|
|
|
"github.com/go-jose/go-jose/v3"
|
|
|
|
|
"github.com/go-jose/go-jose/v3/jwt"
|
|
|
|
|
|
|
|
|
|
"github.com/grafana/grafana/pkg/services/auth"
|
|
|
|
|
"github.com/grafana/grafana/pkg/services/signingkeys"
|
|
|
|
|
)
|
|
|
|
|
|
2023-10-04 16:37:27 +08:00
|
|
|
const idSignerKeyPrefix = "id"
|
|
|
|
|
|
2023-09-27 17:36:23 +08:00
|
|
|
var _ auth.IDSigner = (*LocalSigner)(nil)
|
|
|
|
|
|
|
|
|
|
func ProvideLocalSigner(keyService signingkeys.Service) (*LocalSigner, error) {
|
2023-10-04 16:37:27 +08:00
|
|
|
id, key, err := keyService.GetOrCreatePrivateKey(context.Background(), idSignerKeyPrefix, jose.ES256)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
2023-09-27 17:36:23 +08:00
|
|
|
|
2023-10-04 16:37:27 +08:00
|
|
|
// FIXME: Handle key rotation
|
2023-09-27 17:36:23 +08:00
|
|
|
signer, err := jose.NewSigner(jose.SigningKey{Algorithm: jose.ES256, Key: key}, &jose.SignerOptions{
|
|
|
|
|
ExtraHeaders: map[jose.HeaderKey]interface{}{
|
2023-10-04 16:37:27 +08:00
|
|
|
"kid": id,
|
2023-09-27 17:36:23 +08:00
|
|
|
},
|
|
|
|
|
})
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return &LocalSigner{
|
|
|
|
|
signer: signer,
|
|
|
|
|
}, nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
type LocalSigner struct {
|
|
|
|
|
signer jose.Signer
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (s *LocalSigner) SignIDToken(ctx context.Context, claims *auth.IDClaims) (string, error) {
|
|
|
|
|
builder := jwt.Signed(s.signer).Claims(claims.Claims)
|
|
|
|
|
|
|
|
|
|
token, err := builder.CompactSerialize()
|
|
|
|
|
if err != nil {
|
|
|
|
|
return "", err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return token, nil
|
|
|
|
|
}
|