2022-03-29 23:58:41 +00:00
|
|
|
// Copyright 2021-2022 the Pinniped contributors. All Rights Reserved.
|
2021-10-20 11:59:24 +00:00
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
|
|
|
package integration
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"crypto/tls"
|
|
|
|
"encoding/base64"
|
|
|
|
"fmt"
|
|
|
|
"net/http"
|
|
|
|
"strings"
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
|
|
|
|
"go.pinniped.dev/internal/crypto/ptls"
|
|
|
|
"go.pinniped.dev/internal/testutil/tlsserver"
|
|
|
|
"go.pinniped.dev/test/testlib"
|
|
|
|
)
|
|
|
|
|
|
|
|
// TLS checks safe to run in parallel with serial tests, see main_test.go.
|
|
|
|
func TestSecureTLSPinnipedCLIToKAS_Parallel(t *testing.T) {
|
|
|
|
_ = testlib.IntegrationEnv(t)
|
|
|
|
|
|
|
|
server := tlsserver.TLSTestServer(t, http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
2022-03-31 19:31:20 +00:00
|
|
|
// pinniped CLI uses ptls.Secure when talking to KAS
|
|
|
|
// in FIPs mode the distinction doesn't matter much because
|
|
|
|
// each of the configs is a wrapper for the same base FIPs config
|
|
|
|
tlsserver.AssertTLS(t, r, ptls.Secure)
|
2021-10-20 11:59:24 +00:00
|
|
|
w.Header().Set("content-type", "application/json")
|
|
|
|
fmt.Fprint(w, `{"kind":"TokenCredentialRequest","apiVersion":"login.concierge.pinniped.dev/v1alpha1",`+
|
|
|
|
`"status":{"credential":{"token":"some-fancy-token"}}}`)
|
|
|
|
}), tlsserver.RecordTLSHello)
|
|
|
|
|
|
|
|
ca := tlsserver.TLSTestServerCA(server)
|
|
|
|
|
|
|
|
pinnipedExe := testlib.PinnipedCLIPath(t)
|
|
|
|
|
|
|
|
stdout, stderr := runPinnipedCLI(t, nil, pinnipedExe, "login", "static",
|
|
|
|
"--token", "does-not-matter",
|
|
|
|
"--concierge-authenticator-type", "webhook",
|
|
|
|
"--concierge-authenticator-name", "does-not-matter",
|
|
|
|
"--concierge-ca-bundle-data", base64.StdEncoding.EncodeToString(ca),
|
|
|
|
"--concierge-endpoint", server.URL,
|
|
|
|
"--enable-concierge",
|
|
|
|
"--credential-cache", "",
|
|
|
|
)
|
|
|
|
|
|
|
|
require.Empty(t, stderr)
|
|
|
|
require.Equal(t, `{"kind":"ExecCredential","apiVersion":"client.authentication.k8s.io/v1beta1",`+
|
|
|
|
`"spec":{"interactive":false},"status":{"expirationTimestamp":null,"token":"some-fancy-token"}}
|
|
|
|
`, stdout)
|
|
|
|
}
|
|
|
|
|
|
|
|
// TLS checks safe to run in parallel with serial tests, see main_test.go.
|
|
|
|
func TestSecureTLSPinnipedCLIToSupervisor_Parallel(t *testing.T) {
|
|
|
|
_ = testlib.IntegrationEnv(t)
|
|
|
|
|
|
|
|
server := tlsserver.TLSTestServer(t, http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
2022-03-31 19:31:20 +00:00
|
|
|
// pinniped CLI uses ptls.Default when talking to supervisor
|
|
|
|
// in FIPs mode the distinction doesn't matter much because
|
|
|
|
// each of the configs is a wrapper for the same base FIPs config
|
|
|
|
tlsserver.AssertTLS(t, r, ptls.Default)
|
2021-10-20 11:59:24 +00:00
|
|
|
w.Header().Set("content-type", "application/json")
|
|
|
|
fmt.Fprint(w, `{"issuer":"https://not-a-good-issuer"}`)
|
|
|
|
}), tlsserver.RecordTLSHello)
|
|
|
|
|
|
|
|
ca := tlsserver.TLSTestServerCA(server)
|
|
|
|
|
|
|
|
pinnipedExe := testlib.PinnipedCLIPath(t)
|
|
|
|
|
|
|
|
stdout, stderr := runPinnipedCLI(&fakeT{T: t}, nil, pinnipedExe, "login", "oidc",
|
|
|
|
"--ca-bundle-data", base64.StdEncoding.EncodeToString(ca),
|
|
|
|
"--issuer", server.URL,
|
|
|
|
"--credential-cache", "",
|
|
|
|
"--upstream-identity-provider-flow", "cli_password",
|
|
|
|
"--upstream-identity-provider-name", "does-not-matter",
|
|
|
|
"--upstream-identity-provider-type", "oidc",
|
|
|
|
)
|
|
|
|
|
|
|
|
require.Equal(t, `Error: could not complete Pinniped login: could not perform OIDC discovery for "`+
|
|
|
|
server.URL+`": oidc: issuer did not match the issuer returned by provider, expected "`+
|
|
|
|
server.URL+`" got "https://not-a-good-issuer"
|
|
|
|
`, stderr)
|
|
|
|
require.Empty(t, stdout)
|
|
|
|
}
|
|
|
|
|
|
|
|
// TLS checks safe to run in parallel with serial tests, see main_test.go.
|
|
|
|
func TestSecureTLSConciergeAggregatedAPI_Parallel(t *testing.T) {
|
|
|
|
env := testlib.IntegrationEnv(t)
|
|
|
|
|
|
|
|
cancelCtx, cancel := context.WithCancel(context.Background())
|
|
|
|
t.Cleanup(cancel)
|
|
|
|
|
|
|
|
startKubectlPortForward(cancelCtx, t, "10446", "443", env.ConciergeAppName+"-api", env.ConciergeNamespace)
|
|
|
|
|
2022-03-29 23:58:41 +00:00
|
|
|
stdout, stderr := testlib.RunNmapSSLEnum(t, "127.0.0.1", 10446)
|
2021-10-20 11:59:24 +00:00
|
|
|
|
|
|
|
require.Empty(t, stderr)
|
2022-03-29 23:58:41 +00:00
|
|
|
require.Contains(t, stdout, testlib.GetExpectedCiphers(ptls.Secure(nil)), "stdout:\n%s", stdout)
|
2021-10-20 11:59:24 +00:00
|
|
|
}
|
|
|
|
|
2021-11-18 08:35:30 +00:00
|
|
|
func TestSecureTLSSupervisor(t *testing.T) { // does not run in parallel because of the createSupervisorDefaultTLSCertificateSecretIfNeeded call
|
2021-10-20 11:59:24 +00:00
|
|
|
env := testlib.IntegrationEnv(t)
|
|
|
|
|
|
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
|
|
t.Cleanup(cancel)
|
|
|
|
|
2021-11-18 08:35:30 +00:00
|
|
|
startKubectlPortForward(ctx, t, "10447", "443", env.SupervisorAppName+"-nodeport", env.SupervisorNamespace)
|
2021-10-20 11:59:24 +00:00
|
|
|
|
2022-03-29 23:58:41 +00:00
|
|
|
stdout, stderr := testlib.RunNmapSSLEnum(t, "127.0.0.1", 10447)
|
2021-10-20 11:59:24 +00:00
|
|
|
|
|
|
|
// supervisor's cert is ECDSA
|
2022-03-29 23:58:41 +00:00
|
|
|
defaultECDSAOnly := ptls.Default(nil)
|
|
|
|
ciphers := make([]uint16, 0, len(defaultECDSAOnly.CipherSuites)/2)
|
|
|
|
for _, id := range defaultECDSAOnly.CipherSuites {
|
|
|
|
id := id
|
|
|
|
if !strings.Contains(tls.CipherSuiteName(id), "_ECDSA_") {
|
|
|
|
continue
|
2021-10-20 11:59:24 +00:00
|
|
|
}
|
2022-03-29 23:58:41 +00:00
|
|
|
ciphers = append(ciphers, id)
|
2021-10-20 11:59:24 +00:00
|
|
|
}
|
2022-03-29 23:58:41 +00:00
|
|
|
defaultECDSAOnly.CipherSuites = ciphers
|
2021-10-20 11:59:24 +00:00
|
|
|
|
|
|
|
require.Empty(t, stderr)
|
2022-03-29 23:58:41 +00:00
|
|
|
require.Contains(t, stdout, testlib.GetExpectedCiphers(defaultECDSAOnly), "stdout:\n%s", stdout)
|
2021-10-20 11:59:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type fakeT struct {
|
|
|
|
*testing.T
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t *fakeT) FailNow() {
|
|
|
|
t.Errorf("fakeT ignored FailNow")
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t *fakeT) Errorf(format string, args ...interface{}) {
|
|
|
|
t.Cleanup(func() {
|
|
|
|
if !t.Failed() {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
t.Logf("reporting previously ignored errors since main test failed:\n"+format, args...)
|
|
|
|
})
|
|
|
|
}
|