server.go 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547
  1. package encoding
  2. import (
  3. "bytes"
  4. "crypto/aes"
  5. "crypto/cipher"
  6. "crypto/md5"
  7. "crypto/sha256"
  8. "encoding/binary"
  9. "hash/fnv"
  10. "io"
  11. "io/ioutil"
  12. "sync"
  13. "time"
  14. "golang.org/x/crypto/chacha20poly1305"
  15. "github.com/v2fly/v2ray-core/v4/common"
  16. "github.com/v2fly/v2ray-core/v4/common/bitmask"
  17. "github.com/v2fly/v2ray-core/v4/common/buf"
  18. "github.com/v2fly/v2ray-core/v4/common/crypto"
  19. "github.com/v2fly/v2ray-core/v4/common/dice"
  20. "github.com/v2fly/v2ray-core/v4/common/net"
  21. "github.com/v2fly/v2ray-core/v4/common/protocol"
  22. "github.com/v2fly/v2ray-core/v4/common/task"
  23. "github.com/v2fly/v2ray-core/v4/proxy/vmess"
  24. vmessaead "github.com/v2fly/v2ray-core/v4/proxy/vmess/aead"
  25. )
  26. type sessionID struct {
  27. user [16]byte
  28. key [16]byte
  29. nonce [16]byte
  30. }
  31. // SessionHistory keeps track of historical session ids, to prevent replay attacks.
  32. type SessionHistory struct {
  33. sync.RWMutex
  34. cache map[sessionID]time.Time
  35. task *task.Periodic
  36. }
  37. // NewSessionHistory creates a new SessionHistory object.
  38. func NewSessionHistory() *SessionHistory {
  39. h := &SessionHistory{
  40. cache: make(map[sessionID]time.Time, 128),
  41. }
  42. h.task = &task.Periodic{
  43. Interval: time.Second * 30,
  44. Execute: h.removeExpiredEntries,
  45. }
  46. return h
  47. }
  48. // Close implements common.Closable.
  49. func (h *SessionHistory) Close() error {
  50. return h.task.Close()
  51. }
  52. func (h *SessionHistory) addIfNotExits(session sessionID) bool {
  53. h.Lock()
  54. if expire, found := h.cache[session]; found && expire.After(time.Now()) {
  55. h.Unlock()
  56. return false
  57. }
  58. h.cache[session] = time.Now().Add(time.Minute * 3)
  59. h.Unlock()
  60. common.Must(h.task.Start())
  61. return true
  62. }
  63. func (h *SessionHistory) removeExpiredEntries() error {
  64. now := time.Now()
  65. h.Lock()
  66. defer h.Unlock()
  67. if len(h.cache) == 0 {
  68. return newError("nothing to do")
  69. }
  70. for session, expire := range h.cache {
  71. if expire.Before(now) {
  72. delete(h.cache, session)
  73. }
  74. }
  75. if len(h.cache) == 0 {
  76. h.cache = make(map[sessionID]time.Time, 128)
  77. }
  78. return nil
  79. }
  80. // ServerSession keeps information for a session in VMess server.
  81. type ServerSession struct {
  82. userValidator *vmess.TimedUserValidator
  83. sessionHistory *SessionHistory
  84. requestBodyKey [16]byte
  85. requestBodyIV [16]byte
  86. responseBodyKey [16]byte
  87. responseBodyIV [16]byte
  88. responseWriter io.Writer
  89. responseHeader byte
  90. isAEADRequest bool
  91. isAEADForced bool
  92. }
  93. // NewServerSession creates a new ServerSession, using the given UserValidator.
  94. // The ServerSession instance doesn't take ownership of the validator.
  95. func NewServerSession(validator *vmess.TimedUserValidator, sessionHistory *SessionHistory) *ServerSession {
  96. return &ServerSession{
  97. userValidator: validator,
  98. sessionHistory: sessionHistory,
  99. }
  100. }
  101. // SetAEADForced sets isAEADForced for a ServerSession.
  102. func (s *ServerSession) SetAEADForced(isAEADForced bool) {
  103. s.isAEADForced = isAEADForced
  104. }
  105. func parseSecurityType(b byte) protocol.SecurityType {
  106. if _, f := protocol.SecurityType_name[int32(b)]; f {
  107. st := protocol.SecurityType(b)
  108. // For backward compatibility.
  109. if st == protocol.SecurityType_UNKNOWN {
  110. st = protocol.SecurityType_LEGACY
  111. }
  112. return st
  113. }
  114. return protocol.SecurityType_UNKNOWN
  115. }
  116. // DecodeRequestHeader decodes and returns (if successful) a RequestHeader from an input stream.
  117. func (s *ServerSession) DecodeRequestHeader(reader io.Reader) (*protocol.RequestHeader, error) {
  118. buffer := buf.New()
  119. behaviorRand := dice.NewDeterministicDice(int64(s.userValidator.GetBehaviorSeed()))
  120. BaseDrainSize := behaviorRand.Roll(3266)
  121. RandDrainMax := behaviorRand.Roll(64) + 1
  122. RandDrainRolled := dice.Roll(RandDrainMax)
  123. DrainSize := BaseDrainSize + 16 + 38 + RandDrainRolled
  124. readSizeRemain := DrainSize
  125. drainConnection := func(e error) error {
  126. // We read a deterministic generated length of data before closing the connection to offset padding read pattern
  127. readSizeRemain -= int(buffer.Len())
  128. if readSizeRemain > 0 {
  129. err := s.DrainConnN(reader, readSizeRemain)
  130. if err != nil {
  131. return newError("failed to drain connection DrainSize = ", BaseDrainSize, " ", RandDrainMax, " ", RandDrainRolled).Base(err).Base(e)
  132. }
  133. return newError("connection drained DrainSize = ", BaseDrainSize, " ", RandDrainMax, " ", RandDrainRolled).Base(e)
  134. }
  135. return e
  136. }
  137. defer func() {
  138. buffer.Release()
  139. }()
  140. if _, err := buffer.ReadFullFrom(reader, protocol.IDBytesLen); err != nil {
  141. return nil, newError("failed to read request header").Base(err)
  142. }
  143. var decryptor io.Reader
  144. var vmessAccount *vmess.MemoryAccount
  145. user, foundAEAD, errorAEAD := s.userValidator.GetAEAD(buffer.Bytes())
  146. var fixedSizeAuthID [16]byte
  147. copy(fixedSizeAuthID[:], buffer.Bytes())
  148. switch {
  149. case foundAEAD:
  150. vmessAccount = user.Account.(*vmess.MemoryAccount)
  151. var fixedSizeCmdKey [16]byte
  152. copy(fixedSizeCmdKey[:], vmessAccount.ID.CmdKey())
  153. aeadData, shouldDrain, bytesRead, errorReason := vmessaead.OpenVMessAEADHeader(fixedSizeCmdKey, fixedSizeAuthID, reader)
  154. if errorReason != nil {
  155. if shouldDrain {
  156. readSizeRemain -= bytesRead
  157. return nil, drainConnection(newError("AEAD read failed").Base(errorReason))
  158. }
  159. return nil, drainConnection(newError("AEAD read failed, drain skipped").Base(errorReason))
  160. }
  161. decryptor = bytes.NewReader(aeadData)
  162. s.isAEADRequest = true
  163. case errorAEAD == vmessaead.ErrNotFound:
  164. userLegacy, timestamp, valid, userValidationError := s.userValidator.Get(buffer.Bytes())
  165. if !valid || userValidationError != nil {
  166. return nil, drainConnection(newError("invalid user").Base(userValidationError))
  167. }
  168. if s.isAEADForced {
  169. return nil, drainConnection(newError("invalid user: VMessAEAD is enforced and a non VMessAEAD connection is received. You can still disable this security feature with environment variable v2ray.vmess.aead.forced = false . You will not be able to enable legacy header workaround in the future."))
  170. }
  171. if s.userValidator.ShouldShowLegacyWarn() {
  172. newError("Critical Warning: potentially invalid user: a non VMessAEAD connection is received. From 2022 Jan 1st, this kind of connection will be rejected by default. You should update or replace your client software now. This message will not be shown for further violation on this inbound.").AtWarning().WriteToLog()
  173. }
  174. user = userLegacy
  175. iv := hashTimestamp(md5.New(), timestamp)
  176. vmessAccount = userLegacy.Account.(*vmess.MemoryAccount)
  177. aesStream := crypto.NewAesDecryptionStream(vmessAccount.ID.CmdKey(), iv)
  178. decryptor = crypto.NewCryptionReader(aesStream, reader)
  179. default:
  180. return nil, drainConnection(newError("invalid user").Base(errorAEAD))
  181. }
  182. readSizeRemain -= int(buffer.Len())
  183. buffer.Clear()
  184. if _, err := buffer.ReadFullFrom(decryptor, 38); err != nil {
  185. return nil, newError("failed to read request header").Base(err)
  186. }
  187. request := &protocol.RequestHeader{
  188. User: user,
  189. Version: buffer.Byte(0),
  190. }
  191. copy(s.requestBodyIV[:], buffer.BytesRange(1, 17)) // 16 bytes
  192. copy(s.requestBodyKey[:], buffer.BytesRange(17, 33)) // 16 bytes
  193. var sid sessionID
  194. copy(sid.user[:], vmessAccount.ID.Bytes())
  195. sid.key = s.requestBodyKey
  196. sid.nonce = s.requestBodyIV
  197. if !s.sessionHistory.addIfNotExits(sid) {
  198. if !s.isAEADRequest {
  199. drainErr := s.userValidator.BurnTaintFuse(fixedSizeAuthID[:])
  200. if drainErr != nil {
  201. return nil, drainConnection(newError("duplicated session id, possibly under replay attack, and failed to taint userHash").Base(drainErr))
  202. }
  203. return nil, drainConnection(newError("duplicated session id, possibly under replay attack, userHash tainted"))
  204. }
  205. return nil, newError("duplicated session id, possibly under replay attack, but this is a AEAD request")
  206. }
  207. s.responseHeader = buffer.Byte(33) // 1 byte
  208. request.Option = bitmask.Byte(buffer.Byte(34)) // 1 byte
  209. paddingLen := int(buffer.Byte(35) >> 4)
  210. request.Security = parseSecurityType(buffer.Byte(35) & 0x0F)
  211. // 1 bytes reserved
  212. request.Command = protocol.RequestCommand(buffer.Byte(37))
  213. switch request.Command {
  214. case protocol.RequestCommandMux:
  215. request.Address = net.DomainAddress("v1.mux.cool")
  216. request.Port = 0
  217. case protocol.RequestCommandTCP, protocol.RequestCommandUDP:
  218. if addr, port, err := addrParser.ReadAddressPort(buffer, decryptor); err == nil {
  219. request.Address = addr
  220. request.Port = port
  221. }
  222. }
  223. if paddingLen > 0 {
  224. if _, err := buffer.ReadFullFrom(decryptor, int32(paddingLen)); err != nil {
  225. if !s.isAEADRequest {
  226. burnErr := s.userValidator.BurnTaintFuse(fixedSizeAuthID[:])
  227. if burnErr != nil {
  228. return nil, newError("failed to read padding, failed to taint userHash").Base(burnErr).Base(err)
  229. }
  230. return nil, newError("failed to read padding, userHash tainted").Base(err)
  231. }
  232. return nil, newError("failed to read padding").Base(err)
  233. }
  234. }
  235. if _, err := buffer.ReadFullFrom(decryptor, 4); err != nil {
  236. if !s.isAEADRequest {
  237. burnErr := s.userValidator.BurnTaintFuse(fixedSizeAuthID[:])
  238. if burnErr != nil {
  239. return nil, newError("failed to read checksum, failed to taint userHash").Base(burnErr).Base(err)
  240. }
  241. return nil, newError("failed to read checksum, userHash tainted").Base(err)
  242. }
  243. return nil, newError("failed to read checksum").Base(err)
  244. }
  245. fnv1a := fnv.New32a()
  246. common.Must2(fnv1a.Write(buffer.BytesTo(-4)))
  247. actualHash := fnv1a.Sum32()
  248. expectedHash := binary.BigEndian.Uint32(buffer.BytesFrom(-4))
  249. if actualHash != expectedHash {
  250. if !s.isAEADRequest {
  251. Autherr := newError("invalid auth, legacy userHash tainted")
  252. burnErr := s.userValidator.BurnTaintFuse(fixedSizeAuthID[:])
  253. if burnErr != nil {
  254. Autherr = newError("invalid auth, can't taint legacy userHash").Base(burnErr)
  255. }
  256. // It is possible that we are under attack described in https://github.com/v2ray/v2ray-core/issues/2523
  257. return nil, drainConnection(Autherr)
  258. }
  259. return nil, newError("invalid auth, but this is a AEAD request")
  260. }
  261. if request.Address == nil {
  262. return nil, newError("invalid remote address")
  263. }
  264. if request.Security == protocol.SecurityType_UNKNOWN || request.Security == protocol.SecurityType_AUTO {
  265. return nil, newError("unknown security type: ", request.Security)
  266. }
  267. return request, nil
  268. }
  269. // DecodeRequestBody returns Reader from which caller can fetch decrypted body.
  270. func (s *ServerSession) DecodeRequestBody(request *protocol.RequestHeader, reader io.Reader) buf.Reader {
  271. var sizeParser crypto.ChunkSizeDecoder = crypto.PlainChunkSizeParser{}
  272. if request.Option.Has(protocol.RequestOptionChunkMasking) {
  273. sizeParser = NewShakeSizeParser(s.requestBodyIV[:])
  274. }
  275. var padding crypto.PaddingLengthGenerator
  276. if request.Option.Has(protocol.RequestOptionGlobalPadding) {
  277. padding = sizeParser.(crypto.PaddingLengthGenerator)
  278. }
  279. switch request.Security {
  280. case protocol.SecurityType_NONE:
  281. if request.Option.Has(protocol.RequestOptionChunkStream) {
  282. if request.Command.TransferType() == protocol.TransferTypeStream {
  283. return crypto.NewChunkStreamReader(sizeParser, reader)
  284. }
  285. auth := &crypto.AEADAuthenticator{
  286. AEAD: new(NoOpAuthenticator),
  287. NonceGenerator: crypto.GenerateEmptyBytes(),
  288. AdditionalDataGenerator: crypto.GenerateEmptyBytes(),
  289. }
  290. return crypto.NewAuthenticationReader(auth, sizeParser, reader, protocol.TransferTypePacket, padding)
  291. }
  292. return buf.NewReader(reader)
  293. case protocol.SecurityType_LEGACY:
  294. aesStream := crypto.NewAesDecryptionStream(s.requestBodyKey[:], s.requestBodyIV[:])
  295. cryptionReader := crypto.NewCryptionReader(aesStream, reader)
  296. if request.Option.Has(protocol.RequestOptionChunkStream) {
  297. auth := &crypto.AEADAuthenticator{
  298. AEAD: new(FnvAuthenticator),
  299. NonceGenerator: crypto.GenerateEmptyBytes(),
  300. AdditionalDataGenerator: crypto.GenerateEmptyBytes(),
  301. }
  302. return crypto.NewAuthenticationReader(auth, sizeParser, cryptionReader, request.Command.TransferType(), padding)
  303. }
  304. return buf.NewReader(cryptionReader)
  305. case protocol.SecurityType_AES128_GCM:
  306. aead := crypto.NewAesGcm(s.requestBodyKey[:])
  307. auth := &crypto.AEADAuthenticator{
  308. AEAD: aead,
  309. NonceGenerator: GenerateChunkNonce(s.requestBodyIV[:], uint32(aead.NonceSize())),
  310. AdditionalDataGenerator: crypto.GenerateEmptyBytes(),
  311. }
  312. if request.Option.Has(protocol.RequestOptionAuthenticatedLength) {
  313. AuthenticatedLengthKey := vmessaead.KDF16(s.requestBodyKey[:], "auth_len")
  314. AuthenticatedLengthKeyAEAD := crypto.NewAesGcm(AuthenticatedLengthKey)
  315. lengthAuth := &crypto.AEADAuthenticator{
  316. AEAD: AuthenticatedLengthKeyAEAD,
  317. NonceGenerator: GenerateChunkNonce(s.requestBodyIV[:], uint32(aead.NonceSize())),
  318. AdditionalDataGenerator: crypto.GenerateEmptyBytes(),
  319. }
  320. sizeParser = NewAEADSizeParser(lengthAuth)
  321. }
  322. return crypto.NewAuthenticationReader(auth, sizeParser, reader, request.Command.TransferType(), padding)
  323. case protocol.SecurityType_CHACHA20_POLY1305:
  324. aead, _ := chacha20poly1305.New(GenerateChacha20Poly1305Key(s.requestBodyKey[:]))
  325. auth := &crypto.AEADAuthenticator{
  326. AEAD: aead,
  327. NonceGenerator: GenerateChunkNonce(s.requestBodyIV[:], uint32(aead.NonceSize())),
  328. AdditionalDataGenerator: crypto.GenerateEmptyBytes(),
  329. }
  330. if request.Option.Has(protocol.RequestOptionAuthenticatedLength) {
  331. AuthenticatedLengthKey := vmessaead.KDF16(s.requestBodyKey[:], "auth_len")
  332. AuthenticatedLengthKeyAEAD, err := chacha20poly1305.New(GenerateChacha20Poly1305Key(AuthenticatedLengthKey))
  333. common.Must(err)
  334. lengthAuth := &crypto.AEADAuthenticator{
  335. AEAD: AuthenticatedLengthKeyAEAD,
  336. NonceGenerator: GenerateChunkNonce(s.requestBodyIV[:], uint32(aead.NonceSize())),
  337. AdditionalDataGenerator: crypto.GenerateEmptyBytes(),
  338. }
  339. sizeParser = NewAEADSizeParser(lengthAuth)
  340. }
  341. return crypto.NewAuthenticationReader(auth, sizeParser, reader, request.Command.TransferType(), padding)
  342. default:
  343. panic("Unknown security type.")
  344. }
  345. }
  346. // EncodeResponseHeader writes encoded response header into the given writer.
  347. func (s *ServerSession) EncodeResponseHeader(header *protocol.ResponseHeader, writer io.Writer) {
  348. var encryptionWriter io.Writer
  349. if !s.isAEADRequest {
  350. s.responseBodyKey = md5.Sum(s.requestBodyKey[:])
  351. s.responseBodyIV = md5.Sum(s.requestBodyIV[:])
  352. } else {
  353. BodyKey := sha256.Sum256(s.requestBodyKey[:])
  354. copy(s.responseBodyKey[:], BodyKey[:16])
  355. BodyIV := sha256.Sum256(s.requestBodyIV[:])
  356. copy(s.responseBodyIV[:], BodyIV[:16])
  357. }
  358. aesStream := crypto.NewAesEncryptionStream(s.responseBodyKey[:], s.responseBodyIV[:])
  359. encryptionWriter = crypto.NewCryptionWriter(aesStream, writer)
  360. s.responseWriter = encryptionWriter
  361. aeadEncryptedHeaderBuffer := bytes.NewBuffer(nil)
  362. if s.isAEADRequest {
  363. encryptionWriter = aeadEncryptedHeaderBuffer
  364. }
  365. common.Must2(encryptionWriter.Write([]byte{s.responseHeader, byte(header.Option)}))
  366. err := MarshalCommand(header.Command, encryptionWriter)
  367. if err != nil {
  368. common.Must2(encryptionWriter.Write([]byte{0x00, 0x00}))
  369. }
  370. if s.isAEADRequest {
  371. aeadResponseHeaderLengthEncryptionKey := vmessaead.KDF16(s.responseBodyKey[:], vmessaead.KDFSaltConstAEADRespHeaderLenKey)
  372. aeadResponseHeaderLengthEncryptionIV := vmessaead.KDF(s.responseBodyIV[:], vmessaead.KDFSaltConstAEADRespHeaderLenIV)[:12]
  373. aeadResponseHeaderLengthEncryptionKeyAESBlock := common.Must2(aes.NewCipher(aeadResponseHeaderLengthEncryptionKey)).(cipher.Block)
  374. aeadResponseHeaderLengthEncryptionAEAD := common.Must2(cipher.NewGCM(aeadResponseHeaderLengthEncryptionKeyAESBlock)).(cipher.AEAD)
  375. aeadResponseHeaderLengthEncryptionBuffer := bytes.NewBuffer(nil)
  376. decryptedResponseHeaderLengthBinaryDeserializeBuffer := uint16(aeadEncryptedHeaderBuffer.Len())
  377. common.Must(binary.Write(aeadResponseHeaderLengthEncryptionBuffer, binary.BigEndian, decryptedResponseHeaderLengthBinaryDeserializeBuffer))
  378. AEADEncryptedLength := aeadResponseHeaderLengthEncryptionAEAD.Seal(nil, aeadResponseHeaderLengthEncryptionIV, aeadResponseHeaderLengthEncryptionBuffer.Bytes(), nil)
  379. common.Must2(io.Copy(writer, bytes.NewReader(AEADEncryptedLength)))
  380. aeadResponseHeaderPayloadEncryptionKey := vmessaead.KDF16(s.responseBodyKey[:], vmessaead.KDFSaltConstAEADRespHeaderPayloadKey)
  381. aeadResponseHeaderPayloadEncryptionIV := vmessaead.KDF(s.responseBodyIV[:], vmessaead.KDFSaltConstAEADRespHeaderPayloadIV)[:12]
  382. aeadResponseHeaderPayloadEncryptionKeyAESBlock := common.Must2(aes.NewCipher(aeadResponseHeaderPayloadEncryptionKey)).(cipher.Block)
  383. aeadResponseHeaderPayloadEncryptionAEAD := common.Must2(cipher.NewGCM(aeadResponseHeaderPayloadEncryptionKeyAESBlock)).(cipher.AEAD)
  384. aeadEncryptedHeaderPayload := aeadResponseHeaderPayloadEncryptionAEAD.Seal(nil, aeadResponseHeaderPayloadEncryptionIV, aeadEncryptedHeaderBuffer.Bytes(), nil)
  385. common.Must2(io.Copy(writer, bytes.NewReader(aeadEncryptedHeaderPayload)))
  386. }
  387. }
  388. // EncodeResponseBody returns a Writer that auto-encrypt content written by caller.
  389. func (s *ServerSession) EncodeResponseBody(request *protocol.RequestHeader, writer io.Writer) buf.Writer {
  390. var sizeParser crypto.ChunkSizeEncoder = crypto.PlainChunkSizeParser{}
  391. if request.Option.Has(protocol.RequestOptionChunkMasking) {
  392. sizeParser = NewShakeSizeParser(s.responseBodyIV[:])
  393. }
  394. var padding crypto.PaddingLengthGenerator
  395. if request.Option.Has(protocol.RequestOptionGlobalPadding) {
  396. padding = sizeParser.(crypto.PaddingLengthGenerator)
  397. }
  398. switch request.Security {
  399. case protocol.SecurityType_NONE:
  400. if request.Option.Has(protocol.RequestOptionChunkStream) {
  401. if request.Command.TransferType() == protocol.TransferTypeStream {
  402. return crypto.NewChunkStreamWriter(sizeParser, writer)
  403. }
  404. auth := &crypto.AEADAuthenticator{
  405. AEAD: new(NoOpAuthenticator),
  406. NonceGenerator: crypto.GenerateEmptyBytes(),
  407. AdditionalDataGenerator: crypto.GenerateEmptyBytes(),
  408. }
  409. return crypto.NewAuthenticationWriter(auth, sizeParser, writer, protocol.TransferTypePacket, padding)
  410. }
  411. return buf.NewWriter(writer)
  412. case protocol.SecurityType_LEGACY:
  413. if request.Option.Has(protocol.RequestOptionChunkStream) {
  414. auth := &crypto.AEADAuthenticator{
  415. AEAD: new(FnvAuthenticator),
  416. NonceGenerator: crypto.GenerateEmptyBytes(),
  417. AdditionalDataGenerator: crypto.GenerateEmptyBytes(),
  418. }
  419. return crypto.NewAuthenticationWriter(auth, sizeParser, s.responseWriter, request.Command.TransferType(), padding)
  420. }
  421. return &buf.SequentialWriter{Writer: s.responseWriter}
  422. case protocol.SecurityType_AES128_GCM:
  423. aead := crypto.NewAesGcm(s.responseBodyKey[:])
  424. auth := &crypto.AEADAuthenticator{
  425. AEAD: aead,
  426. NonceGenerator: GenerateChunkNonce(s.responseBodyIV[:], uint32(aead.NonceSize())),
  427. AdditionalDataGenerator: crypto.GenerateEmptyBytes(),
  428. }
  429. if request.Option.Has(protocol.RequestOptionAuthenticatedLength) {
  430. AuthenticatedLengthKey := vmessaead.KDF16(s.requestBodyKey[:], "auth_len")
  431. AuthenticatedLengthKeyAEAD := crypto.NewAesGcm(AuthenticatedLengthKey)
  432. lengthAuth := &crypto.AEADAuthenticator{
  433. AEAD: AuthenticatedLengthKeyAEAD,
  434. NonceGenerator: GenerateChunkNonce(s.requestBodyIV[:], uint32(aead.NonceSize())),
  435. AdditionalDataGenerator: crypto.GenerateEmptyBytes(),
  436. }
  437. sizeParser = NewAEADSizeParser(lengthAuth)
  438. }
  439. return crypto.NewAuthenticationWriter(auth, sizeParser, writer, request.Command.TransferType(), padding)
  440. case protocol.SecurityType_CHACHA20_POLY1305:
  441. aead, _ := chacha20poly1305.New(GenerateChacha20Poly1305Key(s.responseBodyKey[:]))
  442. auth := &crypto.AEADAuthenticator{
  443. AEAD: aead,
  444. NonceGenerator: GenerateChunkNonce(s.responseBodyIV[:], uint32(aead.NonceSize())),
  445. AdditionalDataGenerator: crypto.GenerateEmptyBytes(),
  446. }
  447. if request.Option.Has(protocol.RequestOptionAuthenticatedLength) {
  448. AuthenticatedLengthKey := vmessaead.KDF16(s.requestBodyKey[:], "auth_len")
  449. AuthenticatedLengthKeyAEAD, err := chacha20poly1305.New(GenerateChacha20Poly1305Key(AuthenticatedLengthKey))
  450. common.Must(err)
  451. lengthAuth := &crypto.AEADAuthenticator{
  452. AEAD: AuthenticatedLengthKeyAEAD,
  453. NonceGenerator: GenerateChunkNonce(s.requestBodyIV[:], uint32(aead.NonceSize())),
  454. AdditionalDataGenerator: crypto.GenerateEmptyBytes(),
  455. }
  456. sizeParser = NewAEADSizeParser(lengthAuth)
  457. }
  458. return crypto.NewAuthenticationWriter(auth, sizeParser, writer, request.Command.TransferType(), padding)
  459. default:
  460. panic("Unknown security type.")
  461. }
  462. }
  463. func (s *ServerSession) DrainConnN(reader io.Reader, n int) error {
  464. _, err := io.CopyN(ioutil.Discard, reader, int64(n))
  465. return err
  466. }