2016-02-01 01:56:10 -08:00
|
|
|
package common
|
|
|
|
|
2016-02-15 01:13:44 -08:00
|
|
|
/*
|
|
|
|
I2P RouterInfo
|
2016-06-16 23:17:21 -07:00
|
|
|
https://geti2p.net/spec/common-structures#routerinfo
|
2016-02-15 01:13:44 -08:00
|
|
|
Accurate for version 0.9.24
|
|
|
|
|
|
|
|
+----+----+----+----+----+----+----+----+
|
|
|
|
| router_ident |
|
|
|
|
+ +
|
|
|
|
| |
|
|
|
|
~ ~
|
|
|
|
~ ~
|
|
|
|
| |
|
|
|
|
+----+----+----+----+----+----+----+----+
|
|
|
|
| published |
|
|
|
|
+----+----+----+----+----+----+----+----+
|
|
|
|
|size| RouterAddress 0 |
|
|
|
|
+----+ +
|
|
|
|
| |
|
|
|
|
~ ~
|
|
|
|
~ ~
|
|
|
|
| |
|
|
|
|
+----+----+----+----+----+----+----+----+
|
|
|
|
| RouterAddress 1 |
|
|
|
|
+ +
|
|
|
|
| |
|
|
|
|
~ ~
|
|
|
|
~ ~
|
|
|
|
| |
|
|
|
|
+----+----+----+----+----+----+----+----+
|
|
|
|
| RouterAddress ($size-1) |
|
|
|
|
+ +
|
|
|
|
| |
|
|
|
|
~ ~
|
|
|
|
~ ~
|
|
|
|
| |
|
|
|
|
+----+----+----+----+-//-+----+----+----+
|
|
|
|
|psiz| options |
|
|
|
|
+----+----+----+----+-//-+----+----+----+
|
|
|
|
| signature |
|
|
|
|
+ +
|
|
|
|
| |
|
|
|
|
+ +
|
|
|
|
| |
|
|
|
|
+ +
|
|
|
|
| |
|
|
|
|
+ +
|
|
|
|
| |
|
|
|
|
+----+----+----+----+----+----+----+----+
|
|
|
|
|
|
|
|
router_ident :: RouterIdentity
|
|
|
|
length -> >= 387 bytes
|
|
|
|
|
|
|
|
published :: Date
|
|
|
|
length -> 8 bytes
|
|
|
|
|
|
|
|
size :: Integer
|
|
|
|
length -> 1 byte
|
|
|
|
The number of RouterAddresses to follow, 0-255
|
|
|
|
|
|
|
|
addresses :: [RouterAddress]
|
|
|
|
length -> varies
|
|
|
|
|
|
|
|
peer_size :: Integer
|
|
|
|
length -> 1 byte
|
|
|
|
The number of peer Hashes to follow, 0-255, unused, always zero
|
|
|
|
value -> 0
|
|
|
|
|
|
|
|
options :: Mapping
|
|
|
|
|
|
|
|
signature :: Signature
|
|
|
|
length -> 40 bytes
|
|
|
|
*/
|
|
|
|
|
2016-02-07 02:54:02 -08:00
|
|
|
import (
|
|
|
|
"errors"
|
2022-04-27 10:48:59 -04:00
|
|
|
|
2022-05-22 19:59:20 -04:00
|
|
|
. "github.com/go-i2p/go-i2p/lib/common/data"
|
|
|
|
. "github.com/go-i2p/go-i2p/lib/common/router_address"
|
|
|
|
. "github.com/go-i2p/go-i2p/lib/common/router_identity"
|
|
|
|
. "github.com/go-i2p/go-i2p/lib/common/signature"
|
2021-04-19 20:43:37 -04:00
|
|
|
log "github.com/sirupsen/logrus"
|
2016-02-07 02:54:02 -08:00
|
|
|
)
|
|
|
|
|
2022-05-09 20:43:24 -04:00
|
|
|
const ROUTER_INFO_MIN_SIZE = 439
|
|
|
|
|
|
|
|
type RouterInfo struct {
|
|
|
|
router_identity *RouterIdentity
|
|
|
|
published *Date
|
|
|
|
size *Integer
|
|
|
|
addresses []*RouterAddress
|
|
|
|
peer_size *Integer
|
|
|
|
options *Mapping
|
|
|
|
signature *Signature
|
|
|
|
}
|
|
|
|
|
|
|
|
//[]byte
|
2016-02-01 01:56:10 -08:00
|
|
|
|
2016-02-07 02:54:02 -08:00
|
|
|
//
|
2016-02-15 01:13:44 -08:00
|
|
|
// Read a RouterIdentity from the RouterInfo, returning the RouterIdentity and any errors
|
|
|
|
// encountered parsing the RouterIdentity.
|
2016-02-07 02:54:02 -08:00
|
|
|
//
|
2022-05-09 20:43:24 -04:00
|
|
|
func (router_info *RouterInfo) RouterIdentity() *RouterIdentity {
|
|
|
|
return router_info.router_identity
|
2016-02-01 01:56:10 -08:00
|
|
|
}
|
|
|
|
|
2016-08-17 09:19:56 -04:00
|
|
|
//
|
|
|
|
// Calculate this RouterInfo's Identity Hash (the sha256 of the RouterIdentity)
|
|
|
|
// returns error if the RouterIdentity is malformed
|
|
|
|
//
|
2022-05-09 20:43:24 -04:00
|
|
|
func (router_info *RouterInfo) IdentHash() Hash {
|
|
|
|
ri := router_info.RouterIdentity()
|
|
|
|
h := HashData(ri.KeysAndCert.Certificate().Data())
|
|
|
|
return h
|
2016-08-17 09:19:56 -04:00
|
|
|
}
|
|
|
|
|
2016-02-07 02:54:02 -08:00
|
|
|
//
|
2016-02-15 01:13:44 -08:00
|
|
|
// Return the Date the RouterInfo was published and any errors encountered parsing the RouterInfo.
|
2016-02-07 02:54:02 -08:00
|
|
|
//
|
2022-05-09 20:43:24 -04:00
|
|
|
func (router_info *RouterInfo) Published() *Date {
|
|
|
|
return router_info.published
|
2016-02-01 01:56:10 -08:00
|
|
|
}
|
|
|
|
|
2016-02-07 02:54:02 -08:00
|
|
|
//
|
2016-02-15 01:13:44 -08:00
|
|
|
// Return the Integer representing the number of RouterAddresses that are contained in this RouterInfo.
|
2016-02-07 02:54:02 -08:00
|
|
|
//
|
2022-05-09 20:43:24 -04:00
|
|
|
func (router_info *RouterInfo) RouterAddressCount() int {
|
|
|
|
return router_info.size.Int()
|
2016-02-01 01:56:10 -08:00
|
|
|
}
|
|
|
|
|
2016-02-07 02:54:02 -08:00
|
|
|
//
|
2016-02-15 01:13:44 -08:00
|
|
|
// Read the RouterAddresses inside this RouterInfo and return them in a slice, returning
|
|
|
|
// a partial list if data is missing.
|
2016-02-07 02:54:02 -08:00
|
|
|
//
|
2022-05-09 20:43:24 -04:00
|
|
|
func (router_info *RouterInfo) RouterAddresses() []*RouterAddress {
|
|
|
|
return router_info.addresses
|
2016-02-01 01:56:10 -08:00
|
|
|
}
|
|
|
|
|
2016-02-07 02:54:02 -08:00
|
|
|
//
|
|
|
|
// Return the PeerSize value, currently unused and always zero.
|
|
|
|
//
|
2022-05-09 20:43:24 -04:00
|
|
|
func (router_info *RouterInfo) PeerSize() int {
|
2016-02-07 02:54:02 -08:00
|
|
|
// Peer size is unused:
|
2016-06-16 23:17:21 -07:00
|
|
|
// https://geti2p.net/spec/common-structures#routeraddress
|
2016-02-01 01:56:10 -08:00
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
2016-02-07 02:54:02 -08:00
|
|
|
//
|
2016-02-13 21:00:29 -08:00
|
|
|
// Return the Options Mapping inside this RouterInfo.
|
2016-02-07 02:54:02 -08:00
|
|
|
//
|
2016-02-15 01:13:44 -08:00
|
|
|
func (router_info RouterInfo) Options() (mapping Mapping) {
|
2022-05-09 20:43:24 -04:00
|
|
|
return *router_info.options
|
2016-02-01 01:56:10 -08:00
|
|
|
}
|
|
|
|
|
2016-02-07 02:54:02 -08:00
|
|
|
//
|
2017-04-06 10:20:19 -04:00
|
|
|
// Return the signature of this router info
|
2016-02-07 02:54:02 -08:00
|
|
|
//
|
2016-02-15 01:13:44 -08:00
|
|
|
func (router_info RouterInfo) Signature() (signature Signature) {
|
2022-05-09 20:43:24 -04:00
|
|
|
return *router_info.signature
|
2016-02-01 01:56:10 -08:00
|
|
|
}
|
|
|
|
|
2016-02-07 02:54:02 -08:00
|
|
|
//
|
2016-02-15 01:13:44 -08:00
|
|
|
// Used during parsing to determine where in the RouterInfo the Mapping data begins.
|
2016-02-07 02:54:02 -08:00
|
|
|
//
|
2022-05-09 20:43:24 -04:00
|
|
|
/*func (router_info RouterInfo) optionsLocation() (location int) {
|
2016-07-02 17:22:14 -07:00
|
|
|
data, remainder, err := ReadRouterIdentity(router_info)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
location += len(data)
|
|
|
|
|
|
|
|
remainder_len := len(remainder)
|
|
|
|
if remainder_len < 9 {
|
|
|
|
log.WithFields(log.Fields{
|
|
|
|
"at": "(RouterInfo) optionsLocation",
|
|
|
|
"data_len": remainder_len,
|
|
|
|
"required_len": 9,
|
|
|
|
"reason": "not enough data",
|
|
|
|
}).Error("error parsing router info")
|
|
|
|
err = errors.New("error parsing router addresses: not enough data")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
location += 9
|
|
|
|
|
|
|
|
remaining := remainder[9:]
|
2016-02-01 01:56:10 -08:00
|
|
|
var router_address RouterAddress
|
2016-07-02 17:22:14 -07:00
|
|
|
var router_addresses []RouterAddress
|
|
|
|
addr_count, cerr := router_info.RouterAddressCount()
|
|
|
|
if cerr != nil {
|
|
|
|
err = cerr
|
|
|
|
return
|
|
|
|
}
|
2016-02-07 02:54:02 -08:00
|
|
|
for i := 0; i < addr_count; i++ {
|
2016-07-02 17:22:14 -07:00
|
|
|
router_address, remaining, err = ReadRouterAddress(remaining)
|
|
|
|
if err == nil {
|
|
|
|
location += len(router_address)
|
|
|
|
router_addresses = append(router_addresses, router_address)
|
|
|
|
}
|
2016-02-01 01:56:10 -08:00
|
|
|
}
|
2016-07-02 17:22:14 -07:00
|
|
|
location += 1
|
2016-02-15 01:13:44 -08:00
|
|
|
return
|
2022-05-09 20:43:24 -04:00
|
|
|
}*/
|
2016-02-01 01:56:10 -08:00
|
|
|
|
2016-02-07 02:54:02 -08:00
|
|
|
//
|
2016-02-15 01:13:44 -08:00
|
|
|
// Used during parsing to determine the size of the options in the RouterInfo.
|
2016-02-07 02:54:02 -08:00
|
|
|
//
|
2022-05-09 20:43:24 -04:00
|
|
|
/*func (router_info RouterInfo) optionsSize() (size int) {
|
2016-02-01 01:56:10 -08:00
|
|
|
head := router_info.optionsLocation()
|
2022-05-09 20:43:24 -04:00
|
|
|
s := Integer(router_info[head : head+2])
|
2022-04-27 10:48:59 -04:00
|
|
|
size = s.Int() + 2
|
2016-02-15 01:13:44 -08:00
|
|
|
return
|
2022-05-09 20:43:24 -04:00
|
|
|
}*/
|
|
|
|
|
|
|
|
func ReadRouterInfo(bytes []byte) (info RouterInfo, remainder []byte, err error) {
|
|
|
|
identity, remainder, err := NewRouterIdentity(bytes)
|
|
|
|
info.router_identity = identity
|
|
|
|
if err != nil {
|
|
|
|
log.WithFields(log.Fields{
|
|
|
|
"at": "(RouterInfo) ReadRouterInfo",
|
|
|
|
"data_len": len(bytes),
|
|
|
|
"required_len": ROUTER_INFO_MIN_SIZE,
|
|
|
|
"reason": "not enough data",
|
|
|
|
}).Error("error parsing router info")
|
|
|
|
err = errors.New("error parsing router info: not enough data")
|
|
|
|
}
|
|
|
|
date, remainder, err := NewDate(remainder)
|
|
|
|
info.published = date
|
|
|
|
if err != nil {
|
|
|
|
log.WithFields(log.Fields{
|
|
|
|
"at": "(RouterInfo) ReadRouterInfo",
|
|
|
|
"data_len": len(remainder),
|
|
|
|
"required_len": DATE_SIZE,
|
|
|
|
"reason": "not enough data",
|
|
|
|
}).Error("error parsing router info")
|
|
|
|
err = errors.New("error parsing router info: not enough data")
|
|
|
|
}
|
|
|
|
size, remainder, err := NewInteger(remainder)
|
|
|
|
info.size = size
|
|
|
|
if err != nil {
|
|
|
|
log.WithFields(log.Fields{
|
|
|
|
"at": "(RouterInfo) ReadRouterInfo",
|
|
|
|
"data_len": len(remainder),
|
|
|
|
"required_len": INTEGER_SIZE,
|
|
|
|
"reason": "not enough data",
|
|
|
|
}).Error("error parsing router info")
|
|
|
|
err = errors.New("error parsing router info: not enough data")
|
|
|
|
}
|
|
|
|
addresses := make([]*RouterAddress, size.Int())
|
|
|
|
for i := 0; i < size.Int(); i++ {
|
|
|
|
address, remainder, err := NewRouterAddress(remainder)
|
|
|
|
if err != nil {
|
|
|
|
log.WithFields(log.Fields{
|
|
|
|
"at": "(RouterInfo) ReadRouterInfo",
|
|
|
|
"data_len": len(remainder),
|
|
|
|
//"required_len": ROUTER_ADDRESS_SIZE,
|
|
|
|
"reason": "not enough data",
|
|
|
|
}).Error("error parsing router info")
|
|
|
|
err = errors.New("error parsing router info: not enough data")
|
|
|
|
}
|
|
|
|
addresses = append(addresses, address)
|
|
|
|
}
|
|
|
|
info.addresses = addresses
|
|
|
|
peer_size := Integer(remainder[:1])
|
|
|
|
info.peer_size = &peer_size
|
|
|
|
remainder = remainder[1:]
|
|
|
|
options, remainder, err := NewMapping(remainder)
|
|
|
|
info.options = options
|
|
|
|
if err != nil {
|
|
|
|
log.WithFields(log.Fields{
|
|
|
|
"at": "(RouterInfo) ReadRouterInfo",
|
|
|
|
"data_len": len(remainder),
|
|
|
|
//"required_len": MAPPING_SIZE,
|
|
|
|
"reason": "not enough data",
|
|
|
|
}).Error("error parsing router info")
|
|
|
|
err = errors.New("error parsing router info: not enough data")
|
|
|
|
}
|
|
|
|
return
|
2016-02-01 01:56:10 -08:00
|
|
|
}
|