mirror of
https://gitee.com/rainbond/Rainbond.git
synced 2024-11-30 10:48:15 +08:00
147 lines
3.4 KiB
Go
147 lines
3.4 KiB
Go
|
|
// RAINBOND, Application Management Platform
|
|
// Copyright (C) 2014-2017 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 proxy
|
|
|
|
import (
|
|
"io/ioutil"
|
|
"net/http"
|
|
"strings"
|
|
"sync/atomic"
|
|
|
|
"github.com/Sirupsen/logrus"
|
|
)
|
|
|
|
// RoundRobin round robin loadBalance impl
|
|
type RoundRobin struct {
|
|
ops *uint64
|
|
}
|
|
|
|
//LoadBalance LoadBalance
|
|
type LoadBalance interface {
|
|
Select(r *http.Request, endpoints EndpointList) Endpoint
|
|
}
|
|
|
|
//Endpoint Endpoint
|
|
type Endpoint string
|
|
|
|
func (e Endpoint) String() string {
|
|
return string(e)
|
|
}
|
|
|
|
//EndpointList EndpointList
|
|
type EndpointList []Endpoint
|
|
|
|
//Len Len
|
|
func (e *EndpointList) Len() int {
|
|
return len(*e)
|
|
}
|
|
|
|
//Add Add
|
|
func (e *EndpointList) Add(endpoints ...string) {
|
|
for _, end := range endpoints {
|
|
*e = append(*e, Endpoint(end))
|
|
}
|
|
}
|
|
|
|
//Delete Delete
|
|
func (e *EndpointList) Delete(endpoints ...string) {
|
|
var new EndpointList
|
|
for _, endpoint := range endpoints {
|
|
for _, old := range *e {
|
|
if string(old) != endpoint {
|
|
new = append(new, old)
|
|
}
|
|
}
|
|
}
|
|
*e = new
|
|
}
|
|
|
|
//Selec Selec
|
|
func (e *EndpointList) Selec(i int) Endpoint {
|
|
return (*e)[i]
|
|
}
|
|
|
|
//CreateEndpoints CreateEndpoints
|
|
func CreateEndpoints(endpoints []string) EndpointList {
|
|
var epl EndpointList
|
|
for _, e := range endpoints {
|
|
epl = append(epl, Endpoint(e))
|
|
}
|
|
return epl
|
|
}
|
|
|
|
// NewRoundRobin create a RoundRobin
|
|
func NewRoundRobin() LoadBalance {
|
|
var ops uint64
|
|
ops = 0
|
|
return RoundRobin{
|
|
ops: &ops,
|
|
}
|
|
}
|
|
|
|
// Select select a server from servers using RoundRobin
|
|
func (rr RoundRobin) Select(r *http.Request, endpoints EndpointList) Endpoint {
|
|
l := uint64(endpoints.Len())
|
|
if 0 >= l {
|
|
return ""
|
|
}
|
|
selec := int(atomic.AddUint64(rr.ops, 1) % l)
|
|
return endpoints.Selec(selec)
|
|
}
|
|
|
|
//SelectBalance 选择性负载均衡
|
|
type SelectBalance struct {
|
|
hostIDMap map[string]string
|
|
}
|
|
|
|
//NewSelectBalance 创建选择性负载均衡
|
|
func NewSelectBalance() *SelectBalance {
|
|
body, err := ioutil.ReadFile("/etc/goodrain/host_id_list.conf")
|
|
if err != nil {
|
|
logrus.Error("read host id list error,", err.Error())
|
|
}
|
|
sb := &SelectBalance{
|
|
hostIDMap: map[string]string{"local": "127.0.0.1:6363"},
|
|
}
|
|
if body != nil && len(body) > 0 {
|
|
listStr := string(body)
|
|
hosts := strings.Split(strings.TrimSpace(listStr), ";")
|
|
for _, h := range hosts {
|
|
info := strings.Split(strings.Trim(h, "\r\n"), "=")
|
|
if len(info) == 2 {
|
|
sb.hostIDMap[info[0]] = info[1]
|
|
}
|
|
}
|
|
}
|
|
logrus.Info("Docker log websocket server endpoints:", sb.hostIDMap)
|
|
return sb
|
|
}
|
|
|
|
//Select 负载
|
|
func (s *SelectBalance) Select(r *http.Request, endpoints EndpointList) Endpoint {
|
|
if r.URL != nil {
|
|
hostID := r.URL.Query().Get("host_id")
|
|
if e, ok := s.hostIDMap[hostID]; ok {
|
|
return Endpoint(e)
|
|
}
|
|
}
|
|
return Endpoint(s.hostIDMap["local"])
|
|
}
|