mirror of
https://github.com/apernet/OpenGFW.git
synced 2024-12-23 01:19:21 +08:00
Merge pull request #124 from apernet/wip-tcp-flush
feat: TCP timeout flush
This commit is contained in:
commit
5f447d4e31
58
cmd/root.go
58
cmd/root.go
@ -7,6 +7,7 @@ import (
|
|||||||
"os/signal"
|
"os/signal"
|
||||||
"strings"
|
"strings"
|
||||||
"syscall"
|
"syscall"
|
||||||
|
"time"
|
||||||
|
|
||||||
"github.com/apernet/OpenGFW/analyzer"
|
"github.com/apernet/OpenGFW/analyzer"
|
||||||
"github.com/apernet/OpenGFW/analyzer/tcp"
|
"github.com/apernet/OpenGFW/analyzer/tcp"
|
||||||
@ -176,11 +177,12 @@ type cliConfigIO struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
type cliConfigWorkers struct {
|
type cliConfigWorkers struct {
|
||||||
Count int `mapstructure:"count"`
|
Count int `mapstructure:"count"`
|
||||||
QueueSize int `mapstructure:"queueSize"`
|
QueueSize int `mapstructure:"queueSize"`
|
||||||
TCPMaxBufferedPagesTotal int `mapstructure:"tcpMaxBufferedPagesTotal"`
|
TCPMaxBufferedPagesTotal int `mapstructure:"tcpMaxBufferedPagesTotal"`
|
||||||
TCPMaxBufferedPagesPerConn int `mapstructure:"tcpMaxBufferedPagesPerConn"`
|
TCPMaxBufferedPagesPerConn int `mapstructure:"tcpMaxBufferedPagesPerConn"`
|
||||||
UDPMaxStreams int `mapstructure:"udpMaxStreams"`
|
TCPTimeout time.Duration `mapstructure:"tcpTimeout"`
|
||||||
|
UDPMaxStreams int `mapstructure:"udpMaxStreams"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type cliConfigRuleset struct {
|
type cliConfigRuleset struct {
|
||||||
@ -213,6 +215,7 @@ func (c *cliConfig) fillWorkers(config *engine.Config) error {
|
|||||||
config.WorkerQueueSize = c.Workers.QueueSize
|
config.WorkerQueueSize = c.Workers.QueueSize
|
||||||
config.WorkerTCPMaxBufferedPagesTotal = c.Workers.TCPMaxBufferedPagesTotal
|
config.WorkerTCPMaxBufferedPagesTotal = c.Workers.TCPMaxBufferedPagesTotal
|
||||||
config.WorkerTCPMaxBufferedPagesPerConn = c.Workers.TCPMaxBufferedPagesPerConn
|
config.WorkerTCPMaxBufferedPagesPerConn = c.Workers.TCPMaxBufferedPagesPerConn
|
||||||
|
config.WorkerTCPTimeout = c.Workers.TCPTimeout
|
||||||
config.WorkerUDPMaxStreams = c.Workers.UDPMaxStreams
|
config.WorkerUDPMaxStreams = c.Workers.UDPMaxStreams
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
@ -340,12 +343,26 @@ func (l *engineLogger) TCPStreamPropUpdate(info ruleset.StreamInfo, close bool)
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (l *engineLogger) TCPStreamAction(info ruleset.StreamInfo, action ruleset.Action, noMatch bool) {
|
func (l *engineLogger) TCPStreamAction(info ruleset.StreamInfo, action ruleset.Action, noMatch bool) {
|
||||||
logger.Info("TCP stream action",
|
if noMatch {
|
||||||
zap.Int64("id", info.ID),
|
logger.Debug("TCP stream no match",
|
||||||
zap.String("src", info.SrcString()),
|
zap.Int64("id", info.ID),
|
||||||
zap.String("dst", info.DstString()),
|
zap.String("src", info.SrcString()),
|
||||||
zap.String("action", action.String()),
|
zap.String("dst", info.DstString()),
|
||||||
zap.Bool("noMatch", noMatch))
|
zap.String("action", action.String()))
|
||||||
|
} else {
|
||||||
|
logger.Info("TCP stream action",
|
||||||
|
zap.Int64("id", info.ID),
|
||||||
|
zap.String("src", info.SrcString()),
|
||||||
|
zap.String("dst", info.DstString()),
|
||||||
|
zap.String("action", action.String()))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *engineLogger) TCPFlush(workerID, flushed, closed int) {
|
||||||
|
logger.Debug("TCP flush",
|
||||||
|
zap.Int("workerID", workerID),
|
||||||
|
zap.Int("flushed", flushed),
|
||||||
|
zap.Int("closed", closed))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (l *engineLogger) UDPStreamNew(workerID int, info ruleset.StreamInfo) {
|
func (l *engineLogger) UDPStreamNew(workerID int, info ruleset.StreamInfo) {
|
||||||
@ -366,12 +383,19 @@ func (l *engineLogger) UDPStreamPropUpdate(info ruleset.StreamInfo, close bool)
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (l *engineLogger) UDPStreamAction(info ruleset.StreamInfo, action ruleset.Action, noMatch bool) {
|
func (l *engineLogger) UDPStreamAction(info ruleset.StreamInfo, action ruleset.Action, noMatch bool) {
|
||||||
logger.Info("UDP stream action",
|
if noMatch {
|
||||||
zap.Int64("id", info.ID),
|
logger.Debug("UDP stream no match",
|
||||||
zap.String("src", info.SrcString()),
|
zap.Int64("id", info.ID),
|
||||||
zap.String("dst", info.DstString()),
|
zap.String("src", info.SrcString()),
|
||||||
zap.String("action", action.String()),
|
zap.String("dst", info.DstString()),
|
||||||
zap.Bool("noMatch", noMatch))
|
zap.String("action", action.String()))
|
||||||
|
} else {
|
||||||
|
logger.Info("UDP stream action",
|
||||||
|
zap.Int64("id", info.ID),
|
||||||
|
zap.String("src", info.SrcString()),
|
||||||
|
zap.String("dst", info.DstString()),
|
||||||
|
zap.String("action", action.String()))
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (l *engineLogger) ModifyError(info ruleset.StreamInfo, err error) {
|
func (l *engineLogger) ModifyError(info ruleset.StreamInfo, err error) {
|
||||||
|
@ -34,6 +34,7 @@ func NewEngine(config Config) (Engine, error) {
|
|||||||
Ruleset: config.Ruleset,
|
Ruleset: config.Ruleset,
|
||||||
TCPMaxBufferedPagesTotal: config.WorkerTCPMaxBufferedPagesTotal,
|
TCPMaxBufferedPagesTotal: config.WorkerTCPMaxBufferedPagesTotal,
|
||||||
TCPMaxBufferedPagesPerConn: config.WorkerTCPMaxBufferedPagesPerConn,
|
TCPMaxBufferedPagesPerConn: config.WorkerTCPMaxBufferedPagesPerConn,
|
||||||
|
TCPTimeout: config.WorkerTCPTimeout,
|
||||||
UDPMaxStreams: config.WorkerUDPMaxStreams,
|
UDPMaxStreams: config.WorkerUDPMaxStreams,
|
||||||
})
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -2,6 +2,7 @@ package engine
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
"time"
|
||||||
|
|
||||||
"github.com/apernet/OpenGFW/io"
|
"github.com/apernet/OpenGFW/io"
|
||||||
"github.com/apernet/OpenGFW/ruleset"
|
"github.com/apernet/OpenGFW/ruleset"
|
||||||
@ -25,6 +26,7 @@ type Config struct {
|
|||||||
WorkerQueueSize int
|
WorkerQueueSize int
|
||||||
WorkerTCPMaxBufferedPagesTotal int
|
WorkerTCPMaxBufferedPagesTotal int
|
||||||
WorkerTCPMaxBufferedPagesPerConn int
|
WorkerTCPMaxBufferedPagesPerConn int
|
||||||
|
WorkerTCPTimeout time.Duration
|
||||||
WorkerUDPMaxStreams int
|
WorkerUDPMaxStreams int
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -36,6 +38,7 @@ type Logger interface {
|
|||||||
TCPStreamNew(workerID int, info ruleset.StreamInfo)
|
TCPStreamNew(workerID int, info ruleset.StreamInfo)
|
||||||
TCPStreamPropUpdate(info ruleset.StreamInfo, close bool)
|
TCPStreamPropUpdate(info ruleset.StreamInfo, close bool)
|
||||||
TCPStreamAction(info ruleset.StreamInfo, action ruleset.Action, noMatch bool)
|
TCPStreamAction(info ruleset.StreamInfo, action ruleset.Action, noMatch bool)
|
||||||
|
TCPFlush(workerID, flushed, closed int)
|
||||||
|
|
||||||
UDPStreamNew(workerID int, info ruleset.StreamInfo)
|
UDPStreamNew(workerID int, info ruleset.StreamInfo)
|
||||||
UDPStreamPropUpdate(info ruleset.StreamInfo, close bool)
|
UDPStreamPropUpdate(info ruleset.StreamInfo, close bool)
|
||||||
|
@ -2,6 +2,7 @@ package engine
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
"time"
|
||||||
|
|
||||||
"github.com/apernet/OpenGFW/io"
|
"github.com/apernet/OpenGFW/io"
|
||||||
"github.com/apernet/OpenGFW/ruleset"
|
"github.com/apernet/OpenGFW/ruleset"
|
||||||
@ -14,9 +15,12 @@ import (
|
|||||||
|
|
||||||
const (
|
const (
|
||||||
defaultChanSize = 64
|
defaultChanSize = 64
|
||||||
defaultTCPMaxBufferedPagesTotal = 4096
|
defaultTCPMaxBufferedPagesTotal = 65536
|
||||||
defaultTCPMaxBufferedPagesPerConnection = 64
|
defaultTCPMaxBufferedPagesPerConnection = 16
|
||||||
|
defaultTCPTimeout = 10 * time.Minute
|
||||||
defaultUDPMaxStreams = 4096
|
defaultUDPMaxStreams = 4096
|
||||||
|
|
||||||
|
tcpFlushInterval = 1 * time.Minute
|
||||||
)
|
)
|
||||||
|
|
||||||
type workerPacket struct {
|
type workerPacket struct {
|
||||||
@ -33,6 +37,7 @@ type worker struct {
|
|||||||
tcpStreamFactory *tcpStreamFactory
|
tcpStreamFactory *tcpStreamFactory
|
||||||
tcpStreamPool *reassembly.StreamPool
|
tcpStreamPool *reassembly.StreamPool
|
||||||
tcpAssembler *reassembly.Assembler
|
tcpAssembler *reassembly.Assembler
|
||||||
|
tcpTimeout time.Duration
|
||||||
|
|
||||||
udpStreamFactory *udpStreamFactory
|
udpStreamFactory *udpStreamFactory
|
||||||
udpStreamManager *udpStreamManager
|
udpStreamManager *udpStreamManager
|
||||||
@ -47,6 +52,7 @@ type workerConfig struct {
|
|||||||
Ruleset ruleset.Ruleset
|
Ruleset ruleset.Ruleset
|
||||||
TCPMaxBufferedPagesTotal int
|
TCPMaxBufferedPagesTotal int
|
||||||
TCPMaxBufferedPagesPerConn int
|
TCPMaxBufferedPagesPerConn int
|
||||||
|
TCPTimeout time.Duration
|
||||||
UDPMaxStreams int
|
UDPMaxStreams int
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -60,6 +66,9 @@ func (c *workerConfig) fillDefaults() {
|
|||||||
if c.TCPMaxBufferedPagesPerConn <= 0 {
|
if c.TCPMaxBufferedPagesPerConn <= 0 {
|
||||||
c.TCPMaxBufferedPagesPerConn = defaultTCPMaxBufferedPagesPerConnection
|
c.TCPMaxBufferedPagesPerConn = defaultTCPMaxBufferedPagesPerConnection
|
||||||
}
|
}
|
||||||
|
if c.TCPTimeout <= 0 {
|
||||||
|
c.TCPTimeout = defaultTCPTimeout
|
||||||
|
}
|
||||||
if c.UDPMaxStreams <= 0 {
|
if c.UDPMaxStreams <= 0 {
|
||||||
c.UDPMaxStreams = defaultUDPMaxStreams
|
c.UDPMaxStreams = defaultUDPMaxStreams
|
||||||
}
|
}
|
||||||
@ -98,6 +107,7 @@ func newWorker(config workerConfig) (*worker, error) {
|
|||||||
tcpStreamFactory: tcpSF,
|
tcpStreamFactory: tcpSF,
|
||||||
tcpStreamPool: tcpStreamPool,
|
tcpStreamPool: tcpStreamPool,
|
||||||
tcpAssembler: tcpAssembler,
|
tcpAssembler: tcpAssembler,
|
||||||
|
tcpTimeout: config.TCPTimeout,
|
||||||
udpStreamFactory: udpSF,
|
udpStreamFactory: udpSF,
|
||||||
udpStreamManager: udpSM,
|
udpStreamManager: udpSM,
|
||||||
modSerializeBuffer: gopacket.NewSerializeBuffer(),
|
modSerializeBuffer: gopacket.NewSerializeBuffer(),
|
||||||
@ -111,6 +121,10 @@ func (w *worker) Feed(p *workerPacket) {
|
|||||||
func (w *worker) Run(ctx context.Context) {
|
func (w *worker) Run(ctx context.Context) {
|
||||||
w.logger.WorkerStart(w.id)
|
w.logger.WorkerStart(w.id)
|
||||||
defer w.logger.WorkerStop(w.id)
|
defer w.logger.WorkerStop(w.id)
|
||||||
|
|
||||||
|
tcpFlushTicker := time.NewTicker(tcpFlushInterval)
|
||||||
|
defer tcpFlushTicker.Stop()
|
||||||
|
|
||||||
for {
|
for {
|
||||||
select {
|
select {
|
||||||
case <-ctx.Done():
|
case <-ctx.Done():
|
||||||
@ -122,6 +136,8 @@ func (w *worker) Run(ctx context.Context) {
|
|||||||
}
|
}
|
||||||
v, b := w.handle(wPkt.StreamID, wPkt.Packet)
|
v, b := w.handle(wPkt.StreamID, wPkt.Packet)
|
||||||
_ = wPkt.SetVerdict(v, b)
|
_ = wPkt.SetVerdict(v, b)
|
||||||
|
case <-tcpFlushTicker.C:
|
||||||
|
w.flushTCP(w.tcpTimeout)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -176,6 +192,11 @@ func (w *worker) handleTCP(ipFlow gopacket.Flow, pMeta *gopacket.PacketMetadata,
|
|||||||
return io.Verdict(ctx.Verdict)
|
return io.Verdict(ctx.Verdict)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (w *worker) flushTCP(timeout time.Duration) {
|
||||||
|
flushed, closed := w.tcpAssembler.FlushCloseOlderThan(time.Now().Add(-timeout))
|
||||||
|
w.logger.TCPFlush(w.id, flushed, closed)
|
||||||
|
}
|
||||||
|
|
||||||
func (w *worker) handleUDP(streamID uint32, ipFlow gopacket.Flow, udp *layers.UDP) (io.Verdict, []byte) {
|
func (w *worker) handleUDP(streamID uint32, ipFlow gopacket.Flow, udp *layers.UDP) (io.Verdict, []byte) {
|
||||||
ctx := &udpContext{
|
ctx := &udpContext{
|
||||||
Verdict: udpVerdictAccept,
|
Verdict: udpVerdictAccept,
|
||||||
|
Loading…
Reference in New Issue
Block a user