// Copyright 2019 The go-ethereum Authors // This file is part of the go-ethereum library. // // The go-ethereum library is free software: you can redistribute it and/or modify // it under the terms of the GNU Lesser General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // The go-ethereum library 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 Lesser General Public License for more details. // // You should have received a copy of the GNU Lesser General Public License // along with the go-ethereum library. If not, see . package stream import ( "context" "fmt" "reflect" "sort" "sync" "testing" "time" "github.com/ethereum/go-ethereum/node" "github.com/ethereum/go-ethereum/p2p/enode" "github.com/ethereum/go-ethereum/p2p/simulations/adapters" "github.com/ethereum/go-ethereum/swarm/chunk" "github.com/ethereum/go-ethereum/swarm/network" "github.com/ethereum/go-ethereum/swarm/network/simulation" "github.com/ethereum/go-ethereum/swarm/state" ) // TestSyncSubscriptionsDiff validates the output of syncSubscriptionsDiff // function for various arguments. func TestSyncSubscriptionsDiff(t *testing.T) { max := network.NewKadParams().MaxProxDisplay for _, tc := range []struct { po, prevDepth, newDepth int subBins, quitBins []int }{ { po: 0, prevDepth: -1, newDepth: 0, subBins: []int{0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16}, }, { po: 1, prevDepth: -1, newDepth: 0, subBins: []int{0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16}, }, { po: 2, prevDepth: -1, newDepth: 0, subBins: []int{0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16}, }, { po: 0, prevDepth: -1, newDepth: 1, subBins: []int{0}, }, { po: 1, prevDepth: -1, newDepth: 1, subBins: []int{1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16}, }, { po: 2, prevDepth: -1, newDepth: 2, subBins: []int{2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16}, }, { po: 3, prevDepth: -1, newDepth: 2, subBins: []int{2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16}, }, { po: 1, prevDepth: -1, newDepth: 2, subBins: []int{1}, }, { po: 0, prevDepth: 0, newDepth: 0, // 0-16 -> 0-16 }, { po: 1, prevDepth: 0, newDepth: 0, // 0-16 -> 0-16 }, { po: 0, prevDepth: 0, newDepth: 1, // 0-16 -> 0 quitBins: []int{1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16}, }, { po: 0, prevDepth: 0, newDepth: 2, // 0-16 -> 0 quitBins: []int{1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16}, }, { po: 1, prevDepth: 0, newDepth: 1, // 0-16 -> 1-16 quitBins: []int{0}, }, { po: 1, prevDepth: 1, newDepth: 0, // 1-16 -> 0-16 subBins: []int{0}, }, { po: 4, prevDepth: 0, newDepth: 1, // 0-16 -> 1-16 quitBins: []int{0}, }, { po: 4, prevDepth: 0, newDepth: 4, // 0-16 -> 4-16 quitBins: []int{0, 1, 2, 3}, }, { po: 4, prevDepth: 0, newDepth: 5, // 0-16 -> 4 quitBins: []int{0, 1, 2, 3, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16}, }, { po: 4, prevDepth: 5, newDepth: 0, // 4 -> 0-16 subBins: []int{0, 1, 2, 3, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16}, }, { po: 4, prevDepth: 5, newDepth: 6, // 4 -> 4 }, } { subBins, quitBins := syncSubscriptionsDiff(tc.po, tc.prevDepth, tc.newDepth, max) if fmt.Sprint(subBins) != fmt.Sprint(tc.subBins) { t.Errorf("po: %v, prevDepth: %v, newDepth: %v: got subBins %v, want %v", tc.po, tc.prevDepth, tc.newDepth, subBins, tc.subBins) } if fmt.Sprint(quitBins) != fmt.Sprint(tc.quitBins) { t.Errorf("po: %v, prevDepth: %v, newDepth: %v: got quitBins %v, want %v", tc.po, tc.prevDepth, tc.newDepth, quitBins, tc.quitBins) } } } // TestUpdateSyncingSubscriptions validates that syncing subscriptions are correctly // made on initial node connections and that subscriptions are correctly changed // when kademlia neighbourhood depth is changed by connecting more nodes. func TestUpdateSyncingSubscriptions(t *testing.T) { sim := simulation.New(map[string]simulation.ServiceFunc{ "streamer": func(ctx *adapters.ServiceContext, bucket *sync.Map) (s node.Service, cleanup func(), err error) { addr, netStore, delivery, clean, err := newNetStoreAndDelivery(ctx, bucket) if err != nil { return nil, nil, err } r := NewRegistry(addr.ID(), delivery, netStore, state.NewInmemoryStore(), &RegistryOptions{ SyncUpdateDelay: 100 * time.Millisecond, Syncing: SyncingAutoSubscribe, }, nil) cleanup = func() { r.Close() clean() } bucket.Store("bzz-address", addr) return r, cleanup, nil }, }) defer sim.Close() ctx, cancel := context.WithTimeout(context.Background(), 3*time.Minute) defer cancel() result := sim.Run(ctx, func(ctx context.Context, sim *simulation.Simulation) (err error) { // initial nodes, first one as pivot center of the start ids, err := sim.AddNodesAndConnectStar(10) if err != nil { return err } // pivot values pivotRegistryID := ids[0] pivotRegistry := sim.Service("streamer", pivotRegistryID).(*Registry) pivotKademlia := pivotRegistry.delivery.kad // nodes proximities from the pivot node nodeProximities := make(map[string]int) for _, id := range ids[1:] { bzzAddr, ok := sim.NodeItem(id, "bzz-address") if !ok { t.Fatal("no bzz address for node") } nodeProximities[id.String()] = chunk.Proximity(pivotKademlia.BaseAddr(), bzzAddr.(*network.BzzAddr).Over()) } // wait until sync subscriptions are done for all nodes waitForSubscriptions(t, pivotRegistry, ids[1:]...) // check initial sync streams err = checkSyncStreamsWithRetry(pivotRegistry, nodeProximities) if err != nil { return err } // add more nodes until the depth is changed prevDepth := pivotKademlia.NeighbourhoodDepth() var noDepthChangeChecked bool // true it there was a check when no depth is changed for { ids, err := sim.AddNodes(5) if err != nil { return err } // add new nodes to sync subscriptions check for _, id := range ids { bzzAddr, ok := sim.NodeItem(id, "bzz-address") if !ok { t.Fatal("no bzz address for node") } nodeProximities[id.String()] = chunk.Proximity(pivotKademlia.BaseAddr(), bzzAddr.(*network.BzzAddr).Over()) } err = sim.Net.ConnectNodesStar(ids, pivotRegistryID) if err != nil { return err } waitForSubscriptions(t, pivotRegistry, ids...) newDepth := pivotKademlia.NeighbourhoodDepth() // depth is not changed, check if streams are still correct if newDepth == prevDepth { err = checkSyncStreamsWithRetry(pivotRegistry, nodeProximities) if err != nil { return err } noDepthChangeChecked = true } // do the final check when depth is changed and // there has been at least one check // for the case when depth is not changed if newDepth != prevDepth && noDepthChangeChecked { // check sync streams for changed depth return checkSyncStreamsWithRetry(pivotRegistry, nodeProximities) } prevDepth = newDepth } }) if result.Error != nil { t.Fatal(result.Error) } } // waitForSubscriptions is a test helper function that blocks until // stream server subscriptions are established on the provided registry // to the nodes with provided IDs. func waitForSubscriptions(t *testing.T, r *Registry, ids ...enode.ID) { t.Helper() for retries := 0; retries < 100; retries++ { subs := r.api.GetPeerServerSubscriptions() if allSubscribed(subs, ids) { return } time.Sleep(50 * time.Millisecond) } t.Fatalf("missing subscriptions") } // allSubscribed returns true if nodes with ids have subscriptions // in provided subs map. func allSubscribed(subs map[string][]string, ids []enode.ID) bool { for _, id := range ids { if s, ok := subs[id.String()]; !ok || len(s) == 0 { return false } } return true } // checkSyncStreamsWithRetry is calling checkSyncStreams with retries. func checkSyncStreamsWithRetry(r *Registry, nodeProximities map[string]int) (err error) { for retries := 0; retries < 5; retries++ { err = checkSyncStreams(r, nodeProximities) if err == nil { return nil } time.Sleep(500 * time.Millisecond) } return err } // checkSyncStreams validates that registry contains expected sync // subscriptions to nodes with proximities in a map nodeProximities. func checkSyncStreams(r *Registry, nodeProximities map[string]int) error { depth := r.delivery.kad.NeighbourhoodDepth() maxPO := r.delivery.kad.MaxProxDisplay for id, po := range nodeProximities { wantStreams := syncStreams(po, depth, maxPO) gotStreams := nodeStreams(r, id) if r.getPeer(enode.HexID(id)) == nil { // ignore removed peer continue } if !reflect.DeepEqual(gotStreams, wantStreams) { return fmt.Errorf("node %s got streams %v, want %v", id, gotStreams, wantStreams) } } return nil } // syncStreams returns expected sync streams that need to be // established between a node with kademlia neighbourhood depth // and a node with proximity order po. func syncStreams(po, depth, maxPO int) (streams []string) { start, end := syncBins(po, depth, maxPO) for bin := start; bin < end; bin++ { streams = append(streams, NewStream("SYNC", FormatSyncBinKey(uint8(bin)), false).String()) streams = append(streams, NewStream("SYNC", FormatSyncBinKey(uint8(bin)), true).String()) } return streams } // nodeStreams returns stream server subscriptions on a registry // to the peer with provided id. func nodeStreams(r *Registry, id string) []string { streams := r.api.GetPeerServerSubscriptions()[id] sort.Strings(streams) return streams }