mirror of
https://gitee.com/rainbond/Rainbond.git
synced 2024-12-03 04:07:51 +08:00
162 lines
4.4 KiB
Go
162 lines
4.4 KiB
Go
// RAINBOND, Application Management Platform
|
|
// Copyright (C) 2014-2019 Goodrain Co., Ltd.
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version. For any non-GPL usage of Rainbond,
|
|
// one or multiple Commercial Licenses authorized by Goodrain Co., Ltd.
|
|
// must be obtained first.
|
|
|
|
// This program 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 General Public License for more details.
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
package cluster
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"net"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/Sirupsen/logrus"
|
|
|
|
"github.com/coreos/etcd/clientv3"
|
|
|
|
"github.com/goodrain/rainbond/cmd/gateway/option"
|
|
"github.com/goodrain/rainbond/util"
|
|
)
|
|
|
|
//IPManager ip manager
|
|
//Gets all available IP addresses for synchronizing the current node
|
|
type IPManager interface {
|
|
//Whether the IP address belongs to the current node
|
|
IPInCurrentHost(net.IP) bool
|
|
Start() error
|
|
//An IP pool change triggers a forced update of the gateway policy
|
|
NeedUpdateGatewayPolicy() <-chan util.IPEVENT
|
|
Stop()
|
|
}
|
|
|
|
type ipManager struct {
|
|
ctx context.Context
|
|
cancel context.CancelFunc
|
|
IPPool *util.IPPool
|
|
ipLease map[string]clientv3.LeaseID
|
|
lock sync.Mutex
|
|
etcdCli *clientv3.Client
|
|
config option.Config
|
|
//An IP pool change triggers a forced update of the gateway policy
|
|
needUpdate chan util.IPEVENT
|
|
}
|
|
|
|
//CreateIPManager create ip manage
|
|
func CreateIPManager(ctx context.Context, config option.Config, etcdcli *clientv3.Client) (IPManager, error) {
|
|
newCtx, cancel := context.WithCancel(ctx)
|
|
IPPool := util.NewIPPool(config.IgnoreInterface)
|
|
return &ipManager{
|
|
ctx: newCtx,
|
|
cancel: cancel,
|
|
IPPool: IPPool,
|
|
config: config,
|
|
etcdCli: etcdcli,
|
|
ipLease: make(map[string]clientv3.LeaseID),
|
|
needUpdate: make(chan util.IPEVENT, 10),
|
|
}, nil
|
|
}
|
|
|
|
func (i *ipManager) NeedUpdateGatewayPolicy() <-chan util.IPEVENT {
|
|
return i.needUpdate
|
|
}
|
|
|
|
//IPInCurrentHost Whether the IP address belongs to the current node
|
|
func (i *ipManager) IPInCurrentHost(in net.IP) bool {
|
|
for _, exit := range i.IPPool.GetHostIPs() {
|
|
if exit.Equal(in) {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
func (i *ipManager) Start() error {
|
|
logrus.Info("start ip manager.")
|
|
go i.IPPool.LoopCheckIPs()
|
|
i.IPPool.Ready()
|
|
logrus.Info("ip manager is ready.")
|
|
go i.syncIP()
|
|
return nil
|
|
}
|
|
|
|
func (i *ipManager) syncIP() {
|
|
logrus.Debugf("start syncronizing ip.")
|
|
ips := i.IPPool.GetHostIPs()
|
|
i.updateIP(ips...)
|
|
for ipevent := range i.IPPool.GetWatchIPChan() {
|
|
switch ipevent.Type {
|
|
case util.ADD:
|
|
i.updateIP(ipevent.IP)
|
|
case util.UPDATE:
|
|
i.updateIP(ipevent.IP)
|
|
case util.DEL:
|
|
i.deleteIP(ipevent.IP)
|
|
}
|
|
i.needUpdate <- ipevent
|
|
}
|
|
}
|
|
|
|
func (i *ipManager) updateIP(ips ...net.IP) error {
|
|
ctx, cancel := context.WithTimeout(i.ctx, time.Second*30)
|
|
defer cancel()
|
|
i.lock.Lock()
|
|
defer i.lock.Unlock()
|
|
leaseClient := clientv3.NewLease(i.etcdCli)
|
|
for in := range ips {
|
|
ip := ips[in]
|
|
if id, ok := i.ipLease[ip.String()]; ok {
|
|
if _, err := leaseClient.KeepAliveOnce(ctx, id); err == nil {
|
|
continue
|
|
} else {
|
|
logrus.Warningf("keep alive ip key failure %s", err.Error())
|
|
}
|
|
}
|
|
res, err := leaseClient.Grant(ctx, 10)
|
|
if err != nil {
|
|
logrus.Errorf("put gateway ip to etcd failure %s", err.Error())
|
|
continue
|
|
}
|
|
_, err = i.etcdCli.Put(ctx, fmt.Sprintf("/rainbond/gateway/ips/%s", ip.String()), ip.String(), clientv3.WithLease(res.ID))
|
|
if err != nil {
|
|
logrus.Errorf("put gateway ip to etcd failure %s", err.Error())
|
|
continue
|
|
}
|
|
logrus.Infof("gateway init add ip %s", ip.String())
|
|
i.ipLease[ip.String()] = res.ID
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (i *ipManager) deleteIP(ips ...net.IP) {
|
|
ctx, cancel := context.WithTimeout(i.ctx, time.Second*30)
|
|
defer cancel()
|
|
i.lock.Lock()
|
|
defer i.lock.Unlock()
|
|
for _, ip := range ips {
|
|
_, err := i.etcdCli.Delete(ctx, fmt.Sprintf("/rainbond/gateway/ips/%s", ip.String()))
|
|
if err != nil {
|
|
logrus.Errorf("put gateway ip to etcd failure %s", err.Error())
|
|
}
|
|
delete(i.ipLease, ip.String())
|
|
}
|
|
}
|
|
|
|
func (i *ipManager) Stop() {
|
|
i.cancel()
|
|
}
|