12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788 |
- package sniff
- import (
- "bytes"
- "context"
- "errors"
- "io"
- "net"
- "time"
- "github.com/sagernet/sing-box/adapter"
- C "github.com/sagernet/sing-box/constant"
- "github.com/sagernet/sing/common"
- "github.com/sagernet/sing/common/buf"
- E "github.com/sagernet/sing/common/exceptions"
- )
- type (
- StreamSniffer = func(ctx context.Context, metadata *adapter.InboundContext, reader io.Reader) error
- PacketSniffer = func(ctx context.Context, metadata *adapter.InboundContext, packet []byte) error
- )
- var ErrNeedMoreData = E.New("need more data")
- func Skip(metadata *adapter.InboundContext) bool {
- // skip server first protocols
- switch metadata.Destination.Port {
- case 25, 465, 587:
- // SMTP
- return true
- case 143, 993:
- // IMAP
- return true
- case 110, 995:
- // POP3
- return true
- }
- return false
- }
- func PeekStream(ctx context.Context, metadata *adapter.InboundContext, conn net.Conn, buffers []*buf.Buffer, buffer *buf.Buffer, timeout time.Duration, sniffers ...StreamSniffer) error {
- if timeout == 0 {
- timeout = C.ReadPayloadTimeout
- }
- deadline := time.Now().Add(timeout)
- var sniffError error
- for i := 0; ; i++ {
- err := conn.SetReadDeadline(deadline)
- if err != nil {
- return E.Cause(err, "set read deadline")
- }
- _, err = buffer.ReadOnceFrom(conn)
- _ = conn.SetReadDeadline(time.Time{})
- if err != nil {
- if i > 0 {
- break
- }
- return E.Cause(err, "read payload")
- }
- sniffError = nil
- for _, sniffer := range sniffers {
- reader := io.MultiReader(common.Map(append(buffers, buffer), func(it *buf.Buffer) io.Reader {
- return bytes.NewReader(it.Bytes())
- })...)
- err = sniffer(ctx, metadata, reader)
- if err == nil {
- return nil
- }
- sniffError = E.Errors(sniffError, err)
- }
- if !errors.Is(sniffError, ErrNeedMoreData) {
- break
- }
- }
- return sniffError
- }
- func PeekPacket(ctx context.Context, metadata *adapter.InboundContext, packet []byte, sniffers ...PacketSniffer) error {
- var sniffError []error
- for _, sniffer := range sniffers {
- err := sniffer(ctx, metadata, packet)
- if err == nil {
- return nil
- }
- sniffError = append(sniffError, err)
- }
- return E.Errors(sniffError...)
- }
|