121 lines
3.3 KiB
Go
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,
|
||
|
},
|
||
|
}
|
||
|
}
|