2017-05-23 18:56:10 +00:00
|
|
|
package proxy
|
|
|
|
|
|
|
|
import (
|
|
|
|
"net/http"
|
|
|
|
"net/url"
|
2018-02-23 02:10:26 +00:00
|
|
|
"strings"
|
2017-05-23 18:56:10 +00:00
|
|
|
|
|
|
|
"github.com/orcaman/concurrent-map"
|
|
|
|
"github.com/portainer/portainer"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Manager represents a service used to manage Docker proxies.
|
|
|
|
type Manager struct {
|
2018-02-23 02:10:26 +00:00
|
|
|
proxyFactory *proxyFactory
|
|
|
|
proxies cmap.ConcurrentMap
|
|
|
|
extensionProxies cmap.ConcurrentMap
|
2017-05-23 18:56:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewManager initializes a new proxy Service
|
2018-03-22 22:44:43 +00:00
|
|
|
func NewManager(resourceControlService portainer.ResourceControlService, teamMembershipService portainer.TeamMembershipService, settingsService portainer.SettingsService, registryService portainer.RegistryService, dockerHubService portainer.DockerHubService) *Manager {
|
2017-05-23 18:56:10 +00:00
|
|
|
return &Manager{
|
2018-02-23 02:10:26 +00:00
|
|
|
proxies: cmap.New(),
|
|
|
|
extensionProxies: cmap.New(),
|
2017-05-23 18:56:10 +00:00
|
|
|
proxyFactory: &proxyFactory{
|
|
|
|
ResourceControlService: resourceControlService,
|
|
|
|
TeamMembershipService: teamMembershipService,
|
2017-06-01 08:14:55 +00:00
|
|
|
SettingsService: settingsService,
|
2018-03-22 22:44:43 +00:00
|
|
|
RegistryService: registryService,
|
|
|
|
DockerHubService: dockerHubService,
|
2017-05-23 18:56:10 +00:00
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// CreateAndRegisterProxy creates a new HTTP reverse proxy and adds it to the registered proxies.
|
|
|
|
// It can also be used to create a new HTTP reverse proxy and replace an already registered proxy.
|
|
|
|
func (manager *Manager) CreateAndRegisterProxy(endpoint *portainer.Endpoint) (http.Handler, error) {
|
|
|
|
var proxy http.Handler
|
|
|
|
|
|
|
|
endpointURL, err := url.Parse(endpoint.URL)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if endpointURL.Scheme == "tcp" {
|
2017-09-14 06:08:37 +00:00
|
|
|
if endpoint.TLSConfig.TLS {
|
2018-02-23 02:10:26 +00:00
|
|
|
proxy, err = manager.proxyFactory.newDockerHTTPSProxy(endpointURL, endpoint)
|
2017-05-23 18:56:10 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
} else {
|
2018-02-23 02:10:26 +00:00
|
|
|
proxy = manager.proxyFactory.newDockerHTTPProxy(endpointURL)
|
2017-05-23 18:56:10 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Assume unix:// scheme
|
2018-02-23 02:10:26 +00:00
|
|
|
proxy = manager.proxyFactory.newDockerSocketProxy(endpointURL.Path)
|
2017-05-23 18:56:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
manager.proxies.Set(string(endpoint.ID), proxy)
|
|
|
|
return proxy, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetProxy returns the proxy associated to a key
|
|
|
|
func (manager *Manager) GetProxy(key string) http.Handler {
|
|
|
|
proxy, ok := manager.proxies.Get(key)
|
|
|
|
if !ok {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return proxy.(http.Handler)
|
|
|
|
}
|
|
|
|
|
|
|
|
// DeleteProxy deletes the proxy associated to a key
|
|
|
|
func (manager *Manager) DeleteProxy(key string) {
|
|
|
|
manager.proxies.Remove(key)
|
|
|
|
}
|
2018-02-23 02:10:26 +00:00
|
|
|
|
|
|
|
// CreateAndRegisterExtensionProxy creates a new HTTP reverse proxy for an extension and adds it to the registered proxies.
|
|
|
|
func (manager *Manager) CreateAndRegisterExtensionProxy(key, extensionAPIURL string) (http.Handler, error) {
|
|
|
|
|
|
|
|
extensionURL, err := url.Parse(extensionAPIURL)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
proxy := manager.proxyFactory.newExtensionHTTPPRoxy(extensionURL)
|
|
|
|
manager.extensionProxies.Set(key, proxy)
|
|
|
|
return proxy, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetExtensionProxy returns the extension proxy associated to a key
|
|
|
|
func (manager *Manager) GetExtensionProxy(key string) http.Handler {
|
|
|
|
proxy, ok := manager.extensionProxies.Get(key)
|
|
|
|
if !ok {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return proxy.(http.Handler)
|
|
|
|
}
|
|
|
|
|
|
|
|
// DeleteExtensionProxies deletes all the extension proxies associated to a key
|
|
|
|
func (manager *Manager) DeleteExtensionProxies(key string) {
|
|
|
|
for _, k := range manager.extensionProxies.Keys() {
|
|
|
|
if strings.Contains(k, key+"_") {
|
|
|
|
manager.extensionProxies.Remove(k)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|