| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869 |
- package faketls
-
- import (
- "bytes"
- "fmt"
- "math/rand"
- "net"
-
- "github.com/9seconds/mtg/v2/mtglib/internal/faketls/record"
- )
-
- type Conn struct {
- net.Conn
-
- readBuffer bytes.Buffer
- }
-
- func (c *Conn) Read(p []byte) (int, error) {
- if n, _ := c.readBuffer.Read(p); n > 0 {
- return n, nil
- }
-
- rec := record.AcquireRecord()
- defer record.ReleaseRecord(rec)
-
- for {
- if err := rec.Read(c.Conn); err != nil {
- return 0, err // nolint: wrapcheck
- }
-
- switch rec.Type { // nolint: exhaustive
- case record.TypeApplicationData:
- rec.Payload.WriteTo(&c.readBuffer) // nolint: errcheck
-
- return c.readBuffer.Read(p) // nolint: wrapcheck
- case record.TypeChangeCipherSpec:
- default:
- return 0, fmt.Errorf("unsupported record type %v", rec.Type)
- }
- }
- }
-
- func (c *Conn) Write(p []byte) (int, error) {
- rec := record.AcquireRecord()
- defer record.ReleaseRecord(rec)
-
- rec.Type = record.TypeApplicationData
- rec.Version = record.Version12
- written := 0
-
- for len(p) > 0 {
- chunkSize := rand.Intn(record.TLSMaxRecordSize)
- if chunkSize > len(p) || chunkSize == 0 {
- chunkSize = len(p)
- }
-
- rec.Payload.Reset()
- rec.Payload.Write(p[:chunkSize])
-
- if err := rec.Dump(c.Conn); err != nil {
- return written, err // nolint: wrapcheck
- }
-
- written += chunkSize
- p = p[chunkSize:]
- }
-
- return written, nil
- }
|