forked from TrueCloudLab/frostfs-s3-gw
Roman Loginov
a725c68d06
Salt is used when generating encryption keys for data (tokens) in the access box. Now frostfs-s3-authmate always derivation an encryption key with salt. Signed-off-by: Roman Loginov <r.loginov@yadro.com>
341 lines
9.5 KiB
Go
341 lines
9.5 KiB
Go
package accessbox
|
|
|
|
import (
|
|
"bytes"
|
|
"crypto/cipher"
|
|
"crypto/elliptic"
|
|
"crypto/rand"
|
|
"crypto/sha256"
|
|
"encoding/hex"
|
|
"fmt"
|
|
"io"
|
|
|
|
"git.frostfs.info/TrueCloudLab/frostfs-sdk-go/bearer"
|
|
"git.frostfs.info/TrueCloudLab/frostfs-sdk-go/netmap"
|
|
"git.frostfs.info/TrueCloudLab/frostfs-sdk-go/session"
|
|
"github.com/nspcc-dev/neo-go/pkg/crypto/keys"
|
|
"golang.org/x/crypto/chacha20poly1305"
|
|
"golang.org/x/crypto/hkdf"
|
|
"google.golang.org/protobuf/proto"
|
|
)
|
|
|
|
const (
|
|
secretLength = 32
|
|
saltLength = 16
|
|
)
|
|
|
|
// Box represents friendly AccessBox.
|
|
type Box struct {
|
|
Gate *GateData
|
|
Policies []*ContainerPolicy
|
|
}
|
|
|
|
// ContainerPolicy represents friendly AccessBox_ContainerPolicy.
|
|
type ContainerPolicy struct {
|
|
LocationConstraint string
|
|
Policy netmap.PlacementPolicy
|
|
}
|
|
|
|
// GateData represents gate tokens in AccessBox.
|
|
type GateData struct {
|
|
SecretKey string
|
|
BearerToken *bearer.Token
|
|
SessionTokens []*session.Container
|
|
GateKey *keys.PublicKey
|
|
}
|
|
|
|
// NewGateData returns GateData from the provided bearer token and the public gate key.
|
|
func NewGateData(gateKey *keys.PublicKey, bearerTkn *bearer.Token) *GateData {
|
|
return &GateData{GateKey: gateKey, BearerToken: bearerTkn}
|
|
}
|
|
|
|
// SessionTokenForPut returns the first suitable container session context for PUT operation.
|
|
func (g *GateData) SessionTokenForPut() *session.Container {
|
|
return g.containerSessionToken(session.VerbContainerPut)
|
|
}
|
|
|
|
// SessionTokenForDelete returns the first suitable container session context for DELETE operation.
|
|
func (g *GateData) SessionTokenForDelete() *session.Container {
|
|
return g.containerSessionToken(session.VerbContainerDelete)
|
|
}
|
|
|
|
// SessionToken returns the first container session context.
|
|
func (g *GateData) SessionToken() *session.Container {
|
|
if len(g.SessionTokens) != 0 {
|
|
return g.SessionTokens[0]
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (g *GateData) containerSessionToken(verb session.ContainerVerb) *session.Container {
|
|
for _, sessionToken := range g.SessionTokens {
|
|
if isAppropriateContainerContext(sessionToken, verb) {
|
|
return sessionToken
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func isAppropriateContainerContext(tok *session.Container, verb session.ContainerVerb) bool {
|
|
switch verb {
|
|
case session.VerbContainerDelete, session.VerbContainerPut:
|
|
return tok.AssertVerb(verb)
|
|
default:
|
|
return false
|
|
}
|
|
}
|
|
|
|
// Secrets represents SecretKey and the key to encrypt gate tokens.
|
|
type Secrets struct {
|
|
SecretKey string
|
|
EphemeralKey *keys.PrivateKey
|
|
}
|
|
|
|
// Marshal returns the wire-format of AccessBox.
|
|
func (x *AccessBox) Marshal() ([]byte, error) {
|
|
return proto.Marshal(x)
|
|
}
|
|
|
|
// Unmarshal parses the wire-format message and put data to x.
|
|
func (x *AccessBox) Unmarshal(data []byte) error {
|
|
return proto.Unmarshal(data, x)
|
|
}
|
|
|
|
// PackTokens adds bearer and session tokens to BearerTokens and SessionToken lists respectively.
|
|
// Session token can be nil.
|
|
// Secret can be nil. In such case secret will be generated.
|
|
func PackTokens(gatesData []*GateData, secret []byte, isCustomSecret bool) (*AccessBox, *Secrets, error) {
|
|
box := &AccessBox{}
|
|
ephemeralKey, err := keys.NewPrivateKey()
|
|
if err != nil {
|
|
return nil, nil, fmt.Errorf("create ephemeral key: %w", err)
|
|
}
|
|
box.SeedKey = ephemeralKey.PublicKey().Bytes()
|
|
box.IsCustom = isCustomSecret
|
|
|
|
if secret == nil {
|
|
secret, err = generateRandomBytes(secretLength)
|
|
if err != nil {
|
|
return nil, nil, fmt.Errorf("failed to generate accessKey as hex: %w", err)
|
|
}
|
|
}
|
|
|
|
if err := box.addTokens(gatesData, ephemeralKey, secret); err != nil {
|
|
return nil, nil, fmt.Errorf("failed to add tokens to accessbox: %w", err)
|
|
}
|
|
|
|
secretKey := string(secret)
|
|
if !isCustomSecret {
|
|
secretKey = hex.EncodeToString(secret)
|
|
}
|
|
|
|
return box, &Secrets{SecretKey: secretKey, EphemeralKey: ephemeralKey}, err
|
|
}
|
|
|
|
// GetTokens returns gate tokens from AccessBox.
|
|
func (x *AccessBox) GetTokens(owner *keys.PrivateKey) (*GateData, error) {
|
|
seedKey, err := keys.NewPublicKeyFromBytes(x.SeedKey, elliptic.P256())
|
|
if err != nil {
|
|
return nil, fmt.Errorf("couldn't unmarshal SeedKey: %w", err)
|
|
}
|
|
ownerKey := owner.PublicKey().Bytes()
|
|
for _, gate := range x.Gates {
|
|
if !bytes.Equal(gate.GatePublicKey, ownerKey) {
|
|
continue
|
|
}
|
|
|
|
gateData, err := x.decodeGate(gate, owner, seedKey)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to decode gate: %w", err)
|
|
}
|
|
return gateData, nil
|
|
}
|
|
|
|
return nil, fmt.Errorf("no gate data for key %x was found", ownerKey)
|
|
}
|
|
|
|
// GetPlacementPolicy returns ContainerPolicy from AccessBox.
|
|
func (x *AccessBox) GetPlacementPolicy() ([]*ContainerPolicy, error) {
|
|
var result []*ContainerPolicy
|
|
for _, policy := range x.ContainerPolicy {
|
|
var cnrPolicy ContainerPolicy
|
|
if err := cnrPolicy.Policy.Unmarshal(policy.Policy); err != nil {
|
|
return nil, fmt.Errorf("unmarshal placement policy: %w", err)
|
|
}
|
|
|
|
cnrPolicy.LocationConstraint = policy.LocationConstraint
|
|
|
|
result = append(result, &cnrPolicy)
|
|
}
|
|
|
|
return result, nil
|
|
}
|
|
|
|
// GetBox parses AccessBox to Box.
|
|
func (x *AccessBox) GetBox(owner *keys.PrivateKey) (*Box, error) {
|
|
tokens, err := x.GetTokens(owner)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("get tokens: %w", err)
|
|
}
|
|
|
|
policy, err := x.GetPlacementPolicy()
|
|
if err != nil {
|
|
return nil, fmt.Errorf("get policy: %w", err)
|
|
}
|
|
|
|
return &Box{
|
|
Gate: tokens,
|
|
Policies: policy,
|
|
}, nil
|
|
}
|
|
|
|
func (x *AccessBox) addTokens(gatesData []*GateData, ephemeralKey *keys.PrivateKey, secret []byte) error {
|
|
for _, gate := range gatesData {
|
|
encBearer := gate.BearerToken.Marshal()
|
|
encSessions := make([][]byte, len(gate.SessionTokens))
|
|
for i, sessionToken := range gate.SessionTokens {
|
|
encSessions[i] = sessionToken.Marshal()
|
|
}
|
|
|
|
tokens := new(Tokens)
|
|
tokens.SecretKey = secret
|
|
tokens.BearerToken = encBearer
|
|
tokens.SessionTokens = encSessions
|
|
|
|
boxGate, err := encodeGate(ephemeralKey, gate.GateKey, tokens)
|
|
if err != nil {
|
|
return fmt.Errorf("encode gate: %w", err)
|
|
}
|
|
x.Gates = append(x.Gates, boxGate)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func encodeGate(ephemeralKey *keys.PrivateKey, seedKey *keys.PublicKey, tokens *Tokens) (*AccessBox_Gate, error) {
|
|
data, err := proto.Marshal(tokens)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("encode tokens: %w", err)
|
|
}
|
|
|
|
salt, err := generateRandomBytes(saltLength)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to generate salt for encryption key: %w", err)
|
|
}
|
|
|
|
encrypted, err := encrypt(ephemeralKey, seedKey, data, salt)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("ecrypt tokens: %w", err)
|
|
}
|
|
|
|
gate := new(AccessBox_Gate)
|
|
gate.GatePublicKey = seedKey.Bytes()
|
|
gate.Tokens = encrypted
|
|
gate.EncryptionKeySalt = salt
|
|
return gate, nil
|
|
}
|
|
|
|
func (x *AccessBox) decodeGate(gate *AccessBox_Gate, owner *keys.PrivateKey, seedKey *keys.PublicKey) (*GateData, error) {
|
|
data, err := decrypt(owner, seedKey, gate.Tokens, gate.EncryptionKeySalt)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("decrypt tokens: %w", err)
|
|
}
|
|
tokens := new(Tokens)
|
|
if err = proto.Unmarshal(data, tokens); err != nil {
|
|
return nil, fmt.Errorf("unmarshal tokens: %w", err)
|
|
}
|
|
|
|
var bearerTkn bearer.Token
|
|
if err = bearerTkn.Unmarshal(tokens.BearerToken); err != nil {
|
|
return nil, fmt.Errorf("unmarshal bearer token: %w", err)
|
|
}
|
|
|
|
sessionTkns := make([]*session.Container, len(tokens.SessionTokens))
|
|
for i, encSessionToken := range tokens.SessionTokens {
|
|
sessionTkn := new(session.Container)
|
|
if err = sessionTkn.Unmarshal(encSessionToken); err != nil {
|
|
return nil, fmt.Errorf("unmarshal session token: %w", err)
|
|
}
|
|
sessionTkns[i] = sessionTkn
|
|
}
|
|
|
|
gateData := NewGateData(owner.PublicKey(), &bearerTkn)
|
|
gateData.SessionTokens = sessionTkns
|
|
if x.IsCustom {
|
|
gateData.SecretKey = string(tokens.SecretKey)
|
|
} else {
|
|
gateData.SecretKey = hex.EncodeToString(tokens.SecretKey)
|
|
}
|
|
return gateData, nil
|
|
}
|
|
|
|
func generateShared256(prv *keys.PrivateKey, pub *keys.PublicKey) (sk []byte, err error) {
|
|
if prv.PublicKey().Curve != pub.Curve {
|
|
return nil, fmt.Errorf("not equal curves")
|
|
}
|
|
|
|
x, _ := pub.Curve.ScalarMult(pub.X, pub.Y, prv.D.Bytes())
|
|
if x == nil {
|
|
return nil, fmt.Errorf("shared key is point at infinity")
|
|
}
|
|
|
|
sk = make([]byte, 32)
|
|
skBytes := x.Bytes()
|
|
copy(sk[len(sk)-len(skBytes):], skBytes)
|
|
return sk, nil
|
|
}
|
|
|
|
func deriveKey(secret, salt []byte) ([]byte, error) {
|
|
hash := sha256.New
|
|
kdf := hkdf.New(hash, secret, salt, nil)
|
|
key := make([]byte, 32)
|
|
_, err := io.ReadFull(kdf, key)
|
|
return key, err
|
|
}
|
|
|
|
func encrypt(owner *keys.PrivateKey, seedKey *keys.PublicKey, data, salt []byte) ([]byte, error) {
|
|
enc, err := getCipher(owner, seedKey, salt)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("get chiper: %w", err)
|
|
}
|
|
|
|
nonce := make([]byte, enc.NonceSize())
|
|
if _, err = rand.Read(nonce); err != nil {
|
|
return nil, fmt.Errorf("generate random nonce: %w", err)
|
|
}
|
|
|
|
return enc.Seal(nonce, nonce, data, nil), nil
|
|
}
|
|
|
|
func decrypt(owner *keys.PrivateKey, seedKey *keys.PublicKey, data, salt []byte) ([]byte, error) {
|
|
dec, err := getCipher(owner, seedKey, salt)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("get chiper: %w", err)
|
|
}
|
|
|
|
if ld, ns := len(data), dec.NonceSize(); ld < ns {
|
|
return nil, fmt.Errorf("wrong data size (%d), should be greater than %d", ld, ns)
|
|
}
|
|
|
|
nonce, cypher := data[:dec.NonceSize()], data[dec.NonceSize():]
|
|
return dec.Open(nil, nonce, cypher, nil)
|
|
}
|
|
|
|
func getCipher(owner *keys.PrivateKey, seedKey *keys.PublicKey, salt []byte) (cipher.AEAD, error) {
|
|
secret, err := generateShared256(owner, seedKey)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("generate shared key: %w", err)
|
|
}
|
|
|
|
key, err := deriveKey(secret, salt)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("derive key: %w", err)
|
|
}
|
|
|
|
return chacha20poly1305.NewX(key)
|
|
}
|
|
|
|
func generateRandomBytes(length int) ([]byte, error) {
|
|
b := make([]byte, length)
|
|
_, err := rand.Read(b)
|
|
return b, err
|
|
}
|