package http import ( "bufio" "bytes" "context" "fmt" "io/ioutil" "net" "net/http" "testing" "github.com/asim/go-micro/plugins/client/http/v3/test" "github.com/asim/go-micro/plugins/registry/memory/v3" "github.com/asim/go-micro/v3/client" "github.com/asim/go-micro/v3/registry" "github.com/asim/go-micro/v3/selector" ) func TestHTTPClient(t *testing.T) { r := memory.NewRegistry() s := selector.NewSelector(selector.Registry(r)) l, err := net.Listen("tcp", "127.0.0.1:0") if err != nil { t.Fatal(err) } defer l.Close() mux := http.NewServeMux() mux.HandleFunc("/foo/bar", func(w http.ResponseWriter, r *http.Request) { // only accept post if r.Method != "POST" { http.Error(w, "expect post method", 500) return } // get codec ct := r.Header.Get("Content-Type") codec, ok := defaultHTTPCodecs[ct] if !ok { http.Error(w, "codec not found", 500) return } b, err := ioutil.ReadAll(r.Body) if err != nil { http.Error(w, err.Error(), 500) return } // extract message msg := new(test.Message) if err := codec.Unmarshal(b, msg); err != nil { http.Error(w, err.Error(), 500) return } // marshal response b, err = codec.Marshal(msg) if err != nil { http.Error(w, err.Error(), 500) return } // write response w.Write(b) }) go http.Serve(l, mux) if err := r.Register(®istry.Service{ Name: "test.service", Nodes: []*registry.Node{ { Id: "test.service.1", Address: l.Addr().String(), Metadata: map[string]string{ "protocol": "http", }, }, }, }); err != nil { t.Fatal(err) } c := NewClient(client.Selector(s)) for i := 0; i < 10; i++ { msg := &test.Message{ Seq: int64(i), Data: fmt.Sprintf("message %d", i), } endpoint := "/foo/bar" if i%2 == 0 { endpoint = endpoint + "?pageNum=1&pageSize=2" } req := c.NewRequest("test.service", endpoint, msg) rsp := new(test.Message) err := c.Call(context.TODO(), req, rsp) if err != nil { t.Fatal(err) } if rsp.Seq != msg.Seq { t.Fatalf("invalid seq %d for %d", rsp.Seq, msg.Seq) } } } func TestHTTPClientStream(t *testing.T) { r := memory.NewRegistry() s := selector.NewSelector(selector.Registry(r)) l, err := net.Listen("tcp", "127.0.0.1:0") if err != nil { t.Fatal(err) } defer l.Close() mux := http.NewServeMux() mux.HandleFunc("/foo/bar", func(w http.ResponseWriter, r *http.Request) { // only accept post if r.Method != "POST" { http.Error(w, "expect post method", 500) return } // hijack the connection hj, ok := w.(http.Hijacker) if !ok { http.Error(w, "could not hijack conn", 500) return } // hijacked conn, bufrw, err := hj.Hijack() if err != nil { http.Error(w, err.Error(), 500) return } defer conn.Close() // read off the first request // get codec ct := r.Header.Get("Content-Type") codec, ok := defaultHTTPCodecs[ct] if !ok { http.Error(w, "codec not found", 500) return } b, err := ioutil.ReadAll(r.Body) if err != nil { http.Error(w, err.Error(), 500) return } // extract message msg := new(test.Message) if err := codec.Unmarshal(b, msg); err != nil { http.Error(w, err.Error(), 500) return } // marshal response b, err = codec.Marshal(msg) if err != nil { http.Error(w, err.Error(), 500) return } // write response rsp := &http.Response{ Header: r.Header, Body: &buffer{bytes.NewBuffer(b)}, Status: "200 OK", StatusCode: 200, Proto: "HTTP/1.1", ProtoMajor: 1, ProtoMinor: 1, ContentLength: int64(len(b)), } // write response rsp.Write(bufrw) bufrw.Flush() reader := bufio.NewReader(conn) for { r, err := http.ReadRequest(reader) if err != nil { http.Error(w, err.Error(), 500) return } b, err = ioutil.ReadAll(r.Body) if err != nil { http.Error(w, err.Error(), 500) return } // extract message msg := new(test.Message) if err := codec.Unmarshal(b, msg); err != nil { http.Error(w, err.Error(), 500) return } // marshal response b, err = codec.Marshal(msg) if err != nil { http.Error(w, err.Error(), 500) return } rsp := &http.Response{ Header: r.Header, Body: &buffer{bytes.NewBuffer(b)}, Status: "200 OK", StatusCode: 200, Proto: "HTTP/1.1", ProtoMajor: 1, ProtoMinor: 1, ContentLength: int64(len(b)), } // write response rsp.Write(bufrw) bufrw.Flush() } }) go http.Serve(l, mux) if err := r.Register(®istry.Service{ Name: "test.service", Nodes: []*registry.Node{ { Id: "test.service.1", Address: l.Addr().String(), Metadata: map[string]string{ "protocol": "http", }, }, }, }); err != nil { t.Fatal(err) } c := NewClient(client.Selector(s)) req := c.NewRequest("test.service", "/foo/bar", new(test.Message)) stream, err := c.Stream(context.TODO(), req) if err != nil { t.Fatal(err) } defer stream.Close() for i := 0; i < 10; i++ { msg := &test.Message{ Seq: int64(i), Data: fmt.Sprintf("message %d", i), } err := stream.Send(msg) if err != nil { t.Fatal(err) } rsp := new(test.Message) err = stream.Recv(rsp) if err != nil { t.Fatal(err) } if rsp.Seq != msg.Seq { t.Fatalf("invalid seq %d for %d", rsp.Seq, msg.Seq) } } }