cipher_suite.go 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213
  1. package dtls
  2. import (
  3. "fmt"
  4. "hash"
  5. "github.com/pion/dtls/v2/internal/ciphersuite"
  6. "github.com/pion/dtls/v2/pkg/crypto/clientcertificate"
  7. "github.com/pion/dtls/v2/pkg/protocol/recordlayer"
  8. )
  9. // CipherSuiteID is an ID for our supported CipherSuites
  10. type CipherSuiteID = ciphersuite.ID
  11. // Supported Cipher Suites
  12. const (
  13. // AES-128-CCM
  14. TLS_ECDHE_ECDSA_WITH_AES_128_CCM CipherSuiteID = ciphersuite.TLS_ECDHE_ECDSA_WITH_AES_128_CCM //nolint:golint,stylecheck
  15. TLS_ECDHE_ECDSA_WITH_AES_128_CCM_8 CipherSuiteID = ciphersuite.TLS_ECDHE_ECDSA_WITH_AES_128_CCM_8 //nolint:golint,stylecheck
  16. // AES-128-GCM-SHA256
  17. TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256 CipherSuiteID = ciphersuite.TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256 //nolint:golint,stylecheck
  18. TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256 CipherSuiteID = ciphersuite.TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256 //nolint:golint,stylecheck
  19. // AES-256-CBC-SHA
  20. TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA CipherSuiteID = ciphersuite.TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA //nolint:golint,stylecheck
  21. TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA CipherSuiteID = ciphersuite.TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA //nolint:golint,stylecheck
  22. TLS_PSK_WITH_AES_128_CCM CipherSuiteID = ciphersuite.TLS_PSK_WITH_AES_128_CCM //nolint:golint,stylecheck
  23. TLS_PSK_WITH_AES_128_CCM_8 CipherSuiteID = ciphersuite.TLS_PSK_WITH_AES_128_CCM_8 //nolint:golint,stylecheck
  24. TLS_PSK_WITH_AES_128_GCM_SHA256 CipherSuiteID = ciphersuite.TLS_PSK_WITH_AES_128_GCM_SHA256 //nolint:golint,stylecheck
  25. TLS_PSK_WITH_AES_128_CBC_SHA256 CipherSuiteID = ciphersuite.TLS_PSK_WITH_AES_128_CBC_SHA256 //nolint:golint,stylecheck
  26. )
  27. // CipherSuiteAuthenticationType controls what authentication method is using during the handshake for a CipherSuite
  28. type CipherSuiteAuthenticationType = ciphersuite.AuthenticationType
  29. // AuthenticationType Enums
  30. const (
  31. CipherSuiteAuthenticationTypeCertificate CipherSuiteAuthenticationType = ciphersuite.AuthenticationTypeCertificate
  32. CipherSuiteAuthenticationTypePreSharedKey CipherSuiteAuthenticationType = ciphersuite.AuthenticationTypePreSharedKey
  33. CipherSuiteAuthenticationTypeAnonymous CipherSuiteAuthenticationType = ciphersuite.AuthenticationTypeAnonymous
  34. )
  35. var _ = allCipherSuites() // Necessary until this function isn't only used by Go 1.14
  36. // CipherSuite is an interface that all DTLS CipherSuites must satisfy
  37. type CipherSuite interface {
  38. // String of CipherSuite, only used for logging
  39. String() string
  40. // ID of CipherSuite.
  41. ID() CipherSuiteID
  42. // What type of Certificate does this CipherSuite use
  43. CertificateType() clientcertificate.Type
  44. // What Hash function is used during verification
  45. HashFunc() func() hash.Hash
  46. // AuthenticationType controls what authentication method is using during the handshake
  47. AuthenticationType() CipherSuiteAuthenticationType
  48. // Called when keying material has been generated, should initialize the internal cipher
  49. Init(masterSecret, clientRandom, serverRandom []byte, isClient bool) error
  50. IsInitialized() bool
  51. Encrypt(pkt *recordlayer.RecordLayer, raw []byte) ([]byte, error)
  52. Decrypt(in []byte) ([]byte, error)
  53. }
  54. // CipherSuiteName provides the same functionality as tls.CipherSuiteName
  55. // that appeared first in Go 1.14.
  56. //
  57. // Our implementation differs slightly in that it takes in a CiperSuiteID,
  58. // like the rest of our library, instead of a uint16 like crypto/tls.
  59. func CipherSuiteName(id CipherSuiteID) string {
  60. suite := cipherSuiteForID(id, nil)
  61. if suite != nil {
  62. return suite.String()
  63. }
  64. return fmt.Sprintf("0x%04X", uint16(id))
  65. }
  66. // Taken from https://www.iana.org/assignments/tls-parameters/tls-parameters.xml
  67. // A cipherSuite is a specific combination of key agreement, cipher and MAC
  68. // function.
  69. func cipherSuiteForID(id CipherSuiteID, customCiphers func() []CipherSuite) CipherSuite {
  70. switch id { //nolint:exhaustive
  71. case TLS_ECDHE_ECDSA_WITH_AES_128_CCM:
  72. return ciphersuite.NewTLSEcdheEcdsaWithAes128Ccm()
  73. case TLS_ECDHE_ECDSA_WITH_AES_128_CCM_8:
  74. return ciphersuite.NewTLSEcdheEcdsaWithAes128Ccm8()
  75. case TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256:
  76. return &ciphersuite.TLSEcdheEcdsaWithAes128GcmSha256{}
  77. case TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256:
  78. return &ciphersuite.TLSEcdheRsaWithAes128GcmSha256{}
  79. case TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA:
  80. return &ciphersuite.TLSEcdheEcdsaWithAes256CbcSha{}
  81. case TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA:
  82. return &ciphersuite.TLSEcdheRsaWithAes256CbcSha{}
  83. case TLS_PSK_WITH_AES_128_CCM:
  84. return ciphersuite.NewTLSPskWithAes128Ccm()
  85. case TLS_PSK_WITH_AES_128_CCM_8:
  86. return ciphersuite.NewTLSPskWithAes128Ccm8()
  87. case TLS_PSK_WITH_AES_128_GCM_SHA256:
  88. return &ciphersuite.TLSPskWithAes128GcmSha256{}
  89. case TLS_PSK_WITH_AES_128_CBC_SHA256:
  90. return &ciphersuite.TLSPskWithAes128CbcSha256{}
  91. }
  92. if customCiphers != nil {
  93. for _, c := range customCiphers() {
  94. if c.ID() == id {
  95. return c
  96. }
  97. }
  98. }
  99. return nil
  100. }
  101. // CipherSuites we support in order of preference
  102. func defaultCipherSuites() []CipherSuite {
  103. return []CipherSuite{
  104. &ciphersuite.TLSEcdheEcdsaWithAes128GcmSha256{},
  105. &ciphersuite.TLSEcdheRsaWithAes128GcmSha256{},
  106. &ciphersuite.TLSEcdheEcdsaWithAes256CbcSha{},
  107. &ciphersuite.TLSEcdheRsaWithAes256CbcSha{},
  108. }
  109. }
  110. func allCipherSuites() []CipherSuite {
  111. return []CipherSuite{
  112. ciphersuite.NewTLSEcdheEcdsaWithAes128Ccm(),
  113. ciphersuite.NewTLSEcdheEcdsaWithAes128Ccm8(),
  114. &ciphersuite.TLSEcdheEcdsaWithAes128GcmSha256{},
  115. &ciphersuite.TLSEcdheRsaWithAes128GcmSha256{},
  116. &ciphersuite.TLSEcdheEcdsaWithAes256CbcSha{},
  117. &ciphersuite.TLSEcdheRsaWithAes256CbcSha{},
  118. ciphersuite.NewTLSPskWithAes128Ccm(),
  119. ciphersuite.NewTLSPskWithAes128Ccm8(),
  120. &ciphersuite.TLSPskWithAes128GcmSha256{},
  121. }
  122. }
  123. func cipherSuiteIDs(cipherSuites []CipherSuite) []uint16 {
  124. rtrn := []uint16{}
  125. for _, c := range cipherSuites {
  126. rtrn = append(rtrn, uint16(c.ID()))
  127. }
  128. return rtrn
  129. }
  130. func parseCipherSuites(userSelectedSuites []CipherSuiteID, customCipherSuites func() []CipherSuite, includeCertificateSuites, includePSKSuites bool) ([]CipherSuite, error) {
  131. cipherSuitesForIDs := func(ids []CipherSuiteID) ([]CipherSuite, error) {
  132. cipherSuites := []CipherSuite{}
  133. for _, id := range ids {
  134. c := cipherSuiteForID(id, nil)
  135. if c == nil {
  136. return nil, &invalidCipherSuite{id}
  137. }
  138. cipherSuites = append(cipherSuites, c)
  139. }
  140. return cipherSuites, nil
  141. }
  142. var (
  143. cipherSuites []CipherSuite
  144. err error
  145. i int
  146. )
  147. if userSelectedSuites != nil {
  148. cipherSuites, err = cipherSuitesForIDs(userSelectedSuites)
  149. if err != nil {
  150. return nil, err
  151. }
  152. } else {
  153. cipherSuites = defaultCipherSuites()
  154. }
  155. // Put CustomCipherSuites before ID selected suites
  156. if customCipherSuites != nil {
  157. cipherSuites = append(customCipherSuites(), cipherSuites...)
  158. }
  159. var foundCertificateSuite, foundPSKSuite, foundAnonymousSuite bool
  160. for _, c := range cipherSuites {
  161. switch {
  162. case includeCertificateSuites && c.AuthenticationType() == CipherSuiteAuthenticationTypeCertificate:
  163. foundCertificateSuite = true
  164. case includePSKSuites && c.AuthenticationType() == CipherSuiteAuthenticationTypePreSharedKey:
  165. foundPSKSuite = true
  166. case c.AuthenticationType() == CipherSuiteAuthenticationTypeAnonymous:
  167. foundAnonymousSuite = true
  168. default:
  169. continue
  170. }
  171. cipherSuites[i] = c
  172. i++
  173. }
  174. switch {
  175. case includeCertificateSuites && !foundCertificateSuite && !foundAnonymousSuite:
  176. return nil, errNoAvailableCertificateCipherSuite
  177. case includePSKSuites && !foundPSKSuite:
  178. return nil, errNoAvailablePSKCipherSuite
  179. case i == 0:
  180. return nil, errNoAvailableCipherSuites
  181. }
  182. return cipherSuites[:i], nil
  183. }