mirror of
https://github.com/dokku/dokku.git
synced 2025-12-25 16:29:30 +01:00
261 lines
6.3 KiB
Go
261 lines
6.3 KiB
Go
package proxy
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"os"
|
|
"sort"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"github.com/dokku/dokku/plugins/common"
|
|
"github.com/dokku/dokku/plugins/config"
|
|
columnize "github.com/ryanuber/columnize"
|
|
)
|
|
|
|
// PortMap is a struct that contains a scheme:host-port:container-port mapping
|
|
type PortMap struct {
|
|
ContainerPort int
|
|
HostPort int
|
|
Scheme string
|
|
}
|
|
|
|
func (p PortMap) String() string {
|
|
return fmt.Sprintf("%s:%d:%d", p.Scheme, p.HostPort, p.ContainerPort)
|
|
}
|
|
|
|
// AllowsPersistence returns true if the port map is not to be persisted
|
|
func (p PortMap) AllowsPersistence() bool {
|
|
return p.Scheme == "__internal__"
|
|
}
|
|
|
|
func inRange(value int, min int, max int) bool {
|
|
return min < value && value < max
|
|
}
|
|
|
|
// IsAppProxyEnabled returns true if proxy is enabled; otherwise return false
|
|
func IsAppProxyEnabled(appName string) bool {
|
|
err := common.VerifyAppName(appName)
|
|
if err != nil {
|
|
common.LogFail(err.Error())
|
|
}
|
|
|
|
proxyEnabled := true
|
|
disableProxy := config.GetWithDefault(appName, "DOKKU_DISABLE_PROXY", "")
|
|
if disableProxy != "" {
|
|
proxyEnabled = false
|
|
}
|
|
return proxyEnabled
|
|
}
|
|
|
|
// ReportSingleApp is an internal function that displays the app report for one or more apps
|
|
func ReportSingleApp(appName string, infoFlag string) error {
|
|
if err := common.VerifyAppName(appName); err != nil {
|
|
return err
|
|
}
|
|
|
|
proxyEnabled := "false"
|
|
if IsAppProxyEnabled(appName) {
|
|
proxyEnabled = "true"
|
|
}
|
|
|
|
var proxyPortMap []string
|
|
for _, portMap := range getProxyPortMap(appName) {
|
|
proxyPortMap = append(proxyPortMap, portMap.String())
|
|
}
|
|
|
|
infoFlags := map[string]string{
|
|
"--proxy-enabled": proxyEnabled,
|
|
"--proxy-type": getAppProxyType(appName),
|
|
"--proxy-port-map": strings.Join(proxyPortMap, " "),
|
|
}
|
|
|
|
trimPrefix := false
|
|
uppercaseFirstCharacter := true
|
|
return common.ReportSingleApp("proxy", appName, infoFlag, infoFlags, trimPrefix, uppercaseFirstCharacter)
|
|
}
|
|
|
|
func addProxyPorts(appName string, proxyPortMap []PortMap) error {
|
|
allPortMaps := getProxyPortMap(appName)
|
|
allPortMaps = append(allPortMaps, proxyPortMap...)
|
|
|
|
return setProxyPorts(appName, allPortMaps)
|
|
}
|
|
|
|
func filterAppProxyPorts(appName string, scheme string, hostPort int) []PortMap {
|
|
var filteredProxyMaps []PortMap
|
|
for _, portMap := range getProxyPortMap(appName) {
|
|
if portMap.Scheme == scheme && portMap.HostPort == hostPort {
|
|
filteredProxyMaps = append(filteredProxyMaps, portMap)
|
|
}
|
|
}
|
|
|
|
return filteredProxyMaps
|
|
}
|
|
|
|
func getAppName(args []string) (appName string, err error) {
|
|
if len(args) >= 1 {
|
|
appName = args[0]
|
|
} else {
|
|
err = errors.New("Please specify an app to run the command on")
|
|
}
|
|
|
|
return
|
|
}
|
|
|
|
func getAppProxyType(appName string) string {
|
|
return config.GetWithDefault(appName, "DOKKU_APP_PROXY_TYPE", "nginx")
|
|
}
|
|
|
|
func getProxyPortMap(appName string) []PortMap {
|
|
value := config.GetWithDefault(appName, "DOKKU_PROXY_PORT_MAP", "")
|
|
portMaps, _ := parseProxyPortMapString(value)
|
|
return portMaps
|
|
}
|
|
|
|
func listAppProxyPorts(appName string) error {
|
|
proxyPortMap := getProxyPortMap(appName)
|
|
|
|
if len(proxyPortMap) == 0 {
|
|
return errors.New("No port mappings configured for app")
|
|
}
|
|
|
|
var lines []string
|
|
if os.Getenv("DOKKU_QUIET_OUTPUT") == "" {
|
|
lines = append(lines, "-----> scheme:host port:container port")
|
|
}
|
|
|
|
for _, portMap := range proxyPortMap {
|
|
lines = append(lines, portMap.String())
|
|
}
|
|
|
|
sort.Strings(lines)
|
|
common.LogInfo1Quiet(fmt.Sprintf("Port mappings for %s", appName))
|
|
config := columnize.DefaultConfig()
|
|
config.Delim = ":"
|
|
config.Prefix = " "
|
|
config.Empty = ""
|
|
fmt.Println(columnize.Format(lines, config))
|
|
return nil
|
|
}
|
|
|
|
func setProxyPorts(appName string, proxyPortMap []PortMap) error {
|
|
var value []string
|
|
for _, portMap := range uniqueProxyPortMap(proxyPortMap) {
|
|
if portMap.AllowsPersistence() {
|
|
continue
|
|
}
|
|
|
|
value = append(value, portMap.String())
|
|
}
|
|
|
|
sort.Strings(value)
|
|
entries := map[string]string{
|
|
"DOKKU_PROXY_PORT_MAP": strings.Join(value, " "),
|
|
}
|
|
return config.SetMany(appName, entries, false)
|
|
}
|
|
|
|
func removeProxyPorts(appName string, proxyPortMap []PortMap) error {
|
|
toRemove := map[string]bool{}
|
|
toRemoveByPort := map[int]bool{}
|
|
|
|
for _, portMap := range proxyPortMap {
|
|
if portMap.AllowsPersistence() {
|
|
toRemoveByPort[portMap.HostPort] = true
|
|
continue
|
|
}
|
|
toRemove[portMap.String()] = true
|
|
}
|
|
|
|
var toSet []PortMap
|
|
for _, portMap := range getProxyPortMap(appName) {
|
|
if toRemove[portMap.String()] {
|
|
continue
|
|
}
|
|
|
|
if toRemoveByPort[portMap.HostPort] {
|
|
continue
|
|
}
|
|
|
|
toSet = append(toSet, portMap)
|
|
}
|
|
|
|
if len(toSet) == 0 {
|
|
keys := []string{"DOKKU_PROXY_PORT_MAP"}
|
|
return config.UnsetMany(appName, keys, false)
|
|
}
|
|
|
|
return setProxyPorts(appName, toSet)
|
|
}
|
|
|
|
func parseProxyPortMapString(stringPortMap string) ([]PortMap, error) {
|
|
var proxyPortMap []PortMap
|
|
|
|
for _, v := range strings.Split(strings.TrimSpace(stringPortMap), " ") {
|
|
parts := strings.SplitN(v, ":", 3)
|
|
if len(parts) == 1 {
|
|
hostPort, err := strconv.Atoi(v)
|
|
if err != nil {
|
|
return proxyPortMap, fmt.Errorf("Invalid port map %s [err=%s]", v, err.Error())
|
|
}
|
|
|
|
if !inRange(hostPort, 0, 65536) {
|
|
return proxyPortMap, fmt.Errorf("Invalid port map %s [hostPort=%d]", v, hostPort)
|
|
}
|
|
|
|
proxyPortMap = append(proxyPortMap, PortMap{
|
|
HostPort: hostPort,
|
|
Scheme: "__internal__",
|
|
})
|
|
continue
|
|
}
|
|
|
|
if len(parts) != 3 {
|
|
return proxyPortMap, fmt.Errorf("Invalid port map %s [len=%d]", v, len(parts))
|
|
}
|
|
|
|
hostPort, err := strconv.Atoi(parts[1])
|
|
if err != nil {
|
|
return proxyPortMap, fmt.Errorf("Invalid port map %s [err=%s]", v, err.Error())
|
|
}
|
|
|
|
containerPort, err := strconv.Atoi(parts[2])
|
|
if err != nil {
|
|
return proxyPortMap, fmt.Errorf("Invalid port map %s [err=%s]", v, err.Error())
|
|
}
|
|
|
|
if !inRange(hostPort, 0, 65536) {
|
|
return proxyPortMap, fmt.Errorf("Invalid port map %s [hostPort=%d]", v, hostPort)
|
|
}
|
|
|
|
if !inRange(containerPort, 0, 65536) {
|
|
return proxyPortMap, fmt.Errorf("Invalid port map %s [containerPort=%d]", v, containerPort)
|
|
}
|
|
|
|
proxyPortMap = append(proxyPortMap, PortMap{
|
|
ContainerPort: containerPort,
|
|
HostPort: hostPort,
|
|
Scheme: parts[0],
|
|
})
|
|
}
|
|
|
|
return uniqueProxyPortMap(proxyPortMap), nil
|
|
}
|
|
|
|
func uniqueProxyPortMap(proxyPortMap []PortMap) []PortMap {
|
|
var unique []PortMap
|
|
existingPortMaps := map[string]bool{}
|
|
|
|
for _, portMap := range proxyPortMap {
|
|
if existingPortMaps[portMap.String()] {
|
|
continue
|
|
}
|
|
|
|
existingPortMaps[portMap.String()] = true
|
|
unique = append(unique, portMap)
|
|
}
|
|
|
|
return unique
|
|
}
|