-
Notifications
You must be signed in to change notification settings - Fork 179
/
connection.go
115 lines (97 loc) · 5.54 KB
/
connection.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
package connection
import (
"fmt"
"io"
"net"
"time"
"github.com/rs/zerolog"
"github.com/onflow/flow-go/crypto"
"github.com/onflow/flow-go/module"
"github.com/onflow/flow/protobuf/go/flow/access"
"github.com/onflow/flow/protobuf/go/flow/execution"
)
// ConnectionFactory is an interface for creating access and execution API clients.
type ConnectionFactory interface {
// GetAccessAPIClient gets an access API client for the specified address using the default CollectionGRPCPort, networkPubKey is optional,
// and it is used for secure gRPC connection. Can be nil for an unsecured connection.
// The returned io.Closer should close the connection after the call if no error occurred during client creation.
GetAccessAPIClient(address string, networkPubKey crypto.PublicKey) (access.AccessAPIClient, io.Closer, error)
// GetAccessAPIClientWithPort gets an access API client for the specified address with port, networkPubKey is optional,
// and it is used for secure gRPC connection. Can be nil for an unsecured connection.
// The returned io.Closer should close the connection after the call if no error occurred during client creation.
GetAccessAPIClientWithPort(address string, networkPubKey crypto.PublicKey) (access.AccessAPIClient, io.Closer, error)
// GetExecutionAPIClient gets an execution API client for the specified address using the default ExecutionGRPCPort.
// The returned io.Closer should close the connection after the call if no error occurred during client creation.
GetExecutionAPIClient(address string) (execution.ExecutionAPIClient, io.Closer, error)
}
// ProxyConnectionFactory wraps an existing ConnectionFactory and allows getting API clients for a target address.
type ProxyConnectionFactory struct {
ConnectionFactory
targetAddress string
}
// GetAccessAPIClient gets an access API client for a target address using the default CollectionGRPCPort.
// The networkPubKey is the public key used for a secure gRPC connection. It can be nil for an unsecured connection.
// The returned io.Closer should close the connection after the call if no error occurred during client creation.
func (p *ProxyConnectionFactory) GetAccessAPIClient(address string, networkPubKey crypto.PublicKey) (access.AccessAPIClient, io.Closer, error) {
return p.ConnectionFactory.GetAccessAPIClient(p.targetAddress, networkPubKey)
}
// GetExecutionAPIClient gets an execution API client for a target address using the default ExecutionGRPCPort.
// The returned io.Closer should close the connection after the call if no error occurred during client creation.
func (p *ProxyConnectionFactory) GetExecutionAPIClient(address string) (execution.ExecutionAPIClient, io.Closer, error) {
return p.ConnectionFactory.GetExecutionAPIClient(p.targetAddress)
}
var _ ConnectionFactory = (*ConnectionFactoryImpl)(nil)
type ConnectionFactoryImpl struct {
CollectionGRPCPort uint
ExecutionGRPCPort uint
CollectionNodeGRPCTimeout time.Duration
ExecutionNodeGRPCTimeout time.Duration
AccessMetrics module.AccessMetrics
Log zerolog.Logger
Manager Manager
}
// GetAccessAPIClient gets an access API client for the specified address using the default CollectionGRPCPort.
// The networkPubKey is the public key used for secure gRPC connection. Can be nil for an unsecured connection.
// The returned io.Closer should close the connection after the call if no error occurred during client creation.
func (cf *ConnectionFactoryImpl) GetAccessAPIClient(address string, networkPubKey crypto.PublicKey) (access.AccessAPIClient, io.Closer, error) {
address, err := getGRPCAddress(address, cf.CollectionGRPCPort)
if err != nil {
return nil, nil, err
}
return cf.GetAccessAPIClientWithPort(address, networkPubKey)
}
// GetAccessAPIClientWithPort gets an access API client for the specified address with port.
// The networkPubKey is the public key used for secure gRPC connection. Can be nil for an unsecured connection.
// The returned io.Closer should close the connection after the call if no error occurred during client creation.
func (cf *ConnectionFactoryImpl) GetAccessAPIClientWithPort(address string, networkPubKey crypto.PublicKey) (access.AccessAPIClient, io.Closer, error) {
conn, closer, err := cf.Manager.GetConnection(address, cf.CollectionNodeGRPCTimeout, networkPubKey)
if err != nil {
return nil, nil, err
}
return access.NewAccessAPIClient(conn), closer, nil
}
// GetExecutionAPIClient gets an execution API client for the specified address using the default ExecutionGRPCPort.
// The returned io.Closer should close the connection after the call if no error occurred during client creation.
func (cf *ConnectionFactoryImpl) GetExecutionAPIClient(address string) (execution.ExecutionAPIClient, io.Closer, error) {
grpcAddress, err := getGRPCAddress(address, cf.ExecutionGRPCPort)
if err != nil {
return nil, nil, err
}
conn, closer, err := cf.Manager.GetConnection(grpcAddress, cf.ExecutionNodeGRPCTimeout, nil)
if err != nil {
return nil, nil, err
}
return execution.NewExecutionAPIClient(conn), closer, nil
}
// getGRPCAddress translates the flow.Identity address to the GRPC address of the node by switching the port to the
// GRPC port from the libp2p port.
func getGRPCAddress(address string, grpcPort uint) (string, error) {
// Split hostname and port
hostnameOrIP, _, err := net.SplitHostPort(address)
if err != nil {
return "", err
}
// Use the hostname from the identity list and the GRPC port number as the one passed in as an argument.
grpcAddress := fmt.Sprintf("%s:%d", hostnameOrIP, grpcPort)
return grpcAddress, nil
}