mirror of
https://github.com/v2fly/v2ray-core.git
synced 2025-01-20 08:16:55 -05:00
78 lines
2.3 KiB
Go
78 lines
2.3 KiB
Go
package shadowsocks2022
|
|
|
|
import (
|
|
"crypto/aes"
|
|
"crypto/cipher"
|
|
)
|
|
|
|
func newAES256GCMMethod() *AES256GCMMethod {
|
|
return &AES256GCMMethod{}
|
|
}
|
|
|
|
type AES256GCMMethod struct{}
|
|
|
|
func (a AES256GCMMethod) GetSessionSubKeyAndSaltLength() int {
|
|
return 32
|
|
}
|
|
|
|
func (a AES256GCMMethod) GetStreamAEAD(sessionSubKey []byte) (cipher.AEAD, error) {
|
|
aesCipher, err := aes.NewCipher(sessionSubKey)
|
|
if err != nil {
|
|
return nil, newError("failed to create AES cipher").Base(err)
|
|
}
|
|
aead, err := cipher.NewGCM(aesCipher)
|
|
if err != nil {
|
|
return nil, newError("failed to create AES-GCM AEAD").Base(err)
|
|
}
|
|
return aead, nil
|
|
}
|
|
|
|
func (a AES256GCMMethod) GenerateEIH(currentIdentitySubKey []byte, nextPskHash []byte, out []byte) error {
|
|
aesCipher, err := aes.NewCipher(currentIdentitySubKey)
|
|
if err != nil {
|
|
return newError("failed to create AES cipher").Base(err)
|
|
}
|
|
aesCipher.Encrypt(out, nextPskHash)
|
|
return nil
|
|
}
|
|
|
|
func (a AES256GCMMethod) GetUDPClientProcessor(ipsk [][]byte, psk []byte, derivation KeyDerivation) (UDPClientPacketProcessor, error) {
|
|
reqSeparateHeaderPsk := psk
|
|
if ipsk != nil {
|
|
reqSeparateHeaderPsk = ipsk[0]
|
|
}
|
|
reqSeparateHeaderCipher, err := aes.NewCipher(reqSeparateHeaderPsk)
|
|
if err != nil {
|
|
return nil, newError("failed to create AES cipher").Base(err)
|
|
}
|
|
respSeparateHeaderCipher, err := aes.NewCipher(psk)
|
|
if err != nil {
|
|
return nil, newError("failed to create AES cipher").Base(err)
|
|
}
|
|
getPacketAEAD := func(sessionID []byte) cipher.AEAD {
|
|
sessionKey := make([]byte, a.GetSessionSubKeyAndSaltLength())
|
|
derivation.GetSessionSubKey(psk, sessionID, sessionKey)
|
|
block, err := aes.NewCipher(sessionKey)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
aead, err := cipher.NewGCM(block)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
return aead
|
|
}
|
|
if len(ipsk) == 0 {
|
|
return NewAESUDPClientPacketProcessor(reqSeparateHeaderCipher, respSeparateHeaderCipher, getPacketAEAD, nil), nil
|
|
}
|
|
eihGenerator := newAESEIHGeneratorContainer(len(ipsk), psk, ipsk)
|
|
getEIH := func(mask []byte) ExtensibleIdentityHeaders {
|
|
eih, err := eihGenerator.GenerateEIHUDP(derivation, a, mask)
|
|
if err != nil {
|
|
newError("failed to generate EIH").Base(err).WriteToLog()
|
|
}
|
|
return eih
|
|
}
|
|
return NewAESUDPClientPacketProcessor(reqSeparateHeaderCipher, respSeparateHeaderCipher, getPacketAEAD, getEIH), nil
|
|
}
|