123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156 |
- package group
- import (
- "fmt"
- "net"
- "sync"
- "sync/atomic"
- "github.com/fatedier/frp/pkg/util/vhost"
- )
- type HTTPGroupController struct {
- groups map[string]*HTTPGroup
- vhostRouter *vhost.Routers
- mu sync.Mutex
- }
- func NewHTTPGroupController(vhostRouter *vhost.Routers) *HTTPGroupController {
- return &HTTPGroupController{
- groups: make(map[string]*HTTPGroup),
- vhostRouter: vhostRouter,
- }
- }
- func (ctl *HTTPGroupController) Register(proxyName, group, groupKey string,
- routeConfig vhost.RouteConfig) (err error) {
- indexKey := httpGroupIndex(group, routeConfig.Domain, routeConfig.Location)
- ctl.mu.Lock()
- g, ok := ctl.groups[indexKey]
- if !ok {
- g = NewHTTPGroup(ctl)
- ctl.groups[indexKey] = g
- }
- ctl.mu.Unlock()
- return g.Register(proxyName, group, groupKey, routeConfig)
- }
- func (ctl *HTTPGroupController) UnRegister(proxyName, group, domain, location string) {
- indexKey := httpGroupIndex(group, domain, location)
- ctl.mu.Lock()
- defer ctl.mu.Unlock()
- g, ok := ctl.groups[indexKey]
- if !ok {
- return
- }
- isEmpty := g.UnRegister(proxyName)
- if isEmpty {
- delete(ctl.groups, indexKey)
- }
- }
- type HTTPGroup struct {
- group string
- groupKey string
- domain string
- location string
- createFuncs map[string]vhost.CreateConnFunc
- pxyNames []string
- index uint64
- ctl *HTTPGroupController
- mu sync.RWMutex
- }
- func NewHTTPGroup(ctl *HTTPGroupController) *HTTPGroup {
- return &HTTPGroup{
- createFuncs: make(map[string]vhost.CreateConnFunc),
- pxyNames: make([]string, 0),
- ctl: ctl,
- }
- }
- func (g *HTTPGroup) Register(proxyName, group, groupKey string,
- routeConfig vhost.RouteConfig) (err error) {
- g.mu.Lock()
- defer g.mu.Unlock()
- if len(g.createFuncs) == 0 {
- // the first proxy in this group
- tmp := routeConfig // copy object
- tmp.CreateConnFn = g.createConn
- err = g.ctl.vhostRouter.Add(routeConfig.Domain, routeConfig.Location, &tmp)
- if err != nil {
- return
- }
- g.group = group
- g.groupKey = groupKey
- g.domain = routeConfig.Domain
- g.location = routeConfig.Location
- } else {
- if g.group != group || g.domain != routeConfig.Domain || g.location != routeConfig.Location {
- err = ErrGroupParamsInvalid
- return
- }
- if g.groupKey != groupKey {
- err = ErrGroupAuthFailed
- return
- }
- }
- if _, ok := g.createFuncs[proxyName]; ok {
- err = ErrProxyRepeated
- return
- }
- g.createFuncs[proxyName] = routeConfig.CreateConnFn
- g.pxyNames = append(g.pxyNames, proxyName)
- return nil
- }
- func (g *HTTPGroup) UnRegister(proxyName string) (isEmpty bool) {
- g.mu.Lock()
- defer g.mu.Unlock()
- delete(g.createFuncs, proxyName)
- for i, name := range g.pxyNames {
- if name == proxyName {
- g.pxyNames = append(g.pxyNames[:i], g.pxyNames[i+1:]...)
- break
- }
- }
- if len(g.createFuncs) == 0 {
- isEmpty = true
- g.ctl.vhostRouter.Del(g.domain, g.location)
- }
- return
- }
- func (g *HTTPGroup) createConn(remoteAddr string) (net.Conn, error) {
- var f vhost.CreateConnFunc
- newIndex := atomic.AddUint64(&g.index, 1)
- g.mu.RLock()
- group := g.group
- domain := g.domain
- location := g.location
- if len(g.pxyNames) > 0 {
- name := g.pxyNames[int(newIndex)%len(g.pxyNames)]
- f, _ = g.createFuncs[name]
- }
- g.mu.RUnlock()
- if f == nil {
- return nil, fmt.Errorf("no CreateConnFunc for http group [%s], domain [%s], location [%s]", group, domain, location)
- }
- return f(remoteAddr)
- }
- func httpGroupIndex(group, domain, location string) string {
- return fmt.Sprintf("%s_%s_%s", group, domain, location)
- }
|