| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165 |
- /*
- Licensed under the Apache License, Version 2.0 (the "License");
- you may not use this file except in compliance with the License.
- You may obtain a copy of the License at
- http://www.apache.org/licenses/LICENSE-2.0
- Unless required by applicable law or agreed to in writing, software
- distributed under the License is distributed on an "AS IS" BASIS,
- WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- See the License for the specific language governing permissions and
- limitations under the License.
- */
- package template
- import (
- "bytes"
- "crypto/x509"
- "encoding/base64"
- "encoding/pem"
- "errors"
- "fmt"
- gopkcs12 "software.sslmate.com/src/go-pkcs12"
- )
- func pkcs12keyPass(pass, input string) (string, error) {
- privateKey, _, _, err := gopkcs12.DecodeChain([]byte(input), pass)
- if err != nil {
- return "", fmt.Errorf(errDecodePKCS12WithPass, err)
- }
- marshalPrivateKey, err := x509.MarshalPKCS8PrivateKey(privateKey)
- if err != nil {
- return "", err
- }
- var buf bytes.Buffer
- if err := pem.Encode(&buf, &pem.Block{
- Type: pemTypeKey,
- Bytes: marshalPrivateKey,
- }); err != nil {
- return "", err
- }
- return buf.String(), nil
- }
- func parsePrivateKey(block []byte) (any, error) {
- if k, err := x509.ParsePKCS1PrivateKey(block); err == nil {
- return k, nil
- }
- if k, err := x509.ParsePKCS8PrivateKey(block); err == nil {
- return k, nil
- }
- if k, err := x509.ParseECPrivateKey(block); err == nil {
- return k, nil
- }
- return nil, errors.New(errParsePrivKey)
- }
- func pkcs12key(input string) (string, error) {
- return pkcs12keyPass("", input)
- }
- func pkcs12certPass(pass, input string) (string, error) {
- _, certificate, caCerts, err := gopkcs12.DecodeChain([]byte(input), pass)
- if err != nil {
- return "", fmt.Errorf(errDecodeCertWithPass, err)
- }
- var pemData []byte
- var buf bytes.Buffer
- if err := pem.Encode(&buf, &pem.Block{
- Type: pemTypeCertificate,
- Bytes: certificate.Raw,
- }); err != nil {
- return "", err
- }
- pemData = append(pemData, buf.Bytes()...)
- for _, ca := range caCerts {
- var buf bytes.Buffer
- if err := pem.Encode(&buf, &pem.Block{
- Type: pemTypeCertificate,
- Bytes: ca.Raw,
- }); err != nil {
- return "", err
- }
- pemData = append(pemData, buf.Bytes()...)
- }
- // try to order certificate chain. If it fails we return
- // the unordered raw pem data.
- // This fails if multiple leaf or disjunct certs are provided.
- ordered, err := fetchCertChains(pemData)
- if err != nil {
- return string(pemData), nil
- }
- return string(ordered), nil
- }
- func pkcs12cert(input string) (string, error) {
- return pkcs12certPass("", input)
- }
- func pemToPkcs12(cert, key string) (string, error) {
- return pemToPkcs12Pass(cert, key, "")
- }
- func pemToPkcs12Pass(cert, key, pass string) (string, error) {
- certPem, _ := pem.Decode([]byte(cert))
- parsedCert, err := x509.ParseCertificate(certPem.Bytes)
- if err != nil {
- return "", err
- }
- return certsToPkcs12(parsedCert, key, nil, pass)
- }
- func fullPemToPkcs12(cert, key string) (string, error) {
- return fullPemToPkcs12Pass(cert, key, "")
- }
- func fullPemToPkcs12Pass(cert, key, pass string) (string, error) {
- certPem, rest := pem.Decode([]byte(cert))
- parsedCert, err := x509.ParseCertificate(certPem.Bytes)
- if err != nil {
- return "", err
- }
- caCerts := make([]*x509.Certificate, 0)
- for len(rest) > 0 {
- caPem, restBytes := pem.Decode(rest)
- rest = restBytes
- caCert, err := x509.ParseCertificate(caPem.Bytes)
- if err != nil {
- return "", err
- }
- caCerts = append(caCerts, caCert)
- }
- return certsToPkcs12(parsedCert, key, caCerts, pass)
- }
- func certsToPkcs12(cert *x509.Certificate, key string, caCerts []*x509.Certificate, password string) (string, error) {
- keyPem, _ := pem.Decode([]byte(key))
- parsedKey, err := parsePrivateKey(keyPem.Bytes)
- if err != nil {
- return "", err
- }
- pfx, err := gopkcs12.Modern.Encode(parsedKey, cert, caCerts, password)
- if err != nil {
- return "", err
- }
- return base64.StdEncoding.EncodeToString(pfx), nil
- }
|