Skip to content

Commit

Permalink
ostree: extract httpClientForRef from resolveRef
Browse files Browse the repository at this point in the history
This commit extracts a `httpClientForRef` helper from the `resolveRef()`
function. This helps with readability and testability. It also adds
a test for the proxy setting, CA and MTLS settings.
  • Loading branch information
mvo5 committed Nov 19, 2024
1 parent 8a2c23c commit 34559bc
Show file tree
Hide file tree
Showing 2 changed files with 71 additions and 18 deletions.
41 changes: 24 additions & 17 deletions pkg/ostree/ostree.go
Original file line number Diff line number Diff line change
Expand Up @@ -151,18 +151,9 @@ func verifyChecksum(commit string) bool {
return len(commit) > 0 && ostreeCommitRE.MatchString(commit)
}

// resolveRef resolves the URL path specified by the location and ref
// (location+"refs/heads/"+ref) and returns the commit ID for the named ref. If
// there is an error, it will be of type ResolveRefError.
func resolveRef(ss SourceSpec) (string, error) {
u, err := url.Parse(ss.URL)
if err != nil {
return "", NewResolveRefError("error parsing ostree repository location: %v", err)
}
u.Path = path.Join(u.Path, "refs", "heads", ss.Ref)

func httpClientForRef(scheme string, ss SourceSpec) (*http.Client, error) {
transport := http.DefaultTransport.(*http.Transport).Clone()
if u.Scheme == "https" {
if scheme == "https" {
tlsConf := &tls.Config{
MinVersion: tls.VersionTLS12,
}
Expand All @@ -171,18 +162,18 @@ func resolveRef(ss SourceSpec) (string, error) {
if ss.MTLS != nil && ss.MTLS.CA != "" {
caCertPEM, err := os.ReadFile(ss.MTLS.CA)
if err != nil {
return "", NewResolveRefError("error adding ca certificate when resolving ref: %s", err)
return nil, NewResolveRefError("error adding ca certificate when resolving ref: %s", err)
}
tlsConf.RootCAs = x509.NewCertPool()
if ok := tlsConf.RootCAs.AppendCertsFromPEM(caCertPEM); !ok {
return "", NewResolveRefError("error adding ca certificate when resolving ref")
return nil, NewResolveRefError("error adding ca certificate when resolving ref")
}
}

if ss.MTLS != nil && ss.MTLS.ClientCert != "" && ss.MTLS.ClientKey != "" {
cert, err := tls.LoadX509KeyPair(ss.MTLS.ClientCert, ss.MTLS.ClientKey)
if err != nil {
return "", NewResolveRefError("error adding client certificate when resolving ref: %s", err)
return nil, NewResolveRefError("error adding client certificate when resolving ref: %s", err)
}
tlsConf.Certificates = []tls.Certificate{cert}
}
Expand All @@ -193,22 +184,38 @@ func resolveRef(ss SourceSpec) (string, error) {
if ss.Proxy != "" {
host, port, err := net.SplitHostPort(ss.Proxy)
if err != nil {
return "", NewResolveRefError("error parsing MTLS proxy URL '%s': %v", ss.URL, err)
return nil, NewResolveRefError("error parsing MTLS proxy URL '%s': %v", ss.URL, err)
}

proxyURL, err := url.Parse("http://" + host + ":" + port)
if err != nil {
return "", NewResolveRefError("error parsing MTLS proxy URL '%s': %v", ss.URL, err)
return nil, NewResolveRefError("error parsing MTLS proxy URL '%s': %v", ss.URL, err)
}

transport.Proxy = func(request *http.Request) (*url.URL, error) {
return proxyURL, nil
}
}

client := &http.Client{
return &http.Client{
Transport: transport,
Timeout: 300 * time.Second,
}, nil
}

// resolveRef resolves the URL path specified by the location and ref
// (location+"refs/heads/"+ref) and returns the commit ID for the named ref. If
// there is an error, it will be of type ResolveRefError.
func resolveRef(ss SourceSpec) (string, error) {
u, err := url.Parse(ss.URL)
if err != nil {
return "", NewResolveRefError("error parsing ostree repository location: %v", err)
}
u.Path = path.Join(u.Path, "refs", "heads", ss.Ref)

client, err := httpClientForRef(u.Scheme, ss)
if err != nil {
return "", err
}

req, err := http.NewRequest(http.MethodGet, u.String(), nil)
Expand Down
48 changes: 47 additions & 1 deletion pkg/ostree/ostree_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -85,7 +85,7 @@ func TestOstreeResolveRef(t *testing.T) {
&MTLS{mTLSSrv.CAPath, mTLSSrv.ClientCrtPath, mTLSSrv.ClientKeyPath},
"",
})
assert.NoError(t, err)
require.NoError(t, err)
assert.Equal(t, expOut, out)
}

Expand Down Expand Up @@ -226,3 +226,49 @@ func TestValidate(t *testing.T) {
}

}

func TestClientForRefProxy(t *testing.T) {
ss := SourceSpec{
Proxy: "foo:1234",
}
client, err := httpClientForRef("https", ss)
assert.NoError(t, err)

proxy, err := client.Transport.(*http.Transport).Proxy(&http.Request{})
assert.NoError(t, err)
assert.Equal(t, "foo", proxy.Hostname())
assert.Equal(t, "1234", proxy.Port())
}

func TestClientForRefClientCertKey(t *testing.T) {
ss := SourceSpec{
MTLS: &MTLS{
ClientCert: "test_mtls_server/client.crt",
ClientKey: "test_mtls_server/client.key",
},
}
client, err := httpClientForRef("https", ss)
assert.NoError(t, err)

tlsConf := client.Transport.(*http.Transport).TLSClientConfig
assert.NoError(t, err)
// XXX: improve to check that it's really client.{crt,key}
assert.Equal(t, 1, len(tlsConf.Certificates))
assert.Nil(t, tlsConf.RootCAs)
}

func TestClientForRefCA(t *testing.T) {
ss := SourceSpec{
MTLS: &MTLS{
CA: "test_mtls_server/ca.crt",
},
}
client, err := httpClientForRef("https", ss)
assert.NoError(t, err)

tlsConf := client.Transport.(*http.Transport).TLSClientConfig
assert.NoError(t, err)
// XXX: improve to check that ca.crt above is actually what is
// loaded
assert.Equal(t, 1, len(tlsConf.RootCAs.Subjects()))

Check failure on line 273 in pkg/ostree/ostree_test.go

View workflow job for this annotation

GitHub Actions / ⌨ Lint

SA1019: tlsConf.RootCAs.Subjects has been deprecated since Go 1.18: if s was returned by [SystemCertPool], Subjects will not include the system roots. (staticcheck)
}

0 comments on commit 34559bc

Please sign in to comment.