mirror of
https://github.com/octoleo/syncthing.git
synced 2025-01-24 23:48:26 +00:00
Use persistent (leveldb) storage
This commit is contained in:
parent
6ea8e2525a
commit
db6d3b495b
@ -5,6 +5,7 @@
|
|||||||
package main
|
package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"bytes"
|
||||||
"encoding/binary"
|
"encoding/binary"
|
||||||
"encoding/hex"
|
"encoding/hex"
|
||||||
"flag"
|
"flag"
|
||||||
@ -13,6 +14,7 @@ import (
|
|||||||
"log"
|
"log"
|
||||||
"net"
|
"net"
|
||||||
"os"
|
"os"
|
||||||
|
"path/filepath"
|
||||||
"sync"
|
"sync"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
@ -20,20 +22,13 @@ import (
|
|||||||
"github.com/juju/ratelimit"
|
"github.com/juju/ratelimit"
|
||||||
"github.com/syncthing/syncthing/discover"
|
"github.com/syncthing/syncthing/discover"
|
||||||
"github.com/syncthing/syncthing/protocol"
|
"github.com/syncthing/syncthing/protocol"
|
||||||
|
"github.com/syndtr/goleveldb/leveldb"
|
||||||
|
"github.com/syndtr/goleveldb/leveldb/opt"
|
||||||
)
|
)
|
||||||
|
|
||||||
type node struct {
|
const cacheLimitSeconds = 3600
|
||||||
addresses []address
|
|
||||||
updated time.Time
|
|
||||||
}
|
|
||||||
|
|
||||||
type address struct {
|
|
||||||
ip []byte
|
|
||||||
port uint16
|
|
||||||
}
|
|
||||||
|
|
||||||
var (
|
var (
|
||||||
nodes = make(map[protocol.NodeID]node)
|
|
||||||
lock sync.Mutex
|
lock sync.Mutex
|
||||||
queries = 0
|
queries = 0
|
||||||
announces = 0
|
announces = 0
|
||||||
@ -52,15 +47,17 @@ func main() {
|
|||||||
var timestamp bool
|
var timestamp bool
|
||||||
var statsIntv int
|
var statsIntv int
|
||||||
var statsFile string
|
var statsFile string
|
||||||
|
var dbDir string
|
||||||
|
|
||||||
flag.StringVar(&listen, "listen", ":22026", "Listen address")
|
flag.StringVar(&listen, "listen", ":22026", "Listen address")
|
||||||
flag.BoolVar(&debug, "debug", false, "Enable debug output")
|
flag.BoolVar(&debug, "debug", false, "Enable debug output")
|
||||||
flag.BoolVar(×tamp, "timestamp", true, "Timestamp the log output")
|
flag.BoolVar(×tamp, "timestamp", true, "Timestamp the log output")
|
||||||
flag.IntVar(&statsIntv, "stats-intv", 0, "Statistics output interval (s)")
|
flag.IntVar(&statsIntv, "stats-intv", 0, "Statistics output interval (s)")
|
||||||
flag.StringVar(&statsFile, "stats-file", "/var/log/discosrv.stats", "Statistics file name")
|
flag.StringVar(&statsFile, "stats-file", "/var/discosrv/stats", "Statistics file name")
|
||||||
flag.IntVar(&lruSize, "limit-cache", lruSize, "Limiter cache entries")
|
flag.IntVar(&lruSize, "limit-cache", lruSize, "Limiter cache entries")
|
||||||
flag.IntVar(&limitAvg, "limit-avg", limitAvg, "Allowed average package rate, per 10 s")
|
flag.IntVar(&limitAvg, "limit-avg", limitAvg, "Allowed average package rate, per 10 s")
|
||||||
flag.IntVar(&limitBurst, "limit-burst", limitBurst, "Allowed burst size, packets")
|
flag.IntVar(&limitBurst, "limit-burst", limitBurst, "Allowed burst size, packets")
|
||||||
|
flag.StringVar(&dbDir, "db-dir", "/var/discosrv/db", "Database directory")
|
||||||
flag.Parse()
|
flag.Parse()
|
||||||
|
|
||||||
limiter = lru.New(lruSize)
|
limiter = lru.New(lruSize)
|
||||||
@ -76,9 +73,29 @@ func main() {
|
|||||||
log.Fatal(err)
|
log.Fatal(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
if statsIntv > 0 {
|
parentDir := filepath.Dir(dbDir)
|
||||||
go logStats(statsFile, statsIntv)
|
if _, err := os.Stat(parentDir); err != nil && os.IsNotExist(err) {
|
||||||
|
err = os.MkdirAll(parentDir, 0755)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
db, err := leveldb.OpenFile(dbDir, &opt.Options{CachedOpenFiles: 32})
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
statsLog, err := os.OpenFile(statsFile, os.O_WRONLY|os.O_CREATE|os.O_APPEND, 0644)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if statsIntv > 0 {
|
||||||
|
go logStats(statsLog, statsIntv)
|
||||||
|
}
|
||||||
|
|
||||||
|
go clean(statsLog, db)
|
||||||
|
|
||||||
var buf = make([]byte, 1024)
|
var buf = make([]byte, 1024)
|
||||||
for {
|
for {
|
||||||
@ -104,10 +121,10 @@ func main() {
|
|||||||
|
|
||||||
switch magic {
|
switch magic {
|
||||||
case discover.AnnouncementMagic:
|
case discover.AnnouncementMagic:
|
||||||
handleAnnounceV2(addr, buf)
|
handleAnnounceV2(db, addr, buf)
|
||||||
|
|
||||||
case discover.QueryMagic:
|
case discover.QueryMagic:
|
||||||
handleQueryV2(conn, addr, buf)
|
handleQueryV2(db, conn, addr, buf)
|
||||||
|
|
||||||
default:
|
default:
|
||||||
lock.Lock()
|
lock.Lock()
|
||||||
@ -145,7 +162,7 @@ func limit(addr *net.UDPAddr) bool {
|
|||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
func handleAnnounceV2(addr *net.UDPAddr, buf []byte) {
|
func handleAnnounceV2(db *leveldb.DB, addr *net.UDPAddr, buf []byte) {
|
||||||
var pkt discover.Announce
|
var pkt discover.Announce
|
||||||
err := pkt.UnmarshalXDR(buf)
|
err := pkt.UnmarshalXDR(buf)
|
||||||
if err != nil && err != io.EOF {
|
if err != nil && err != io.EOF {
|
||||||
@ -167,6 +184,7 @@ func handleAnnounceV2(addr *net.UDPAddr, buf []byte) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
var addrs []address
|
var addrs []address
|
||||||
|
now := time.Now().Unix()
|
||||||
for _, addr := range pkt.This.Addresses {
|
for _, addr := range pkt.This.Addresses {
|
||||||
tip := addr.IP
|
tip := addr.IP
|
||||||
if len(tip) == 0 {
|
if len(tip) == 0 {
|
||||||
@ -175,14 +193,10 @@ func handleAnnounceV2(addr *net.UDPAddr, buf []byte) {
|
|||||||
addrs = append(addrs, address{
|
addrs = append(addrs, address{
|
||||||
ip: tip,
|
ip: tip,
|
||||||
port: addr.Port,
|
port: addr.Port,
|
||||||
|
seen: now,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
node := node{
|
|
||||||
addresses: addrs,
|
|
||||||
updated: time.Now(),
|
|
||||||
}
|
|
||||||
|
|
||||||
var id protocol.NodeID
|
var id protocol.NodeID
|
||||||
if len(pkt.This.ID) == 32 {
|
if len(pkt.This.ID) == 32 {
|
||||||
// Raw node ID
|
// Raw node ID
|
||||||
@ -191,12 +205,10 @@ func handleAnnounceV2(addr *net.UDPAddr, buf []byte) {
|
|||||||
id.UnmarshalText(pkt.This.ID)
|
id.UnmarshalText(pkt.This.ID)
|
||||||
}
|
}
|
||||||
|
|
||||||
lock.Lock()
|
update(db, id, addrs)
|
||||||
nodes[id] = node
|
|
||||||
lock.Unlock()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func handleQueryV2(conn *net.UDPConn, addr *net.UDPAddr, buf []byte) {
|
func handleQueryV2(db *leveldb.DB, conn *net.UDPConn, addr *net.UDPAddr, buf []byte) {
|
||||||
var pkt discover.Query
|
var pkt discover.Query
|
||||||
err := pkt.UnmarshalXDR(buf)
|
err := pkt.UnmarshalXDR(buf)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -217,24 +229,33 @@ func handleQueryV2(conn *net.UDPConn, addr *net.UDPAddr, buf []byte) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
lock.Lock()
|
lock.Lock()
|
||||||
node, ok := nodes[id]
|
|
||||||
queries++
|
queries++
|
||||||
lock.Unlock()
|
lock.Unlock()
|
||||||
|
|
||||||
if ok && len(node.addresses) > 0 {
|
addrs := get(db, id)
|
||||||
|
|
||||||
|
now := time.Now().Unix()
|
||||||
|
if len(addrs) > 0 {
|
||||||
ann := discover.Announce{
|
ann := discover.Announce{
|
||||||
Magic: discover.AnnouncementMagic,
|
Magic: discover.AnnouncementMagic,
|
||||||
This: discover.Node{
|
This: discover.Node{
|
||||||
ID: pkt.NodeID,
|
ID: pkt.NodeID,
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
for _, addr := range node.addresses {
|
for _, addr := range addrs {
|
||||||
|
if now-addr.seen > cacheLimitSeconds {
|
||||||
|
continue
|
||||||
|
}
|
||||||
ann.This.Addresses = append(ann.This.Addresses, discover.Address{IP: addr.ip, Port: addr.port})
|
ann.This.Addresses = append(ann.This.Addresses, discover.Address{IP: addr.ip, Port: addr.port})
|
||||||
}
|
}
|
||||||
if debug {
|
if debug {
|
||||||
log.Printf("-> %v %#v", addr, pkt)
|
log.Printf("-> %v %#v", addr, pkt)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if len(ann.This.Addresses) == 0 {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
tb := ann.MarshalXDR()
|
tb := ann.MarshalXDR()
|
||||||
_, _, err = conn.WriteMsgUDP(tb, nil, addr)
|
_, _, err = conn.WriteMsgUDP(tb, nil, addr)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -255,27 +276,14 @@ func next(intv int) time.Time {
|
|||||||
return t1
|
return t1
|
||||||
}
|
}
|
||||||
|
|
||||||
func logStats(file string, intv int) {
|
func logStats(statsLog io.Writer, intv int) {
|
||||||
f, err := os.OpenFile(file, os.O_WRONLY|os.O_CREATE|os.O_APPEND, 0644)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatal(err)
|
|
||||||
}
|
|
||||||
for {
|
for {
|
||||||
t := next(intv)
|
t := next(intv)
|
||||||
|
|
||||||
lock.Lock()
|
lock.Lock()
|
||||||
|
|
||||||
var deleted = 0
|
fmt.Fprintf(statsLog, "%d Queries:%d Answered:%d Announces:%d Unknown:%d Limited:%d\n",
|
||||||
for id, node := range nodes {
|
t.Unix(), queries, answered, announces, unknowns, limited)
|
||||||
if time.Since(node.updated) > 60*time.Minute {
|
|
||||||
delete(nodes, id)
|
|
||||||
deleted++
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fmt.Fprintf(f, "%d Nr:%d Ne:%d Qt:%d Qa:%d A:%d U:%d Lq:%d Lc:%d\n",
|
|
||||||
t.Unix(), len(nodes), deleted, queries, answered, announces, unknowns, limited, limiter.Len())
|
|
||||||
f.Sync()
|
|
||||||
|
|
||||||
queries = 0
|
queries = 0
|
||||||
announces = 0
|
announces = 0
|
||||||
@ -286,3 +294,76 @@ func logStats(file string, intv int) {
|
|||||||
lock.Unlock()
|
lock.Unlock()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func get(db *leveldb.DB, id protocol.NodeID) []address {
|
||||||
|
var addrs addressList
|
||||||
|
val, err := db.Get(id[:], nil)
|
||||||
|
if err == nil {
|
||||||
|
addrs.UnmarshalXDR(val)
|
||||||
|
}
|
||||||
|
return addrs.addresses
|
||||||
|
}
|
||||||
|
|
||||||
|
func update(db *leveldb.DB, id protocol.NodeID, addrs []address) {
|
||||||
|
var newAddrs addressList
|
||||||
|
|
||||||
|
val, err := db.Get(id[:], nil)
|
||||||
|
if err == nil {
|
||||||
|
newAddrs.UnmarshalXDR(val)
|
||||||
|
}
|
||||||
|
|
||||||
|
nextAddr:
|
||||||
|
for _, newAddr := range addrs {
|
||||||
|
for i, exAddr := range newAddrs.addresses {
|
||||||
|
if bytes.Compare(newAddr.ip, exAddr.ip) == 0 {
|
||||||
|
newAddrs.addresses[i] = newAddr
|
||||||
|
continue nextAddr
|
||||||
|
}
|
||||||
|
}
|
||||||
|
newAddrs.addresses = append(newAddrs.addresses, newAddr)
|
||||||
|
}
|
||||||
|
|
||||||
|
db.Put(id[:], newAddrs.MarshalXDR(), nil)
|
||||||
|
}
|
||||||
|
|
||||||
|
func clean(statsLog io.Writer, db *leveldb.DB) {
|
||||||
|
for {
|
||||||
|
now := time.Now()
|
||||||
|
nowSecs := now.Unix()
|
||||||
|
|
||||||
|
var kept, deleted int64
|
||||||
|
iter := db.NewIterator(nil, nil)
|
||||||
|
for iter.Next() {
|
||||||
|
var addrs addressList
|
||||||
|
addrs.UnmarshalXDR(iter.Value())
|
||||||
|
|
||||||
|
// Remove expired addresses
|
||||||
|
newAddrs := addrs.addresses
|
||||||
|
for i := 0; i < len(newAddrs); i++ {
|
||||||
|
if nowSecs-newAddrs[i].seen > cacheLimitSeconds {
|
||||||
|
newAddrs[i] = newAddrs[len(newAddrs)-1]
|
||||||
|
newAddrs = newAddrs[:len(newAddrs)-1]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Delete empty records
|
||||||
|
if len(newAddrs) == 0 {
|
||||||
|
db.Delete(iter.Key(), nil)
|
||||||
|
deleted++
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
// Update changed records
|
||||||
|
if len(newAddrs) != len(addrs.addresses) {
|
||||||
|
addrs.addresses = newAddrs
|
||||||
|
db.Put(iter.Key(), addrs.MarshalXDR(), nil)
|
||||||
|
}
|
||||||
|
kept++
|
||||||
|
}
|
||||||
|
iter.Release()
|
||||||
|
|
||||||
|
fmt.Fprintf(statsLog, "%d Kept:%d Deleted:%d Took:%0.04fs\n", nowSecs, kept, deleted, time.Since(now).Seconds())
|
||||||
|
|
||||||
|
time.Sleep(cacheLimitSeconds * time.Second / 2)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
15
cmd/discosrv/types.go
Normal file
15
cmd/discosrv/types.go
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
// Copyright (C) 2014 Jakob Borg and Contributors (see the CONTRIBUTORS file).
|
||||||
|
// All rights reserved. Use of this source code is governed by an MIT-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
package main
|
||||||
|
|
||||||
|
type address struct {
|
||||||
|
ip []byte
|
||||||
|
port uint16
|
||||||
|
seen int64 // epoch seconds
|
||||||
|
}
|
||||||
|
|
||||||
|
type addressList struct {
|
||||||
|
addresses []address
|
||||||
|
}
|
151
cmd/discosrv/types_xdr.go
Normal file
151
cmd/discosrv/types_xdr.go
Normal file
@ -0,0 +1,151 @@
|
|||||||
|
// Copyright (C) 2014 Jakob Borg and Contributors (see the CONTRIBUTORS file).
|
||||||
|
// All rights reserved. Use of this source code is governed by an MIT-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
// ************************************************************
|
||||||
|
// This file is automatically generated by genxdr. Do not edit.
|
||||||
|
// ************************************************************
|
||||||
|
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"io"
|
||||||
|
|
||||||
|
"github.com/calmh/xdr"
|
||||||
|
)
|
||||||
|
|
||||||
|
/*
|
||||||
|
|
||||||
|
address Structure:
|
||||||
|
|
||||||
|
0 1 2 3
|
||||||
|
0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
|
||||||
|
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
||||||
|
| Length of ip |
|
||||||
|
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
||||||
|
/ /
|
||||||
|
\ ip (variable length) \
|
||||||
|
/ /
|
||||||
|
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
||||||
|
| 0x0000 | port |
|
||||||
|
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
||||||
|
| |
|
||||||
|
+ seen (64 bits) +
|
||||||
|
| |
|
||||||
|
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
||||||
|
|
||||||
|
|
||||||
|
struct address {
|
||||||
|
opaque ip<>;
|
||||||
|
unsigned int port;
|
||||||
|
hyper seen;
|
||||||
|
}
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
func (o address) EncodeXDR(w io.Writer) (int, error) {
|
||||||
|
var xw = xdr.NewWriter(w)
|
||||||
|
return o.encodeXDR(xw)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (o address) MarshalXDR() []byte {
|
||||||
|
return o.AppendXDR(make([]byte, 0, 128))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (o address) AppendXDR(bs []byte) []byte {
|
||||||
|
var aw = xdr.AppendWriter(bs)
|
||||||
|
var xw = xdr.NewWriter(&aw)
|
||||||
|
o.encodeXDR(xw)
|
||||||
|
return []byte(aw)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (o address) encodeXDR(xw *xdr.Writer) (int, error) {
|
||||||
|
xw.WriteBytes(o.ip)
|
||||||
|
xw.WriteUint16(o.port)
|
||||||
|
xw.WriteUint64(uint64(o.seen))
|
||||||
|
return xw.Tot(), xw.Error()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (o *address) DecodeXDR(r io.Reader) error {
|
||||||
|
xr := xdr.NewReader(r)
|
||||||
|
return o.decodeXDR(xr)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (o *address) UnmarshalXDR(bs []byte) error {
|
||||||
|
var br = bytes.NewReader(bs)
|
||||||
|
var xr = xdr.NewReader(br)
|
||||||
|
return o.decodeXDR(xr)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (o *address) decodeXDR(xr *xdr.Reader) error {
|
||||||
|
o.ip = xr.ReadBytes()
|
||||||
|
o.port = xr.ReadUint16()
|
||||||
|
o.seen = int64(xr.ReadUint64())
|
||||||
|
return xr.Error()
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
|
||||||
|
addressList Structure:
|
||||||
|
|
||||||
|
0 1 2 3
|
||||||
|
0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
|
||||||
|
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
||||||
|
| Number of addresses |
|
||||||
|
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
||||||
|
/ /
|
||||||
|
\ Zero or more address Structures \
|
||||||
|
/ /
|
||||||
|
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
||||||
|
|
||||||
|
|
||||||
|
struct addressList {
|
||||||
|
address addresses<>;
|
||||||
|
}
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
func (o addressList) EncodeXDR(w io.Writer) (int, error) {
|
||||||
|
var xw = xdr.NewWriter(w)
|
||||||
|
return o.encodeXDR(xw)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (o addressList) MarshalXDR() []byte {
|
||||||
|
return o.AppendXDR(make([]byte, 0, 128))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (o addressList) AppendXDR(bs []byte) []byte {
|
||||||
|
var aw = xdr.AppendWriter(bs)
|
||||||
|
var xw = xdr.NewWriter(&aw)
|
||||||
|
o.encodeXDR(xw)
|
||||||
|
return []byte(aw)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (o addressList) encodeXDR(xw *xdr.Writer) (int, error) {
|
||||||
|
xw.WriteUint32(uint32(len(o.addresses)))
|
||||||
|
for i := range o.addresses {
|
||||||
|
o.addresses[i].encodeXDR(xw)
|
||||||
|
}
|
||||||
|
return xw.Tot(), xw.Error()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (o *addressList) DecodeXDR(r io.Reader) error {
|
||||||
|
xr := xdr.NewReader(r)
|
||||||
|
return o.decodeXDR(xr)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (o *addressList) UnmarshalXDR(bs []byte) error {
|
||||||
|
var br = bytes.NewReader(bs)
|
||||||
|
var xr = xdr.NewReader(br)
|
||||||
|
return o.decodeXDR(xr)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (o *addressList) decodeXDR(xr *xdr.Reader) error {
|
||||||
|
_addressesSize := int(xr.ReadUint32())
|
||||||
|
o.addresses = make([]address, _addressesSize)
|
||||||
|
for i := range o.addresses {
|
||||||
|
(&o.addresses[i]).decodeXDR(xr)
|
||||||
|
}
|
||||||
|
return xr.Error()
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user