Makes porter more conservative by trying to connect to ports before handing them out.

pull/3495/head
James Phillips 2017-09-25 17:42:53 -07:00
parent 73951d8319
commit e19f547529
No known key found for this signature in database
GPG Key ID: 77183E682AC5FC11
1 changed files with 28 additions and 12 deletions

View File

@ -6,6 +6,7 @@ import (
"fmt" "fmt"
"io/ioutil" "io/ioutil"
"log" "log"
"net"
"net/http" "net/http"
"os" "os"
"os/exec" "os/exec"
@ -97,7 +98,6 @@ func run(args []string) error {
return cmd.Run() return cmd.Run()
} }
// todo(fs): check which ports are currently bound and exclude them
func servePort(w http.ResponseWriter, r *http.Request) { func servePort(w http.ResponseWriter, r *http.Request) {
var count int var count int
n, err := strconv.Atoi(r.RequestURI[1:]) n, err := strconv.Atoi(r.RequestURI[1:])
@ -108,26 +108,42 @@ func servePort(w http.ResponseWriter, r *http.Request) {
count = 1 count = 1
} }
mu.Lock() // getPort assumes the lock is already held and tries to return a port
if port < firstPort { // that's not in use. It will panic if it has to try too many times.
port = firstPort getPort := func() int {
for i := 0; i < 10; i++ {
port++
if port < firstPort {
port = firstPort
}
if port >= lastPort {
port = firstPort
}
conn, err := net.Dial("tcp", fmt.Sprintf("127.0.0.1:%d", port))
if err != nil {
return port
}
conn.Close()
if verbose {
log.Printf("porter: skipping port %d, already in use", port)
}
}
panic(fmt.Errorf("could not find a free port"))
} }
if port+count >= lastPort {
port = firstPort
}
from, to := port, port+count
port = to
mu.Unlock()
p := make([]int, count) p := make([]int, count)
mu.Lock()
for i := 0; i < count; i++ { for i := 0; i < count; i++ {
p[i] = from + i p[i] = getPort()
} }
mu.Unlock()
if err := json.NewEncoder(w).Encode(p); err != nil { if err := json.NewEncoder(w).Encode(p); err != nil {
// this shouldn't happen so we panic since we can't recover // this shouldn't happen so we panic since we can't recover
panic(err) panic(err)
} }
if verbose { if verbose {
log.Printf("porter: allocated ports %d-%d (%d)", from, to, count) log.Printf("porter: allocated ports %v", p)
} }
} }