1
0
mirror of https://github.com/v2fly/v2ray-core.git synced 2025-02-20 23:47:21 -05:00

130 lines
2.7 KiB
Go
Raw Normal View History

package shadowsocks
import (
2016-05-24 22:09:22 +02:00
"bytes"
"crypto/hmac"
"crypto/sha1"
2016-01-29 20:54:06 +01:00
"io"
2016-12-08 11:21:24 +01:00
2017-11-29 22:57:18 +01:00
"v2ray.com/core/common"
2016-12-09 11:35:27 +01:00
"v2ray.com/core/common/buf"
2016-08-20 20:55:45 +02:00
"v2ray.com/core/common/serial"
)
const (
2016-12-21 23:07:35 +01:00
// AuthSize is the number of extra bytes for Shadowsocks OTA.
AuthSize = 10
)
type KeyGenerator func() []byte
type Authenticator struct {
key KeyGenerator
}
func NewAuthenticator(keygen KeyGenerator) *Authenticator {
return &Authenticator{
key: keygen,
}
}
2016-12-09 12:08:25 +01:00
func (v *Authenticator) Authenticate(data []byte) buf.Supplier {
2016-11-27 21:39:09 +01:00
hasher := hmac.New(sha1.New, v.key())
2017-11-29 22:57:18 +01:00
common.Must2(hasher.Write(data))
res := hasher.Sum(nil)
2016-12-09 12:08:25 +01:00
return func(b []byte) (int, error) {
return copy(b, res[:AuthSize]), nil
2016-12-06 11:03:42 +01:00
}
}
func HeaderKeyGenerator(key []byte, iv []byte) func() []byte {
return func() []byte {
newKey := make([]byte, 0, len(key)+len(iv))
newKey = append(newKey, iv...)
2016-02-28 21:00:53 +01:00
newKey = append(newKey, key...)
return newKey
}
}
func ChunkKeyGenerator(iv []byte) func() []byte {
2016-12-21 23:07:35 +01:00
chunkID := 0
return func() []byte {
newKey := make([]byte, 0, len(iv)+4)
newKey = append(newKey, iv...)
2016-12-21 23:07:35 +01:00
newKey = serial.IntToBytes(chunkID, newKey)
chunkID++
return newKey
}
}
2016-01-29 20:54:06 +01:00
type ChunkReader struct {
reader io.Reader
auth *Authenticator
}
func NewChunkReader(reader io.Reader, auth *Authenticator) *ChunkReader {
return &ChunkReader{
reader: reader,
auth: auth,
}
}
2017-11-09 22:33:15 +01:00
func (v *ChunkReader) ReadMultiBuffer() (buf.MultiBuffer, error) {
2018-03-12 22:10:13 +01:00
size, err := serial.ReadUint16(v.reader)
if err != nil {
return nil, newError("failed to read size")
2016-11-19 10:57:00 +01:00
}
2018-03-12 22:10:13 +01:00
size += AuthSize
2016-12-05 17:05:47 +01:00
2018-04-02 22:01:55 +02:00
buffer := buf.NewSize(int32(size))
2018-04-02 00:44:47 +02:00
if err := buffer.AppendSupplier(buf.ReadFullFrom(v.reader, int32(size))); err != nil {
2016-02-01 12:22:29 +01:00
buffer.Release()
2016-01-29 20:54:06 +01:00
return nil, err
}
2016-12-05 15:19:14 +01:00
authBytes := buffer.BytesTo(AuthSize)
payload := buffer.BytesFrom(AuthSize)
2016-01-29 20:54:06 +01:00
2016-12-06 11:03:42 +01:00
actualAuthBytes := make([]byte, AuthSize)
v.auth.Authenticate(payload)(actualAuthBytes)
2016-05-24 22:09:22 +02:00
if !bytes.Equal(authBytes, actualAuthBytes) {
2016-02-01 12:22:29 +01:00
buffer.Release()
2017-04-09 01:43:25 +02:00
return nil, newError("invalid auth")
2016-01-29 20:54:06 +01:00
}
2016-07-13 11:38:14 +02:00
buffer.SliceFrom(AuthSize)
2016-01-29 20:54:06 +01:00
2017-04-16 13:17:35 +02:00
return buf.NewMultiBufferValue(buffer), nil
2016-01-29 20:54:06 +01:00
}
2016-10-21 00:33:23 +02:00
type ChunkWriter struct {
writer io.Writer
auth *Authenticator
2016-12-08 11:21:24 +01:00
buffer []byte
2016-10-21 00:33:23 +02:00
}
func NewChunkWriter(writer io.Writer, auth *Authenticator) *ChunkWriter {
return &ChunkWriter{
writer: writer,
auth: auth,
2016-12-08 11:21:24 +01:00
buffer: make([]byte, 32*1024),
2016-10-21 00:33:23 +02:00
}
}
2017-11-09 22:33:15 +01:00
// WriteMultiBuffer implements buf.Writer.
func (w *ChunkWriter) WriteMultiBuffer(mb buf.MultiBuffer) error {
2017-04-15 21:07:23 +02:00
defer mb.Release()
2017-04-15 21:49:18 +02:00
for {
2017-04-19 21:27:21 +02:00
payloadLen, _ := mb.Read(w.buffer[2+AuthSize:])
2017-04-15 21:49:18 +02:00
serial.Uint16ToBytes(uint16(payloadLen), w.buffer[:0])
w.auth.Authenticate(w.buffer[2+AuthSize : 2+AuthSize+payloadLen])(w.buffer[2:])
if _, err := w.writer.Write(w.buffer[:2+AuthSize+payloadLen]); err != nil {
return err
}
if mb.IsEmpty() {
break
}
2017-04-15 21:07:23 +02:00
}
return nil
}