package vhost import ( "sort" "strings" "sync" ) type VhostRouters struct { RouterByDomain map[string][]*VhostRouter mutex sync.RWMutex } type VhostRouter struct { domain string location string listener *Listener } func NewVhostRouters() *VhostRouters { return &VhostRouters{ RouterByDomain: make(map[string][]*VhostRouter), } } func (r *VhostRouters) Add(domain, location string, l *Listener) { r.mutex.Lock() defer r.mutex.Unlock() vrs, found := r.RouterByDomain[domain] if !found { vrs = make([]*VhostRouter, 0) } vr := &VhostRouter{ domain: domain, location: location, listener: l, } vrs = append(vrs, vr) sort.Reverse(ByLocation(vrs)) r.RouterByDomain[domain] = vrs } func (r *VhostRouters) Del(l *Listener) { r.mutex.Lock() defer r.mutex.Unlock() vrs, found := r.RouterByDomain[l.name] if !found { return } for i, vr := range vrs { if vr.listener == l { if len(vrs) > i+1 { r.RouterByDomain[l.name] = append(vrs[:i], vrs[i+1:]...) } else { r.RouterByDomain[l.name] = vrs[:i] } } } } func (r *VhostRouters) Get(host, path string) (vr *VhostRouter, exist bool) { r.mutex.RLock() defer r.mutex.RUnlock() vrs, found := r.RouterByDomain[host] if !found { return } //can't support load balance,will to do for _, vr = range vrs { if strings.HasPrefix(path, vr.location) { return vr, true } } return } //sort by location type ByLocation []*VhostRouter func (a ByLocation) Len() int { return len(a) } func (a ByLocation) Swap(i, j int) { a[i], a[j] = a[j], a[i] } func (a ByLocation) Less(i, j int) bool { return strings.Compare(a[i].location, a[j].location) < 0 }