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