pkcs12.go 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164
  1. /*
  2. Licensed under the Apache License, Version 2.0 (the "License");
  3. you may not use this file except in compliance with the License.
  4. You may obtain a copy of the License at
  5. http://www.apache.org/licenses/LICENSE-2.0
  6. Unless required by applicable law or agreed to in writing, software
  7. distributed under the License is distributed on an "AS IS" BASIS,
  8. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  9. See the License for the specific language governing permissions and
  10. limitations under the License.
  11. */
  12. package template
  13. import (
  14. "bytes"
  15. "crypto/x509"
  16. "encoding/base64"
  17. "encoding/pem"
  18. "fmt"
  19. gopkcs12 "software.sslmate.com/src/go-pkcs12"
  20. )
  21. func pkcs12keyPass(pass, input string) (string, error) {
  22. privateKey, _, _, err := gopkcs12.DecodeChain([]byte(input), pass)
  23. if err != nil {
  24. return "", fmt.Errorf(errDecodePKCS12WithPass, err)
  25. }
  26. marshalPrivateKey, err := x509.MarshalPKCS8PrivateKey(privateKey)
  27. if err != nil {
  28. return "", err
  29. }
  30. var buf bytes.Buffer
  31. if err := pem.Encode(&buf, &pem.Block{
  32. Type: pemTypeKey,
  33. Bytes: marshalPrivateKey,
  34. }); err != nil {
  35. return "", err
  36. }
  37. return buf.String(), nil
  38. }
  39. func parsePrivateKey(block []byte) (any, error) {
  40. if k, err := x509.ParsePKCS1PrivateKey(block); err == nil {
  41. return k, nil
  42. }
  43. if k, err := x509.ParsePKCS8PrivateKey(block); err == nil {
  44. return k, nil
  45. }
  46. if k, err := x509.ParseECPrivateKey(block); err == nil {
  47. return k, nil
  48. }
  49. return nil, fmt.Errorf(errParsePrivKey)
  50. }
  51. func pkcs12key(input string) (string, error) {
  52. return pkcs12keyPass("", input)
  53. }
  54. func pkcs12certPass(pass, input string) (string, error) {
  55. _, certificate, caCerts, err := gopkcs12.DecodeChain([]byte(input), pass)
  56. if err != nil {
  57. return "", fmt.Errorf(errDecodeCertWithPass, err)
  58. }
  59. var pemData []byte
  60. var buf bytes.Buffer
  61. if err := pem.Encode(&buf, &pem.Block{
  62. Type: pemTypeCertificate,
  63. Bytes: certificate.Raw,
  64. }); err != nil {
  65. return "", err
  66. }
  67. pemData = append(pemData, buf.Bytes()...)
  68. for _, ca := range caCerts {
  69. var buf bytes.Buffer
  70. if err := pem.Encode(&buf, &pem.Block{
  71. Type: pemTypeCertificate,
  72. Bytes: ca.Raw,
  73. }); err != nil {
  74. return "", err
  75. }
  76. pemData = append(pemData, buf.Bytes()...)
  77. }
  78. // try to order certificate chain. If it fails we return
  79. // the unordered raw pem data.
  80. // This fails if multiple leaf or disjunct certs are provided.
  81. ordered, err := fetchCertChains(pemData)
  82. if err != nil {
  83. return string(pemData), nil
  84. }
  85. return string(ordered), nil
  86. }
  87. func pkcs12cert(input string) (string, error) {
  88. return pkcs12certPass("", input)
  89. }
  90. func pemToPkcs12(cert, key string) (string, error) {
  91. return pemToPkcs12Pass(cert, key, "")
  92. }
  93. func pemToPkcs12Pass(cert, key, pass string) (string, error) {
  94. certPem, _ := pem.Decode([]byte(cert))
  95. parsedCert, err := x509.ParseCertificate(certPem.Bytes)
  96. if err != nil {
  97. return "", err
  98. }
  99. return certsToPkcs12(parsedCert, key, nil, pass)
  100. }
  101. func fullPemToPkcs12(cert, key string) (string, error) {
  102. return fullPemToPkcs12Pass(cert, key, "")
  103. }
  104. func fullPemToPkcs12Pass(cert, key, pass string) (string, error) {
  105. certPem, rest := pem.Decode([]byte(cert))
  106. parsedCert, err := x509.ParseCertificate(certPem.Bytes)
  107. if err != nil {
  108. return "", err
  109. }
  110. caCerts := make([]*x509.Certificate, 0)
  111. for len(rest) > 0 {
  112. caPem, restBytes := pem.Decode(rest)
  113. rest = restBytes
  114. caCert, err := x509.ParseCertificate(caPem.Bytes)
  115. if err != nil {
  116. return "", err
  117. }
  118. caCerts = append(caCerts, caCert)
  119. }
  120. return certsToPkcs12(parsedCert, key, caCerts, pass)
  121. }
  122. func certsToPkcs12(cert *x509.Certificate, key string, caCerts []*x509.Certificate, password string) (string, error) {
  123. keyPem, _ := pem.Decode([]byte(key))
  124. parsedKey, err := parsePrivateKey(keyPem.Bytes)
  125. if err != nil {
  126. return "", err
  127. }
  128. pfx, err := gopkcs12.Modern.Encode(parsedKey, cert, caCerts, password)
  129. if err != nil {
  130. return "", err
  131. }
  132. return base64.StdEncoding.EncodeToString(pfx), nil
  133. }