adding context
parent
630fe8ee45
commit
de64f65984
@ -0,0 +1,24 @@
|
||||
# Using Context
|
||||
|
||||
Remember that a common pattern of signaling events or completion is using a blocking channel that receives and empty struct. An empty struct has zero memory footprint. It is typically used to signal completion or stopping.
|
||||
|
||||
The Context.Done() returns a channel that is closed when this Context is canceled `Done() <-chan struct{}`. Contexts have a root and can form a parent child tree structure. A child is "derived" from a parent context and when a Context is cancelled all derived contexts are also canceled. WithCancel and WithTimeout return derived Context values that can be canceled sooner than the parent Context.
|
||||
|
||||
> Incoming requests to a server should create a Context, and outgoing calls to servers should accept a Context. The chain of function calls between them must propagate the Context, optionally replacing it with a derived Context created using WithCancel, WithDeadline, WithTimeout, or WithValue. When a Context is canceled, all Contexts derived from it are also canceled.
|
||||
|
||||
## What about Context.Value
|
||||
|
||||
The problem with context.Values is that it's just an untyped map so you have no type-safety and you have to handle it not actually containing your value. You have to create a coupling of map keys from one module to another and if someone changes something things start breaking.
|
||||
|
||||
In short, if a function needs some values, put them as typed parameters rather than trying to fetch them from
|
||||
context.Value
|
||||
. This makes it statically checked and documented for everyone to see.
|
||||
|
||||
Context.Value is good for things like "A trace ID". It's information that is not needed for by every function in the call stack.
|
||||
|
||||
> Context.Value should inform not control. The content of context.Value is for maintainers not users. It should never be required input for documented or expected results.
|
||||
|
||||
## Readings
|
||||
|
||||
https://go.dev/blog/context
|
||||
https://go.dev/blog/pipelines
|
@ -0,0 +1,47 @@
|
||||
package my_context
|
||||
|
||||
import (
|
||||
"context"
|
||||
"fmt"
|
||||
"net/http"
|
||||
)
|
||||
|
||||
type Store interface {
|
||||
Fetch(ctx context.Context) (string, error)
|
||||
// instead of calling cancel we want to use context above to cross API boundries
|
||||
// Cancel()
|
||||
}
|
||||
|
||||
func Server(store Store) http.HandlerFunc {
|
||||
return func(w http.ResponseWriter, r *http.Request) {
|
||||
|
||||
// THIS WAS WHAT WE DID WHEN WE DIDN"T PROPOGATE CONTEXT
|
||||
// // This is the requests ctx not a derived context
|
||||
// ctx := r.Context()
|
||||
|
||||
// data := make(chan string, 1)
|
||||
|
||||
// go func() {
|
||||
// data <- store.Fetch()
|
||||
// }()
|
||||
|
||||
// // Here we are simply waiting on the which channel
|
||||
// // will close out first. If the Done() channel on the request
|
||||
// // does store.Cancel() is called.
|
||||
|
||||
// // Remember select lets use wait on multiple async operations and execute cases
|
||||
// // based on which one completes first as signaled by data being received on that channel.
|
||||
// select {
|
||||
// case d := <-data:
|
||||
// fmt.Fprint(w, d)
|
||||
// case <-ctx.Done():
|
||||
// store.Cancel()
|
||||
// }
|
||||
data, err := store.Fetch(r.Context())
|
||||
|
||||
if err != nil {
|
||||
return // todo log the error however you like
|
||||
}
|
||||
fmt.Fprint(w, data)
|
||||
}
|
||||
}
|
@ -0,0 +1,175 @@
|
||||
package my_context
|
||||
|
||||
import (
|
||||
"context"
|
||||
"errors"
|
||||
"log"
|
||||
"net/http"
|
||||
"net/http/httptest"
|
||||
"testing"
|
||||
"time"
|
||||
)
|
||||
|
||||
// THIS WAS WHAT WE DID WHEN WE DIDN"T PROPOGATE CONTEXT
|
||||
// type SpyStore struct {
|
||||
// response string
|
||||
// cancelled bool
|
||||
// t *testing.T
|
||||
// }
|
||||
|
||||
// func (s *SpyStore) assertWasCancelled() {
|
||||
// s.t.Helper()
|
||||
// if !s.cancelled {
|
||||
// s.t.Error("store was not told to cancel")
|
||||
// }
|
||||
// }
|
||||
|
||||
// func (s *SpyStore) assertWasNotCancelled() {
|
||||
// s.t.Helper()
|
||||
// if s.cancelled {
|
||||
// s.t.Error("store was told to cancel")
|
||||
// }
|
||||
// }
|
||||
|
||||
// func (s *SpyStore) Fetch() string {
|
||||
// time.Sleep(100 * time.Millisecond)
|
||||
// return s.response
|
||||
// }
|
||||
|
||||
// func (s *SpyStore) Cancel() {
|
||||
// s.cancelled = true
|
||||
// }
|
||||
|
||||
// func TestServer(t *testing.T) {
|
||||
// data := "Hello, Gopher"
|
||||
// t.Run("Happy path", func(t *testing.T) {
|
||||
// store := &SpyStore{response: data, t: t}
|
||||
// svr := Server(store)
|
||||
|
||||
// request := httptest.NewRequest(http.MethodGet, "/", nil)
|
||||
// response := httptest.NewRecorder()
|
||||
|
||||
// svr.ServeHTTP(response, request)
|
||||
|
||||
// if response.Body.String() != data {
|
||||
// t.Errorf(`got "%s" want "%s"`, response.Body.String(), data)
|
||||
// }
|
||||
|
||||
// store.assertWasNotCancelled()
|
||||
|
||||
// })
|
||||
// t.Run("Should cancel", func(t *testing.T) {
|
||||
// store := &SpyStore{response: data, t: t}
|
||||
// svr := Server(store)
|
||||
|
||||
// request := httptest.NewRequest(http.MethodGet, "/", nil)
|
||||
// // we are copying the parent context but with a new Done channel.
|
||||
// // The returned context's Done channel is closed when the returned cancel
|
||||
// // function is called or when the parent context's Done channel is closed,
|
||||
// // whichever happens first.
|
||||
// cancellingCtx, cancel := context.WithCancel(request.Context())
|
||||
// time.AfterFunc(5*time.Millisecond, cancel)
|
||||
// // changing the current request context to our new cancellingCtx
|
||||
// request = request.WithContext(cancellingCtx)
|
||||
// response := httptest.NewRecorder()
|
||||
|
||||
// svr.ServeHTTP(response, request)
|
||||
|
||||
// store.a
|
||||
|
||||
// })
|
||||
// }
|
||||
|
||||
type SpyStore struct {
|
||||
response string
|
||||
t *testing.T
|
||||
}
|
||||
|
||||
func (s *SpyStore) Fetch(ctx context.Context) (string, error) {
|
||||
data := make(chan string, 1)
|
||||
|
||||
go func() {
|
||||
var result string
|
||||
for _, c := range s.response {
|
||||
select {
|
||||
case <-ctx.Done():
|
||||
log.Println("spy store got cancelled")
|
||||
return
|
||||
default:
|
||||
// simulates some network traffic?
|
||||
time.Sleep(10 * time.Millisecond)
|
||||
result += string(c)
|
||||
}
|
||||
}
|
||||
data <- result
|
||||
}()
|
||||
|
||||
select {
|
||||
case <-ctx.Done():
|
||||
return "", ctx.Err()
|
||||
case res := <-data:
|
||||
return res, nil
|
||||
}
|
||||
}
|
||||
|
||||
// Rolling our own Spy for ResponseWriter because httptest.ResponseRecorder can't be
|
||||
// used to determine if any response was writing in the error case
|
||||
|
||||
// SpyResponse Writer implements http.ResponseWriter so we can use it in the test.
|
||||
type SpyResponseWriter struct {
|
||||
written bool
|
||||
}
|
||||
|
||||
func (s *SpyResponseWriter) Header() http.Header {
|
||||
s.written = true
|
||||
return nil
|
||||
}
|
||||
|
||||
func (s *SpyResponseWriter) Write([]byte) (int, error) {
|
||||
s.written = true
|
||||
return 0, errors.New("not implemented")
|
||||
}
|
||||
|
||||
func (s *SpyResponseWriter) WriteHeader(statusCode int) {
|
||||
s.written = true
|
||||
}
|
||||
|
||||
func TestServer(t *testing.T) {
|
||||
data := "Hello, Gopher"
|
||||
|
||||
t.Run("Happy Path", func(t *testing.T) {
|
||||
|
||||
store := &SpyStore{response: data, t: t}
|
||||
svr := Server(store)
|
||||
|
||||
request := httptest.NewRequest(http.MethodGet, "/", nil)
|
||||
response := httptest.NewRecorder()
|
||||
|
||||
svr.ServeHTTP(response, request)
|
||||
|
||||
if response.Body.String() != data {
|
||||
t.Errorf(`got "%s", want "%s"`, response.Body.String(), data)
|
||||
}
|
||||
})
|
||||
|
||||
t.Run("tells store to cancel work if request is cancelled", func(t *testing.T) {
|
||||
store := &SpyStore{response: data, t: t}
|
||||
svr := Server(store)
|
||||
|
||||
request := httptest.NewRequest(http.MethodGet, "/", nil)
|
||||
|
||||
cancellingCtx, cancel := context.WithCancel(request.Context())
|
||||
time.AfterFunc(5*time.Millisecond, cancel)
|
||||
request = request.WithContext(cancellingCtx)
|
||||
|
||||
response := &SpyResponseWriter{}
|
||||
|
||||
svr.ServeHTTP(response, request)
|
||||
|
||||
if response.written {
|
||||
t.Errorf("A response should not have been written")
|
||||
}
|
||||
|
||||
})
|
||||
|
||||
}
|
Loading…
Reference in New Issue