io.Copy timeout when keepalive is disabled
Feb 13, 2024 pm 09:36 PM#php editor Yuzai will introduce to you today a problem about "io.Copy timeout when keepalive is disabled". When using the io.Copy function in the Go language for data transmission, if keepalive is disabled, transmission timeout may occur. This problem may cause some trouble to developers, so we need to know some solutions to avoid this problem from happening. Let’s take a look at how to solve this problem!
Question content
When I disable linux keepalive
sudo sysctl -w net.ipv4.tcp_keepalive_probes=0
And run the following code
package main import ( "fmt" "io" "net" "os" "sync" "time" ) func main() { go func() { listen, err := net.Listen("tcp", "127.0.0.1:9390") if err != nil { fmt.Printf("net listen fail, reason: [%s]\n", err.Error()) os.Exit(1) } defer listen.Close() for { conn, err := listen.Accept() if err != nil { fmt.Printf("net accept fail, reason: [%s]\n", err.Error()) continue } _, err = io.Copy(conn, conn) if err != nil { fmt.Printf("net Copy fail, reason: [%s]\n", err.Error()) continue } conn.Close() } }() var wg sync.WaitGroup wg.Add(1) go func() { defer wg.Done() for { time.Sleep(1 * time.Second) tcpAddr, _ := net.ResolveTCPAddr("tcp", "127.0.0.1:9390") fmt.Println("tcp client resolve success: ", tcpAddr.String()) tcpConn, err := net.DialTCP("tcp", nil, tcpAddr) if err != nil { fmt.Println("tcp connect fail: ", err.Error()) return } time.Sleep(100 * time.Second) tcpConn.Close() fmt.Println("tcp end") } }() wg.Wait() }
io.Copy will return "splice: connection timeout"
If I enable keepalive
sudo sysctl -w net.ipv4.tcp_keepalive_probes=3
io.Copy is enough
I tried the tcp client to send packets at 1 second intervals and disable tcp_keepalive, that worked too.
I wrote another code to replace io.Copy
buf := make([]byte, 10) for { conn.SetReadDeadline(time.Now().Add(30 * time.Second)) n, err := conn.Read(buf) if err != nil { fmt.Println("read fail: ", err.Error()) break } _, err = conn.Write(buf[:n]) if err != nil { fmt.Println("write fail: ", err.Error()) break } } conn.Close()
Read timeout after 15 seconds, not working for 30 seconds
Why is this so strange?
Solution
Answer: When "kernel tcp stack" uses keepalive to detect the network, when "kernel tcp stack" needs to send keepalive, net.ipv4.tcp_keepalive_probes = 0 will time out
The above is the detailed content of io.Copy timeout when keepalive is disabled. For more information, please follow other related articles on the PHP Chinese website!

Hot Article

Hot tools Tags

Hot Article

Hot Article Tags

Notepad++7.3.1
Easy-to-use and free code editor

SublimeText3 Chinese version
Chinese version, very easy to use

Zend Studio 13.0.1
Powerful PHP integrated development environment

Dreamweaver CS6
Visual web development tools

SublimeText3 Mac version
God-level code editing software (SublimeText3)

Hot Topics

How to use reflection to access private fields and methods in golang

Tips for dynamically creating new functions in golang functions

The difference between performance testing and unit testing in Go language

What pitfalls should we pay attention to when designing distributed systems with Golang technology?

Golang technology libraries and tools used in machine learning

The evolution of golang function naming convention

The role of Golang technology in mobile IoT development

Can golang variable parameters be used for function return values?
