package caddytls import ( "crypto/ecdsa" "crypto/elliptic" "crypto/rand" "crypto/rsa" "crypto/tls" "crypto/x509" "crypto/x509/pkix" "fmt" "math/big" "net" "strings" "time" "github.com/xenolf/lego/certcrypto" ) // newSelfSignedCertificate returns a new self-signed certificate. func newSelfSignedCertificate(ssconfig selfSignedConfig) (tls.Certificate, error) { // start by generating private key var privKey interface{} var err error switch ssconfig.KeyType { case "", certcrypto.EC256: privKey, err = ecdsa.GenerateKey(elliptic.P256(), rand.Reader) case certcrypto.EC384: privKey, err = ecdsa.GenerateKey(elliptic.P384(), rand.Reader) case certcrypto.RSA2048: privKey, err = rsa.GenerateKey(rand.Reader, 2048) case certcrypto.RSA4096: privKey, err = rsa.GenerateKey(rand.Reader, 4096) case certcrypto.RSA8192: privKey, err = rsa.GenerateKey(rand.Reader, 8192) default: return tls.Certificate{}, fmt.Errorf("cannot generate private key; unknown key type %v", ssconfig.KeyType) } if err != nil { return tls.Certificate{}, fmt.Errorf("failed to generate private key: %v", err) } // create certificate structure with proper values notBefore := time.Now() notAfter := ssconfig.Expire if notAfter.IsZero() || notAfter.Before(notBefore) { notAfter = notBefore.Add(24 * time.Hour * 7) } serialNumberLimit := new(big.Int).Lsh(big.NewInt(1), 128) serialNumber, err := rand.Int(rand.Reader, serialNumberLimit) if err != nil { return tls.Certificate{}, fmt.Errorf("failed to generate serial number: %v", err) } cert := &x509.Certificate{ SerialNumber: serialNumber, Subject: pkix.Name{Organization: []string{"Caddy Self-Signed"}}, NotBefore: notBefore, NotAfter: notAfter, KeyUsage: x509.KeyUsageKeyEncipherment | x509.KeyUsageDigitalSignature, ExtKeyUsage: []x509.ExtKeyUsage{x509.ExtKeyUsageServerAuth}, } if len(ssconfig.SAN) == 0 { ssconfig.SAN = []string{""} } for _, san := range ssconfig.SAN { if ip := net.ParseIP(san); ip != nil { cert.IPAddresses = append(cert.IPAddresses, ip) } else { cert.DNSNames = append(cert.DNSNames, strings.ToLower(san)) } } // generate the associated public key publicKey := func(privKey interface{}) interface{} { switch k := privKey.(type) { case *rsa.PrivateKey: return &k.PublicKey case *ecdsa.PrivateKey: return &k.PublicKey default: return fmt.Errorf("unknown key type") } } derBytes, err := x509.CreateCertificate(rand.Reader, cert, cert, publicKey(privKey), privKey) if err != nil { return tls.Certificate{}, fmt.Errorf("could not create certificate: %v", err) } chain := [][]byte{derBytes} return tls.Certificate{ Certificate: chain, PrivateKey: privKey, Leaf: cert, }, nil } // selfSignedConfig configures a self-signed certificate. type selfSignedConfig struct { SAN []string KeyType certcrypto.KeyType Expire time.Time }