mirror of
https://github.com/slackhq/nebula.git
synced 2025-01-28 18:49:37 +00:00
695 lines
24 KiB
Go
695 lines
24 KiB
Go
package cert
|
|
|
|
import (
|
|
"crypto/ecdh"
|
|
"crypto/ecdsa"
|
|
"crypto/elliptic"
|
|
"crypto/rand"
|
|
"fmt"
|
|
"io"
|
|
"net/netip"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/slackhq/nebula/test"
|
|
"github.com/stretchr/testify/assert"
|
|
"golang.org/x/crypto/curve25519"
|
|
"golang.org/x/crypto/ed25519"
|
|
)
|
|
|
|
func TestMarshalingNebulaCertificate(t *testing.T) {
|
|
before := time.Now().Add(time.Second * -60).Round(time.Second)
|
|
after := time.Now().Add(time.Second * 60).Round(time.Second)
|
|
pubKey := []byte("1234567890abcedfghij1234567890ab")
|
|
|
|
nc := certificateV1{
|
|
details: detailsV1{
|
|
Name: "testing",
|
|
Ips: []netip.Prefix{
|
|
mustParsePrefixUnmapped("10.1.1.1/24"),
|
|
mustParsePrefixUnmapped("10.1.1.2/16"),
|
|
},
|
|
Subnets: []netip.Prefix{
|
|
mustParsePrefixUnmapped("9.1.1.2/24"),
|
|
mustParsePrefixUnmapped("9.1.1.3/16"),
|
|
},
|
|
Groups: []string{"test-group1", "test-group2", "test-group3"},
|
|
NotBefore: before,
|
|
NotAfter: after,
|
|
PublicKey: pubKey,
|
|
IsCA: false,
|
|
Issuer: "1234567890abcedfghij1234567890ab",
|
|
},
|
|
signature: []byte("1234567890abcedfghij1234567890ab"),
|
|
}
|
|
|
|
b, err := nc.Marshal()
|
|
assert.Nil(t, err)
|
|
//t.Log("Cert size:", len(b))
|
|
|
|
nc2, err := unmarshalCertificateV1(b, true)
|
|
assert.Nil(t, err)
|
|
|
|
assert.Equal(t, nc.signature, nc2.Signature())
|
|
assert.Equal(t, nc.details.Name, nc2.Name())
|
|
assert.Equal(t, nc.details.NotBefore, nc2.NotBefore())
|
|
assert.Equal(t, nc.details.NotAfter, nc2.NotAfter())
|
|
assert.Equal(t, nc.details.PublicKey, nc2.PublicKey())
|
|
assert.Equal(t, nc.details.IsCA, nc2.IsCA())
|
|
|
|
assert.Equal(t, nc.details.Ips, nc2.Networks())
|
|
assert.Equal(t, nc.details.Subnets, nc2.UnsafeNetworks())
|
|
|
|
assert.Equal(t, nc.details.Groups, nc2.Groups())
|
|
}
|
|
|
|
//func TestNebulaCertificate_Sign(t *testing.T) {
|
|
// before := time.Now().Add(time.Second * -60).Round(time.Second)
|
|
// after := time.Now().Add(time.Second * 60).Round(time.Second)
|
|
// pubKey := []byte("1234567890abcedfghij1234567890ab")
|
|
//
|
|
// nc := certificateV1{
|
|
// details: detailsV1{
|
|
// Name: "testing",
|
|
// Ips: []netip.Prefix{
|
|
// mustParsePrefixUnmapped("10.1.1.1/24"),
|
|
// mustParsePrefixUnmapped("10.1.1.2/16"),
|
|
// //TODO: netip cant do it
|
|
// //{IP: net.ParseIP("10.1.1.3"), Mask: net.IPMask(net.ParseIP("255.0.255.0"))},
|
|
// },
|
|
// Subnets: []netip.Prefix{
|
|
// //TODO: netip cant do it
|
|
// //{IP: net.ParseIP("9.1.1.1"), Mask: net.IPMask(net.ParseIP("255.0.255.0"))},
|
|
// mustParsePrefixUnmapped("9.1.1.2/24"),
|
|
// mustParsePrefixUnmapped("9.1.1.3/24"),
|
|
// },
|
|
// Groups: []string{"test-group1", "test-group2", "test-group3"},
|
|
// NotBefore: before,
|
|
// NotAfter: after,
|
|
// PublicKey: pubKey,
|
|
// IsCA: false,
|
|
// Issuer: "1234567890abcedfghij1234567890ab",
|
|
// },
|
|
// }
|
|
//
|
|
// pub, priv, err := ed25519.GenerateKey(rand.Reader)
|
|
// assert.Nil(t, err)
|
|
// assert.False(t, nc.CheckSignature(pub))
|
|
// assert.Nil(t, nc.Sign(Curve_CURVE25519, priv))
|
|
// assert.True(t, nc.CheckSignature(pub))
|
|
//
|
|
// _, err = nc.Marshal()
|
|
// assert.Nil(t, err)
|
|
// //t.Log("Cert size:", len(b))
|
|
//}
|
|
|
|
//func TestNebulaCertificate_SignP256(t *testing.T) {
|
|
// before := time.Now().Add(time.Second * -60).Round(time.Second)
|
|
// after := time.Now().Add(time.Second * 60).Round(time.Second)
|
|
// pubKey := []byte("01234567890abcedfghij1234567890ab1234567890abcedfghij1234567890ab")
|
|
//
|
|
// nc := certificateV1{
|
|
// details: detailsV1{
|
|
// Name: "testing",
|
|
// Ips: []netip.Prefix{
|
|
// mustParsePrefixUnmapped("10.1.1.1/24"),
|
|
// mustParsePrefixUnmapped("10.1.1.2/16"),
|
|
// //TODO: netip no can do
|
|
// //{IP: net.ParseIP("10.1.1.3"), Mask: net.IPMask(net.ParseIP("255.0.255.0"))},
|
|
// },
|
|
// Subnets: []netip.Prefix{
|
|
// //TODO: netip bad
|
|
// //{IP: net.ParseIP("9.1.1.1"), Mask: net.IPMask(net.ParseIP("255.0.255.0"))},
|
|
// mustParsePrefixUnmapped("9.1.1.2/24"),
|
|
// mustParsePrefixUnmapped("9.1.1.3/16"),
|
|
// },
|
|
// Groups: []string{"test-group1", "test-group2", "test-group3"},
|
|
// NotBefore: before,
|
|
// NotAfter: after,
|
|
// PublicKey: pubKey,
|
|
// IsCA: false,
|
|
// Curve: Curve_P256,
|
|
// Issuer: "1234567890abcedfghij1234567890ab",
|
|
// },
|
|
// }
|
|
//
|
|
// priv, err := ecdsa.GenerateKey(elliptic.P256(), rand.Reader)
|
|
// pub := elliptic.Marshal(elliptic.P256(), priv.PublicKey.X, priv.PublicKey.Y)
|
|
// rawPriv := priv.D.FillBytes(make([]byte, 32))
|
|
//
|
|
// assert.Nil(t, err)
|
|
// assert.False(t, nc.CheckSignature(pub))
|
|
// assert.Nil(t, nc.Sign(Curve_P256, rawPriv))
|
|
// assert.True(t, nc.CheckSignature(pub))
|
|
//
|
|
// _, err = nc.Marshal()
|
|
// assert.Nil(t, err)
|
|
// //t.Log("Cert size:", len(b))
|
|
//}
|
|
|
|
func TestNebulaCertificate_Expired(t *testing.T) {
|
|
nc := certificateV1{
|
|
details: detailsV1{
|
|
NotBefore: time.Now().Add(time.Second * -60).Round(time.Second),
|
|
NotAfter: time.Now().Add(time.Second * 60).Round(time.Second),
|
|
},
|
|
}
|
|
|
|
assert.True(t, nc.Expired(time.Now().Add(time.Hour)))
|
|
assert.True(t, nc.Expired(time.Now().Add(-time.Hour)))
|
|
assert.False(t, nc.Expired(time.Now()))
|
|
}
|
|
|
|
func TestNebulaCertificate_MarshalJSON(t *testing.T) {
|
|
time.Local = time.UTC
|
|
pubKey := []byte("1234567890abcedfghij1234567890ab")
|
|
|
|
nc := certificateV1{
|
|
details: detailsV1{
|
|
Name: "testing",
|
|
Ips: []netip.Prefix{
|
|
mustParsePrefixUnmapped("10.1.1.1/24"),
|
|
mustParsePrefixUnmapped("10.1.1.2/16"),
|
|
},
|
|
Subnets: []netip.Prefix{
|
|
mustParsePrefixUnmapped("9.1.1.2/24"),
|
|
mustParsePrefixUnmapped("9.1.1.3/16"),
|
|
},
|
|
Groups: []string{"test-group1", "test-group2", "test-group3"},
|
|
NotBefore: time.Date(1, 0, 0, 1, 0, 0, 0, time.UTC),
|
|
NotAfter: time.Date(1, 0, 0, 2, 0, 0, 0, time.UTC),
|
|
PublicKey: pubKey,
|
|
IsCA: false,
|
|
Issuer: "1234567890abcedfghij1234567890ab",
|
|
},
|
|
signature: []byte("1234567890abcedfghij1234567890ab"),
|
|
}
|
|
|
|
b, err := nc.MarshalJSON()
|
|
assert.Nil(t, err)
|
|
assert.Equal(
|
|
t,
|
|
"{\"details\":{\"curve\":\"CURVE25519\",\"groups\":[\"test-group1\",\"test-group2\",\"test-group3\"],\"ips\":[\"10.1.1.1/24\",\"10.1.1.2/16\"],\"isCa\":false,\"issuer\":\"1234567890abcedfghij1234567890ab\",\"name\":\"testing\",\"notAfter\":\"0000-11-30T02:00:00Z\",\"notBefore\":\"0000-11-30T01:00:00Z\",\"publicKey\":\"313233343536373839306162636564666768696a313233343536373839306162\",\"subnets\":[\"9.1.1.2/24\",\"9.1.1.3/16\"]},\"fingerprint\":\"3944c53d4267a229295b56cb2d27d459164c010ac97d655063ba421e0670f4ba\",\"signature\":\"313233343536373839306162636564666768696a313233343536373839306162\"}",
|
|
string(b),
|
|
)
|
|
}
|
|
|
|
func TestNebulaCertificate_Verify(t *testing.T) {
|
|
ca, _, caKey, err := newTestCaCert(time.Now(), time.Now().Add(10*time.Minute), nil, nil, nil)
|
|
assert.Nil(t, err)
|
|
|
|
c, _, _, err := newTestCert(ca, caKey, time.Now(), time.Now().Add(5*time.Minute), nil, nil, nil)
|
|
assert.Nil(t, err)
|
|
|
|
caPool := NewCAPool()
|
|
assert.NoError(t, caPool.AddCA(ca))
|
|
|
|
f, err := c.Fingerprint()
|
|
assert.Nil(t, err)
|
|
caPool.BlocklistFingerprint(f)
|
|
|
|
_, err = caPool.VerifyCertificate(time.Now(), c)
|
|
assert.EqualError(t, err, "certificate is in the block list")
|
|
|
|
caPool.ResetCertBlocklist()
|
|
_, err = caPool.VerifyCertificate(time.Now(), c)
|
|
assert.Nil(t, err)
|
|
|
|
_, err = caPool.VerifyCertificate(time.Now().Add(time.Hour*1000), c)
|
|
assert.EqualError(t, err, "root certificate is expired")
|
|
|
|
c, _, _, err = newTestCert(ca, caKey, time.Time{}, time.Time{}, nil, nil, nil)
|
|
assert.EqualError(t, err, "certificate is valid before the signing certificate")
|
|
|
|
// Test group assertion
|
|
ca, _, caKey, err = newTestCaCert(time.Now(), time.Now().Add(10*time.Minute), nil, nil, []string{"test1", "test2"})
|
|
assert.Nil(t, err)
|
|
|
|
caPem, err := ca.MarshalPEM()
|
|
assert.Nil(t, err)
|
|
|
|
caPool = NewCAPool()
|
|
b, err := caPool.AddCAFromPEM(caPem)
|
|
assert.NoError(t, err)
|
|
assert.Empty(t, b)
|
|
|
|
c, _, _, err = newTestCert(ca, caKey, time.Now(), time.Now().Add(5*time.Minute), nil, nil, []string{"test1", "bad"})
|
|
assert.EqualError(t, err, "certificate contained a group not present on the signing ca: bad")
|
|
|
|
c, _, _, err = newTestCert(ca, caKey, time.Now(), time.Now().Add(5*time.Minute), nil, nil, []string{"test1"})
|
|
assert.Nil(t, err)
|
|
_, err = caPool.VerifyCertificate(time.Now(), c)
|
|
assert.Nil(t, err)
|
|
}
|
|
|
|
func TestNebulaCertificate_VerifyP256(t *testing.T) {
|
|
ca, _, caKey, err := newTestCaCertP256(time.Now(), time.Now().Add(10*time.Minute), nil, nil, nil)
|
|
assert.Nil(t, err)
|
|
|
|
c, _, _, err := newTestCert(ca, caKey, time.Now(), time.Now().Add(5*time.Minute), nil, nil, nil)
|
|
assert.Nil(t, err)
|
|
|
|
caPool := NewCAPool()
|
|
assert.NoError(t, caPool.AddCA(ca))
|
|
|
|
f, err := c.Fingerprint()
|
|
assert.Nil(t, err)
|
|
caPool.BlocklistFingerprint(f)
|
|
|
|
_, err = caPool.VerifyCertificate(time.Now(), c)
|
|
assert.EqualError(t, err, "certificate is in the block list")
|
|
|
|
caPool.ResetCertBlocklist()
|
|
_, err = caPool.VerifyCertificate(time.Now(), c)
|
|
assert.Nil(t, err)
|
|
|
|
_, err = caPool.VerifyCertificate(time.Now().Add(time.Hour*1000), c)
|
|
assert.EqualError(t, err, "root certificate is expired")
|
|
|
|
c, _, _, err = newTestCert(ca, caKey, time.Time{}, time.Time{}, nil, nil, nil)
|
|
assert.EqualError(t, err, "certificate is valid before the signing certificate")
|
|
|
|
// Test group assertion
|
|
ca, _, caKey, err = newTestCaCertP256(time.Now(), time.Now().Add(10*time.Minute), nil, nil, []string{"test1", "test2"})
|
|
assert.Nil(t, err)
|
|
|
|
caPem, err := ca.MarshalPEM()
|
|
assert.Nil(t, err)
|
|
|
|
caPool = NewCAPool()
|
|
b, err := caPool.AddCAFromPEM(caPem)
|
|
assert.NoError(t, err)
|
|
assert.Empty(t, b)
|
|
|
|
c, _, _, err = newTestCert(ca, caKey, time.Now(), time.Now().Add(5*time.Minute), nil, nil, []string{"test1", "bad"})
|
|
assert.EqualError(t, err, "certificate contained a group not present on the signing ca: bad")
|
|
|
|
c, _, _, err = newTestCert(ca, caKey, time.Now(), time.Now().Add(5*time.Minute), nil, nil, []string{"test1"})
|
|
assert.Nil(t, err)
|
|
_, err = caPool.VerifyCertificate(time.Now(), c)
|
|
assert.Nil(t, err)
|
|
}
|
|
|
|
func TestNebulaCertificate_Verify_IPs(t *testing.T) {
|
|
caIp1 := mustParsePrefixUnmapped("10.0.0.0/16")
|
|
caIp2 := mustParsePrefixUnmapped("192.168.0.0/24")
|
|
ca, _, caKey, err := newTestCaCert(time.Now(), time.Now().Add(10*time.Minute), []netip.Prefix{caIp1, caIp2}, nil, []string{"test"})
|
|
assert.Nil(t, err)
|
|
|
|
caPem, err := ca.MarshalPEM()
|
|
assert.Nil(t, err)
|
|
|
|
caPool := NewCAPool()
|
|
b, err := caPool.AddCAFromPEM(caPem)
|
|
assert.NoError(t, err)
|
|
assert.Empty(t, b)
|
|
|
|
// ip is outside the network
|
|
cIp1 := mustParsePrefixUnmapped("10.1.0.0/24")
|
|
cIp2 := mustParsePrefixUnmapped("192.168.0.1/16")
|
|
c, _, _, err := newTestCert(ca, caKey, time.Now(), time.Now().Add(5*time.Minute), []netip.Prefix{cIp1, cIp2}, nil, []string{"test"})
|
|
assert.EqualError(t, err, "certificate contained a network assignment outside the limitations of the signing ca: 10.1.0.0/24")
|
|
|
|
// ip is outside the network reversed order of above
|
|
cIp1 = mustParsePrefixUnmapped("192.168.0.1/24")
|
|
cIp2 = mustParsePrefixUnmapped("10.1.0.0/24")
|
|
c, _, _, err = newTestCert(ca, caKey, time.Now(), time.Now().Add(5*time.Minute), []netip.Prefix{cIp1, cIp2}, nil, []string{"test"})
|
|
assert.EqualError(t, err, "certificate contained a network assignment outside the limitations of the signing ca: 10.1.0.0/24")
|
|
|
|
// ip is within the network but mask is outside
|
|
cIp1 = mustParsePrefixUnmapped("10.0.1.0/15")
|
|
cIp2 = mustParsePrefixUnmapped("192.168.0.1/24")
|
|
c, _, _, err = newTestCert(ca, caKey, time.Now(), time.Now().Add(5*time.Minute), []netip.Prefix{cIp1, cIp2}, nil, []string{"test"})
|
|
assert.EqualError(t, err, "certificate contained a network assignment outside the limitations of the signing ca: 10.0.1.0/15")
|
|
|
|
// ip is within the network but mask is outside reversed order of above
|
|
cIp1 = mustParsePrefixUnmapped("192.168.0.1/24")
|
|
cIp2 = mustParsePrefixUnmapped("10.0.1.0/15")
|
|
c, _, _, err = newTestCert(ca, caKey, time.Now(), time.Now().Add(5*time.Minute), []netip.Prefix{cIp1, cIp2}, nil, []string{"test"})
|
|
assert.EqualError(t, err, "certificate contained a network assignment outside the limitations of the signing ca: 10.0.1.0/15")
|
|
|
|
// ip and mask are within the network
|
|
cIp1 = mustParsePrefixUnmapped("10.0.1.0/16")
|
|
cIp2 = mustParsePrefixUnmapped("192.168.0.1/25")
|
|
c, _, _, err = newTestCert(ca, caKey, time.Now(), time.Now().Add(5*time.Minute), []netip.Prefix{cIp1, cIp2}, nil, []string{"test"})
|
|
assert.Nil(t, err)
|
|
_, err = caPool.VerifyCertificate(time.Now(), c)
|
|
assert.Nil(t, err)
|
|
|
|
// Exact matches
|
|
c, _, _, err = newTestCert(ca, caKey, time.Now(), time.Now().Add(5*time.Minute), []netip.Prefix{caIp1, caIp2}, nil, []string{"test"})
|
|
assert.Nil(t, err)
|
|
_, err = caPool.VerifyCertificate(time.Now(), c)
|
|
assert.Nil(t, err)
|
|
|
|
// Exact matches reversed
|
|
c, _, _, err = newTestCert(ca, caKey, time.Now(), time.Now().Add(5*time.Minute), []netip.Prefix{caIp2, caIp1}, nil, []string{"test"})
|
|
assert.Nil(t, err)
|
|
_, err = caPool.VerifyCertificate(time.Now(), c)
|
|
assert.Nil(t, err)
|
|
|
|
// Exact matches reversed with just 1
|
|
c, _, _, err = newTestCert(ca, caKey, time.Now(), time.Now().Add(5*time.Minute), []netip.Prefix{caIp1}, nil, []string{"test"})
|
|
assert.Nil(t, err)
|
|
_, err = caPool.VerifyCertificate(time.Now(), c)
|
|
assert.Nil(t, err)
|
|
}
|
|
|
|
func TestNebulaCertificate_Verify_Subnets(t *testing.T) {
|
|
caIp1 := mustParsePrefixUnmapped("10.0.0.0/16")
|
|
caIp2 := mustParsePrefixUnmapped("192.168.0.0/24")
|
|
ca, _, caKey, err := newTestCaCert(time.Now(), time.Now().Add(10*time.Minute), nil, []netip.Prefix{caIp1, caIp2}, []string{"test"})
|
|
assert.Nil(t, err)
|
|
|
|
caPem, err := ca.MarshalPEM()
|
|
assert.Nil(t, err)
|
|
|
|
caPool := NewCAPool()
|
|
b, err := caPool.AddCAFromPEM(caPem)
|
|
assert.NoError(t, err)
|
|
assert.Empty(t, b)
|
|
|
|
// ip is outside the network
|
|
cIp1 := mustParsePrefixUnmapped("10.1.0.0/24")
|
|
cIp2 := mustParsePrefixUnmapped("192.168.0.1/16")
|
|
c, _, _, err := newTestCert(ca, caKey, time.Now(), time.Now().Add(5*time.Minute), nil, []netip.Prefix{cIp1, cIp2}, []string{"test"})
|
|
assert.EqualError(t, err, "certificate contained an unsafe network assignment outside the limitations of the signing ca: 10.1.0.0/24")
|
|
|
|
// ip is outside the network reversed order of above
|
|
cIp1 = mustParsePrefixUnmapped("192.168.0.1/24")
|
|
cIp2 = mustParsePrefixUnmapped("10.1.0.0/24")
|
|
c, _, _, err = newTestCert(ca, caKey, time.Now(), time.Now().Add(5*time.Minute), nil, []netip.Prefix{cIp1, cIp2}, []string{"test"})
|
|
assert.EqualError(t, err, "certificate contained an unsafe network assignment outside the limitations of the signing ca: 10.1.0.0/24")
|
|
|
|
// ip is within the network but mask is outside
|
|
cIp1 = mustParsePrefixUnmapped("10.0.1.0/15")
|
|
cIp2 = mustParsePrefixUnmapped("192.168.0.1/24")
|
|
c, _, _, err = newTestCert(ca, caKey, time.Now(), time.Now().Add(5*time.Minute), nil, []netip.Prefix{cIp1, cIp2}, []string{"test"})
|
|
assert.EqualError(t, err, "certificate contained an unsafe network assignment outside the limitations of the signing ca: 10.0.1.0/15")
|
|
|
|
// ip is within the network but mask is outside reversed order of above
|
|
cIp1 = mustParsePrefixUnmapped("192.168.0.1/24")
|
|
cIp2 = mustParsePrefixUnmapped("10.0.1.0/15")
|
|
c, _, _, err = newTestCert(ca, caKey, time.Now(), time.Now().Add(5*time.Minute), nil, []netip.Prefix{cIp1, cIp2}, []string{"test"})
|
|
assert.EqualError(t, err, "certificate contained an unsafe network assignment outside the limitations of the signing ca: 10.0.1.0/15")
|
|
|
|
// ip and mask are within the network
|
|
cIp1 = mustParsePrefixUnmapped("10.0.1.0/16")
|
|
cIp2 = mustParsePrefixUnmapped("192.168.0.1/25")
|
|
c, _, _, err = newTestCert(ca, caKey, time.Now(), time.Now().Add(5*time.Minute), nil, []netip.Prefix{cIp1, cIp2}, []string{"test"})
|
|
assert.Nil(t, err)
|
|
_, err = caPool.VerifyCertificate(time.Now(), c)
|
|
assert.Nil(t, err)
|
|
|
|
// Exact matches
|
|
c, _, _, err = newTestCert(ca, caKey, time.Now(), time.Now().Add(5*time.Minute), nil, []netip.Prefix{caIp1, caIp2}, []string{"test"})
|
|
assert.Nil(t, err)
|
|
_, err = caPool.VerifyCertificate(time.Now(), c)
|
|
assert.Nil(t, err)
|
|
|
|
// Exact matches reversed
|
|
c, _, _, err = newTestCert(ca, caKey, time.Now(), time.Now().Add(5*time.Minute), nil, []netip.Prefix{caIp2, caIp1}, []string{"test"})
|
|
assert.Nil(t, err)
|
|
_, err = caPool.VerifyCertificate(time.Now(), c)
|
|
assert.Nil(t, err)
|
|
|
|
// Exact matches reversed with just 1
|
|
c, _, _, err = newTestCert(ca, caKey, time.Now(), time.Now().Add(5*time.Minute), nil, []netip.Prefix{caIp1}, []string{"test"})
|
|
assert.Nil(t, err)
|
|
_, err = caPool.VerifyCertificate(time.Now(), c)
|
|
assert.Nil(t, err)
|
|
}
|
|
|
|
func TestNebulaCertificate_VerifyPrivateKey(t *testing.T) {
|
|
ca, _, caKey, err := newTestCaCert(time.Time{}, time.Time{}, nil, nil, nil)
|
|
assert.Nil(t, err)
|
|
err = ca.VerifyPrivateKey(Curve_CURVE25519, caKey)
|
|
assert.Nil(t, err)
|
|
|
|
_, _, caKey2, err := newTestCaCert(time.Time{}, time.Time{}, nil, nil, nil)
|
|
assert.Nil(t, err)
|
|
err = ca.VerifyPrivateKey(Curve_CURVE25519, caKey2)
|
|
assert.NotNil(t, err)
|
|
|
|
c, _, priv, err := newTestCert(ca, caKey, time.Time{}, time.Time{}, nil, nil, nil)
|
|
err = c.VerifyPrivateKey(Curve_CURVE25519, priv)
|
|
assert.Nil(t, err)
|
|
|
|
_, priv2 := x25519Keypair()
|
|
err = c.VerifyPrivateKey(Curve_CURVE25519, priv2)
|
|
assert.NotNil(t, err)
|
|
}
|
|
|
|
func TestNebulaCertificate_VerifyPrivateKeyP256(t *testing.T) {
|
|
ca, _, caKey, err := newTestCaCertP256(time.Time{}, time.Time{}, nil, nil, nil)
|
|
assert.Nil(t, err)
|
|
err = ca.VerifyPrivateKey(Curve_P256, caKey)
|
|
assert.Nil(t, err)
|
|
|
|
_, _, caKey2, err := newTestCaCertP256(time.Time{}, time.Time{}, nil, nil, nil)
|
|
assert.Nil(t, err)
|
|
err = ca.VerifyPrivateKey(Curve_P256, caKey2)
|
|
assert.NotNil(t, err)
|
|
|
|
c, _, priv, err := newTestCert(ca, caKey, time.Time{}, time.Time{}, nil, nil, nil)
|
|
err = c.VerifyPrivateKey(Curve_P256, priv)
|
|
assert.Nil(t, err)
|
|
|
|
_, priv2 := p256Keypair()
|
|
err = c.VerifyPrivateKey(Curve_P256, priv2)
|
|
assert.NotNil(t, err)
|
|
}
|
|
|
|
func appendByteSlices(b ...[]byte) []byte {
|
|
retSlice := []byte{}
|
|
for _, v := range b {
|
|
retSlice = append(retSlice, v...)
|
|
}
|
|
return retSlice
|
|
}
|
|
|
|
// Ensure that upgrading the protobuf library does not change how certificates
|
|
// are marshalled, since this would break signature verification
|
|
//TODO: since netip cant represent 255.0.255.0 netmask we can't verify the old certs are ok
|
|
//func TestMarshalingNebulaCertificateConsistency(t *testing.T) {
|
|
// before := time.Date(2009, time.November, 10, 23, 0, 0, 0, time.UTC)
|
|
// after := time.Date(2017, time.January, 18, 28, 40, 0, 0, time.UTC)
|
|
// pubKey := []byte("1234567890abcedfghij1234567890ab")
|
|
//
|
|
// nc := certificateV1{
|
|
// details: detailsV1{
|
|
// Name: "testing",
|
|
// Ips: []netip.Prefix{
|
|
// mustParsePrefixUnmapped("10.1.1.1/24"),
|
|
// mustParsePrefixUnmapped("10.1.1.2/16"),
|
|
// //TODO: netip bad
|
|
// //{IP: net.ParseIP("10.1.1.3"), Mask: net.IPMask(net.ParseIP("255.0.255.0"))},
|
|
// },
|
|
// Subnets: []netip.Prefix{
|
|
// //TODO: netip bad
|
|
// //{IP: net.ParseIP("9.1.1.1"), Mask: net.IPMask(net.ParseIP("255.0.255.0"))},
|
|
// mustParsePrefixUnmapped("9.1.1.2/24"),
|
|
// mustParsePrefixUnmapped("9.1.1.3/16"),
|
|
// },
|
|
// Groups: []string{"test-group1", "test-group2", "test-group3"},
|
|
// NotBefore: before,
|
|
// NotAfter: after,
|
|
// PublicKey: pubKey,
|
|
// IsCA: false,
|
|
// Issuer: "1234567890abcedfghij1234567890ab",
|
|
// },
|
|
// signature: []byte("1234567890abcedfghij1234567890ab"),
|
|
// }
|
|
//
|
|
// b, err := nc.Marshal()
|
|
// assert.Nil(t, err)
|
|
// //t.Log("Cert size:", len(b))
|
|
// assert.Equal(t, "0aa2010a0774657374696e67121b8182845080feffff0f828284508080fcff0f8382845080fe83f80f1a1b8182844880fe83f80f8282844880feffff0f838284488080fcff0f220b746573742d67726f757031220b746573742d67726f757032220b746573742d67726f75703328f0e0e7d70430a08681c4053a20313233343536373839306162636564666768696a3132333435363738393061624a081234567890abcedf1220313233343536373839306162636564666768696a313233343536373839306162", fmt.Sprintf("%x", b))
|
|
//
|
|
// b, err = proto.Marshal(nc.getRawDetails())
|
|
// assert.Nil(t, err)
|
|
// //t.Log("Raw cert size:", len(b))
|
|
// assert.Equal(t, "0a0774657374696e67121b8182845080feffff0f828284508080fcff0f8382845080fe83f80f1a1b8182844880fe83f80f8282844880feffff0f838284488080fcff0f220b746573742d67726f757031220b746573742d67726f757032220b746573742d67726f75703328f0e0e7d70430a08681c4053a20313233343536373839306162636564666768696a3132333435363738393061624a081234567890abcedf", fmt.Sprintf("%x", b))
|
|
//}
|
|
|
|
func TestNebulaCertificate_Copy(t *testing.T) {
|
|
ca, _, caKey, err := newTestCaCert(time.Now(), time.Now().Add(10*time.Minute), nil, nil, nil)
|
|
assert.Nil(t, err)
|
|
|
|
c, _, _, err := newTestCert(ca, caKey, time.Now(), time.Now().Add(5*time.Minute), nil, nil, nil)
|
|
assert.Nil(t, err)
|
|
cc := c.Copy()
|
|
|
|
test.AssertDeepCopyEqual(t, c, cc)
|
|
}
|
|
|
|
func TestUnmarshalNebulaCertificate(t *testing.T) {
|
|
// Test that we don't panic with an invalid certificate (#332)
|
|
data := []byte("\x98\x00\x00")
|
|
_, err := unmarshalCertificateV1(data, true)
|
|
assert.EqualError(t, err, "encoded Details was nil")
|
|
}
|
|
|
|
func newTestCaCert(before, after time.Time, ips, subnets []netip.Prefix, groups []string) (Certificate, []byte, []byte, error) {
|
|
pub, priv, err := ed25519.GenerateKey(rand.Reader)
|
|
if before.IsZero() {
|
|
before = time.Now().Add(time.Second * -60).Round(time.Second)
|
|
}
|
|
if after.IsZero() {
|
|
after = time.Now().Add(time.Second * 60).Round(time.Second)
|
|
}
|
|
|
|
tbs := &TBSCertificate{
|
|
Version: Version1,
|
|
Name: "test ca",
|
|
IsCA: true,
|
|
NotBefore: time.Unix(before.Unix(), 0),
|
|
NotAfter: time.Unix(after.Unix(), 0),
|
|
PublicKey: pub,
|
|
}
|
|
|
|
if len(ips) > 0 {
|
|
tbs.Networks = ips
|
|
}
|
|
|
|
if len(subnets) > 0 {
|
|
tbs.UnsafeNetworks = subnets
|
|
}
|
|
|
|
if len(groups) > 0 {
|
|
tbs.Groups = groups
|
|
}
|
|
|
|
nc, err := tbs.Sign(nil, Curve_CURVE25519, priv)
|
|
if err != nil {
|
|
return nil, nil, nil, err
|
|
}
|
|
return nc, pub, priv, nil
|
|
}
|
|
|
|
func newTestCaCertP256(before, after time.Time, ips, subnets []netip.Prefix, groups []string) (Certificate, []byte, []byte, error) {
|
|
priv, err := ecdsa.GenerateKey(elliptic.P256(), rand.Reader)
|
|
pub := elliptic.Marshal(elliptic.P256(), priv.PublicKey.X, priv.PublicKey.Y)
|
|
rawPriv := priv.D.FillBytes(make([]byte, 32))
|
|
|
|
if before.IsZero() {
|
|
before = time.Now().Add(time.Second * -60).Round(time.Second)
|
|
}
|
|
if after.IsZero() {
|
|
after = time.Now().Add(time.Second * 60).Round(time.Second)
|
|
}
|
|
|
|
tbs := &TBSCertificate{
|
|
Version: Version1,
|
|
Name: "test ca",
|
|
IsCA: true,
|
|
NotBefore: time.Unix(before.Unix(), 0),
|
|
NotAfter: time.Unix(after.Unix(), 0),
|
|
PublicKey: pub,
|
|
Curve: Curve_P256,
|
|
}
|
|
|
|
if len(ips) > 0 {
|
|
tbs.Networks = ips
|
|
}
|
|
|
|
if len(subnets) > 0 {
|
|
tbs.UnsafeNetworks = subnets
|
|
}
|
|
|
|
if len(groups) > 0 {
|
|
tbs.Groups = groups
|
|
}
|
|
|
|
nc, err := tbs.Sign(nil, Curve_P256, rawPriv)
|
|
if err != nil {
|
|
return nil, nil, nil, err
|
|
}
|
|
return nc, pub, rawPriv, nil
|
|
}
|
|
|
|
func newTestCert(ca Certificate, key []byte, before, after time.Time, ips, subnets []netip.Prefix, groups []string) (Certificate, []byte, []byte, error) {
|
|
if before.IsZero() {
|
|
before = time.Now().Add(time.Second * -60).Round(time.Second)
|
|
}
|
|
if after.IsZero() {
|
|
after = time.Now().Add(time.Second * 60).Round(time.Second)
|
|
}
|
|
|
|
if len(groups) == 0 {
|
|
groups = []string{"test-group1", "test-group2", "test-group3"}
|
|
}
|
|
|
|
if len(ips) == 0 {
|
|
ips = []netip.Prefix{
|
|
mustParsePrefixUnmapped("10.1.1.1/24"),
|
|
mustParsePrefixUnmapped("10.1.1.2/16"),
|
|
}
|
|
}
|
|
|
|
if len(subnets) == 0 {
|
|
subnets = []netip.Prefix{
|
|
mustParsePrefixUnmapped("9.1.1.2/24"),
|
|
mustParsePrefixUnmapped("9.1.1.3/16"),
|
|
}
|
|
}
|
|
|
|
var pub, rawPriv []byte
|
|
|
|
switch ca.Curve() {
|
|
case Curve_CURVE25519:
|
|
pub, rawPriv = x25519Keypair()
|
|
case Curve_P256:
|
|
pub, rawPriv = p256Keypair()
|
|
default:
|
|
return nil, nil, nil, fmt.Errorf("unknown curve: %v", ca.Curve())
|
|
}
|
|
|
|
tbs := &TBSCertificate{
|
|
Version: Version1,
|
|
Name: "testing",
|
|
Networks: ips,
|
|
UnsafeNetworks: subnets,
|
|
Groups: groups,
|
|
IsCA: false,
|
|
NotBefore: time.Unix(before.Unix(), 0),
|
|
NotAfter: time.Unix(after.Unix(), 0),
|
|
PublicKey: pub,
|
|
Curve: ca.Curve(),
|
|
}
|
|
|
|
nc, err := tbs.Sign(ca, ca.Curve(), key)
|
|
if err != nil {
|
|
return nil, nil, nil, err
|
|
}
|
|
|
|
return nc, pub, rawPriv, nil
|
|
}
|
|
|
|
func x25519Keypair() ([]byte, []byte) {
|
|
privkey := make([]byte, 32)
|
|
if _, err := io.ReadFull(rand.Reader, privkey); err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
pubkey, err := curve25519.X25519(privkey, curve25519.Basepoint)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
return pubkey, privkey
|
|
}
|
|
|
|
func p256Keypair() ([]byte, []byte) {
|
|
privkey, err := ecdh.P256().GenerateKey(rand.Reader)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
pubkey := privkey.PublicKey()
|
|
return pubkey.Bytes(), privkey.Bytes()
|
|
}
|
|
|
|
func mustParsePrefixUnmapped(s string) netip.Prefix {
|
|
prefix := netip.MustParsePrefix(s)
|
|
return netip.PrefixFrom(prefix.Addr().Unmap(), prefix.Bits())
|
|
}
|