aboutsummaryrefslogblamecommitdiffstats
path: root/cmd/swarm/access_test.go
blob: ed589f9f474f4aad7d02b7d328d90960c21c7333 (plain) (tree)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15














                                                                       


                  

















                                                
                                                      





                                                                

                                






























































































































































































































































































































                                                                                                                           









                                                                                                                                                                                                


                                                                                                                         


                                                                                                                               


















































                                                                                        


















                                                                                                                                             


                            

                                                      





                                                                                 

























































































































































                                                                                                                      
// Copyright 2018 The go-ethereum Authors
// This file is part of go-ethereum.
//
// go-ethereum is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// go-ethereum is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with go-ethereum. If not, see <http://www.gnu.org/licenses/>.

// +build !windows

package main

import (
    "bytes"
    "crypto/rand"
    "encoding/hex"
    "encoding/json"
    "io"
    "io/ioutil"
    gorand "math/rand"
    "net/http"
    "os"
    "path/filepath"
    "strings"
    "testing"
    "time"

    "github.com/ethereum/go-ethereum/crypto"
    "github.com/ethereum/go-ethereum/crypto/ecies"
    "github.com/ethereum/go-ethereum/crypto/sha3"
    "github.com/ethereum/go-ethereum/log"
    "github.com/ethereum/go-ethereum/swarm/api"
    swarm "github.com/ethereum/go-ethereum/swarm/api/client"
)

var DefaultCurve = crypto.S256()

// TestAccessPassword tests for the correct creation of an ACT manifest protected by a password.
// The test creates bogus content, uploads it encrypted, then creates the wrapping manifest with the Access entry
// The parties participating - node (publisher), uploads to second node then disappears. Content which was uploaded
// is then fetched through 2nd node. since the tested code is not key-aware - we can just
// fetch from the 2nd node using HTTP BasicAuth
func TestAccessPassword(t *testing.T) {
    cluster := newTestCluster(t, 1)
    defer cluster.Shutdown()
    proxyNode := cluster.Nodes[0]

    // create a tmp file
    tmp, err := ioutil.TempDir("", "swarm-test")
    if err != nil {
        t.Fatal(err)
    }
    defer os.RemoveAll(tmp)

    // write data to file
    data := "notsorandomdata"
    dataFilename := filepath.Join(tmp, "data.txt")

    err = ioutil.WriteFile(dataFilename, []byte(data), 0666)
    if err != nil {
        t.Fatal(err)
    }

    hashRegexp := `[a-f\d]{128}`

    // upload the file with 'swarm up' and expect a hash
    up := runSwarm(t,
        "--bzzapi",
        proxyNode.URL, //it doesn't matter through which node we upload content
        "up",
        "--encrypt",
        dataFilename)
    _, matches := up.ExpectRegexp(hashRegexp)
    up.ExpectExit()

    if len(matches) < 1 {
        t.Fatal("no matches found")
    }

    ref := matches[0]

    password := "smth"
    passwordFilename := filepath.Join(tmp, "password.txt")

    err = ioutil.WriteFile(passwordFilename, []byte(password), 0666)
    if err != nil {
        t.Fatal(err)
    }

    up = runSwarm(t,
        "access",
        "new",
        "pass",
        "--dry-run",
        "--password",
        passwordFilename,
        ref,
    )

    _, matches = up.ExpectRegexp(".+")
    up.ExpectExit()

    if len(matches) == 0 {
        t.Fatalf("stdout not matched")
    }

    var m api.Manifest

    err = json.Unmarshal([]byte(matches[0]), &m)
    if err != nil {
        t.Fatalf("unmarshal manifest: %v", err)
    }

    if len(m.Entries) != 1 {
        t.Fatalf("expected one manifest entry, got %v", len(m.Entries))
    }

    e := m.Entries[0]

    ct := "application/bzz-manifest+json"
    if e.ContentType != ct {
        t.Errorf("expected %q content type, got %q", ct, e.ContentType)
    }

    if e.Access == nil {
        t.Fatal("manifest access is nil")
    }

    a := e.Access

    if a.Type != "pass" {
        t.Errorf(`got access type %q, expected "pass"`, a.Type)
    }
    if len(a.Salt) < 32 {
        t.Errorf(`got salt with length %v, expected not less the 32 bytes`, len(a.Salt))
    }
    if a.KdfParams == nil {
        t.Fatal("manifest access kdf params is nil")
    }

    client := swarm.NewClient(cluster.Nodes[0].URL)

    hash, err := client.UploadManifest(&m, false)
    if err != nil {
        t.Fatal(err)
    }

    httpClient := &http.Client{}

    url := cluster.Nodes[0].URL + "/" + "bzz:/" + hash
    response, err := httpClient.Get(url)
    if err != nil {
        t.Fatal(err)
    }
    if response.StatusCode != http.StatusUnauthorized {
        t.Fatal("should be a 401")
    }
    authHeader := response.Header.Get("WWW-Authenticate")
    if authHeader == "" {
        t.Fatal("should be something here")
    }

    req, err := http.NewRequest(http.MethodGet, url, nil)
    if err != nil {
        t.Fatal(err)
    }
    req.SetBasicAuth("", password)

    response, err = http.DefaultClient.Do(req)
    if err != nil {
        t.Fatal(err)
    }
    defer response.Body.Close()

    if response.StatusCode != http.StatusOK {
        t.Errorf("expected status %v, got %v", http.StatusOK, response.StatusCode)
    }
    d, err := ioutil.ReadAll(response.Body)
    if err != nil {
        t.Fatal(err)
    }
    if string(d) != data {
        t.Errorf("expected decrypted data %q, got %q", data, string(d))
    }

    wrongPasswordFilename := filepath.Join(tmp, "password-wrong.txt")

    err = ioutil.WriteFile(wrongPasswordFilename, []byte("just wr0ng"), 0666)
    if err != nil {
        t.Fatal(err)
    }

    //download file with 'swarm down' with wrong password
    up = runSwarm(t,
        "--bzzapi",
        proxyNode.URL,
        "down",
        "bzz:/"+hash,
        tmp,
        "--password",
        wrongPasswordFilename)

    _, matches = up.ExpectRegexp("unauthorized")
    if len(matches) != 1 && matches[0] != "unauthorized" {
        t.Fatal(`"unauthorized" not found in output"`)
    }
    up.ExpectExit()
}

// TestAccessPK tests for the correct creation of an ACT manifest between two parties (publisher and grantee).
// The test creates bogus content, uploads it encrypted, then creates the wrapping manifest with the Access entry
// The parties participating - node (publisher), uploads to second node (which is also the grantee) then disappears.
// Content which was uploaded is then fetched through the grantee's http proxy. Since the tested code is private-key aware,
// the test will fail if the proxy's given private key is not granted on the ACT.
func TestAccessPK(t *testing.T) {
    // Setup Swarm and upload a test file to it
    cluster := newTestCluster(t, 1)
    defer cluster.Shutdown()

    // create a tmp file
    tmp, err := ioutil.TempFile("", "swarm-test")
    if err != nil {
        t.Fatal(err)
    }
    defer tmp.Close()
    defer os.Remove(tmp.Name())

    // write data to file
    data := "notsorandomdata"
    _, err = io.WriteString(tmp, data)
    if err != nil {
        t.Fatal(err)
    }

    hashRegexp := `[a-f\d]{128}`

    // upload the file with 'swarm up' and expect a hash
    up := runSwarm(t,
        "--bzzapi",
        cluster.Nodes[0].URL,
        "up",
        "--encrypt",
        tmp.Name())
    _, matches := up.ExpectRegexp(hashRegexp)
    up.ExpectExit()

    if len(matches) < 1 {
        t.Fatal("no matches found")
    }

    ref := matches[0]

    pk := cluster.Nodes[0].PrivateKey
    granteePubKey := crypto.CompressPubkey(&pk.PublicKey)

    publisherDir, err := ioutil.TempDir("", "swarm-account-dir-temp")
    if err != nil {
        t.Fatal(err)
    }

    passFile, err := ioutil.TempFile("", "swarm-test")
    if err != nil {
        t.Fatal(err)
    }
    defer passFile.Close()
    defer os.Remove(passFile.Name())
    _, err = io.WriteString(passFile, testPassphrase)
    if err != nil {
        t.Fatal(err)
    }
    _, publisherAccount := getTestAccount(t, publisherDir)
    up = runSwarm(t,
        "--bzzaccount",
        publisherAccount.Address.String(),
        "--password",
        passFile.Name(),
        "--datadir",
        publisherDir,
        "--bzzapi",
        cluster.Nodes[0].URL,
        "access",
        "new",
        "pk",
        "--dry-run",
        "--grant-key",
        hex.EncodeToString(granteePubKey),
        ref,
    )

    _, matches = up.ExpectRegexp(".+")
    up.ExpectExit()

    if len(matches) == 0 {
        t.Fatalf("stdout not matched")
    }

    var m api.Manifest

    err = json.Unmarshal([]byte(matches[0]), &m)
    if err != nil {
        t.Fatalf("unmarshal manifest: %v", err)
    }

    if len(m.Entries) != 1 {
        t.Fatalf("expected one manifest entry, got %v", len(m.Entries))
    }

    e := m.Entries[0]

    ct := "application/bzz-manifest+json"
    if e.ContentType != ct {
        t.Errorf("expected %q content type, got %q", ct, e.ContentType)
    }

    if e.Access == nil {
        t.Fatal("manifest access is nil")
    }

    a := e.Access

    if a.Type != "pk" {
        t.Errorf(`got access type %q, expected "pk"`, a.Type)
    }
    if len(a.Salt) < 32 {
        t.Errorf(`got salt with length %v, expected not less the 32 bytes`, len(a.Salt))
    }
    if a.KdfParams != nil {
        t.Fatal("manifest access kdf params should be nil")
    }

    client := swarm.NewClient(cluster.Nodes[0].URL)

    hash, err := client.UploadManifest(&m, false)
    if err != nil {
        t.Fatal(err)
    }

    httpClient := &http.Client{}

    url := cluster.Nodes[0].URL + "/" + "bzz:/" + hash
    response, err := httpClient.Get(url)
    if err != nil {
        t.Fatal(err)
    }
    if response.StatusCode != http.StatusOK {
        t.Fatal("should be a 200")
    }
    d, err := ioutil.ReadAll(response.Body)
    if err != nil {
        t.Fatal(err)
    }
    if string(d) != data {
        t.Errorf("expected decrypted data %q, got %q", data, string(d))
    }
}

// TestAccessACT tests the creation of the ACT manifest end-to-end, without any bogus entries (i.e. default scenario = 3 nodes 1 unauthorized)
func TestAccessACT(t *testing.T) {
    testAccessACT(t, 0)
}

// TestAccessACTScale tests the creation of the ACT manifest end-to-end, with 1000 bogus entries (i.e. 1000 EC keys + default scenario = 3 nodes 1 unauthorized = 1003 keys in the ACT manifest)
func TestAccessACTScale(t *testing.T) {
    testAccessACT(t, 1000)
}

// TestAccessACT tests the e2e creation, uploading and downloading of an ACT type access control
// the test fires up a 3 node cluster, then randomly picks 2 nodes which will be acting as grantees to the data
// set. the third node should fail decoding the reference as it will not be granted access. the publisher uploads through
// one of the nodes then disappears. If `bogusEntries` is bigger than 0, the test will generate the number of bogus act entries
// to test what happens at scale
func testAccessACT(t *testing.T, bogusEntries int) {
    // Setup Swarm and upload a test file to it
    cluster := newTestCluster(t, 3)
    defer cluster.Shutdown()

    var uploadThroughNode = cluster.Nodes[0]
    client := swarm.NewClient(uploadThroughNode.URL)

    r1 := gorand.New(gorand.NewSource(time.Now().UnixNano()))
    nodeToSkip := r1.Intn(3) // a number between 0 and 2 (node indices in `cluster`)
    // create a tmp file
    tmp, err := ioutil.TempFile("", "swarm-test")
    if err != nil {
        t.Fatal(err)
    }
    defer tmp.Close()
    defer os.Remove(tmp.Name())

    // write data to file
    data := "notsorandomdata"
    _, err = io.WriteString(tmp, data)
    if err != nil {
        t.Fatal(err)
    }

    hashRegexp := `[a-f\d]{128}`

    // upload the file with 'swarm up' and expect a hash
    up := runSwarm(t,
        "--bzzapi",
        cluster.Nodes[0].URL,
        "up",
        "--encrypt",
        tmp.Name())
    _, matches := up.ExpectRegexp(hashRegexp)
    up.ExpectExit()

    if len(matches) < 1 {
        t.Fatal("no matches found")
    }

    ref := matches[0]
    grantees := []string{}
    for i, v := range cluster.Nodes {
        if i == nodeToSkip {
            continue
        }
        pk := v.PrivateKey
        granteePubKey := crypto.CompressPubkey(&pk.PublicKey)
        grantees = append(grantees, hex.EncodeToString(granteePubKey))
    }

    if bogusEntries > 0 {
        bogusGrantees := []string{}

        for i := 0; i < bogusEntries; i++ {
            prv, err := ecies.GenerateKey(rand.Reader, DefaultCurve, nil)
            if err != nil {
                t.Fatal(err)
            }
            bogusGrantees = append(bogusGrantees, hex.EncodeToString(crypto.CompressPubkey(&prv.ExportECDSA().PublicKey)))
        }
        r2 := gorand.New(gorand.NewSource(time.Now().UnixNano()))
        for i := 0; i < len(grantees); i++ {
            insertAtIdx := r2.Intn(len(bogusGrantees))
            bogusGrantees = append(bogusGrantees[:insertAtIdx], append([]string{grantees[i]}, bogusGrantees[insertAtIdx:]...)...)
        }
        grantees = bogusGrantees
    }

    granteesPubkeyListFile, err := ioutil.TempFile("", "grantees-pubkey-list")
    if err != nil {
        t.Fatal(err)
    }
    defer granteesPubkeyListFile.Close()
    defer os.Remove(granteesPubkeyListFile.Name())

    _, err = granteesPubkeyListFile.WriteString(strings.Join(grantees, "\n"))
    if err != nil {
        t.Fatal(err)
    }

    publisherDir, err := ioutil.TempDir("", "swarm-account-dir-temp")
    if err != nil {
        t.Fatal(err)
    }

    passFile, err := ioutil.TempFile("", "swarm-test")
    if err != nil {
        t.Fatal(err)
    }
    defer passFile.Close()
    defer os.Remove(passFile.Name())
    _, err = io.WriteString(passFile, testPassphrase)
    if err != nil {
        t.Fatal(err)
    }

    _, publisherAccount := getTestAccount(t, publisherDir)
    up = runSwarm(t,
        "--bzzaccount",
        publisherAccount.Address.String(),
        "--password",
        passFile.Name(),
        "--datadir",
        publisherDir,
        "--bzzapi",
        cluster.Nodes[0].URL,
        "access",
        "new",
        "act",
        "--grant-keys",
        granteesPubkeyListFile.Name(),
        ref,
    )

    _, matches = up.ExpectRegexp(`[a-f\d]{64}`)
    up.ExpectExit()

    if len(matches) == 0 {
        t.Fatalf("stdout not matched")
    }
    hash := matches[0]
    m, _, err := client.DownloadManifest(hash)
    if err != nil {
        t.Fatalf("unmarshal manifest: %v", err)
    }

    if len(m.Entries) != 1 {
        t.Fatalf("expected one manifest entry, got %v", len(m.Entries))
    }

    e := m.Entries[0]

    ct := "application/bzz-manifest+json"
    if e.ContentType != ct {
        t.Errorf("expected %q content type, got %q", ct, e.ContentType)
    }

    if e.Access == nil {
        t.Fatal("manifest access is nil")
    }

    a := e.Access

    if a.Type != "act" {
        t.Fatalf(`got access type %q, expected "act"`, a.Type)
    }
    if len(a.Salt) < 32 {
        t.Fatalf(`got salt with length %v, expected not less the 32 bytes`, len(a.Salt))
    }
    if a.KdfParams != nil {
        t.Fatal("manifest access kdf params should be nil")
    }

    httpClient := &http.Client{}

    // all nodes except the skipped node should be able to decrypt the content
    for i, node := range cluster.Nodes {
        log.Debug("trying to fetch from node", "node index", i)

        url := node.URL + "/" + "bzz:/" + hash
        response, err := httpClient.Get(url)
        if err != nil {
            t.Fatal(err)
        }
        log.Debug("got response from node", "response code", response.StatusCode)

        if i == nodeToSkip {
            log.Debug("reached node to skip", "status code", response.StatusCode)

            if response.StatusCode != http.StatusUnauthorized {
                t.Fatalf("should be a 401")
            }

            continue
        }

        if response.StatusCode != http.StatusOK {
            t.Fatal("should be a 200")
        }
        d, err := ioutil.ReadAll(response.Body)
        if err != nil {
            t.Fatal(err)
        }
        if string(d) != data {
            t.Errorf("expected decrypted data %q, got %q", data, string(d))
        }
    }
}

// TestKeypairSanity is a sanity test for the crypto scheme for ACT. it asserts the correct shared secret according to
// the specs at https://github.com/ethersphere/swarm-docs/blob/eb857afda906c6e7bb90d37f3f334ccce5eef230/act.md
func TestKeypairSanity(t *testing.T) {
    salt := make([]byte, 32)
    if _, err := io.ReadFull(rand.Reader, salt); err != nil {
        t.Fatalf("reading from crypto/rand failed: %v", err.Error())
    }
    sharedSecret := "a85586744a1ddd56a7ed9f33fa24f40dd745b3a941be296a0d60e329dbdb896d"

    for i, v := range []struct {
        publisherPriv string
        granteePub    string
    }{
        {
            publisherPriv: "ec5541555f3bc6376788425e9d1a62f55a82901683fd7062c5eddcc373a73459",
            granteePub:    "0226f213613e843a413ad35b40f193910d26eb35f00154afcde9ded57479a6224a",
        },
        {
            publisherPriv: "70c7a73011aa56584a0009ab874794ee7e5652fd0c6911cd02f8b6267dd82d2d",
            granteePub:    "02e6f8d5e28faaa899744972bb847b6eb805a160494690c9ee7197ae9f619181db",
        },
    } {
        b, _ := hex.DecodeString(v.granteePub)
        granteePub, _ := crypto.DecompressPubkey(b)
        publisherPrivate, _ := crypto.HexToECDSA(v.publisherPriv)

        ssKey, err := api.NewSessionKeyPK(publisherPrivate, granteePub, salt)
        if err != nil {
            t.Fatal(err)
        }

        hasher := sha3.NewKeccak256()
        hasher.Write(salt)
        shared, err := hex.DecodeString(sharedSecret)
        if err != nil {
            t.Fatal(err)
        }
        hasher.Write(shared)
        sum := hasher.Sum(nil)

        if !bytes.Equal(ssKey, sum) {
            t.Fatalf("%d: got a session key mismatch", i)
        }
    }
}