1
0
mirror of https://github.com/bingohuang/docker-labs.git synced 2025-07-14 18:27:25 +08:00

Add support for setting alias when creatign instance. (#140)

* Add support for setting alias when creatign instance.

The POST to create a instance now provides an `alias` field which
then can be used to access the instance services through the following
URL:

`http://<alias>-<short_session>-<port>.<tld>`

When creating a session you can now send an `alias`

* Remove unnecessary function

* Add alias support for DNS resolution
This commit is contained in:
Marcos Nils 2017-05-11 17:39:17 -03:00 committed by GitHub
parent 62c5d3761d
commit 61a0bb4db1
5 changed files with 59 additions and 6 deletions

2
api.go
View File

@ -65,6 +65,8 @@ func main() {
// Specific routes // Specific routes
r.Host(`{subdomain:.*}{node:pwd[0-9]{1,3}_[0-9]{1,3}_[0-9]{1,3}_[0-9]{1,3}}-{port:[0-9]*}.{tld:.*}`).Handler(tcpHandler) r.Host(`{subdomain:.*}{node:pwd[0-9]{1,3}_[0-9]{1,3}_[0-9]{1,3}_[0-9]{1,3}}-{port:[0-9]*}.{tld:.*}`).Handler(tcpHandler)
r.Host(`{subdomain:.*}{node:pwd[0-9]{1,3}_[0-9]{1,3}_[0-9]{1,3}_[0-9]{1,3}}.{tld:.*}`).Handler(tcpHandler) r.Host(`{subdomain:.*}{node:pwd[0-9]{1,3}_[0-9]{1,3}_[0-9]{1,3}_[0-9]{1,3}}.{tld:.*}`).Handler(tcpHandler)
r.Host(`pwd{alias:.*}-{session:.*}-{port:[0-9]*}.{tld:.*}`).Handler(tcpHandler)
r.Host(`pwd{alias:.*}-{session:.*}.{tld:.*}`).Handler(tcpHandler)
r.HandleFunc("/ping", handlers.Ping).Methods("GET") r.HandleFunc("/ping", handlers.Ping).Methods("GET")
corsRouter.HandleFunc("/instances/images", handlers.GetInstanceImages).Methods("GET") corsRouter.HandleFunc("/instances/images", handlers.GetInstanceImages).Methods("GET")
corsRouter.HandleFunc("/sessions/{sessionId}", handlers.GetSession).Methods("GET") corsRouter.HandleFunc("/sessions/{sessionId}", handlers.GetSession).Methods("GET")

View File

@ -8,16 +8,20 @@ import (
"strings" "strings"
"github.com/miekg/dns" "github.com/miekg/dns"
"github.com/play-with-docker/play-with-docker/services"
) )
var dnsFilter = regexp.MustCompile(`pwd[0-9]{1,3}_[0-9]{1,3}_[0-9]{1,3}_[0-9]{1,3}`) var dnsFilter = regexp.MustCompile(`^.*pwd([0-9]{1,3}_[0-9]{1,3}_[0-9]{1,3}_[0-9]{1,3}(?:-[0-9]{1,5})?)\..*$`)
var aliasFilter = regexp.MustCompile(`^.*pwd(.*?)-(.*?)[\.-].*`)
func DnsRequest(w dns.ResponseWriter, r *dns.Msg) { func DnsRequest(w dns.ResponseWriter, r *dns.Msg) {
if len(r.Question) > 0 && dnsFilter.MatchString(r.Question[0].Name) { if len(r.Question) > 0 && dnsFilter.MatchString(r.Question[0].Name) {
// this is something we know about and we should try to handle // this is something we know about and we should try to handle
question := r.Question[0].Name question := r.Question[0].Name
domainChunks := strings.Split(question, ".")
tldChunks := strings.Split(strings.TrimPrefix(domainChunks[0], "pwd"), "-") match := dnsFilter.FindStringSubmatch(question)
tldChunks := strings.Split(match[1], "-")
ip := strings.Replace(tldChunks[0], "_", ".", -1) ip := strings.Replace(tldChunks[0], "_", ".", -1)
m := new(dns.Msg) m := new(dns.Msg)
@ -31,6 +35,25 @@ func DnsRequest(w dns.ResponseWriter, r *dns.Msg) {
m.Answer = append(m.Answer, a) m.Answer = append(m.Answer, a)
w.WriteMsg(m) w.WriteMsg(m)
return return
} else if len(r.Question) > 0 && aliasFilter.MatchString(r.Question[0].Name) {
// this is something we know about and we should try to handle
question := r.Question[0].Name
match := aliasFilter.FindStringSubmatch(question)
i := services.FindInstanceByAlias(match[2], match[1])
m := new(dns.Msg)
m.SetReply(r)
m.Authoritative = true
m.RecursionAvailable = true
a, err := dns.NewRR(fmt.Sprintf("%s 60 IN A %s", question, i.IP))
if err != nil {
log.Fatal(err)
}
m.Answer = append(m.Answer, a)
w.WriteMsg(m)
return
} else { } else {
if len(r.Question) > 0 { if len(r.Question) > 0 {
question := r.Question[0].Name question := r.Question[0].Name

View File

@ -13,7 +13,7 @@ func NewInstance(rw http.ResponseWriter, req *http.Request) {
vars := mux.Vars(req) vars := mux.Vars(req)
sessionId := vars["sessionId"] sessionId := vars["sessionId"]
body := struct{ ImageName string }{} body := struct{ ImageName, Alias string }{}
json.NewDecoder(req.Body).Decode(&body) json.NewDecoder(req.Body).Decode(&body)
@ -26,7 +26,7 @@ func NewInstance(rw http.ResponseWriter, req *http.Request) {
return return
} }
i, err := services.NewInstance(s, body.ImageName) i, err := services.NewInstance(s, body.ImageName, body.Alias)
if err != nil { if err != nil {
log.Println(err) log.Println(err)
rw.WriteHeader(http.StatusInternalServerError) rw.WriteHeader(http.StatusInternalServerError)

View File

@ -11,11 +11,14 @@ import (
"github.com/gorilla/mux" "github.com/gorilla/mux"
"github.com/play-with-docker/play-with-docker/config" "github.com/play-with-docker/play-with-docker/config"
"github.com/play-with-docker/play-with-docker/services"
) )
func getTargetInfo(vars map[string]string, req *http.Request) (string, string) { func getTargetInfo(vars map[string]string, req *http.Request) (string, string) {
node := vars["node"] node := vars["node"]
port := vars["port"] port := vars["port"]
alias := vars["alias"]
sessionPrefix := vars["session"]
hostPort := strings.Split(req.Host, ":") hostPort := strings.Split(req.Host, ":")
// give priority to the URL host port // give priority to the URL host port
@ -25,6 +28,14 @@ func getTargetInfo(vars map[string]string, req *http.Request) (string, string) {
port = "80" port = "80"
} }
if alias != "" {
instance := services.FindInstanceByAlias(sessionPrefix, alias)
if instance != nil {
node = instance.IP
return node, port
}
}
if strings.HasPrefix(node, "pwd") { if strings.HasPrefix(node, "pwd") {
// Node is actually an ip, need to convert underscores by dots. // Node is actually an ip, need to convert underscores by dots.
ip := strings.Replace(strings.TrimPrefix(node, "pwd"), "_", ".", -1) ip := strings.Replace(strings.TrimPrefix(node, "pwd"), "_", ".", -1)

View File

@ -34,6 +34,7 @@ type Instance struct {
IsManager *bool `json:"is_manager"` IsManager *bool `json:"is_manager"`
Mem string `json:"mem"` Mem string `json:"mem"`
Cpu string `json:"cpu"` Cpu string `json:"cpu"`
Alias string `json:"alias"`
Ports UInt16Slice Ports UInt16Slice
tempPorts []uint16 `json:"-"` tempPorts []uint16 `json:"-"`
ServerCert []byte `json:"server_cert"` ServerCert []byte `json:"server_cert"`
@ -96,7 +97,7 @@ func getDindImageName() string {
return dindImage return dindImage
} }
func NewInstance(session *Session, imageName string) (*Instance, error) { func NewInstance(session *Session, imageName, alias string) (*Instance, error) {
if imageName == "" { if imageName == "" {
imageName = dindImage imageName = dindImage
} }
@ -105,6 +106,9 @@ func NewInstance(session *Session, imageName string) (*Instance, error) {
if err != nil { if err != nil {
return nil, err return nil, err
} }
instance.Alias = alias
instance.session = session instance.session = session
if session.Instances == nil { if session.Instances == nil {
@ -174,6 +178,19 @@ func FindInstanceByIP(ip string) *Instance {
return nil return nil
} }
func FindInstanceByAlias(sessionPrefix, alias string) *Instance {
for id, s := range sessions {
if strings.HasPrefix(id, sessionPrefix) {
for _, i := range s.Instances {
if i.Alias == alias {
return i
}
}
}
}
return nil
}
func DeleteInstance(session *Session, instance *Instance) error { func DeleteInstance(session *Session, instance *Instance) error {
if instance.conn != nil { if instance.conn != nil {
instance.conn.Close() instance.conn.Close()