mirror of
https://github.com/yusing/godoxy.git
synced 2025-05-20 12:42:34 +02:00
133 lines
3.6 KiB
Go
133 lines
3.6 KiB
Go
package provider
|
|
|
|
import (
|
|
"github.com/yusing/go-proxy/internal/common"
|
|
E "github.com/yusing/go-proxy/internal/error"
|
|
"github.com/yusing/go-proxy/internal/proxy/entry"
|
|
"github.com/yusing/go-proxy/internal/route"
|
|
"github.com/yusing/go-proxy/internal/task"
|
|
"github.com/yusing/go-proxy/internal/watcher"
|
|
)
|
|
|
|
type EventHandler struct {
|
|
provider *Provider
|
|
|
|
errs *E.Builder
|
|
added *E.Builder
|
|
removed *E.Builder
|
|
updated *E.Builder
|
|
}
|
|
|
|
func (provider *Provider) newEventHandler() *EventHandler {
|
|
return &EventHandler{
|
|
provider: provider,
|
|
errs: E.NewBuilder("event errors"),
|
|
added: E.NewBuilder("added"),
|
|
removed: E.NewBuilder("removed"),
|
|
updated: E.NewBuilder("updated"),
|
|
}
|
|
}
|
|
|
|
func (handler *EventHandler) Handle(parent task.Task, events []watcher.Event) {
|
|
oldRoutes := handler.provider.routes
|
|
newRoutes, err := handler.provider.loadRoutesImpl()
|
|
if err != nil {
|
|
handler.errs.Add(err)
|
|
if newRoutes.Size() == 0 {
|
|
return
|
|
}
|
|
}
|
|
|
|
if common.IsDebug {
|
|
eventsLog := E.NewBuilder("events")
|
|
for _, event := range events {
|
|
eventsLog.Addf("event %s, actor: name=%s, id=%s", event.Action, event.ActorName, event.ActorID)
|
|
}
|
|
E.LogDebug(eventsLog.About(), eventsLog.Error(), handler.provider.Logger())
|
|
|
|
oldRoutesLog := E.NewBuilder("old routes")
|
|
oldRoutes.RangeAllParallel(func(k string, r *route.Route) {
|
|
oldRoutesLog.Adds(k)
|
|
})
|
|
E.LogDebug(oldRoutesLog.About(), oldRoutesLog.Error(), handler.provider.Logger())
|
|
|
|
newRoutesLog := E.NewBuilder("new routes")
|
|
newRoutes.RangeAllParallel(func(k string, r *route.Route) {
|
|
newRoutesLog.Adds(k)
|
|
})
|
|
E.LogDebug(newRoutesLog.About(), newRoutesLog.Error(), handler.provider.Logger())
|
|
}
|
|
|
|
oldRoutes.RangeAll(func(k string, oldr *route.Route) {
|
|
newr, ok := newRoutes.Load(k)
|
|
if !ok {
|
|
handler.Remove(oldr)
|
|
} else if handler.matchAny(events, newr) {
|
|
handler.Update(parent, oldr, newr)
|
|
} else if entry.ShouldNotServe(newr) {
|
|
handler.Remove(oldr)
|
|
}
|
|
})
|
|
newRoutes.RangeAll(func(k string, newr *route.Route) {
|
|
if !(oldRoutes.Has(k) || entry.ShouldNotServe(newr)) {
|
|
handler.Add(parent, newr)
|
|
}
|
|
})
|
|
}
|
|
|
|
func (handler *EventHandler) matchAny(events []watcher.Event, route *route.Route) bool {
|
|
for _, event := range events {
|
|
if handler.match(event, route) {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
func (handler *EventHandler) match(event watcher.Event, route *route.Route) bool {
|
|
switch handler.provider.GetType() {
|
|
case ProviderTypeDocker:
|
|
return route.Entry.Container.ContainerID == event.ActorID ||
|
|
route.Entry.Container.ContainerName == event.ActorName
|
|
case ProviderTypeFile:
|
|
return true
|
|
}
|
|
// should never happen
|
|
return false
|
|
}
|
|
|
|
func (handler *EventHandler) Add(parent task.Task, route *route.Route) {
|
|
err := handler.provider.startRoute(parent, route)
|
|
if err != nil {
|
|
handler.errs.Add(err.Subject("add"))
|
|
} else {
|
|
handler.added.Adds(route.Entry.Alias)
|
|
}
|
|
}
|
|
|
|
func (handler *EventHandler) Remove(route *route.Route) {
|
|
route.Finish("route removed")
|
|
handler.provider.routes.Delete(route.Entry.Alias)
|
|
handler.removed.Adds(route.Entry.Alias)
|
|
}
|
|
|
|
func (handler *EventHandler) Update(parent task.Task, oldRoute *route.Route, newRoute *route.Route) {
|
|
oldRoute.Finish("route update")
|
|
err := handler.provider.startRoute(parent, newRoute)
|
|
if err != nil {
|
|
handler.errs.Add(err.Subject("update"))
|
|
} else {
|
|
handler.updated.Adds(newRoute.Entry.Alias)
|
|
}
|
|
}
|
|
|
|
func (handler *EventHandler) Log() {
|
|
results := E.NewBuilder("event occured")
|
|
results.Add(handler.added.Error())
|
|
results.Add(handler.removed.Error())
|
|
results.Add(handler.updated.Error())
|
|
results.Add(handler.errs.Error())
|
|
if result := results.String(); result != "" {
|
|
handler.provider.Logger().Info().Msg(result)
|
|
}
|
|
}
|