From 46e2136078c33f3c0b76e279999393f5cbf50ee3 Mon Sep 17 00:00:00 2001 From: Dylan Date: Mon, 31 Aug 2020 21:24:51 -0400 Subject: [PATCH 1/5] add fetch test from http --- cmd/commander/commander.go | 6 ++--- pkg/app/test_command.go | 49 +++++++++++++++++++++++++++++--------- 2 files changed, 41 insertions(+), 14 deletions(-) diff --git a/cmd/commander/commander.go b/cmd/commander/commander.go index 785a029a..f8e007b8 100644 --- a/cmd/commander/commander.go +++ b/cmd/commander/commander.go @@ -45,8 +45,8 @@ func createCliApp() *cli.App { func createTestCommand() cli.Command { return cli.Command{ - Name: "test", - Usage: `Execute cli app tests + Name: "test", + Usage: `Execute cli app tests By default it will use the commander.yaml from your current directory. Tests are always executed in alphabetical order. @@ -79,7 +79,7 @@ test commander.yaml --filter="^filter1$" Usage: "Execute all test files in a directory sorted by file name, this is not recursive - e.g. /path/to/test_files/", }, cli.StringSliceFlag{ - Name: "filter", + Name: "filter", Usage: `Filter tests by a given regex pattern. Tests are filtered by its title.`, }, }, diff --git a/pkg/app/test_command.go b/pkg/app/test_command.go index df735a09..85f74ba3 100644 --- a/pkg/app/test_command.go +++ b/pkg/app/test_command.go @@ -4,8 +4,10 @@ import ( "fmt" "io/ioutil" "log" + "net/http" "os" "path" + "strings" "github.com/commander-cli/commander/pkg/output" "github.com/commander-cli/commander/pkg/runtime" @@ -36,6 +38,10 @@ func TestCommand(testPath string, ctx TestCommandContext) error { fmt.Println("Starting test against directory: " + testPath + "...") fmt.Println("") result, err = testDir(testPath, ctx.Filters) + case isURL(testPath): + fmt.Println("Starting test from " + testPath + "...") + fmt.Println("") + result, err = testURL(testPath, ctx.Filters) default: fmt.Println("Starting test file " + testPath + "...") fmt.Println("") @@ -53,6 +59,15 @@ func TestCommand(testPath string, ctx TestCommandContext) error { return nil } +func testFile(filePath string, fileName string, filters runtime.Filters) (runtime.Result, error) { + s, err := readFile(filePath, fileName) + if err != nil { + return runtime.Result{}, fmt.Errorf("Error " + err.Error()) + } + + return execute(s, filters) +} + func testDir(directory string, filters runtime.Filters) (runtime.Result, error) { result := runtime.Result{} files, err := ioutil.ReadDir(directory) @@ -77,6 +92,27 @@ func testDir(directory string, filters runtime.Filters) (runtime.Result, error) return result, nil } +func testURL(url string, filters runtime.Filters) (runtime.Result, error) { + resp, err := http.Get(url) + if err != nil { + return runtime.Result{}, err + } + defer resp.Body.Close() + + body, err := ioutil.ReadAll(resp.Body) + if err != nil { + return runtime.Result{}, err + } + + s := suite.ParseYAML(body, "") + + return execute(s, filters) +} + +func isURL(s string) bool { + return strings.HasPrefix(s, "http://") || strings.HasPrefix(s, "https://") +} + func convergeResults(result runtime.Result, new runtime.Result) runtime.Result { result.TestResults = append(result.TestResults, new.TestResults...) result.Failed += new.Failed @@ -85,15 +121,6 @@ func convergeResults(result runtime.Result, new runtime.Result) runtime.Result { return result } -func testFile(filePath string, fileName string, filters runtime.Filters) (runtime.Result, error) { - s, err := readFile(filePath, fileName) - if err != nil { - return runtime.Result{}, fmt.Errorf("Error " + err.Error()) - } - - return execute(s, filters) -} - func execute(s suite.Suite, filters runtime.Filters) (runtime.Result, error) { tests := s.GetTests() if len(filters) != 0 { @@ -115,7 +142,7 @@ func execute(s suite.Suite, filters runtime.Filters) (runtime.Result, error) { return result, nil } -func readFile(filePath string, filName string) (suite.Suite, error) { +func readFile(filePath string, fileName string) (suite.Suite, error) { s := suite.Suite{} f, err := os.Stat(filePath) @@ -132,7 +159,7 @@ func readFile(filePath string, filName string) (suite.Suite, error) { return s, err } - s = suite.ParseYAML(content, filName) + s = suite.ParseYAML(content, fileName) return s, nil } From 9d02a5df08e0e11d25d240f2ee64caac7dc41c45 Mon Sep 17 00:00:00 2001 From: Dylan Date: Sun, 13 Sep 2020 20:30:19 -0400 Subject: [PATCH 2/5] add unit and integration test --- commander_unix.yaml | 12 ++++++++++++ go.mod | 1 + go.sum | 6 ++++++ integration/unix/remote_http.yaml | 5 +++++ pkg/app/test_command.go | 6 +++--- pkg/app/test_command_test.go | 20 ++++++++++++++++++++ 6 files changed, 47 insertions(+), 3 deletions(-) create mode 100644 integration/unix/remote_http.yaml diff --git a/commander_unix.yaml b/commander_unix.yaml index 248dfed7..b5842a21 100644 --- a/commander_unix.yaml +++ b/commander_unix.yaml @@ -123,3 +123,15 @@ tests: ✓ [local] a ✓ [local] b exit-code: 0 + + it should execute test from url: + config: + env: + USER: from_parent + command: ./commander test https://raw.githubusercontent.com/commander-cli/commander/master/integration/unix/commander_test.yaml + stdout: + contains: + - ✓ [local] it should exit with error code + - "- [local] it should skip, was skipped" + line-count: 17 + exit-code: 0 \ No newline at end of file diff --git a/go.mod b/go.mod index 592793ce..45eef86a 100644 --- a/go.mod +++ b/go.mod @@ -20,6 +20,7 @@ require ( golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2 golang.org/x/net v0.0.0-20191014212845-da9a3fd4c582 // indirect gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15 // indirect + gopkg.in/h2non/gock.v1 v1.0.15 gopkg.in/yaml.v2 v2.2.4 ) diff --git a/go.sum b/go.sum index 77f7ef61..aa5241e0 100644 --- a/go.sum +++ b/go.sum @@ -18,6 +18,8 @@ github.com/docker/go-connections v0.4.0 h1:El9xVISelRB7BuFusrZozjnkIM5YnzCViNKoh github.com/docker/go-connections v0.4.0/go.mod h1:Gbd7IOopHjR8Iph03tsViu4nIes5XhDvyHbTtUxmeec= github.com/docker/go-units v0.4.0 h1:3uh0PgVws3nIA0Q+MwDC8yjEPf9zjRfZZWXZYDct3Tw= github.com/docker/go-units v0.4.0/go.mod h1:fgPhTUdO+D/Jk86RDLlptpiXQzgHJF7gydDDbaIK4Dk= +github.com/h2non/parth v0.0.0-20190131123155-b4df798d6542 h1:2VTzZjLZBgl62/EtslCrtky5vbi9dd7HrQPQIx6wqiw= +github.com/h2non/parth v0.0.0-20190131123155-b4df798d6542/go.mod h1:Ow0tF8D4Kplbc8s8sSb3V2oUCygFHVp8gC3Dn6U4MNI= github.com/konsorten/go-windows-terminal-sequences v1.0.1/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= github.com/kr/pretty v0.2.0 h1:s5hAObm+yFO5uHYt5dYjxi2rXrsnmRpJx4OYvIWUaQs= github.com/kr/pretty v0.2.0/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI= @@ -26,6 +28,8 @@ github.com/kr/text v0.1.0 h1:45sCR5RtlFHMR4UwH9sdQ5TC8v0qDQCHnXt+kaKSTVE= github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= github.com/logrusorgru/aurora v0.0.0-20181002194514-a7b3b318ed4e h1:9MlwzLdW7QSDrhDjFlsEYmxpFyIoXmYRon3dt0io31k= github.com/logrusorgru/aurora v0.0.0-20181002194514-a7b3b318ed4e/go.mod h1:7rIyQOR62GCctdiQpZ/zOJlFyk6y+94wXzv6RNZgaR4= +github.com/nbio/st v0.0.0-20140626010706-e9e8d9816f32 h1:W6apQkHrMkS0Muv8G/TipAy/FJl/rCYT0+EuS8+Z0z4= +github.com/nbio/st v0.0.0-20140626010706-e9e8d9816f32/go.mod h1:9wM+0iRr9ahx58uYLpLIr5fm8diHn0JbqRycJi6w0Ms= github.com/opencontainers/go-digest v1.0.0-rc1 h1:WzifXhOVOEOuFYOJAW6aQqW0TooG2iki3E3Ii+WN7gQ= github.com/opencontainers/go-digest v1.0.0-rc1/go.mod h1:cMLVZDEM3+U2I4VmLI6N8jQYUd2OVphdqWwCJHrFt2s= github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= @@ -61,6 +65,8 @@ golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15 h1:YR8cESwS4TdDjEe65xsg0ogRM/Nc3DYOhEAlW+xobZo= gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= +gopkg.in/h2non/gock.v1 v1.0.15 h1:SzLqcIlb/fDfg7UvukMpNcWsu7sI5tWwL+KCATZqks0= +gopkg.in/h2non/gock.v1 v1.0.15/go.mod h1:sX4zAkdYX1TRGJ2JY156cFspQn4yRWn6p9EMdODlynE= gopkg.in/yaml.v2 v2.2.2 h1:ZCJp+EgiOT7lHqUV2J862kp8Qj64Jo6az82+3Td9dZw= gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.2.4 h1:/eiJrUcujPVeJ3xlSWaiNi3uSVmDGBK1pDHUHAnao1I= diff --git a/integration/unix/remote_http.yaml b/integration/unix/remote_http.yaml new file mode 100644 index 00000000..f9c6b545 --- /dev/null +++ b/integration/unix/remote_http.yaml @@ -0,0 +1,5 @@ +tests: + hello world: + command: echo hello world + stdout: hello world + exit-code: 0 \ No newline at end of file diff --git a/pkg/app/test_command.go b/pkg/app/test_command.go index 85f74ba3..8d92aaae 100644 --- a/pkg/app/test_command.go +++ b/pkg/app/test_command.go @@ -18,8 +18,8 @@ var out output.OutputWriter // TestCommand executes the test argument // testPath is the path to the test suite config, it can be a dir or file -// ctx holds the command flags. If directory scanning is enabled with --dir it is -// not supported to filter tests, therefore testFilterTitle is an empty string +// ctx holds the command flags. If directory scanning is enabled with --dir, +// test filtering is not supported func TestCommand(testPath string, ctx TestCommandContext) error { if ctx.Verbose { log.SetOutput(os.Stdout) @@ -72,7 +72,7 @@ func testDir(directory string, filters runtime.Filters) (runtime.Result, error) result := runtime.Result{} files, err := ioutil.ReadDir(directory) if err != nil { - return result, fmt.Errorf(err.Error()) + return result, fmt.Errorf("Error: Input is not a directory") } for _, f := range files { diff --git a/pkg/app/test_command_test.go b/pkg/app/test_command_test.go index 904a8a49..217b70a5 100644 --- a/pkg/app/test_command_test.go +++ b/pkg/app/test_command_test.go @@ -12,6 +12,7 @@ import ( commanderRuntime "github.com/commander-cli/commander/pkg/runtime" "github.com/stretchr/testify/assert" + "gopkg.in/h2non/gock.v1" ) func Test_TestCommand_Verbose(t *testing.T) { @@ -68,6 +69,25 @@ func Test_TestCommand_Dir(t *testing.T) { } } +func Test_TestCommand_Http(t *testing.T) { + defer gock.Off() + + gock.New("http://foo.com"). + Get("/bar"). + Reply(200). + BodyString(` +tests: + echo hello: + exit-code: 0 +`) + + out := captureOutput(func() { + TestCommand("http://foo.com/bar", TestCommandContext{Verbose: true}) + }) + + assert.Contains(t, out, "✓ [local] echo hello") +} + func Test_ConvergeResults(t *testing.T) { duration, _ := time.ParseDuration("5s") From f53aa81680cd7b45abe295b64d7ae2b68d9d1497 Mon Sep 17 00:00:00 2001 From: Dylan Date: Sun, 13 Sep 2020 21:01:48 -0400 Subject: [PATCH 3/5] add tests for invalid inputs --- pkg/app/test_command_test.go | 20 ++++++++++++++++++++ 1 file changed, 20 insertions(+) diff --git a/pkg/app/test_command_test.go b/pkg/app/test_command_test.go index 217b70a5..66b82d22 100644 --- a/pkg/app/test_command_test.go +++ b/pkg/app/test_command_test.go @@ -69,6 +69,16 @@ func Test_TestCommand_Dir(t *testing.T) { } } +func Test_TestCommand_Dir_Err(t *testing.T) { + err := TestCommand("http://foo.com/bar", TestCommandContext{Dir: true}) + + if runtime.GOOS == "windows" { + assert.Contains(t, err.Error(), "Error: Input is not a directory") + } else { + assert.Equal(t, "Error: Input is not a directory", err.Error()) + } +} + func Test_TestCommand_Http(t *testing.T) { defer gock.Off() @@ -88,6 +98,16 @@ tests: assert.Contains(t, out, "✓ [local] echo hello") } +func Test_TestCommand_Http_Err(t *testing.T) { + err := TestCommand("http://error/not/a/url", TestCommandContext{Dir: false}) + + if runtime.GOOS == "windows" { + assert.NotNil(t, err) + } else { + assert.NotNil(t, err) + } +} + func Test_ConvergeResults(t *testing.T) { duration, _ := time.ParseDuration("5s") From 451b0b6d1c8d710a7dca38de3ba003ed53a3efb4 Mon Sep 17 00:00:00 2001 From: Dylan Date: Tue, 15 Sep 2020 19:27:58 -0400 Subject: [PATCH 4/5] update changeloog and add readme docs --- CHANGELOG.md | 4 ++++ README.md | 3 +++ 2 files changed, 7 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 43246ef9..83468b67 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,3 +1,7 @@ +# v2.4.0 + + - Add ability to test suite from a url + # v2.3.0 - Preserve test execution order alphabetical order diff --git a/README.md b/README.md index 20c4b1de..6bba6136 100644 --- a/README.md +++ b/README.md @@ -203,6 +203,9 @@ $ ./commander test /tmp/test.yaml # Execute a single test $ ./commander test /tmp/test.yaml "my test" +# Execute suite from url +$ ./commander test https://raw.githubusercontent.com/commander-cli/commander/master/integration/unix/commander_test.yaml + # Execute suites within a test directory $ ./commander test --dir /tmp ``` From f32aa9fd00cf253c05dcb138da542c64537c787c Mon Sep 17 00:00:00 2001 From: Dylan Date: Sat, 26 Sep 2020 17:39:13 -0400 Subject: [PATCH 5/5] vendoring gock --- vendor/github.com/h2non/parth/LICENSE | 22 ++ vendor/github.com/h2non/parth/README.md | 196 +++++++++++ vendor/github.com/h2non/parth/go.mod | 1 + vendor/github.com/h2non/parth/parth.go | 349 ++++++++++++++++++++ vendor/github.com/h2non/parth/segindex.go | 118 +++++++ vendor/github.com/h2non/parth/segtostr.go | 305 +++++++++++++++++ vendor/gopkg.in/h2non/gock.v1/.editorconfig | 12 + vendor/gopkg.in/h2non/gock.v1/.gitignore | 30 ++ vendor/gopkg.in/h2non/gock.v1/.travis.yml | 26 ++ vendor/gopkg.in/h2non/gock.v1/History.md | 124 +++++++ vendor/gopkg.in/h2non/gock.v1/LICENSE | 24 ++ vendor/gopkg.in/h2non/gock.v1/README.md | 341 +++++++++++++++++++ vendor/gopkg.in/h2non/gock.v1/go.mod | 6 + vendor/gopkg.in/h2non/gock.v1/gock.go | 176 ++++++++++ vendor/gopkg.in/h2non/gock.v1/matcher.go | 135 ++++++++ vendor/gopkg.in/h2non/gock.v1/matchers.go | 256 ++++++++++++++ vendor/gopkg.in/h2non/gock.v1/mock.go | 146 ++++++++ vendor/gopkg.in/h2non/gock.v1/options.go | 8 + vendor/gopkg.in/h2non/gock.v1/request.go | 325 ++++++++++++++++++ vendor/gopkg.in/h2non/gock.v1/responder.go | 87 +++++ vendor/gopkg.in/h2non/gock.v1/response.go | 186 +++++++++++ vendor/gopkg.in/h2non/gock.v1/store.go | 100 ++++++ vendor/gopkg.in/h2non/gock.v1/transport.go | 112 +++++++ vendor/gopkg.in/h2non/gock.v1/version.go | 4 + vendor/modules.txt | 4 + 25 files changed, 3093 insertions(+) create mode 100644 vendor/github.com/h2non/parth/LICENSE create mode 100644 vendor/github.com/h2non/parth/README.md create mode 100644 vendor/github.com/h2non/parth/go.mod create mode 100644 vendor/github.com/h2non/parth/parth.go create mode 100644 vendor/github.com/h2non/parth/segindex.go create mode 100644 vendor/github.com/h2non/parth/segtostr.go create mode 100644 vendor/gopkg.in/h2non/gock.v1/.editorconfig create mode 100644 vendor/gopkg.in/h2non/gock.v1/.gitignore create mode 100644 vendor/gopkg.in/h2non/gock.v1/.travis.yml create mode 100644 vendor/gopkg.in/h2non/gock.v1/History.md create mode 100644 vendor/gopkg.in/h2non/gock.v1/LICENSE create mode 100644 vendor/gopkg.in/h2non/gock.v1/README.md create mode 100644 vendor/gopkg.in/h2non/gock.v1/go.mod create mode 100644 vendor/gopkg.in/h2non/gock.v1/gock.go create mode 100644 vendor/gopkg.in/h2non/gock.v1/matcher.go create mode 100644 vendor/gopkg.in/h2non/gock.v1/matchers.go create mode 100644 vendor/gopkg.in/h2non/gock.v1/mock.go create mode 100644 vendor/gopkg.in/h2non/gock.v1/options.go create mode 100644 vendor/gopkg.in/h2non/gock.v1/request.go create mode 100644 vendor/gopkg.in/h2non/gock.v1/responder.go create mode 100644 vendor/gopkg.in/h2non/gock.v1/response.go create mode 100644 vendor/gopkg.in/h2non/gock.v1/store.go create mode 100644 vendor/gopkg.in/h2non/gock.v1/transport.go create mode 100644 vendor/gopkg.in/h2non/gock.v1/version.go diff --git a/vendor/github.com/h2non/parth/LICENSE b/vendor/github.com/h2non/parth/LICENSE new file mode 100644 index 00000000..a5325703 --- /dev/null +++ b/vendor/github.com/h2non/parth/LICENSE @@ -0,0 +1,22 @@ +The MIT License (MIT) + +Copyright (c) 2018 codemodus + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. + diff --git a/vendor/github.com/h2non/parth/README.md b/vendor/github.com/h2non/parth/README.md new file mode 100644 index 00000000..f8dd8df8 --- /dev/null +++ b/vendor/github.com/h2non/parth/README.md @@ -0,0 +1,196 @@ +# parth + + go get -u github.com/h2non/parth + +Package parth provides path parsing for segment unmarshaling and slicing. In +other words, parth provides simple and flexible access to (URL) path parameters. + +Along with string, all basic non-alias types are supported. An interface is +available for implementation by user-defined types. When handling an int, uint, +or float of any size, the first valid value within the specified segment will be +used. + +## Usage + +```go +Variables +func Segment(path string, i int, v interface{}) error +func Sequent(path, key string, v interface{}) error +func Span(path string, i, j int) (string, error) +func SubSeg(path, key string, i int, v interface{}) error +func SubSpan(path, key string, i, j int) (string, error) +type Parth + func New(path string) *Parth + func NewBySpan(path string, i, j int) *Parth + func NewBySubSpan(path, key string, i, j int) *Parth + func (p *Parth) Err() error + func (p *Parth) Segment(i int, v interface{}) + func (p *Parth) Sequent(key string, v interface{}) + func (p *Parth) Span(i, j int) string + func (p *Parth) SubSeg(key string, i int, v interface{}) + func (p *Parth) SubSpan(key string, i, j int) string +type Unmarshaler +``` + +### Setup ("By Index") + +```go +import ( + "fmt" + + "github.com/codemodus/parth" +) + +func handler(w http.ResponseWriter, r *http.Request) { + var s string + if err := parth.Segment(r.URL.Path, 4, &s); err != nil { + fmt.Fprintln(os.Stderr, err) + } + + fmt.Println(r.URL.Path) + fmt.Printf("%v (%T)\n", s, s) + + // Output: + // /some/path/things/42/others/3 + // others (string) +} +``` + +### Setup ("By Key") + +```go +import ( + "fmt" + + "github.com/codemodus/parth" +) + +func handler(w http.ResponseWriter, r *http.Request) { + var i int64 + if err := parth.Sequent(r.URL.Path, "things", &i); err != nil { + fmt.Fprintln(os.Stderr, err) + } + + fmt.Println(r.URL.Path) + fmt.Printf("%v (%T)\n", i, i) + + // Output: + // /some/path/things/42/others/3 + // 42 (int64) +} +``` + +### Setup (Parth Type) + +```go +import ( + "fmt" + + "github.com/codemodus/parth" +) + +func handler(w http.ResponseWriter, r *http.Request) { + var s string + var f float32 + + p := parth.New(r.URL.Path) + p.Segment(2, &s) + p.SubSeg("key", 1, &f) + if err := p.Err(); err != nil { + fmt.Fprintln(os.Stderr, err) + } + + fmt.Println(r.URL.Path) + fmt.Printf("%v (%T)\n", s, s) + fmt.Printf("%v (%T)\n", f, f) + + // Output: + // /zero/one/two/key/four/5.5/six + // two (string) + // 5.5 (float32) +} +``` + +### Setup (Unmarshaler) + +```go +import ( + "fmt" + + "github.com/codemodus/parth" +) + +func handler(w http.ResponseWriter, r *http.Request) { + /* + type mytype []byte + + func (m *mytype) UnmarshalSegment(seg string) error { + *m = []byte(seg) + } + */ + + var m mytype + if err := parth.Segment(r.URL.Path, 4, &m); err != nil { + fmt.Fprintln(os.Stderr, err) + } + + fmt.Println(r.URL.Path) + fmt.Printf("%v == %q (%T)\n", m, m, m) + + // Output: + // /zero/one/two/key/four/5.5/six + // [102 111 117 114] == "four" (mypkg.mytype) +} +``` + +## More Info + +### Keep Using http.HandlerFunc And Minimize context.Context Usage + +The most obvious use case for parth is when working with any URL path such as +the one found at http.Request.URL.Path. parth is fast enough that it can be used +multiple times in place of a single use of similar router-parameter schemes or +even context.Context. There is no need to use an alternate http handler function +definition in order to pass data that is already being passed. The http.Request +type already holds URL data and parth is great at handling it. Additionally, +parth takes care of parsing selected path segments into the types actually +needed. Parth not only does more, it's usually faster and less intrusive than +the alternatives. + +### Indexes + +If an index is negative, the negative count begins with the last segment. +Providing a 0 for the second index is a special case which acts as an alias for +the end of the path. An error is returned if: 1. Any index is out of range of +the path; 2. When there are two indexes, the first index does not precede the +second index. + +### Keys + +If a key is involved, functions will only handle the portion of the path +subsequent to the provided key. An error is returned if the key cannot be found +in the path. + +### First Whole, First Decimal (Restated - Important!) + +When handling an int, uint, or float of any size, the first valid value within +the specified segment will be used. + +## Documentation + +View the [GoDoc](http://godoc.org/github.com/codemodus/parth) + +## Benchmarks + + Go 1.11 + benchmark iter time/iter bytes alloc allocs + --------- ---- --------- ----------- ------ + BenchmarkSegmentString-8 30000000 39.60 ns/op 0 B/op 0 allocs/op + BenchmarkSegmentInt-8 20000000 65.60 ns/op 0 B/op 0 allocs/op + BenchmarkSegmentIntNegIndex-8 20000000 86.60 ns/op 0 B/op 0 allocs/op + BenchmarkSpan-8 100000000 18.20 ns/op 0 B/op 0 allocs/op + BenchmarkStdlibSegmentString-8 5000000 454.00 ns/op 50 B/op 2 allocs/op + BenchmarkStdlibSegmentInt-8 3000000 526.00 ns/op 50 B/op 2 allocs/op + BenchmarkStdlibSpan-8 3000000 518.00 ns/op 69 B/op 2 allocs/op + BenchmarkContextLookupSetGet-8 1000000 1984.00 ns/op 480 B/op 6 allocs/op + diff --git a/vendor/github.com/h2non/parth/go.mod b/vendor/github.com/h2non/parth/go.mod new file mode 100644 index 00000000..c25e6633 --- /dev/null +++ b/vendor/github.com/h2non/parth/go.mod @@ -0,0 +1 @@ +module github.com/h2non/parth diff --git a/vendor/github.com/h2non/parth/parth.go b/vendor/github.com/h2non/parth/parth.go new file mode 100644 index 00000000..3e64d8fe --- /dev/null +++ b/vendor/github.com/h2non/parth/parth.go @@ -0,0 +1,349 @@ +// Package parth provides path parsing for segment unmarshaling and slicing. In +// other words, parth provides simple and flexible access to (URL) path +// parameters. +// +// Along with string, all basic non-alias types are supported. An interface is +// available for implementation by user-defined types. When handling an int, +// uint, or float of any size, the first valid value within the specified +// segment will be used. +package parth + +import ( + "errors" +) + +// Unmarshaler is the interface implemented by types that can unmarshal a path +// segment representation of themselves. It is safe to assume that the segment +// data will not include slashes. +type Unmarshaler interface { + UnmarshalSegment(string) error +} + +// Err{Name} values facilitate error identification. +var ( + ErrUnknownType = errors.New("unknown type provided") + + ErrFirstSegNotFound = errors.New("first segment not found by index") + ErrLastSegNotFound = errors.New("last segment not found by index") + ErrSegOrderReversed = errors.New("first segment must precede last segment") + ErrKeySegNotFound = errors.New("segment not found by key") + + ErrDataUnparsable = errors.New("data cannot be parsed") +) + +// Segment locates the path segment indicated by the index i and unmarshals it +// into the provided type v. If the index is negative, the negative count +// begins with the last segment. An error is returned if: 1. The type is not a +// pointer to an instance of one of the basic non-alias types and does not +// implement the Unmarshaler interface; 2. The index is out of range of the +// path; 3. The located path segment data cannot be parsed as the provided type +// or if an error is returned when using a provided Unmarshaler implementation. +func Segment(path string, i int, v interface{}) error { //nolint + var err error + + switch v := v.(type) { + case *bool: + *v, err = segmentToBool(path, i) + + case *float32: + var f float64 + f, err = segmentToFloatN(path, i, 32) + *v = float32(f) + + case *float64: + *v, err = segmentToFloatN(path, i, 64) + + case *int: + var n int64 + n, err = segmentToIntN(path, i, 0) + *v = int(n) + + case *int16: + var n int64 + n, err = segmentToIntN(path, i, 16) + *v = int16(n) + + case *int32: + var n int64 + n, err = segmentToIntN(path, i, 32) + *v = int32(n) + + case *int64: + *v, err = segmentToIntN(path, i, 64) + + case *int8: + var n int64 + n, err = segmentToIntN(path, i, 8) + *v = int8(n) + + case *string: + *v, err = segmentToString(path, i) + + case *uint: + var n uint64 + n, err = segmentToUintN(path, i, 0) + *v = uint(n) + + case *uint16: + var n uint64 + n, err = segmentToUintN(path, i, 16) + *v = uint16(n) + + case *uint32: + var n uint64 + n, err = segmentToUintN(path, i, 32) + *v = uint32(n) + + case *uint64: + *v, err = segmentToUintN(path, i, 64) + + case *uint8: + var n uint64 + n, err = segmentToUintN(path, i, 8) + *v = uint8(n) + + case Unmarshaler: + var s string + s, err = segmentToString(path, i) + if err == nil { + err = v.UnmarshalSegment(s) + } + + default: + err = ErrUnknownType + } + + return err +} + +// Sequent is similar to Segment, but uses a key to locate a segment and then +// unmarshal the subsequent segment. It is a simple wrapper over SubSeg with an +// index of 0. +func Sequent(path, key string, v interface{}) error { + return SubSeg(path, key, 0, v) +} + +// Span returns the path segments between two segment indexes i and j including +// the first segment. If an index is negative, the negative count begins with +// the last segment. Providing a 0 for the last index j is a special case which +// acts as an alias for the end of the path. If the first segment does not begin +// with a slash and it is part of the requested span, no slash will be added. An +// error is returned if: 1. Either index is out of range of the path; 2. The +// first index i does not precede the last index j. +func Span(path string, i, j int) (string, error) { + var f, l int + var ok bool + + if i < 0 { + f, ok = segStartIndexFromEnd(path, i) + } else { + f, ok = segStartIndexFromStart(path, i) + } + if !ok { + return "", ErrFirstSegNotFound + } + + if j > 0 { + l, ok = segEndIndexFromStart(path, j) + } else { + l, ok = segEndIndexFromEnd(path, j) + } + if !ok { + return "", ErrLastSegNotFound + } + + if f == l { + return "", nil + } + + if f > l { + return "", ErrSegOrderReversed + } + + return path[f:l], nil +} + +// SubSeg is similar to Segment, but only handles the portion of the path +// subsequent to the provided key. For example, to access the segment +// immediately after a key, an index of 0 should be provided (see Sequent). An +// error is returned if the key cannot be found in the path. +func SubSeg(path, key string, i int, v interface{}) error { //nolint + var err error + + switch v := v.(type) { + case *bool: + *v, err = subSegToBool(path, key, i) + + case *float32: + var f float64 + f, err = subSegToFloatN(path, key, i, 32) + *v = float32(f) + + case *float64: + *v, err = subSegToFloatN(path, key, i, 64) + + case *int: + var n int64 + n, err = subSegToIntN(path, key, i, 0) + *v = int(n) + + case *int16: + var n int64 + n, err = subSegToIntN(path, key, i, 16) + *v = int16(n) + + case *int32: + var n int64 + n, err = subSegToIntN(path, key, i, 32) + *v = int32(n) + + case *int64: + *v, err = subSegToIntN(path, key, i, 64) + + case *int8: + var n int64 + n, err = subSegToIntN(path, key, i, 8) + *v = int8(n) + + case *string: + *v, err = subSegToString(path, key, i) + + case *uint: + var n uint64 + n, err = subSegToUintN(path, key, i, 0) + *v = uint(n) + + case *uint16: + var n uint64 + n, err = subSegToUintN(path, key, i, 16) + *v = uint16(n) + + case *uint32: + var n uint64 + n, err = subSegToUintN(path, key, i, 32) + *v = uint32(n) + + case *uint64: + *v, err = subSegToUintN(path, key, i, 64) + + case *uint8: + var n uint64 + n, err = subSegToUintN(path, key, i, 8) + *v = uint8(n) + + case Unmarshaler: + var s string + s, err = subSegToString(path, key, i) + if err == nil { + err = v.UnmarshalSegment(s) + } + + default: + err = ErrUnknownType + } + + return err +} + +// SubSpan is similar to Span, but only handles the portion of the path +// subsequent to the provided key. An error is returned if the key cannot be +// found in the path. +func SubSpan(path, key string, i, j int) (string, error) { + si, ok := segIndexByKey(path, key) + if !ok { + return "", ErrKeySegNotFound + } + + if i >= 0 { + i++ + } + if j > 0 { + j++ + } + + s, err := Span(path[si:], i, j) + if err != nil { + return "", err + } + + return s, nil +} + +// Parth manages path and error data for processing a single path multiple +// times while error checking only once. Only the first encountered error is +// stored as all subsequent calls to Parth methods that can error are elided. +type Parth struct { + path string + err error +} + +// New constructs a pointer to an instance of Parth around the provided path. +func New(path string) *Parth { + return &Parth{path: path} +} + +// NewBySpan constructs a pointer to an instance of Parth after preprocessing +// the provided path with Span. +func NewBySpan(path string, i, j int) *Parth { + s, err := Span(path, i, j) + return &Parth{s, err} +} + +// NewBySubSpan constructs a pointer to an instance of Parth after +// preprocessing the provided path with SubSpan. +func NewBySubSpan(path, key string, i, j int) *Parth { + s, err := SubSpan(path, key, i, j) + return &Parth{s, err} +} + +// Err returns the first error encountered by the *Parth receiver. +func (p *Parth) Err() error { + return p.err +} + +// Segment operates the same as the package-level function Segment. +func (p *Parth) Segment(i int, v interface{}) { + if p.err != nil { + return + } + + p.err = Segment(p.path, i, v) +} + +// Sequent operates the same as the package-level function Sequent. +func (p *Parth) Sequent(key string, v interface{}) { + p.SubSeg(key, 0, v) +} + +// Span operates the same as the package-level function Span. +func (p *Parth) Span(i, j int) string { + if p.err != nil { + return "" + } + + s, err := Span(p.path, i, j) + p.err = err + + return s +} + +// SubSeg operates the same as the package-level function SubSeg. +func (p *Parth) SubSeg(key string, i int, v interface{}) { + if p.err != nil { + return + } + + p.err = SubSeg(p.path, key, i, v) +} + +// SubSpan operates the same as the package-level function SubSpan. +func (p *Parth) SubSpan(key string, i, j int) string { + if p.err != nil { + return "" + } + + s, err := SubSpan(p.path, key, i, j) + p.err = err + + return s +} diff --git a/vendor/github.com/h2non/parth/segindex.go b/vendor/github.com/h2non/parth/segindex.go new file mode 100644 index 00000000..77861f99 --- /dev/null +++ b/vendor/github.com/h2non/parth/segindex.go @@ -0,0 +1,118 @@ +package parth + +func segStartIndexFromStart(path string, seg int) (int, bool) { + if seg < 0 { + return 0, false + } + + for n, ct := 0, 0; n < len(path); n++ { + if n > 0 && path[n] == '/' { + ct++ + } + + if ct == seg { + return n, true + } + } + + return 0, false +} + +func segStartIndexFromEnd(path string, seg int) (int, bool) { + if seg > -1 { + return 0, false + } + + for n, ct := len(path)-1, 0; n >= 0; n-- { + if path[n] == '/' || n == 0 { + ct-- + } + + if ct == seg { + return n, true + } + } + + return 0, false +} + +func segEndIndexFromStart(path string, seg int) (int, bool) { + if seg < 1 { + return 0, false + } + + for n, ct := 0, 0; n < len(path); n++ { + if path[n] == '/' && n > 0 { + ct++ + } + + if ct == seg { + return n, true + } + + if n+1 == len(path) && ct+1 == seg { + return n + 1, true + } + } + + return 0, false +} + +func segEndIndexFromEnd(path string, seg int) (int, bool) { + if seg > 0 { + return 0, false + } + + if seg == 0 { + return len(path), true + } + + if len(path) == 1 && path[0] == '/' { + return 0, true + } + + for n, ct := len(path)-1, 0; n >= 0; n-- { + if n == 0 || path[n] == '/' { + ct-- + } + + if ct == seg { + return n, true + } + + } + + return 0, false +} + +func segIndexByKey(path, key string) (int, bool) { //nolint + if path == "" || key == "" { + return 0, false + } + + for n := 0; n < len(path); n++ { + si, ok := segStartIndexFromStart(path, n) + if !ok { + return 0, false + } + + if len(path[si:]) == len(key)+1 { + if path[si+1:] == key { + return si, true + } + + return 0, false + } + + tmpEI, ok := segStartIndexFromStart(path[si:], 1) + if !ok { + return 0, false + } + + if path[si+1:tmpEI+si] == key || n == 0 && path[0] != '/' && path[si:tmpEI+si] == key { + return si, true + } + } + + return 0, false +} diff --git a/vendor/github.com/h2non/parth/segtostr.go b/vendor/github.com/h2non/parth/segtostr.go new file mode 100644 index 00000000..6e16a911 --- /dev/null +++ b/vendor/github.com/h2non/parth/segtostr.go @@ -0,0 +1,305 @@ +package parth + +import ( + "strconv" + "unicode" +) + +func segmentToBool(path string, i int) (bool, error) { + s, err := segmentToString(path, i) + if err != nil { + return false, err + } + + v, err := strconv.ParseBool(s) + if err != nil { + return false, ErrDataUnparsable + } + + return v, nil +} + +func segmentToFloatN(path string, i, size int) (float64, error) { + ss, err := segmentToString(path, i) + if err != nil { + return 0.0, err + } + + s, ok := firstFloatFromString(ss) + if !ok { + return 0.0, err + } + + v, err := strconv.ParseFloat(s, size) + if err != nil { + return 0.0, ErrDataUnparsable + } + + return v, nil +} + +func segmentToIntN(path string, i, size int) (int64, error) { + ss, err := segmentToString(path, i) + if err != nil { + return 0, err + } + + s, ok := firstIntFromString(ss) + if !ok { + return 0, ErrDataUnparsable + } + + v, err := strconv.ParseInt(s, 10, size) + if err != nil { + return 0, ErrDataUnparsable + } + + return v, nil +} + +func segmentToString(path string, i int) (string, error) { + j := i + 1 + if i < 0 { + i-- + } + + s, err := Span(path, i, j) + if err != nil { + return "", err + } + + if s[0] == '/' { + s = s[1:] + } + + return s, nil +} + +func segmentToUintN(path string, i, size int) (uint64, error) { + ss, err := segmentToString(path, i) + if err != nil { + return 0, err + } + + s, ok := firstUintFromString(ss) + if !ok { + return 0, ErrDataUnparsable + } + + v, err := strconv.ParseUint(s, 10, size) + if err != nil { + return 0, ErrDataUnparsable + } + + return v, nil +} + +func subSegToBool(path, key string, i int) (bool, error) { + s, err := subSegToString(path, key, i) + if err != nil { + return false, err + } + + v, err := strconv.ParseBool(s) + if err != nil { + return false, ErrDataUnparsable + } + + return v, nil +} + +func subSegToFloatN(path, key string, i, size int) (float64, error) { + ss, err := subSegToString(path, key, i) + if err != nil { + return 0.0, err + } + + s, ok := firstFloatFromString(ss) + if !ok { + return 0.0, ErrDataUnparsable + } + + v, err := strconv.ParseFloat(s, size) + if err != nil { + return 0.0, ErrDataUnparsable + } + + return v, nil +} + +func subSegToIntN(path, key string, i, size int) (int64, error) { + ss, err := subSegToString(path, key, i) + if err != nil { + return 0, err + } + + s, ok := firstIntFromString(ss) + if !ok { + return 0, ErrDataUnparsable + } + + v, err := strconv.ParseInt(s, 10, size) + if err != nil { + return 0, ErrDataUnparsable + } + + return v, nil +} + +func subSegToString(path, key string, i int) (string, error) { + ki, ok := segIndexByKey(path, key) + if !ok { + return "", ErrKeySegNotFound + } + + i++ + + s, err := segmentToString(path[ki:], i) + if err != nil { + return "", err + } + + return s, nil +} + +func subSegToUintN(path, key string, i, size int) (uint64, error) { + ss, err := subSegToString(path, key, i) + if err != nil { + return 0, err + } + + s, ok := firstUintFromString(ss) + if !ok { + return 0, ErrDataUnparsable + } + + v, err := strconv.ParseUint(s, 10, size) + if err != nil { + return 0, ErrDataUnparsable + } + + return v, nil +} + +func firstUintFromString(s string) (string, bool) { + ind, l := 0, 0 + + for n := 0; n < len(s); n++ { + if unicode.IsDigit(rune(s[n])) { + if l == 0 { + ind = n + } + + l++ + } else { + if l == 0 && s[n] == '.' { + if n+1 < len(s) && unicode.IsDigit(rune(s[n+1])) { + return "0", true + } + + break + } + + if l > 0 { + break + } + } + } + + if l == 0 { + return "", false + } + + return s[ind : ind+l], true +} + +func firstIntFromString(s string) (string, bool) { //nolint + ind, l := 0, 0 + + for n := 0; n < len(s); n++ { + if unicode.IsDigit(rune(s[n])) { + if l == 0 { + ind = n + } + + l++ + } else if s[n] == '-' { + if l == 0 { + ind = n + l++ + } else { + break + } + } else { + if l == 0 && s[n] == '.' { + if n+1 < len(s) && unicode.IsDigit(rune(s[n+1])) { + return "0", true + } + + break + } + + if l > 0 { + break + } + } + } + + if l == 0 { + return "", false + } + + return s[ind : ind+l], true +} + +func firstFloatFromString(s string) (string, bool) { //nolint + c, ind, l := 0, 0, 0 + + for n := 0; n < len(s); n++ { + if unicode.IsDigit(rune(s[n])) { + if l == 0 { + ind = n + } + + l++ + } else if s[n] == '-' { + if l == 0 { + ind = n + l++ + } else { + break + } + } else if s[n] == '.' { + if l == 0 { + ind = n + } + + if c > 0 { + break + } + + l++ + c++ + } else if s[n] == 'e' && l > 0 && n+1 < len(s) && s[n+1] == '+' { + l++ + } else if s[n] == '+' && l > 0 && s[n-1] == 'e' { + if n+1 < len(s) && unicode.IsDigit(rune(s[n+1])) { + l++ + continue + } + + l-- + break + } else { + if l > 0 { + break + } + } + } + + if l == 0 || s[ind:ind+l] == "." { + return "", false + } + + return s[ind : ind+l], true +} diff --git a/vendor/gopkg.in/h2non/gock.v1/.editorconfig b/vendor/gopkg.in/h2non/gock.v1/.editorconfig new file mode 100644 index 00000000..b570cb16 --- /dev/null +++ b/vendor/gopkg.in/h2non/gock.v1/.editorconfig @@ -0,0 +1,12 @@ +root = true + +[*] +indent_style = tab +indent_size = 2 +end_of_line = lf +charset = utf-8 +trim_trailing_whitespace = true +insert_final_newline = true + +[*.md] +trim_trailing_whitespace = false diff --git a/vendor/gopkg.in/h2non/gock.v1/.gitignore b/vendor/gopkg.in/h2non/gock.v1/.gitignore new file mode 100644 index 00000000..173b0298 --- /dev/null +++ b/vendor/gopkg.in/h2non/gock.v1/.gitignore @@ -0,0 +1,30 @@ +# Compiled Object files, Static and Dynamic libs (Shared Objects) +*.o +*.a +*.so +go.sum + +# Folders +_obj +_test + +# Architecture specific extensions/prefixes +*.[568vq] +[568vq].out + +*.cgo1.go +*.cgo2.c +_cgo_defun.c +_cgo_gotypes.go +_cgo_export.* + +_testmain.go + +*.exe +*.test +*.prof + +.idea/ +*.iml +*.out +*.tmp diff --git a/vendor/gopkg.in/h2non/gock.v1/.travis.yml b/vendor/gopkg.in/h2non/gock.v1/.travis.yml new file mode 100644 index 00000000..f12f2c5f --- /dev/null +++ b/vendor/gopkg.in/h2non/gock.v1/.travis.yml @@ -0,0 +1,26 @@ +language: go + +go: + - "1.11" + - "1.10" + - "stable" + +before_install: + - go get -u github.com/nbio/st + - go get -u github.com/codemodus/parth + - go get -u gopkg.in/h2non/gentleman.v1 + - go get -u -v github.com/axw/gocov/gocov + - go get -u -v github.com/mattn/goveralls + - go get -v -u golang.org/x/lint/golint + - mkdir -p $GOPATH/src/gopkg.in/h2non/gock.v1 + - cp -r . $GOPATH/src/gopkg.in/h2non/gock.v1 + +script: + - diff -u <(echo -n) <(gofmt -s -d ./) + - diff -u <(echo -n) <(go vet ./...) + - diff -u <(echo -n) <(golint ./...) + - go test -v -race -covermode=atomic -coverprofile=coverage.out + - go test ./_examples/*/ -v -race + +after_success: + - goveralls -coverprofile=coverage.out -service=travis-ci -repotoken $COVERALLS_TOKEN diff --git a/vendor/gopkg.in/h2non/gock.v1/History.md b/vendor/gopkg.in/h2non/gock.v1/History.md new file mode 100644 index 00000000..e9e4172e --- /dev/null +++ b/vendor/gopkg.in/h2non/gock.v1/History.md @@ -0,0 +1,124 @@ + +## v1.0.15 / 2019-07-03 + + * NewMatcher() will now return objects that completely separate one another. (#55) + * add request Options (#49) + * fix typo: function -> func (#52) + * feat(docs): change note + * feat(docs): add net/http support + * Add Basic Auth (#47) + * Update LICENSE (#46) + +## v1.0.14 / 2019-01-31 + + * feat(version): bump to v1.0.14 + * feat: add go.mod + +## v1.0.13 / 2019-01-30 + + * Add PathParam matcher (#42) + +## v1.0.12 / 2018-11-13 + + * Fix possible data race. (#41) + +## v1.0.11 / 2018-10-29 + + * Do not reset response body (#40) + * refactor(travis): remove unsupported versions for golint based on Go release policy support + * feat(gock): add gock.Observe to support inspection of the outgoing intercepted HTTP traffic (#38) + +## v1.0.10 / 2018-09-09 + + * Support multiple response headers with same name #35 (#36) + +## v1.0.9 / 2018-06-14 + + * fix(url-encoding) add exact match test in MatchPath (#34) + * fix(travis): use string notation for Go versions + +## v1.0.8 / 2018-02-28 + + * chore(LICENSE): update year ;) + * feat(docs): add additional tips and examples + * feat(gock): ignore already intercepted http.Client + +## v1.0.7 / 2017-12-21 + + * Make MatchHost case insensitive. (#31) + * refactor(docs): remove codesponsor :( + * add example when request reply with error (#28) + * feat(docs): add sponsor ad + * Add example networking partially enabled (#23) + +## v1.0.6 / 2017-07-27 + + * fix(#23): mock transport deadlock + +## v1.0.5 / 2017-07-26 + + * feat(#25, #24): use content type only if missing while matching JSON/XML + * feat(#24): add CleanUnmatchedRequests() and OffAll() public functions + * feat(version): bump to v1.0.5 + * fix(store): use proper indent style + * fix(mutex): use different mutex for store + * feat(travis): add Go 1.8 CI support + +## v1.0.4 / 2017-02-14 + + * Update README to include most up to date version (#17) + * Update MatchBody() to compare if key + value pairs of JSON match regardless of order they are in. (#16) + * feat(examples): add new example for unmatch case + * refactor(docs): add pook reference + +## 1.0.3 / 14-11-2016 + +- feat(#13): adds `GetUnmatchedRequests()` and `HasUnmatchedRequests()` API functions. + +## 1.0.2 / 10-11-2016 + +- fix(#11): adds `Compression()` method for output HTTP traffic body compression processing and matching. + +## 1.0.1 / 07-09-2016 + +- fix(#9): missing URL query param matcher. + +## 1.0.0 / 19-04-2016 + +- feat(version): first major version release. + +## 0.1.6 / 19-04-2016 + +- fix(#7): if error configured, RoundTripper should reply with `nil` response. + +## 0.1.5 / 09-04-2016 + +- feat(#5): support `ReplyFunc` for convenience. + +## 0.1.4 / 16-03-2016 + +- feat(api): add `IsDone()` method. +- fix(responder): return mock error if present. +- feat(#4): support define request/response body from file disk. + +## 0.1.3 / 09-03-2016 + +- feat(matcher): add content type matcher helper method supporting aliases. +- feat(interceptor): add function to restore HTTP client transport. +- feat(matcher): add URL scheme matcher function. +- fix(request): ignore base slash path. +- feat(api): add Off() method for easier restore and clean up. +- feat(store): add public API for pending mocks. + +## 0.1.2 / 04-03-2016 + +- fix(matcher): body matchers no used by default. +- feat(matcher): add matcher factories for multiple cases. + +## 0.1.1 / 04-03-2016 + +- fix(params): persist query params accordingly. + +## 0.1.0 / 02-03-2016 + +- First release. diff --git a/vendor/gopkg.in/h2non/gock.v1/LICENSE b/vendor/gopkg.in/h2non/gock.v1/LICENSE new file mode 100644 index 00000000..1911b177 --- /dev/null +++ b/vendor/gopkg.in/h2non/gock.v1/LICENSE @@ -0,0 +1,24 @@ +The MIT License + +Copyright (c) 2016-2019 Tomas Aparicio + +Permission is hereby granted, free of charge, to any person +obtaining a copy of this software and associated documentation +files (the "Software"), to deal in the Software without +restriction, including without limitation the rights to use, +copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the +Software is furnished to do so, subject to the following +conditions: + +The above copyright notice and this permission notice shall be +included in all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, +EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES +OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND +NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT +HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, +WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING +FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR +OTHER DEALINGS IN THE SOFTWARE. diff --git a/vendor/gopkg.in/h2non/gock.v1/README.md b/vendor/gopkg.in/h2non/gock.v1/README.md new file mode 100644 index 00000000..cbc7e8fa --- /dev/null +++ b/vendor/gopkg.in/h2non/gock.v1/README.md @@ -0,0 +1,341 @@ +# 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) for `net/http` stdlib package. + +Heavily inspired by [nock](https://github.com/node-nock/nock). +There is also its Python port, [pook](https://github.com/h2non/pook). + +To get started, take a look to the [examples](#examples). + +## Features + +- Simple, expressive, fluent API. +- Semantic API DSL for declarative HTTP mock declarations. +- Built-in helpers for easy JSON/XML mocking. +- Supports persistent and volatile TTL-limited mocks. +- Full regular expressions capable HTTP request mock 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 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. + +#### Disable `gock` traffic interception once done + +In other to minimize potential side effects within your test code, it's a good practice +disabling `gock` once you are done with your HTTP testing logic. + +A Go idiomatic approach for doing this can be using it in a `defer` statement, such as: + +```go +func TestGock (t *testing.T) { + defer gock.Off() + + // ... my test code goes here +} +``` + +#### Intercept an `http.Client` just once + +You don't need to intercept multiple times the same `http.Client` instance. + +Just call `gock.InterceptClient(client)` once, typically at the beginning of your test scenarios. + +#### Restore an `http.Client` after interception + +**NOTE**: this is not required is you are using `http.DefaultClient` or `http.DefaultTransport`. + +As a good testing pattern, you should call `gock.RestoreClient(client)` after running your test scenario, typically as after clean up hook. + +You can also use a `defer` statement for doing it, as you do with `gock.Off()`, such as: + +```go +func TestGock (t *testing.T) { + defer gock.Off() + defer gock.RestoreClient(client) + + // ... my test code goes here +} +``` + +## 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) +} +``` + +#### Debug intercepted http requests + +```go +package main + +import ( + "bytes" + "gopkg.in/h2non/gock.v1" + "net/http" +) + +func main() { + defer gock.Off() + gock.Observe(gock.DumpRequest) + + gock.New("http://foo.com"). + Post("/bar"). + MatchType("json"). + JSON(map[string]string{"foo": "bar"}). + Reply(200) + + body := bytes.NewBuffer([]byte(`{"foo":"bar"}`)) + http.Post("http://foo.com/bar", "application/json", body) +} + +``` + +## 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 diff --git a/vendor/gopkg.in/h2non/gock.v1/go.mod b/vendor/gopkg.in/h2non/gock.v1/go.mod new file mode 100644 index 00000000..7aabc340 --- /dev/null +++ b/vendor/gopkg.in/h2non/gock.v1/go.mod @@ -0,0 +1,6 @@ +module gopkg.in/h2non/gock.v1 + +require ( + github.com/h2non/parth v0.0.0-20190131123155-b4df798d6542 + github.com/nbio/st v0.0.0-20140626010706-e9e8d9816f32 +) diff --git a/vendor/gopkg.in/h2non/gock.v1/gock.go b/vendor/gopkg.in/h2non/gock.v1/gock.go new file mode 100644 index 00000000..08e9534c --- /dev/null +++ b/vendor/gopkg.in/h2non/gock.v1/gock.go @@ -0,0 +1,176 @@ +package gock + +import ( + "fmt" + "net/http" + "net/http/httputil" + "net/url" + "regexp" + "sync" +) + +// mutex is used interally for locking thread-sensitive functions. +var mutex = &sync.Mutex{} + +// config global singleton store. +var config = struct { + Networking bool + NetworkingFilters []FilterRequestFunc + Observer ObserverFunc +}{} + +// ObserverFunc is implemented by users to inspect the outgoing intercepted HTTP traffic +type ObserverFunc func(*http.Request, Mock) + +// DumpRequest is a default implementation of ObserverFunc that dumps +// the HTTP/1.x wire representation of the http request +var DumpRequest ObserverFunc = func(request *http.Request, mock Mock) { + bytes, _ := httputil.DumpRequestOut(request, true) + fmt.Println(string(bytes)) + fmt.Printf("\nMatches: %v\n---\n", mock != nil) +} + +// track unmatched requests so they can be tested for +var unmatchedRequests = []*http.Request{} + +// New creates and registers a new HTTP mock with +// default settings and returns the Request DSL for HTTP mock +// definition and set up. +func New(uri string) *Request { + Intercept() + + res := NewResponse() + req := NewRequest() + req.URLStruct, res.Error = url.Parse(normalizeURI(uri)) + + // Create the new mock expectation + exp := NewMock(req, res) + Register(exp) + + return req +} + +// Intercepting returns true if gock is currently able to intercept. +func Intercepting() bool { + mutex.Lock() + defer mutex.Unlock() + return http.DefaultTransport == DefaultTransport +} + +// Intercept enables HTTP traffic interception via http.DefaultTransport. +// If you are using a custom HTTP transport, you have to use `gock.Transport()` +func Intercept() { + if !Intercepting() { + http.DefaultTransport = DefaultTransport + } +} + +// InterceptClient allows the developer to intercept HTTP traffic using +// a custom http.Client who uses a non default http.Transport/http.RoundTripper implementation. +func InterceptClient(cli *http.Client) { + _, ok := cli.Transport.(*Transport) + if ok { + return // if transport already intercepted, just ignore it + } + trans := NewTransport() + trans.Transport = cli.Transport + cli.Transport = trans +} + +// RestoreClient allows the developer to disable and restore the +// original transport in the given http.Client. +func RestoreClient(cli *http.Client) { + trans, ok := cli.Transport.(*Transport) + if !ok { + return + } + cli.Transport = trans.Transport +} + +// Disable disables HTTP traffic interception by gock. +func Disable() { + mutex.Lock() + defer mutex.Unlock() + http.DefaultTransport = NativeTransport +} + +// Off disables the default HTTP interceptors and removes +// all the registered mocks, even if they has not been intercepted yet. +func Off() { + Flush() + Disable() +} + +// OffAll is like `Off()`, but it also removes the unmatched requests registry. +func OffAll() { + Flush() + Disable() + CleanUnmatchedRequest() +} + +// Observe provides a hook to support inspection of the request and matched mock +func Observe(fn ObserverFunc) { + mutex.Lock() + defer mutex.Unlock() + config.Observer = fn +} + +// EnableNetworking enables real HTTP networking +func EnableNetworking() { + mutex.Lock() + defer mutex.Unlock() + config.Networking = true +} + +// DisableNetworking disables real HTTP networking +func DisableNetworking() { + mutex.Lock() + defer mutex.Unlock() + config.Networking = false +} + +// NetworkingFilter determines if an http.Request should be triggered or not. +func NetworkingFilter(fn FilterRequestFunc) { + mutex.Lock() + defer mutex.Unlock() + config.NetworkingFilters = append(config.NetworkingFilters, fn) +} + +// DisableNetworkingFilters disables registered networking filters. +func DisableNetworkingFilters() { + mutex.Lock() + defer mutex.Unlock() + config.NetworkingFilters = []FilterRequestFunc{} +} + +// GetUnmatchedRequests returns all requests that have been received but haven't matched any mock +func GetUnmatchedRequests() []*http.Request { + mutex.Lock() + defer mutex.Unlock() + return unmatchedRequests +} + +// HasUnmatchedRequest returns true if gock has received any requests that didn't match a mock +func HasUnmatchedRequest() bool { + return len(GetUnmatchedRequests()) > 0 +} + +// CleanUnmatchedRequest cleans the unmatched requests internal registry. +func CleanUnmatchedRequest() { + mutex.Lock() + defer mutex.Unlock() + unmatchedRequests = []*http.Request{} +} + +func trackUnmatchedRequest(req *http.Request) { + mutex.Lock() + defer mutex.Unlock() + unmatchedRequests = append(unmatchedRequests, req) +} + +func normalizeURI(uri string) string { + if ok, _ := regexp.MatchString("^http[s]?", uri); !ok { + return "http://" + uri + } + return uri +} diff --git a/vendor/gopkg.in/h2non/gock.v1/matcher.go b/vendor/gopkg.in/h2non/gock.v1/matcher.go new file mode 100644 index 00000000..95763346 --- /dev/null +++ b/vendor/gopkg.in/h2non/gock.v1/matcher.go @@ -0,0 +1,135 @@ +package gock + +import "net/http" + +// MatchersHeader exposes an slice of HTTP header specific mock matchers. +var MatchersHeader = []MatchFunc{ + MatchMethod, + MatchScheme, + MatchHost, + MatchPath, + MatchHeaders, + MatchQueryParams, + MatchPathParams, +} + +// MatchersBody exposes an slice of HTTP body specific built-in mock matchers. +var MatchersBody = []MatchFunc{ + MatchBody, +} + +// Matchers stores all the built-in mock matchers. +var Matchers = append(MatchersHeader, MatchersBody...) + +// DefaultMatcher stores the default Matcher instance used to match mocks. +var DefaultMatcher = NewMatcher() + +// MatchFunc represents the required function +// interface implemented by matchers. +type MatchFunc func(*http.Request, *Request) (bool, error) + +// Matcher represents the required interface implemented by mock matchers. +type Matcher interface { + // Get returns a slice of registered function matchers. + Get() []MatchFunc + + // Add adds a new matcher function. + Add(MatchFunc) + + // Set sets the matchers functions stack. + Set([]MatchFunc) + + // Flush flushes the current matchers function stack. + Flush() + + // Match matches the given http.Request with a mock Request. + Match(*http.Request, *Request) (bool, error) +} + +// MockMatcher implements a mock matcher +type MockMatcher struct { + Matchers []MatchFunc +} + +// NewMatcher creates a new mock matcher +// using the default matcher functions. +func NewMatcher() *MockMatcher { + m := NewEmptyMatcher() + for _, matchFn := range Matchers { + m.Add(matchFn) + } + return m +} + +// NewBasicMatcher creates a new matcher with header only mock matchers. +func NewBasicMatcher() *MockMatcher { + m := NewEmptyMatcher() + for _, matchFn := range MatchersHeader { + m.Add(matchFn) + } + return m +} + +// NewEmptyMatcher creates a new empty matcher without default matchers. +func NewEmptyMatcher() *MockMatcher { + return &MockMatcher{Matchers: []MatchFunc{}} +} + +// Get returns a slice of registered function matchers. +func (m *MockMatcher) Get() []MatchFunc { + return m.Matchers +} + +// Add adds a new function matcher. +func (m *MockMatcher) Add(fn MatchFunc) { + m.Matchers = append(m.Matchers, fn) +} + +// Set sets a new stack of matchers functions. +func (m *MockMatcher) Set(stack []MatchFunc) { + m.Matchers = stack +} + +// Flush flushes the current matcher +func (m *MockMatcher) Flush() { + m.Matchers = []MatchFunc{} +} + +// Clone returns a separate MockMatcher instance that has a copy of the same MatcherFuncs +func (m *MockMatcher) Clone() *MockMatcher { + m2 := NewEmptyMatcher() + for _, mFn := range m.Get() { + m2.Add(mFn) + } + return m2 +} + +// Match matches the given http.Request with a mock request +// returning true in case that the request matches, otherwise false. +func (m *MockMatcher) Match(req *http.Request, ereq *Request) (bool, error) { + for _, matcher := range m.Matchers { + matches, err := matcher(req, ereq) + if err != nil { + return false, err + } + if !matches { + return false, nil + } + } + return true, nil +} + +// MatchMock is a helper function that matches the given http.Request +// in the list of registered mocks, returning it if matches or error if it fails. +func MatchMock(req *http.Request) (Mock, error) { + for _, mock := range GetAll() { + matches, err := mock.Match(req) + if err != nil { + return nil, err + } + if matches { + return mock, nil + } + } + return nil, nil +} diff --git a/vendor/gopkg.in/h2non/gock.v1/matchers.go b/vendor/gopkg.in/h2non/gock.v1/matchers.go new file mode 100644 index 00000000..cd6dec10 --- /dev/null +++ b/vendor/gopkg.in/h2non/gock.v1/matchers.go @@ -0,0 +1,256 @@ +package gock + +import ( + "compress/gzip" + "encoding/json" + "io" + "io/ioutil" + "net/http" + "reflect" + "regexp" + "strings" + + "github.com/h2non/parth" +) + +// EOL represents the end of line character. +const EOL = 0xa + +// BodyTypes stores the supported MIME body types for matching. +// Currently only text-based types. +var BodyTypes = []string{ + "text/html", + "text/plain", + "application/json", + "application/xml", + "multipart/form-data", + "application/x-www-form-urlencoded", +} + +// BodyTypeAliases stores a generic MIME type by alias. +var BodyTypeAliases = map[string]string{ + "html": "text/html", + "text": "text/plain", + "json": "application/json", + "xml": "application/xml", + "form": "multipart/form-data", + "url": "application/x-www-form-urlencoded", +} + +// CompressionSchemes stores the supported Content-Encoding types for decompression. +var CompressionSchemes = []string{ + "gzip", +} + +// MatchMethod matches the HTTP method of the given request. +func MatchMethod(req *http.Request, ereq *Request) (bool, error) { + return ereq.Method == "" || req.Method == ereq.Method, nil +} + +// MatchScheme matches the request URL protocol scheme. +func MatchScheme(req *http.Request, ereq *Request) (bool, error) { + return ereq.URLStruct.Scheme == "" || req.URL.Scheme == "" || ereq.URLStruct.Scheme == req.URL.Scheme, nil +} + +// MatchHost matches the HTTP host header field of the given request. +func MatchHost(req *http.Request, ereq *Request) (bool, error) { + url := ereq.URLStruct + if strings.EqualFold(url.Host, req.URL.Host) { + return true, nil + } + if !ereq.Options.DisableRegexpHost { + return regexp.MatchString(url.Host, req.URL.Host) + } + return false, nil +} + +// MatchPath matches the HTTP URL path of the given request. +func MatchPath(req *http.Request, ereq *Request) (bool, error) { + if req.URL.Path == ereq.URLStruct.Path { + return true, nil + } + return regexp.MatchString(ereq.URLStruct.Path, req.URL.Path) +} + +// MatchHeaders matches the headers fields of the given request. +func MatchHeaders(req *http.Request, ereq *Request) (bool, error) { + for key, value := range ereq.Header { + var err error + var match bool + + for _, field := range req.Header[key] { + match, err = regexp.MatchString(value[0], field) + if err != nil { + return false, err + } + if match { + break + } + } + + if !match { + return false, nil + } + } + return true, nil +} + +// MatchQueryParams matches the URL query params fields of the given request. +func MatchQueryParams(req *http.Request, ereq *Request) (bool, error) { + for key, value := range ereq.URLStruct.Query() { + var err error + var match bool + + for _, field := range req.URL.Query()[key] { + match, err = regexp.MatchString(value[0], field) + if err != nil { + return false, err + } + if match { + break + } + } + + if !match { + return false, nil + } + } + return true, nil +} + +// MatchPathParams matches the URL path parameters of the given request. +func MatchPathParams(req *http.Request, ereq *Request) (bool, error) { + for key, value := range ereq.PathParams { + var s string + + if err := parth.Sequent(req.URL.Path, key, &s); err != nil { + return false, nil + } + + if s != value { + return false, nil + } + } + return true, nil +} + +// MatchBody tries to match the request body. +// TODO: not too smart now, needs several improvements. +func MatchBody(req *http.Request, ereq *Request) (bool, error) { + // If match body is empty, just continue + if req.Method == "HEAD" || len(ereq.BodyBuffer) == 0 { + return true, nil + } + + // Only can match certain MIME body types + if !supportedType(req) { + return false, nil + } + + // Can only match certain compression schemes + if !supportedCompressionScheme(req) { + return false, nil + } + + // Create a reader for the body depending on compression type + bodyReader := req.Body + if ereq.CompressionScheme != "" { + if ereq.CompressionScheme != req.Header.Get("Content-Encoding") { + return false, nil + } + compressedBodyReader, err := compressionReader(req.Body, ereq.CompressionScheme) + if err != nil { + return false, err + } + bodyReader = compressedBodyReader + } + + // Read the whole request body + body, err := ioutil.ReadAll(bodyReader) + if err != nil { + return false, err + } + + // Restore body reader stream + req.Body = createReadCloser(body) + + // If empty, ignore the match + if len(body) == 0 && len(ereq.BodyBuffer) != 0 { + return false, nil + } + + // Match body by atomic string comparison + bodyStr := castToString(body) + matchStr := castToString(ereq.BodyBuffer) + if bodyStr == matchStr { + return true, nil + } + + // Match request body by regexp + match, _ := regexp.MatchString(matchStr, bodyStr) + if match == true { + return true, nil + } + + // todo - add conditional do only perform the conversion of body bytes + // representation of JSON to a map and then compare them for equality. + + // Check if the key + value pairs match + var bodyMap map[string]interface{} + var matchMap map[string]interface{} + + // Ensure that both byte bodies that that should be JSON can be converted to maps. + umErr := json.Unmarshal(body, &bodyMap) + umErr2 := json.Unmarshal(ereq.BodyBuffer, &matchMap) + if umErr == nil && umErr2 == nil && reflect.DeepEqual(bodyMap, matchMap) { + return true, nil + } + + return false, nil +} + +func supportedType(req *http.Request) bool { + mime := req.Header.Get("Content-Type") + if mime == "" { + return true + } + + for _, kind := range BodyTypes { + if match, _ := regexp.MatchString(kind, mime); match { + return true + } + } + return false +} + +func supportedCompressionScheme(req *http.Request) bool { + encoding := req.Header.Get("Content-Encoding") + if encoding == "" { + return true + } + + for _, kind := range CompressionSchemes { + if match, _ := regexp.MatchString(kind, encoding); match { + return true + } + } + return false +} + +func castToString(buf []byte) string { + str := string(buf) + tail := len(str) - 1 + if str[tail] == EOL { + str = str[:tail] + } + return str +} + +func compressionReader(r io.ReadCloser, scheme string) (io.ReadCloser, error) { + switch scheme { + case "gzip": + return gzip.NewReader(r) + default: + return r, nil + } +} diff --git a/vendor/gopkg.in/h2non/gock.v1/mock.go b/vendor/gopkg.in/h2non/gock.v1/mock.go new file mode 100644 index 00000000..75ad5923 --- /dev/null +++ b/vendor/gopkg.in/h2non/gock.v1/mock.go @@ -0,0 +1,146 @@ +package gock + +import ( + "net/http" + "sync" +) + +// Mock represents the required interface that must +// be implemented by HTTP mock instances. +type Mock interface { + // Disable disables the current mock manually. + Disable() + + // Done returns true if the current mock is disabled. + Done() bool + + // Request returns the mock Request instance. + Request() *Request + + // Response returns the mock Response instance. + Response() *Response + + // Match matches the given http.Request with the current mock. + Match(*http.Request) (bool, error) + + // AddMatcher adds a new matcher function. + AddMatcher(MatchFunc) + + // SetMatcher uses a new matcher implementation. + SetMatcher(Matcher) +} + +// Mocker implements a Mock capable interface providing +// a default mock configuration used internally to store mocks. +type Mocker struct { + // disabled stores if the current mock is disabled. + disabled bool + + // mutex stores the mock mutex for thread safity. + mutex sync.Mutex + + // matcher stores a Matcher capable instance to match the given http.Request. + matcher Matcher + + // request stores the mock Request to match. + request *Request + + // response stores the mock Response to use in case of match. + response *Response +} + +// NewMock creates a new HTTP mock based on the given request and response instances. +// It's mostly used internally. +func NewMock(req *Request, res *Response) *Mocker { + mock := &Mocker{ + request: req, + response: res, + matcher: DefaultMatcher.Clone(), + } + res.Mock = mock + req.Mock = mock + req.Response = res + return mock +} + +// Disable disables the current mock manually. +func (m *Mocker) Disable() { + m.disabled = true +} + +// Done returns true in case that the current mock +// instance is disabled and therefore must be removed. +func (m *Mocker) Done() bool { + m.mutex.Lock() + defer m.mutex.Unlock() + return m.disabled || (!m.request.Persisted && m.request.Counter == 0) +} + +// Request returns the Request instance +// configured for the current HTTP mock. +func (m *Mocker) Request() *Request { + return m.request +} + +// Response returns the Response instance +// configured for the current HTTP mock. +func (m *Mocker) Response() *Response { + return m.response +} + +// Match matches the given http.Request with the current Request +// mock expectation, returning true if matches. +func (m *Mocker) Match(req *http.Request) (bool, error) { + if m.disabled { + return false, nil + } + + // Filter + for _, filter := range m.request.Filters { + if !filter(req) { + return false, nil + } + } + + // Map + for _, mapper := range m.request.Mappers { + if treq := mapper(req); treq != nil { + req = treq + } + } + + // Match + matches, err := m.matcher.Match(req, m.request) + if matches { + m.decrement() + } + + return matches, err +} + +// SetMatcher sets a new matcher implementation +// for the current mock expectation. +func (m *Mocker) SetMatcher(matcher Matcher) { + m.matcher = matcher +} + +// AddMatcher adds a new matcher function +// for the current mock expectation. +func (m *Mocker) AddMatcher(fn MatchFunc) { + m.matcher.Add(fn) +} + +// decrement decrements the current mock Request counter. +func (m *Mocker) decrement() { + if m.request.Persisted { + return + } + + m.mutex.Lock() + defer m.mutex.Unlock() + + m.request.Counter-- + if m.request.Counter == 0 { + m.disabled = true + } +} diff --git a/vendor/gopkg.in/h2non/gock.v1/options.go b/vendor/gopkg.in/h2non/gock.v1/options.go new file mode 100644 index 00000000..188aa58d --- /dev/null +++ b/vendor/gopkg.in/h2non/gock.v1/options.go @@ -0,0 +1,8 @@ +package gock + +// Options represents customized option for gock +type Options struct { + // DisableRegexpHost stores if the host is only a plain string rather than regular expression, + // if DisableRegexpHost is true, host sets in gock.New(...) will be treated as plain string + DisableRegexpHost bool +} diff --git a/vendor/gopkg.in/h2non/gock.v1/request.go b/vendor/gopkg.in/h2non/gock.v1/request.go new file mode 100644 index 00000000..4259169e --- /dev/null +++ b/vendor/gopkg.in/h2non/gock.v1/request.go @@ -0,0 +1,325 @@ +package gock + +import ( + "encoding/base64" + "io" + "io/ioutil" + "net/http" + "net/url" + "strings" +) + +// MapRequestFunc represents the required function interface for request mappers. +type MapRequestFunc func(*http.Request) *http.Request + +// FilterRequestFunc represents the required function interface for request filters. +type FilterRequestFunc func(*http.Request) bool + +// Request represents the high-level HTTP request used to store +// request fields used to match intercepted requests. +type Request struct { + // Mock stores the parent mock reference for the current request mock used for method delegation. + Mock Mock + + // Response stores the current Response instance for the current matches Request. + Response *Response + + // Error stores the latest mock request configuration error. + Error error + + // Counter stores the pending times that the current mock should be active. + Counter int + + // Persisted stores if the current mock should be always active. + Persisted bool + + // Options stores options for current Request. + Options Options + + // URLStruct stores the parsed URL as *url.URL struct. + URLStruct *url.URL + + // Method stores the Request HTTP method to match. + Method string + + // CompressionScheme stores the Request Compression scheme to match and use for decompression. + CompressionScheme string + + // Header stores the HTTP header fields to match. + Header http.Header + + // Cookies stores the Request HTTP cookies values to match. + Cookies []*http.Cookie + + // PathParams stores the path parameters to match. + PathParams map[string]string + + // BodyBuffer stores the body data to match. + BodyBuffer []byte + + // Mappers stores the request functions mappers used for matching. + Mappers []MapRequestFunc + + // Filters stores the request functions filters used for matching. + Filters []FilterRequestFunc +} + +// NewRequest creates a new Request instance. +func NewRequest() *Request { + return &Request{ + Counter: 1, + URLStruct: &url.URL{}, + Header: make(http.Header), + PathParams: make(map[string]string), + } +} + +// URL defines the mock URL to match. +func (r *Request) URL(uri string) *Request { + r.URLStruct, r.Error = url.Parse(uri) + return r +} + +// SetURL defines the url.URL struct to be used for matching. +func (r *Request) SetURL(u *url.URL) *Request { + r.URLStruct = u + return r +} + +// Path defines the mock URL path value to match. +func (r *Request) Path(path string) *Request { + r.URLStruct.Path = path + return r +} + +// Get specifies the GET method and the given URL path to match. +func (r *Request) Get(path string) *Request { + return r.method("GET", path) +} + +// Post specifies the POST method and the given URL path to match. +func (r *Request) Post(path string) *Request { + return r.method("POST", path) +} + +// Put specifies the PUT method and the given URL path to match. +func (r *Request) Put(path string) *Request { + return r.method("PUT", path) +} + +// Delete specifies the DELETE method and the given URL path to match. +func (r *Request) Delete(path string) *Request { + return r.method("DELETE", path) +} + +// Patch specifies the PATCH method and the given URL path to match. +func (r *Request) Patch(path string) *Request { + return r.method("PATCH", path) +} + +// Head specifies the HEAD method and the given URL path to match. +func (r *Request) Head(path string) *Request { + return r.method("HEAD", path) +} + +// method is a DRY shortcut used to declare the expected HTTP method and URL path. +func (r *Request) method(method, path string) *Request { + if path != "/" { + r.URLStruct.Path = path + } + r.Method = strings.ToUpper(method) + return r +} + +// Body defines the body data to match based on a io.Reader interface. +func (r *Request) Body(body io.Reader) *Request { + r.BodyBuffer, r.Error = ioutil.ReadAll(body) + return r +} + +// BodyString defines the body to match based on a given string. +func (r *Request) BodyString(body string) *Request { + r.BodyBuffer = []byte(body) + return r +} + +// File defines the body to match based on the given file path string. +func (r *Request) File(path string) *Request { + r.BodyBuffer, r.Error = ioutil.ReadFile(path) + return r +} + +// Compression defines the request compression scheme, and enables automatic body decompression. +// Supports only the "gzip" scheme so far. +func (r *Request) Compression(scheme string) *Request { + r.Header.Set("Content-Encoding", scheme) + r.CompressionScheme = scheme + return r +} + +// JSON defines the JSON body to match based on a given structure. +func (r *Request) JSON(data interface{}) *Request { + if r.Header.Get("Content-Type") == "" { + r.Header.Set("Content-Type", "application/json") + } + r.BodyBuffer, r.Error = readAndDecode(data, "json") + return r +} + +// XML defines the XML body to match based on a given structure. +func (r *Request) XML(data interface{}) *Request { + if r.Header.Get("Content-Type") == "" { + r.Header.Set("Content-Type", "application/xml") + } + r.BodyBuffer, r.Error = readAndDecode(data, "xml") + return r +} + +// MatchType defines the request Content-Type MIME header field. +// Supports type alias. E.g: json, xml, form, text... +func (r *Request) MatchType(kind string) *Request { + mime := BodyTypeAliases[kind] + if mime != "" { + kind = mime + } + r.Header.Set("Content-Type", kind) + return r +} + +// BasicAuth defines a username and password for HTTP Basic Authentication +func (r *Request) BasicAuth(username, password string) *Request { + r.Header.Set("Authorization", "Basic "+basicAuth(username, password)) + return r +} + +// MatchHeader defines a new key and value header to match. +func (r *Request) MatchHeader(key, value string) *Request { + r.Header.Set(key, value) + return r +} + +// HeaderPresent defines that a header field must be present in the request. +func (r *Request) HeaderPresent(key string) *Request { + r.Header.Set(key, ".*") + return r +} + +// MatchHeaders defines a map of key-value headers to match. +func (r *Request) MatchHeaders(headers map[string]string) *Request { + for key, value := range headers { + r.Header.Set(key, value) + } + return r +} + +// MatchParam defines a new key and value URL query param to match. +func (r *Request) MatchParam(key, value string) *Request { + query := r.URLStruct.Query() + query.Set(key, value) + r.URLStruct.RawQuery = query.Encode() + return r +} + +// MatchParams defines a map of URL query param key-value to match. +func (r *Request) MatchParams(params map[string]string) *Request { + query := r.URLStruct.Query() + for key, value := range params { + query.Set(key, value) + } + r.URLStruct.RawQuery = query.Encode() + return r +} + +// ParamPresent matches if the given query param key is present in the URL. +func (r *Request) ParamPresent(key string) *Request { + r.MatchParam(key, ".*") + return r +} + +// PathParam matches if a given path parameter key is present in the URL. +// +// The value is representative of the restful resource the key defines, e.g. +// // /users/123/name +// r.PathParam("users", "123") +// would match. +func (r *Request) PathParam(key, val string) *Request { + r.PathParams[key] = val + + return r +} + +// Persist defines the current HTTP mock as persistent and won't be removed after intercepting it. +func (r *Request) Persist() *Request { + r.Persisted = true + return r +} + +// WithOptions sets the options for the request. +func (r *Request) WithOptions(options Options) *Request { + r.Options = options + return r +} + +// Times defines the number of times that the current HTTP mock should remain active. +func (r *Request) Times(num int) *Request { + r.Counter = num + return r +} + +// AddMatcher adds a new matcher function to match the request. +func (r *Request) AddMatcher(fn MatchFunc) *Request { + r.Mock.AddMatcher(fn) + return r +} + +// SetMatcher sets a new matcher function to match the request. +func (r *Request) SetMatcher(matcher Matcher) *Request { + r.Mock.SetMatcher(matcher) + return r +} + +// Map adds a new request mapper function to map http.Request before the matching process. +func (r *Request) Map(fn MapRequestFunc) *Request { + r.Mappers = append(r.Mappers, fn) + return r +} + +// Filter filters a new request filter function to filter http.Request before the matching process. +func (r *Request) Filter(fn FilterRequestFunc) *Request { + r.Filters = append(r.Filters, fn) + return r +} + +// EnableNetworking enables the use real networking for the current mock. +func (r *Request) EnableNetworking() *Request { + if r.Response != nil { + r.Response.UseNetwork = true + } + return r +} + +// Reply defines the Response status code and returns the mock Response DSL. +func (r *Request) Reply(status int) *Response { + return r.Response.Status(status) +} + +// ReplyError defines the Response simulated error. +func (r *Request) ReplyError(err error) *Response { + return r.Response.SetError(err) +} + +// ReplyFunc allows the developer to define the mock response via a custom function. +func (r *Request) ReplyFunc(replier func(*Response)) *Response { + replier(r.Response) + return r.Response +} + +// See 2 (end of page 4) https://www.ietf.org/rfc/rfc2617.txt +// "To receive authorization, the client sends the userid and password, +// separated by a single colon (":") character, within a base64 +// encoded string in the credentials." +// It is not meant to be urlencoded. +func basicAuth(username, password string) string { + auth := username + ":" + password + return base64.StdEncoding.EncodeToString([]byte(auth)) +} diff --git a/vendor/gopkg.in/h2non/gock.v1/responder.go b/vendor/gopkg.in/h2non/gock.v1/responder.go new file mode 100644 index 00000000..35eb8719 --- /dev/null +++ b/vendor/gopkg.in/h2non/gock.v1/responder.go @@ -0,0 +1,87 @@ +package gock + +import ( + "bytes" + "io" + "io/ioutil" + "net/http" + "strconv" + "time" +) + +// Responder builds a mock http.Response based on the given Response mock. +func Responder(req *http.Request, mock *Response, res *http.Response) (*http.Response, error) { + // If error present, reply it + err := mock.Error + if err != nil { + return nil, err + } + + if res == nil { + res = createResponse(req) + } + + // Apply response filter + for _, filter := range mock.Filters { + if !filter(res) { + return res, nil + } + } + + // Define mock status code + if mock.StatusCode != 0 { + res.Status = strconv.Itoa(mock.StatusCode) + " " + http.StatusText(mock.StatusCode) + res.StatusCode = mock.StatusCode + } + + // Define headers by merging fields + res.Header = mergeHeaders(res, mock) + + // Define mock body, if present + if len(mock.BodyBuffer) > 0 { + res.ContentLength = int64(len(mock.BodyBuffer)) + res.Body = createReadCloser(mock.BodyBuffer) + } + + // Apply response mappers + for _, mapper := range mock.Mappers { + if tres := mapper(res); tres != nil { + res = tres + } + } + + // Sleep to simulate delay, if necessary + if mock.ResponseDelay > 0 { + time.Sleep(mock.ResponseDelay) + } + + return res, err +} + +// createResponse creates a new http.Response with default fields. +func createResponse(req *http.Request) *http.Response { + return &http.Response{ + ProtoMajor: 1, + ProtoMinor: 1, + Proto: "HTTP/1.1", + Request: req, + Header: make(http.Header), + Body: createReadCloser([]byte{}), + } +} + +// mergeHeaders copies the mock headers. +func mergeHeaders(res *http.Response, mres *Response) http.Header { + for key, values := range mres.Header { + for _, value := range values { + res.Header.Add(key, value) + } + } + return res.Header +} + +// createReadCloser creates an io.ReadCloser from a byte slice that is suitable for use as an +// http response body. +func createReadCloser(body []byte) io.ReadCloser { + return ioutil.NopCloser(bytes.NewReader(body)) +} diff --git a/vendor/gopkg.in/h2non/gock.v1/response.go b/vendor/gopkg.in/h2non/gock.v1/response.go new file mode 100644 index 00000000..07b71580 --- /dev/null +++ b/vendor/gopkg.in/h2non/gock.v1/response.go @@ -0,0 +1,186 @@ +package gock + +import ( + "bytes" + "encoding/json" + "encoding/xml" + "io" + "io/ioutil" + "net/http" + "time" +) + +// MapResponseFunc represents the required function interface impletemed by response mappers. +type MapResponseFunc func(*http.Response) *http.Response + +// FilterResponseFunc represents the required function interface impletemed by response filters. +type FilterResponseFunc func(*http.Response) bool + +// Response represents high-level HTTP fields to configure +// and define HTTP responses intercepted by gock. +type Response struct { + // Mock stores the parent mock reference for the current response mock used for method delegation. + Mock Mock + + // Error stores the latest response configuration or injected error. + Error error + + // UseNetwork enables the use of real network for the current mock. + UseNetwork bool + + // StatusCode stores the response status code. + StatusCode int + + // Headers stores the response headers. + Header http.Header + + // Cookies stores the response cookie fields. + Cookies []*http.Cookie + + // BodyBuffer stores the array of bytes to use as body. + BodyBuffer []byte + + // ResponseDelay stores the simulated response delay. + ResponseDelay time.Duration + + // Mappers stores the request functions mappers used for matching. + Mappers []MapResponseFunc + + // Filters stores the request functions filters used for matching. + Filters []FilterResponseFunc +} + +// NewResponse creates a new Response. +func NewResponse() *Response { + return &Response{Header: make(http.Header)} +} + +// Status defines the desired HTTP status code to reply in the current response. +func (r *Response) Status(code int) *Response { + r.StatusCode = code + return r +} + +// Type defines the response Content-Type MIME header field. +// Supports type alias. E.g: json, xml, form, text... +func (r *Response) Type(kind string) *Response { + mime := BodyTypeAliases[kind] + if mime != "" { + kind = mime + } + r.Header.Set("Content-Type", kind) + return r +} + +// SetHeader sets a new header field in the mock response. +func (r *Response) SetHeader(key, value string) *Response { + r.Header.Set(key, value) + return r +} + +// AddHeader adds a new header field in the mock response +// with out removing an existent one. +func (r *Response) AddHeader(key, value string) *Response { + r.Header.Add(key, value) + return r +} + +// SetHeaders sets a map of header fields in the mock response. +func (r *Response) SetHeaders(headers map[string]string) *Response { + for key, value := range headers { + r.Header.Add(key, value) + } + return r +} + +// Body sets the HTTP response body to be used. +func (r *Response) Body(body io.Reader) *Response { + r.BodyBuffer, r.Error = ioutil.ReadAll(body) + return r +} + +// BodyString defines the response body as string. +func (r *Response) BodyString(body string) *Response { + r.BodyBuffer = []byte(body) + return r +} + +// File defines the response body reading the data +// from disk based on the file path string. +func (r *Response) File(path string) *Response { + r.BodyBuffer, r.Error = ioutil.ReadFile(path) + return r +} + +// JSON defines the response body based on a JSON based input. +func (r *Response) JSON(data interface{}) *Response { + r.Header.Set("Content-Type", "application/json") + r.BodyBuffer, r.Error = readAndDecode(data, "json") + return r +} + +// XML defines the response body based on a XML based input. +func (r *Response) XML(data interface{}) *Response { + r.Header.Set("Content-Type", "application/xml") + r.BodyBuffer, r.Error = readAndDecode(data, "xml") + return r +} + +// SetError defines the response simulated error. +func (r *Response) SetError(err error) *Response { + r.Error = err + return r +} + +// Delay defines the response simulated delay. +// This feature is still experimental and will be improved in the future. +func (r *Response) Delay(delay time.Duration) *Response { + r.ResponseDelay = delay + return r +} + +// Map adds a new response mapper function to map http.Response before the matching process. +func (r *Response) Map(fn MapResponseFunc) *Response { + r.Mappers = append(r.Mappers, fn) + return r +} + +// Filter filters a new request filter function to filter http.Request before the matching process. +func (r *Response) Filter(fn FilterResponseFunc) *Response { + r.Filters = append(r.Filters, fn) + return r +} + +// EnableNetworking enables the use real networking for the current mock. +func (r *Response) EnableNetworking() *Response { + r.UseNetwork = true + return r +} + +// Done returns true if the mock was done and disabled. +func (r *Response) Done() bool { + return r.Mock.Done() +} + +func readAndDecode(data interface{}, kind string) ([]byte, error) { + buf := &bytes.Buffer{} + + switch data.(type) { + case string: + buf.WriteString(data.(string)) + case []byte: + buf.Write(data.([]byte)) + default: + var err error + if kind == "xml" { + err = xml.NewEncoder(buf).Encode(data) + } else { + err = json.NewEncoder(buf).Encode(data) + } + if err != nil { + return nil, err + } + } + + return ioutil.ReadAll(buf) +} diff --git a/vendor/gopkg.in/h2non/gock.v1/store.go b/vendor/gopkg.in/h2non/gock.v1/store.go new file mode 100644 index 00000000..71a20a80 --- /dev/null +++ b/vendor/gopkg.in/h2non/gock.v1/store.go @@ -0,0 +1,100 @@ +package gock + +import ( + "sync" +) + +// storeMutex is used interally for store synchronization. +var storeMutex = sync.RWMutex{} + +// mocks is internally used to store registered mocks. +var mocks = []Mock{} + +// Register registers a new mock in the current mocks stack. +func Register(mock Mock) { + if Exists(mock) { + return + } + + // Make ops thread safe + mutex.Lock() + defer mutex.Unlock() + + // Expose mock in request/response for delegation + mock.Request().Mock = mock + mock.Response().Mock = mock + + // Registers the mock in the global store + mocks = append(mocks, mock) +} + +// GetAll returns the current stack of registed mocks. +func GetAll() []Mock { + storeMutex.RLock() + defer storeMutex.RUnlock() + return mocks +} + +// Exists checks if the given Mock is already registered. +func Exists(m Mock) bool { + storeMutex.RLock() + defer storeMutex.RUnlock() + for _, mock := range mocks { + if mock == m { + return true + } + } + return false +} + +// Remove removes a registered mock by reference. +func Remove(m Mock) { + for i, mock := range mocks { + if mock == m { + storeMutex.Lock() + mocks = append(mocks[:i], mocks[i+1:]...) + storeMutex.Unlock() + } + } +} + +// Flush flushes the current stack of registered mocks. +func Flush() { + storeMutex.Lock() + defer storeMutex.Unlock() + mocks = []Mock{} +} + +// Pending returns an slice of pending mocks. +func Pending() []Mock { + Clean() + storeMutex.RLock() + defer storeMutex.RUnlock() + return mocks +} + +// IsDone returns true if all the registered mocks has been triggered successfully. +func IsDone() bool { + return !IsPending() +} + +// IsPending returns true if there are pending mocks. +func IsPending() bool { + return len(Pending()) > 0 +} + +// Clean cleans the mocks store removing disabled or obsolete mocks. +func Clean() { + storeMutex.Lock() + defer storeMutex.Unlock() + + buf := []Mock{} + for _, mock := range mocks { + if mock.Done() { + continue + } + buf = append(buf, mock) + } + + mocks = buf +} diff --git a/vendor/gopkg.in/h2non/gock.v1/transport.go b/vendor/gopkg.in/h2non/gock.v1/transport.go new file mode 100644 index 00000000..5b2bba28 --- /dev/null +++ b/vendor/gopkg.in/h2non/gock.v1/transport.go @@ -0,0 +1,112 @@ +package gock + +import ( + "errors" + "net/http" + "sync" +) + +// var mutex *sync.Mutex = &sync.Mutex{} + +var ( + // DefaultTransport stores the default mock transport used by gock. + DefaultTransport = NewTransport() + + // NativeTransport stores the native net/http default transport + // in order to restore it when needed. + NativeTransport = http.DefaultTransport +) + +var ( + // ErrCannotMatch store the error returned in case of no matches. + ErrCannotMatch = errors.New("gock: cannot match any request") +) + +// Transport implements http.RoundTripper, which fulfills single http requests issued by +// an http.Client. +// +// gock's Transport encapsulates a given or default http.Transport for further +// delegation, if needed. +type Transport struct { + // mutex is used to make transport thread-safe of concurrent uses across goroutines. + mutex sync.Mutex + + // Transport encapsulates the original http.RoundTripper transport interface for delegation. + Transport http.RoundTripper +} + +// NewTransport creates a new *Transport with no responders. +func NewTransport() *Transport { + return &Transport{Transport: NativeTransport} +} + +// RoundTrip receives HTTP requests and routes them to the appropriate responder. It is required to +// implement the http.RoundTripper interface. You will not interact with this directly, instead +// the *http.Client you are using will call it for you. +func (m *Transport) RoundTrip(req *http.Request) (*http.Response, error) { + // Just act as a proxy if not intercepting + if !Intercepting() { + return m.Transport.RoundTrip(req) + } + + m.mutex.Lock() + defer Clean() + + var err error + var res *http.Response + + // Match mock for the incoming http.Request + mock, err := MatchMock(req) + if err != nil { + m.mutex.Unlock() + return nil, err + } + + // Invoke the observer with the intercepted http.Request and matched mock + if config.Observer != nil { + config.Observer(req, mock) + } + + // Verify if should use real networking + networking := shouldUseNetwork(req, mock) + if !networking && mock == nil { + m.mutex.Unlock() + trackUnmatchedRequest(req) + return nil, ErrCannotMatch + } + + // Ensure me unlock the mutex before building the response + m.mutex.Unlock() + + // Perform real networking via original transport + if networking { + res, err = m.Transport.RoundTrip(req) + // In no mock matched, continue with the response + if err != nil || mock == nil { + return res, err + } + } + + return Responder(req, mock.Response(), res) +} + +// CancelRequest is a no-op function. +func (m *Transport) CancelRequest(req *http.Request) {} + +func shouldUseNetwork(req *http.Request, mock Mock) bool { + if mock != nil && mock.Response().UseNetwork { + return true + } + if !config.Networking { + return false + } + if len(config.NetworkingFilters) == 0 { + return true + } + for _, filter := range config.NetworkingFilters { + if !filter(req) { + return false + } + } + return true +} diff --git a/vendor/gopkg.in/h2non/gock.v1/version.go b/vendor/gopkg.in/h2non/gock.v1/version.go new file mode 100644 index 00000000..b3205d70 --- /dev/null +++ b/vendor/gopkg.in/h2non/gock.v1/version.go @@ -0,0 +1,4 @@ +package gock + +// Version defines the current package semantic version. +const Version = "1.0.15" diff --git a/vendor/modules.txt b/vendor/modules.txt index 23bb564e..b55087fb 100644 --- a/vendor/modules.txt +++ b/vendor/modules.txt @@ -36,6 +36,8 @@ github.com/docker/go-connections/sockets github.com/docker/go-connections/tlsconfig # github.com/docker/go-units v0.4.0 github.com/docker/go-units +# github.com/h2non/parth v0.0.0-20190131123155-b4df798d6542 +github.com/h2non/parth # github.com/logrusorgru/aurora v0.0.0-20181002194514-a7b3b318ed4e github.com/logrusorgru/aurora # github.com/opencontainers/go-digest v1.0.0-rc1 @@ -90,5 +92,7 @@ golang.org/x/text/internal/utf8internal golang.org/x/text/language golang.org/x/text/runes golang.org/x/text/transform +# gopkg.in/h2non/gock.v1 v1.0.15 +gopkg.in/h2non/gock.v1 # gopkg.in/yaml.v2 v2.2.4 gopkg.in/yaml.v2