2019-06-20 16:36:40 -07:00
|
|
|
package api
|
|
|
|
|
|
|
|
import (
|
|
|
|
"crypto/tls"
|
|
|
|
"crypto/x509"
|
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
|
|
|
"net"
|
|
|
|
"net/http"
|
2020-02-17 13:57:15 -08:00
|
|
|
"os"
|
2020-06-25 01:21:47 -07:00
|
|
|
"time"
|
2019-06-20 16:36:40 -07:00
|
|
|
|
2019-08-27 15:01:29 -07:00
|
|
|
"github.com/anuvu/zot/errors"
|
2020-06-25 01:21:47 -07:00
|
|
|
cveinfo "github.com/anuvu/zot/pkg/extensions/search/cve"
|
2019-11-25 14:33:58 -08:00
|
|
|
"github.com/anuvu/zot/pkg/log"
|
2019-06-20 16:36:40 -07:00
|
|
|
"github.com/anuvu/zot/pkg/storage"
|
2019-11-25 14:33:58 -08:00
|
|
|
"github.com/gorilla/handlers"
|
2019-07-09 22:23:59 -07:00
|
|
|
"github.com/gorilla/mux"
|
2019-06-20 16:36:40 -07:00
|
|
|
)
|
|
|
|
|
|
|
|
type Controller struct {
|
|
|
|
Config *Config
|
2019-07-09 22:23:59 -07:00
|
|
|
Router *mux.Router
|
2019-06-20 16:36:40 -07:00
|
|
|
ImageStore *storage.ImageStore
|
2019-11-25 14:33:58 -08:00
|
|
|
Log log.Logger
|
2019-06-20 16:36:40 -07:00
|
|
|
Server *http.Server
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewController(config *Config) *Controller {
|
2019-11-25 14:33:58 -08:00
|
|
|
return &Controller{Config: config, Log: log.NewLogger(config.Log.Level, config.Log.Output)}
|
2019-06-20 16:36:40 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Controller) Run() error {
|
2019-08-15 09:34:54 -07:00
|
|
|
// validate configuration
|
|
|
|
if err := c.Config.Validate(c.Log); err != nil {
|
|
|
|
c.Log.Error().Err(err).Msg("configuration validation failed")
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// print the current configuration, but strip secrets
|
|
|
|
c.Log.Info().Interface("params", c.Config.Sanitize()).Msg("configuration settings")
|
|
|
|
|
2019-07-09 22:23:59 -07:00
|
|
|
engine := mux.NewRouter()
|
2019-11-25 14:33:58 -08:00
|
|
|
engine.Use(log.SessionLogger(c.Log), handlers.RecoveryHandler(handlers.RecoveryLogger(c.Log),
|
|
|
|
handlers.PrintRecoveryStack(false)))
|
2019-12-13 00:53:18 -05:00
|
|
|
|
2020-04-15 16:24:05 -07:00
|
|
|
c.ImageStore = storage.NewImageStore(c.Config.Storage.RootDirectory, c.Config.Storage.GC,
|
|
|
|
c.Config.Storage.Dedupe, c.Log)
|
2020-02-17 13:57:15 -08:00
|
|
|
if c.ImageStore == nil {
|
|
|
|
// we can't proceed without at least a image store
|
|
|
|
os.Exit(1)
|
|
|
|
}
|
|
|
|
|
2020-06-25 01:21:47 -07:00
|
|
|
// Updating the CVE Database
|
|
|
|
if c.Config != nil && c.Config.Extensions != nil && c.Config.Extensions.Search != nil &&
|
|
|
|
c.Config.Extensions.Search.CVE != nil {
|
|
|
|
defaultUpdateInterval, _ := time.ParseDuration("2h")
|
|
|
|
|
|
|
|
if c.Config.Extensions.Search.CVE.UpdateInterval < defaultUpdateInterval {
|
|
|
|
c.Config.Extensions.Search.CVE.UpdateInterval = defaultUpdateInterval
|
|
|
|
c.Log.Warn().Msg("CVE update interval set to too-short interval <= 1, changing update duration to 2 hours and continuing.") // nolint: lll
|
|
|
|
}
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
for {
|
|
|
|
c.Log.Info().Msg("Updating the CVE database")
|
|
|
|
|
|
|
|
err := cveinfo.UpdateCVEDb(c.Config.Storage.RootDirectory, c.Log)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
c.Log.Info().Str("Db update completed, next update scheduled after", c.Config.Extensions.Search.CVE.UpdateInterval.String()).Msg("") //nolint: lll
|
|
|
|
|
|
|
|
time.Sleep(c.Config.Extensions.Search.CVE.UpdateInterval)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
} else {
|
|
|
|
c.Log.Info().Msg("Cve config not provided, skipping cve update")
|
|
|
|
}
|
|
|
|
|
2019-06-20 16:36:40 -07:00
|
|
|
c.Router = engine
|
2019-12-23 22:32:52 -08:00
|
|
|
c.Router.UseEncodedPath()
|
2019-06-20 16:36:40 -07:00
|
|
|
_ = NewRouteHandler(c)
|
|
|
|
|
|
|
|
addr := fmt.Sprintf("%s:%s", c.Config.HTTP.Address, c.Config.HTTP.Port)
|
|
|
|
server := &http.Server{Addr: addr, Handler: c.Router}
|
|
|
|
c.Server = server
|
|
|
|
|
|
|
|
// Create the listener
|
|
|
|
l, err := net.Listen("tcp", addr)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2019-08-15 09:34:54 -07:00
|
|
|
if c.Config.HTTP.TLS != nil && c.Config.HTTP.TLS.Key != "" && c.Config.HTTP.TLS.Cert != "" {
|
2019-06-20 16:36:40 -07:00
|
|
|
if c.Config.HTTP.TLS.CACert != "" {
|
2019-08-28 14:05:16 -07:00
|
|
|
clientAuth := tls.VerifyClientCertIfGiven
|
2019-08-15 09:34:54 -07:00
|
|
|
if (c.Config.HTTP.Auth == nil || c.Config.HTTP.Auth.HTPasswd.Path == "") && !c.Config.HTTP.AllowReadAccess {
|
2019-08-28 14:05:16 -07:00
|
|
|
clientAuth = tls.RequireAndVerifyClientCert
|
|
|
|
}
|
|
|
|
|
2019-06-20 16:36:40 -07:00
|
|
|
caCert, err := ioutil.ReadFile(c.Config.HTTP.TLS.CACert)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2019-12-13 00:53:18 -05:00
|
|
|
|
2019-06-20 16:36:40 -07:00
|
|
|
caCertPool := x509.NewCertPool()
|
2019-12-13 00:53:18 -05:00
|
|
|
|
2019-08-27 15:01:29 -07:00
|
|
|
if !caCertPool.AppendCertsFromPEM(caCert) {
|
|
|
|
panic(errors.ErrBadCACert)
|
|
|
|
}
|
2019-12-13 00:53:18 -05:00
|
|
|
|
2019-06-20 16:36:40 -07:00
|
|
|
server.TLSConfig = &tls.Config{
|
2019-08-27 15:01:29 -07:00
|
|
|
ClientAuth: clientAuth,
|
|
|
|
ClientCAs: caCertPool,
|
|
|
|
PreferServerCipherSuites: true,
|
|
|
|
MinVersion: tls.VersionTLS12,
|
2019-06-20 16:36:40 -07:00
|
|
|
}
|
2020-05-11 15:13:24 -07:00
|
|
|
server.TLSConfig.BuildNameToCertificate() // nolint: staticcheck
|
2019-06-20 16:36:40 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
return server.ServeTLS(l, c.Config.HTTP.TLS.Cert, c.Config.HTTP.TLS.Key)
|
|
|
|
}
|
2019-12-13 00:53:18 -05:00
|
|
|
|
2019-06-20 16:36:40 -07:00
|
|
|
return server.Serve(l)
|
|
|
|
}
|