clair/cmd/clairctl/server/server.go
Julien Garcia Gonzalez c52c952f0a add clairctl
2017-02-27 15:02:08 +01:00

121 lines
3.3 KiB
Go

package server
import (
"crypto/tls"
"net"
"net/http"
"net/http/httputil"
"net/url"
"os"
"regexp"
"strings"
"time"
"github.com/coreos/pkg/capnslog"
"github.com/jgsqware/clairctl/clair"
"github.com/jgsqware/clairctl/config"
"github.com/jgsqware/clairctl/docker/dockerdist"
"github.com/spf13/viper"
)
var log = capnslog.NewPackageLogger("github.com/jgsqware/clairctl", "server")
//Serve run a local server with the fileserver and the reverse proxy
func Serve(sURL string) error {
go func() {
http.Handle("/v2/", newSingleHostReverseProxy())
http.Handle("/local/", http.StripPrefix("/local", restrictedFileServer(config.TmpLocal())))
listener := tcpListener(sURL)
log.Info("Starting Server on ", listener.Addr())
if err := http.Serve(listener, nil); err != nil {
log.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
}
func tcpListener(sURL string) (listener net.Listener) {
listener, err := net.Listen("tcp", sURL)
if err != nil {
log.Fatalf("cannot instanciate listener: %v", err)
}
if viper.GetInt("clairctl.port") == 0 {
port := strings.Split(listener.Addr().String(), ":")[1]
log.Debugf("Update local server port from %q to %q", "0", port)
viper.Set("clairctl.port", port)
}
return
}
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
log.Debugf("request url: %v", u)
log.Debugf("request for image: %v", config.ImageName)
if !validID.MatchString(u) {
log.Errorf("cannot parse url: %v", u)
}
var host string
host, err := clair.GetRegistryMapping(validID.FindStringSubmatch(u)[1])
log.Debugf("host retreived: %v", host)
if err != nil {
log.Errorf("response error: %v", err)
return
}
out, _ := url.Parse(host)
request.URL.Scheme = out.Scheme
request.URL.Host = out.Host
client := &http.Client{Transport: &http.Transport{
TLSClientConfig: &tls.Config{InsecureSkipVerify: viper.GetBool("auth.insecureSkipVerify")},
DisableCompression: true,
}}
log.Debugf("auth.insecureSkipVerify: %v", viper.GetBool("auth.insecureSkipVerify"))
log.Debugf("request.URL.String(): %v", request.URL.String())
req, _ := http.NewRequest("HEAD", request.URL.String(), nil)
resp, err := client.Do(req)
if err != nil {
log.Errorf("response error: %v", err)
return
}
if resp.StatusCode == http.StatusUnauthorized {
log.Info("pull from clair is unauthorized")
dockerdist.AuthenticateResponse(client, resp, request)
}
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,
Transport: &http.Transport{
TLSClientConfig: &tls.Config{InsecureSkipVerify: viper.GetBool("auth.insecureSkipVerify")},
DisableCompression: true,
},
}
}