package sniff import ( "bytes" "context" "io" "net" "time" "github.com/sagernet/sing-box/adapter" C "github.com/sagernet/sing-box/constant" "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 ) 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, buffer *buf.Buffer, timeout time.Duration, sniffers ...StreamSniffer) error { if timeout == 0 { timeout = C.ReadPayloadTimeout } deadline := time.Now().Add(timeout) var errors []error err := conn.SetReadDeadline(deadline) if err != nil { return E.Cause(err, "set read deadline") } defer conn.SetReadDeadline(time.Time{}) for _, sniffer := range sniffers { if buffer.IsEmpty() { err = sniffer(ctx, metadata, io.TeeReader(conn, buffer)) } else { err = sniffer(ctx, metadata, io.MultiReader(bytes.NewReader(buffer.Bytes()), io.TeeReader(conn, buffer))) } if err == nil { return nil } errors = append(errors, err) } return E.Errors(errors...) } func PeekPacket(ctx context.Context, metadata *adapter.InboundContext, packet []byte, sniffers ...PacketSniffer) error { var errors []error for _, sniffer := range sniffers { err := sniffer(ctx, metadata, packet) if err == nil { return nil } errors = append(errors, err) } return E.Errors(errors...) }