# gock [![Build Status](https://travis-ci.org/h2non/gock.svg?branch=master)](https://travis-ci.org/h2non/gock) [![GitHub release](https://img.shields.io/badge/version-v1.0-orange.svg?style=flat)](https://github.com/h2non/gock/releases) [![GoDoc](https://godoc.org/github.com/h2non/gock?status.svg)](https://godoc.org/github.com/h2non/gock) [![Coverage Status](https://coveralls.io/repos/github/h2non/gock/badge.svg?branch=master)](https://coveralls.io/github/h2non/gock?branch=master) [![Go Report Card](https://img.shields.io/badge/go_report-A+-brightgreen.svg)](https://goreportcard.com/report/github.com/h2non/gock) [![license](https://img.shields.io/badge/license-MIT-blue.svg)]() Versatile HTTP mocking made easy in [Go](https://golang.org). Heavily inspired by [nock](https://github.com/node-nock/nock). See also its Python port, [pook](https://github.com/h2non/pook). Take a look to the [examples](#examples) to get started. ## Features - Simple, expressive, fluent API. - Semantic DSL for easy HTTP mocks definition. - Built-in helpers for easy JSON/XML mocking. - Supports persistent and volatile mocks. - Full regexp capable HTTP request matching. - Designed for both testing and runtime scenarios. - Match request by method, URL params, headers and bodies. - Extensible and pluggable HTTP matching rules. - Ability to switch between mock and real networking modes. - Ability to filter/map HTTP requests for accurate mock matching. - Supports map and filters to handle mocks easily. - Wide compatible HTTP interceptor using `http.RoundTripper` interface. - Works with any `net/http` compatible client, such as [gentleman](https://github.com/h2non/gentleman). - Network delay simulation (beta). - Extensible and hackable API. - Dependency free. ## Installation ```bash go get -u gopkg.in/h2non/gock.v1 ``` ## API See [godoc reference](https://godoc.org/github.com/h2non/gock) for detailed API documentation. ## How it mocks 1. Intercepts any HTTP outgoing request via `http.DefaultTransport` or custom `http.Transport` used by any `http.Client`. 2. Matches outgoing HTTP requests against a pool of defined HTTP mock expectations in FIFO declaration order. 3. If at least one mock matches, it will be used in order to compose the mock HTTP response. 4. If no mock can be matched, it will resolve the request with an error, unless real networking mode is enable, in which case a real HTTP request will be performed. ## Tips #### Testing Declare your mocks before you start declaring the concrete test logic: ```go func TestFoo(t *testing.T) { defer gock.Off() // Flush pending mocks after test execution gock.New("http://server.com"). Get("/bar"). Reply(200). JSON(map[string]string{"foo": "bar"}) // Your test code starts here... } ``` #### Race conditions If you're running concurrent code, be aware that your mocks are declared first to avoid unexpected race conditions while configuring `gock` or intercepting custom HTTP clients. `gock` is not fully thread-safe, but sensible parts are. Any help making `gock` more reliable in this sense is highly appreciated. #### Define complex mocks first If you're mocking a bunch of mocks in the same test suite, it's recommended to define the more concrete mocks first, and then the generic ones. This approach usually avoids matching unexpected generic mocks (e.g: specific header, body payload...) instead of the generic ones that performs less complex matches. ## Examples See [examples](https://github.com/h2non/gock/tree/master/_examples) directory for more featured use cases. #### Simple mocking via tests ```go package test import ( "github.com/nbio/st" "gopkg.in/h2non/gock.v1" "io/ioutil" "net/http" "testing" ) func TestSimple(t *testing.T) { defer gock.Off() gock.New("http://foo.com"). Get("/bar"). Reply(200). JSON(map[string]string{"foo": "bar"}) res, err := http.Get("http://foo.com/bar") st.Expect(t, err, nil) st.Expect(t, res.StatusCode, 200) body, _ := ioutil.ReadAll(res.Body) st.Expect(t, string(body)[:13], `{"foo":"bar"}`) // Verify that we don't have pending mocks st.Expect(t, gock.IsDone(), true) } ``` #### Request headers matching ```go package test import ( "github.com/nbio/st" "gopkg.in/h2non/gock.v1" "io/ioutil" "net/http" "testing" ) func TestMatchHeaders(t *testing.T) { defer gock.Off() gock.New("http://foo.com"). MatchHeader("Authorization", "^foo bar$"). MatchHeader("API", "1.[0-9]+"). HeaderPresent("Accept"). Reply(200). BodyString("foo foo") req, err := http.NewRequest("GET", "http://foo.com", nil) req.Header.Set("Authorization", "foo bar") req.Header.Set("API", "1.0") req.Header.Set("Accept", "text/plain") res, err := (&http.Client{}).Do(req) st.Expect(t, err, nil) st.Expect(t, res.StatusCode, 200) body, _ := ioutil.ReadAll(res.Body) st.Expect(t, string(body), "foo foo") // Verify that we don't have pending mocks st.Expect(t, gock.IsDone(), true) } ``` #### JSON body matching and response ```go package test import ( "bytes" "github.com/nbio/st" "gopkg.in/h2non/gock.v1" "io/ioutil" "net/http" "testing" ) func TestMockSimple(t *testing.T) { defer gock.Off() gock.New("http://foo.com"). Post("/bar"). MatchType("json"). JSON(map[string]string{"foo": "bar"}). Reply(201). JSON(map[string]string{"bar": "foo"}) body := bytes.NewBuffer([]byte(`{"foo":"bar"}`)) res, err := http.Post("http://foo.com/bar", "application/json", body) st.Expect(t, err, nil) st.Expect(t, res.StatusCode, 201) resBody, _ := ioutil.ReadAll(res.Body) st.Expect(t, string(resBody)[:13], `{"bar":"foo"}`) // Verify that we don't have pending mocks st.Expect(t, gock.IsDone(), true) } ``` #### Mocking a custom http.Client and http.RoundTripper ```go package test import ( "github.com/nbio/st" "gopkg.in/h2non/gock.v1" "io/ioutil" "net/http" "testing" ) func TestClient(t *testing.T) { defer gock.Off() gock.New("http://foo.com"). Reply(200). BodyString("foo foo") req, err := http.NewRequest("GET", "http://foo.com", nil) client := &http.Client{Transport: &http.Transport{}} gock.InterceptClient(client) res, err := client.Do(req) st.Expect(t, err, nil) st.Expect(t, res.StatusCode, 200) body, _ := ioutil.ReadAll(res.Body) st.Expect(t, string(body), "foo foo") // Verify that we don't have pending mocks st.Expect(t, gock.IsDone(), true) } ``` #### Enable real networking ```go package main import ( "fmt" "gopkg.in/h2non/gock.v1" "io/ioutil" "net/http" ) func main() { defer gock.Off() defer gock.DisableNetworking() gock.EnableNetworking() gock.New("http://httpbin.org"). Get("/get"). Reply(201). SetHeader("Server", "gock") res, err := http.Get("http://httpbin.org/get") if err != nil { fmt.Errorf("Error: %s", err) } // The response status comes from the mock fmt.Printf("Status: %d\n", res.StatusCode) // The server header comes from mock as well fmt.Printf("Server header: %s\n", res.Header.Get("Server")) // Response body is the original body, _ := ioutil.ReadAll(res.Body) fmt.Printf("Body: %s", string(body)) // Verify that we don't have pending mocks st.Expect(t, gock.IsDone(), true) } ``` ## Hacking it! You can easily hack `gock` defining custom matcher functions with own matching rules. See [add matcher functions](https://github.com/h2non/gock/blob/master/_examples/add_matchers/matchers.go) and [custom matching layer](https://github.com/h2non/gock/blob/master/_examples/custom_matcher/matcher.go) examples for further details. ## License MIT - Tomas Aparicio