mirror of
https://github.com/kemko/reproxy.git
synced 2026-01-06 18:25:49 +03:00
316 lines
7.6 KiB
Go
316 lines
7.6 KiB
Go
// Package discovery provides a common interface for all providers and Match to
|
|
// transform source to destination URL.
|
|
// Run func starts event loop checking all providers and retrieving lists of rules.
|
|
// All lists combined into a merged one.
|
|
package discovery
|
|
|
|
import (
|
|
"context"
|
|
"regexp"
|
|
"sort"
|
|
"strings"
|
|
"sync"
|
|
"time"
|
|
|
|
log "github.com/go-pkgz/lgr"
|
|
)
|
|
|
|
//go:generate moq -out provider_mock.go -fmt goimports . Provider
|
|
|
|
// Service implements discovery with multiple providers and url matcher
|
|
type Service struct {
|
|
providers []Provider
|
|
mappers map[string][]URLMapper
|
|
lock sync.RWMutex
|
|
interval time.Duration
|
|
}
|
|
|
|
// URLMapper contains all info about source and destination routes
|
|
type URLMapper struct {
|
|
Server string
|
|
SrcMatch regexp.Regexp
|
|
Dst string
|
|
ProviderID ProviderID
|
|
PingURL string
|
|
MatchType MatchType
|
|
|
|
AssetsLocation string
|
|
AssetsWebRoot string
|
|
|
|
dead bool
|
|
}
|
|
|
|
// Provider defines sources of mappers
|
|
type Provider interface {
|
|
Events(ctx context.Context) (res <-chan ProviderID)
|
|
List() (res []URLMapper, err error)
|
|
}
|
|
|
|
// ProviderID holds provider identifier to emulate enum of them
|
|
type ProviderID string
|
|
|
|
// enum of all provider ids
|
|
const (
|
|
PIDocker ProviderID = "docker"
|
|
PIStatic ProviderID = "static"
|
|
PIFile ProviderID = "file"
|
|
PIConsulCatalog ProviderID = "consul-catalog"
|
|
)
|
|
|
|
// MatchType defines the type of mapper (rule)
|
|
type MatchType int
|
|
|
|
// enum of all match types
|
|
const (
|
|
MTProxy MatchType = iota
|
|
MTStatic
|
|
)
|
|
|
|
func (m MatchType) String() string {
|
|
switch m {
|
|
case MTProxy:
|
|
return "proxy"
|
|
case MTStatic:
|
|
return "static"
|
|
default:
|
|
return "unknown"
|
|
}
|
|
}
|
|
|
|
// NewService makes service with given providers
|
|
func NewService(providers []Provider, interval time.Duration) *Service {
|
|
return &Service{providers: providers, interval: interval}
|
|
}
|
|
|
|
// Run runs blocking loop getting events from all providers
|
|
// and updating all mappers on each event
|
|
func (s *Service) Run(ctx context.Context) error {
|
|
|
|
evChs := make([]<-chan ProviderID, 0, len(s.providers))
|
|
for _, p := range s.providers {
|
|
evChs = append(evChs, p.Events(ctx))
|
|
}
|
|
ch := s.mergeEvents(ctx, evChs...)
|
|
var evRecv bool
|
|
for {
|
|
select {
|
|
case <-ctx.Done():
|
|
return ctx.Err()
|
|
case ev := <-ch:
|
|
log.Printf("[DEBUG] new update event received, %s", ev)
|
|
evRecv = true
|
|
case <-time.After(s.interval):
|
|
if !evRecv {
|
|
continue
|
|
}
|
|
evRecv = false
|
|
lst := s.mergeLists()
|
|
for _, m := range lst {
|
|
if m.MatchType == MTProxy {
|
|
log.Printf("[INFO] proxy %s: %s %s -> %s", m.ProviderID, m.Server, m.SrcMatch.String(), m.Dst)
|
|
}
|
|
if m.MatchType == MTStatic {
|
|
log.Printf("[INFO] assets %s: %s %s -> %s", m.ProviderID, m.Server, m.AssetsWebRoot, m.AssetsLocation)
|
|
}
|
|
}
|
|
s.lock.Lock()
|
|
s.mappers = make(map[string][]URLMapper)
|
|
for _, m := range lst {
|
|
s.mappers[m.Server] = append(s.mappers[m.Server], m)
|
|
}
|
|
s.lock.Unlock()
|
|
}
|
|
}
|
|
}
|
|
|
|
// Match url to all mappers
|
|
func (s *Service) Match(srv, src string) (string, MatchType, bool) {
|
|
|
|
s.lock.RLock()
|
|
defer s.lock.RUnlock()
|
|
|
|
for _, srvName := range []string{srv, "*", ""} {
|
|
for _, m := range s.mappers[srvName] {
|
|
|
|
switch m.MatchType {
|
|
case MTProxy:
|
|
dest := m.SrcMatch.ReplaceAllString(src, m.Dst)
|
|
if src != dest {
|
|
return dest, m.MatchType, true
|
|
}
|
|
case MTStatic:
|
|
if src == m.AssetsWebRoot || strings.HasPrefix(src, m.AssetsWebRoot+"/") {
|
|
return m.AssetsWebRoot + ":" + m.AssetsLocation, MTStatic, true
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return src, MTProxy, false
|
|
}
|
|
|
|
// ScheduleHealthCheck starts background loop with health-check
|
|
func (s *Service) ScheduleHealthCheck(ctx context.Context, interval time.Duration) {
|
|
log.Printf("health-check scheduled every %s", interval)
|
|
|
|
go func() {
|
|
hloop:
|
|
for {
|
|
timer := time.NewTimer(interval)
|
|
select {
|
|
case <-timer.C:
|
|
s.lock.RLock()
|
|
cres := CheckHealth(s.Mappers())
|
|
s.lock.RUnlock()
|
|
|
|
// alive services would be picked up first
|
|
sort.SliceStable(cres.mappers, func(i, j int) bool {
|
|
return cres.mappers[j].dead
|
|
})
|
|
s.lock.Lock()
|
|
s.mappers = make(map[string][]URLMapper)
|
|
for _, m := range cres.mappers {
|
|
s.mappers[m.Server] = append(s.mappers[m.Server], m)
|
|
}
|
|
s.lock.Unlock()
|
|
case <-ctx.Done():
|
|
break hloop
|
|
}
|
|
}
|
|
}()
|
|
}
|
|
|
|
// Servers return list of all servers, skips "*" (catch-all/default)
|
|
func (s *Service) Servers() (servers []string) {
|
|
s.lock.RLock()
|
|
defer s.lock.RUnlock()
|
|
for key, ms := range s.mappers {
|
|
if key == "*" || key == "" {
|
|
continue
|
|
}
|
|
for _, m := range ms {
|
|
servers = append(servers, m.Server)
|
|
}
|
|
}
|
|
sort.Strings(servers)
|
|
return servers
|
|
}
|
|
|
|
// Mappers return list of all mappers
|
|
func (s *Service) Mappers() (mappers []URLMapper) {
|
|
s.lock.RLock()
|
|
defer s.lock.RUnlock()
|
|
for _, m := range s.mappers {
|
|
mappers = append(mappers, m...)
|
|
}
|
|
sort.Slice(mappers, func(i, j int) bool {
|
|
return len(mappers[i].SrcMatch.String()) > len(mappers[j].SrcMatch.String())
|
|
})
|
|
return mappers
|
|
}
|
|
|
|
func (s *Service) mergeLists() (res []URLMapper) {
|
|
for _, p := range s.providers {
|
|
lst, err := p.List()
|
|
if err != nil {
|
|
log.Printf("[DEBUG] can't get list for %s, %v", p, err)
|
|
continue
|
|
}
|
|
for i := range lst {
|
|
lst[i] = s.extendMapper(lst[i])
|
|
}
|
|
res = append(res, lst...)
|
|
}
|
|
return res
|
|
}
|
|
|
|
// extendMapper from /something/blah->http://example.com/api to ^/something/blah/(.*)->http://example.com/api/$1
|
|
// also substitutes @ in dest by $. The reason for this substitution - some providers, for example docker
|
|
// treat $ in a special way for variable substitution and user has to escape $, like this reproxy.dest: '/$$1'
|
|
// It can be simplified with @, i.e. reproxy.dest: '/@1'
|
|
func (s *Service) extendMapper(m URLMapper) URLMapper {
|
|
|
|
src := m.SrcMatch.String()
|
|
m.Dst = strings.Replace(m.Dst, "@", "$", -1) // allow group defined as @n instead of $n (yaml friendly)
|
|
|
|
// static match with assets uses AssetsWebRoot and AssetsLocation
|
|
if m.MatchType == MTStatic && m.AssetsWebRoot != "" && m.AssetsLocation != "" {
|
|
m.AssetsWebRoot = strings.TrimSuffix(m.AssetsWebRoot, "/")
|
|
m.AssetsLocation = strings.TrimSuffix(m.AssetsLocation, "/") + "/"
|
|
}
|
|
|
|
// static match without assets defined defaulted to src:dst/
|
|
if m.MatchType == MTStatic && m.AssetsWebRoot == "" && m.AssetsLocation == "" {
|
|
m.AssetsWebRoot = strings.TrimSuffix(src, "/")
|
|
m.AssetsLocation = strings.TrimSuffix(m.Dst, "/") + "/"
|
|
}
|
|
|
|
// don't extend src and dst with dst or src regex groups
|
|
if strings.Contains(m.Dst, "$") || strings.Contains(m.Dst, "@") || strings.Contains(src, "(") {
|
|
return m
|
|
}
|
|
|
|
if !strings.HasSuffix(src, "/") && m.MatchType == MTProxy {
|
|
return m
|
|
}
|
|
|
|
res := URLMapper{
|
|
Server: m.Server,
|
|
Dst: strings.TrimSuffix(m.Dst, "/") + "/$1",
|
|
ProviderID: m.ProviderID,
|
|
PingURL: m.PingURL,
|
|
MatchType: m.MatchType,
|
|
AssetsWebRoot: m.AssetsWebRoot,
|
|
AssetsLocation: m.AssetsLocation,
|
|
}
|
|
|
|
rx, err := regexp.Compile("^" + strings.TrimSuffix(src, "/") + "/(.*)")
|
|
if err != nil {
|
|
log.Printf("[WARN] can't extend %s, %v", m.SrcMatch.String(), err)
|
|
return m
|
|
}
|
|
res.SrcMatch = *rx
|
|
return res
|
|
}
|
|
|
|
func (s *Service) mergeEvents(ctx context.Context, chs ...<-chan ProviderID) <-chan ProviderID {
|
|
var wg sync.WaitGroup
|
|
out := make(chan ProviderID)
|
|
|
|
output := func(ctx context.Context, c <-chan ProviderID) {
|
|
defer wg.Done()
|
|
for {
|
|
select {
|
|
case <-ctx.Done():
|
|
return
|
|
case v, ok := <-c:
|
|
if !ok {
|
|
return
|
|
}
|
|
out <- v
|
|
}
|
|
}
|
|
}
|
|
|
|
wg.Add(len(chs))
|
|
for _, c := range chs {
|
|
go output(ctx, c)
|
|
}
|
|
|
|
go func() {
|
|
wg.Wait()
|
|
close(out)
|
|
}()
|
|
return out
|
|
}
|
|
|
|
// Contains checks if the input string (e) in the given slice
|
|
func Contains(e string, s []string) bool {
|
|
for _, a := range s {
|
|
if a == e {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|