Move remotes to core/remotes
Signed-off-by: Derek McGowan <derek@mcg.dev>
This commit is contained in:
42
core/remotes/docker/config/config_unix.go
Normal file
42
core/remotes/docker/config/config_unix.go
Normal file
@@ -0,0 +1,42 @@
|
||||
//go:build !windows
|
||||
|
||||
/*
|
||||
Copyright The containerd Authors.
|
||||
|
||||
Licensed under the Apache License, Version 2.0 (the "License");
|
||||
you may not use this file except in compliance with the License.
|
||||
You may obtain a copy of the License at
|
||||
|
||||
http://www.apache.org/licenses/LICENSE-2.0
|
||||
|
||||
Unless required by applicable law or agreed to in writing, software
|
||||
distributed under the License is distributed on an "AS IS" BASIS,
|
||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
See the License for the specific language governing permissions and
|
||||
limitations under the License.
|
||||
*/
|
||||
|
||||
package config
|
||||
|
||||
import (
|
||||
"crypto/x509"
|
||||
"path/filepath"
|
||||
)
|
||||
|
||||
func hostPaths(root, host string) (hosts []string) {
|
||||
ch := hostDirectory(host)
|
||||
if ch != host {
|
||||
hosts = append(hosts, filepath.Join(root, ch))
|
||||
}
|
||||
|
||||
hosts = append(hosts,
|
||||
filepath.Join(root, host),
|
||||
filepath.Join(root, "_default"),
|
||||
)
|
||||
|
||||
return
|
||||
}
|
||||
|
||||
func rootSystemPool() (*x509.CertPool, error) {
|
||||
return x509.SystemCertPool()
|
||||
}
|
||||
41
core/remotes/docker/config/config_windows.go
Normal file
41
core/remotes/docker/config/config_windows.go
Normal file
@@ -0,0 +1,41 @@
|
||||
/*
|
||||
Copyright The containerd Authors.
|
||||
|
||||
Licensed under the Apache License, Version 2.0 (the "License");
|
||||
you may not use this file except in compliance with the License.
|
||||
You may obtain a copy of the License at
|
||||
|
||||
http://www.apache.org/licenses/LICENSE-2.0
|
||||
|
||||
Unless required by applicable law or agreed to in writing, software
|
||||
distributed under the License is distributed on an "AS IS" BASIS,
|
||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
See the License for the specific language governing permissions and
|
||||
limitations under the License.
|
||||
*/
|
||||
|
||||
package config
|
||||
|
||||
import (
|
||||
"crypto/x509"
|
||||
"path/filepath"
|
||||
"strings"
|
||||
)
|
||||
|
||||
func hostPaths(root, host string) (hosts []string) {
|
||||
ch := hostDirectory(host)
|
||||
if ch != host {
|
||||
hosts = append(hosts, filepath.Join(root, strings.Replace(ch, ":", "", -1)))
|
||||
}
|
||||
|
||||
hosts = append(hosts,
|
||||
filepath.Join(root, strings.Replace(host, ":", "", -1)),
|
||||
filepath.Join(root, "_default"),
|
||||
)
|
||||
|
||||
return
|
||||
}
|
||||
|
||||
func rootSystemPool() (*x509.CertPool, error) {
|
||||
return x509.NewCertPool(), nil
|
||||
}
|
||||
44
core/remotes/docker/config/docker_fuzzer_internal.go
Normal file
44
core/remotes/docker/config/docker_fuzzer_internal.go
Normal file
@@ -0,0 +1,44 @@
|
||||
//go:build gofuzz
|
||||
|
||||
/*
|
||||
Copyright The containerd Authors.
|
||||
|
||||
Licensed under the Apache License, Version 2.0 (the "License");
|
||||
you may not use this file except in compliance with the License.
|
||||
You may obtain a copy of the License at
|
||||
|
||||
http://www.apache.org/licenses/LICENSE-2.0
|
||||
|
||||
Unless required by applicable law or agreed to in writing, software
|
||||
distributed under the License is distributed on an "AS IS" BASIS,
|
||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
See the License for the specific language governing permissions and
|
||||
limitations under the License.
|
||||
*/
|
||||
|
||||
package config
|
||||
|
||||
import (
|
||||
"os"
|
||||
|
||||
fuzz "github.com/AdaLogics/go-fuzz-headers"
|
||||
)
|
||||
|
||||
func FuzzParseHostsFile(data []byte) int {
|
||||
f := fuzz.NewConsumer(data)
|
||||
dir, err := os.MkdirTemp("", "fuzz-")
|
||||
if err != nil {
|
||||
return 0
|
||||
}
|
||||
err = f.CreateFiles(dir)
|
||||
if err != nil {
|
||||
return 0
|
||||
}
|
||||
defer os.RemoveAll(dir)
|
||||
b, err := f.GetBytes()
|
||||
if err != nil {
|
||||
return 0
|
||||
}
|
||||
_, _ = parseHostsFile(dir, b)
|
||||
return 1
|
||||
}
|
||||
617
core/remotes/docker/config/hosts.go
Normal file
617
core/remotes/docker/config/hosts.go
Normal file
@@ -0,0 +1,617 @@
|
||||
/*
|
||||
Copyright The containerd Authors.
|
||||
|
||||
Licensed under the Apache License, Version 2.0 (the "License");
|
||||
you may not use this file except in compliance with the License.
|
||||
You may obtain a copy of the License at
|
||||
|
||||
http://www.apache.org/licenses/LICENSE-2.0
|
||||
|
||||
Unless required by applicable law or agreed to in writing, software
|
||||
distributed under the License is distributed on an "AS IS" BASIS,
|
||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
See the License for the specific language governing permissions and
|
||||
limitations under the License.
|
||||
*/
|
||||
|
||||
// Package config contains utilities for helping configure the Docker resolver
|
||||
package config
|
||||
|
||||
import (
|
||||
"context"
|
||||
"crypto/tls"
|
||||
"fmt"
|
||||
"net"
|
||||
"net/http"
|
||||
"net/url"
|
||||
"os"
|
||||
"path"
|
||||
"path/filepath"
|
||||
"strings"
|
||||
"time"
|
||||
|
||||
"github.com/containerd/containerd/v2/core/remotes/docker"
|
||||
"github.com/containerd/containerd/v2/errdefs"
|
||||
"github.com/containerd/log"
|
||||
"github.com/pelletier/go-toml/v2"
|
||||
tomlu "github.com/pelletier/go-toml/v2/unstable"
|
||||
)
|
||||
|
||||
// UpdateClientFunc is a function that lets you to amend http Client behavior used by registry clients.
|
||||
type UpdateClientFunc func(client *http.Client) error
|
||||
|
||||
type hostConfig struct {
|
||||
scheme string
|
||||
host string
|
||||
path string
|
||||
|
||||
capabilities docker.HostCapabilities
|
||||
|
||||
caCerts []string
|
||||
clientPairs [][2]string
|
||||
skipVerify *bool
|
||||
|
||||
header http.Header
|
||||
|
||||
// TODO: Add credential configuration (domain alias, username)
|
||||
}
|
||||
|
||||
// HostOptions is used to configure registry hosts
|
||||
type HostOptions struct {
|
||||
HostDir func(string) (string, error)
|
||||
Credentials func(host string) (string, string, error)
|
||||
DefaultTLS *tls.Config
|
||||
DefaultScheme string
|
||||
// UpdateClient will be called after creating http.Client object, so clients can provide extra configuration
|
||||
UpdateClient UpdateClientFunc
|
||||
AuthorizerOpts []docker.AuthorizerOpt
|
||||
}
|
||||
|
||||
// ConfigureHosts creates a registry hosts function from the provided
|
||||
// host creation options. The host directory can read hosts.toml or
|
||||
// certificate files laid out in the Docker specific layout.
|
||||
// If a `HostDir` function is not required, defaults are used.
|
||||
func ConfigureHosts(ctx context.Context, options HostOptions) docker.RegistryHosts {
|
||||
return func(host string) ([]docker.RegistryHost, error) {
|
||||
var hosts []hostConfig
|
||||
if options.HostDir != nil {
|
||||
dir, err := options.HostDir(host)
|
||||
if err != nil && !errdefs.IsNotFound(err) {
|
||||
return nil, err
|
||||
}
|
||||
if dir != "" {
|
||||
log.G(ctx).WithField("dir", dir).Debug("loading host directory")
|
||||
hosts, err = loadHostDir(ctx, dir)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// If hosts was not set, add a default host
|
||||
// NOTE: Check nil here and not empty, the host may be
|
||||
// intentionally configured to not have any endpoints
|
||||
if hosts == nil {
|
||||
hosts = make([]hostConfig, 1)
|
||||
}
|
||||
if len(hosts) > 0 && hosts[len(hosts)-1].host == "" {
|
||||
if host == "docker.io" {
|
||||
hosts[len(hosts)-1].scheme = "https"
|
||||
hosts[len(hosts)-1].host = "registry-1.docker.io"
|
||||
} else if docker.IsLocalhost(host) {
|
||||
hosts[len(hosts)-1].host = host
|
||||
if options.DefaultScheme == "" {
|
||||
_, port, _ := net.SplitHostPort(host)
|
||||
if port == "" || port == "443" {
|
||||
// If port is default or 443, only use https
|
||||
hosts[len(hosts)-1].scheme = "https"
|
||||
} else {
|
||||
// HTTP fallback logic will be used when protocol is ambiguous
|
||||
hosts[len(hosts)-1].scheme = "http"
|
||||
}
|
||||
|
||||
// When port is 80, protocol is not ambiguous
|
||||
if port != "80" {
|
||||
// Skipping TLS verification for localhost
|
||||
var skipVerify = true
|
||||
hosts[len(hosts)-1].skipVerify = &skipVerify
|
||||
}
|
||||
} else {
|
||||
hosts[len(hosts)-1].scheme = options.DefaultScheme
|
||||
}
|
||||
} else {
|
||||
hosts[len(hosts)-1].host = host
|
||||
if options.DefaultScheme != "" {
|
||||
hosts[len(hosts)-1].scheme = options.DefaultScheme
|
||||
} else {
|
||||
hosts[len(hosts)-1].scheme = "https"
|
||||
}
|
||||
}
|
||||
hosts[len(hosts)-1].path = "/v2"
|
||||
hosts[len(hosts)-1].capabilities = docker.HostCapabilityPull | docker.HostCapabilityResolve | docker.HostCapabilityPush
|
||||
}
|
||||
|
||||
// tlsConfigured indicates that TLS was configured and HTTP endpoints should
|
||||
// attempt to use the TLS configuration before falling back to HTTP
|
||||
var tlsConfigured bool
|
||||
|
||||
var defaultTLSConfig *tls.Config
|
||||
if options.DefaultTLS != nil {
|
||||
tlsConfigured = true
|
||||
defaultTLSConfig = options.DefaultTLS
|
||||
} else {
|
||||
defaultTLSConfig = &tls.Config{}
|
||||
}
|
||||
|
||||
defaultTransport := &http.Transport{
|
||||
Proxy: http.ProxyFromEnvironment,
|
||||
DialContext: (&net.Dialer{
|
||||
Timeout: 30 * time.Second,
|
||||
KeepAlive: 30 * time.Second,
|
||||
FallbackDelay: 300 * time.Millisecond,
|
||||
}).DialContext,
|
||||
MaxIdleConns: 10,
|
||||
IdleConnTimeout: 30 * time.Second,
|
||||
TLSHandshakeTimeout: 10 * time.Second,
|
||||
TLSClientConfig: defaultTLSConfig,
|
||||
ExpectContinueTimeout: 5 * time.Second,
|
||||
}
|
||||
|
||||
client := &http.Client{
|
||||
Transport: defaultTransport,
|
||||
}
|
||||
if options.UpdateClient != nil {
|
||||
if err := options.UpdateClient(client); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
}
|
||||
|
||||
authOpts := []docker.AuthorizerOpt{docker.WithAuthClient(client)}
|
||||
if options.Credentials != nil {
|
||||
authOpts = append(authOpts, docker.WithAuthCreds(options.Credentials))
|
||||
}
|
||||
authOpts = append(authOpts, options.AuthorizerOpts...)
|
||||
authorizer := docker.NewDockerAuthorizer(authOpts...)
|
||||
|
||||
rhosts := make([]docker.RegistryHost, len(hosts))
|
||||
for i, host := range hosts {
|
||||
// Allow setting for each host as well
|
||||
explicitTLS := tlsConfigured
|
||||
|
||||
if host.caCerts != nil || host.clientPairs != nil || host.skipVerify != nil {
|
||||
explicitTLS = true
|
||||
tr := defaultTransport.Clone()
|
||||
tlsConfig := tr.TLSClientConfig
|
||||
if host.skipVerify != nil {
|
||||
tlsConfig.InsecureSkipVerify = *host.skipVerify
|
||||
}
|
||||
if host.caCerts != nil {
|
||||
if tlsConfig.RootCAs == nil {
|
||||
rootPool, err := rootSystemPool()
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("unable to initialize cert pool: %w", err)
|
||||
}
|
||||
tlsConfig.RootCAs = rootPool
|
||||
}
|
||||
for _, f := range host.caCerts {
|
||||
data, err := os.ReadFile(f)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("unable to read CA cert %q: %w", f, err)
|
||||
}
|
||||
if !tlsConfig.RootCAs.AppendCertsFromPEM(data) {
|
||||
return nil, fmt.Errorf("unable to load CA cert %q", f)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
for _, pair := range host.clientPairs {
|
||||
certPEMBlock, err := os.ReadFile(pair[0])
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("unable to read CERT file %q: %w", pair[0], err)
|
||||
}
|
||||
var keyPEMBlock []byte
|
||||
if pair[1] != "" {
|
||||
keyPEMBlock, err = os.ReadFile(pair[1])
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("unable to read CERT file %q: %w", pair[1], err)
|
||||
}
|
||||
} else {
|
||||
// Load key block from same PEM file
|
||||
keyPEMBlock = certPEMBlock
|
||||
}
|
||||
cert, err := tls.X509KeyPair(certPEMBlock, keyPEMBlock)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("failed to load X509 key pair: %w", err)
|
||||
}
|
||||
|
||||
tlsConfig.Certificates = append(tlsConfig.Certificates, cert)
|
||||
}
|
||||
|
||||
c := *client
|
||||
c.Transport = tr
|
||||
if options.UpdateClient != nil {
|
||||
if err := options.UpdateClient(&c); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
}
|
||||
|
||||
rhosts[i].Client = &c
|
||||
rhosts[i].Authorizer = docker.NewDockerAuthorizer(append(authOpts, docker.WithAuthClient(&c))...)
|
||||
} else {
|
||||
rhosts[i].Client = client
|
||||
rhosts[i].Authorizer = authorizer
|
||||
}
|
||||
|
||||
// When TLS has been configured for the operation or host and
|
||||
// the protocol from the port number is ambiguous, use the
|
||||
// docker.HTTPFallback roundtripper to catch TLS errors and re-attempt the
|
||||
// request as http. This allows preference for https when configured but
|
||||
// also catches TLS errors early enough in the request to avoid sending
|
||||
// the request twice or consuming the request body.
|
||||
if host.scheme == "http" && explicitTLS {
|
||||
_, port, _ := net.SplitHostPort(host.host)
|
||||
if port != "" && port != "80" {
|
||||
log.G(ctx).WithField("host", host.host).Info("host will try HTTPS first since it is configured for HTTP with a TLS configuration, consider changing host to HTTPS or removing unused TLS configuration")
|
||||
host.scheme = "https"
|
||||
rhosts[i].Client.Transport = docker.HTTPFallback{RoundTripper: rhosts[i].Client.Transport}
|
||||
}
|
||||
}
|
||||
|
||||
rhosts[i].Scheme = host.scheme
|
||||
rhosts[i].Host = host.host
|
||||
rhosts[i].Path = host.path
|
||||
rhosts[i].Capabilities = host.capabilities
|
||||
rhosts[i].Header = host.header
|
||||
}
|
||||
|
||||
return rhosts, nil
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
// HostDirFromRoot returns a function which finds a host directory
|
||||
// based at the given root.
|
||||
func HostDirFromRoot(root string) func(string) (string, error) {
|
||||
return func(host string) (string, error) {
|
||||
for _, p := range hostPaths(root, host) {
|
||||
if _, err := os.Stat(p); err == nil {
|
||||
return p, nil
|
||||
} else if !os.IsNotExist(err) {
|
||||
return "", err
|
||||
}
|
||||
}
|
||||
return "", errdefs.ErrNotFound
|
||||
}
|
||||
}
|
||||
|
||||
// hostDirectory converts ":port" to "_port_" in directory names
|
||||
func hostDirectory(host string) string {
|
||||
idx := strings.LastIndex(host, ":")
|
||||
if idx > 0 {
|
||||
return host[:idx] + "_" + host[idx+1:] + "_"
|
||||
}
|
||||
return host
|
||||
}
|
||||
|
||||
func loadHostDir(ctx context.Context, hostsDir string) ([]hostConfig, error) {
|
||||
b, err := os.ReadFile(filepath.Join(hostsDir, "hosts.toml"))
|
||||
if err != nil && !os.IsNotExist(err) {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
if len(b) == 0 {
|
||||
// If hosts.toml does not exist, fallback to checking for
|
||||
// certificate files based on Docker's certificate file
|
||||
// pattern (".crt", ".cert", ".key" files)
|
||||
return loadCertFiles(ctx, hostsDir)
|
||||
}
|
||||
|
||||
hosts, err := parseHostsFile(hostsDir, b)
|
||||
if err != nil {
|
||||
log.G(ctx).WithError(err).Error("failed to decode hosts.toml")
|
||||
// Fallback to checking certificate files
|
||||
return loadCertFiles(ctx, hostsDir)
|
||||
}
|
||||
|
||||
return hosts, nil
|
||||
}
|
||||
|
||||
type hostFileConfig struct {
|
||||
// Capabilities determine what operations a host is
|
||||
// capable of performing. Allowed values
|
||||
// - pull
|
||||
// - resolve
|
||||
// - push
|
||||
Capabilities []string `toml:"capabilities"`
|
||||
|
||||
// CACert are the public key certificates for TLS
|
||||
// Accepted types
|
||||
// - string - Single file with certificate(s)
|
||||
// - []string - Multiple files with certificates
|
||||
CACert interface{} `toml:"ca"`
|
||||
|
||||
// Client keypair(s) for TLS with client authentication
|
||||
// Accepted types
|
||||
// - string - Single file with public and private keys
|
||||
// - []string - Multiple files with public and private keys
|
||||
// - [][2]string - Multiple keypairs with public and private keys in separate files
|
||||
Client interface{} `toml:"client"`
|
||||
|
||||
// SkipVerify skips verification of the server's certificate chain
|
||||
// and host name. This should only be used for testing or in
|
||||
// combination with other methods of verifying connections.
|
||||
SkipVerify *bool `toml:"skip_verify"`
|
||||
|
||||
// Header are additional header files to send to the server
|
||||
Header map[string]interface{} `toml:"header"`
|
||||
|
||||
// OverridePath indicates the API root endpoint is defined in the URL
|
||||
// path rather than by the API specification.
|
||||
// This may be used with non-compliant OCI registries to override the
|
||||
// API root endpoint.
|
||||
OverridePath bool `toml:"override_path"`
|
||||
|
||||
// TODO: Credentials: helper? name? username? alternate domain? token?
|
||||
}
|
||||
|
||||
func parseHostsFile(baseDir string, b []byte) ([]hostConfig, error) {
|
||||
orderedHosts, err := getSortedHosts(b)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
c := struct {
|
||||
hostFileConfig
|
||||
// Server specifies the default server. When `host` is
|
||||
// also specified, those hosts are tried first.
|
||||
Server string `toml:"server"`
|
||||
// HostConfigs store the per-host configuration
|
||||
HostConfigs map[string]hostFileConfig `toml:"host"`
|
||||
}{}
|
||||
|
||||
var (
|
||||
hosts []hostConfig
|
||||
)
|
||||
|
||||
if err := toml.Unmarshal(b, &c); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
// Parse hosts array
|
||||
for _, host := range orderedHosts {
|
||||
config := c.HostConfigs[host]
|
||||
|
||||
parsed, err := parseHostConfig(host, baseDir, config)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
hosts = append(hosts, parsed)
|
||||
}
|
||||
|
||||
// Parse root host config and append it as the last element
|
||||
parsed, err := parseHostConfig(c.Server, baseDir, c.hostFileConfig)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
hosts = append(hosts, parsed)
|
||||
|
||||
return hosts, nil
|
||||
}
|
||||
|
||||
func parseHostConfig(server string, baseDir string, config hostFileConfig) (hostConfig, error) {
|
||||
var (
|
||||
result = hostConfig{}
|
||||
err error
|
||||
)
|
||||
|
||||
if server != "" {
|
||||
if !strings.HasPrefix(server, "http") {
|
||||
server = "https://" + server
|
||||
}
|
||||
u, err := url.Parse(server)
|
||||
if err != nil {
|
||||
return hostConfig{}, fmt.Errorf("unable to parse server %v: %w", server, err)
|
||||
}
|
||||
result.scheme = u.Scheme
|
||||
result.host = u.Host
|
||||
if len(u.Path) > 0 {
|
||||
u.Path = path.Clean(u.Path)
|
||||
if !strings.HasSuffix(u.Path, "/v2") && !config.OverridePath {
|
||||
u.Path = u.Path + "/v2"
|
||||
}
|
||||
} else if !config.OverridePath {
|
||||
u.Path = "/v2"
|
||||
}
|
||||
result.path = u.Path
|
||||
}
|
||||
|
||||
result.skipVerify = config.SkipVerify
|
||||
|
||||
if len(config.Capabilities) > 0 {
|
||||
for _, c := range config.Capabilities {
|
||||
switch strings.ToLower(c) {
|
||||
case "pull":
|
||||
result.capabilities |= docker.HostCapabilityPull
|
||||
case "resolve":
|
||||
result.capabilities |= docker.HostCapabilityResolve
|
||||
case "push":
|
||||
result.capabilities |= docker.HostCapabilityPush
|
||||
default:
|
||||
return hostConfig{}, fmt.Errorf("unknown capability %v", c)
|
||||
}
|
||||
}
|
||||
} else {
|
||||
result.capabilities = docker.HostCapabilityPull | docker.HostCapabilityResolve | docker.HostCapabilityPush
|
||||
}
|
||||
|
||||
if config.CACert != nil {
|
||||
switch cert := config.CACert.(type) {
|
||||
case string:
|
||||
result.caCerts = []string{makeAbsPath(cert, baseDir)}
|
||||
case []interface{}:
|
||||
result.caCerts, err = makeStringSlice(cert, func(p string) string {
|
||||
return makeAbsPath(p, baseDir)
|
||||
})
|
||||
if err != nil {
|
||||
return hostConfig{}, err
|
||||
}
|
||||
default:
|
||||
return hostConfig{}, fmt.Errorf("invalid type %v for \"ca\"", cert)
|
||||
}
|
||||
}
|
||||
|
||||
if config.Client != nil {
|
||||
switch client := config.Client.(type) {
|
||||
case string:
|
||||
result.clientPairs = [][2]string{{makeAbsPath(client, baseDir), ""}}
|
||||
case []interface{}:
|
||||
// []string or [][2]string
|
||||
for _, pairs := range client {
|
||||
switch p := pairs.(type) {
|
||||
case string:
|
||||
result.clientPairs = append(result.clientPairs, [2]string{makeAbsPath(p, baseDir), ""})
|
||||
case []interface{}:
|
||||
slice, err := makeStringSlice(p, func(s string) string {
|
||||
return makeAbsPath(s, baseDir)
|
||||
})
|
||||
if err != nil {
|
||||
return hostConfig{}, err
|
||||
}
|
||||
if len(slice) != 2 {
|
||||
return hostConfig{}, fmt.Errorf("invalid pair %v for \"client\"", p)
|
||||
}
|
||||
|
||||
var pair [2]string
|
||||
copy(pair[:], slice)
|
||||
result.clientPairs = append(result.clientPairs, pair)
|
||||
default:
|
||||
return hostConfig{}, fmt.Errorf("invalid type %T for \"client\"", p)
|
||||
}
|
||||
}
|
||||
default:
|
||||
return hostConfig{}, fmt.Errorf("invalid type %v for \"client\"", client)
|
||||
}
|
||||
}
|
||||
|
||||
if config.Header != nil {
|
||||
header := http.Header{}
|
||||
for key, ty := range config.Header {
|
||||
switch value := ty.(type) {
|
||||
case string:
|
||||
header[key] = []string{value}
|
||||
case []interface{}:
|
||||
header[key], err = makeStringSlice(value, nil)
|
||||
if err != nil {
|
||||
return hostConfig{}, err
|
||||
}
|
||||
default:
|
||||
return hostConfig{}, fmt.Errorf("invalid type %v for header %q", ty, key)
|
||||
}
|
||||
}
|
||||
result.header = header
|
||||
}
|
||||
|
||||
return result, nil
|
||||
}
|
||||
|
||||
// getSortedHosts returns the list of hosts in the order are they defined in the file.
|
||||
func getSortedHosts(b []byte) ([]string, error) {
|
||||
var hostsInOrder []string
|
||||
|
||||
// Use toml unstable package for directly parsing toml
|
||||
// See https://github.com/pelletier/go-toml/discussions/801#discussioncomment-7083586
|
||||
p := tomlu.Parser{}
|
||||
p.Reset(b)
|
||||
|
||||
var host string
|
||||
// iterate over all top level expressions
|
||||
for p.NextExpression() {
|
||||
e := p.Expression()
|
||||
|
||||
if e.Kind != tomlu.Table {
|
||||
continue
|
||||
}
|
||||
|
||||
// Let's look at the key. It's an iterator over the multiple dotted parts of the key.
|
||||
var parts []string
|
||||
for it := e.Key(); it.Next(); {
|
||||
parts = append(parts, string(it.Node().Data))
|
||||
}
|
||||
|
||||
// only consider keys that look like `hosts.XXX`
|
||||
// and skip subtables such as `hosts.XXX.header`
|
||||
if len(parts) < 2 || parts[0] != "host" || parts[1] == host {
|
||||
continue
|
||||
}
|
||||
|
||||
host = parts[1]
|
||||
hostsInOrder = append(hostsInOrder, host)
|
||||
}
|
||||
|
||||
return hostsInOrder, nil
|
||||
}
|
||||
|
||||
// makeStringSlice is a helper func to convert from []interface{} to []string.
|
||||
// Additionally an optional cb func may be passed to perform string mapping.
|
||||
func makeStringSlice(slice []interface{}, cb func(string) string) ([]string, error) {
|
||||
out := make([]string, len(slice))
|
||||
for i, value := range slice {
|
||||
str, ok := value.(string)
|
||||
if !ok {
|
||||
return nil, fmt.Errorf("unable to cast %v to string", value)
|
||||
}
|
||||
|
||||
if cb != nil {
|
||||
out[i] = cb(str)
|
||||
} else {
|
||||
out[i] = str
|
||||
}
|
||||
}
|
||||
return out, nil
|
||||
}
|
||||
|
||||
func makeAbsPath(p string, base string) string {
|
||||
if filepath.IsAbs(p) {
|
||||
return p
|
||||
}
|
||||
return filepath.Join(base, p)
|
||||
}
|
||||
|
||||
// loadCertsDir loads certs from certsDir like "/etc/docker/certs.d" .
|
||||
// Compatible with Docker file layout
|
||||
// - files ending with ".crt" are treated as CA certificate files
|
||||
// - files ending with ".cert" are treated as client certificates, and
|
||||
// files with the same name but ending with ".key" are treated as the
|
||||
// corresponding private key.
|
||||
// NOTE: If a ".key" file is missing, this function will just return
|
||||
// the ".cert", which may contain the private key. If the ".cert" file
|
||||
// does not contain the private key, the caller should detect and error.
|
||||
func loadCertFiles(ctx context.Context, certsDir string) ([]hostConfig, error) {
|
||||
fs, err := os.ReadDir(certsDir)
|
||||
if err != nil && !os.IsNotExist(err) {
|
||||
return nil, err
|
||||
}
|
||||
hosts := make([]hostConfig, 1)
|
||||
for _, f := range fs {
|
||||
if f.IsDir() {
|
||||
continue
|
||||
}
|
||||
if strings.HasSuffix(f.Name(), ".crt") {
|
||||
hosts[0].caCerts = append(hosts[0].caCerts, filepath.Join(certsDir, f.Name()))
|
||||
}
|
||||
if strings.HasSuffix(f.Name(), ".cert") {
|
||||
var pair [2]string
|
||||
certFile := f.Name()
|
||||
pair[0] = filepath.Join(certsDir, certFile)
|
||||
// Check if key also exists
|
||||
keyFile := filepath.Join(certsDir, certFile[:len(certFile)-5]+".key")
|
||||
if _, err := os.Stat(keyFile); err == nil {
|
||||
pair[1] = keyFile
|
||||
} else if !os.IsNotExist(err) {
|
||||
return nil, err
|
||||
}
|
||||
hosts[0].clientPairs = append(hosts[0].clientPairs, pair)
|
||||
}
|
||||
}
|
||||
return hosts, nil
|
||||
}
|
||||
609
core/remotes/docker/config/hosts_test.go
Normal file
609
core/remotes/docker/config/hosts_test.go
Normal file
@@ -0,0 +1,609 @@
|
||||
/*
|
||||
Copyright The containerd Authors.
|
||||
|
||||
Licensed under the Apache License, Version 2.0 (the "License");
|
||||
you may not use this file except in compliance with the License.
|
||||
You may obtain a copy of the License at
|
||||
|
||||
http://www.apache.org/licenses/LICENSE-2.0
|
||||
|
||||
Unless required by applicable law or agreed to in writing, software
|
||||
distributed under the License is distributed on an "AS IS" BASIS,
|
||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
See the License for the specific language governing permissions and
|
||||
limitations under the License.
|
||||
*/
|
||||
|
||||
package config
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"context"
|
||||
"crypto/tls"
|
||||
"fmt"
|
||||
"net/http"
|
||||
"os"
|
||||
"path/filepath"
|
||||
"testing"
|
||||
|
||||
"github.com/containerd/containerd/v2/core/remotes/docker"
|
||||
"github.com/containerd/log/logtest"
|
||||
)
|
||||
|
||||
const allCaps = docker.HostCapabilityPull | docker.HostCapabilityResolve | docker.HostCapabilityPush
|
||||
|
||||
func TestDefaultHosts(t *testing.T) {
|
||||
ctx := logtest.WithT(context.Background(), t)
|
||||
resolve := ConfigureHosts(ctx, HostOptions{})
|
||||
|
||||
for _, tc := range []struct {
|
||||
host string
|
||||
expected []docker.RegistryHost
|
||||
}{
|
||||
{
|
||||
host: "docker.io",
|
||||
expected: []docker.RegistryHost{
|
||||
{
|
||||
Scheme: "https",
|
||||
Host: "registry-1.docker.io",
|
||||
Path: "/v2",
|
||||
Capabilities: allCaps,
|
||||
},
|
||||
},
|
||||
},
|
||||
} {
|
||||
hosts, err := resolve(tc.host)
|
||||
if err != nil {
|
||||
t.Errorf("[%s] resolve failed: %v", tc.host, err)
|
||||
continue
|
||||
}
|
||||
if len(hosts) != len(tc.expected) {
|
||||
t.Errorf("[%s] unexpected number of hosts %d, expected %d", tc.host, len(hosts), len(tc.expected))
|
||||
continue
|
||||
}
|
||||
for j := range hosts {
|
||||
if !compareRegistryHost(hosts[j], tc.expected[j]) {
|
||||
|
||||
t.Errorf("[%s] [%d] unexpected host %v, expected %v", tc.host, j, hosts[j], tc.expected[j])
|
||||
break
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
func TestParseHostFile(t *testing.T) {
|
||||
const testtoml = `
|
||||
server = "https://test-default.registry"
|
||||
ca = "/etc/path/default"
|
||||
[header]
|
||||
x-custom-1 = "custom header"
|
||||
|
||||
[host."https://mirror.registry"]
|
||||
capabilities = ["pull"]
|
||||
ca = "/etc/certs/mirror.pem"
|
||||
skip_verify = false
|
||||
[host."https://mirror.registry".header]
|
||||
x-custom-2 = ["value1", "value2"]
|
||||
|
||||
[host."https://mirror-bak.registry/us"]
|
||||
capabilities = ["pull"]
|
||||
skip_verify = true
|
||||
|
||||
[host."http://mirror.registry"]
|
||||
capabilities = ["pull"]
|
||||
|
||||
[host."https://test-1.registry"]
|
||||
capabilities = ["pull", "resolve", "push"]
|
||||
ca = ["/etc/certs/test-1-ca.pem", "/etc/certs/special.pem"]
|
||||
client = [["/etc/certs/client.cert", "/etc/certs/client.key"],["/etc/certs/client.pem", ""]]
|
||||
|
||||
[host."https://test-2.registry"]
|
||||
client = "/etc/certs/client.pem"
|
||||
|
||||
[host."https://test-3.registry"]
|
||||
client = ["/etc/certs/client-1.pem", "/etc/certs/client-2.pem"]
|
||||
|
||||
[host."https://noncompliantmirror.registry/v2/namespaceprefix"]
|
||||
capabilities = ["pull"]
|
||||
override_path = true
|
||||
|
||||
[host."https://noprefixnoncompliant.registry"]
|
||||
override_path = true
|
||||
|
||||
[host."https://onlyheader.registry".header]
|
||||
x-custom-1 = "justaheader"
|
||||
`
|
||||
var tb, fb = true, false
|
||||
expected := []hostConfig{
|
||||
{
|
||||
scheme: "https",
|
||||
host: "mirror.registry",
|
||||
path: "/v2",
|
||||
capabilities: docker.HostCapabilityPull,
|
||||
caCerts: []string{filepath.FromSlash("/etc/certs/mirror.pem")},
|
||||
skipVerify: &fb,
|
||||
header: http.Header{"x-custom-2": {"value1", "value2"}},
|
||||
},
|
||||
{
|
||||
scheme: "https",
|
||||
host: "mirror-bak.registry",
|
||||
path: "/us/v2",
|
||||
capabilities: docker.HostCapabilityPull,
|
||||
skipVerify: &tb,
|
||||
},
|
||||
{
|
||||
scheme: "http",
|
||||
host: "mirror.registry",
|
||||
path: "/v2",
|
||||
capabilities: docker.HostCapabilityPull,
|
||||
},
|
||||
{
|
||||
scheme: "https",
|
||||
host: "test-1.registry",
|
||||
path: "/v2",
|
||||
capabilities: allCaps,
|
||||
caCerts: []string{filepath.FromSlash("/etc/certs/test-1-ca.pem"), filepath.FromSlash("/etc/certs/special.pem")},
|
||||
clientPairs: [][2]string{
|
||||
{filepath.FromSlash("/etc/certs/client.cert"), filepath.FromSlash("/etc/certs/client.key")},
|
||||
{filepath.FromSlash("/etc/certs/client.pem"), ""},
|
||||
},
|
||||
},
|
||||
{
|
||||
scheme: "https",
|
||||
host: "test-2.registry",
|
||||
path: "/v2",
|
||||
capabilities: allCaps,
|
||||
clientPairs: [][2]string{
|
||||
{filepath.FromSlash("/etc/certs/client.pem")},
|
||||
},
|
||||
},
|
||||
{
|
||||
scheme: "https",
|
||||
host: "test-3.registry",
|
||||
path: "/v2",
|
||||
capabilities: allCaps,
|
||||
clientPairs: [][2]string{
|
||||
{filepath.FromSlash("/etc/certs/client-1.pem")},
|
||||
{filepath.FromSlash("/etc/certs/client-2.pem")},
|
||||
},
|
||||
},
|
||||
{
|
||||
scheme: "https",
|
||||
host: "noncompliantmirror.registry",
|
||||
path: "/v2/namespaceprefix",
|
||||
capabilities: docker.HostCapabilityPull,
|
||||
},
|
||||
{
|
||||
scheme: "https",
|
||||
host: "noprefixnoncompliant.registry",
|
||||
capabilities: allCaps,
|
||||
},
|
||||
{
|
||||
scheme: "https",
|
||||
host: "onlyheader.registry",
|
||||
path: "/v2",
|
||||
capabilities: allCaps,
|
||||
header: http.Header{"x-custom-1": {"justaheader"}},
|
||||
},
|
||||
{
|
||||
scheme: "https",
|
||||
host: "test-default.registry",
|
||||
path: "/v2",
|
||||
capabilities: allCaps,
|
||||
caCerts: []string{filepath.FromSlash("/etc/path/default")},
|
||||
header: http.Header{"x-custom-1": {"custom header"}},
|
||||
},
|
||||
}
|
||||
hosts, err := parseHostsFile("", []byte(testtoml))
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
||||
defer func() {
|
||||
if t.Failed() {
|
||||
t.Log("HostConfigs...\nActual:\n" + printHostConfig(hosts) + "Expected:\n" + printHostConfig(expected))
|
||||
}
|
||||
}()
|
||||
|
||||
if len(hosts) != len(expected) {
|
||||
t.Fatalf("Unexpected number of hosts %d, expected %d", len(hosts), len(expected))
|
||||
}
|
||||
|
||||
for i := range hosts {
|
||||
if !compareHostConfig(hosts[i], expected[i]) {
|
||||
t.Fatalf("Mismatch at host %d", i)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func TestLoadCertFiles(t *testing.T) {
|
||||
dir := t.TempDir()
|
||||
|
||||
type testCase struct {
|
||||
input hostConfig
|
||||
}
|
||||
cases := map[string]testCase{
|
||||
"crt only": {
|
||||
input: hostConfig{host: "testing.io", caCerts: []string{filepath.Join(dir, "testing.io", "ca.crt")}},
|
||||
},
|
||||
"crt and cert pair": {
|
||||
input: hostConfig{
|
||||
host: "testing.io",
|
||||
caCerts: []string{filepath.Join(dir, "testing.io", "ca.crt")},
|
||||
clientPairs: [][2]string{
|
||||
{
|
||||
filepath.Join(dir, "testing.io", "client.cert"),
|
||||
filepath.Join(dir, "testing.io", "client.key"),
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
"cert pair only": {
|
||||
input: hostConfig{
|
||||
host: "testing.io",
|
||||
clientPairs: [][2]string{
|
||||
{
|
||||
filepath.Join(dir, "testing.io", "client.cert"),
|
||||
filepath.Join(dir, "testing.io", "client.key"),
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
}
|
||||
|
||||
for name, tc := range cases {
|
||||
t.Run(name, func(t *testing.T) {
|
||||
|
||||
hostDir := filepath.Join(dir, tc.input.host)
|
||||
if err := os.MkdirAll(hostDir, 0700); err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
defer os.RemoveAll(hostDir)
|
||||
|
||||
for _, f := range tc.input.caCerts {
|
||||
if err := os.WriteFile(f, testKey, 0600); err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
}
|
||||
|
||||
for _, pair := range tc.input.clientPairs {
|
||||
if err := os.WriteFile(pair[0], testKey, 0600); err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
if err := os.WriteFile(pair[1], testKey, 0600); err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
}
|
||||
|
||||
configs, err := loadHostDir(context.Background(), hostDir)
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
if len(configs) != 1 {
|
||||
t.Fatalf("\nexpected:\n%+v\ngot:\n%+v", tc.input, configs)
|
||||
}
|
||||
|
||||
cfg := configs[0]
|
||||
cfg.host = tc.input.host
|
||||
|
||||
if !compareHostConfig(cfg, tc.input) {
|
||||
t.Errorf("\nexpected:\n%+v:\n\ngot:\n%+v", tc.input, cfg)
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
func TestHTTPFallback(t *testing.T) {
|
||||
for _, tc := range []struct {
|
||||
host string
|
||||
opts HostOptions
|
||||
expectedScheme string
|
||||
usesFallback bool
|
||||
}{
|
||||
{
|
||||
host: "localhost:8080",
|
||||
opts: HostOptions{
|
||||
DefaultScheme: "http",
|
||||
DefaultTLS: &tls.Config{
|
||||
InsecureSkipVerify: true,
|
||||
},
|
||||
},
|
||||
expectedScheme: "https",
|
||||
usesFallback: true,
|
||||
},
|
||||
{
|
||||
host: "localhost:8080",
|
||||
opts: HostOptions{
|
||||
DefaultScheme: "https",
|
||||
DefaultTLS: &tls.Config{
|
||||
InsecureSkipVerify: true,
|
||||
},
|
||||
},
|
||||
expectedScheme: "https",
|
||||
usesFallback: false,
|
||||
},
|
||||
{
|
||||
host: "localhost:8080",
|
||||
opts: HostOptions{},
|
||||
expectedScheme: "https",
|
||||
usesFallback: true,
|
||||
},
|
||||
{
|
||||
host: "localhost:80",
|
||||
opts: HostOptions{},
|
||||
expectedScheme: "http",
|
||||
usesFallback: false,
|
||||
},
|
||||
{
|
||||
host: "localhost:443",
|
||||
opts: HostOptions{},
|
||||
expectedScheme: "https",
|
||||
usesFallback: false,
|
||||
},
|
||||
{
|
||||
host: "localhost:80",
|
||||
opts: HostOptions{
|
||||
DefaultScheme: "http",
|
||||
DefaultTLS: &tls.Config{
|
||||
InsecureSkipVerify: true,
|
||||
},
|
||||
},
|
||||
expectedScheme: "http",
|
||||
usesFallback: false,
|
||||
},
|
||||
{
|
||||
host: "localhost",
|
||||
opts: HostOptions{
|
||||
DefaultScheme: "http",
|
||||
DefaultTLS: &tls.Config{
|
||||
InsecureSkipVerify: true,
|
||||
},
|
||||
},
|
||||
expectedScheme: "http",
|
||||
usesFallback: false,
|
||||
},
|
||||
{
|
||||
host: "localhost",
|
||||
opts: HostOptions{
|
||||
DefaultScheme: "https",
|
||||
DefaultTLS: &tls.Config{
|
||||
InsecureSkipVerify: true,
|
||||
},
|
||||
},
|
||||
expectedScheme: "https",
|
||||
usesFallback: false,
|
||||
},
|
||||
{
|
||||
host: "localhost",
|
||||
opts: HostOptions{},
|
||||
expectedScheme: "https",
|
||||
usesFallback: false,
|
||||
},
|
||||
{
|
||||
host: "localhost:5000",
|
||||
opts: HostOptions{},
|
||||
expectedScheme: "https",
|
||||
usesFallback: true,
|
||||
},
|
||||
{
|
||||
host: "example.com",
|
||||
opts: HostOptions{},
|
||||
expectedScheme: "https",
|
||||
usesFallback: false,
|
||||
},
|
||||
|
||||
{
|
||||
host: "example.com",
|
||||
opts: HostOptions{
|
||||
DefaultScheme: "http",
|
||||
DefaultTLS: &tls.Config{
|
||||
InsecureSkipVerify: true,
|
||||
},
|
||||
},
|
||||
expectedScheme: "http",
|
||||
usesFallback: false,
|
||||
},
|
||||
{
|
||||
host: "example.com:5000",
|
||||
opts: HostOptions{
|
||||
DefaultScheme: "http",
|
||||
DefaultTLS: &tls.Config{
|
||||
InsecureSkipVerify: true,
|
||||
},
|
||||
},
|
||||
expectedScheme: "https",
|
||||
usesFallback: true,
|
||||
},
|
||||
{
|
||||
host: "example.com:5000",
|
||||
opts: HostOptions{},
|
||||
expectedScheme: "https",
|
||||
usesFallback: false,
|
||||
},
|
||||
{
|
||||
host: "example2.com",
|
||||
opts: HostOptions{
|
||||
DefaultScheme: "http",
|
||||
},
|
||||
expectedScheme: "http",
|
||||
usesFallback: false,
|
||||
},
|
||||
{
|
||||
host: "127.0.0.254:5000",
|
||||
opts: HostOptions{},
|
||||
expectedScheme: "https",
|
||||
usesFallback: true,
|
||||
},
|
||||
{
|
||||
host: "127.0.0.254",
|
||||
opts: HostOptions{},
|
||||
expectedScheme: "https",
|
||||
usesFallback: false,
|
||||
},
|
||||
{
|
||||
host: "[::1]:5000",
|
||||
opts: HostOptions{},
|
||||
expectedScheme: "https",
|
||||
usesFallback: true,
|
||||
},
|
||||
{
|
||||
host: "::1",
|
||||
opts: HostOptions{},
|
||||
expectedScheme: "https",
|
||||
usesFallback: false,
|
||||
},
|
||||
} {
|
||||
testName := tc.host
|
||||
if tc.opts.DefaultScheme != "" {
|
||||
testName = testName + "-default-" + tc.opts.DefaultScheme
|
||||
}
|
||||
t.Run(testName, func(t *testing.T) {
|
||||
ctx := logtest.WithT(context.TODO(), t)
|
||||
hosts := ConfigureHosts(ctx, tc.opts)
|
||||
testHosts, err := hosts(tc.host)
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
if len(testHosts) != 1 {
|
||||
t.Fatalf("expected a single host for localhost config, got %d hosts", len(testHosts))
|
||||
}
|
||||
if testHosts[0].Scheme != tc.expectedScheme {
|
||||
t.Fatalf("expected %s scheme for localhost with tls config, got %q", tc.expectedScheme, testHosts[0].Scheme)
|
||||
}
|
||||
_, ok := testHosts[0].Client.Transport.(docker.HTTPFallback)
|
||||
if tc.usesFallback && !ok {
|
||||
t.Fatal("expected http fallback configured for defaulted localhost endpoint")
|
||||
} else if ok && !tc.usesFallback {
|
||||
t.Fatal("expected no http fallback configured for defaulted localhost endpoint")
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
func compareRegistryHost(j, k docker.RegistryHost) bool {
|
||||
if j.Scheme != k.Scheme {
|
||||
return false
|
||||
}
|
||||
if j.Host != k.Host {
|
||||
return false
|
||||
}
|
||||
if j.Path != k.Path {
|
||||
return false
|
||||
}
|
||||
if j.Capabilities != k.Capabilities {
|
||||
return false
|
||||
}
|
||||
// Not comparing TLS configs or authorizations
|
||||
return true
|
||||
}
|
||||
|
||||
func compareHostConfig(j, k hostConfig) bool {
|
||||
if j.scheme != k.scheme {
|
||||
return false
|
||||
}
|
||||
if j.host != k.host {
|
||||
return false
|
||||
}
|
||||
if j.path != k.path {
|
||||
return false
|
||||
}
|
||||
if j.capabilities != k.capabilities {
|
||||
return false
|
||||
}
|
||||
|
||||
if len(j.caCerts) != len(k.caCerts) {
|
||||
return false
|
||||
}
|
||||
for i := range j.caCerts {
|
||||
if j.caCerts[i] != k.caCerts[i] {
|
||||
return false
|
||||
}
|
||||
}
|
||||
if len(j.clientPairs) != len(k.clientPairs) {
|
||||
return false
|
||||
}
|
||||
for i := range j.clientPairs {
|
||||
if j.clientPairs[i][0] != k.clientPairs[i][0] {
|
||||
return false
|
||||
}
|
||||
if j.clientPairs[i][1] != k.clientPairs[i][1] {
|
||||
return false
|
||||
}
|
||||
}
|
||||
if j.skipVerify != nil && k.skipVerify != nil {
|
||||
if *j.skipVerify != *k.skipVerify {
|
||||
return false
|
||||
}
|
||||
} else if j.skipVerify != nil || k.skipVerify != nil {
|
||||
return false
|
||||
}
|
||||
|
||||
if len(j.header) != len(k.header) {
|
||||
return false
|
||||
}
|
||||
for key := range j.header {
|
||||
if len(j.header[key]) != len(k.header[key]) {
|
||||
return false
|
||||
}
|
||||
for i := range j.header[key] {
|
||||
if j.header[key][i] != k.header[key][i] {
|
||||
return false
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return true
|
||||
}
|
||||
|
||||
func printHostConfig(hc []hostConfig) string {
|
||||
b := bytes.NewBuffer(nil)
|
||||
for i := range hc {
|
||||
fmt.Fprintf(b, "\t[%d]\tscheme: %q\n", i, hc[i].scheme)
|
||||
fmt.Fprintf(b, "\t\thost: %q\n", hc[i].host)
|
||||
fmt.Fprintf(b, "\t\tpath: %q\n", hc[i].path)
|
||||
fmt.Fprintf(b, "\t\tcaps: %03b\n", hc[i].capabilities)
|
||||
fmt.Fprintf(b, "\t\tca: %#v\n", hc[i].caCerts)
|
||||
fmt.Fprintf(b, "\t\tclients: %#v\n", hc[i].clientPairs)
|
||||
if hc[i].skipVerify == nil {
|
||||
fmt.Fprintf(b, "\t\tskip-verify: %v\n", hc[i].skipVerify)
|
||||
} else {
|
||||
fmt.Fprintf(b, "\t\tskip-verify: %t\n", *hc[i].skipVerify)
|
||||
}
|
||||
fmt.Fprintf(b, "\t\theader: %#v\n", hc[i].header)
|
||||
}
|
||||
return b.String()
|
||||
}
|
||||
|
||||
var (
|
||||
testKey = []byte(`-----BEGIN PRIVATE KEY-----
|
||||
MIIEvwIBADANBgkqhkiG9w0BAQEFAASCBKkwggSlAgEAAoIBAQDa+zvPgFXwra4S
|
||||
0DzEWRgZHxVTDG1sJsnN/jOaHCNpRyABGVW5kdei9WFWv3dpiELI+guQMjdUL++w
|
||||
M68bs6cXKW+1nW6u5uWuGwklOwkoKoeHkkn/vHef7ybk+5qdk6AYY0DKQsrBBOvj
|
||||
f0WAnG+1xi8VIOEBmce0/47MexOiuILVkjokgdmDCOc8ShkT6/EJTCsI1wDew/4G
|
||||
9IiRzw2xSM0ZATAtEC3HEBRLJGWZQtuKlLCuzJ+erOWUcg2cjnSgR3PmaAXE//5g
|
||||
SoeqEbtTo1satf9AR4VvreIAI8m0eyo8ABMLTkZovEFcUUHetL63hdqItjCeRfrQ
|
||||
zK4LMRFbAgMBAAECggEBAJtP6UHo0gtcA8SQMSlJz4+xvhwjClDUyfjyPIMnRe5b
|
||||
ZdWhtG1jhT+tLhaqwfT1kfidcCobk6aAQU4FukK5jt8cooB7Yo9mcKylvDzNvFbi
|
||||
ozGCjj113JpwsnNiCG2O0NO7Qa6y5L810GCQWik3yvtvzuD7atsJyN0VDKD3Ahw7
|
||||
1X8z76grZFlhVMCTAA3vAJ2y2p3sd+TGC/PIhnsvChwxEorGCnMj93mBaUI7zZRY
|
||||
EZhlk4ZvC9sUvlVUuYC+wAHjasgN9s3AzsOBSx+Xt3NaXQHzhL0mVo/vu/pjjFBs
|
||||
WBLR1PBoIfveTJPOp+Hrr4cuCK0NuX9sWlWPYLl5A2ECgYEA5fq3n4PhbJ2BuTS5
|
||||
AVgOmjRpk1eogb6aSY+cx7Mr++ADF9EYXc5tgKoUsDeeiiyK2lv6IKavoTWT1kdd
|
||||
shiclyEzp2CxG5GtbC/g2XHiBLepgo1fjfev3btCmIeGVBjglOx4F3gEsRygrAID
|
||||
zcz94m2I+uqLT8hvWnccIqScglkCgYEA88H2ji4Nvx6TmqCLcER0vNDVoxxDfgGb
|
||||
iohvenD2jmmdTnezTddsgECAI8L0BPNS/0vBCduTjs5BqhKbIfQvuK5CANMUcxuQ
|
||||
twWH8kPvTYJVgsmWP6sSXSz3PohWC5EA9xACExGtyN6d7sLUCV0SBhjlcgMvGuDM
|
||||
lP6NjyyWctMCgYBKdfGr+QQsqZaNw48+6ybXMK8aIKCTWYYU2SW21sEf7PizZmTQ
|
||||
Qnzb0rWeFHQFYsSWTH9gwPdOZ8107GheuG9C02IpCDpvpawTwjC31pKKWnjMpz9P
|
||||
9OkBDpdSUVbhtahJL4L2fkpumck/x+s5X+y3uiVGsFfovgmnrbbzVH7ECQKBgQCC
|
||||
MYs7DaYR+obkA/P2FtozL2esIyB5YOpu58iDIWrPTeHTU2PVo8Y0Cj9m2m3zZvNh
|
||||
oFiOp1T85XV1HVL2o7IJdimSvyshAAwfdTjTUS2zvHVn0bwKbZj1Y1r7b15l9yEI
|
||||
1OgGv16O9zhrmmweRDOoRgvnBYRXWtJqkjuRyULiOQKBgQC/lSYigV32Eb8Eg1pv
|
||||
7OcPWv4qV4880lRE0MXuQ4VFa4+pqvdziYFYQD4jDYJ4IX9l//bsobL0j7z0P0Gk
|
||||
wDFti9bRwRoO1ntqoA8n2pDLlLRGl0dyjB6fHzp27oqtyf1HRlHiow7Gqx5b5JOk
|
||||
tycYKwA3DuaSyqPe6MthLneq8w==
|
||||
-----END PRIVATE KEY-----
|
||||
`)
|
||||
)
|
||||
Reference in New Issue
Block a user