package turn import ( "crypto/ecdsa" "crypto/elliptic" "crypto/rand" "crypto/x509" "crypto/x509/pkix" "encoding/pem" "fmt" "math/big" "net" "os" "time" ) // GenerateSelfSignedCert generates a self-signed TLS certificate for TURNS. // The certificate is valid for 1 year and includes the public IP as a SAN. func GenerateSelfSignedCert(certPath, keyPath, publicIP string) error { key, err := ecdsa.GenerateKey(elliptic.P256(), rand.Reader) if err != nil { return fmt.Errorf("failed to generate private key: %w", err) } serialNumber, err := rand.Int(rand.Reader, new(big.Int).Lsh(big.NewInt(1), 128)) if err != nil { return fmt.Errorf("failed to generate serial number: %w", err) } template := x509.Certificate{ SerialNumber: serialNumber, Subject: pkix.Name{ Organization: []string{"Orama Network"}, CommonName: "TURN Server", }, NotBefore: time.Now(), NotAfter: time.Now().Add(365 * 24 * time.Hour), KeyUsage: x509.KeyUsageDigitalSignature | x509.KeyUsageKeyEncipherment, ExtKeyUsage: []x509.ExtKeyUsage{x509.ExtKeyUsageServerAuth}, BasicConstraintsValid: true, } if ip := net.ParseIP(publicIP); ip != nil { template.IPAddresses = append(template.IPAddresses, ip) } certDER, err := x509.CreateCertificate(rand.Reader, &template, &template, &key.PublicKey, key) if err != nil { return fmt.Errorf("failed to create certificate: %w", err) } certFile, err := os.Create(certPath) if err != nil { return fmt.Errorf("failed to create cert file: %w", err) } defer certFile.Close() if err := pem.Encode(certFile, &pem.Block{Type: "CERTIFICATE", Bytes: certDER}); err != nil { return fmt.Errorf("failed to write cert PEM: %w", err) } keyDER, err := x509.MarshalECPrivateKey(key) if err != nil { return fmt.Errorf("failed to marshal private key: %w", err) } keyFile, err := os.Create(keyPath) if err != nil { return fmt.Errorf("failed to create key file: %w", err) } defer keyFile.Close() if err := pem.Encode(keyFile, &pem.Block{Type: "EC PRIVATE KEY", Bytes: keyDER}); err != nil { return fmt.Errorf("failed to write key PEM: %w", err) } // Restrict key file permissions if err := os.Chmod(keyPath, 0600); err != nil { return fmt.Errorf("failed to set key file permissions: %w", err) } return nil }