| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538 |
- package api
- import (
- "context"
- "crypto/ecdsa"
- "crypto/rand"
- "encoding/hex"
- "encoding/json"
- "errors"
- "fmt"
- "io"
- "strings"
- "time"
- "github.com/ethereum/go-ethereum/common"
- "github.com/ethereum/go-ethereum/crypto"
- "github.com/ethereum/go-ethereum/crypto/ecies"
- "github.com/ethereum/go-ethereum/swarm/log"
- "github.com/ethereum/go-ethereum/swarm/sctx"
- "github.com/ethereum/go-ethereum/swarm/storage"
- "golang.org/x/crypto/scrypt"
- "golang.org/x/crypto/sha3"
- cli "gopkg.in/urfave/cli.v1"
- )
- var (
- ErrDecrypt = errors.New("cant decrypt - forbidden")
- ErrUnknownAccessType = errors.New("unknown access type (or not implemented)")
- ErrDecryptDomainForbidden = errors.New("decryption request domain forbidden - can only decrypt on localhost")
- AllowedDecryptDomains = []string{
- "localhost",
- "127.0.0.1",
- }
- )
- const EMPTY_CREDENTIALS = ""
- type AccessEntry struct {
- Type AccessType
- Publisher string
- Salt []byte
- Act string
- KdfParams *KdfParams
- }
- type DecryptFunc func(*ManifestEntry) error
- func (a *AccessEntry) MarshalJSON() (out []byte, err error) {
- return json.Marshal(struct {
- Type AccessType `json:"type,omitempty"`
- Publisher string `json:"publisher,omitempty"`
- Salt string `json:"salt,omitempty"`
- Act string `json:"act,omitempty"`
- KdfParams *KdfParams `json:"kdf_params,omitempty"`
- }{
- Type: a.Type,
- Publisher: a.Publisher,
- Salt: hex.EncodeToString(a.Salt),
- Act: a.Act,
- KdfParams: a.KdfParams,
- })
- }
- func (a *AccessEntry) UnmarshalJSON(value []byte) error {
- v := struct {
- Type AccessType `json:"type,omitempty"`
- Publisher string `json:"publisher,omitempty"`
- Salt string `json:"salt,omitempty"`
- Act string `json:"act,omitempty"`
- KdfParams *KdfParams `json:"kdf_params,omitempty"`
- }{}
- err := json.Unmarshal(value, &v)
- if err != nil {
- return err
- }
- a.Act = v.Act
- a.KdfParams = v.KdfParams
- a.Publisher = v.Publisher
- a.Salt, err = hex.DecodeString(v.Salt)
- if err != nil {
- return err
- }
- if len(a.Salt) != 32 {
- return errors.New("salt should be 32 bytes long")
- }
- a.Type = v.Type
- return nil
- }
- type KdfParams struct {
- N int `json:"n"`
- P int `json:"p"`
- R int `json:"r"`
- }
- type AccessType string
- const AccessTypePass = AccessType("pass")
- const AccessTypePK = AccessType("pk")
- const AccessTypeACT = AccessType("act")
- // NewAccessEntryPassword creates a manifest AccessEntry in order to create an ACT protected by a password
- func NewAccessEntryPassword(salt []byte, kdfParams *KdfParams) (*AccessEntry, error) {
- if len(salt) != 32 {
- return nil, fmt.Errorf("salt should be 32 bytes long")
- }
- return &AccessEntry{
- Type: AccessTypePass,
- Salt: salt,
- KdfParams: kdfParams,
- }, nil
- }
- // NewAccessEntryPK creates a manifest AccessEntry in order to create an ACT protected by a pair of Elliptic Curve keys
- func NewAccessEntryPK(publisher string, salt []byte) (*AccessEntry, error) {
- if len(publisher) != 66 {
- return nil, fmt.Errorf("publisher should be 66 characters long, got %d", len(publisher))
- }
- if len(salt) != 32 {
- return nil, fmt.Errorf("salt should be 32 bytes long")
- }
- return &AccessEntry{
- Type: AccessTypePK,
- Publisher: publisher,
- Salt: salt,
- }, nil
- }
- // NewAccessEntryACT creates a manifest AccessEntry in order to create an ACT protected by a combination of EC keys and passwords
- func NewAccessEntryACT(publisher string, salt []byte, act string) (*AccessEntry, error) {
- if len(salt) != 32 {
- return nil, fmt.Errorf("salt should be 32 bytes long")
- }
- if len(publisher) != 66 {
- return nil, fmt.Errorf("publisher should be 66 characters long")
- }
- return &AccessEntry{
- Type: AccessTypeACT,
- Publisher: publisher,
- Salt: salt,
- Act: act,
- KdfParams: DefaultKdfParams,
- }, nil
- }
- // NOOPDecrypt is a generic decrypt function that is passed into the API in places where real ACT decryption capabilities are
- // either unwanted, or alternatively, cannot be implemented in the immediate scope
- func NOOPDecrypt(*ManifestEntry) error {
- return nil
- }
- var DefaultKdfParams = NewKdfParams(262144, 1, 8)
- // NewKdfParams returns a KdfParams struct with the given scrypt params
- func NewKdfParams(n, p, r int) *KdfParams {
- return &KdfParams{
- N: n,
- P: p,
- R: r,
- }
- }
- // NewSessionKeyPassword creates a session key based on a shared secret (password) and the given salt
- // and kdf parameters in the access entry
- func NewSessionKeyPassword(password string, accessEntry *AccessEntry) ([]byte, error) {
- if accessEntry.Type != AccessTypePass && accessEntry.Type != AccessTypeACT {
- return nil, errors.New("incorrect access entry type")
- }
- return sessionKeyPassword(password, accessEntry.Salt, accessEntry.KdfParams)
- }
- func sessionKeyPassword(password string, salt []byte, kdfParams *KdfParams) ([]byte, error) {
- return scrypt.Key(
- []byte(password),
- salt,
- kdfParams.N,
- kdfParams.R,
- kdfParams.P,
- 32,
- )
- }
- // NewSessionKeyPK creates a new ACT Session Key using an ECDH shared secret for the given key pair and the given salt value
- func NewSessionKeyPK(private *ecdsa.PrivateKey, public *ecdsa.PublicKey, salt []byte) ([]byte, error) {
- granteePubEcies := ecies.ImportECDSAPublic(public)
- privateKey := ecies.ImportECDSA(private)
- bytes, err := privateKey.GenerateShared(granteePubEcies, 16, 16)
- if err != nil {
- return nil, err
- }
- bytes = append(salt, bytes...)
- sessionKey := crypto.Keccak256(bytes)
- return sessionKey, nil
- }
- func (a *API) doDecrypt(ctx context.Context, credentials string, pk *ecdsa.PrivateKey) DecryptFunc {
- return func(m *ManifestEntry) error {
- if m.Access == nil {
- return nil
- }
- allowed := false
- requestDomain := sctx.GetHost(ctx)
- for _, v := range AllowedDecryptDomains {
- if strings.Contains(requestDomain, v) {
- allowed = true
- }
- }
- if !allowed {
- return ErrDecryptDomainForbidden
- }
- switch m.Access.Type {
- case "pass":
- if credentials != "" {
- key, err := NewSessionKeyPassword(credentials, m.Access)
- if err != nil {
- return err
- }
- ref, err := hex.DecodeString(m.Hash)
- if err != nil {
- return err
- }
- enc := NewRefEncryption(len(ref) - 8)
- decodedRef, err := enc.Decrypt(ref, key)
- if err != nil {
- return ErrDecrypt
- }
- m.Hash = hex.EncodeToString(decodedRef)
- m.Access = nil
- return nil
- }
- return ErrDecrypt
- case "pk":
- publisherBytes, err := hex.DecodeString(m.Access.Publisher)
- if err != nil {
- return ErrDecrypt
- }
- publisher, err := crypto.DecompressPubkey(publisherBytes)
- if err != nil {
- return ErrDecrypt
- }
- key, err := NewSessionKeyPK(pk, publisher, m.Access.Salt)
- if err != nil {
- return ErrDecrypt
- }
- ref, err := hex.DecodeString(m.Hash)
- if err != nil {
- return err
- }
- enc := NewRefEncryption(len(ref) - 8)
- decodedRef, err := enc.Decrypt(ref, key)
- if err != nil {
- return ErrDecrypt
- }
- m.Hash = hex.EncodeToString(decodedRef)
- m.Access = nil
- return nil
- case "act":
- var (
- sessionKey []byte
- err error
- )
- publisherBytes, err := hex.DecodeString(m.Access.Publisher)
- if err != nil {
- return ErrDecrypt
- }
- publisher, err := crypto.DecompressPubkey(publisherBytes)
- if err != nil {
- return ErrDecrypt
- }
- sessionKey, err = NewSessionKeyPK(pk, publisher, m.Access.Salt)
- if err != nil {
- return ErrDecrypt
- }
- found, ciphertext, decryptionKey, err := a.getACTDecryptionKey(ctx, storage.Address(common.Hex2Bytes(m.Access.Act)), sessionKey)
- if err != nil {
- return err
- }
- if !found {
- // try to fall back to password
- if credentials != "" {
- sessionKey, err = NewSessionKeyPassword(credentials, m.Access)
- if err != nil {
- return err
- }
- found, ciphertext, decryptionKey, err = a.getACTDecryptionKey(ctx, storage.Address(common.Hex2Bytes(m.Access.Act)), sessionKey)
- if err != nil {
- return err
- }
- if !found {
- return ErrDecrypt
- }
- } else {
- return ErrDecrypt
- }
- }
- enc := NewRefEncryption(len(ciphertext) - 8)
- decodedRef, err := enc.Decrypt(ciphertext, decryptionKey)
- if err != nil {
- return ErrDecrypt
- }
- ref, err := hex.DecodeString(m.Hash)
- if err != nil {
- return err
- }
- enc = NewRefEncryption(len(ref) - 8)
- decodedMainRef, err := enc.Decrypt(ref, decodedRef)
- if err != nil {
- return ErrDecrypt
- }
- m.Hash = hex.EncodeToString(decodedMainRef)
- m.Access = nil
- return nil
- }
- return ErrUnknownAccessType
- }
- }
- func (a *API) getACTDecryptionKey(ctx context.Context, actManifestAddress storage.Address, sessionKey []byte) (found bool, ciphertext, decryptionKey []byte, err error) {
- hasher := sha3.NewLegacyKeccak256()
- hasher.Write(append(sessionKey, 0))
- lookupKey := hasher.Sum(nil)
- hasher.Reset()
- hasher.Write(append(sessionKey, 1))
- accessKeyDecryptionKey := hasher.Sum(nil)
- hasher.Reset()
- lk := hex.EncodeToString(lookupKey)
- list, err := a.GetManifestList(ctx, NOOPDecrypt, actManifestAddress, lk)
- if err != nil {
- return false, nil, nil, err
- }
- for _, v := range list.Entries {
- if v.Path == lk {
- cipherTextBytes, err := hex.DecodeString(v.Hash)
- if err != nil {
- return false, nil, nil, err
- }
- return true, cipherTextBytes, accessKeyDecryptionKey, nil
- }
- }
- return false, nil, nil, nil
- }
- func GenerateAccessControlManifest(ctx *cli.Context, ref string, accessKey []byte, ae *AccessEntry) (*Manifest, error) {
- refBytes, err := hex.DecodeString(ref)
- if err != nil {
- return nil, err
- }
- // encrypt ref with accessKey
- enc := NewRefEncryption(len(refBytes))
- encrypted, err := enc.Encrypt(refBytes, accessKey)
- if err != nil {
- return nil, err
- }
- m := &Manifest{
- Entries: []ManifestEntry{
- {
- Hash: hex.EncodeToString(encrypted),
- ContentType: ManifestType,
- ModTime: time.Now(),
- Access: ae,
- },
- },
- }
- return m, nil
- }
- // DoPK is a helper function to the CLI API that handles the entire business logic for
- // creating a session key and access entry given the cli context, ec keys and salt
- func DoPK(ctx *cli.Context, privateKey *ecdsa.PrivateKey, granteePublicKey string, salt []byte) (sessionKey []byte, ae *AccessEntry, err error) {
- if granteePublicKey == "" {
- return nil, nil, errors.New("need a grantee Public Key")
- }
- b, err := hex.DecodeString(granteePublicKey)
- if err != nil {
- log.Error("error decoding grantee public key", "err", err)
- return nil, nil, err
- }
- granteePub, err := crypto.DecompressPubkey(b)
- if err != nil {
- log.Error("error decompressing grantee public key", "err", err)
- return nil, nil, err
- }
- sessionKey, err = NewSessionKeyPK(privateKey, granteePub, salt)
- if err != nil {
- log.Error("error getting session key", "err", err)
- return nil, nil, err
- }
- ae, err = NewAccessEntryPK(hex.EncodeToString(crypto.CompressPubkey(&privateKey.PublicKey)), salt)
- if err != nil {
- log.Error("error generating access entry", "err", err)
- return nil, nil, err
- }
- return sessionKey, ae, nil
- }
- // DoACT is a helper function to the CLI API that handles the entire business logic for
- // creating a access key, access entry and ACT manifest (including uploading it) given the cli context, ec keys, password grantees and salt
- func DoACT(ctx *cli.Context, privateKey *ecdsa.PrivateKey, salt []byte, grantees []string, encryptPasswords []string) (accessKey []byte, ae *AccessEntry, actManifest *Manifest, err error) {
- if len(grantees) == 0 && len(encryptPasswords) == 0 {
- return nil, nil, nil, errors.New("did not get any grantee public keys or any encryption passwords")
- }
- publisherPub := hex.EncodeToString(crypto.CompressPubkey(&privateKey.PublicKey))
- grantees = append(grantees, publisherPub)
- accessKey = make([]byte, 32)
- if _, err := io.ReadFull(rand.Reader, salt); err != nil {
- panic("reading from crypto/rand failed: " + err.Error())
- }
- if _, err := io.ReadFull(rand.Reader, accessKey); err != nil {
- panic("reading from crypto/rand failed: " + err.Error())
- }
- lookupPathEncryptedAccessKeyMap := make(map[string]string)
- i := 0
- for _, v := range grantees {
- i++
- if v == "" {
- return nil, nil, nil, errors.New("need a grantee Public Key")
- }
- b, err := hex.DecodeString(v)
- if err != nil {
- log.Error("error decoding grantee public key", "err", err)
- return nil, nil, nil, err
- }
- granteePub, err := crypto.DecompressPubkey(b)
- if err != nil {
- log.Error("error decompressing grantee public key", "err", err)
- return nil, nil, nil, err
- }
- sessionKey, err := NewSessionKeyPK(privateKey, granteePub, salt)
- if err != nil {
- return nil, nil, nil, err
- }
- hasher := sha3.NewLegacyKeccak256()
- hasher.Write(append(sessionKey, 0))
- lookupKey := hasher.Sum(nil)
- hasher.Reset()
- hasher.Write(append(sessionKey, 1))
- accessKeyEncryptionKey := hasher.Sum(nil)
- enc := NewRefEncryption(len(accessKey))
- encryptedAccessKey, err := enc.Encrypt(accessKey, accessKeyEncryptionKey)
- if err != nil {
- return nil, nil, nil, err
- }
- lookupPathEncryptedAccessKeyMap[hex.EncodeToString(lookupKey)] = hex.EncodeToString(encryptedAccessKey)
- }
- for _, pass := range encryptPasswords {
- sessionKey, err := sessionKeyPassword(pass, salt, DefaultKdfParams)
- if err != nil {
- return nil, nil, nil, err
- }
- hasher := sha3.NewLegacyKeccak256()
- hasher.Write(append(sessionKey, 0))
- lookupKey := hasher.Sum(nil)
- hasher.Reset()
- hasher.Write(append(sessionKey, 1))
- accessKeyEncryptionKey := hasher.Sum(nil)
- enc := NewRefEncryption(len(accessKey))
- encryptedAccessKey, err := enc.Encrypt(accessKey, accessKeyEncryptionKey)
- if err != nil {
- return nil, nil, nil, err
- }
- lookupPathEncryptedAccessKeyMap[hex.EncodeToString(lookupKey)] = hex.EncodeToString(encryptedAccessKey)
- }
- m := &Manifest{
- Entries: []ManifestEntry{},
- }
- for k, v := range lookupPathEncryptedAccessKeyMap {
- m.Entries = append(m.Entries, ManifestEntry{
- Path: k,
- Hash: v,
- ContentType: "text/plain",
- })
- }
- ae, err = NewAccessEntryACT(hex.EncodeToString(crypto.CompressPubkey(&privateKey.PublicKey)), salt, "")
- if err != nil {
- return nil, nil, nil, err
- }
- return accessKey, ae, m, nil
- }
- // DoPassword is a helper function to the CLI API that handles the entire business logic for
- // creating a session key and an access entry given the cli context, password and salt.
- // By default - DefaultKdfParams are used as the scrypt params
- func DoPassword(ctx *cli.Context, password string, salt []byte) (sessionKey []byte, ae *AccessEntry, err error) {
- ae, err = NewAccessEntryPassword(salt, DefaultKdfParams)
- if err != nil {
- return nil, nil, err
- }
- sessionKey, err = NewSessionKeyPassword(password, ae)
- if err != nil {
- return nil, nil, err
- }
- return sessionKey, ae, nil
- }
|