6.4 KiB
Load Balancing Module - WIP
The LCD will be an important bridge between service providers and cosmos blockchain network. Suppose a service provider wants to monitor token information for millions of accounts. Then it has to keep sending a large mount of requests to LCD to query token information. As a result, LCD will send huge requests to full node to get token information and necessary proof which will cost full node much computing and bandwidth resource. Too many requests to a single full node may result in some bad situations:
1. The full node crash possibility increases.
2. The reply delay increases.
3. The system reliability will decrease.
4. As the full node may belong to other people or associates, they may deny too frequent access from a single client.
It is very urgent to solve this problems. Here we consider to import load balancing into LCD. By the help of load balancing, LCD can distribute millions of requests to a set of full nodes. Thus the load of each full node won't be too heavy and the unavailable full nodes will be wiped out of query list. In addition, the system reliability will increase.
Design
This module need combine with client to realize the real load balancing. It can embed the
HTTP Client. In other
words,we realise the new httpclient based on HTTP
.
type HTTPLoadBalancer struct {
rpcs map[string]*rpcclient.JSONRPCClient
*WSEvents
}
The Diagram of LCD RPC WorkFlow with LoadBalance
In the above sequence diagram, application calls the Request()
, and LCD finally call the
HTTP.Request()
through the SecureClient Wrapper
. In every HTTP.Request()
, Getclient()
selects the current working rpcclient by the load balancing algorithm,then run the
JSONRPCClient.Call()
to request from the Full Node, finally UpdateClient()
updates the weight of
the current rpcclient according to the status that is returned by the full node. The GetAddr()
and UpdateAddrWeight()
are realized in the load balancing module.
There are some abilities to do:
- Add the Remote Address
- Delete the Remote Address
- Update the weights of the addresses
Load balancing Strategies
We can design some strategies like nginx to combine the different load balancing algorithms to get the final remote. We can also get the status of the remote server to add or delete the addresses and update weights of the addresses.
In a word,it can make the entire LCD work more effective in actual conditions. We are working this module independently in this Github Repository.
Interface And Type
Balancer
This interface Balancer
is the core of the package. Every load balancing algorithm should realize
it,and it defined two interfaces.
init
initialize the balancer, assigns the variables whichDoBalance
needs.DoBalance
load balance the full node addresses according to the current situation.
package balance
type Balancer interface {
init(NodeAddrs)
DoBalance(NodeAddrs) (*NodeAddr,int,error)
}
NodeAddr
- host: ip address
- port: the number of port
- weight: the weight of this full node address,default:1
This NodeAddr is the base struct of the address.
type NodeAddr struct{
host string
port int
weight int
}
func (p *NodeAddr) GetHost() string
func (p *NodeAddr) GetPort() int
func (p *NodeAddr) GetWeight() int
func (p *NodeAddr) updateWeight(weight int)
The weight
is the important factor that schedules which full node the LCD calls. The weight can be
changed by the information from the full node. So we have the function updateWegiht
.
NodeAddrs
in
balance/types.go
NodeAddrs
is the list of the full node address. This is the member variable in the
BalanceManager(BalancerMgr
).
type NodeAddrs []*NodeAddr
Load Balancing Algorithm
Random
in
balance/random.go
Random algorithm selects a remote address randomly to process the request. The probability of them being selected is the same.
RandomWeight
in
balance/random.go
RandomWeight Algorithm also selects a remote address randomly to process the request. But the higher the weight, the greater the probability.
RoundRobin
in
balance/roundrobin.go
RoundRobin Algorithm selects a remote address orderly. Every remote address have the same probability to be selected.
RoundRobinWeight
in
balance/roundrobin.go
RoundRobinWeight Algorthm selects a remote address orderly. But every remote address have different probability to be selected which are determined by their weight.
Hash
//TODO
Load Balancing Manager
BalanceMgr
in
balance/manager.go
- addrs: the set of the remote full node addresses
- balancers: map the string of balancer name to the specific balancer
- change: record whether the machine reinitialize after the
addrs
changes
BalanceMgr
is the manager of many balancer. It is the access of load balancing. Its main function
is to maintain the NodeAddrs
and to call the specific load balancing algorithm above.
type BalanceMgr struct{
addrs NodeAddrs
balancers map[string]Balancer
change map[string]bool
}
func (p *BalanceMgr) RegisterBalancer(name string,balancer Balancer)
func (p *BalanceMgr) updateBalancer(name string)
func (p *BalanceMgr) AddNodeAddr(addr *NodeAddr)
func (p *BalanceMgr) DeleteNodeAddr(i int)
func (p *BalanceMgr) UpdateWeightNodeAddr(i int,weight int)
func (p *BalanceMgr) GetAddr(name string)(*NodeAddr,int,error) {
// if addrs change,update the balancer which we use.
if p.change[name]{
p.updateBalancer(name)
}
// get the balancer by name
balancer := p.balancers[name]
// use the load balancing algorithm
addr,index,err := balancer.DoBalance(p.addrs)
return addr,index,err
}
RegisterBalancer
: register the basic balancer implementing theBalancer
interface and initialize them.updateBalancer
: update the specific balancer after theaddrs
change.AddNodeAddr
: add the remote address and set all the values of thechange
to true.DeleteNodeAddr
: delete the remote address and set all the values of thechange
to true.UpdateWeightNodeAddr
: update the weight of the remote address and set all the values of thechange
to true.GetAddr
:select the address by the balancer thename
decides.