net & http & timeout === <!-- .slide: data-background-color="pink" --> <!-- .slide: data-transition="zoom" --> > [name=郭學聰 Hsueh-Tsung Kuo] > [time=Sun, 02 Aug 2020] [color=red] ###### CC BY-SA 4.0 --- <!-- .slide: data-transition="convex" --> ## Who am I? ![fieliapm](https://www.gravatar.com/avatar/2aef78f04240a6ac9ccd473ba1cbd1e3?size=2048 =384x384) <small>Someone (who?) said: a game programmer should be able to draw cute anime character(?)</small> ---- <!-- .slide: data-transition="convex" --> * A programmer from game company in Taiwan * Backend (and temporary frontend) engineer, focus on common service * Usually develop something related to my work in Python, Ruby, ECMAScript, Golang, C# * ECMAScript hater since **Netscape** is dead ---- <!-- .slide: data-transition="convex" --> * Built CDN-aware game asset update system * Built almost entire VOEZ game server by myself only * Supported Sdorica backend development * Verify and tweak Soul of Eden backend performance --- <!-- .slide: data-transition="convex" --> ## Outline ---- <!-- .slide: data-transition="convex" --> 4. net/http 5. net/http Timeout * Reason * Spec * Server * Client 6. Streaming & Timeout 7. Implementation 8. Timer Issues 9. Conclusion 10. Reference 11. Q&A --- <!-- .slide: data-transition="convex" --> ## net/http ---- <!-- .slide: data-transition="convex" --> ### Information https://golang.org/pkg/net/http/ ---- <!-- .slide: data-transition="convex" --> ### Feature * HTTP * Client * Server * Hijack * WebSocket --- <!-- .slide: data-transition="convex" --> ## net/http Timeout ---- <!-- .slide: data-transition="convex" --> ### Reason Leak File Descriptors ``` http: Accept error: accept tcp [::]:80: accept4: too many open files; retrying in 5ms ``` ---- <!-- .slide: data-transition="convex" --> ### Reason Slowloris <small>https://en.wikipedia.org/wiki/Slowloris_(computer_security)</small> ---- <!-- .slide: data-transition="convex" --> ### Spec The complete guide to Go net/http timeouts <small>https://blog.cloudflare.com/the-complete-guide-to-golang-net-http-timeouts/</small> ---- <!-- .slide: data-transition="convex" --> #### Server ![http server timeout](https://blog.cloudflare.com/content/images/2016/06/Timeouts-001.png) ---- <!-- .slide: data-transition="convex" --> ##### Phase Timeout ```go= srv := &http.Server{ ReadTimeout: 5 * time.Second, WriteTimeout: 10 * time.Second, IdleTimeout: 60 * time.Second, ReadHeaderTimeout: 1 * time.Second, } log.Println(srv.ListenAndServe()) ``` ---- <!-- .slide: data-transition="convex" --> #### Client ![http client timeout](https://blog.cloudflare.com/content/images/2016/06/Timeouts-002.png) ---- <!-- .slide: data-transition="convex" --> ##### Entire Timeout ```go= c := &http.Client{ Timeout: 15 * time.Second, } resp, err := c.Get("https://blog.filippo.io/") ``` ---- <!-- .slide: data-transition="convex" --> ##### Phase Timeout ```go= c := &http.Client{ Transport: &http.Transport{ Dial: (&net.Dialer{ Timeout: 30 * time.Second, KeepAlive: 30 * time.Second, }).Dial, TLSHandshakeTimeout: 10 * time.Second, ResponseHeaderTimeout: 10 * time.Second, ExpectContinueTimeout: 1 * time.Second, IdleConnTimeout: 60 * time.Second, } } ``` --- <!-- .slide: data-transition="convex" --> ## Streaming & Timeout ---- <!-- .slide: data-transition="convex" --> ### Phase Timeout ![http server timeout](https://blog.cloudflare.com/content/images/2016/06/Timeouts-001.png =738x216) ![http client timeout](https://blog.cloudflare.com/content/images/2016/06/Timeouts-002.png =745x201) ---- <!-- .slide: data-transition="convex" --> ### Streaming Timeout <small>net/http: no way of manipulating timeouts in Handler #16100</small> https://github.com/golang/go/issues/16100 ---- <!-- .slide: data-transition="convex" --> ### Streaming Timeout per Entrypoint # It does not exist! ---- <!-- .slide: data-transition="zoom" --> #### Game Over? * Good Game... GG * 人生オワタ \(^ o ^)/ # :shit: :shit: :shit: ---- <!-- .slide: data-transition="zoom" --> #### Open Source # Do It Yourself --- <!-- .slide: data-transition="convex" --> ## Implementation ---- <!-- .slide: data-transition="convex" --> ### fieliapm/stream-http-go * Repository * https://github.com/fieliapm/stream-http-go * GoDoc * https://godoc.org/github.com/fieliapm/stream-http-go/pkg/stream_http ---- <!-- .slide: data-transition="convex" --> #### Client ---- <!-- .slide: data-transition="convex" --> ##### Client Example ```go= ctx := context.Background() client := http.DefaultClient requestMod := stream_http.RequestMod(func(req *http.Request) { req.Header.Set("Content-Type", "application/x-www-form-urlencoded") req.Header.Set("Authorization", "Bearer qawsedrftgyhujikolp") query := req.URL.Query() query.Add("param", "pvalue") req.URL.RawQuery = query.Encode() }) requestTimeout := stream_http.Timeout(500 * time.Millisecond) reqBody := bytes.NewBuffer([]byte("form=fvalue")) var respBody bytes.Buffer resp, err := stream_http.DoRequest(ctx, client, http.MethodPost, "http://httpbin.org/anything", reqBody, &respBody, requestMod, requestTimeout) //resp, err := stream_http.DoRequest(ctx, client, http.MethodGet, "http://httpbin.org/status/401", nil, &respBody, requestMod, requestTimeout) if err != nil { return err } fmt.Printf("get response with status code: %d\n%s\n", resp.StatusCode, respBody.String()) ``` ---- <!-- .slide: data-transition="convex" --> ##### Spirit * context :arrow_forward: context, cancel() * :warning: timeout event :arrow_right: cancel() * :heavy_check_mark: chunk transmitted :arrow_right: reset timeout <!-- .element: class="fragment" data-fragment-index="1" --> ---- <!-- .slide: data-transition="convex" --> ##### Client Control Flow ```graphviz digraph client_control_flow { nodesep=1.0 // increases the separation between nodes node [color="hotpink" shape=box] edge [color="hotpink" style=dashed] with_cancel [shape="oval" label="context.WithCancel()"] context_parent [label="parent context"] context [label="context"] cancel [color="gray" label="cancel()"] client_do [shape="oval" label="client.Do()"] timer [shape="oval" label="time.AfterFunc()"] context_parent->with_cancel [style=solid] with_cancel->context [style=solid] with_cancel->cancel [style=solid] cancel->context [color="gray"] context->client_do [style=solid] timer->cancel [color="orange"] } ``` ---- <!-- .slide: data-transition="convex" --> #### Server ---- <!-- .slide: data-transition="convex" --> ##### Server Example ```go= func handleFunc(w http.ResponseWriter, req *http.Request) { n, err := stream_http.TimeoutCopy(&reqBody, req.Body, time.Duration(-1), 500*time.Millisecond) n, err = stream_http.TimeoutCopy(w, &respBody, 500*time.Millisecond, time.Duration(-1)) } ``` ---- <!-- .slide: data-transition="convex" --> ##### Spirit * TimeoutCopy() :arrow_forward: copy goroutine * :heavy_check_mark: chunk transmitted :arrow_right: reset timeout <!-- .element: class="fragment" data-fragment-index="1" --> * :warning: timeout event :arrow_right: TimeoutCopy() <!-- .element: class="fragment" data-fragment-index="2" --> * TimeoutCopy() :arrow_right: :x: copy goroutine <!-- .element: class="fragment" data-fragment-index="3" --> ---- <!-- .slide: data-transition="convex" --> ##### Server Handle Flow ```sequence note left of handler: receive request handler->TimeoutCopy(): call TimeoutCopy()->runtimeTimer: time.NewTimer() TimeoutCopy()->copy goroutine: request/response_writer note right of copy goroutine: chunk transmitted copy goroutine->runtimeTimer: timer.Reset()/timer.Stop() note right of copy goroutine: copy finished copy goroutine->TimeoutCopy(): copyResult/panic TimeoutCopy()->handler: copy result note left of handler: send ok/error response note right of runtimeTimer: timeout runtimeTimer->TimeoutCopy(): timer.C TimeoutCopy()->handler: error timeout TimeoutCopy()->copy goroutine: close(doneChan) note left of handler: send timeout response ``` ---- <!-- .slide: data-transition="convex" --> ##### Question How to correctly stop server handler? ---- <!-- .slide: data-transition="convex" --> ##### Answer # I don't know ¯\\\_(ツ)\_/¯ --- <!-- .slide: data-transition="convex" --> ## Timer Issues ---- <!-- .slide: data-transition="convex" --> <small>time: Timer.C can still trigger even after Timer.Reset is called #11513</small> https://github.com/golang/go/issues/11513 ---- <!-- .slide: data-transition="convex" --> Timer.Reset() note https://golang.org/pkg/time/#Timer.Reset <small> Reset should be invoked only on stopped or expired timers with drained channels. If a program has already received a value from t.C, the timer is known to have expired and the channel drained, so t.Reset can be used directly. If a program has not yet received a value from t.C, however, the timer must be stopped and—if Stop reports that the timer expired before being stopped—the channel explicitly drained: </small> ```go= if !t.Stop() { <-t.C } t.Reset(d) ``` <small> This should not be done concurrent to other receives from the Timer's channel. </small> ---- <!-- .slide: data-transition="convex" --> <small>time: Timer.Reset is not possible to use correctly #14038</small> https://github.com/golang/go/issues/14038 ---- <!-- .slide: data-transition="convex" --> Timer.Reset() note https://golang.org/pkg/time/#Timer.Reset <small> Note that it is not possible to use Reset's return value correctly, as there is a race condition between draining the channel and the new timer expiring. Reset should always be invoked on stopped or expired channels, as described above. The return value exists to preserve compatibility with existing programs. </small> ---- <!-- .slide: data-transition="convex" --> ### Understand Timer Behaviour ```go= if !t.Stop() { <-t.C // don't do this if t.C is drained from another goroutine } _ = t.Reset(d) // don't use return value ``` ---- <!-- .slide: data-transition="convex" --> ### Broken Timer Implementation * Go version * go \< 1.8 :arrow_right: crap <!-- .element: class="fragment" data-fragment-index="1" --> * go \>= 1.8 (added note) :arrow_right: crap <!-- .element: class="fragment" data-fragment-index="2" --> ---- <!-- .slide: data-transition="convex" --> ### History Repeats Itself * <small>time: Timer.C can still trigger even after Timer.Reset is called #11513</small> * <small>time: Timer.Reset is not possible to use correctly #14038</small> * <small>... ...</small> ---- <!-- .slide: data-transition="convex" --> ![Georg Wilhelm Friedrich Hegel](https://upload.wikimedia.org/wikipedia/commons/b/bf/1831_Schlesinger_Philosoph_Georg_Friedrich_Wilhelm_Hegel_anagoria.JPG =379x480) <small>Georg Wilhelm Friedrich Hegel https://de.wikipedia.org/wiki/Georg_Wilhelm_Friedrich_Hegel</small> ---- <!-- .slide: data-transition="convex" --> > <small> Was die Erfahrung aber und die Geschichte lehren, ist dieses, daß **Völker und Regierungen niemals etwas aus der Geschichte gelernt** und nach Lehren, die aus derselben zu ziehen gewesen wären, gehandelt haben. </small> > [name=Georg Wilhelm Friedrich Hegel] :dizzy: ---- <!-- .slide: data-transition="convex" --> > <small> We learn from history that **we do not learn from history**. </small> > [name=Georg Wilhelm Friedrich Hegel] --- <!-- .slide: data-transition="convex" --> ## Conclusion ---- <!-- .slide: data-transition="convex" --> ### Contribute! * fieliapm/stream-http-go * Better client interface * Server side timeout handling example * How to correctly stop server handler > [name=郭學聰 Hsueh-Tsung Kuo] [time=2020_08_02] [color=red] :notebook: --- <!-- .slide: data-transition="convex" --> ## Reference ---- <!-- .slide: data-transition="convex" --> ### Blog * The complete guide to Go net/http timeouts \[en\] * <small>https://blog.cloudflare.com/the-complete-guide-to-golang-net-http-timeouts/</small> * How to use timer reset in Golang correctly \[zh\] * <small>https://tonybai.com/2016/12/21/how-to-use-timer-reset-in-golang-correctly/</small> --- <!-- .slide: data-transition="zoom" --> ## Q&A --- <style> .reveal { background: #FFDFEF; color: black; } .reveal h2, .reveal h3, .reveal h4, .reveal h5, .reveal h6 { color: black; } .reveal code { font-size: 18px !important; line-height: 1.2; } .progress div{ height:14px !important; background: hotpink !important; } // original template .rightpart{ float:right; width:50%; } .leftpart{ margin-right: 50% !important; height:50%; } .reveal section img { background:none; border:none; box-shadow:none; } p.blo { font-size: 50px !important; background:#B6BDBB; border:1px solid silver; display:inline-block; padding:0.5em 0.75em; border-radius: 10px; box-shadow: 5px 5px 5px #666; } p.blo1 { background: #c7c2bb; } p.blo2 { background: #b8c0c8; } p.blo3 { background: #c7cedd; } p.bloT { font-size: 60px !important; background:#B6BDD3; border:1px solid silver; display:inline-block; padding:0.5em 0.75em; border-radius: 8px; box-shadow: 1px 2px 5px #333; } p.bloA { background: #B6BDE3; } p.bloB { background: #E3BDB3; } .slide-number{ margin-bottom:10px !important; width:100%; text-align:center; font-size:25px !important; background-color:transparent !important; } iframe.myclass{ width:100px; height:100px; bottom:0; left:0; position:fixed; border:none; z-index:99999; } h1.raw { color: #fff; background-image: linear-gradient(90deg,#f35626,#feab3a); -webkit-background-clip: text; -webkit-text-fill-color: transparent; animation: hue 5s infinite linear; } @keyframes hue { from { filter: hue-rotate(0deg); } to { filter: hue-rotate(360deg); } } .progress{ height:14px !important; } .progress span{ height:14px !important; background: url("") repeat-x !important; } .progress span:after, .progress span.nyancat{ content: ""; background: url('') !important; width: 34px !important; height: 21px !important; border: none !important; float:right; margin-top:-7px; margin-right:-10px; } </style>
{"metaMigratedAt":"2023-06-15T11:14:48.424Z","metaMigratedFrom":"YAML","title":"net & http & timeout","breaks":true,"description":"View the slide with \"Slide Mode\".","slideOptions":"{\"spotlight\":{\"enabled\":false},\"allottedMinutes\":25}","contributors":"[{\"id\":\"ea27dcd7-a3f2-47c2-b25e-6760e7936c38\",\"add\":22328,\"del\":1979}]"}
    818 views