2016-05-02 16:23:38 +00:00
|
|
|
package server
|
|
|
|
|
|
|
|
import (
|
|
|
|
"net"
|
|
|
|
"net/http"
|
2016-05-02 19:47:14 +00:00
|
|
|
"net/http/httputil"
|
|
|
|
"net/url"
|
2016-05-02 16:23:38 +00:00
|
|
|
"os"
|
2016-05-02 19:47:14 +00:00
|
|
|
"regexp"
|
2016-05-02 16:23:38 +00:00
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/Sirupsen/logrus"
|
|
|
|
"github.com/coreos/clair/cmd/clairctl/docker"
|
2016-05-02 19:47:14 +00:00
|
|
|
"github.com/coreos/clair/cmd/clairctl/docker/httpclient"
|
2016-05-02 16:23:38 +00:00
|
|
|
"github.com/spf13/viper"
|
|
|
|
)
|
|
|
|
|
|
|
|
//Serve run a local server with the fileserver and the reverse proxy
|
|
|
|
func Serve(sURL string) error {
|
|
|
|
|
|
|
|
go func() {
|
2016-05-02 19:47:14 +00:00
|
|
|
http.Handle("/v2/", newSingleHostReverseProxy())
|
|
|
|
http.Handle("/local/", http.StripPrefix("/local", restrictedFileServer(docker.TmpLocal())))
|
2016-05-02 16:23:38 +00:00
|
|
|
|
2016-05-02 19:47:14 +00:00
|
|
|
listener := tcpListener(sURL)
|
2016-05-02 16:23:38 +00:00
|
|
|
logrus.Info("Starting Server on ", listener.Addr())
|
|
|
|
|
|
|
|
if err := http.Serve(listener, nil); err != nil {
|
|
|
|
logrus.Fatalf("local server error: %v", err)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
//sleep needed to wait the server start. Maybe use a channel for that
|
|
|
|
time.Sleep(5 * time.Millisecond)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2016-05-02 19:47:14 +00:00
|
|
|
func tcpListener(sURL string) (listener net.Listener) {
|
|
|
|
listener, err := net.Listen("tcp", sURL)
|
|
|
|
if err != nil {
|
|
|
|
logrus.Fatalf("cannot instanciate listener: %v", err)
|
|
|
|
}
|
|
|
|
|
2016-05-17 16:31:53 +00:00
|
|
|
if viper.GetInt("clairctl.port") == 0 {
|
2016-05-02 19:47:14 +00:00
|
|
|
port := strings.Split(listener.Addr().String(), ":")[1]
|
|
|
|
logrus.Debugf("Update local server port from %q to %q", "0", port)
|
2016-05-17 16:31:53 +00:00
|
|
|
viper.Set("clairctl.port", port)
|
2016-05-02 19:47:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return
|
2016-05-02 16:23:38 +00:00
|
|
|
}
|
|
|
|
|
2016-05-02 19:47:14 +00:00
|
|
|
func restrictedFileServer(path string) http.Handler {
|
|
|
|
if _, err := os.Stat(path); os.IsNotExist(err) {
|
|
|
|
os.Mkdir(path, 0777)
|
|
|
|
}
|
|
|
|
|
|
|
|
fc := func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
http.FileServer(http.Dir(path)).ServeHTTP(w, r)
|
|
|
|
}
|
|
|
|
return http.HandlerFunc(fc)
|
|
|
|
}
|
|
|
|
|
|
|
|
func newSingleHostReverseProxy() *httputil.ReverseProxy {
|
|
|
|
director := func(request *http.Request) {
|
|
|
|
|
|
|
|
var validID = regexp.MustCompile(`.*/blobs/(.*)$`)
|
|
|
|
u := request.URL.Path
|
|
|
|
logrus.Debugf("request url: %v", u)
|
|
|
|
if !validID.MatchString(u) {
|
|
|
|
logrus.Errorf("cannot parse url: %v", u)
|
|
|
|
}
|
|
|
|
|
|
|
|
host, _ := docker.GetRegistryMapping(validID.FindStringSubmatch(u)[1])
|
|
|
|
out, _ := url.Parse(host)
|
|
|
|
request.URL.Scheme = out.Scheme
|
|
|
|
request.URL.Host = out.Host
|
|
|
|
client := httpclient.Get()
|
|
|
|
req, _ := http.NewRequest("HEAD", request.URL.String(), nil)
|
|
|
|
resp, err := client.Do(req)
|
|
|
|
if err != nil {
|
|
|
|
logrus.Errorf("response error: %v", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if resp.StatusCode == http.StatusUnauthorized {
|
|
|
|
logrus.Info("pull from clair is unauthorized")
|
|
|
|
docker.AuthenticateResponse(resp, request)
|
|
|
|
}
|
2016-05-02 16:23:38 +00:00
|
|
|
|
2016-05-02 19:47:14 +00:00
|
|
|
r, _ := http.NewRequest("GET", request.URL.String(), nil)
|
|
|
|
r.Header.Set("Authorization", request.Header.Get("Authorization"))
|
|
|
|
r.Header.Set("Accept-Encoding", request.Header.Get("Accept-Encoding"))
|
|
|
|
*request = *r
|
|
|
|
}
|
|
|
|
return &httputil.ReverseProxy{
|
|
|
|
Director: director,
|
|
|
|
}
|
2016-05-02 16:23:38 +00:00
|
|
|
}
|