mirror of
https://github.com/kardolus/chatgpt-cli.git
synced 2024-09-08 23:15:00 +03:00
POC
This commit is contained in:
4
.gitignore
vendored
Normal file
4
.gitignore
vendored
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
.idea
|
||||||
|
.DS_Store
|
||||||
|
bin/*
|
||||||
|
scripts/env.sh
|
||||||
27
README.md
Normal file
27
README.md
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
# ChatGPT CLI
|
||||||
|
A Proof of Concept (POC) for building ChatGPT clients.
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
## Development
|
||||||
|
|
||||||
|
To start developing, set the CHAT_GPT_SECRET_KEY environment variable to your [ChatGPT secret key](https://platform.openai.com/account/api-keys). Follow these steps for
|
||||||
|
running tests and building the application:
|
||||||
|
|
||||||
|
1. Run the unit tests using the following script:
|
||||||
|
```shell
|
||||||
|
./scripts/unit.sh
|
||||||
|
```
|
||||||
|
2. Build the app using the installation script:
|
||||||
|
```shell
|
||||||
|
./scripts/install.sh
|
||||||
|
```
|
||||||
|
3. After a successful build, test the application with the following command:
|
||||||
|
```shell
|
||||||
|
./bin/chatgpt what type of dog is a Jack Russel?
|
||||||
|
```
|
||||||
|
|
||||||
|
## Links
|
||||||
|
|
||||||
|
* [ChatGPT API](https://platform.openai.com/docs/introduction/overview)
|
||||||
|
* [Key Usage Dashboard](https://platform.openai.com/account/usage)
|
||||||
69
client/client.go
Normal file
69
client/client.go
Normal file
@@ -0,0 +1,69 @@
|
|||||||
|
package client
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"github.com/kardolus/chatgpt-poc/http"
|
||||||
|
"github.com/kardolus/chatgpt-poc/types"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
model = "gpt-3.5-turbo"
|
||||||
|
role = "user"
|
||||||
|
URL = "https://api.openai.com/v1/chat/completions"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Client struct {
|
||||||
|
caller http.Caller
|
||||||
|
}
|
||||||
|
|
||||||
|
func New(caller http.Caller) *Client {
|
||||||
|
return &Client{caller: caller}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Client) Query(input string) (string, error) {
|
||||||
|
body, err := CreateBody(input)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
raw, err := c.caller.Post(URL, body)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
if raw == nil {
|
||||||
|
return "", errors.New("empty response")
|
||||||
|
}
|
||||||
|
|
||||||
|
var response types.Response
|
||||||
|
if err := json.Unmarshal(raw, &response); err != nil {
|
||||||
|
return "", fmt.Errorf("failed to decode response: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(response.Choices) == 0 {
|
||||||
|
return "", errors.New("no responses returned")
|
||||||
|
}
|
||||||
|
|
||||||
|
return response.Choices[0].Message.Content, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func CreateBody(query string) ([]byte, error) {
|
||||||
|
message := types.Message{
|
||||||
|
Role: role,
|
||||||
|
Content: query,
|
||||||
|
}
|
||||||
|
|
||||||
|
body := types.Request{
|
||||||
|
Model: model,
|
||||||
|
Messages: []types.Message{message},
|
||||||
|
}
|
||||||
|
|
||||||
|
result, err := json.Marshal(body)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return result, nil
|
||||||
|
}
|
||||||
121
client/client_test.go
Normal file
121
client/client_test.go
Normal file
@@ -0,0 +1,121 @@
|
|||||||
|
package client_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"errors"
|
||||||
|
"github.com/golang/mock/gomock"
|
||||||
|
_ "github.com/golang/mock/mockgen/model"
|
||||||
|
"github.com/kardolus/chatgpt-poc/client"
|
||||||
|
"github.com/kardolus/chatgpt-poc/types"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
. "github.com/onsi/gomega"
|
||||||
|
"github.com/sclevine/spec"
|
||||||
|
"github.com/sclevine/spec/report"
|
||||||
|
)
|
||||||
|
|
||||||
|
//go:generate mockgen -destination=mocks_test.go -package=client_test github.com/kardolus/chatgpt-poc/http Caller
|
||||||
|
|
||||||
|
var (
|
||||||
|
mockCtrl *gomock.Controller
|
||||||
|
mockCaller *MockCaller
|
||||||
|
subject *client.Client
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestUnitClient(t *testing.T) {
|
||||||
|
spec.Run(t, "Testing the client package", testClient, spec.Report(report.Terminal{}))
|
||||||
|
}
|
||||||
|
|
||||||
|
func testClient(t *testing.T, when spec.G, it spec.S) {
|
||||||
|
it.Before(func() {
|
||||||
|
RegisterTestingT(t)
|
||||||
|
mockCtrl = gomock.NewController(t)
|
||||||
|
mockCaller = NewMockCaller(mockCtrl)
|
||||||
|
|
||||||
|
subject = client.New(mockCaller)
|
||||||
|
})
|
||||||
|
|
||||||
|
it.After(func() {
|
||||||
|
mockCtrl.Finish()
|
||||||
|
})
|
||||||
|
|
||||||
|
when("Query()", func() {
|
||||||
|
const query = "test query"
|
||||||
|
|
||||||
|
var (
|
||||||
|
err error
|
||||||
|
body []byte
|
||||||
|
)
|
||||||
|
|
||||||
|
it.Before(func() {
|
||||||
|
body, err = client.CreateBody(query)
|
||||||
|
Expect(err).NotTo(HaveOccurred())
|
||||||
|
})
|
||||||
|
|
||||||
|
it("throws an error when the http callout fails", func() {
|
||||||
|
errorMsg := "error message"
|
||||||
|
mockCaller.EXPECT().Post(client.URL, body).Return(nil, errors.New(errorMsg))
|
||||||
|
|
||||||
|
_, err = subject.Query(query)
|
||||||
|
Expect(err).To(HaveOccurred())
|
||||||
|
Expect(err.Error()).To(Equal(errorMsg))
|
||||||
|
})
|
||||||
|
it("throws an error when the response is empty", func() {
|
||||||
|
mockCaller.EXPECT().Post(client.URL, body).Return(nil, nil)
|
||||||
|
|
||||||
|
_, err = subject.Query(query)
|
||||||
|
Expect(err).To(HaveOccurred())
|
||||||
|
Expect(err.Error()).To(Equal("empty response"))
|
||||||
|
})
|
||||||
|
it("throws an error when the response is a malformed json", func() {
|
||||||
|
malformed := "{no"
|
||||||
|
mockCaller.EXPECT().Post(client.URL, body).Return([]byte(malformed), nil)
|
||||||
|
|
||||||
|
_, err = subject.Query(query)
|
||||||
|
Expect(err).To(HaveOccurred())
|
||||||
|
Expect(err.Error()).Should(HavePrefix("failed to decode response:"))
|
||||||
|
})
|
||||||
|
it("throws an error when the response is missing Choices", func() {
|
||||||
|
response := &types.Response{
|
||||||
|
ID: "id",
|
||||||
|
Object: "object",
|
||||||
|
Created: 0,
|
||||||
|
Model: "model",
|
||||||
|
Choices: []types.Choice{},
|
||||||
|
}
|
||||||
|
|
||||||
|
respBytes, err := json.Marshal(response)
|
||||||
|
Expect(err).NotTo(HaveOccurred())
|
||||||
|
mockCaller.EXPECT().Post(client.URL, body).Return(respBytes, nil)
|
||||||
|
|
||||||
|
_, err = subject.Query(query)
|
||||||
|
Expect(err).To(HaveOccurred())
|
||||||
|
Expect(err.Error()).To(Equal("no responses returned"))
|
||||||
|
})
|
||||||
|
it("parses a valid http response as expected", func() {
|
||||||
|
choice := types.Choice{
|
||||||
|
Message: types.Message{
|
||||||
|
Role: "role",
|
||||||
|
Content: "content",
|
||||||
|
},
|
||||||
|
FinishReason: "",
|
||||||
|
Index: 0,
|
||||||
|
}
|
||||||
|
response := &types.Response{
|
||||||
|
ID: "id",
|
||||||
|
Object: "object",
|
||||||
|
Created: 0,
|
||||||
|
Model: "model",
|
||||||
|
Choices: []types.Choice{choice},
|
||||||
|
}
|
||||||
|
|
||||||
|
respBytes, err := json.Marshal(response)
|
||||||
|
Expect(err).NotTo(HaveOccurred())
|
||||||
|
mockCaller.EXPECT().Post(client.URL, body).Return(respBytes, nil)
|
||||||
|
|
||||||
|
result, err := subject.Query(query)
|
||||||
|
Expect(err).NotTo(HaveOccurred())
|
||||||
|
Expect(result).To(Equal("content"))
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
49
client/mocks_test.go
Normal file
49
client/mocks_test.go
Normal file
@@ -0,0 +1,49 @@
|
|||||||
|
// Code generated by MockGen. DO NOT EDIT.
|
||||||
|
// Source: github.com/kardolus/chatgpt-poc/http (interfaces: Caller)
|
||||||
|
|
||||||
|
// Package client_test is a generated GoMock package.
|
||||||
|
package client_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
reflect "reflect"
|
||||||
|
|
||||||
|
gomock "github.com/golang/mock/gomock"
|
||||||
|
)
|
||||||
|
|
||||||
|
// MockCaller is a mock of Caller interface.
|
||||||
|
type MockCaller struct {
|
||||||
|
ctrl *gomock.Controller
|
||||||
|
recorder *MockCallerMockRecorder
|
||||||
|
}
|
||||||
|
|
||||||
|
// MockCallerMockRecorder is the mock recorder for MockCaller.
|
||||||
|
type MockCallerMockRecorder struct {
|
||||||
|
mock *MockCaller
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewMockCaller creates a new mock instance.
|
||||||
|
func NewMockCaller(ctrl *gomock.Controller) *MockCaller {
|
||||||
|
mock := &MockCaller{ctrl: ctrl}
|
||||||
|
mock.recorder = &MockCallerMockRecorder{mock}
|
||||||
|
return mock
|
||||||
|
}
|
||||||
|
|
||||||
|
// EXPECT returns an object that allows the caller to indicate expected use.
|
||||||
|
func (m *MockCaller) EXPECT() *MockCallerMockRecorder {
|
||||||
|
return m.recorder
|
||||||
|
}
|
||||||
|
|
||||||
|
// Post mocks base method.
|
||||||
|
func (m *MockCaller) Post(arg0 string, arg1 []byte) ([]byte, error) {
|
||||||
|
m.ctrl.T.Helper()
|
||||||
|
ret := m.ctrl.Call(m, "Post", arg0, arg1)
|
||||||
|
ret0, _ := ret[0].([]byte)
|
||||||
|
ret1, _ := ret[1].(error)
|
||||||
|
return ret0, ret1
|
||||||
|
}
|
||||||
|
|
||||||
|
// Post indicates an expected call of Post.
|
||||||
|
func (mr *MockCallerMockRecorder) Post(arg0, arg1 interface{}) *gomock.Call {
|
||||||
|
mr.mock.ctrl.T.Helper()
|
||||||
|
return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "Post", reflect.TypeOf((*MockCaller)(nil).Post), arg0, arg1)
|
||||||
|
}
|
||||||
45
cmd/chatgpt/main.go
Normal file
45
cmd/chatgpt/main.go
Normal file
@@ -0,0 +1,45 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"github.com/kardolus/chatgpt-poc/client"
|
||||||
|
"github.com/kardolus/chatgpt-poc/http"
|
||||||
|
"log"
|
||||||
|
"os"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
const secretEnv = "CHAT_GPT_SECRET_KEY"
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
exit(run())
|
||||||
|
}
|
||||||
|
|
||||||
|
func exit(err error) {
|
||||||
|
if err == nil {
|
||||||
|
os.Exit(0)
|
||||||
|
}
|
||||||
|
log.Printf("Error: %s\n", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
|
||||||
|
func run() error {
|
||||||
|
if len(os.Args) <= 1 {
|
||||||
|
return errors.New("you must specify your query")
|
||||||
|
}
|
||||||
|
|
||||||
|
secret := os.Getenv(secretEnv)
|
||||||
|
if secret == "" {
|
||||||
|
return errors.New("missing environment variable: " + secretEnv)
|
||||||
|
}
|
||||||
|
client := client.New(http.New().WithSecret(secret))
|
||||||
|
|
||||||
|
result, err := client.Query(strings.Join(os.Args[1:], " "))
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
fmt.Println(result)
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
71
doc/chatgpt.md
Normal file
71
doc/chatgpt.md
Normal file
@@ -0,0 +1,71 @@
|
|||||||
|
# ChatGPT API
|
||||||
|
|
||||||
|
### cURL davinci
|
||||||
|
```shell
|
||||||
|
curl https://api.openai.com/v1/completions \
|
||||||
|
-H 'Content-Type: application/json' \
|
||||||
|
-H "Authorization: Bearer ${CHAT_GPT_SECRET_KEY}" \
|
||||||
|
-d '{
|
||||||
|
"model": "text-davinci-003",
|
||||||
|
"prompt": "What is your name?",
|
||||||
|
"max_tokens": 4000,
|
||||||
|
"temperature": 1.0
|
||||||
|
}' \
|
||||||
|
--insecure | jq .
|
||||||
|
```
|
||||||
|
Output:
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"id": "cmpl-7BQi5QXWoy83V1HR8VcC7MzrtArGp",
|
||||||
|
"object": "text_completion",
|
||||||
|
"created": 1682958637,
|
||||||
|
"model": "text-davinci-003",
|
||||||
|
"choices": [
|
||||||
|
{
|
||||||
|
"text": "\n\nMy name is John.",
|
||||||
|
"index": 0,
|
||||||
|
"logprobs": null,
|
||||||
|
"finish_reason": "stop"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"usage": {
|
||||||
|
"prompt_tokens": 5,
|
||||||
|
"completion_tokens": 7,
|
||||||
|
"total_tokens": 12
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
### cURL gpt
|
||||||
|
```shell
|
||||||
|
curl --location --insecure --request POST 'https://api.openai.com/v1/chat/completions' \
|
||||||
|
--header "Authorization: Bearer ${CHAT_GPT_SECRET_KEY}" \
|
||||||
|
--header 'Content-Type: application/json' \
|
||||||
|
--data-raw '{
|
||||||
|
"model": "gpt-3.5-turbo",
|
||||||
|
"messages": [{"role": "user", "content": "What is the OpenAI mission?"}]
|
||||||
|
}' | jq .
|
||||||
|
```
|
||||||
|
Output:
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"id": "chatcmpl-7BQnIwmXhD6ohmwS6BjRHJrw9rJ7K",
|
||||||
|
"object": "chat.completion",
|
||||||
|
"created": 1682958960,
|
||||||
|
"model": "gpt-3.5-turbo-0301",
|
||||||
|
"usage": {
|
||||||
|
"prompt_tokens": 15,
|
||||||
|
"completion_tokens": 96,
|
||||||
|
"total_tokens": 111
|
||||||
|
},
|
||||||
|
"choices": [
|
||||||
|
{
|
||||||
|
"message": {
|
||||||
|
"role": "assistant",
|
||||||
|
"content": "The OpenAI mission is to create and develop advanced Artificial Intelligence in a way that is safe and beneficial to humanity. Their goal is to build systems capable of performing tasks that would normally require human intelligence, such as natural language processing, understanding, and decision-making. The organization aims to foster research and development that is accessible and open to the public while maintaining ethical considerations and prioritizing safety. The ultimate objective is to use AI to enhance human life while minimizing the potential for negative consequences."
|
||||||
|
},
|
||||||
|
"finish_reason": "stop",
|
||||||
|
"index": 0
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
13
go.mod
Normal file
13
go.mod
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
module github.com/kardolus/chatgpt-poc
|
||||||
|
|
||||||
|
go 1.18
|
||||||
|
|
||||||
|
require (
|
||||||
|
github.com/golang/mock v1.6.0 // indirect
|
||||||
|
github.com/google/go-cmp v0.5.9 // indirect
|
||||||
|
github.com/onsi/gomega v1.27.6 // indirect
|
||||||
|
github.com/sclevine/spec v1.4.0 // indirect
|
||||||
|
golang.org/x/net v0.8.0 // indirect
|
||||||
|
golang.org/x/text v0.8.0 // indirect
|
||||||
|
gopkg.in/yaml.v3 v3.0.1 // indirect
|
||||||
|
)
|
||||||
38
go.sum
Normal file
38
go.sum
Normal file
@@ -0,0 +1,38 @@
|
|||||||
|
github.com/golang/mock v1.6.0 h1:ErTB+efbowRARo13NNdxyJji2egdxLGQhRaY+DUumQc=
|
||||||
|
github.com/golang/mock v1.6.0/go.mod h1:p6yTPP+5HYm5mzsMV8JkE6ZKdX+/wYM6Hr+LicevLPs=
|
||||||
|
github.com/google/go-cmp v0.5.9 h1:O2Tfq5qg4qc4AmwVlvv0oLiVAGB7enBSJ2x2DqQFi38=
|
||||||
|
github.com/google/go-cmp v0.5.9/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY=
|
||||||
|
github.com/onsi/gomega v1.27.6 h1:ENqfyGeS5AX/rlXDd/ETokDz93u0YufY1Pgxuy/PvWE=
|
||||||
|
github.com/onsi/gomega v1.27.6/go.mod h1:PIQNjfQwkP3aQAH7lf7j87O/5FiNr+ZR8+ipb+qQlhg=
|
||||||
|
github.com/sclevine/spec v1.4.0 h1:z/Q9idDcay5m5irkZ28M7PtQM4aOISzOpj4bUPkDee8=
|
||||||
|
github.com/sclevine/spec v1.4.0/go.mod h1:LvpgJaFyvQzRvc1kaDs0bulYwzC70PbiYjC4QnFHkOM=
|
||||||
|
github.com/yuin/goldmark v1.3.5/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k=
|
||||||
|
golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
|
||||||
|
golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
|
||||||
|
golang.org/x/mod v0.4.2/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA=
|
||||||
|
golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
|
||||||
|
golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
|
||||||
|
golang.org/x/net v0.0.0-20210405180319-a5a99cb37ef4/go.mod h1:p54w0d4576C0XHj96bSt6lcn1PtDYWL6XObtHCRCNQM=
|
||||||
|
golang.org/x/net v0.8.0 h1:Zrh2ngAOFYneWTAIAPethzeaQLuHwhuBkuV6ZiRnUaQ=
|
||||||
|
golang.org/x/net v0.8.0/go.mod h1:QVkue5JL9kW//ek3r6jTKnTFis1tRmNAW2P1shuFdJc=
|
||||||
|
golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||||
|
golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||||
|
golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
||||||
|
golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||||
|
golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||||
|
golang.org/x/sys v0.0.0-20210330210617-4fbd30eecc44/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||||
|
golang.org/x/sys v0.0.0-20210510120138-977fb7262007/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||||
|
golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo=
|
||||||
|
golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
|
||||||
|
golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
|
||||||
|
golang.org/x/text v0.8.0 h1:57P1ETyNKtuIjB4SRd15iJxuhj8Gc416Y78H3qgMh68=
|
||||||
|
golang.org/x/text v0.8.0/go.mod h1:e1OnstbJyHTd6l/uOt8jFFHp6TRDWZR/bV3emEE/zU8=
|
||||||
|
golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
||||||
|
golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
|
||||||
|
golang.org/x/tools v0.1.1/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk=
|
||||||
|
golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
||||||
|
golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
||||||
|
golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
||||||
|
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
||||||
|
gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=
|
||||||
|
gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
|
||||||
59
http/http.go
Normal file
59
http/http.go
Normal file
@@ -0,0 +1,59 @@
|
|||||||
|
package http
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"fmt"
|
||||||
|
"io/ioutil"
|
||||||
|
"net/http"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Caller interface {
|
||||||
|
Post(url string, body []byte) ([]byte, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
type RestCaller struct {
|
||||||
|
client *http.Client
|
||||||
|
secret string
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ensure RestCaller implements Caller interface
|
||||||
|
var _ Caller = &RestCaller{}
|
||||||
|
|
||||||
|
func New() *RestCaller {
|
||||||
|
return &RestCaller{
|
||||||
|
client: &http.Client{},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *RestCaller) WithSecret(secret string) *RestCaller {
|
||||||
|
r.secret = secret
|
||||||
|
return r
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *RestCaller) Post(url string, body []byte) ([]byte, error) {
|
||||||
|
req, err := http.NewRequest(http.MethodPost, url, bytes.NewBuffer(body))
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to create request: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if r.secret != "" {
|
||||||
|
req.Header.Set("Authorization", fmt.Sprintf("Bearer %s", r.secret))
|
||||||
|
}
|
||||||
|
req.Header.Set("Content-Type", "application/json")
|
||||||
|
|
||||||
|
response, err := r.client.Do(req)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to make request: %w", err)
|
||||||
|
}
|
||||||
|
defer response.Body.Close()
|
||||||
|
|
||||||
|
if response.StatusCode >= 200 && response.StatusCode < 300 {
|
||||||
|
result, err := ioutil.ReadAll(response.Body)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to read response: %w", err)
|
||||||
|
}
|
||||||
|
return result, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil, fmt.Errorf("http error: %d", response.StatusCode)
|
||||||
|
}
|
||||||
BIN
resources/screenshot.png
Normal file
BIN
resources/screenshot.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 224 KiB |
5
scripts/all-tests.sh
Executable file
5
scripts/all-tests.sh
Executable file
@@ -0,0 +1,5 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
cd "$( dirname "${BASH_SOURCE[0]}" )/.."
|
||||||
|
./scripts/unit.sh && ./scripts/integration.sh
|
||||||
24
scripts/install.sh
Executable file
24
scripts/install.sh
Executable file
@@ -0,0 +1,24 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
cd "$( dirname "${BASH_SOURCE[0]}" )/.."
|
||||||
|
mkdir -p bin
|
||||||
|
|
||||||
|
TARGET_OS=${1:-darwin}
|
||||||
|
GIT_COMMIT=$(git rev-list -1 HEAD)
|
||||||
|
GIT_TAGS=$(git rev-list --tags --max-count=1)
|
||||||
|
|
||||||
|
echo "Target OS: $TARGET_OS"
|
||||||
|
for b in $(ls cmd); do
|
||||||
|
echo -n "Building $b..."
|
||||||
|
|
||||||
|
if [ ! -z "$GIT_TAGS" ]
|
||||||
|
then
|
||||||
|
GIT_VERSION=$(git describe --tags $GIT_TAGS)
|
||||||
|
GOOS=$TARGET_OS go build -mod=vendor -ldflags="-s -w -X main.GitCommit=$GIT_COMMIT -X main.GitVersion=$GIT_VERSION" -o bin/$b -a cmd/$b/main.go
|
||||||
|
else
|
||||||
|
GOOS=$TARGET_OS go build -mod=vendor -ldflags="-s -w -X main.GitCommit=$GIT_COMMIT" -o bin/$b -a cmd/$b/main.go
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "done"
|
||||||
|
done
|
||||||
22
scripts/integration.sh
Executable file
22
scripts/integration.sh
Executable file
@@ -0,0 +1,22 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
cd "$( dirname "${BASH_SOURCE[0]}" )/.."
|
||||||
|
|
||||||
|
if [[ ! -d integration ]]; then
|
||||||
|
echo -e "\n\033[0;31m** WARNING No Integration tests **\033[0m"
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "Run Integration Tests"
|
||||||
|
set +e
|
||||||
|
go test -parallel 1 -timeout 0 -mod=vendor ./integration/... -v -run Integration
|
||||||
|
exit_code=$?
|
||||||
|
|
||||||
|
if [ "$exit_code" != "0" ]; then
|
||||||
|
echo -e "\n\033[0;31m** GO Test Failed **\033[0m"
|
||||||
|
else
|
||||||
|
echo -e "\n\033[0;32m** GO Test Succeeded **\033[0m"
|
||||||
|
fi
|
||||||
|
|
||||||
|
exit $exit_code
|
||||||
9
scripts/reinstall.sh
Executable file
9
scripts/reinstall.sh
Executable file
@@ -0,0 +1,9 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
mkdir -p bin
|
||||||
|
rm -f bin/*
|
||||||
|
|
||||||
|
TARGET_OS=${1:-darwin}
|
||||||
|
|
||||||
|
source ./scripts/install.sh $TARGET_OS
|
||||||
23
scripts/unit.sh
Executable file
23
scripts/unit.sh
Executable file
@@ -0,0 +1,23 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
set -o pipefail
|
||||||
|
|
||||||
|
cd "$( dirname "${BASH_SOURCE[0]}" )/.."
|
||||||
|
|
||||||
|
go clean -testcache
|
||||||
|
|
||||||
|
echo "Run Unit Tests"
|
||||||
|
if [ -z "$1" ]
|
||||||
|
then
|
||||||
|
CONFIG_PATH="file://$PWD" TESTING=true go test -mod=vendor ./... -v -run Unit
|
||||||
|
else
|
||||||
|
CONFIG_PATH="file://$PWD" TESTING=true go test -mod=vendor ./"$1" -v -run Unit
|
||||||
|
fi
|
||||||
|
exit_code=$?
|
||||||
|
|
||||||
|
if [ "$exit_code" != "0" ]; then
|
||||||
|
echo -e "\n\033[0;31m** GO Test Failed **\033[0m"
|
||||||
|
else
|
||||||
|
echo -e "\n\033[0;32m** GO Test Succeeded **\033[0m"
|
||||||
|
fi
|
||||||
|
|
||||||
|
exit $exit_code
|
||||||
14
scripts/updatedeps.sh
Executable file
14
scripts/updatedeps.sh
Executable file
@@ -0,0 +1,14 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
set -o pipefail
|
||||||
|
|
||||||
|
go get -u ./...
|
||||||
|
go mod vendor
|
||||||
|
|
||||||
|
if [[ `git status --porcelain` ]]; then
|
||||||
|
echo "Updated dependencies"
|
||||||
|
git add .
|
||||||
|
git ci -m "Bump dependencies"
|
||||||
|
git push
|
||||||
|
else
|
||||||
|
echo "Dependencies up to date"
|
||||||
|
fi
|
||||||
30
types/types.go
Normal file
30
types/types.go
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
package types
|
||||||
|
|
||||||
|
type Request struct {
|
||||||
|
Model string `json:"model"`
|
||||||
|
Messages []Message `json:"messages"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type Message struct {
|
||||||
|
Role string `json:"role"`
|
||||||
|
Content string `json:"content"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type Response struct {
|
||||||
|
ID string `json:"id"`
|
||||||
|
Object string `json:"object"`
|
||||||
|
Created int `json:"created"`
|
||||||
|
Model string `json:"model"`
|
||||||
|
Usage struct {
|
||||||
|
PromptTokens int `json:"prompt_tokens"`
|
||||||
|
CompletionTokens int `json:"completion_tokens"`
|
||||||
|
TotalTokens int `json:"total_tokens"`
|
||||||
|
} `json:"usage"`
|
||||||
|
Choices []Choice `json:"choices"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type Choice struct {
|
||||||
|
Message Message
|
||||||
|
FinishReason string `json:"finish_reason"`
|
||||||
|
Index int `json:"index"`
|
||||||
|
}
|
||||||
12
vendor/github.com/golang/mock/AUTHORS
generated
vendored
Normal file
12
vendor/github.com/golang/mock/AUTHORS
generated
vendored
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
# This is the official list of GoMock authors for copyright purposes.
|
||||||
|
# This file is distinct from the CONTRIBUTORS files.
|
||||||
|
# See the latter for an explanation.
|
||||||
|
|
||||||
|
# Names should be added to this file as
|
||||||
|
# Name or Organization <email address>
|
||||||
|
# The email address is not required for organizations.
|
||||||
|
|
||||||
|
# Please keep the list sorted.
|
||||||
|
|
||||||
|
Alex Reece <awreece@gmail.com>
|
||||||
|
Google Inc.
|
||||||
37
vendor/github.com/golang/mock/CONTRIBUTORS
generated
vendored
Normal file
37
vendor/github.com/golang/mock/CONTRIBUTORS
generated
vendored
Normal file
@@ -0,0 +1,37 @@
|
|||||||
|
# This is the official list of people who can contribute (and typically
|
||||||
|
# have contributed) code to the gomock repository.
|
||||||
|
# The AUTHORS file lists the copyright holders; this file
|
||||||
|
# lists people. For example, Google employees are listed here
|
||||||
|
# but not in AUTHORS, because Google holds the copyright.
|
||||||
|
#
|
||||||
|
# The submission process automatically checks to make sure
|
||||||
|
# that people submitting code are listed in this file (by email address).
|
||||||
|
#
|
||||||
|
# Names should be added to this file only after verifying that
|
||||||
|
# the individual or the individual's organization has agreed to
|
||||||
|
# the appropriate Contributor License Agreement, found here:
|
||||||
|
#
|
||||||
|
# http://code.google.com/legal/individual-cla-v1.0.html
|
||||||
|
# http://code.google.com/legal/corporate-cla-v1.0.html
|
||||||
|
#
|
||||||
|
# The agreement for individuals can be filled out on the web.
|
||||||
|
#
|
||||||
|
# When adding J Random Contributor's name to this file,
|
||||||
|
# either J's name or J's organization's name should be
|
||||||
|
# added to the AUTHORS file, depending on whether the
|
||||||
|
# individual or corporate CLA was used.
|
||||||
|
|
||||||
|
# Names should be added to this file like so:
|
||||||
|
# Name <email address>
|
||||||
|
#
|
||||||
|
# An entry with two email addresses specifies that the
|
||||||
|
# first address should be used in the submit logs and
|
||||||
|
# that the second address should be recognized as the
|
||||||
|
# same person when interacting with Rietveld.
|
||||||
|
|
||||||
|
# Please keep the list sorted.
|
||||||
|
|
||||||
|
Aaron Jacobs <jacobsa@google.com> <aaronjjacobs@gmail.com>
|
||||||
|
Alex Reece <awreece@gmail.com>
|
||||||
|
David Symonds <dsymonds@golang.org>
|
||||||
|
Ryan Barrett <ryanb@google.com>
|
||||||
202
vendor/github.com/golang/mock/LICENSE
generated
vendored
Normal file
202
vendor/github.com/golang/mock/LICENSE
generated
vendored
Normal file
@@ -0,0 +1,202 @@
|
|||||||
|
|
||||||
|
Apache License
|
||||||
|
Version 2.0, January 2004
|
||||||
|
http://www.apache.org/licenses/
|
||||||
|
|
||||||
|
TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
|
||||||
|
|
||||||
|
1. Definitions.
|
||||||
|
|
||||||
|
"License" shall mean the terms and conditions for use, reproduction,
|
||||||
|
and distribution as defined by Sections 1 through 9 of this document.
|
||||||
|
|
||||||
|
"Licensor" shall mean the copyright owner or entity authorized by
|
||||||
|
the copyright owner that is granting the License.
|
||||||
|
|
||||||
|
"Legal Entity" shall mean the union of the acting entity and all
|
||||||
|
other entities that control, are controlled by, or are under common
|
||||||
|
control with that entity. For the purposes of this definition,
|
||||||
|
"control" means (i) the power, direct or indirect, to cause the
|
||||||
|
direction or management of such entity, whether by contract or
|
||||||
|
otherwise, or (ii) ownership of fifty percent (50%) or more of the
|
||||||
|
outstanding shares, or (iii) beneficial ownership of such entity.
|
||||||
|
|
||||||
|
"You" (or "Your") shall mean an individual or Legal Entity
|
||||||
|
exercising permissions granted by this License.
|
||||||
|
|
||||||
|
"Source" form shall mean the preferred form for making modifications,
|
||||||
|
including but not limited to software source code, documentation
|
||||||
|
source, and configuration files.
|
||||||
|
|
||||||
|
"Object" form shall mean any form resulting from mechanical
|
||||||
|
transformation or translation of a Source form, including but
|
||||||
|
not limited to compiled object code, generated documentation,
|
||||||
|
and conversions to other media types.
|
||||||
|
|
||||||
|
"Work" shall mean the work of authorship, whether in Source or
|
||||||
|
Object form, made available under the License, as indicated by a
|
||||||
|
copyright notice that is included in or attached to the work
|
||||||
|
(an example is provided in the Appendix below).
|
||||||
|
|
||||||
|
"Derivative Works" shall mean any work, whether in Source or Object
|
||||||
|
form, that is based on (or derived from) the Work and for which the
|
||||||
|
editorial revisions, annotations, elaborations, or other modifications
|
||||||
|
represent, as a whole, an original work of authorship. For the purposes
|
||||||
|
of this License, Derivative Works shall not include works that remain
|
||||||
|
separable from, or merely link (or bind by name) to the interfaces of,
|
||||||
|
the Work and Derivative Works thereof.
|
||||||
|
|
||||||
|
"Contribution" shall mean any work of authorship, including
|
||||||
|
the original version of the Work and any modifications or additions
|
||||||
|
to that Work or Derivative Works thereof, that is intentionally
|
||||||
|
submitted to Licensor for inclusion in the Work by the copyright owner
|
||||||
|
or by an individual or Legal Entity authorized to submit on behalf of
|
||||||
|
the copyright owner. For the purposes of this definition, "submitted"
|
||||||
|
means any form of electronic, verbal, or written communication sent
|
||||||
|
to the Licensor or its representatives, including but not limited to
|
||||||
|
communication on electronic mailing lists, source code control systems,
|
||||||
|
and issue tracking systems that are managed by, or on behalf of, the
|
||||||
|
Licensor for the purpose of discussing and improving the Work, but
|
||||||
|
excluding communication that is conspicuously marked or otherwise
|
||||||
|
designated in writing by the copyright owner as "Not a Contribution."
|
||||||
|
|
||||||
|
"Contributor" shall mean Licensor and any individual or Legal Entity
|
||||||
|
on behalf of whom a Contribution has been received by Licensor and
|
||||||
|
subsequently incorporated within the Work.
|
||||||
|
|
||||||
|
2. Grant of Copyright License. Subject to the terms and conditions of
|
||||||
|
this License, each Contributor hereby grants to You a perpetual,
|
||||||
|
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
||||||
|
copyright license to reproduce, prepare Derivative Works of,
|
||||||
|
publicly display, publicly perform, sublicense, and distribute the
|
||||||
|
Work and such Derivative Works in Source or Object form.
|
||||||
|
|
||||||
|
3. Grant of Patent License. Subject to the terms and conditions of
|
||||||
|
this License, each Contributor hereby grants to You a perpetual,
|
||||||
|
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
||||||
|
(except as stated in this section) patent license to make, have made,
|
||||||
|
use, offer to sell, sell, import, and otherwise transfer the Work,
|
||||||
|
where such license applies only to those patent claims licensable
|
||||||
|
by such Contributor that are necessarily infringed by their
|
||||||
|
Contribution(s) alone or by combination of their Contribution(s)
|
||||||
|
with the Work to which such Contribution(s) was submitted. If You
|
||||||
|
institute patent litigation against any entity (including a
|
||||||
|
cross-claim or counterclaim in a lawsuit) alleging that the Work
|
||||||
|
or a Contribution incorporated within the Work constitutes direct
|
||||||
|
or contributory patent infringement, then any patent licenses
|
||||||
|
granted to You under this License for that Work shall terminate
|
||||||
|
as of the date such litigation is filed.
|
||||||
|
|
||||||
|
4. Redistribution. You may reproduce and distribute copies of the
|
||||||
|
Work or Derivative Works thereof in any medium, with or without
|
||||||
|
modifications, and in Source or Object form, provided that You
|
||||||
|
meet the following conditions:
|
||||||
|
|
||||||
|
(a) You must give any other recipients of the Work or
|
||||||
|
Derivative Works a copy of this License; and
|
||||||
|
|
||||||
|
(b) You must cause any modified files to carry prominent notices
|
||||||
|
stating that You changed the files; and
|
||||||
|
|
||||||
|
(c) You must retain, in the Source form of any Derivative Works
|
||||||
|
that You distribute, all copyright, patent, trademark, and
|
||||||
|
attribution notices from the Source form of the Work,
|
||||||
|
excluding those notices that do not pertain to any part of
|
||||||
|
the Derivative Works; and
|
||||||
|
|
||||||
|
(d) If the Work includes a "NOTICE" text file as part of its
|
||||||
|
distribution, then any Derivative Works that You distribute must
|
||||||
|
include a readable copy of the attribution notices contained
|
||||||
|
within such NOTICE file, excluding those notices that do not
|
||||||
|
pertain to any part of the Derivative Works, in at least one
|
||||||
|
of the following places: within a NOTICE text file distributed
|
||||||
|
as part of the Derivative Works; within the Source form or
|
||||||
|
documentation, if provided along with the Derivative Works; or,
|
||||||
|
within a display generated by the Derivative Works, if and
|
||||||
|
wherever such third-party notices normally appear. The contents
|
||||||
|
of the NOTICE file are for informational purposes only and
|
||||||
|
do not modify the License. You may add Your own attribution
|
||||||
|
notices within Derivative Works that You distribute, alongside
|
||||||
|
or as an addendum to the NOTICE text from the Work, provided
|
||||||
|
that such additional attribution notices cannot be construed
|
||||||
|
as modifying the License.
|
||||||
|
|
||||||
|
You may add Your own copyright statement to Your modifications and
|
||||||
|
may provide additional or different license terms and conditions
|
||||||
|
for use, reproduction, or distribution of Your modifications, or
|
||||||
|
for any such Derivative Works as a whole, provided Your use,
|
||||||
|
reproduction, and distribution of the Work otherwise complies with
|
||||||
|
the conditions stated in this License.
|
||||||
|
|
||||||
|
5. Submission of Contributions. Unless You explicitly state otherwise,
|
||||||
|
any Contribution intentionally submitted for inclusion in the Work
|
||||||
|
by You to the Licensor shall be under the terms and conditions of
|
||||||
|
this License, without any additional terms or conditions.
|
||||||
|
Notwithstanding the above, nothing herein shall supersede or modify
|
||||||
|
the terms of any separate license agreement you may have executed
|
||||||
|
with Licensor regarding such Contributions.
|
||||||
|
|
||||||
|
6. Trademarks. This License does not grant permission to use the trade
|
||||||
|
names, trademarks, service marks, or product names of the Licensor,
|
||||||
|
except as required for reasonable and customary use in describing the
|
||||||
|
origin of the Work and reproducing the content of the NOTICE file.
|
||||||
|
|
||||||
|
7. Disclaimer of Warranty. Unless required by applicable law or
|
||||||
|
agreed to in writing, Licensor provides the Work (and each
|
||||||
|
Contributor provides its Contributions) on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
|
||||||
|
implied, including, without limitation, any warranties or conditions
|
||||||
|
of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
|
||||||
|
PARTICULAR PURPOSE. You are solely responsible for determining the
|
||||||
|
appropriateness of using or redistributing the Work and assume any
|
||||||
|
risks associated with Your exercise of permissions under this License.
|
||||||
|
|
||||||
|
8. Limitation of Liability. In no event and under no legal theory,
|
||||||
|
whether in tort (including negligence), contract, or otherwise,
|
||||||
|
unless required by applicable law (such as deliberate and grossly
|
||||||
|
negligent acts) or agreed to in writing, shall any Contributor be
|
||||||
|
liable to You for damages, including any direct, indirect, special,
|
||||||
|
incidental, or consequential damages of any character arising as a
|
||||||
|
result of this License or out of the use or inability to use the
|
||||||
|
Work (including but not limited to damages for loss of goodwill,
|
||||||
|
work stoppage, computer failure or malfunction, or any and all
|
||||||
|
other commercial damages or losses), even if such Contributor
|
||||||
|
has been advised of the possibility of such damages.
|
||||||
|
|
||||||
|
9. Accepting Warranty or Additional Liability. While redistributing
|
||||||
|
the Work or Derivative Works thereof, You may choose to offer,
|
||||||
|
and charge a fee for, acceptance of support, warranty, indemnity,
|
||||||
|
or other liability obligations and/or rights consistent with this
|
||||||
|
License. However, in accepting such obligations, You may act only
|
||||||
|
on Your own behalf and on Your sole responsibility, not on behalf
|
||||||
|
of any other Contributor, and only if You agree to indemnify,
|
||||||
|
defend, and hold each Contributor harmless for any liability
|
||||||
|
incurred by, or claims asserted against, such Contributor by reason
|
||||||
|
of your accepting any such warranty or additional liability.
|
||||||
|
|
||||||
|
END OF TERMS AND CONDITIONS
|
||||||
|
|
||||||
|
APPENDIX: How to apply the Apache License to your work.
|
||||||
|
|
||||||
|
To apply the Apache License to your work, attach the following
|
||||||
|
boilerplate notice, with the fields enclosed by brackets "[]"
|
||||||
|
replaced with your own identifying information. (Don't include
|
||||||
|
the brackets!) The text should be enclosed in the appropriate
|
||||||
|
comment syntax for the file format. We also recommend that a
|
||||||
|
file or class name and description of purpose be included on the
|
||||||
|
same "printed page" as the copyright notice for easier
|
||||||
|
identification within third-party archives.
|
||||||
|
|
||||||
|
Copyright [yyyy] [name of copyright owner]
|
||||||
|
|
||||||
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
you may not use this file except in compliance with the License.
|
||||||
|
You may obtain a copy of the License at
|
||||||
|
|
||||||
|
http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
|
||||||
|
Unless required by applicable law or agreed to in writing, software
|
||||||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
See the License for the specific language governing permissions and
|
||||||
|
limitations under the License.
|
||||||
445
vendor/github.com/golang/mock/gomock/call.go
generated
vendored
Normal file
445
vendor/github.com/golang/mock/gomock/call.go
generated
vendored
Normal file
@@ -0,0 +1,445 @@
|
|||||||
|
// Copyright 2010 Google Inc.
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License.
|
||||||
|
// You may obtain a copy of the License at
|
||||||
|
//
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
|
||||||
|
package gomock
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"reflect"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Call represents an expected call to a mock.
|
||||||
|
type Call struct {
|
||||||
|
t TestHelper // for triggering test failures on invalid call setup
|
||||||
|
|
||||||
|
receiver interface{} // the receiver of the method call
|
||||||
|
method string // the name of the method
|
||||||
|
methodType reflect.Type // the type of the method
|
||||||
|
args []Matcher // the args
|
||||||
|
origin string // file and line number of call setup
|
||||||
|
|
||||||
|
preReqs []*Call // prerequisite calls
|
||||||
|
|
||||||
|
// Expectations
|
||||||
|
minCalls, maxCalls int
|
||||||
|
|
||||||
|
numCalls int // actual number made
|
||||||
|
|
||||||
|
// actions are called when this Call is called. Each action gets the args and
|
||||||
|
// can set the return values by returning a non-nil slice. Actions run in the
|
||||||
|
// order they are created.
|
||||||
|
actions []func([]interface{}) []interface{}
|
||||||
|
}
|
||||||
|
|
||||||
|
// newCall creates a *Call. It requires the method type in order to support
|
||||||
|
// unexported methods.
|
||||||
|
func newCall(t TestHelper, receiver interface{}, method string, methodType reflect.Type, args ...interface{}) *Call {
|
||||||
|
t.Helper()
|
||||||
|
|
||||||
|
// TODO: check arity, types.
|
||||||
|
mArgs := make([]Matcher, len(args))
|
||||||
|
for i, arg := range args {
|
||||||
|
if m, ok := arg.(Matcher); ok {
|
||||||
|
mArgs[i] = m
|
||||||
|
} else if arg == nil {
|
||||||
|
// Handle nil specially so that passing a nil interface value
|
||||||
|
// will match the typed nils of concrete args.
|
||||||
|
mArgs[i] = Nil()
|
||||||
|
} else {
|
||||||
|
mArgs[i] = Eq(arg)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// callerInfo's skip should be updated if the number of calls between the user's test
|
||||||
|
// and this line changes, i.e. this code is wrapped in another anonymous function.
|
||||||
|
// 0 is us, 1 is RecordCallWithMethodType(), 2 is the generated recorder, and 3 is the user's test.
|
||||||
|
origin := callerInfo(3)
|
||||||
|
actions := []func([]interface{}) []interface{}{func([]interface{}) []interface{} {
|
||||||
|
// Synthesize the zero value for each of the return args' types.
|
||||||
|
rets := make([]interface{}, methodType.NumOut())
|
||||||
|
for i := 0; i < methodType.NumOut(); i++ {
|
||||||
|
rets[i] = reflect.Zero(methodType.Out(i)).Interface()
|
||||||
|
}
|
||||||
|
return rets
|
||||||
|
}}
|
||||||
|
return &Call{t: t, receiver: receiver, method: method, methodType: methodType,
|
||||||
|
args: mArgs, origin: origin, minCalls: 1, maxCalls: 1, actions: actions}
|
||||||
|
}
|
||||||
|
|
||||||
|
// AnyTimes allows the expectation to be called 0 or more times
|
||||||
|
func (c *Call) AnyTimes() *Call {
|
||||||
|
c.minCalls, c.maxCalls = 0, 1e8 // close enough to infinity
|
||||||
|
return c
|
||||||
|
}
|
||||||
|
|
||||||
|
// MinTimes requires the call to occur at least n times. If AnyTimes or MaxTimes have not been called or if MaxTimes
|
||||||
|
// was previously called with 1, MinTimes also sets the maximum number of calls to infinity.
|
||||||
|
func (c *Call) MinTimes(n int) *Call {
|
||||||
|
c.minCalls = n
|
||||||
|
if c.maxCalls == 1 {
|
||||||
|
c.maxCalls = 1e8
|
||||||
|
}
|
||||||
|
return c
|
||||||
|
}
|
||||||
|
|
||||||
|
// MaxTimes limits the number of calls to n times. If AnyTimes or MinTimes have not been called or if MinTimes was
|
||||||
|
// previously called with 1, MaxTimes also sets the minimum number of calls to 0.
|
||||||
|
func (c *Call) MaxTimes(n int) *Call {
|
||||||
|
c.maxCalls = n
|
||||||
|
if c.minCalls == 1 {
|
||||||
|
c.minCalls = 0
|
||||||
|
}
|
||||||
|
return c
|
||||||
|
}
|
||||||
|
|
||||||
|
// DoAndReturn declares the action to run when the call is matched.
|
||||||
|
// The return values from this function are returned by the mocked function.
|
||||||
|
// It takes an interface{} argument to support n-arity functions.
|
||||||
|
func (c *Call) DoAndReturn(f interface{}) *Call {
|
||||||
|
// TODO: Check arity and types here, rather than dying badly elsewhere.
|
||||||
|
v := reflect.ValueOf(f)
|
||||||
|
|
||||||
|
c.addAction(func(args []interface{}) []interface{} {
|
||||||
|
c.t.Helper()
|
||||||
|
vArgs := make([]reflect.Value, len(args))
|
||||||
|
ft := v.Type()
|
||||||
|
if c.methodType.NumIn() != ft.NumIn() {
|
||||||
|
c.t.Fatalf("wrong number of arguments in DoAndReturn func for %T.%v: got %d, want %d [%s]",
|
||||||
|
c.receiver, c.method, ft.NumIn(), c.methodType.NumIn(), c.origin)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
for i := 0; i < len(args); i++ {
|
||||||
|
if args[i] != nil {
|
||||||
|
vArgs[i] = reflect.ValueOf(args[i])
|
||||||
|
} else {
|
||||||
|
// Use the zero value for the arg.
|
||||||
|
vArgs[i] = reflect.Zero(ft.In(i))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
vRets := v.Call(vArgs)
|
||||||
|
rets := make([]interface{}, len(vRets))
|
||||||
|
for i, ret := range vRets {
|
||||||
|
rets[i] = ret.Interface()
|
||||||
|
}
|
||||||
|
return rets
|
||||||
|
})
|
||||||
|
return c
|
||||||
|
}
|
||||||
|
|
||||||
|
// Do declares the action to run when the call is matched. The function's
|
||||||
|
// return values are ignored to retain backward compatibility. To use the
|
||||||
|
// return values call DoAndReturn.
|
||||||
|
// It takes an interface{} argument to support n-arity functions.
|
||||||
|
func (c *Call) Do(f interface{}) *Call {
|
||||||
|
// TODO: Check arity and types here, rather than dying badly elsewhere.
|
||||||
|
v := reflect.ValueOf(f)
|
||||||
|
|
||||||
|
c.addAction(func(args []interface{}) []interface{} {
|
||||||
|
c.t.Helper()
|
||||||
|
if c.methodType.NumIn() != v.Type().NumIn() {
|
||||||
|
c.t.Fatalf("wrong number of arguments in Do func for %T.%v: got %d, want %d [%s]",
|
||||||
|
c.receiver, c.method, v.Type().NumIn(), c.methodType.NumIn(), c.origin)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
vArgs := make([]reflect.Value, len(args))
|
||||||
|
ft := v.Type()
|
||||||
|
for i := 0; i < len(args); i++ {
|
||||||
|
if args[i] != nil {
|
||||||
|
vArgs[i] = reflect.ValueOf(args[i])
|
||||||
|
} else {
|
||||||
|
// Use the zero value for the arg.
|
||||||
|
vArgs[i] = reflect.Zero(ft.In(i))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
v.Call(vArgs)
|
||||||
|
return nil
|
||||||
|
})
|
||||||
|
return c
|
||||||
|
}
|
||||||
|
|
||||||
|
// Return declares the values to be returned by the mocked function call.
|
||||||
|
func (c *Call) Return(rets ...interface{}) *Call {
|
||||||
|
c.t.Helper()
|
||||||
|
|
||||||
|
mt := c.methodType
|
||||||
|
if len(rets) != mt.NumOut() {
|
||||||
|
c.t.Fatalf("wrong number of arguments to Return for %T.%v: got %d, want %d [%s]",
|
||||||
|
c.receiver, c.method, len(rets), mt.NumOut(), c.origin)
|
||||||
|
}
|
||||||
|
for i, ret := range rets {
|
||||||
|
if got, want := reflect.TypeOf(ret), mt.Out(i); got == want {
|
||||||
|
// Identical types; nothing to do.
|
||||||
|
} else if got == nil {
|
||||||
|
// Nil needs special handling.
|
||||||
|
switch want.Kind() {
|
||||||
|
case reflect.Chan, reflect.Func, reflect.Interface, reflect.Map, reflect.Ptr, reflect.Slice:
|
||||||
|
// ok
|
||||||
|
default:
|
||||||
|
c.t.Fatalf("argument %d to Return for %T.%v is nil, but %v is not nillable [%s]",
|
||||||
|
i, c.receiver, c.method, want, c.origin)
|
||||||
|
}
|
||||||
|
} else if got.AssignableTo(want) {
|
||||||
|
// Assignable type relation. Make the assignment now so that the generated code
|
||||||
|
// can return the values with a type assertion.
|
||||||
|
v := reflect.New(want).Elem()
|
||||||
|
v.Set(reflect.ValueOf(ret))
|
||||||
|
rets[i] = v.Interface()
|
||||||
|
} else {
|
||||||
|
c.t.Fatalf("wrong type of argument %d to Return for %T.%v: %v is not assignable to %v [%s]",
|
||||||
|
i, c.receiver, c.method, got, want, c.origin)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
c.addAction(func([]interface{}) []interface{} {
|
||||||
|
return rets
|
||||||
|
})
|
||||||
|
|
||||||
|
return c
|
||||||
|
}
|
||||||
|
|
||||||
|
// Times declares the exact number of times a function call is expected to be executed.
|
||||||
|
func (c *Call) Times(n int) *Call {
|
||||||
|
c.minCalls, c.maxCalls = n, n
|
||||||
|
return c
|
||||||
|
}
|
||||||
|
|
||||||
|
// SetArg declares an action that will set the nth argument's value,
|
||||||
|
// indirected through a pointer. Or, in the case of a slice, SetArg
|
||||||
|
// will copy value's elements into the nth argument.
|
||||||
|
func (c *Call) SetArg(n int, value interface{}) *Call {
|
||||||
|
c.t.Helper()
|
||||||
|
|
||||||
|
mt := c.methodType
|
||||||
|
// TODO: This will break on variadic methods.
|
||||||
|
// We will need to check those at invocation time.
|
||||||
|
if n < 0 || n >= mt.NumIn() {
|
||||||
|
c.t.Fatalf("SetArg(%d, ...) called for a method with %d args [%s]",
|
||||||
|
n, mt.NumIn(), c.origin)
|
||||||
|
}
|
||||||
|
// Permit setting argument through an interface.
|
||||||
|
// In the interface case, we don't (nay, can't) check the type here.
|
||||||
|
at := mt.In(n)
|
||||||
|
switch at.Kind() {
|
||||||
|
case reflect.Ptr:
|
||||||
|
dt := at.Elem()
|
||||||
|
if vt := reflect.TypeOf(value); !vt.AssignableTo(dt) {
|
||||||
|
c.t.Fatalf("SetArg(%d, ...) argument is a %v, not assignable to %v [%s]",
|
||||||
|
n, vt, dt, c.origin)
|
||||||
|
}
|
||||||
|
case reflect.Interface:
|
||||||
|
// nothing to do
|
||||||
|
case reflect.Slice:
|
||||||
|
// nothing to do
|
||||||
|
default:
|
||||||
|
c.t.Fatalf("SetArg(%d, ...) referring to argument of non-pointer non-interface non-slice type %v [%s]",
|
||||||
|
n, at, c.origin)
|
||||||
|
}
|
||||||
|
|
||||||
|
c.addAction(func(args []interface{}) []interface{} {
|
||||||
|
v := reflect.ValueOf(value)
|
||||||
|
switch reflect.TypeOf(args[n]).Kind() {
|
||||||
|
case reflect.Slice:
|
||||||
|
setSlice(args[n], v)
|
||||||
|
default:
|
||||||
|
reflect.ValueOf(args[n]).Elem().Set(v)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
})
|
||||||
|
return c
|
||||||
|
}
|
||||||
|
|
||||||
|
// isPreReq returns true if other is a direct or indirect prerequisite to c.
|
||||||
|
func (c *Call) isPreReq(other *Call) bool {
|
||||||
|
for _, preReq := range c.preReqs {
|
||||||
|
if other == preReq || preReq.isPreReq(other) {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
// After declares that the call may only match after preReq has been exhausted.
|
||||||
|
func (c *Call) After(preReq *Call) *Call {
|
||||||
|
c.t.Helper()
|
||||||
|
|
||||||
|
if c == preReq {
|
||||||
|
c.t.Fatalf("A call isn't allowed to be its own prerequisite")
|
||||||
|
}
|
||||||
|
if preReq.isPreReq(c) {
|
||||||
|
c.t.Fatalf("Loop in call order: %v is a prerequisite to %v (possibly indirectly).", c, preReq)
|
||||||
|
}
|
||||||
|
|
||||||
|
c.preReqs = append(c.preReqs, preReq)
|
||||||
|
return c
|
||||||
|
}
|
||||||
|
|
||||||
|
// Returns true if the minimum number of calls have been made.
|
||||||
|
func (c *Call) satisfied() bool {
|
||||||
|
return c.numCalls >= c.minCalls
|
||||||
|
}
|
||||||
|
|
||||||
|
// Returns true if the maximum number of calls have been made.
|
||||||
|
func (c *Call) exhausted() bool {
|
||||||
|
return c.numCalls >= c.maxCalls
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Call) String() string {
|
||||||
|
args := make([]string, len(c.args))
|
||||||
|
for i, arg := range c.args {
|
||||||
|
args[i] = arg.String()
|
||||||
|
}
|
||||||
|
arguments := strings.Join(args, ", ")
|
||||||
|
return fmt.Sprintf("%T.%v(%s) %s", c.receiver, c.method, arguments, c.origin)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Tests if the given call matches the expected call.
|
||||||
|
// If yes, returns nil. If no, returns error with message explaining why it does not match.
|
||||||
|
func (c *Call) matches(args []interface{}) error {
|
||||||
|
if !c.methodType.IsVariadic() {
|
||||||
|
if len(args) != len(c.args) {
|
||||||
|
return fmt.Errorf("expected call at %s has the wrong number of arguments. Got: %d, want: %d",
|
||||||
|
c.origin, len(args), len(c.args))
|
||||||
|
}
|
||||||
|
|
||||||
|
for i, m := range c.args {
|
||||||
|
if !m.Matches(args[i]) {
|
||||||
|
return fmt.Errorf(
|
||||||
|
"expected call at %s doesn't match the argument at index %d.\nGot: %v\nWant: %v",
|
||||||
|
c.origin, i, formatGottenArg(m, args[i]), m,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if len(c.args) < c.methodType.NumIn()-1 {
|
||||||
|
return fmt.Errorf("expected call at %s has the wrong number of matchers. Got: %d, want: %d",
|
||||||
|
c.origin, len(c.args), c.methodType.NumIn()-1)
|
||||||
|
}
|
||||||
|
if len(c.args) != c.methodType.NumIn() && len(args) != len(c.args) {
|
||||||
|
return fmt.Errorf("expected call at %s has the wrong number of arguments. Got: %d, want: %d",
|
||||||
|
c.origin, len(args), len(c.args))
|
||||||
|
}
|
||||||
|
if len(args) < len(c.args)-1 {
|
||||||
|
return fmt.Errorf("expected call at %s has the wrong number of arguments. Got: %d, want: greater than or equal to %d",
|
||||||
|
c.origin, len(args), len(c.args)-1)
|
||||||
|
}
|
||||||
|
|
||||||
|
for i, m := range c.args {
|
||||||
|
if i < c.methodType.NumIn()-1 {
|
||||||
|
// Non-variadic args
|
||||||
|
if !m.Matches(args[i]) {
|
||||||
|
return fmt.Errorf("expected call at %s doesn't match the argument at index %s.\nGot: %v\nWant: %v",
|
||||||
|
c.origin, strconv.Itoa(i), formatGottenArg(m, args[i]), m)
|
||||||
|
}
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
// The last arg has a possibility of a variadic argument, so let it branch
|
||||||
|
|
||||||
|
// sample: Foo(a int, b int, c ...int)
|
||||||
|
if i < len(c.args) && i < len(args) {
|
||||||
|
if m.Matches(args[i]) {
|
||||||
|
// Got Foo(a, b, c) want Foo(matcherA, matcherB, gomock.Any())
|
||||||
|
// Got Foo(a, b, c) want Foo(matcherA, matcherB, someSliceMatcher)
|
||||||
|
// Got Foo(a, b, c) want Foo(matcherA, matcherB, matcherC)
|
||||||
|
// Got Foo(a, b) want Foo(matcherA, matcherB)
|
||||||
|
// Got Foo(a, b, c, d) want Foo(matcherA, matcherB, matcherC, matcherD)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// The number of actual args don't match the number of matchers,
|
||||||
|
// or the last matcher is a slice and the last arg is not.
|
||||||
|
// If this function still matches it is because the last matcher
|
||||||
|
// matches all the remaining arguments or the lack of any.
|
||||||
|
// Convert the remaining arguments, if any, into a slice of the
|
||||||
|
// expected type.
|
||||||
|
vArgsType := c.methodType.In(c.methodType.NumIn() - 1)
|
||||||
|
vArgs := reflect.MakeSlice(vArgsType, 0, len(args)-i)
|
||||||
|
for _, arg := range args[i:] {
|
||||||
|
vArgs = reflect.Append(vArgs, reflect.ValueOf(arg))
|
||||||
|
}
|
||||||
|
if m.Matches(vArgs.Interface()) {
|
||||||
|
// Got Foo(a, b, c, d, e) want Foo(matcherA, matcherB, gomock.Any())
|
||||||
|
// Got Foo(a, b, c, d, e) want Foo(matcherA, matcherB, someSliceMatcher)
|
||||||
|
// Got Foo(a, b) want Foo(matcherA, matcherB, gomock.Any())
|
||||||
|
// Got Foo(a, b) want Foo(matcherA, matcherB, someEmptySliceMatcher)
|
||||||
|
break
|
||||||
|
}
|
||||||
|
// Wrong number of matchers or not match. Fail.
|
||||||
|
// Got Foo(a, b) want Foo(matcherA, matcherB, matcherC, matcherD)
|
||||||
|
// Got Foo(a, b, c) want Foo(matcherA, matcherB, matcherC, matcherD)
|
||||||
|
// Got Foo(a, b, c, d) want Foo(matcherA, matcherB, matcherC, matcherD, matcherE)
|
||||||
|
// Got Foo(a, b, c, d, e) want Foo(matcherA, matcherB, matcherC, matcherD)
|
||||||
|
// Got Foo(a, b, c) want Foo(matcherA, matcherB)
|
||||||
|
|
||||||
|
return fmt.Errorf("expected call at %s doesn't match the argument at index %s.\nGot: %v\nWant: %v",
|
||||||
|
c.origin, strconv.Itoa(i), formatGottenArg(m, args[i:]), c.args[i])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check that all prerequisite calls have been satisfied.
|
||||||
|
for _, preReqCall := range c.preReqs {
|
||||||
|
if !preReqCall.satisfied() {
|
||||||
|
return fmt.Errorf("expected call at %s doesn't have a prerequisite call satisfied:\n%v\nshould be called before:\n%v",
|
||||||
|
c.origin, preReqCall, c)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check that the call is not exhausted.
|
||||||
|
if c.exhausted() {
|
||||||
|
return fmt.Errorf("expected call at %s has already been called the max number of times", c.origin)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// dropPrereqs tells the expected Call to not re-check prerequisite calls any
|
||||||
|
// longer, and to return its current set.
|
||||||
|
func (c *Call) dropPrereqs() (preReqs []*Call) {
|
||||||
|
preReqs = c.preReqs
|
||||||
|
c.preReqs = nil
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Call) call() []func([]interface{}) []interface{} {
|
||||||
|
c.numCalls++
|
||||||
|
return c.actions
|
||||||
|
}
|
||||||
|
|
||||||
|
// InOrder declares that the given calls should occur in order.
|
||||||
|
func InOrder(calls ...*Call) {
|
||||||
|
for i := 1; i < len(calls); i++ {
|
||||||
|
calls[i].After(calls[i-1])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func setSlice(arg interface{}, v reflect.Value) {
|
||||||
|
va := reflect.ValueOf(arg)
|
||||||
|
for i := 0; i < v.Len(); i++ {
|
||||||
|
va.Index(i).Set(v.Index(i))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Call) addAction(action func([]interface{}) []interface{}) {
|
||||||
|
c.actions = append(c.actions, action)
|
||||||
|
}
|
||||||
|
|
||||||
|
func formatGottenArg(m Matcher, arg interface{}) string {
|
||||||
|
got := fmt.Sprintf("%v (%T)", arg, arg)
|
||||||
|
if gs, ok := m.(GotFormatter); ok {
|
||||||
|
got = gs.Got(arg)
|
||||||
|
}
|
||||||
|
return got
|
||||||
|
}
|
||||||
113
vendor/github.com/golang/mock/gomock/callset.go
generated
vendored
Normal file
113
vendor/github.com/golang/mock/gomock/callset.go
generated
vendored
Normal file
@@ -0,0 +1,113 @@
|
|||||||
|
// Copyright 2011 Google Inc.
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License.
|
||||||
|
// You may obtain a copy of the License at
|
||||||
|
//
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
|
||||||
|
package gomock
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
)
|
||||||
|
|
||||||
|
// callSet represents a set of expected calls, indexed by receiver and method
|
||||||
|
// name.
|
||||||
|
type callSet struct {
|
||||||
|
// Calls that are still expected.
|
||||||
|
expected map[callSetKey][]*Call
|
||||||
|
// Calls that have been exhausted.
|
||||||
|
exhausted map[callSetKey][]*Call
|
||||||
|
}
|
||||||
|
|
||||||
|
// callSetKey is the key in the maps in callSet
|
||||||
|
type callSetKey struct {
|
||||||
|
receiver interface{}
|
||||||
|
fname string
|
||||||
|
}
|
||||||
|
|
||||||
|
func newCallSet() *callSet {
|
||||||
|
return &callSet{make(map[callSetKey][]*Call), make(map[callSetKey][]*Call)}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add adds a new expected call.
|
||||||
|
func (cs callSet) Add(call *Call) {
|
||||||
|
key := callSetKey{call.receiver, call.method}
|
||||||
|
m := cs.expected
|
||||||
|
if call.exhausted() {
|
||||||
|
m = cs.exhausted
|
||||||
|
}
|
||||||
|
m[key] = append(m[key], call)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Remove removes an expected call.
|
||||||
|
func (cs callSet) Remove(call *Call) {
|
||||||
|
key := callSetKey{call.receiver, call.method}
|
||||||
|
calls := cs.expected[key]
|
||||||
|
for i, c := range calls {
|
||||||
|
if c == call {
|
||||||
|
// maintain order for remaining calls
|
||||||
|
cs.expected[key] = append(calls[:i], calls[i+1:]...)
|
||||||
|
cs.exhausted[key] = append(cs.exhausted[key], call)
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// FindMatch searches for a matching call. Returns error with explanation message if no call matched.
|
||||||
|
func (cs callSet) FindMatch(receiver interface{}, method string, args []interface{}) (*Call, error) {
|
||||||
|
key := callSetKey{receiver, method}
|
||||||
|
|
||||||
|
// Search through the expected calls.
|
||||||
|
expected := cs.expected[key]
|
||||||
|
var callsErrors bytes.Buffer
|
||||||
|
for _, call := range expected {
|
||||||
|
err := call.matches(args)
|
||||||
|
if err != nil {
|
||||||
|
_, _ = fmt.Fprintf(&callsErrors, "\n%v", err)
|
||||||
|
} else {
|
||||||
|
return call, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// If we haven't found a match then search through the exhausted calls so we
|
||||||
|
// get useful error messages.
|
||||||
|
exhausted := cs.exhausted[key]
|
||||||
|
for _, call := range exhausted {
|
||||||
|
if err := call.matches(args); err != nil {
|
||||||
|
_, _ = fmt.Fprintf(&callsErrors, "\n%v", err)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
_, _ = fmt.Fprintf(
|
||||||
|
&callsErrors, "all expected calls for method %q have been exhausted", method,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(expected)+len(exhausted) == 0 {
|
||||||
|
_, _ = fmt.Fprintf(&callsErrors, "there are no expected calls of the method %q for that receiver", method)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil, errors.New(callsErrors.String())
|
||||||
|
}
|
||||||
|
|
||||||
|
// Failures returns the calls that are not satisfied.
|
||||||
|
func (cs callSet) Failures() []*Call {
|
||||||
|
failures := make([]*Call, 0, len(cs.expected))
|
||||||
|
for _, calls := range cs.expected {
|
||||||
|
for _, call := range calls {
|
||||||
|
if !call.satisfied() {
|
||||||
|
failures = append(failures, call)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return failures
|
||||||
|
}
|
||||||
336
vendor/github.com/golang/mock/gomock/controller.go
generated
vendored
Normal file
336
vendor/github.com/golang/mock/gomock/controller.go
generated
vendored
Normal file
@@ -0,0 +1,336 @@
|
|||||||
|
// Copyright 2010 Google Inc.
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License.
|
||||||
|
// You may obtain a copy of the License at
|
||||||
|
//
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
|
||||||
|
// Package gomock is a mock framework for Go.
|
||||||
|
//
|
||||||
|
// Standard usage:
|
||||||
|
// (1) Define an interface that you wish to mock.
|
||||||
|
// type MyInterface interface {
|
||||||
|
// SomeMethod(x int64, y string)
|
||||||
|
// }
|
||||||
|
// (2) Use mockgen to generate a mock from the interface.
|
||||||
|
// (3) Use the mock in a test:
|
||||||
|
// func TestMyThing(t *testing.T) {
|
||||||
|
// mockCtrl := gomock.NewController(t)
|
||||||
|
// defer mockCtrl.Finish()
|
||||||
|
//
|
||||||
|
// mockObj := something.NewMockMyInterface(mockCtrl)
|
||||||
|
// mockObj.EXPECT().SomeMethod(4, "blah")
|
||||||
|
// // pass mockObj to a real object and play with it.
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// By default, expected calls are not enforced to run in any particular order.
|
||||||
|
// Call order dependency can be enforced by use of InOrder and/or Call.After.
|
||||||
|
// Call.After can create more varied call order dependencies, but InOrder is
|
||||||
|
// often more convenient.
|
||||||
|
//
|
||||||
|
// The following examples create equivalent call order dependencies.
|
||||||
|
//
|
||||||
|
// Example of using Call.After to chain expected call order:
|
||||||
|
//
|
||||||
|
// firstCall := mockObj.EXPECT().SomeMethod(1, "first")
|
||||||
|
// secondCall := mockObj.EXPECT().SomeMethod(2, "second").After(firstCall)
|
||||||
|
// mockObj.EXPECT().SomeMethod(3, "third").After(secondCall)
|
||||||
|
//
|
||||||
|
// Example of using InOrder to declare expected call order:
|
||||||
|
//
|
||||||
|
// gomock.InOrder(
|
||||||
|
// mockObj.EXPECT().SomeMethod(1, "first"),
|
||||||
|
// mockObj.EXPECT().SomeMethod(2, "second"),
|
||||||
|
// mockObj.EXPECT().SomeMethod(3, "third"),
|
||||||
|
// )
|
||||||
|
package gomock
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
"reflect"
|
||||||
|
"runtime"
|
||||||
|
"sync"
|
||||||
|
)
|
||||||
|
|
||||||
|
// A TestReporter is something that can be used to report test failures. It
|
||||||
|
// is satisfied by the standard library's *testing.T.
|
||||||
|
type TestReporter interface {
|
||||||
|
Errorf(format string, args ...interface{})
|
||||||
|
Fatalf(format string, args ...interface{})
|
||||||
|
}
|
||||||
|
|
||||||
|
// TestHelper is a TestReporter that has the Helper method. It is satisfied
|
||||||
|
// by the standard library's *testing.T.
|
||||||
|
type TestHelper interface {
|
||||||
|
TestReporter
|
||||||
|
Helper()
|
||||||
|
}
|
||||||
|
|
||||||
|
// cleanuper is used to check if TestHelper also has the `Cleanup` method. A
|
||||||
|
// common pattern is to pass in a `*testing.T` to
|
||||||
|
// `NewController(t TestReporter)`. In Go 1.14+, `*testing.T` has a cleanup
|
||||||
|
// method. This can be utilized to call `Finish()` so the caller of this library
|
||||||
|
// does not have to.
|
||||||
|
type cleanuper interface {
|
||||||
|
Cleanup(func())
|
||||||
|
}
|
||||||
|
|
||||||
|
// A Controller represents the top-level control of a mock ecosystem. It
|
||||||
|
// defines the scope and lifetime of mock objects, as well as their
|
||||||
|
// expectations. It is safe to call Controller's methods from multiple
|
||||||
|
// goroutines. Each test should create a new Controller and invoke Finish via
|
||||||
|
// defer.
|
||||||
|
//
|
||||||
|
// func TestFoo(t *testing.T) {
|
||||||
|
// ctrl := gomock.NewController(t)
|
||||||
|
// defer ctrl.Finish()
|
||||||
|
// // ..
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// func TestBar(t *testing.T) {
|
||||||
|
// t.Run("Sub-Test-1", st) {
|
||||||
|
// ctrl := gomock.NewController(st)
|
||||||
|
// defer ctrl.Finish()
|
||||||
|
// // ..
|
||||||
|
// })
|
||||||
|
// t.Run("Sub-Test-2", st) {
|
||||||
|
// ctrl := gomock.NewController(st)
|
||||||
|
// defer ctrl.Finish()
|
||||||
|
// // ..
|
||||||
|
// })
|
||||||
|
// })
|
||||||
|
type Controller struct {
|
||||||
|
// T should only be called within a generated mock. It is not intended to
|
||||||
|
// be used in user code and may be changed in future versions. T is the
|
||||||
|
// TestReporter passed in when creating the Controller via NewController.
|
||||||
|
// If the TestReporter does not implement a TestHelper it will be wrapped
|
||||||
|
// with a nopTestHelper.
|
||||||
|
T TestHelper
|
||||||
|
mu sync.Mutex
|
||||||
|
expectedCalls *callSet
|
||||||
|
finished bool
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewController returns a new Controller. It is the preferred way to create a
|
||||||
|
// Controller.
|
||||||
|
//
|
||||||
|
// New in go1.14+, if you are passing a *testing.T into this function you no
|
||||||
|
// longer need to call ctrl.Finish() in your test methods.
|
||||||
|
func NewController(t TestReporter) *Controller {
|
||||||
|
h, ok := t.(TestHelper)
|
||||||
|
if !ok {
|
||||||
|
h = &nopTestHelper{t}
|
||||||
|
}
|
||||||
|
ctrl := &Controller{
|
||||||
|
T: h,
|
||||||
|
expectedCalls: newCallSet(),
|
||||||
|
}
|
||||||
|
if c, ok := isCleanuper(ctrl.T); ok {
|
||||||
|
c.Cleanup(func() {
|
||||||
|
ctrl.T.Helper()
|
||||||
|
ctrl.finish(true, nil)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
return ctrl
|
||||||
|
}
|
||||||
|
|
||||||
|
type cancelReporter struct {
|
||||||
|
t TestHelper
|
||||||
|
cancel func()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *cancelReporter) Errorf(format string, args ...interface{}) {
|
||||||
|
r.t.Errorf(format, args...)
|
||||||
|
}
|
||||||
|
func (r *cancelReporter) Fatalf(format string, args ...interface{}) {
|
||||||
|
defer r.cancel()
|
||||||
|
r.t.Fatalf(format, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *cancelReporter) Helper() {
|
||||||
|
r.t.Helper()
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithContext returns a new Controller and a Context, which is cancelled on any
|
||||||
|
// fatal failure.
|
||||||
|
func WithContext(ctx context.Context, t TestReporter) (*Controller, context.Context) {
|
||||||
|
h, ok := t.(TestHelper)
|
||||||
|
if !ok {
|
||||||
|
h = &nopTestHelper{t: t}
|
||||||
|
}
|
||||||
|
|
||||||
|
ctx, cancel := context.WithCancel(ctx)
|
||||||
|
return NewController(&cancelReporter{t: h, cancel: cancel}), ctx
|
||||||
|
}
|
||||||
|
|
||||||
|
type nopTestHelper struct {
|
||||||
|
t TestReporter
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *nopTestHelper) Errorf(format string, args ...interface{}) {
|
||||||
|
h.t.Errorf(format, args...)
|
||||||
|
}
|
||||||
|
func (h *nopTestHelper) Fatalf(format string, args ...interface{}) {
|
||||||
|
h.t.Fatalf(format, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h nopTestHelper) Helper() {}
|
||||||
|
|
||||||
|
// RecordCall is called by a mock. It should not be called by user code.
|
||||||
|
func (ctrl *Controller) RecordCall(receiver interface{}, method string, args ...interface{}) *Call {
|
||||||
|
ctrl.T.Helper()
|
||||||
|
|
||||||
|
recv := reflect.ValueOf(receiver)
|
||||||
|
for i := 0; i < recv.Type().NumMethod(); i++ {
|
||||||
|
if recv.Type().Method(i).Name == method {
|
||||||
|
return ctrl.RecordCallWithMethodType(receiver, method, recv.Method(i).Type(), args...)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
ctrl.T.Fatalf("gomock: failed finding method %s on %T", method, receiver)
|
||||||
|
panic("unreachable")
|
||||||
|
}
|
||||||
|
|
||||||
|
// RecordCallWithMethodType is called by a mock. It should not be called by user code.
|
||||||
|
func (ctrl *Controller) RecordCallWithMethodType(receiver interface{}, method string, methodType reflect.Type, args ...interface{}) *Call {
|
||||||
|
ctrl.T.Helper()
|
||||||
|
|
||||||
|
call := newCall(ctrl.T, receiver, method, methodType, args...)
|
||||||
|
|
||||||
|
ctrl.mu.Lock()
|
||||||
|
defer ctrl.mu.Unlock()
|
||||||
|
ctrl.expectedCalls.Add(call)
|
||||||
|
|
||||||
|
return call
|
||||||
|
}
|
||||||
|
|
||||||
|
// Call is called by a mock. It should not be called by user code.
|
||||||
|
func (ctrl *Controller) Call(receiver interface{}, method string, args ...interface{}) []interface{} {
|
||||||
|
ctrl.T.Helper()
|
||||||
|
|
||||||
|
// Nest this code so we can use defer to make sure the lock is released.
|
||||||
|
actions := func() []func([]interface{}) []interface{} {
|
||||||
|
ctrl.T.Helper()
|
||||||
|
ctrl.mu.Lock()
|
||||||
|
defer ctrl.mu.Unlock()
|
||||||
|
|
||||||
|
expected, err := ctrl.expectedCalls.FindMatch(receiver, method, args)
|
||||||
|
if err != nil {
|
||||||
|
// callerInfo's skip should be updated if the number of calls between the user's test
|
||||||
|
// and this line changes, i.e. this code is wrapped in another anonymous function.
|
||||||
|
// 0 is us, 1 is controller.Call(), 2 is the generated mock, and 3 is the user's test.
|
||||||
|
origin := callerInfo(3)
|
||||||
|
ctrl.T.Fatalf("Unexpected call to %T.%v(%v) at %s because: %s", receiver, method, args, origin, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Two things happen here:
|
||||||
|
// * the matching call no longer needs to check prerequite calls,
|
||||||
|
// * and the prerequite calls are no longer expected, so remove them.
|
||||||
|
preReqCalls := expected.dropPrereqs()
|
||||||
|
for _, preReqCall := range preReqCalls {
|
||||||
|
ctrl.expectedCalls.Remove(preReqCall)
|
||||||
|
}
|
||||||
|
|
||||||
|
actions := expected.call()
|
||||||
|
if expected.exhausted() {
|
||||||
|
ctrl.expectedCalls.Remove(expected)
|
||||||
|
}
|
||||||
|
return actions
|
||||||
|
}()
|
||||||
|
|
||||||
|
var rets []interface{}
|
||||||
|
for _, action := range actions {
|
||||||
|
if r := action(args); r != nil {
|
||||||
|
rets = r
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return rets
|
||||||
|
}
|
||||||
|
|
||||||
|
// Finish checks to see if all the methods that were expected to be called
|
||||||
|
// were called. It should be invoked for each Controller. It is not idempotent
|
||||||
|
// and therefore can only be invoked once.
|
||||||
|
//
|
||||||
|
// New in go1.14+, if you are passing a *testing.T into NewController function you no
|
||||||
|
// longer need to call ctrl.Finish() in your test methods.
|
||||||
|
func (ctrl *Controller) Finish() {
|
||||||
|
// If we're currently panicking, probably because this is a deferred call.
|
||||||
|
// This must be recovered in the deferred function.
|
||||||
|
err := recover()
|
||||||
|
ctrl.finish(false, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ctrl *Controller) finish(cleanup bool, panicErr interface{}) {
|
||||||
|
ctrl.T.Helper()
|
||||||
|
|
||||||
|
ctrl.mu.Lock()
|
||||||
|
defer ctrl.mu.Unlock()
|
||||||
|
|
||||||
|
if ctrl.finished {
|
||||||
|
if _, ok := isCleanuper(ctrl.T); !ok {
|
||||||
|
ctrl.T.Fatalf("Controller.Finish was called more than once. It has to be called exactly once.")
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
ctrl.finished = true
|
||||||
|
|
||||||
|
// Short-circuit, pass through the panic.
|
||||||
|
if panicErr != nil {
|
||||||
|
panic(panicErr)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check that all remaining expected calls are satisfied.
|
||||||
|
failures := ctrl.expectedCalls.Failures()
|
||||||
|
for _, call := range failures {
|
||||||
|
ctrl.T.Errorf("missing call(s) to %v", call)
|
||||||
|
}
|
||||||
|
if len(failures) != 0 {
|
||||||
|
if !cleanup {
|
||||||
|
ctrl.T.Fatalf("aborting test due to missing call(s)")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
ctrl.T.Errorf("aborting test due to missing call(s)")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// callerInfo returns the file:line of the call site. skip is the number
|
||||||
|
// of stack frames to skip when reporting. 0 is callerInfo's call site.
|
||||||
|
func callerInfo(skip int) string {
|
||||||
|
if _, file, line, ok := runtime.Caller(skip + 1); ok {
|
||||||
|
return fmt.Sprintf("%s:%d", file, line)
|
||||||
|
}
|
||||||
|
return "unknown file"
|
||||||
|
}
|
||||||
|
|
||||||
|
// isCleanuper checks it if t's base TestReporter has a Cleanup method.
|
||||||
|
func isCleanuper(t TestReporter) (cleanuper, bool) {
|
||||||
|
tr := unwrapTestReporter(t)
|
||||||
|
c, ok := tr.(cleanuper)
|
||||||
|
return c, ok
|
||||||
|
}
|
||||||
|
|
||||||
|
// unwrapTestReporter unwraps TestReporter to the base implementation.
|
||||||
|
func unwrapTestReporter(t TestReporter) TestReporter {
|
||||||
|
tr := t
|
||||||
|
switch nt := t.(type) {
|
||||||
|
case *cancelReporter:
|
||||||
|
tr = nt.t
|
||||||
|
if h, check := tr.(*nopTestHelper); check {
|
||||||
|
tr = h.t
|
||||||
|
}
|
||||||
|
case *nopTestHelper:
|
||||||
|
tr = nt.t
|
||||||
|
default:
|
||||||
|
// not wrapped
|
||||||
|
}
|
||||||
|
return tr
|
||||||
|
}
|
||||||
341
vendor/github.com/golang/mock/gomock/matchers.go
generated
vendored
Normal file
341
vendor/github.com/golang/mock/gomock/matchers.go
generated
vendored
Normal file
@@ -0,0 +1,341 @@
|
|||||||
|
// Copyright 2010 Google Inc.
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License.
|
||||||
|
// You may obtain a copy of the License at
|
||||||
|
//
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
|
||||||
|
package gomock
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"reflect"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
// A Matcher is a representation of a class of values.
|
||||||
|
// It is used to represent the valid or expected arguments to a mocked method.
|
||||||
|
type Matcher interface {
|
||||||
|
// Matches returns whether x is a match.
|
||||||
|
Matches(x interface{}) bool
|
||||||
|
|
||||||
|
// String describes what the matcher matches.
|
||||||
|
String() string
|
||||||
|
}
|
||||||
|
|
||||||
|
// WantFormatter modifies the given Matcher's String() method to the given
|
||||||
|
// Stringer. This allows for control on how the "Want" is formatted when
|
||||||
|
// printing .
|
||||||
|
func WantFormatter(s fmt.Stringer, m Matcher) Matcher {
|
||||||
|
type matcher interface {
|
||||||
|
Matches(x interface{}) bool
|
||||||
|
}
|
||||||
|
|
||||||
|
return struct {
|
||||||
|
matcher
|
||||||
|
fmt.Stringer
|
||||||
|
}{
|
||||||
|
matcher: m,
|
||||||
|
Stringer: s,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// StringerFunc type is an adapter to allow the use of ordinary functions as
|
||||||
|
// a Stringer. If f is a function with the appropriate signature,
|
||||||
|
// StringerFunc(f) is a Stringer that calls f.
|
||||||
|
type StringerFunc func() string
|
||||||
|
|
||||||
|
// String implements fmt.Stringer.
|
||||||
|
func (f StringerFunc) String() string {
|
||||||
|
return f()
|
||||||
|
}
|
||||||
|
|
||||||
|
// GotFormatter is used to better print failure messages. If a matcher
|
||||||
|
// implements GotFormatter, it will use the result from Got when printing
|
||||||
|
// the failure message.
|
||||||
|
type GotFormatter interface {
|
||||||
|
// Got is invoked with the received value. The result is used when
|
||||||
|
// printing the failure message.
|
||||||
|
Got(got interface{}) string
|
||||||
|
}
|
||||||
|
|
||||||
|
// GotFormatterFunc type is an adapter to allow the use of ordinary
|
||||||
|
// functions as a GotFormatter. If f is a function with the appropriate
|
||||||
|
// signature, GotFormatterFunc(f) is a GotFormatter that calls f.
|
||||||
|
type GotFormatterFunc func(got interface{}) string
|
||||||
|
|
||||||
|
// Got implements GotFormatter.
|
||||||
|
func (f GotFormatterFunc) Got(got interface{}) string {
|
||||||
|
return f(got)
|
||||||
|
}
|
||||||
|
|
||||||
|
// GotFormatterAdapter attaches a GotFormatter to a Matcher.
|
||||||
|
func GotFormatterAdapter(s GotFormatter, m Matcher) Matcher {
|
||||||
|
return struct {
|
||||||
|
GotFormatter
|
||||||
|
Matcher
|
||||||
|
}{
|
||||||
|
GotFormatter: s,
|
||||||
|
Matcher: m,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
type anyMatcher struct{}
|
||||||
|
|
||||||
|
func (anyMatcher) Matches(interface{}) bool {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
func (anyMatcher) String() string {
|
||||||
|
return "is anything"
|
||||||
|
}
|
||||||
|
|
||||||
|
type eqMatcher struct {
|
||||||
|
x interface{}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (e eqMatcher) Matches(x interface{}) bool {
|
||||||
|
// In case, some value is nil
|
||||||
|
if e.x == nil || x == nil {
|
||||||
|
return reflect.DeepEqual(e.x, x)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if types assignable and convert them to common type
|
||||||
|
x1Val := reflect.ValueOf(e.x)
|
||||||
|
x2Val := reflect.ValueOf(x)
|
||||||
|
|
||||||
|
if x1Val.Type().AssignableTo(x2Val.Type()) {
|
||||||
|
x1ValConverted := x1Val.Convert(x2Val.Type())
|
||||||
|
return reflect.DeepEqual(x1ValConverted.Interface(), x2Val.Interface())
|
||||||
|
}
|
||||||
|
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
func (e eqMatcher) String() string {
|
||||||
|
return fmt.Sprintf("is equal to %v (%T)", e.x, e.x)
|
||||||
|
}
|
||||||
|
|
||||||
|
type nilMatcher struct{}
|
||||||
|
|
||||||
|
func (nilMatcher) Matches(x interface{}) bool {
|
||||||
|
if x == nil {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
v := reflect.ValueOf(x)
|
||||||
|
switch v.Kind() {
|
||||||
|
case reflect.Chan, reflect.Func, reflect.Interface, reflect.Map,
|
||||||
|
reflect.Ptr, reflect.Slice:
|
||||||
|
return v.IsNil()
|
||||||
|
}
|
||||||
|
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
func (nilMatcher) String() string {
|
||||||
|
return "is nil"
|
||||||
|
}
|
||||||
|
|
||||||
|
type notMatcher struct {
|
||||||
|
m Matcher
|
||||||
|
}
|
||||||
|
|
||||||
|
func (n notMatcher) Matches(x interface{}) bool {
|
||||||
|
return !n.m.Matches(x)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (n notMatcher) String() string {
|
||||||
|
return "not(" + n.m.String() + ")"
|
||||||
|
}
|
||||||
|
|
||||||
|
type assignableToTypeOfMatcher struct {
|
||||||
|
targetType reflect.Type
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m assignableToTypeOfMatcher) Matches(x interface{}) bool {
|
||||||
|
return reflect.TypeOf(x).AssignableTo(m.targetType)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m assignableToTypeOfMatcher) String() string {
|
||||||
|
return "is assignable to " + m.targetType.Name()
|
||||||
|
}
|
||||||
|
|
||||||
|
type allMatcher struct {
|
||||||
|
matchers []Matcher
|
||||||
|
}
|
||||||
|
|
||||||
|
func (am allMatcher) Matches(x interface{}) bool {
|
||||||
|
for _, m := range am.matchers {
|
||||||
|
if !m.Matches(x) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
func (am allMatcher) String() string {
|
||||||
|
ss := make([]string, 0, len(am.matchers))
|
||||||
|
for _, matcher := range am.matchers {
|
||||||
|
ss = append(ss, matcher.String())
|
||||||
|
}
|
||||||
|
return strings.Join(ss, "; ")
|
||||||
|
}
|
||||||
|
|
||||||
|
type lenMatcher struct {
|
||||||
|
i int
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m lenMatcher) Matches(x interface{}) bool {
|
||||||
|
v := reflect.ValueOf(x)
|
||||||
|
switch v.Kind() {
|
||||||
|
case reflect.Array, reflect.Chan, reflect.Map, reflect.Slice, reflect.String:
|
||||||
|
return v.Len() == m.i
|
||||||
|
default:
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m lenMatcher) String() string {
|
||||||
|
return fmt.Sprintf("has length %d", m.i)
|
||||||
|
}
|
||||||
|
|
||||||
|
type inAnyOrderMatcher struct {
|
||||||
|
x interface{}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m inAnyOrderMatcher) Matches(x interface{}) bool {
|
||||||
|
given, ok := m.prepareValue(x)
|
||||||
|
if !ok {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
wanted, ok := m.prepareValue(m.x)
|
||||||
|
if !ok {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
if given.Len() != wanted.Len() {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
usedFromGiven := make([]bool, given.Len())
|
||||||
|
foundFromWanted := make([]bool, wanted.Len())
|
||||||
|
for i := 0; i < wanted.Len(); i++ {
|
||||||
|
wantedMatcher := Eq(wanted.Index(i).Interface())
|
||||||
|
for j := 0; j < given.Len(); j++ {
|
||||||
|
if usedFromGiven[j] {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if wantedMatcher.Matches(given.Index(j).Interface()) {
|
||||||
|
foundFromWanted[i] = true
|
||||||
|
usedFromGiven[j] = true
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
missingFromWanted := 0
|
||||||
|
for _, found := range foundFromWanted {
|
||||||
|
if !found {
|
||||||
|
missingFromWanted++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
extraInGiven := 0
|
||||||
|
for _, used := range usedFromGiven {
|
||||||
|
if !used {
|
||||||
|
extraInGiven++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return extraInGiven == 0 && missingFromWanted == 0
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m inAnyOrderMatcher) prepareValue(x interface{}) (reflect.Value, bool) {
|
||||||
|
xValue := reflect.ValueOf(x)
|
||||||
|
switch xValue.Kind() {
|
||||||
|
case reflect.Slice, reflect.Array:
|
||||||
|
return xValue, true
|
||||||
|
default:
|
||||||
|
return reflect.Value{}, false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m inAnyOrderMatcher) String() string {
|
||||||
|
return fmt.Sprintf("has the same elements as %v", m.x)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Constructors
|
||||||
|
|
||||||
|
// All returns a composite Matcher that returns true if and only all of the
|
||||||
|
// matchers return true.
|
||||||
|
func All(ms ...Matcher) Matcher { return allMatcher{ms} }
|
||||||
|
|
||||||
|
// Any returns a matcher that always matches.
|
||||||
|
func Any() Matcher { return anyMatcher{} }
|
||||||
|
|
||||||
|
// Eq returns a matcher that matches on equality.
|
||||||
|
//
|
||||||
|
// Example usage:
|
||||||
|
// Eq(5).Matches(5) // returns true
|
||||||
|
// Eq(5).Matches(4) // returns false
|
||||||
|
func Eq(x interface{}) Matcher { return eqMatcher{x} }
|
||||||
|
|
||||||
|
// Len returns a matcher that matches on length. This matcher returns false if
|
||||||
|
// is compared to a type that is not an array, chan, map, slice, or string.
|
||||||
|
func Len(i int) Matcher {
|
||||||
|
return lenMatcher{i}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Nil returns a matcher that matches if the received value is nil.
|
||||||
|
//
|
||||||
|
// Example usage:
|
||||||
|
// var x *bytes.Buffer
|
||||||
|
// Nil().Matches(x) // returns true
|
||||||
|
// x = &bytes.Buffer{}
|
||||||
|
// Nil().Matches(x) // returns false
|
||||||
|
func Nil() Matcher { return nilMatcher{} }
|
||||||
|
|
||||||
|
// Not reverses the results of its given child matcher.
|
||||||
|
//
|
||||||
|
// Example usage:
|
||||||
|
// Not(Eq(5)).Matches(4) // returns true
|
||||||
|
// Not(Eq(5)).Matches(5) // returns false
|
||||||
|
func Not(x interface{}) Matcher {
|
||||||
|
if m, ok := x.(Matcher); ok {
|
||||||
|
return notMatcher{m}
|
||||||
|
}
|
||||||
|
return notMatcher{Eq(x)}
|
||||||
|
}
|
||||||
|
|
||||||
|
// AssignableToTypeOf is a Matcher that matches if the parameter to the mock
|
||||||
|
// function is assignable to the type of the parameter to this function.
|
||||||
|
//
|
||||||
|
// Example usage:
|
||||||
|
// var s fmt.Stringer = &bytes.Buffer{}
|
||||||
|
// AssignableToTypeOf(s).Matches(time.Second) // returns true
|
||||||
|
// AssignableToTypeOf(s).Matches(99) // returns false
|
||||||
|
//
|
||||||
|
// var ctx = reflect.TypeOf((*context.Context)(nil)).Elem()
|
||||||
|
// AssignableToTypeOf(ctx).Matches(context.Background()) // returns true
|
||||||
|
func AssignableToTypeOf(x interface{}) Matcher {
|
||||||
|
if xt, ok := x.(reflect.Type); ok {
|
||||||
|
return assignableToTypeOfMatcher{xt}
|
||||||
|
}
|
||||||
|
return assignableToTypeOfMatcher{reflect.TypeOf(x)}
|
||||||
|
}
|
||||||
|
|
||||||
|
// InAnyOrder is a Matcher that returns true for collections of the same elements ignoring the order.
|
||||||
|
//
|
||||||
|
// Example usage:
|
||||||
|
// InAnyOrder([]int{1, 2, 3}).Matches([]int{1, 3, 2}) // returns true
|
||||||
|
// InAnyOrder([]int{1, 2, 3}).Matches([]int{1, 2}) // returns false
|
||||||
|
func InAnyOrder(x interface{}) Matcher {
|
||||||
|
return inAnyOrderMatcher{x}
|
||||||
|
}
|
||||||
495
vendor/github.com/golang/mock/mockgen/model/model.go
generated
vendored
Normal file
495
vendor/github.com/golang/mock/mockgen/model/model.go
generated
vendored
Normal file
@@ -0,0 +1,495 @@
|
|||||||
|
// Copyright 2012 Google Inc.
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License.
|
||||||
|
// You may obtain a copy of the License at
|
||||||
|
//
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
|
||||||
|
// Package model contains the data model necessary for generating mock implementations.
|
||||||
|
package model
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/gob"
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
"reflect"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
// pkgPath is the importable path for package model
|
||||||
|
const pkgPath = "github.com/golang/mock/mockgen/model"
|
||||||
|
|
||||||
|
// Package is a Go package. It may be a subset.
|
||||||
|
type Package struct {
|
||||||
|
Name string
|
||||||
|
PkgPath string
|
||||||
|
Interfaces []*Interface
|
||||||
|
DotImports []string
|
||||||
|
}
|
||||||
|
|
||||||
|
// Print writes the package name and its exported interfaces.
|
||||||
|
func (pkg *Package) Print(w io.Writer) {
|
||||||
|
_, _ = fmt.Fprintf(w, "package %s\n", pkg.Name)
|
||||||
|
for _, intf := range pkg.Interfaces {
|
||||||
|
intf.Print(w)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Imports returns the imports needed by the Package as a set of import paths.
|
||||||
|
func (pkg *Package) Imports() map[string]bool {
|
||||||
|
im := make(map[string]bool)
|
||||||
|
for _, intf := range pkg.Interfaces {
|
||||||
|
intf.addImports(im)
|
||||||
|
}
|
||||||
|
return im
|
||||||
|
}
|
||||||
|
|
||||||
|
// Interface is a Go interface.
|
||||||
|
type Interface struct {
|
||||||
|
Name string
|
||||||
|
Methods []*Method
|
||||||
|
}
|
||||||
|
|
||||||
|
// Print writes the interface name and its methods.
|
||||||
|
func (intf *Interface) Print(w io.Writer) {
|
||||||
|
_, _ = fmt.Fprintf(w, "interface %s\n", intf.Name)
|
||||||
|
for _, m := range intf.Methods {
|
||||||
|
m.Print(w)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (intf *Interface) addImports(im map[string]bool) {
|
||||||
|
for _, m := range intf.Methods {
|
||||||
|
m.addImports(im)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// AddMethod adds a new method, de-duplicating by method name.
|
||||||
|
func (intf *Interface) AddMethod(m *Method) {
|
||||||
|
for _, me := range intf.Methods {
|
||||||
|
if me.Name == m.Name {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
intf.Methods = append(intf.Methods, m)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Method is a single method of an interface.
|
||||||
|
type Method struct {
|
||||||
|
Name string
|
||||||
|
In, Out []*Parameter
|
||||||
|
Variadic *Parameter // may be nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Print writes the method name and its signature.
|
||||||
|
func (m *Method) Print(w io.Writer) {
|
||||||
|
_, _ = fmt.Fprintf(w, " - method %s\n", m.Name)
|
||||||
|
if len(m.In) > 0 {
|
||||||
|
_, _ = fmt.Fprintf(w, " in:\n")
|
||||||
|
for _, p := range m.In {
|
||||||
|
p.Print(w)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if m.Variadic != nil {
|
||||||
|
_, _ = fmt.Fprintf(w, " ...:\n")
|
||||||
|
m.Variadic.Print(w)
|
||||||
|
}
|
||||||
|
if len(m.Out) > 0 {
|
||||||
|
_, _ = fmt.Fprintf(w, " out:\n")
|
||||||
|
for _, p := range m.Out {
|
||||||
|
p.Print(w)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Method) addImports(im map[string]bool) {
|
||||||
|
for _, p := range m.In {
|
||||||
|
p.Type.addImports(im)
|
||||||
|
}
|
||||||
|
if m.Variadic != nil {
|
||||||
|
m.Variadic.Type.addImports(im)
|
||||||
|
}
|
||||||
|
for _, p := range m.Out {
|
||||||
|
p.Type.addImports(im)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Parameter is an argument or return parameter of a method.
|
||||||
|
type Parameter struct {
|
||||||
|
Name string // may be empty
|
||||||
|
Type Type
|
||||||
|
}
|
||||||
|
|
||||||
|
// Print writes a method parameter.
|
||||||
|
func (p *Parameter) Print(w io.Writer) {
|
||||||
|
n := p.Name
|
||||||
|
if n == "" {
|
||||||
|
n = `""`
|
||||||
|
}
|
||||||
|
_, _ = fmt.Fprintf(w, " - %v: %v\n", n, p.Type.String(nil, ""))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Type is a Go type.
|
||||||
|
type Type interface {
|
||||||
|
String(pm map[string]string, pkgOverride string) string
|
||||||
|
addImports(im map[string]bool)
|
||||||
|
}
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
gob.Register(&ArrayType{})
|
||||||
|
gob.Register(&ChanType{})
|
||||||
|
gob.Register(&FuncType{})
|
||||||
|
gob.Register(&MapType{})
|
||||||
|
gob.Register(&NamedType{})
|
||||||
|
gob.Register(&PointerType{})
|
||||||
|
|
||||||
|
// Call gob.RegisterName to make sure it has the consistent name registered
|
||||||
|
// for both gob decoder and encoder.
|
||||||
|
//
|
||||||
|
// For a non-pointer type, gob.Register will try to get package full path by
|
||||||
|
// calling rt.PkgPath() for a name to register. If your project has vendor
|
||||||
|
// directory, it is possible that PkgPath will get a path like this:
|
||||||
|
// ../../../vendor/github.com/golang/mock/mockgen/model
|
||||||
|
gob.RegisterName(pkgPath+".PredeclaredType", PredeclaredType(""))
|
||||||
|
}
|
||||||
|
|
||||||
|
// ArrayType is an array or slice type.
|
||||||
|
type ArrayType struct {
|
||||||
|
Len int // -1 for slices, >= 0 for arrays
|
||||||
|
Type Type
|
||||||
|
}
|
||||||
|
|
||||||
|
func (at *ArrayType) String(pm map[string]string, pkgOverride string) string {
|
||||||
|
s := "[]"
|
||||||
|
if at.Len > -1 {
|
||||||
|
s = fmt.Sprintf("[%d]", at.Len)
|
||||||
|
}
|
||||||
|
return s + at.Type.String(pm, pkgOverride)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (at *ArrayType) addImports(im map[string]bool) { at.Type.addImports(im) }
|
||||||
|
|
||||||
|
// ChanType is a channel type.
|
||||||
|
type ChanType struct {
|
||||||
|
Dir ChanDir // 0, 1 or 2
|
||||||
|
Type Type
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ct *ChanType) String(pm map[string]string, pkgOverride string) string {
|
||||||
|
s := ct.Type.String(pm, pkgOverride)
|
||||||
|
if ct.Dir == RecvDir {
|
||||||
|
return "<-chan " + s
|
||||||
|
}
|
||||||
|
if ct.Dir == SendDir {
|
||||||
|
return "chan<- " + s
|
||||||
|
}
|
||||||
|
return "chan " + s
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ct *ChanType) addImports(im map[string]bool) { ct.Type.addImports(im) }
|
||||||
|
|
||||||
|
// ChanDir is a channel direction.
|
||||||
|
type ChanDir int
|
||||||
|
|
||||||
|
// Constants for channel directions.
|
||||||
|
const (
|
||||||
|
RecvDir ChanDir = 1
|
||||||
|
SendDir ChanDir = 2
|
||||||
|
)
|
||||||
|
|
||||||
|
// FuncType is a function type.
|
||||||
|
type FuncType struct {
|
||||||
|
In, Out []*Parameter
|
||||||
|
Variadic *Parameter // may be nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ft *FuncType) String(pm map[string]string, pkgOverride string) string {
|
||||||
|
args := make([]string, len(ft.In))
|
||||||
|
for i, p := range ft.In {
|
||||||
|
args[i] = p.Type.String(pm, pkgOverride)
|
||||||
|
}
|
||||||
|
if ft.Variadic != nil {
|
||||||
|
args = append(args, "..."+ft.Variadic.Type.String(pm, pkgOverride))
|
||||||
|
}
|
||||||
|
rets := make([]string, len(ft.Out))
|
||||||
|
for i, p := range ft.Out {
|
||||||
|
rets[i] = p.Type.String(pm, pkgOverride)
|
||||||
|
}
|
||||||
|
retString := strings.Join(rets, ", ")
|
||||||
|
if nOut := len(ft.Out); nOut == 1 {
|
||||||
|
retString = " " + retString
|
||||||
|
} else if nOut > 1 {
|
||||||
|
retString = " (" + retString + ")"
|
||||||
|
}
|
||||||
|
return "func(" + strings.Join(args, ", ") + ")" + retString
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ft *FuncType) addImports(im map[string]bool) {
|
||||||
|
for _, p := range ft.In {
|
||||||
|
p.Type.addImports(im)
|
||||||
|
}
|
||||||
|
if ft.Variadic != nil {
|
||||||
|
ft.Variadic.Type.addImports(im)
|
||||||
|
}
|
||||||
|
for _, p := range ft.Out {
|
||||||
|
p.Type.addImports(im)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// MapType is a map type.
|
||||||
|
type MapType struct {
|
||||||
|
Key, Value Type
|
||||||
|
}
|
||||||
|
|
||||||
|
func (mt *MapType) String(pm map[string]string, pkgOverride string) string {
|
||||||
|
return "map[" + mt.Key.String(pm, pkgOverride) + "]" + mt.Value.String(pm, pkgOverride)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (mt *MapType) addImports(im map[string]bool) {
|
||||||
|
mt.Key.addImports(im)
|
||||||
|
mt.Value.addImports(im)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NamedType is an exported type in a package.
|
||||||
|
type NamedType struct {
|
||||||
|
Package string // may be empty
|
||||||
|
Type string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (nt *NamedType) String(pm map[string]string, pkgOverride string) string {
|
||||||
|
if pkgOverride == nt.Package {
|
||||||
|
return nt.Type
|
||||||
|
}
|
||||||
|
prefix := pm[nt.Package]
|
||||||
|
if prefix != "" {
|
||||||
|
return prefix + "." + nt.Type
|
||||||
|
}
|
||||||
|
|
||||||
|
return nt.Type
|
||||||
|
}
|
||||||
|
|
||||||
|
func (nt *NamedType) addImports(im map[string]bool) {
|
||||||
|
if nt.Package != "" {
|
||||||
|
im[nt.Package] = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// PointerType is a pointer to another type.
|
||||||
|
type PointerType struct {
|
||||||
|
Type Type
|
||||||
|
}
|
||||||
|
|
||||||
|
func (pt *PointerType) String(pm map[string]string, pkgOverride string) string {
|
||||||
|
return "*" + pt.Type.String(pm, pkgOverride)
|
||||||
|
}
|
||||||
|
func (pt *PointerType) addImports(im map[string]bool) { pt.Type.addImports(im) }
|
||||||
|
|
||||||
|
// PredeclaredType is a predeclared type such as "int".
|
||||||
|
type PredeclaredType string
|
||||||
|
|
||||||
|
func (pt PredeclaredType) String(map[string]string, string) string { return string(pt) }
|
||||||
|
func (pt PredeclaredType) addImports(map[string]bool) {}
|
||||||
|
|
||||||
|
// The following code is intended to be called by the program generated by ../reflect.go.
|
||||||
|
|
||||||
|
// InterfaceFromInterfaceType returns a pointer to an interface for the
|
||||||
|
// given reflection interface type.
|
||||||
|
func InterfaceFromInterfaceType(it reflect.Type) (*Interface, error) {
|
||||||
|
if it.Kind() != reflect.Interface {
|
||||||
|
return nil, fmt.Errorf("%v is not an interface", it)
|
||||||
|
}
|
||||||
|
intf := &Interface{}
|
||||||
|
|
||||||
|
for i := 0; i < it.NumMethod(); i++ {
|
||||||
|
mt := it.Method(i)
|
||||||
|
// TODO: need to skip unexported methods? or just raise an error?
|
||||||
|
m := &Method{
|
||||||
|
Name: mt.Name,
|
||||||
|
}
|
||||||
|
|
||||||
|
var err error
|
||||||
|
m.In, m.Variadic, m.Out, err = funcArgsFromType(mt.Type)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
intf.AddMethod(m)
|
||||||
|
}
|
||||||
|
|
||||||
|
return intf, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// t's Kind must be a reflect.Func.
|
||||||
|
func funcArgsFromType(t reflect.Type) (in []*Parameter, variadic *Parameter, out []*Parameter, err error) {
|
||||||
|
nin := t.NumIn()
|
||||||
|
if t.IsVariadic() {
|
||||||
|
nin--
|
||||||
|
}
|
||||||
|
var p *Parameter
|
||||||
|
for i := 0; i < nin; i++ {
|
||||||
|
p, err = parameterFromType(t.In(i))
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
in = append(in, p)
|
||||||
|
}
|
||||||
|
if t.IsVariadic() {
|
||||||
|
p, err = parameterFromType(t.In(nin).Elem())
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
variadic = p
|
||||||
|
}
|
||||||
|
for i := 0; i < t.NumOut(); i++ {
|
||||||
|
p, err = parameterFromType(t.Out(i))
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
out = append(out, p)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func parameterFromType(t reflect.Type) (*Parameter, error) {
|
||||||
|
tt, err := typeFromType(t)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return &Parameter{Type: tt}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var errorType = reflect.TypeOf((*error)(nil)).Elem()
|
||||||
|
|
||||||
|
var byteType = reflect.TypeOf(byte(0))
|
||||||
|
|
||||||
|
func typeFromType(t reflect.Type) (Type, error) {
|
||||||
|
// Hack workaround for https://golang.org/issue/3853.
|
||||||
|
// This explicit check should not be necessary.
|
||||||
|
if t == byteType {
|
||||||
|
return PredeclaredType("byte"), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if imp := t.PkgPath(); imp != "" {
|
||||||
|
return &NamedType{
|
||||||
|
Package: impPath(imp),
|
||||||
|
Type: t.Name(),
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// only unnamed or predeclared types after here
|
||||||
|
|
||||||
|
// Lots of types have element types. Let's do the parsing and error checking for all of them.
|
||||||
|
var elemType Type
|
||||||
|
switch t.Kind() {
|
||||||
|
case reflect.Array, reflect.Chan, reflect.Map, reflect.Ptr, reflect.Slice:
|
||||||
|
var err error
|
||||||
|
elemType, err = typeFromType(t.Elem())
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
switch t.Kind() {
|
||||||
|
case reflect.Array:
|
||||||
|
return &ArrayType{
|
||||||
|
Len: t.Len(),
|
||||||
|
Type: elemType,
|
||||||
|
}, nil
|
||||||
|
case reflect.Bool, reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64,
|
||||||
|
reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr,
|
||||||
|
reflect.Float32, reflect.Float64, reflect.Complex64, reflect.Complex128, reflect.String:
|
||||||
|
return PredeclaredType(t.Kind().String()), nil
|
||||||
|
case reflect.Chan:
|
||||||
|
var dir ChanDir
|
||||||
|
switch t.ChanDir() {
|
||||||
|
case reflect.RecvDir:
|
||||||
|
dir = RecvDir
|
||||||
|
case reflect.SendDir:
|
||||||
|
dir = SendDir
|
||||||
|
}
|
||||||
|
return &ChanType{
|
||||||
|
Dir: dir,
|
||||||
|
Type: elemType,
|
||||||
|
}, nil
|
||||||
|
case reflect.Func:
|
||||||
|
in, variadic, out, err := funcArgsFromType(t)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return &FuncType{
|
||||||
|
In: in,
|
||||||
|
Out: out,
|
||||||
|
Variadic: variadic,
|
||||||
|
}, nil
|
||||||
|
case reflect.Interface:
|
||||||
|
// Two special interfaces.
|
||||||
|
if t.NumMethod() == 0 {
|
||||||
|
return PredeclaredType("interface{}"), nil
|
||||||
|
}
|
||||||
|
if t == errorType {
|
||||||
|
return PredeclaredType("error"), nil
|
||||||
|
}
|
||||||
|
case reflect.Map:
|
||||||
|
kt, err := typeFromType(t.Key())
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return &MapType{
|
||||||
|
Key: kt,
|
||||||
|
Value: elemType,
|
||||||
|
}, nil
|
||||||
|
case reflect.Ptr:
|
||||||
|
return &PointerType{
|
||||||
|
Type: elemType,
|
||||||
|
}, nil
|
||||||
|
case reflect.Slice:
|
||||||
|
return &ArrayType{
|
||||||
|
Len: -1,
|
||||||
|
Type: elemType,
|
||||||
|
}, nil
|
||||||
|
case reflect.Struct:
|
||||||
|
if t.NumField() == 0 {
|
||||||
|
return PredeclaredType("struct{}"), nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: Struct, UnsafePointer
|
||||||
|
return nil, fmt.Errorf("can't yet turn %v (%v) into a model.Type", t, t.Kind())
|
||||||
|
}
|
||||||
|
|
||||||
|
// impPath sanitizes the package path returned by `PkgPath` method of a reflect Type so that
|
||||||
|
// it is importable. PkgPath might return a path that includes "vendor". These paths do not
|
||||||
|
// compile, so we need to remove everything up to and including "/vendor/".
|
||||||
|
// See https://github.com/golang/go/issues/12019.
|
||||||
|
func impPath(imp string) string {
|
||||||
|
if strings.HasPrefix(imp, "vendor/") {
|
||||||
|
imp = "/" + imp
|
||||||
|
}
|
||||||
|
if i := strings.LastIndex(imp, "/vendor/"); i != -1 {
|
||||||
|
imp = imp[i+len("/vendor/"):]
|
||||||
|
}
|
||||||
|
return imp
|
||||||
|
}
|
||||||
|
|
||||||
|
// ErrorInterface represent built-in error interface.
|
||||||
|
var ErrorInterface = Interface{
|
||||||
|
Name: "error",
|
||||||
|
Methods: []*Method{
|
||||||
|
{
|
||||||
|
Name: "Error",
|
||||||
|
Out: []*Parameter{
|
||||||
|
{
|
||||||
|
Name: "",
|
||||||
|
Type: PredeclaredType("string"),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
27
vendor/github.com/google/go-cmp/LICENSE
generated
vendored
Normal file
27
vendor/github.com/google/go-cmp/LICENSE
generated
vendored
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
Copyright (c) 2017 The Go Authors. All rights reserved.
|
||||||
|
|
||||||
|
Redistribution and use in source and binary forms, with or without
|
||||||
|
modification, are permitted provided that the following conditions are
|
||||||
|
met:
|
||||||
|
|
||||||
|
* Redistributions of source code must retain the above copyright
|
||||||
|
notice, this list of conditions and the following disclaimer.
|
||||||
|
* Redistributions in binary form must reproduce the above
|
||||||
|
copyright notice, this list of conditions and the following disclaimer
|
||||||
|
in the documentation and/or other materials provided with the
|
||||||
|
distribution.
|
||||||
|
* Neither the name of Google Inc. nor the names of its
|
||||||
|
contributors may be used to endorse or promote products derived from
|
||||||
|
this software without specific prior written permission.
|
||||||
|
|
||||||
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||||
|
"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||||
|
LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||||
|
A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||||
|
OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||||
|
SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||||
|
LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||||
|
DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||||
|
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||||
|
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
669
vendor/github.com/google/go-cmp/cmp/compare.go
generated
vendored
Normal file
669
vendor/github.com/google/go-cmp/cmp/compare.go
generated
vendored
Normal file
@@ -0,0 +1,669 @@
|
|||||||
|
// Copyright 2017, The Go Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a BSD-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
// Package cmp determines equality of values.
|
||||||
|
//
|
||||||
|
// This package is intended to be a more powerful and safer alternative to
|
||||||
|
// reflect.DeepEqual for comparing whether two values are semantically equal.
|
||||||
|
// It is intended to only be used in tests, as performance is not a goal and
|
||||||
|
// it may panic if it cannot compare the values. Its propensity towards
|
||||||
|
// panicking means that its unsuitable for production environments where a
|
||||||
|
// spurious panic may be fatal.
|
||||||
|
//
|
||||||
|
// The primary features of cmp are:
|
||||||
|
//
|
||||||
|
// - When the default behavior of equality does not suit the test's needs,
|
||||||
|
// custom equality functions can override the equality operation.
|
||||||
|
// For example, an equality function may report floats as equal so long as
|
||||||
|
// they are within some tolerance of each other.
|
||||||
|
//
|
||||||
|
// - Types with an Equal method may use that method to determine equality.
|
||||||
|
// This allows package authors to determine the equality operation
|
||||||
|
// for the types that they define.
|
||||||
|
//
|
||||||
|
// - If no custom equality functions are used and no Equal method is defined,
|
||||||
|
// equality is determined by recursively comparing the primitive kinds on
|
||||||
|
// both values, much like reflect.DeepEqual. Unlike reflect.DeepEqual,
|
||||||
|
// unexported fields are not compared by default; they result in panics
|
||||||
|
// unless suppressed by using an Ignore option (see cmpopts.IgnoreUnexported)
|
||||||
|
// or explicitly compared using the Exporter option.
|
||||||
|
package cmp
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"reflect"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/google/go-cmp/cmp/internal/diff"
|
||||||
|
"github.com/google/go-cmp/cmp/internal/function"
|
||||||
|
"github.com/google/go-cmp/cmp/internal/value"
|
||||||
|
)
|
||||||
|
|
||||||
|
// TODO(≥go1.18): Use any instead of interface{}.
|
||||||
|
|
||||||
|
// Equal reports whether x and y are equal by recursively applying the
|
||||||
|
// following rules in the given order to x and y and all of their sub-values:
|
||||||
|
//
|
||||||
|
// - Let S be the set of all Ignore, Transformer, and Comparer options that
|
||||||
|
// remain after applying all path filters, value filters, and type filters.
|
||||||
|
// If at least one Ignore exists in S, then the comparison is ignored.
|
||||||
|
// If the number of Transformer and Comparer options in S is non-zero,
|
||||||
|
// then Equal panics because it is ambiguous which option to use.
|
||||||
|
// If S contains a single Transformer, then use that to transform
|
||||||
|
// the current values and recursively call Equal on the output values.
|
||||||
|
// If S contains a single Comparer, then use that to compare the current values.
|
||||||
|
// Otherwise, evaluation proceeds to the next rule.
|
||||||
|
//
|
||||||
|
// - If the values have an Equal method of the form "(T) Equal(T) bool" or
|
||||||
|
// "(T) Equal(I) bool" where T is assignable to I, then use the result of
|
||||||
|
// x.Equal(y) even if x or y is nil. Otherwise, no such method exists and
|
||||||
|
// evaluation proceeds to the next rule.
|
||||||
|
//
|
||||||
|
// - Lastly, try to compare x and y based on their basic kinds.
|
||||||
|
// Simple kinds like booleans, integers, floats, complex numbers, strings,
|
||||||
|
// and channels are compared using the equivalent of the == operator in Go.
|
||||||
|
// Functions are only equal if they are both nil, otherwise they are unequal.
|
||||||
|
//
|
||||||
|
// Structs are equal if recursively calling Equal on all fields report equal.
|
||||||
|
// If a struct contains unexported fields, Equal panics unless an Ignore option
|
||||||
|
// (e.g., cmpopts.IgnoreUnexported) ignores that field or the Exporter option
|
||||||
|
// explicitly permits comparing the unexported field.
|
||||||
|
//
|
||||||
|
// Slices are equal if they are both nil or both non-nil, where recursively
|
||||||
|
// calling Equal on all non-ignored slice or array elements report equal.
|
||||||
|
// Empty non-nil slices and nil slices are not equal; to equate empty slices,
|
||||||
|
// consider using cmpopts.EquateEmpty.
|
||||||
|
//
|
||||||
|
// Maps are equal if they are both nil or both non-nil, where recursively
|
||||||
|
// calling Equal on all non-ignored map entries report equal.
|
||||||
|
// Map keys are equal according to the == operator.
|
||||||
|
// To use custom comparisons for map keys, consider using cmpopts.SortMaps.
|
||||||
|
// Empty non-nil maps and nil maps are not equal; to equate empty maps,
|
||||||
|
// consider using cmpopts.EquateEmpty.
|
||||||
|
//
|
||||||
|
// Pointers and interfaces are equal if they are both nil or both non-nil,
|
||||||
|
// where they have the same underlying concrete type and recursively
|
||||||
|
// calling Equal on the underlying values reports equal.
|
||||||
|
//
|
||||||
|
// Before recursing into a pointer, slice element, or map, the current path
|
||||||
|
// is checked to detect whether the address has already been visited.
|
||||||
|
// If there is a cycle, then the pointed at values are considered equal
|
||||||
|
// only if both addresses were previously visited in the same path step.
|
||||||
|
func Equal(x, y interface{}, opts ...Option) bool {
|
||||||
|
s := newState(opts)
|
||||||
|
s.compareAny(rootStep(x, y))
|
||||||
|
return s.result.Equal()
|
||||||
|
}
|
||||||
|
|
||||||
|
// Diff returns a human-readable report of the differences between two values:
|
||||||
|
// y - x. It returns an empty string if and only if Equal returns true for the
|
||||||
|
// same input values and options.
|
||||||
|
//
|
||||||
|
// The output is displayed as a literal in pseudo-Go syntax.
|
||||||
|
// At the start of each line, a "-" prefix indicates an element removed from x,
|
||||||
|
// a "+" prefix to indicates an element added from y, and the lack of a prefix
|
||||||
|
// indicates an element common to both x and y. If possible, the output
|
||||||
|
// uses fmt.Stringer.String or error.Error methods to produce more humanly
|
||||||
|
// readable outputs. In such cases, the string is prefixed with either an
|
||||||
|
// 's' or 'e' character, respectively, to indicate that the method was called.
|
||||||
|
//
|
||||||
|
// Do not depend on this output being stable. If you need the ability to
|
||||||
|
// programmatically interpret the difference, consider using a custom Reporter.
|
||||||
|
func Diff(x, y interface{}, opts ...Option) string {
|
||||||
|
s := newState(opts)
|
||||||
|
|
||||||
|
// Optimization: If there are no other reporters, we can optimize for the
|
||||||
|
// common case where the result is equal (and thus no reported difference).
|
||||||
|
// This avoids the expensive construction of a difference tree.
|
||||||
|
if len(s.reporters) == 0 {
|
||||||
|
s.compareAny(rootStep(x, y))
|
||||||
|
if s.result.Equal() {
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
s.result = diff.Result{} // Reset results
|
||||||
|
}
|
||||||
|
|
||||||
|
r := new(defaultReporter)
|
||||||
|
s.reporters = append(s.reporters, reporter{r})
|
||||||
|
s.compareAny(rootStep(x, y))
|
||||||
|
d := r.String()
|
||||||
|
if (d == "") != s.result.Equal() {
|
||||||
|
panic("inconsistent difference and equality results")
|
||||||
|
}
|
||||||
|
return d
|
||||||
|
}
|
||||||
|
|
||||||
|
// rootStep constructs the first path step. If x and y have differing types,
|
||||||
|
// then they are stored within an empty interface type.
|
||||||
|
func rootStep(x, y interface{}) PathStep {
|
||||||
|
vx := reflect.ValueOf(x)
|
||||||
|
vy := reflect.ValueOf(y)
|
||||||
|
|
||||||
|
// If the inputs are different types, auto-wrap them in an empty interface
|
||||||
|
// so that they have the same parent type.
|
||||||
|
var t reflect.Type
|
||||||
|
if !vx.IsValid() || !vy.IsValid() || vx.Type() != vy.Type() {
|
||||||
|
t = anyType
|
||||||
|
if vx.IsValid() {
|
||||||
|
vvx := reflect.New(t).Elem()
|
||||||
|
vvx.Set(vx)
|
||||||
|
vx = vvx
|
||||||
|
}
|
||||||
|
if vy.IsValid() {
|
||||||
|
vvy := reflect.New(t).Elem()
|
||||||
|
vvy.Set(vy)
|
||||||
|
vy = vvy
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
t = vx.Type()
|
||||||
|
}
|
||||||
|
|
||||||
|
return &pathStep{t, vx, vy}
|
||||||
|
}
|
||||||
|
|
||||||
|
type state struct {
|
||||||
|
// These fields represent the "comparison state".
|
||||||
|
// Calling statelessCompare must not result in observable changes to these.
|
||||||
|
result diff.Result // The current result of comparison
|
||||||
|
curPath Path // The current path in the value tree
|
||||||
|
curPtrs pointerPath // The current set of visited pointers
|
||||||
|
reporters []reporter // Optional reporters
|
||||||
|
|
||||||
|
// recChecker checks for infinite cycles applying the same set of
|
||||||
|
// transformers upon the output of itself.
|
||||||
|
recChecker recChecker
|
||||||
|
|
||||||
|
// dynChecker triggers pseudo-random checks for option correctness.
|
||||||
|
// It is safe for statelessCompare to mutate this value.
|
||||||
|
dynChecker dynChecker
|
||||||
|
|
||||||
|
// These fields, once set by processOption, will not change.
|
||||||
|
exporters []exporter // List of exporters for structs with unexported fields
|
||||||
|
opts Options // List of all fundamental and filter options
|
||||||
|
}
|
||||||
|
|
||||||
|
func newState(opts []Option) *state {
|
||||||
|
// Always ensure a validator option exists to validate the inputs.
|
||||||
|
s := &state{opts: Options{validator{}}}
|
||||||
|
s.curPtrs.Init()
|
||||||
|
s.processOption(Options(opts))
|
||||||
|
return s
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *state) processOption(opt Option) {
|
||||||
|
switch opt := opt.(type) {
|
||||||
|
case nil:
|
||||||
|
case Options:
|
||||||
|
for _, o := range opt {
|
||||||
|
s.processOption(o)
|
||||||
|
}
|
||||||
|
case coreOption:
|
||||||
|
type filtered interface {
|
||||||
|
isFiltered() bool
|
||||||
|
}
|
||||||
|
if fopt, ok := opt.(filtered); ok && !fopt.isFiltered() {
|
||||||
|
panic(fmt.Sprintf("cannot use an unfiltered option: %v", opt))
|
||||||
|
}
|
||||||
|
s.opts = append(s.opts, opt)
|
||||||
|
case exporter:
|
||||||
|
s.exporters = append(s.exporters, opt)
|
||||||
|
case reporter:
|
||||||
|
s.reporters = append(s.reporters, opt)
|
||||||
|
default:
|
||||||
|
panic(fmt.Sprintf("unknown option %T", opt))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// statelessCompare compares two values and returns the result.
|
||||||
|
// This function is stateless in that it does not alter the current result,
|
||||||
|
// or output to any registered reporters.
|
||||||
|
func (s *state) statelessCompare(step PathStep) diff.Result {
|
||||||
|
// We do not save and restore curPath and curPtrs because all of the
|
||||||
|
// compareX methods should properly push and pop from them.
|
||||||
|
// It is an implementation bug if the contents of the paths differ from
|
||||||
|
// when calling this function to when returning from it.
|
||||||
|
|
||||||
|
oldResult, oldReporters := s.result, s.reporters
|
||||||
|
s.result = diff.Result{} // Reset result
|
||||||
|
s.reporters = nil // Remove reporters to avoid spurious printouts
|
||||||
|
s.compareAny(step)
|
||||||
|
res := s.result
|
||||||
|
s.result, s.reporters = oldResult, oldReporters
|
||||||
|
return res
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *state) compareAny(step PathStep) {
|
||||||
|
// Update the path stack.
|
||||||
|
s.curPath.push(step)
|
||||||
|
defer s.curPath.pop()
|
||||||
|
for _, r := range s.reporters {
|
||||||
|
r.PushStep(step)
|
||||||
|
defer r.PopStep()
|
||||||
|
}
|
||||||
|
s.recChecker.Check(s.curPath)
|
||||||
|
|
||||||
|
// Cycle-detection for slice elements (see NOTE in compareSlice).
|
||||||
|
t := step.Type()
|
||||||
|
vx, vy := step.Values()
|
||||||
|
if si, ok := step.(SliceIndex); ok && si.isSlice && vx.IsValid() && vy.IsValid() {
|
||||||
|
px, py := vx.Addr(), vy.Addr()
|
||||||
|
if eq, visited := s.curPtrs.Push(px, py); visited {
|
||||||
|
s.report(eq, reportByCycle)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
defer s.curPtrs.Pop(px, py)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Rule 1: Check whether an option applies on this node in the value tree.
|
||||||
|
if s.tryOptions(t, vx, vy) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Rule 2: Check whether the type has a valid Equal method.
|
||||||
|
if s.tryMethod(t, vx, vy) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Rule 3: Compare based on the underlying kind.
|
||||||
|
switch t.Kind() {
|
||||||
|
case reflect.Bool:
|
||||||
|
s.report(vx.Bool() == vy.Bool(), 0)
|
||||||
|
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
|
||||||
|
s.report(vx.Int() == vy.Int(), 0)
|
||||||
|
case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr:
|
||||||
|
s.report(vx.Uint() == vy.Uint(), 0)
|
||||||
|
case reflect.Float32, reflect.Float64:
|
||||||
|
s.report(vx.Float() == vy.Float(), 0)
|
||||||
|
case reflect.Complex64, reflect.Complex128:
|
||||||
|
s.report(vx.Complex() == vy.Complex(), 0)
|
||||||
|
case reflect.String:
|
||||||
|
s.report(vx.String() == vy.String(), 0)
|
||||||
|
case reflect.Chan, reflect.UnsafePointer:
|
||||||
|
s.report(vx.Pointer() == vy.Pointer(), 0)
|
||||||
|
case reflect.Func:
|
||||||
|
s.report(vx.IsNil() && vy.IsNil(), 0)
|
||||||
|
case reflect.Struct:
|
||||||
|
s.compareStruct(t, vx, vy)
|
||||||
|
case reflect.Slice, reflect.Array:
|
||||||
|
s.compareSlice(t, vx, vy)
|
||||||
|
case reflect.Map:
|
||||||
|
s.compareMap(t, vx, vy)
|
||||||
|
case reflect.Ptr:
|
||||||
|
s.comparePtr(t, vx, vy)
|
||||||
|
case reflect.Interface:
|
||||||
|
s.compareInterface(t, vx, vy)
|
||||||
|
default:
|
||||||
|
panic(fmt.Sprintf("%v kind not handled", t.Kind()))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *state) tryOptions(t reflect.Type, vx, vy reflect.Value) bool {
|
||||||
|
// Evaluate all filters and apply the remaining options.
|
||||||
|
if opt := s.opts.filter(s, t, vx, vy); opt != nil {
|
||||||
|
opt.apply(s, vx, vy)
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *state) tryMethod(t reflect.Type, vx, vy reflect.Value) bool {
|
||||||
|
// Check if this type even has an Equal method.
|
||||||
|
m, ok := t.MethodByName("Equal")
|
||||||
|
if !ok || !function.IsType(m.Type, function.EqualAssignable) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
eq := s.callTTBFunc(m.Func, vx, vy)
|
||||||
|
s.report(eq, reportByMethod)
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *state) callTRFunc(f, v reflect.Value, step Transform) reflect.Value {
|
||||||
|
if !s.dynChecker.Next() {
|
||||||
|
return f.Call([]reflect.Value{v})[0]
|
||||||
|
}
|
||||||
|
|
||||||
|
// Run the function twice and ensure that we get the same results back.
|
||||||
|
// We run in goroutines so that the race detector (if enabled) can detect
|
||||||
|
// unsafe mutations to the input.
|
||||||
|
c := make(chan reflect.Value)
|
||||||
|
go detectRaces(c, f, v)
|
||||||
|
got := <-c
|
||||||
|
want := f.Call([]reflect.Value{v})[0]
|
||||||
|
if step.vx, step.vy = got, want; !s.statelessCompare(step).Equal() {
|
||||||
|
// To avoid false-positives with non-reflexive equality operations,
|
||||||
|
// we sanity check whether a value is equal to itself.
|
||||||
|
if step.vx, step.vy = want, want; !s.statelessCompare(step).Equal() {
|
||||||
|
return want
|
||||||
|
}
|
||||||
|
panic(fmt.Sprintf("non-deterministic function detected: %s", function.NameOf(f)))
|
||||||
|
}
|
||||||
|
return want
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *state) callTTBFunc(f, x, y reflect.Value) bool {
|
||||||
|
if !s.dynChecker.Next() {
|
||||||
|
return f.Call([]reflect.Value{x, y})[0].Bool()
|
||||||
|
}
|
||||||
|
|
||||||
|
// Swapping the input arguments is sufficient to check that
|
||||||
|
// f is symmetric and deterministic.
|
||||||
|
// We run in goroutines so that the race detector (if enabled) can detect
|
||||||
|
// unsafe mutations to the input.
|
||||||
|
c := make(chan reflect.Value)
|
||||||
|
go detectRaces(c, f, y, x)
|
||||||
|
got := <-c
|
||||||
|
want := f.Call([]reflect.Value{x, y})[0].Bool()
|
||||||
|
if !got.IsValid() || got.Bool() != want {
|
||||||
|
panic(fmt.Sprintf("non-deterministic or non-symmetric function detected: %s", function.NameOf(f)))
|
||||||
|
}
|
||||||
|
return want
|
||||||
|
}
|
||||||
|
|
||||||
|
func detectRaces(c chan<- reflect.Value, f reflect.Value, vs ...reflect.Value) {
|
||||||
|
var ret reflect.Value
|
||||||
|
defer func() {
|
||||||
|
recover() // Ignore panics, let the other call to f panic instead
|
||||||
|
c <- ret
|
||||||
|
}()
|
||||||
|
ret = f.Call(vs)[0]
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *state) compareStruct(t reflect.Type, vx, vy reflect.Value) {
|
||||||
|
var addr bool
|
||||||
|
var vax, vay reflect.Value // Addressable versions of vx and vy
|
||||||
|
|
||||||
|
var mayForce, mayForceInit bool
|
||||||
|
step := StructField{&structField{}}
|
||||||
|
for i := 0; i < t.NumField(); i++ {
|
||||||
|
step.typ = t.Field(i).Type
|
||||||
|
step.vx = vx.Field(i)
|
||||||
|
step.vy = vy.Field(i)
|
||||||
|
step.name = t.Field(i).Name
|
||||||
|
step.idx = i
|
||||||
|
step.unexported = !isExported(step.name)
|
||||||
|
if step.unexported {
|
||||||
|
if step.name == "_" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
// Defer checking of unexported fields until later to give an
|
||||||
|
// Ignore a chance to ignore the field.
|
||||||
|
if !vax.IsValid() || !vay.IsValid() {
|
||||||
|
// For retrieveUnexportedField to work, the parent struct must
|
||||||
|
// be addressable. Create a new copy of the values if
|
||||||
|
// necessary to make them addressable.
|
||||||
|
addr = vx.CanAddr() || vy.CanAddr()
|
||||||
|
vax = makeAddressable(vx)
|
||||||
|
vay = makeAddressable(vy)
|
||||||
|
}
|
||||||
|
if !mayForceInit {
|
||||||
|
for _, xf := range s.exporters {
|
||||||
|
mayForce = mayForce || xf(t)
|
||||||
|
}
|
||||||
|
mayForceInit = true
|
||||||
|
}
|
||||||
|
step.mayForce = mayForce
|
||||||
|
step.paddr = addr
|
||||||
|
step.pvx = vax
|
||||||
|
step.pvy = vay
|
||||||
|
step.field = t.Field(i)
|
||||||
|
}
|
||||||
|
s.compareAny(step)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *state) compareSlice(t reflect.Type, vx, vy reflect.Value) {
|
||||||
|
isSlice := t.Kind() == reflect.Slice
|
||||||
|
if isSlice && (vx.IsNil() || vy.IsNil()) {
|
||||||
|
s.report(vx.IsNil() && vy.IsNil(), 0)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// NOTE: It is incorrect to call curPtrs.Push on the slice header pointer
|
||||||
|
// since slices represents a list of pointers, rather than a single pointer.
|
||||||
|
// The pointer checking logic must be handled on a per-element basis
|
||||||
|
// in compareAny.
|
||||||
|
//
|
||||||
|
// A slice header (see reflect.SliceHeader) in Go is a tuple of a starting
|
||||||
|
// pointer P, a length N, and a capacity C. Supposing each slice element has
|
||||||
|
// a memory size of M, then the slice is equivalent to the list of pointers:
|
||||||
|
// [P+i*M for i in range(N)]
|
||||||
|
//
|
||||||
|
// For example, v[:0] and v[:1] are slices with the same starting pointer,
|
||||||
|
// but they are clearly different values. Using the slice pointer alone
|
||||||
|
// violates the assumption that equal pointers implies equal values.
|
||||||
|
|
||||||
|
step := SliceIndex{&sliceIndex{pathStep: pathStep{typ: t.Elem()}, isSlice: isSlice}}
|
||||||
|
withIndexes := func(ix, iy int) SliceIndex {
|
||||||
|
if ix >= 0 {
|
||||||
|
step.vx, step.xkey = vx.Index(ix), ix
|
||||||
|
} else {
|
||||||
|
step.vx, step.xkey = reflect.Value{}, -1
|
||||||
|
}
|
||||||
|
if iy >= 0 {
|
||||||
|
step.vy, step.ykey = vy.Index(iy), iy
|
||||||
|
} else {
|
||||||
|
step.vy, step.ykey = reflect.Value{}, -1
|
||||||
|
}
|
||||||
|
return step
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ignore options are able to ignore missing elements in a slice.
|
||||||
|
// However, detecting these reliably requires an optimal differencing
|
||||||
|
// algorithm, for which diff.Difference is not.
|
||||||
|
//
|
||||||
|
// Instead, we first iterate through both slices to detect which elements
|
||||||
|
// would be ignored if standing alone. The index of non-discarded elements
|
||||||
|
// are stored in a separate slice, which diffing is then performed on.
|
||||||
|
var indexesX, indexesY []int
|
||||||
|
var ignoredX, ignoredY []bool
|
||||||
|
for ix := 0; ix < vx.Len(); ix++ {
|
||||||
|
ignored := s.statelessCompare(withIndexes(ix, -1)).NumDiff == 0
|
||||||
|
if !ignored {
|
||||||
|
indexesX = append(indexesX, ix)
|
||||||
|
}
|
||||||
|
ignoredX = append(ignoredX, ignored)
|
||||||
|
}
|
||||||
|
for iy := 0; iy < vy.Len(); iy++ {
|
||||||
|
ignored := s.statelessCompare(withIndexes(-1, iy)).NumDiff == 0
|
||||||
|
if !ignored {
|
||||||
|
indexesY = append(indexesY, iy)
|
||||||
|
}
|
||||||
|
ignoredY = append(ignoredY, ignored)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Compute an edit-script for slices vx and vy (excluding ignored elements).
|
||||||
|
edits := diff.Difference(len(indexesX), len(indexesY), func(ix, iy int) diff.Result {
|
||||||
|
return s.statelessCompare(withIndexes(indexesX[ix], indexesY[iy]))
|
||||||
|
})
|
||||||
|
|
||||||
|
// Replay the ignore-scripts and the edit-script.
|
||||||
|
var ix, iy int
|
||||||
|
for ix < vx.Len() || iy < vy.Len() {
|
||||||
|
var e diff.EditType
|
||||||
|
switch {
|
||||||
|
case ix < len(ignoredX) && ignoredX[ix]:
|
||||||
|
e = diff.UniqueX
|
||||||
|
case iy < len(ignoredY) && ignoredY[iy]:
|
||||||
|
e = diff.UniqueY
|
||||||
|
default:
|
||||||
|
e, edits = edits[0], edits[1:]
|
||||||
|
}
|
||||||
|
switch e {
|
||||||
|
case diff.UniqueX:
|
||||||
|
s.compareAny(withIndexes(ix, -1))
|
||||||
|
ix++
|
||||||
|
case diff.UniqueY:
|
||||||
|
s.compareAny(withIndexes(-1, iy))
|
||||||
|
iy++
|
||||||
|
default:
|
||||||
|
s.compareAny(withIndexes(ix, iy))
|
||||||
|
ix++
|
||||||
|
iy++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *state) compareMap(t reflect.Type, vx, vy reflect.Value) {
|
||||||
|
if vx.IsNil() || vy.IsNil() {
|
||||||
|
s.report(vx.IsNil() && vy.IsNil(), 0)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Cycle-detection for maps.
|
||||||
|
if eq, visited := s.curPtrs.Push(vx, vy); visited {
|
||||||
|
s.report(eq, reportByCycle)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
defer s.curPtrs.Pop(vx, vy)
|
||||||
|
|
||||||
|
// We combine and sort the two map keys so that we can perform the
|
||||||
|
// comparisons in a deterministic order.
|
||||||
|
step := MapIndex{&mapIndex{pathStep: pathStep{typ: t.Elem()}}}
|
||||||
|
for _, k := range value.SortKeys(append(vx.MapKeys(), vy.MapKeys()...)) {
|
||||||
|
step.vx = vx.MapIndex(k)
|
||||||
|
step.vy = vy.MapIndex(k)
|
||||||
|
step.key = k
|
||||||
|
if !step.vx.IsValid() && !step.vy.IsValid() {
|
||||||
|
// It is possible for both vx and vy to be invalid if the
|
||||||
|
// key contained a NaN value in it.
|
||||||
|
//
|
||||||
|
// Even with the ability to retrieve NaN keys in Go 1.12,
|
||||||
|
// there still isn't a sensible way to compare the values since
|
||||||
|
// a NaN key may map to multiple unordered values.
|
||||||
|
// The most reasonable way to compare NaNs would be to compare the
|
||||||
|
// set of values. However, this is impossible to do efficiently
|
||||||
|
// since set equality is provably an O(n^2) operation given only
|
||||||
|
// an Equal function. If we had a Less function or Hash function,
|
||||||
|
// this could be done in O(n*log(n)) or O(n), respectively.
|
||||||
|
//
|
||||||
|
// Rather than adding complex logic to deal with NaNs, make it
|
||||||
|
// the user's responsibility to compare such obscure maps.
|
||||||
|
const help = "consider providing a Comparer to compare the map"
|
||||||
|
panic(fmt.Sprintf("%#v has map key with NaNs\n%s", s.curPath, help))
|
||||||
|
}
|
||||||
|
s.compareAny(step)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *state) comparePtr(t reflect.Type, vx, vy reflect.Value) {
|
||||||
|
if vx.IsNil() || vy.IsNil() {
|
||||||
|
s.report(vx.IsNil() && vy.IsNil(), 0)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Cycle-detection for pointers.
|
||||||
|
if eq, visited := s.curPtrs.Push(vx, vy); visited {
|
||||||
|
s.report(eq, reportByCycle)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
defer s.curPtrs.Pop(vx, vy)
|
||||||
|
|
||||||
|
vx, vy = vx.Elem(), vy.Elem()
|
||||||
|
s.compareAny(Indirect{&indirect{pathStep{t.Elem(), vx, vy}}})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *state) compareInterface(t reflect.Type, vx, vy reflect.Value) {
|
||||||
|
if vx.IsNil() || vy.IsNil() {
|
||||||
|
s.report(vx.IsNil() && vy.IsNil(), 0)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
vx, vy = vx.Elem(), vy.Elem()
|
||||||
|
if vx.Type() != vy.Type() {
|
||||||
|
s.report(false, 0)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
s.compareAny(TypeAssertion{&typeAssertion{pathStep{vx.Type(), vx, vy}}})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *state) report(eq bool, rf resultFlags) {
|
||||||
|
if rf&reportByIgnore == 0 {
|
||||||
|
if eq {
|
||||||
|
s.result.NumSame++
|
||||||
|
rf |= reportEqual
|
||||||
|
} else {
|
||||||
|
s.result.NumDiff++
|
||||||
|
rf |= reportUnequal
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for _, r := range s.reporters {
|
||||||
|
r.Report(Result{flags: rf})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// recChecker tracks the state needed to periodically perform checks that
|
||||||
|
// user provided transformers are not stuck in an infinitely recursive cycle.
|
||||||
|
type recChecker struct{ next int }
|
||||||
|
|
||||||
|
// Check scans the Path for any recursive transformers and panics when any
|
||||||
|
// recursive transformers are detected. Note that the presence of a
|
||||||
|
// recursive Transformer does not necessarily imply an infinite cycle.
|
||||||
|
// As such, this check only activates after some minimal number of path steps.
|
||||||
|
func (rc *recChecker) Check(p Path) {
|
||||||
|
const minLen = 1 << 16
|
||||||
|
if rc.next == 0 {
|
||||||
|
rc.next = minLen
|
||||||
|
}
|
||||||
|
if len(p) < rc.next {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
rc.next <<= 1
|
||||||
|
|
||||||
|
// Check whether the same transformer has appeared at least twice.
|
||||||
|
var ss []string
|
||||||
|
m := map[Option]int{}
|
||||||
|
for _, ps := range p {
|
||||||
|
if t, ok := ps.(Transform); ok {
|
||||||
|
t := t.Option()
|
||||||
|
if m[t] == 1 { // Transformer was used exactly once before
|
||||||
|
tf := t.(*transformer).fnc.Type()
|
||||||
|
ss = append(ss, fmt.Sprintf("%v: %v => %v", t, tf.In(0), tf.Out(0)))
|
||||||
|
}
|
||||||
|
m[t]++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if len(ss) > 0 {
|
||||||
|
const warning = "recursive set of Transformers detected"
|
||||||
|
const help = "consider using cmpopts.AcyclicTransformer"
|
||||||
|
set := strings.Join(ss, "\n\t")
|
||||||
|
panic(fmt.Sprintf("%s:\n\t%s\n%s", warning, set, help))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// dynChecker tracks the state needed to periodically perform checks that
|
||||||
|
// user provided functions are symmetric and deterministic.
|
||||||
|
// The zero value is safe for immediate use.
|
||||||
|
type dynChecker struct{ curr, next int }
|
||||||
|
|
||||||
|
// Next increments the state and reports whether a check should be performed.
|
||||||
|
//
|
||||||
|
// Checks occur every Nth function call, where N is a triangular number:
|
||||||
|
//
|
||||||
|
// 0 1 3 6 10 15 21 28 36 45 55 66 78 91 105 120 136 153 171 190 ...
|
||||||
|
//
|
||||||
|
// See https://en.wikipedia.org/wiki/Triangular_number
|
||||||
|
//
|
||||||
|
// This sequence ensures that the cost of checks drops significantly as
|
||||||
|
// the number of functions calls grows larger.
|
||||||
|
func (dc *dynChecker) Next() bool {
|
||||||
|
ok := dc.curr == dc.next
|
||||||
|
if ok {
|
||||||
|
dc.curr = 0
|
||||||
|
dc.next++
|
||||||
|
}
|
||||||
|
dc.curr++
|
||||||
|
return ok
|
||||||
|
}
|
||||||
|
|
||||||
|
// makeAddressable returns a value that is always addressable.
|
||||||
|
// It returns the input verbatim if it is already addressable,
|
||||||
|
// otherwise it creates a new value and returns an addressable copy.
|
||||||
|
func makeAddressable(v reflect.Value) reflect.Value {
|
||||||
|
if v.CanAddr() {
|
||||||
|
return v
|
||||||
|
}
|
||||||
|
vc := reflect.New(v.Type()).Elem()
|
||||||
|
vc.Set(v)
|
||||||
|
return vc
|
||||||
|
}
|
||||||
16
vendor/github.com/google/go-cmp/cmp/export_panic.go
generated
vendored
Normal file
16
vendor/github.com/google/go-cmp/cmp/export_panic.go
generated
vendored
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
// Copyright 2017, The Go Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a BSD-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
//go:build purego
|
||||||
|
// +build purego
|
||||||
|
|
||||||
|
package cmp
|
||||||
|
|
||||||
|
import "reflect"
|
||||||
|
|
||||||
|
const supportExporters = false
|
||||||
|
|
||||||
|
func retrieveUnexportedField(reflect.Value, reflect.StructField, bool) reflect.Value {
|
||||||
|
panic("no support for forcibly accessing unexported fields")
|
||||||
|
}
|
||||||
36
vendor/github.com/google/go-cmp/cmp/export_unsafe.go
generated
vendored
Normal file
36
vendor/github.com/google/go-cmp/cmp/export_unsafe.go
generated
vendored
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
// Copyright 2017, The Go Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a BSD-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
//go:build !purego
|
||||||
|
// +build !purego
|
||||||
|
|
||||||
|
package cmp
|
||||||
|
|
||||||
|
import (
|
||||||
|
"reflect"
|
||||||
|
"unsafe"
|
||||||
|
)
|
||||||
|
|
||||||
|
const supportExporters = true
|
||||||
|
|
||||||
|
// retrieveUnexportedField uses unsafe to forcibly retrieve any field from
|
||||||
|
// a struct such that the value has read-write permissions.
|
||||||
|
//
|
||||||
|
// The parent struct, v, must be addressable, while f must be a StructField
|
||||||
|
// describing the field to retrieve. If addr is false,
|
||||||
|
// then the returned value will be shallowed copied to be non-addressable.
|
||||||
|
func retrieveUnexportedField(v reflect.Value, f reflect.StructField, addr bool) reflect.Value {
|
||||||
|
ve := reflect.NewAt(f.Type, unsafe.Pointer(uintptr(unsafe.Pointer(v.UnsafeAddr()))+f.Offset)).Elem()
|
||||||
|
if !addr {
|
||||||
|
// A field is addressable if and only if the struct is addressable.
|
||||||
|
// If the original parent value was not addressable, shallow copy the
|
||||||
|
// value to make it non-addressable to avoid leaking an implementation
|
||||||
|
// detail of how forcibly exporting a field works.
|
||||||
|
if ve.Kind() == reflect.Interface && ve.IsNil() {
|
||||||
|
return reflect.Zero(f.Type)
|
||||||
|
}
|
||||||
|
return reflect.ValueOf(ve.Interface()).Convert(f.Type)
|
||||||
|
}
|
||||||
|
return ve
|
||||||
|
}
|
||||||
18
vendor/github.com/google/go-cmp/cmp/internal/diff/debug_disable.go
generated
vendored
Normal file
18
vendor/github.com/google/go-cmp/cmp/internal/diff/debug_disable.go
generated
vendored
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
// Copyright 2017, The Go Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a BSD-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
//go:build !cmp_debug
|
||||||
|
// +build !cmp_debug
|
||||||
|
|
||||||
|
package diff
|
||||||
|
|
||||||
|
var debug debugger
|
||||||
|
|
||||||
|
type debugger struct{}
|
||||||
|
|
||||||
|
func (debugger) Begin(_, _ int, f EqualFunc, _, _ *EditScript) EqualFunc {
|
||||||
|
return f
|
||||||
|
}
|
||||||
|
func (debugger) Update() {}
|
||||||
|
func (debugger) Finish() {}
|
||||||
123
vendor/github.com/google/go-cmp/cmp/internal/diff/debug_enable.go
generated
vendored
Normal file
123
vendor/github.com/google/go-cmp/cmp/internal/diff/debug_enable.go
generated
vendored
Normal file
@@ -0,0 +1,123 @@
|
|||||||
|
// Copyright 2017, The Go Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a BSD-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
//go:build cmp_debug
|
||||||
|
// +build cmp_debug
|
||||||
|
|
||||||
|
package diff
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"strings"
|
||||||
|
"sync"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
// The algorithm can be seen running in real-time by enabling debugging:
|
||||||
|
// go test -tags=cmp_debug -v
|
||||||
|
//
|
||||||
|
// Example output:
|
||||||
|
// === RUN TestDifference/#34
|
||||||
|
// ┌───────────────────────────────┐
|
||||||
|
// │ \ · · · · · · · · · · · · · · │
|
||||||
|
// │ · # · · · · · · · · · · · · · │
|
||||||
|
// │ · \ · · · · · · · · · · · · · │
|
||||||
|
// │ · · \ · · · · · · · · · · · · │
|
||||||
|
// │ · · · X # · · · · · · · · · · │
|
||||||
|
// │ · · · # \ · · · · · · · · · · │
|
||||||
|
// │ · · · · · # # · · · · · · · · │
|
||||||
|
// │ · · · · · # \ · · · · · · · · │
|
||||||
|
// │ · · · · · · · \ · · · · · · · │
|
||||||
|
// │ · · · · · · · · \ · · · · · · │
|
||||||
|
// │ · · · · · · · · · \ · · · · · │
|
||||||
|
// │ · · · · · · · · · · \ · · # · │
|
||||||
|
// │ · · · · · · · · · · · \ # # · │
|
||||||
|
// │ · · · · · · · · · · · # # # · │
|
||||||
|
// │ · · · · · · · · · · # # # # · │
|
||||||
|
// │ · · · · · · · · · # # # # # · │
|
||||||
|
// │ · · · · · · · · · · · · · · \ │
|
||||||
|
// └───────────────────────────────┘
|
||||||
|
// [.Y..M.XY......YXYXY.|]
|
||||||
|
//
|
||||||
|
// The grid represents the edit-graph where the horizontal axis represents
|
||||||
|
// list X and the vertical axis represents list Y. The start of the two lists
|
||||||
|
// is the top-left, while the ends are the bottom-right. The '·' represents
|
||||||
|
// an unexplored node in the graph. The '\' indicates that the two symbols
|
||||||
|
// from list X and Y are equal. The 'X' indicates that two symbols are similar
|
||||||
|
// (but not exactly equal) to each other. The '#' indicates that the two symbols
|
||||||
|
// are different (and not similar). The algorithm traverses this graph trying to
|
||||||
|
// make the paths starting in the top-left and the bottom-right connect.
|
||||||
|
//
|
||||||
|
// The series of '.', 'X', 'Y', and 'M' characters at the bottom represents
|
||||||
|
// the currently established path from the forward and reverse searches,
|
||||||
|
// separated by a '|' character.
|
||||||
|
|
||||||
|
const (
|
||||||
|
updateDelay = 100 * time.Millisecond
|
||||||
|
finishDelay = 500 * time.Millisecond
|
||||||
|
ansiTerminal = true // ANSI escape codes used to move terminal cursor
|
||||||
|
)
|
||||||
|
|
||||||
|
var debug debugger
|
||||||
|
|
||||||
|
type debugger struct {
|
||||||
|
sync.Mutex
|
||||||
|
p1, p2 EditScript
|
||||||
|
fwdPath, revPath *EditScript
|
||||||
|
grid []byte
|
||||||
|
lines int
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbg *debugger) Begin(nx, ny int, f EqualFunc, p1, p2 *EditScript) EqualFunc {
|
||||||
|
dbg.Lock()
|
||||||
|
dbg.fwdPath, dbg.revPath = p1, p2
|
||||||
|
top := "┌─" + strings.Repeat("──", nx) + "┐\n"
|
||||||
|
row := "│ " + strings.Repeat("· ", nx) + "│\n"
|
||||||
|
btm := "└─" + strings.Repeat("──", nx) + "┘\n"
|
||||||
|
dbg.grid = []byte(top + strings.Repeat(row, ny) + btm)
|
||||||
|
dbg.lines = strings.Count(dbg.String(), "\n")
|
||||||
|
fmt.Print(dbg)
|
||||||
|
|
||||||
|
// Wrap the EqualFunc so that we can intercept each result.
|
||||||
|
return func(ix, iy int) (r Result) {
|
||||||
|
cell := dbg.grid[len(top)+iy*len(row):][len("│ ")+len("· ")*ix:][:len("·")]
|
||||||
|
for i := range cell {
|
||||||
|
cell[i] = 0 // Zero out the multiple bytes of UTF-8 middle-dot
|
||||||
|
}
|
||||||
|
switch r = f(ix, iy); {
|
||||||
|
case r.Equal():
|
||||||
|
cell[0] = '\\'
|
||||||
|
case r.Similar():
|
||||||
|
cell[0] = 'X'
|
||||||
|
default:
|
||||||
|
cell[0] = '#'
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbg *debugger) Update() {
|
||||||
|
dbg.print(updateDelay)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbg *debugger) Finish() {
|
||||||
|
dbg.print(finishDelay)
|
||||||
|
dbg.Unlock()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbg *debugger) String() string {
|
||||||
|
dbg.p1, dbg.p2 = *dbg.fwdPath, dbg.p2[:0]
|
||||||
|
for i := len(*dbg.revPath) - 1; i >= 0; i-- {
|
||||||
|
dbg.p2 = append(dbg.p2, (*dbg.revPath)[i])
|
||||||
|
}
|
||||||
|
return fmt.Sprintf("%s[%v|%v]\n\n", dbg.grid, dbg.p1, dbg.p2)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (dbg *debugger) print(d time.Duration) {
|
||||||
|
if ansiTerminal {
|
||||||
|
fmt.Printf("\x1b[%dA", dbg.lines) // Reset terminal cursor
|
||||||
|
}
|
||||||
|
fmt.Print(dbg)
|
||||||
|
time.Sleep(d)
|
||||||
|
}
|
||||||
402
vendor/github.com/google/go-cmp/cmp/internal/diff/diff.go
generated
vendored
Normal file
402
vendor/github.com/google/go-cmp/cmp/internal/diff/diff.go
generated
vendored
Normal file
@@ -0,0 +1,402 @@
|
|||||||
|
// Copyright 2017, The Go Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a BSD-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
// Package diff implements an algorithm for producing edit-scripts.
|
||||||
|
// The edit-script is a sequence of operations needed to transform one list
|
||||||
|
// of symbols into another (or vice-versa). The edits allowed are insertions,
|
||||||
|
// deletions, and modifications. The summation of all edits is called the
|
||||||
|
// Levenshtein distance as this problem is well-known in computer science.
|
||||||
|
//
|
||||||
|
// This package prioritizes performance over accuracy. That is, the run time
|
||||||
|
// is more important than obtaining a minimal Levenshtein distance.
|
||||||
|
package diff
|
||||||
|
|
||||||
|
import (
|
||||||
|
"math/rand"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/google/go-cmp/cmp/internal/flags"
|
||||||
|
)
|
||||||
|
|
||||||
|
// EditType represents a single operation within an edit-script.
|
||||||
|
type EditType uint8
|
||||||
|
|
||||||
|
const (
|
||||||
|
// Identity indicates that a symbol pair is identical in both list X and Y.
|
||||||
|
Identity EditType = iota
|
||||||
|
// UniqueX indicates that a symbol only exists in X and not Y.
|
||||||
|
UniqueX
|
||||||
|
// UniqueY indicates that a symbol only exists in Y and not X.
|
||||||
|
UniqueY
|
||||||
|
// Modified indicates that a symbol pair is a modification of each other.
|
||||||
|
Modified
|
||||||
|
)
|
||||||
|
|
||||||
|
// EditScript represents the series of differences between two lists.
|
||||||
|
type EditScript []EditType
|
||||||
|
|
||||||
|
// String returns a human-readable string representing the edit-script where
|
||||||
|
// Identity, UniqueX, UniqueY, and Modified are represented by the
|
||||||
|
// '.', 'X', 'Y', and 'M' characters, respectively.
|
||||||
|
func (es EditScript) String() string {
|
||||||
|
b := make([]byte, len(es))
|
||||||
|
for i, e := range es {
|
||||||
|
switch e {
|
||||||
|
case Identity:
|
||||||
|
b[i] = '.'
|
||||||
|
case UniqueX:
|
||||||
|
b[i] = 'X'
|
||||||
|
case UniqueY:
|
||||||
|
b[i] = 'Y'
|
||||||
|
case Modified:
|
||||||
|
b[i] = 'M'
|
||||||
|
default:
|
||||||
|
panic("invalid edit-type")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return string(b)
|
||||||
|
}
|
||||||
|
|
||||||
|
// stats returns a histogram of the number of each type of edit operation.
|
||||||
|
func (es EditScript) stats() (s struct{ NI, NX, NY, NM int }) {
|
||||||
|
for _, e := range es {
|
||||||
|
switch e {
|
||||||
|
case Identity:
|
||||||
|
s.NI++
|
||||||
|
case UniqueX:
|
||||||
|
s.NX++
|
||||||
|
case UniqueY:
|
||||||
|
s.NY++
|
||||||
|
case Modified:
|
||||||
|
s.NM++
|
||||||
|
default:
|
||||||
|
panic("invalid edit-type")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Dist is the Levenshtein distance and is guaranteed to be 0 if and only if
|
||||||
|
// lists X and Y are equal.
|
||||||
|
func (es EditScript) Dist() int { return len(es) - es.stats().NI }
|
||||||
|
|
||||||
|
// LenX is the length of the X list.
|
||||||
|
func (es EditScript) LenX() int { return len(es) - es.stats().NY }
|
||||||
|
|
||||||
|
// LenY is the length of the Y list.
|
||||||
|
func (es EditScript) LenY() int { return len(es) - es.stats().NX }
|
||||||
|
|
||||||
|
// EqualFunc reports whether the symbols at indexes ix and iy are equal.
|
||||||
|
// When called by Difference, the index is guaranteed to be within nx and ny.
|
||||||
|
type EqualFunc func(ix int, iy int) Result
|
||||||
|
|
||||||
|
// Result is the result of comparison.
|
||||||
|
// NumSame is the number of sub-elements that are equal.
|
||||||
|
// NumDiff is the number of sub-elements that are not equal.
|
||||||
|
type Result struct{ NumSame, NumDiff int }
|
||||||
|
|
||||||
|
// BoolResult returns a Result that is either Equal or not Equal.
|
||||||
|
func BoolResult(b bool) Result {
|
||||||
|
if b {
|
||||||
|
return Result{NumSame: 1} // Equal, Similar
|
||||||
|
} else {
|
||||||
|
return Result{NumDiff: 2} // Not Equal, not Similar
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Equal indicates whether the symbols are equal. Two symbols are equal
|
||||||
|
// if and only if NumDiff == 0. If Equal, then they are also Similar.
|
||||||
|
func (r Result) Equal() bool { return r.NumDiff == 0 }
|
||||||
|
|
||||||
|
// Similar indicates whether two symbols are similar and may be represented
|
||||||
|
// by using the Modified type. As a special case, we consider binary comparisons
|
||||||
|
// (i.e., those that return Result{1, 0} or Result{0, 1}) to be similar.
|
||||||
|
//
|
||||||
|
// The exact ratio of NumSame to NumDiff to determine similarity may change.
|
||||||
|
func (r Result) Similar() bool {
|
||||||
|
// Use NumSame+1 to offset NumSame so that binary comparisons are similar.
|
||||||
|
return r.NumSame+1 >= r.NumDiff
|
||||||
|
}
|
||||||
|
|
||||||
|
var randBool = rand.New(rand.NewSource(time.Now().Unix())).Intn(2) == 0
|
||||||
|
|
||||||
|
// Difference reports whether two lists of lengths nx and ny are equal
|
||||||
|
// given the definition of equality provided as f.
|
||||||
|
//
|
||||||
|
// This function returns an edit-script, which is a sequence of operations
|
||||||
|
// needed to convert one list into the other. The following invariants for
|
||||||
|
// the edit-script are maintained:
|
||||||
|
// - eq == (es.Dist()==0)
|
||||||
|
// - nx == es.LenX()
|
||||||
|
// - ny == es.LenY()
|
||||||
|
//
|
||||||
|
// This algorithm is not guaranteed to be an optimal solution (i.e., one that
|
||||||
|
// produces an edit-script with a minimal Levenshtein distance). This algorithm
|
||||||
|
// favors performance over optimality. The exact output is not guaranteed to
|
||||||
|
// be stable and may change over time.
|
||||||
|
func Difference(nx, ny int, f EqualFunc) (es EditScript) {
|
||||||
|
// This algorithm is based on traversing what is known as an "edit-graph".
|
||||||
|
// See Figure 1 from "An O(ND) Difference Algorithm and Its Variations"
|
||||||
|
// by Eugene W. Myers. Since D can be as large as N itself, this is
|
||||||
|
// effectively O(N^2). Unlike the algorithm from that paper, we are not
|
||||||
|
// interested in the optimal path, but at least some "decent" path.
|
||||||
|
//
|
||||||
|
// For example, let X and Y be lists of symbols:
|
||||||
|
// X = [A B C A B B A]
|
||||||
|
// Y = [C B A B A C]
|
||||||
|
//
|
||||||
|
// The edit-graph can be drawn as the following:
|
||||||
|
// A B C A B B A
|
||||||
|
// ┌─────────────┐
|
||||||
|
// C │_|_|\|_|_|_|_│ 0
|
||||||
|
// B │_|\|_|_|\|\|_│ 1
|
||||||
|
// A │\|_|_|\|_|_|\│ 2
|
||||||
|
// B │_|\|_|_|\|\|_│ 3
|
||||||
|
// A │\|_|_|\|_|_|\│ 4
|
||||||
|
// C │ | |\| | | | │ 5
|
||||||
|
// └─────────────┘ 6
|
||||||
|
// 0 1 2 3 4 5 6 7
|
||||||
|
//
|
||||||
|
// List X is written along the horizontal axis, while list Y is written
|
||||||
|
// along the vertical axis. At any point on this grid, if the symbol in
|
||||||
|
// list X matches the corresponding symbol in list Y, then a '\' is drawn.
|
||||||
|
// The goal of any minimal edit-script algorithm is to find a path from the
|
||||||
|
// top-left corner to the bottom-right corner, while traveling through the
|
||||||
|
// fewest horizontal or vertical edges.
|
||||||
|
// A horizontal edge is equivalent to inserting a symbol from list X.
|
||||||
|
// A vertical edge is equivalent to inserting a symbol from list Y.
|
||||||
|
// A diagonal edge is equivalent to a matching symbol between both X and Y.
|
||||||
|
|
||||||
|
// Invariants:
|
||||||
|
// - 0 ≤ fwdPath.X ≤ (fwdFrontier.X, revFrontier.X) ≤ revPath.X ≤ nx
|
||||||
|
// - 0 ≤ fwdPath.Y ≤ (fwdFrontier.Y, revFrontier.Y) ≤ revPath.Y ≤ ny
|
||||||
|
//
|
||||||
|
// In general:
|
||||||
|
// - fwdFrontier.X < revFrontier.X
|
||||||
|
// - fwdFrontier.Y < revFrontier.Y
|
||||||
|
//
|
||||||
|
// Unless, it is time for the algorithm to terminate.
|
||||||
|
fwdPath := path{+1, point{0, 0}, make(EditScript, 0, (nx+ny)/2)}
|
||||||
|
revPath := path{-1, point{nx, ny}, make(EditScript, 0)}
|
||||||
|
fwdFrontier := fwdPath.point // Forward search frontier
|
||||||
|
revFrontier := revPath.point // Reverse search frontier
|
||||||
|
|
||||||
|
// Search budget bounds the cost of searching for better paths.
|
||||||
|
// The longest sequence of non-matching symbols that can be tolerated is
|
||||||
|
// approximately the square-root of the search budget.
|
||||||
|
searchBudget := 4 * (nx + ny) // O(n)
|
||||||
|
|
||||||
|
// Running the tests with the "cmp_debug" build tag prints a visualization
|
||||||
|
// of the algorithm running in real-time. This is educational for
|
||||||
|
// understanding how the algorithm works. See debug_enable.go.
|
||||||
|
f = debug.Begin(nx, ny, f, &fwdPath.es, &revPath.es)
|
||||||
|
|
||||||
|
// The algorithm below is a greedy, meet-in-the-middle algorithm for
|
||||||
|
// computing sub-optimal edit-scripts between two lists.
|
||||||
|
//
|
||||||
|
// The algorithm is approximately as follows:
|
||||||
|
// - Searching for differences switches back-and-forth between
|
||||||
|
// a search that starts at the beginning (the top-left corner), and
|
||||||
|
// a search that starts at the end (the bottom-right corner).
|
||||||
|
// The goal of the search is connect with the search
|
||||||
|
// from the opposite corner.
|
||||||
|
// - As we search, we build a path in a greedy manner,
|
||||||
|
// where the first match seen is added to the path (this is sub-optimal,
|
||||||
|
// but provides a decent result in practice). When matches are found,
|
||||||
|
// we try the next pair of symbols in the lists and follow all matches
|
||||||
|
// as far as possible.
|
||||||
|
// - When searching for matches, we search along a diagonal going through
|
||||||
|
// through the "frontier" point. If no matches are found,
|
||||||
|
// we advance the frontier towards the opposite corner.
|
||||||
|
// - This algorithm terminates when either the X coordinates or the
|
||||||
|
// Y coordinates of the forward and reverse frontier points ever intersect.
|
||||||
|
|
||||||
|
// This algorithm is correct even if searching only in the forward direction
|
||||||
|
// or in the reverse direction. We do both because it is commonly observed
|
||||||
|
// that two lists commonly differ because elements were added to the front
|
||||||
|
// or end of the other list.
|
||||||
|
//
|
||||||
|
// Non-deterministically start with either the forward or reverse direction
|
||||||
|
// to introduce some deliberate instability so that we have the flexibility
|
||||||
|
// to change this algorithm in the future.
|
||||||
|
if flags.Deterministic || randBool {
|
||||||
|
goto forwardSearch
|
||||||
|
} else {
|
||||||
|
goto reverseSearch
|
||||||
|
}
|
||||||
|
|
||||||
|
forwardSearch:
|
||||||
|
{
|
||||||
|
// Forward search from the beginning.
|
||||||
|
if fwdFrontier.X >= revFrontier.X || fwdFrontier.Y >= revFrontier.Y || searchBudget == 0 {
|
||||||
|
goto finishSearch
|
||||||
|
}
|
||||||
|
for stop1, stop2, i := false, false, 0; !(stop1 && stop2) && searchBudget > 0; i++ {
|
||||||
|
// Search in a diagonal pattern for a match.
|
||||||
|
z := zigzag(i)
|
||||||
|
p := point{fwdFrontier.X + z, fwdFrontier.Y - z}
|
||||||
|
switch {
|
||||||
|
case p.X >= revPath.X || p.Y < fwdPath.Y:
|
||||||
|
stop1 = true // Hit top-right corner
|
||||||
|
case p.Y >= revPath.Y || p.X < fwdPath.X:
|
||||||
|
stop2 = true // Hit bottom-left corner
|
||||||
|
case f(p.X, p.Y).Equal():
|
||||||
|
// Match found, so connect the path to this point.
|
||||||
|
fwdPath.connect(p, f)
|
||||||
|
fwdPath.append(Identity)
|
||||||
|
// Follow sequence of matches as far as possible.
|
||||||
|
for fwdPath.X < revPath.X && fwdPath.Y < revPath.Y {
|
||||||
|
if !f(fwdPath.X, fwdPath.Y).Equal() {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
fwdPath.append(Identity)
|
||||||
|
}
|
||||||
|
fwdFrontier = fwdPath.point
|
||||||
|
stop1, stop2 = true, true
|
||||||
|
default:
|
||||||
|
searchBudget-- // Match not found
|
||||||
|
}
|
||||||
|
debug.Update()
|
||||||
|
}
|
||||||
|
// Advance the frontier towards reverse point.
|
||||||
|
if revPath.X-fwdFrontier.X >= revPath.Y-fwdFrontier.Y {
|
||||||
|
fwdFrontier.X++
|
||||||
|
} else {
|
||||||
|
fwdFrontier.Y++
|
||||||
|
}
|
||||||
|
goto reverseSearch
|
||||||
|
}
|
||||||
|
|
||||||
|
reverseSearch:
|
||||||
|
{
|
||||||
|
// Reverse search from the end.
|
||||||
|
if fwdFrontier.X >= revFrontier.X || fwdFrontier.Y >= revFrontier.Y || searchBudget == 0 {
|
||||||
|
goto finishSearch
|
||||||
|
}
|
||||||
|
for stop1, stop2, i := false, false, 0; !(stop1 && stop2) && searchBudget > 0; i++ {
|
||||||
|
// Search in a diagonal pattern for a match.
|
||||||
|
z := zigzag(i)
|
||||||
|
p := point{revFrontier.X - z, revFrontier.Y + z}
|
||||||
|
switch {
|
||||||
|
case fwdPath.X >= p.X || revPath.Y < p.Y:
|
||||||
|
stop1 = true // Hit bottom-left corner
|
||||||
|
case fwdPath.Y >= p.Y || revPath.X < p.X:
|
||||||
|
stop2 = true // Hit top-right corner
|
||||||
|
case f(p.X-1, p.Y-1).Equal():
|
||||||
|
// Match found, so connect the path to this point.
|
||||||
|
revPath.connect(p, f)
|
||||||
|
revPath.append(Identity)
|
||||||
|
// Follow sequence of matches as far as possible.
|
||||||
|
for fwdPath.X < revPath.X && fwdPath.Y < revPath.Y {
|
||||||
|
if !f(revPath.X-1, revPath.Y-1).Equal() {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
revPath.append(Identity)
|
||||||
|
}
|
||||||
|
revFrontier = revPath.point
|
||||||
|
stop1, stop2 = true, true
|
||||||
|
default:
|
||||||
|
searchBudget-- // Match not found
|
||||||
|
}
|
||||||
|
debug.Update()
|
||||||
|
}
|
||||||
|
// Advance the frontier towards forward point.
|
||||||
|
if revFrontier.X-fwdPath.X >= revFrontier.Y-fwdPath.Y {
|
||||||
|
revFrontier.X--
|
||||||
|
} else {
|
||||||
|
revFrontier.Y--
|
||||||
|
}
|
||||||
|
goto forwardSearch
|
||||||
|
}
|
||||||
|
|
||||||
|
finishSearch:
|
||||||
|
// Join the forward and reverse paths and then append the reverse path.
|
||||||
|
fwdPath.connect(revPath.point, f)
|
||||||
|
for i := len(revPath.es) - 1; i >= 0; i-- {
|
||||||
|
t := revPath.es[i]
|
||||||
|
revPath.es = revPath.es[:i]
|
||||||
|
fwdPath.append(t)
|
||||||
|
}
|
||||||
|
debug.Finish()
|
||||||
|
return fwdPath.es
|
||||||
|
}
|
||||||
|
|
||||||
|
type path struct {
|
||||||
|
dir int // +1 if forward, -1 if reverse
|
||||||
|
point // Leading point of the EditScript path
|
||||||
|
es EditScript
|
||||||
|
}
|
||||||
|
|
||||||
|
// connect appends any necessary Identity, Modified, UniqueX, or UniqueY types
|
||||||
|
// to the edit-script to connect p.point to dst.
|
||||||
|
func (p *path) connect(dst point, f EqualFunc) {
|
||||||
|
if p.dir > 0 {
|
||||||
|
// Connect in forward direction.
|
||||||
|
for dst.X > p.X && dst.Y > p.Y {
|
||||||
|
switch r := f(p.X, p.Y); {
|
||||||
|
case r.Equal():
|
||||||
|
p.append(Identity)
|
||||||
|
case r.Similar():
|
||||||
|
p.append(Modified)
|
||||||
|
case dst.X-p.X >= dst.Y-p.Y:
|
||||||
|
p.append(UniqueX)
|
||||||
|
default:
|
||||||
|
p.append(UniqueY)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for dst.X > p.X {
|
||||||
|
p.append(UniqueX)
|
||||||
|
}
|
||||||
|
for dst.Y > p.Y {
|
||||||
|
p.append(UniqueY)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Connect in reverse direction.
|
||||||
|
for p.X > dst.X && p.Y > dst.Y {
|
||||||
|
switch r := f(p.X-1, p.Y-1); {
|
||||||
|
case r.Equal():
|
||||||
|
p.append(Identity)
|
||||||
|
case r.Similar():
|
||||||
|
p.append(Modified)
|
||||||
|
case p.Y-dst.Y >= p.X-dst.X:
|
||||||
|
p.append(UniqueY)
|
||||||
|
default:
|
||||||
|
p.append(UniqueX)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for p.X > dst.X {
|
||||||
|
p.append(UniqueX)
|
||||||
|
}
|
||||||
|
for p.Y > dst.Y {
|
||||||
|
p.append(UniqueY)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *path) append(t EditType) {
|
||||||
|
p.es = append(p.es, t)
|
||||||
|
switch t {
|
||||||
|
case Identity, Modified:
|
||||||
|
p.add(p.dir, p.dir)
|
||||||
|
case UniqueX:
|
||||||
|
p.add(p.dir, 0)
|
||||||
|
case UniqueY:
|
||||||
|
p.add(0, p.dir)
|
||||||
|
}
|
||||||
|
debug.Update()
|
||||||
|
}
|
||||||
|
|
||||||
|
type point struct{ X, Y int }
|
||||||
|
|
||||||
|
func (p *point) add(dx, dy int) { p.X += dx; p.Y += dy }
|
||||||
|
|
||||||
|
// zigzag maps a consecutive sequence of integers to a zig-zag sequence.
|
||||||
|
//
|
||||||
|
// [0 1 2 3 4 5 ...] => [0 -1 +1 -2 +2 ...]
|
||||||
|
func zigzag(x int) int {
|
||||||
|
if x&1 != 0 {
|
||||||
|
x = ^x
|
||||||
|
}
|
||||||
|
return x >> 1
|
||||||
|
}
|
||||||
9
vendor/github.com/google/go-cmp/cmp/internal/flags/flags.go
generated
vendored
Normal file
9
vendor/github.com/google/go-cmp/cmp/internal/flags/flags.go
generated
vendored
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
// Copyright 2019, The Go Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a BSD-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
package flags
|
||||||
|
|
||||||
|
// Deterministic controls whether the output of Diff should be deterministic.
|
||||||
|
// This is only used for testing.
|
||||||
|
var Deterministic bool
|
||||||
99
vendor/github.com/google/go-cmp/cmp/internal/function/func.go
generated
vendored
Normal file
99
vendor/github.com/google/go-cmp/cmp/internal/function/func.go
generated
vendored
Normal file
@@ -0,0 +1,99 @@
|
|||||||
|
// Copyright 2017, The Go Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a BSD-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
// Package function provides functionality for identifying function types.
|
||||||
|
package function
|
||||||
|
|
||||||
|
import (
|
||||||
|
"reflect"
|
||||||
|
"regexp"
|
||||||
|
"runtime"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
type funcType int
|
||||||
|
|
||||||
|
const (
|
||||||
|
_ funcType = iota
|
||||||
|
|
||||||
|
tbFunc // func(T) bool
|
||||||
|
ttbFunc // func(T, T) bool
|
||||||
|
trbFunc // func(T, R) bool
|
||||||
|
tibFunc // func(T, I) bool
|
||||||
|
trFunc // func(T) R
|
||||||
|
|
||||||
|
Equal = ttbFunc // func(T, T) bool
|
||||||
|
EqualAssignable = tibFunc // func(T, I) bool; encapsulates func(T, T) bool
|
||||||
|
Transformer = trFunc // func(T) R
|
||||||
|
ValueFilter = ttbFunc // func(T, T) bool
|
||||||
|
Less = ttbFunc // func(T, T) bool
|
||||||
|
ValuePredicate = tbFunc // func(T) bool
|
||||||
|
KeyValuePredicate = trbFunc // func(T, R) bool
|
||||||
|
)
|
||||||
|
|
||||||
|
var boolType = reflect.TypeOf(true)
|
||||||
|
|
||||||
|
// IsType reports whether the reflect.Type is of the specified function type.
|
||||||
|
func IsType(t reflect.Type, ft funcType) bool {
|
||||||
|
if t == nil || t.Kind() != reflect.Func || t.IsVariadic() {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
ni, no := t.NumIn(), t.NumOut()
|
||||||
|
switch ft {
|
||||||
|
case tbFunc: // func(T) bool
|
||||||
|
if ni == 1 && no == 1 && t.Out(0) == boolType {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
case ttbFunc: // func(T, T) bool
|
||||||
|
if ni == 2 && no == 1 && t.In(0) == t.In(1) && t.Out(0) == boolType {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
case trbFunc: // func(T, R) bool
|
||||||
|
if ni == 2 && no == 1 && t.Out(0) == boolType {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
case tibFunc: // func(T, I) bool
|
||||||
|
if ni == 2 && no == 1 && t.In(0).AssignableTo(t.In(1)) && t.Out(0) == boolType {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
case trFunc: // func(T) R
|
||||||
|
if ni == 1 && no == 1 {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
var lastIdentRx = regexp.MustCompile(`[_\p{L}][_\p{L}\p{N}]*$`)
|
||||||
|
|
||||||
|
// NameOf returns the name of the function value.
|
||||||
|
func NameOf(v reflect.Value) string {
|
||||||
|
fnc := runtime.FuncForPC(v.Pointer())
|
||||||
|
if fnc == nil {
|
||||||
|
return "<unknown>"
|
||||||
|
}
|
||||||
|
fullName := fnc.Name() // e.g., "long/path/name/mypkg.(*MyType).(long/path/name/mypkg.myMethod)-fm"
|
||||||
|
|
||||||
|
// Method closures have a "-fm" suffix.
|
||||||
|
fullName = strings.TrimSuffix(fullName, "-fm")
|
||||||
|
|
||||||
|
var name string
|
||||||
|
for len(fullName) > 0 {
|
||||||
|
inParen := strings.HasSuffix(fullName, ")")
|
||||||
|
fullName = strings.TrimSuffix(fullName, ")")
|
||||||
|
|
||||||
|
s := lastIdentRx.FindString(fullName)
|
||||||
|
if s == "" {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
name = s + "." + name
|
||||||
|
fullName = strings.TrimSuffix(fullName, s)
|
||||||
|
|
||||||
|
if i := strings.LastIndexByte(fullName, '('); inParen && i >= 0 {
|
||||||
|
fullName = fullName[:i]
|
||||||
|
}
|
||||||
|
fullName = strings.TrimSuffix(fullName, ".")
|
||||||
|
}
|
||||||
|
return strings.TrimSuffix(name, ".")
|
||||||
|
}
|
||||||
164
vendor/github.com/google/go-cmp/cmp/internal/value/name.go
generated
vendored
Normal file
164
vendor/github.com/google/go-cmp/cmp/internal/value/name.go
generated
vendored
Normal file
@@ -0,0 +1,164 @@
|
|||||||
|
// Copyright 2020, The Go Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a BSD-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
package value
|
||||||
|
|
||||||
|
import (
|
||||||
|
"reflect"
|
||||||
|
"strconv"
|
||||||
|
)
|
||||||
|
|
||||||
|
var anyType = reflect.TypeOf((*interface{})(nil)).Elem()
|
||||||
|
|
||||||
|
// TypeString is nearly identical to reflect.Type.String,
|
||||||
|
// but has an additional option to specify that full type names be used.
|
||||||
|
func TypeString(t reflect.Type, qualified bool) string {
|
||||||
|
return string(appendTypeName(nil, t, qualified, false))
|
||||||
|
}
|
||||||
|
|
||||||
|
func appendTypeName(b []byte, t reflect.Type, qualified, elideFunc bool) []byte {
|
||||||
|
// BUG: Go reflection provides no way to disambiguate two named types
|
||||||
|
// of the same name and within the same package,
|
||||||
|
// but declared within the namespace of different functions.
|
||||||
|
|
||||||
|
// Use the "any" alias instead of "interface{}" for better readability.
|
||||||
|
if t == anyType {
|
||||||
|
return append(b, "any"...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Named type.
|
||||||
|
if t.Name() != "" {
|
||||||
|
if qualified && t.PkgPath() != "" {
|
||||||
|
b = append(b, '"')
|
||||||
|
b = append(b, t.PkgPath()...)
|
||||||
|
b = append(b, '"')
|
||||||
|
b = append(b, '.')
|
||||||
|
b = append(b, t.Name()...)
|
||||||
|
} else {
|
||||||
|
b = append(b, t.String()...)
|
||||||
|
}
|
||||||
|
return b
|
||||||
|
}
|
||||||
|
|
||||||
|
// Unnamed type.
|
||||||
|
switch k := t.Kind(); k {
|
||||||
|
case reflect.Bool, reflect.String, reflect.UnsafePointer,
|
||||||
|
reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64,
|
||||||
|
reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr,
|
||||||
|
reflect.Float32, reflect.Float64, reflect.Complex64, reflect.Complex128:
|
||||||
|
b = append(b, k.String()...)
|
||||||
|
case reflect.Chan:
|
||||||
|
if t.ChanDir() == reflect.RecvDir {
|
||||||
|
b = append(b, "<-"...)
|
||||||
|
}
|
||||||
|
b = append(b, "chan"...)
|
||||||
|
if t.ChanDir() == reflect.SendDir {
|
||||||
|
b = append(b, "<-"...)
|
||||||
|
}
|
||||||
|
b = append(b, ' ')
|
||||||
|
b = appendTypeName(b, t.Elem(), qualified, false)
|
||||||
|
case reflect.Func:
|
||||||
|
if !elideFunc {
|
||||||
|
b = append(b, "func"...)
|
||||||
|
}
|
||||||
|
b = append(b, '(')
|
||||||
|
for i := 0; i < t.NumIn(); i++ {
|
||||||
|
if i > 0 {
|
||||||
|
b = append(b, ", "...)
|
||||||
|
}
|
||||||
|
if i == t.NumIn()-1 && t.IsVariadic() {
|
||||||
|
b = append(b, "..."...)
|
||||||
|
b = appendTypeName(b, t.In(i).Elem(), qualified, false)
|
||||||
|
} else {
|
||||||
|
b = appendTypeName(b, t.In(i), qualified, false)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
b = append(b, ')')
|
||||||
|
switch t.NumOut() {
|
||||||
|
case 0:
|
||||||
|
// Do nothing
|
||||||
|
case 1:
|
||||||
|
b = append(b, ' ')
|
||||||
|
b = appendTypeName(b, t.Out(0), qualified, false)
|
||||||
|
default:
|
||||||
|
b = append(b, " ("...)
|
||||||
|
for i := 0; i < t.NumOut(); i++ {
|
||||||
|
if i > 0 {
|
||||||
|
b = append(b, ", "...)
|
||||||
|
}
|
||||||
|
b = appendTypeName(b, t.Out(i), qualified, false)
|
||||||
|
}
|
||||||
|
b = append(b, ')')
|
||||||
|
}
|
||||||
|
case reflect.Struct:
|
||||||
|
b = append(b, "struct{ "...)
|
||||||
|
for i := 0; i < t.NumField(); i++ {
|
||||||
|
if i > 0 {
|
||||||
|
b = append(b, "; "...)
|
||||||
|
}
|
||||||
|
sf := t.Field(i)
|
||||||
|
if !sf.Anonymous {
|
||||||
|
if qualified && sf.PkgPath != "" {
|
||||||
|
b = append(b, '"')
|
||||||
|
b = append(b, sf.PkgPath...)
|
||||||
|
b = append(b, '"')
|
||||||
|
b = append(b, '.')
|
||||||
|
}
|
||||||
|
b = append(b, sf.Name...)
|
||||||
|
b = append(b, ' ')
|
||||||
|
}
|
||||||
|
b = appendTypeName(b, sf.Type, qualified, false)
|
||||||
|
if sf.Tag != "" {
|
||||||
|
b = append(b, ' ')
|
||||||
|
b = strconv.AppendQuote(b, string(sf.Tag))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if b[len(b)-1] == ' ' {
|
||||||
|
b = b[:len(b)-1]
|
||||||
|
} else {
|
||||||
|
b = append(b, ' ')
|
||||||
|
}
|
||||||
|
b = append(b, '}')
|
||||||
|
case reflect.Slice, reflect.Array:
|
||||||
|
b = append(b, '[')
|
||||||
|
if k == reflect.Array {
|
||||||
|
b = strconv.AppendUint(b, uint64(t.Len()), 10)
|
||||||
|
}
|
||||||
|
b = append(b, ']')
|
||||||
|
b = appendTypeName(b, t.Elem(), qualified, false)
|
||||||
|
case reflect.Map:
|
||||||
|
b = append(b, "map["...)
|
||||||
|
b = appendTypeName(b, t.Key(), qualified, false)
|
||||||
|
b = append(b, ']')
|
||||||
|
b = appendTypeName(b, t.Elem(), qualified, false)
|
||||||
|
case reflect.Ptr:
|
||||||
|
b = append(b, '*')
|
||||||
|
b = appendTypeName(b, t.Elem(), qualified, false)
|
||||||
|
case reflect.Interface:
|
||||||
|
b = append(b, "interface{ "...)
|
||||||
|
for i := 0; i < t.NumMethod(); i++ {
|
||||||
|
if i > 0 {
|
||||||
|
b = append(b, "; "...)
|
||||||
|
}
|
||||||
|
m := t.Method(i)
|
||||||
|
if qualified && m.PkgPath != "" {
|
||||||
|
b = append(b, '"')
|
||||||
|
b = append(b, m.PkgPath...)
|
||||||
|
b = append(b, '"')
|
||||||
|
b = append(b, '.')
|
||||||
|
}
|
||||||
|
b = append(b, m.Name...)
|
||||||
|
b = appendTypeName(b, m.Type, qualified, true)
|
||||||
|
}
|
||||||
|
if b[len(b)-1] == ' ' {
|
||||||
|
b = b[:len(b)-1]
|
||||||
|
} else {
|
||||||
|
b = append(b, ' ')
|
||||||
|
}
|
||||||
|
b = append(b, '}')
|
||||||
|
default:
|
||||||
|
panic("invalid kind: " + k.String())
|
||||||
|
}
|
||||||
|
return b
|
||||||
|
}
|
||||||
34
vendor/github.com/google/go-cmp/cmp/internal/value/pointer_purego.go
generated
vendored
Normal file
34
vendor/github.com/google/go-cmp/cmp/internal/value/pointer_purego.go
generated
vendored
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
// Copyright 2018, The Go Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a BSD-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
//go:build purego
|
||||||
|
// +build purego
|
||||||
|
|
||||||
|
package value
|
||||||
|
|
||||||
|
import "reflect"
|
||||||
|
|
||||||
|
// Pointer is an opaque typed pointer and is guaranteed to be comparable.
|
||||||
|
type Pointer struct {
|
||||||
|
p uintptr
|
||||||
|
t reflect.Type
|
||||||
|
}
|
||||||
|
|
||||||
|
// PointerOf returns a Pointer from v, which must be a
|
||||||
|
// reflect.Ptr, reflect.Slice, or reflect.Map.
|
||||||
|
func PointerOf(v reflect.Value) Pointer {
|
||||||
|
// NOTE: Storing a pointer as an uintptr is technically incorrect as it
|
||||||
|
// assumes that the GC implementation does not use a moving collector.
|
||||||
|
return Pointer{v.Pointer(), v.Type()}
|
||||||
|
}
|
||||||
|
|
||||||
|
// IsNil reports whether the pointer is nil.
|
||||||
|
func (p Pointer) IsNil() bool {
|
||||||
|
return p.p == 0
|
||||||
|
}
|
||||||
|
|
||||||
|
// Uintptr returns the pointer as a uintptr.
|
||||||
|
func (p Pointer) Uintptr() uintptr {
|
||||||
|
return p.p
|
||||||
|
}
|
||||||
37
vendor/github.com/google/go-cmp/cmp/internal/value/pointer_unsafe.go
generated
vendored
Normal file
37
vendor/github.com/google/go-cmp/cmp/internal/value/pointer_unsafe.go
generated
vendored
Normal file
@@ -0,0 +1,37 @@
|
|||||||
|
// Copyright 2018, The Go Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a BSD-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
//go:build !purego
|
||||||
|
// +build !purego
|
||||||
|
|
||||||
|
package value
|
||||||
|
|
||||||
|
import (
|
||||||
|
"reflect"
|
||||||
|
"unsafe"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Pointer is an opaque typed pointer and is guaranteed to be comparable.
|
||||||
|
type Pointer struct {
|
||||||
|
p unsafe.Pointer
|
||||||
|
t reflect.Type
|
||||||
|
}
|
||||||
|
|
||||||
|
// PointerOf returns a Pointer from v, which must be a
|
||||||
|
// reflect.Ptr, reflect.Slice, or reflect.Map.
|
||||||
|
func PointerOf(v reflect.Value) Pointer {
|
||||||
|
// The proper representation of a pointer is unsafe.Pointer,
|
||||||
|
// which is necessary if the GC ever uses a moving collector.
|
||||||
|
return Pointer{unsafe.Pointer(v.Pointer()), v.Type()}
|
||||||
|
}
|
||||||
|
|
||||||
|
// IsNil reports whether the pointer is nil.
|
||||||
|
func (p Pointer) IsNil() bool {
|
||||||
|
return p.p == nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Uintptr returns the pointer as a uintptr.
|
||||||
|
func (p Pointer) Uintptr() uintptr {
|
||||||
|
return uintptr(p.p)
|
||||||
|
}
|
||||||
106
vendor/github.com/google/go-cmp/cmp/internal/value/sort.go
generated
vendored
Normal file
106
vendor/github.com/google/go-cmp/cmp/internal/value/sort.go
generated
vendored
Normal file
@@ -0,0 +1,106 @@
|
|||||||
|
// Copyright 2017, The Go Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a BSD-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
package value
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"math"
|
||||||
|
"reflect"
|
||||||
|
"sort"
|
||||||
|
)
|
||||||
|
|
||||||
|
// SortKeys sorts a list of map keys, deduplicating keys if necessary.
|
||||||
|
// The type of each value must be comparable.
|
||||||
|
func SortKeys(vs []reflect.Value) []reflect.Value {
|
||||||
|
if len(vs) == 0 {
|
||||||
|
return vs
|
||||||
|
}
|
||||||
|
|
||||||
|
// Sort the map keys.
|
||||||
|
sort.SliceStable(vs, func(i, j int) bool { return isLess(vs[i], vs[j]) })
|
||||||
|
|
||||||
|
// Deduplicate keys (fails for NaNs).
|
||||||
|
vs2 := vs[:1]
|
||||||
|
for _, v := range vs[1:] {
|
||||||
|
if isLess(vs2[len(vs2)-1], v) {
|
||||||
|
vs2 = append(vs2, v)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return vs2
|
||||||
|
}
|
||||||
|
|
||||||
|
// isLess is a generic function for sorting arbitrary map keys.
|
||||||
|
// The inputs must be of the same type and must be comparable.
|
||||||
|
func isLess(x, y reflect.Value) bool {
|
||||||
|
switch x.Type().Kind() {
|
||||||
|
case reflect.Bool:
|
||||||
|
return !x.Bool() && y.Bool()
|
||||||
|
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
|
||||||
|
return x.Int() < y.Int()
|
||||||
|
case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr:
|
||||||
|
return x.Uint() < y.Uint()
|
||||||
|
case reflect.Float32, reflect.Float64:
|
||||||
|
// NOTE: This does not sort -0 as less than +0
|
||||||
|
// since Go maps treat -0 and +0 as equal keys.
|
||||||
|
fx, fy := x.Float(), y.Float()
|
||||||
|
return fx < fy || math.IsNaN(fx) && !math.IsNaN(fy)
|
||||||
|
case reflect.Complex64, reflect.Complex128:
|
||||||
|
cx, cy := x.Complex(), y.Complex()
|
||||||
|
rx, ix, ry, iy := real(cx), imag(cx), real(cy), imag(cy)
|
||||||
|
if rx == ry || (math.IsNaN(rx) && math.IsNaN(ry)) {
|
||||||
|
return ix < iy || math.IsNaN(ix) && !math.IsNaN(iy)
|
||||||
|
}
|
||||||
|
return rx < ry || math.IsNaN(rx) && !math.IsNaN(ry)
|
||||||
|
case reflect.Ptr, reflect.UnsafePointer, reflect.Chan:
|
||||||
|
return x.Pointer() < y.Pointer()
|
||||||
|
case reflect.String:
|
||||||
|
return x.String() < y.String()
|
||||||
|
case reflect.Array:
|
||||||
|
for i := 0; i < x.Len(); i++ {
|
||||||
|
if isLess(x.Index(i), y.Index(i)) {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
if isLess(y.Index(i), x.Index(i)) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
case reflect.Struct:
|
||||||
|
for i := 0; i < x.NumField(); i++ {
|
||||||
|
if isLess(x.Field(i), y.Field(i)) {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
if isLess(y.Field(i), x.Field(i)) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
case reflect.Interface:
|
||||||
|
vx, vy := x.Elem(), y.Elem()
|
||||||
|
if !vx.IsValid() || !vy.IsValid() {
|
||||||
|
return !vx.IsValid() && vy.IsValid()
|
||||||
|
}
|
||||||
|
tx, ty := vx.Type(), vy.Type()
|
||||||
|
if tx == ty {
|
||||||
|
return isLess(x.Elem(), y.Elem())
|
||||||
|
}
|
||||||
|
if tx.Kind() != ty.Kind() {
|
||||||
|
return vx.Kind() < vy.Kind()
|
||||||
|
}
|
||||||
|
if tx.String() != ty.String() {
|
||||||
|
return tx.String() < ty.String()
|
||||||
|
}
|
||||||
|
if tx.PkgPath() != ty.PkgPath() {
|
||||||
|
return tx.PkgPath() < ty.PkgPath()
|
||||||
|
}
|
||||||
|
// This can happen in rare situations, so we fallback to just comparing
|
||||||
|
// the unique pointer for a reflect.Type. This guarantees deterministic
|
||||||
|
// ordering within a program, but it is obviously not stable.
|
||||||
|
return reflect.ValueOf(vx.Type()).Pointer() < reflect.ValueOf(vy.Type()).Pointer()
|
||||||
|
default:
|
||||||
|
// Must be Func, Map, or Slice; which are not comparable.
|
||||||
|
panic(fmt.Sprintf("%T is not comparable", x.Type()))
|
||||||
|
}
|
||||||
|
}
|
||||||
554
vendor/github.com/google/go-cmp/cmp/options.go
generated
vendored
Normal file
554
vendor/github.com/google/go-cmp/cmp/options.go
generated
vendored
Normal file
@@ -0,0 +1,554 @@
|
|||||||
|
// Copyright 2017, The Go Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a BSD-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
package cmp
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"reflect"
|
||||||
|
"regexp"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/google/go-cmp/cmp/internal/function"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Option configures for specific behavior of Equal and Diff. In particular,
|
||||||
|
// the fundamental Option functions (Ignore, Transformer, and Comparer),
|
||||||
|
// configure how equality is determined.
|
||||||
|
//
|
||||||
|
// The fundamental options may be composed with filters (FilterPath and
|
||||||
|
// FilterValues) to control the scope over which they are applied.
|
||||||
|
//
|
||||||
|
// The cmp/cmpopts package provides helper functions for creating options that
|
||||||
|
// may be used with Equal and Diff.
|
||||||
|
type Option interface {
|
||||||
|
// filter applies all filters and returns the option that remains.
|
||||||
|
// Each option may only read s.curPath and call s.callTTBFunc.
|
||||||
|
//
|
||||||
|
// An Options is returned only if multiple comparers or transformers
|
||||||
|
// can apply simultaneously and will only contain values of those types
|
||||||
|
// or sub-Options containing values of those types.
|
||||||
|
filter(s *state, t reflect.Type, vx, vy reflect.Value) applicableOption
|
||||||
|
}
|
||||||
|
|
||||||
|
// applicableOption represents the following types:
|
||||||
|
//
|
||||||
|
// Fundamental: ignore | validator | *comparer | *transformer
|
||||||
|
// Grouping: Options
|
||||||
|
type applicableOption interface {
|
||||||
|
Option
|
||||||
|
|
||||||
|
// apply executes the option, which may mutate s or panic.
|
||||||
|
apply(s *state, vx, vy reflect.Value)
|
||||||
|
}
|
||||||
|
|
||||||
|
// coreOption represents the following types:
|
||||||
|
//
|
||||||
|
// Fundamental: ignore | validator | *comparer | *transformer
|
||||||
|
// Filters: *pathFilter | *valuesFilter
|
||||||
|
type coreOption interface {
|
||||||
|
Option
|
||||||
|
isCore()
|
||||||
|
}
|
||||||
|
|
||||||
|
type core struct{}
|
||||||
|
|
||||||
|
func (core) isCore() {}
|
||||||
|
|
||||||
|
// Options is a list of Option values that also satisfies the Option interface.
|
||||||
|
// Helper comparison packages may return an Options value when packing multiple
|
||||||
|
// Option values into a single Option. When this package processes an Options,
|
||||||
|
// it will be implicitly expanded into a flat list.
|
||||||
|
//
|
||||||
|
// Applying a filter on an Options is equivalent to applying that same filter
|
||||||
|
// on all individual options held within.
|
||||||
|
type Options []Option
|
||||||
|
|
||||||
|
func (opts Options) filter(s *state, t reflect.Type, vx, vy reflect.Value) (out applicableOption) {
|
||||||
|
for _, opt := range opts {
|
||||||
|
switch opt := opt.filter(s, t, vx, vy); opt.(type) {
|
||||||
|
case ignore:
|
||||||
|
return ignore{} // Only ignore can short-circuit evaluation
|
||||||
|
case validator:
|
||||||
|
out = validator{} // Takes precedence over comparer or transformer
|
||||||
|
case *comparer, *transformer, Options:
|
||||||
|
switch out.(type) {
|
||||||
|
case nil:
|
||||||
|
out = opt
|
||||||
|
case validator:
|
||||||
|
// Keep validator
|
||||||
|
case *comparer, *transformer, Options:
|
||||||
|
out = Options{out, opt} // Conflicting comparers or transformers
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return out
|
||||||
|
}
|
||||||
|
|
||||||
|
func (opts Options) apply(s *state, _, _ reflect.Value) {
|
||||||
|
const warning = "ambiguous set of applicable options"
|
||||||
|
const help = "consider using filters to ensure at most one Comparer or Transformer may apply"
|
||||||
|
var ss []string
|
||||||
|
for _, opt := range flattenOptions(nil, opts) {
|
||||||
|
ss = append(ss, fmt.Sprint(opt))
|
||||||
|
}
|
||||||
|
set := strings.Join(ss, "\n\t")
|
||||||
|
panic(fmt.Sprintf("%s at %#v:\n\t%s\n%s", warning, s.curPath, set, help))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (opts Options) String() string {
|
||||||
|
var ss []string
|
||||||
|
for _, opt := range opts {
|
||||||
|
ss = append(ss, fmt.Sprint(opt))
|
||||||
|
}
|
||||||
|
return fmt.Sprintf("Options{%s}", strings.Join(ss, ", "))
|
||||||
|
}
|
||||||
|
|
||||||
|
// FilterPath returns a new Option where opt is only evaluated if filter f
|
||||||
|
// returns true for the current Path in the value tree.
|
||||||
|
//
|
||||||
|
// This filter is called even if a slice element or map entry is missing and
|
||||||
|
// provides an opportunity to ignore such cases. The filter function must be
|
||||||
|
// symmetric such that the filter result is identical regardless of whether the
|
||||||
|
// missing value is from x or y.
|
||||||
|
//
|
||||||
|
// The option passed in may be an Ignore, Transformer, Comparer, Options, or
|
||||||
|
// a previously filtered Option.
|
||||||
|
func FilterPath(f func(Path) bool, opt Option) Option {
|
||||||
|
if f == nil {
|
||||||
|
panic("invalid path filter function")
|
||||||
|
}
|
||||||
|
if opt := normalizeOption(opt); opt != nil {
|
||||||
|
return &pathFilter{fnc: f, opt: opt}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
type pathFilter struct {
|
||||||
|
core
|
||||||
|
fnc func(Path) bool
|
||||||
|
opt Option
|
||||||
|
}
|
||||||
|
|
||||||
|
func (f pathFilter) filter(s *state, t reflect.Type, vx, vy reflect.Value) applicableOption {
|
||||||
|
if f.fnc(s.curPath) {
|
||||||
|
return f.opt.filter(s, t, vx, vy)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (f pathFilter) String() string {
|
||||||
|
return fmt.Sprintf("FilterPath(%s, %v)", function.NameOf(reflect.ValueOf(f.fnc)), f.opt)
|
||||||
|
}
|
||||||
|
|
||||||
|
// FilterValues returns a new Option where opt is only evaluated if filter f,
|
||||||
|
// which is a function of the form "func(T, T) bool", returns true for the
|
||||||
|
// current pair of values being compared. If either value is invalid or
|
||||||
|
// the type of the values is not assignable to T, then this filter implicitly
|
||||||
|
// returns false.
|
||||||
|
//
|
||||||
|
// The filter function must be
|
||||||
|
// symmetric (i.e., agnostic to the order of the inputs) and
|
||||||
|
// deterministic (i.e., produces the same result when given the same inputs).
|
||||||
|
// If T is an interface, it is possible that f is called with two values with
|
||||||
|
// different concrete types that both implement T.
|
||||||
|
//
|
||||||
|
// The option passed in may be an Ignore, Transformer, Comparer, Options, or
|
||||||
|
// a previously filtered Option.
|
||||||
|
func FilterValues(f interface{}, opt Option) Option {
|
||||||
|
v := reflect.ValueOf(f)
|
||||||
|
if !function.IsType(v.Type(), function.ValueFilter) || v.IsNil() {
|
||||||
|
panic(fmt.Sprintf("invalid values filter function: %T", f))
|
||||||
|
}
|
||||||
|
if opt := normalizeOption(opt); opt != nil {
|
||||||
|
vf := &valuesFilter{fnc: v, opt: opt}
|
||||||
|
if ti := v.Type().In(0); ti.Kind() != reflect.Interface || ti.NumMethod() > 0 {
|
||||||
|
vf.typ = ti
|
||||||
|
}
|
||||||
|
return vf
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
type valuesFilter struct {
|
||||||
|
core
|
||||||
|
typ reflect.Type // T
|
||||||
|
fnc reflect.Value // func(T, T) bool
|
||||||
|
opt Option
|
||||||
|
}
|
||||||
|
|
||||||
|
func (f valuesFilter) filter(s *state, t reflect.Type, vx, vy reflect.Value) applicableOption {
|
||||||
|
if !vx.IsValid() || !vx.CanInterface() || !vy.IsValid() || !vy.CanInterface() {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
if (f.typ == nil || t.AssignableTo(f.typ)) && s.callTTBFunc(f.fnc, vx, vy) {
|
||||||
|
return f.opt.filter(s, t, vx, vy)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (f valuesFilter) String() string {
|
||||||
|
return fmt.Sprintf("FilterValues(%s, %v)", function.NameOf(f.fnc), f.opt)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ignore is an Option that causes all comparisons to be ignored.
|
||||||
|
// This value is intended to be combined with FilterPath or FilterValues.
|
||||||
|
// It is an error to pass an unfiltered Ignore option to Equal.
|
||||||
|
func Ignore() Option { return ignore{} }
|
||||||
|
|
||||||
|
type ignore struct{ core }
|
||||||
|
|
||||||
|
func (ignore) isFiltered() bool { return false }
|
||||||
|
func (ignore) filter(_ *state, _ reflect.Type, _, _ reflect.Value) applicableOption { return ignore{} }
|
||||||
|
func (ignore) apply(s *state, _, _ reflect.Value) { s.report(true, reportByIgnore) }
|
||||||
|
func (ignore) String() string { return "Ignore()" }
|
||||||
|
|
||||||
|
// validator is a sentinel Option type to indicate that some options could not
|
||||||
|
// be evaluated due to unexported fields, missing slice elements, or
|
||||||
|
// missing map entries. Both values are validator only for unexported fields.
|
||||||
|
type validator struct{ core }
|
||||||
|
|
||||||
|
func (validator) filter(_ *state, _ reflect.Type, vx, vy reflect.Value) applicableOption {
|
||||||
|
if !vx.IsValid() || !vy.IsValid() {
|
||||||
|
return validator{}
|
||||||
|
}
|
||||||
|
if !vx.CanInterface() || !vy.CanInterface() {
|
||||||
|
return validator{}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
func (validator) apply(s *state, vx, vy reflect.Value) {
|
||||||
|
// Implies missing slice element or map entry.
|
||||||
|
if !vx.IsValid() || !vy.IsValid() {
|
||||||
|
s.report(vx.IsValid() == vy.IsValid(), 0)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Unable to Interface implies unexported field without visibility access.
|
||||||
|
if !vx.CanInterface() || !vy.CanInterface() {
|
||||||
|
help := "consider using a custom Comparer; if you control the implementation of type, you can also consider using an Exporter, AllowUnexported, or cmpopts.IgnoreUnexported"
|
||||||
|
var name string
|
||||||
|
if t := s.curPath.Index(-2).Type(); t.Name() != "" {
|
||||||
|
// Named type with unexported fields.
|
||||||
|
name = fmt.Sprintf("%q.%v", t.PkgPath(), t.Name()) // e.g., "path/to/package".MyType
|
||||||
|
if _, ok := reflect.New(t).Interface().(error); ok {
|
||||||
|
help = "consider using cmpopts.EquateErrors to compare error values"
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Unnamed type with unexported fields. Derive PkgPath from field.
|
||||||
|
var pkgPath string
|
||||||
|
for i := 0; i < t.NumField() && pkgPath == ""; i++ {
|
||||||
|
pkgPath = t.Field(i).PkgPath
|
||||||
|
}
|
||||||
|
name = fmt.Sprintf("%q.(%v)", pkgPath, t.String()) // e.g., "path/to/package".(struct { a int })
|
||||||
|
}
|
||||||
|
panic(fmt.Sprintf("cannot handle unexported field at %#v:\n\t%v\n%s", s.curPath, name, help))
|
||||||
|
}
|
||||||
|
|
||||||
|
panic("not reachable")
|
||||||
|
}
|
||||||
|
|
||||||
|
// identRx represents a valid identifier according to the Go specification.
|
||||||
|
const identRx = `[_\p{L}][_\p{L}\p{N}]*`
|
||||||
|
|
||||||
|
var identsRx = regexp.MustCompile(`^` + identRx + `(\.` + identRx + `)*$`)
|
||||||
|
|
||||||
|
// Transformer returns an Option that applies a transformation function that
|
||||||
|
// converts values of a certain type into that of another.
|
||||||
|
//
|
||||||
|
// The transformer f must be a function "func(T) R" that converts values of
|
||||||
|
// type T to those of type R and is implicitly filtered to input values
|
||||||
|
// assignable to T. The transformer must not mutate T in any way.
|
||||||
|
//
|
||||||
|
// To help prevent some cases of infinite recursive cycles applying the
|
||||||
|
// same transform to the output of itself (e.g., in the case where the
|
||||||
|
// input and output types are the same), an implicit filter is added such that
|
||||||
|
// a transformer is applicable only if that exact transformer is not already
|
||||||
|
// in the tail of the Path since the last non-Transform step.
|
||||||
|
// For situations where the implicit filter is still insufficient,
|
||||||
|
// consider using cmpopts.AcyclicTransformer, which adds a filter
|
||||||
|
// to prevent the transformer from being recursively applied upon itself.
|
||||||
|
//
|
||||||
|
// The name is a user provided label that is used as the Transform.Name in the
|
||||||
|
// transformation PathStep (and eventually shown in the Diff output).
|
||||||
|
// The name must be a valid identifier or qualified identifier in Go syntax.
|
||||||
|
// If empty, an arbitrary name is used.
|
||||||
|
func Transformer(name string, f interface{}) Option {
|
||||||
|
v := reflect.ValueOf(f)
|
||||||
|
if !function.IsType(v.Type(), function.Transformer) || v.IsNil() {
|
||||||
|
panic(fmt.Sprintf("invalid transformer function: %T", f))
|
||||||
|
}
|
||||||
|
if name == "" {
|
||||||
|
name = function.NameOf(v)
|
||||||
|
if !identsRx.MatchString(name) {
|
||||||
|
name = "λ" // Lambda-symbol as placeholder name
|
||||||
|
}
|
||||||
|
} else if !identsRx.MatchString(name) {
|
||||||
|
panic(fmt.Sprintf("invalid name: %q", name))
|
||||||
|
}
|
||||||
|
tr := &transformer{name: name, fnc: reflect.ValueOf(f)}
|
||||||
|
if ti := v.Type().In(0); ti.Kind() != reflect.Interface || ti.NumMethod() > 0 {
|
||||||
|
tr.typ = ti
|
||||||
|
}
|
||||||
|
return tr
|
||||||
|
}
|
||||||
|
|
||||||
|
type transformer struct {
|
||||||
|
core
|
||||||
|
name string
|
||||||
|
typ reflect.Type // T
|
||||||
|
fnc reflect.Value // func(T) R
|
||||||
|
}
|
||||||
|
|
||||||
|
func (tr *transformer) isFiltered() bool { return tr.typ != nil }
|
||||||
|
|
||||||
|
func (tr *transformer) filter(s *state, t reflect.Type, _, _ reflect.Value) applicableOption {
|
||||||
|
for i := len(s.curPath) - 1; i >= 0; i-- {
|
||||||
|
if t, ok := s.curPath[i].(Transform); !ok {
|
||||||
|
break // Hit most recent non-Transform step
|
||||||
|
} else if tr == t.trans {
|
||||||
|
return nil // Cannot directly use same Transform
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if tr.typ == nil || t.AssignableTo(tr.typ) {
|
||||||
|
return tr
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (tr *transformer) apply(s *state, vx, vy reflect.Value) {
|
||||||
|
step := Transform{&transform{pathStep{typ: tr.fnc.Type().Out(0)}, tr}}
|
||||||
|
vvx := s.callTRFunc(tr.fnc, vx, step)
|
||||||
|
vvy := s.callTRFunc(tr.fnc, vy, step)
|
||||||
|
step.vx, step.vy = vvx, vvy
|
||||||
|
s.compareAny(step)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (tr transformer) String() string {
|
||||||
|
return fmt.Sprintf("Transformer(%s, %s)", tr.name, function.NameOf(tr.fnc))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Comparer returns an Option that determines whether two values are equal
|
||||||
|
// to each other.
|
||||||
|
//
|
||||||
|
// The comparer f must be a function "func(T, T) bool" and is implicitly
|
||||||
|
// filtered to input values assignable to T. If T is an interface, it is
|
||||||
|
// possible that f is called with two values of different concrete types that
|
||||||
|
// both implement T.
|
||||||
|
//
|
||||||
|
// The equality function must be:
|
||||||
|
// - Symmetric: equal(x, y) == equal(y, x)
|
||||||
|
// - Deterministic: equal(x, y) == equal(x, y)
|
||||||
|
// - Pure: equal(x, y) does not modify x or y
|
||||||
|
func Comparer(f interface{}) Option {
|
||||||
|
v := reflect.ValueOf(f)
|
||||||
|
if !function.IsType(v.Type(), function.Equal) || v.IsNil() {
|
||||||
|
panic(fmt.Sprintf("invalid comparer function: %T", f))
|
||||||
|
}
|
||||||
|
cm := &comparer{fnc: v}
|
||||||
|
if ti := v.Type().In(0); ti.Kind() != reflect.Interface || ti.NumMethod() > 0 {
|
||||||
|
cm.typ = ti
|
||||||
|
}
|
||||||
|
return cm
|
||||||
|
}
|
||||||
|
|
||||||
|
type comparer struct {
|
||||||
|
core
|
||||||
|
typ reflect.Type // T
|
||||||
|
fnc reflect.Value // func(T, T) bool
|
||||||
|
}
|
||||||
|
|
||||||
|
func (cm *comparer) isFiltered() bool { return cm.typ != nil }
|
||||||
|
|
||||||
|
func (cm *comparer) filter(_ *state, t reflect.Type, _, _ reflect.Value) applicableOption {
|
||||||
|
if cm.typ == nil || t.AssignableTo(cm.typ) {
|
||||||
|
return cm
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (cm *comparer) apply(s *state, vx, vy reflect.Value) {
|
||||||
|
eq := s.callTTBFunc(cm.fnc, vx, vy)
|
||||||
|
s.report(eq, reportByFunc)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (cm comparer) String() string {
|
||||||
|
return fmt.Sprintf("Comparer(%s)", function.NameOf(cm.fnc))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Exporter returns an Option that specifies whether Equal is allowed to
|
||||||
|
// introspect into the unexported fields of certain struct types.
|
||||||
|
//
|
||||||
|
// Users of this option must understand that comparing on unexported fields
|
||||||
|
// from external packages is not safe since changes in the internal
|
||||||
|
// implementation of some external package may cause the result of Equal
|
||||||
|
// to unexpectedly change. However, it may be valid to use this option on types
|
||||||
|
// defined in an internal package where the semantic meaning of an unexported
|
||||||
|
// field is in the control of the user.
|
||||||
|
//
|
||||||
|
// In many cases, a custom Comparer should be used instead that defines
|
||||||
|
// equality as a function of the public API of a type rather than the underlying
|
||||||
|
// unexported implementation.
|
||||||
|
//
|
||||||
|
// For example, the reflect.Type documentation defines equality to be determined
|
||||||
|
// by the == operator on the interface (essentially performing a shallow pointer
|
||||||
|
// comparison) and most attempts to compare *regexp.Regexp types are interested
|
||||||
|
// in only checking that the regular expression strings are equal.
|
||||||
|
// Both of these are accomplished using Comparers:
|
||||||
|
//
|
||||||
|
// Comparer(func(x, y reflect.Type) bool { return x == y })
|
||||||
|
// Comparer(func(x, y *regexp.Regexp) bool { return x.String() == y.String() })
|
||||||
|
//
|
||||||
|
// In other cases, the cmpopts.IgnoreUnexported option can be used to ignore
|
||||||
|
// all unexported fields on specified struct types.
|
||||||
|
func Exporter(f func(reflect.Type) bool) Option {
|
||||||
|
if !supportExporters {
|
||||||
|
panic("Exporter is not supported on purego builds")
|
||||||
|
}
|
||||||
|
return exporter(f)
|
||||||
|
}
|
||||||
|
|
||||||
|
type exporter func(reflect.Type) bool
|
||||||
|
|
||||||
|
func (exporter) filter(_ *state, _ reflect.Type, _, _ reflect.Value) applicableOption {
|
||||||
|
panic("not implemented")
|
||||||
|
}
|
||||||
|
|
||||||
|
// AllowUnexported returns an Options that allows Equal to forcibly introspect
|
||||||
|
// unexported fields of the specified struct types.
|
||||||
|
//
|
||||||
|
// See Exporter for the proper use of this option.
|
||||||
|
func AllowUnexported(types ...interface{}) Option {
|
||||||
|
m := make(map[reflect.Type]bool)
|
||||||
|
for _, typ := range types {
|
||||||
|
t := reflect.TypeOf(typ)
|
||||||
|
if t.Kind() != reflect.Struct {
|
||||||
|
panic(fmt.Sprintf("invalid struct type: %T", typ))
|
||||||
|
}
|
||||||
|
m[t] = true
|
||||||
|
}
|
||||||
|
return exporter(func(t reflect.Type) bool { return m[t] })
|
||||||
|
}
|
||||||
|
|
||||||
|
// Result represents the comparison result for a single node and
|
||||||
|
// is provided by cmp when calling Report (see Reporter).
|
||||||
|
type Result struct {
|
||||||
|
_ [0]func() // Make Result incomparable
|
||||||
|
flags resultFlags
|
||||||
|
}
|
||||||
|
|
||||||
|
// Equal reports whether the node was determined to be equal or not.
|
||||||
|
// As a special case, ignored nodes are considered equal.
|
||||||
|
func (r Result) Equal() bool {
|
||||||
|
return r.flags&(reportEqual|reportByIgnore) != 0
|
||||||
|
}
|
||||||
|
|
||||||
|
// ByIgnore reports whether the node is equal because it was ignored.
|
||||||
|
// This never reports true if Equal reports false.
|
||||||
|
func (r Result) ByIgnore() bool {
|
||||||
|
return r.flags&reportByIgnore != 0
|
||||||
|
}
|
||||||
|
|
||||||
|
// ByMethod reports whether the Equal method determined equality.
|
||||||
|
func (r Result) ByMethod() bool {
|
||||||
|
return r.flags&reportByMethod != 0
|
||||||
|
}
|
||||||
|
|
||||||
|
// ByFunc reports whether a Comparer function determined equality.
|
||||||
|
func (r Result) ByFunc() bool {
|
||||||
|
return r.flags&reportByFunc != 0
|
||||||
|
}
|
||||||
|
|
||||||
|
// ByCycle reports whether a reference cycle was detected.
|
||||||
|
func (r Result) ByCycle() bool {
|
||||||
|
return r.flags&reportByCycle != 0
|
||||||
|
}
|
||||||
|
|
||||||
|
type resultFlags uint
|
||||||
|
|
||||||
|
const (
|
||||||
|
_ resultFlags = (1 << iota) / 2
|
||||||
|
|
||||||
|
reportEqual
|
||||||
|
reportUnequal
|
||||||
|
reportByIgnore
|
||||||
|
reportByMethod
|
||||||
|
reportByFunc
|
||||||
|
reportByCycle
|
||||||
|
)
|
||||||
|
|
||||||
|
// Reporter is an Option that can be passed to Equal. When Equal traverses
|
||||||
|
// the value trees, it calls PushStep as it descends into each node in the
|
||||||
|
// tree and PopStep as it ascend out of the node. The leaves of the tree are
|
||||||
|
// either compared (determined to be equal or not equal) or ignored and reported
|
||||||
|
// as such by calling the Report method.
|
||||||
|
func Reporter(r interface {
|
||||||
|
// PushStep is called when a tree-traversal operation is performed.
|
||||||
|
// The PathStep itself is only valid until the step is popped.
|
||||||
|
// The PathStep.Values are valid for the duration of the entire traversal
|
||||||
|
// and must not be mutated.
|
||||||
|
//
|
||||||
|
// Equal always calls PushStep at the start to provide an operation-less
|
||||||
|
// PathStep used to report the root values.
|
||||||
|
//
|
||||||
|
// Within a slice, the exact set of inserted, removed, or modified elements
|
||||||
|
// is unspecified and may change in future implementations.
|
||||||
|
// The entries of a map are iterated through in an unspecified order.
|
||||||
|
PushStep(PathStep)
|
||||||
|
|
||||||
|
// Report is called exactly once on leaf nodes to report whether the
|
||||||
|
// comparison identified the node as equal, unequal, or ignored.
|
||||||
|
// A leaf node is one that is immediately preceded by and followed by
|
||||||
|
// a pair of PushStep and PopStep calls.
|
||||||
|
Report(Result)
|
||||||
|
|
||||||
|
// PopStep ascends back up the value tree.
|
||||||
|
// There is always a matching pop call for every push call.
|
||||||
|
PopStep()
|
||||||
|
}) Option {
|
||||||
|
return reporter{r}
|
||||||
|
}
|
||||||
|
|
||||||
|
type reporter struct{ reporterIface }
|
||||||
|
type reporterIface interface {
|
||||||
|
PushStep(PathStep)
|
||||||
|
Report(Result)
|
||||||
|
PopStep()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (reporter) filter(_ *state, _ reflect.Type, _, _ reflect.Value) applicableOption {
|
||||||
|
panic("not implemented")
|
||||||
|
}
|
||||||
|
|
||||||
|
// normalizeOption normalizes the input options such that all Options groups
|
||||||
|
// are flattened and groups with a single element are reduced to that element.
|
||||||
|
// Only coreOptions and Options containing coreOptions are allowed.
|
||||||
|
func normalizeOption(src Option) Option {
|
||||||
|
switch opts := flattenOptions(nil, Options{src}); len(opts) {
|
||||||
|
case 0:
|
||||||
|
return nil
|
||||||
|
case 1:
|
||||||
|
return opts[0]
|
||||||
|
default:
|
||||||
|
return opts
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// flattenOptions copies all options in src to dst as a flat list.
|
||||||
|
// Only coreOptions and Options containing coreOptions are allowed.
|
||||||
|
func flattenOptions(dst, src Options) Options {
|
||||||
|
for _, opt := range src {
|
||||||
|
switch opt := opt.(type) {
|
||||||
|
case nil:
|
||||||
|
continue
|
||||||
|
case Options:
|
||||||
|
dst = flattenOptions(dst, opt)
|
||||||
|
case coreOption:
|
||||||
|
dst = append(dst, opt)
|
||||||
|
default:
|
||||||
|
panic(fmt.Sprintf("invalid option type: %T", opt))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return dst
|
||||||
|
}
|
||||||
380
vendor/github.com/google/go-cmp/cmp/path.go
generated
vendored
Normal file
380
vendor/github.com/google/go-cmp/cmp/path.go
generated
vendored
Normal file
@@ -0,0 +1,380 @@
|
|||||||
|
// Copyright 2017, The Go Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a BSD-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
package cmp
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"reflect"
|
||||||
|
"strings"
|
||||||
|
"unicode"
|
||||||
|
"unicode/utf8"
|
||||||
|
|
||||||
|
"github.com/google/go-cmp/cmp/internal/value"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Path is a list of PathSteps describing the sequence of operations to get
|
||||||
|
// from some root type to the current position in the value tree.
|
||||||
|
// The first Path element is always an operation-less PathStep that exists
|
||||||
|
// simply to identify the initial type.
|
||||||
|
//
|
||||||
|
// When traversing structs with embedded structs, the embedded struct will
|
||||||
|
// always be accessed as a field before traversing the fields of the
|
||||||
|
// embedded struct themselves. That is, an exported field from the
|
||||||
|
// embedded struct will never be accessed directly from the parent struct.
|
||||||
|
type Path []PathStep
|
||||||
|
|
||||||
|
// PathStep is a union-type for specific operations to traverse
|
||||||
|
// a value's tree structure. Users of this package never need to implement
|
||||||
|
// these types as values of this type will be returned by this package.
|
||||||
|
//
|
||||||
|
// Implementations of this interface are
|
||||||
|
// StructField, SliceIndex, MapIndex, Indirect, TypeAssertion, and Transform.
|
||||||
|
type PathStep interface {
|
||||||
|
String() string
|
||||||
|
|
||||||
|
// Type is the resulting type after performing the path step.
|
||||||
|
Type() reflect.Type
|
||||||
|
|
||||||
|
// Values is the resulting values after performing the path step.
|
||||||
|
// The type of each valid value is guaranteed to be identical to Type.
|
||||||
|
//
|
||||||
|
// In some cases, one or both may be invalid or have restrictions:
|
||||||
|
// - For StructField, both are not interface-able if the current field
|
||||||
|
// is unexported and the struct type is not explicitly permitted by
|
||||||
|
// an Exporter to traverse unexported fields.
|
||||||
|
// - For SliceIndex, one may be invalid if an element is missing from
|
||||||
|
// either the x or y slice.
|
||||||
|
// - For MapIndex, one may be invalid if an entry is missing from
|
||||||
|
// either the x or y map.
|
||||||
|
//
|
||||||
|
// The provided values must not be mutated.
|
||||||
|
Values() (vx, vy reflect.Value)
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
_ PathStep = StructField{}
|
||||||
|
_ PathStep = SliceIndex{}
|
||||||
|
_ PathStep = MapIndex{}
|
||||||
|
_ PathStep = Indirect{}
|
||||||
|
_ PathStep = TypeAssertion{}
|
||||||
|
_ PathStep = Transform{}
|
||||||
|
)
|
||||||
|
|
||||||
|
func (pa *Path) push(s PathStep) {
|
||||||
|
*pa = append(*pa, s)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (pa *Path) pop() {
|
||||||
|
*pa = (*pa)[:len(*pa)-1]
|
||||||
|
}
|
||||||
|
|
||||||
|
// Last returns the last PathStep in the Path.
|
||||||
|
// If the path is empty, this returns a non-nil PathStep that reports a nil Type.
|
||||||
|
func (pa Path) Last() PathStep {
|
||||||
|
return pa.Index(-1)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Index returns the ith step in the Path and supports negative indexing.
|
||||||
|
// A negative index starts counting from the tail of the Path such that -1
|
||||||
|
// refers to the last step, -2 refers to the second-to-last step, and so on.
|
||||||
|
// If index is invalid, this returns a non-nil PathStep that reports a nil Type.
|
||||||
|
func (pa Path) Index(i int) PathStep {
|
||||||
|
if i < 0 {
|
||||||
|
i = len(pa) + i
|
||||||
|
}
|
||||||
|
if i < 0 || i >= len(pa) {
|
||||||
|
return pathStep{}
|
||||||
|
}
|
||||||
|
return pa[i]
|
||||||
|
}
|
||||||
|
|
||||||
|
// String returns the simplified path to a node.
|
||||||
|
// The simplified path only contains struct field accesses.
|
||||||
|
//
|
||||||
|
// For example:
|
||||||
|
//
|
||||||
|
// MyMap.MySlices.MyField
|
||||||
|
func (pa Path) String() string {
|
||||||
|
var ss []string
|
||||||
|
for _, s := range pa {
|
||||||
|
if _, ok := s.(StructField); ok {
|
||||||
|
ss = append(ss, s.String())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return strings.TrimPrefix(strings.Join(ss, ""), ".")
|
||||||
|
}
|
||||||
|
|
||||||
|
// GoString returns the path to a specific node using Go syntax.
|
||||||
|
//
|
||||||
|
// For example:
|
||||||
|
//
|
||||||
|
// (*root.MyMap["key"].(*mypkg.MyStruct).MySlices)[2][3].MyField
|
||||||
|
func (pa Path) GoString() string {
|
||||||
|
var ssPre, ssPost []string
|
||||||
|
var numIndirect int
|
||||||
|
for i, s := range pa {
|
||||||
|
var nextStep PathStep
|
||||||
|
if i+1 < len(pa) {
|
||||||
|
nextStep = pa[i+1]
|
||||||
|
}
|
||||||
|
switch s := s.(type) {
|
||||||
|
case Indirect:
|
||||||
|
numIndirect++
|
||||||
|
pPre, pPost := "(", ")"
|
||||||
|
switch nextStep.(type) {
|
||||||
|
case Indirect:
|
||||||
|
continue // Next step is indirection, so let them batch up
|
||||||
|
case StructField:
|
||||||
|
numIndirect-- // Automatic indirection on struct fields
|
||||||
|
case nil:
|
||||||
|
pPre, pPost = "", "" // Last step; no need for parenthesis
|
||||||
|
}
|
||||||
|
if numIndirect > 0 {
|
||||||
|
ssPre = append(ssPre, pPre+strings.Repeat("*", numIndirect))
|
||||||
|
ssPost = append(ssPost, pPost)
|
||||||
|
}
|
||||||
|
numIndirect = 0
|
||||||
|
continue
|
||||||
|
case Transform:
|
||||||
|
ssPre = append(ssPre, s.trans.name+"(")
|
||||||
|
ssPost = append(ssPost, ")")
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
ssPost = append(ssPost, s.String())
|
||||||
|
}
|
||||||
|
for i, j := 0, len(ssPre)-1; i < j; i, j = i+1, j-1 {
|
||||||
|
ssPre[i], ssPre[j] = ssPre[j], ssPre[i]
|
||||||
|
}
|
||||||
|
return strings.Join(ssPre, "") + strings.Join(ssPost, "")
|
||||||
|
}
|
||||||
|
|
||||||
|
type pathStep struct {
|
||||||
|
typ reflect.Type
|
||||||
|
vx, vy reflect.Value
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ps pathStep) Type() reflect.Type { return ps.typ }
|
||||||
|
func (ps pathStep) Values() (vx, vy reflect.Value) { return ps.vx, ps.vy }
|
||||||
|
func (ps pathStep) String() string {
|
||||||
|
if ps.typ == nil {
|
||||||
|
return "<nil>"
|
||||||
|
}
|
||||||
|
s := value.TypeString(ps.typ, false)
|
||||||
|
if s == "" || strings.ContainsAny(s, "{}\n") {
|
||||||
|
return "root" // Type too simple or complex to print
|
||||||
|
}
|
||||||
|
return fmt.Sprintf("{%s}", s)
|
||||||
|
}
|
||||||
|
|
||||||
|
// StructField represents a struct field access on a field called Name.
|
||||||
|
type StructField struct{ *structField }
|
||||||
|
type structField struct {
|
||||||
|
pathStep
|
||||||
|
name string
|
||||||
|
idx int
|
||||||
|
|
||||||
|
// These fields are used for forcibly accessing an unexported field.
|
||||||
|
// pvx, pvy, and field are only valid if unexported is true.
|
||||||
|
unexported bool
|
||||||
|
mayForce bool // Forcibly allow visibility
|
||||||
|
paddr bool // Was parent addressable?
|
||||||
|
pvx, pvy reflect.Value // Parent values (always addressable)
|
||||||
|
field reflect.StructField // Field information
|
||||||
|
}
|
||||||
|
|
||||||
|
func (sf StructField) Type() reflect.Type { return sf.typ }
|
||||||
|
func (sf StructField) Values() (vx, vy reflect.Value) {
|
||||||
|
if !sf.unexported {
|
||||||
|
return sf.vx, sf.vy // CanInterface reports true
|
||||||
|
}
|
||||||
|
|
||||||
|
// Forcibly obtain read-write access to an unexported struct field.
|
||||||
|
if sf.mayForce {
|
||||||
|
vx = retrieveUnexportedField(sf.pvx, sf.field, sf.paddr)
|
||||||
|
vy = retrieveUnexportedField(sf.pvy, sf.field, sf.paddr)
|
||||||
|
return vx, vy // CanInterface reports true
|
||||||
|
}
|
||||||
|
return sf.vx, sf.vy // CanInterface reports false
|
||||||
|
}
|
||||||
|
func (sf StructField) String() string { return fmt.Sprintf(".%s", sf.name) }
|
||||||
|
|
||||||
|
// Name is the field name.
|
||||||
|
func (sf StructField) Name() string { return sf.name }
|
||||||
|
|
||||||
|
// Index is the index of the field in the parent struct type.
|
||||||
|
// See reflect.Type.Field.
|
||||||
|
func (sf StructField) Index() int { return sf.idx }
|
||||||
|
|
||||||
|
// SliceIndex is an index operation on a slice or array at some index Key.
|
||||||
|
type SliceIndex struct{ *sliceIndex }
|
||||||
|
type sliceIndex struct {
|
||||||
|
pathStep
|
||||||
|
xkey, ykey int
|
||||||
|
isSlice bool // False for reflect.Array
|
||||||
|
}
|
||||||
|
|
||||||
|
func (si SliceIndex) Type() reflect.Type { return si.typ }
|
||||||
|
func (si SliceIndex) Values() (vx, vy reflect.Value) { return si.vx, si.vy }
|
||||||
|
func (si SliceIndex) String() string {
|
||||||
|
switch {
|
||||||
|
case si.xkey == si.ykey:
|
||||||
|
return fmt.Sprintf("[%d]", si.xkey)
|
||||||
|
case si.ykey == -1:
|
||||||
|
// [5->?] means "I don't know where X[5] went"
|
||||||
|
return fmt.Sprintf("[%d->?]", si.xkey)
|
||||||
|
case si.xkey == -1:
|
||||||
|
// [?->3] means "I don't know where Y[3] came from"
|
||||||
|
return fmt.Sprintf("[?->%d]", si.ykey)
|
||||||
|
default:
|
||||||
|
// [5->3] means "X[5] moved to Y[3]"
|
||||||
|
return fmt.Sprintf("[%d->%d]", si.xkey, si.ykey)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Key is the index key; it may return -1 if in a split state
|
||||||
|
func (si SliceIndex) Key() int {
|
||||||
|
if si.xkey != si.ykey {
|
||||||
|
return -1
|
||||||
|
}
|
||||||
|
return si.xkey
|
||||||
|
}
|
||||||
|
|
||||||
|
// SplitKeys are the indexes for indexing into slices in the
|
||||||
|
// x and y values, respectively. These indexes may differ due to the
|
||||||
|
// insertion or removal of an element in one of the slices, causing
|
||||||
|
// all of the indexes to be shifted. If an index is -1, then that
|
||||||
|
// indicates that the element does not exist in the associated slice.
|
||||||
|
//
|
||||||
|
// Key is guaranteed to return -1 if and only if the indexes returned
|
||||||
|
// by SplitKeys are not the same. SplitKeys will never return -1 for
|
||||||
|
// both indexes.
|
||||||
|
func (si SliceIndex) SplitKeys() (ix, iy int) { return si.xkey, si.ykey }
|
||||||
|
|
||||||
|
// MapIndex is an index operation on a map at some index Key.
|
||||||
|
type MapIndex struct{ *mapIndex }
|
||||||
|
type mapIndex struct {
|
||||||
|
pathStep
|
||||||
|
key reflect.Value
|
||||||
|
}
|
||||||
|
|
||||||
|
func (mi MapIndex) Type() reflect.Type { return mi.typ }
|
||||||
|
func (mi MapIndex) Values() (vx, vy reflect.Value) { return mi.vx, mi.vy }
|
||||||
|
func (mi MapIndex) String() string { return fmt.Sprintf("[%#v]", mi.key) }
|
||||||
|
|
||||||
|
// Key is the value of the map key.
|
||||||
|
func (mi MapIndex) Key() reflect.Value { return mi.key }
|
||||||
|
|
||||||
|
// Indirect represents pointer indirection on the parent type.
|
||||||
|
type Indirect struct{ *indirect }
|
||||||
|
type indirect struct {
|
||||||
|
pathStep
|
||||||
|
}
|
||||||
|
|
||||||
|
func (in Indirect) Type() reflect.Type { return in.typ }
|
||||||
|
func (in Indirect) Values() (vx, vy reflect.Value) { return in.vx, in.vy }
|
||||||
|
func (in Indirect) String() string { return "*" }
|
||||||
|
|
||||||
|
// TypeAssertion represents a type assertion on an interface.
|
||||||
|
type TypeAssertion struct{ *typeAssertion }
|
||||||
|
type typeAssertion struct {
|
||||||
|
pathStep
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ta TypeAssertion) Type() reflect.Type { return ta.typ }
|
||||||
|
func (ta TypeAssertion) Values() (vx, vy reflect.Value) { return ta.vx, ta.vy }
|
||||||
|
func (ta TypeAssertion) String() string { return fmt.Sprintf(".(%v)", value.TypeString(ta.typ, false)) }
|
||||||
|
|
||||||
|
// Transform is a transformation from the parent type to the current type.
|
||||||
|
type Transform struct{ *transform }
|
||||||
|
type transform struct {
|
||||||
|
pathStep
|
||||||
|
trans *transformer
|
||||||
|
}
|
||||||
|
|
||||||
|
func (tf Transform) Type() reflect.Type { return tf.typ }
|
||||||
|
func (tf Transform) Values() (vx, vy reflect.Value) { return tf.vx, tf.vy }
|
||||||
|
func (tf Transform) String() string { return fmt.Sprintf("%s()", tf.trans.name) }
|
||||||
|
|
||||||
|
// Name is the name of the Transformer.
|
||||||
|
func (tf Transform) Name() string { return tf.trans.name }
|
||||||
|
|
||||||
|
// Func is the function pointer to the transformer function.
|
||||||
|
func (tf Transform) Func() reflect.Value { return tf.trans.fnc }
|
||||||
|
|
||||||
|
// Option returns the originally constructed Transformer option.
|
||||||
|
// The == operator can be used to detect the exact option used.
|
||||||
|
func (tf Transform) Option() Option { return tf.trans }
|
||||||
|
|
||||||
|
// pointerPath represents a dual-stack of pointers encountered when
|
||||||
|
// recursively traversing the x and y values. This data structure supports
|
||||||
|
// detection of cycles and determining whether the cycles are equal.
|
||||||
|
// In Go, cycles can occur via pointers, slices, and maps.
|
||||||
|
//
|
||||||
|
// The pointerPath uses a map to represent a stack; where descension into a
|
||||||
|
// pointer pushes the address onto the stack, and ascension from a pointer
|
||||||
|
// pops the address from the stack. Thus, when traversing into a pointer from
|
||||||
|
// reflect.Ptr, reflect.Slice element, or reflect.Map, we can detect cycles
|
||||||
|
// by checking whether the pointer has already been visited. The cycle detection
|
||||||
|
// uses a separate stack for the x and y values.
|
||||||
|
//
|
||||||
|
// If a cycle is detected we need to determine whether the two pointers
|
||||||
|
// should be considered equal. The definition of equality chosen by Equal
|
||||||
|
// requires two graphs to have the same structure. To determine this, both the
|
||||||
|
// x and y values must have a cycle where the previous pointers were also
|
||||||
|
// encountered together as a pair.
|
||||||
|
//
|
||||||
|
// Semantically, this is equivalent to augmenting Indirect, SliceIndex, and
|
||||||
|
// MapIndex with pointer information for the x and y values.
|
||||||
|
// Suppose px and py are two pointers to compare, we then search the
|
||||||
|
// Path for whether px was ever encountered in the Path history of x, and
|
||||||
|
// similarly so with py. If either side has a cycle, the comparison is only
|
||||||
|
// equal if both px and py have a cycle resulting from the same PathStep.
|
||||||
|
//
|
||||||
|
// Using a map as a stack is more performant as we can perform cycle detection
|
||||||
|
// in O(1) instead of O(N) where N is len(Path).
|
||||||
|
type pointerPath struct {
|
||||||
|
// mx is keyed by x pointers, where the value is the associated y pointer.
|
||||||
|
mx map[value.Pointer]value.Pointer
|
||||||
|
// my is keyed by y pointers, where the value is the associated x pointer.
|
||||||
|
my map[value.Pointer]value.Pointer
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *pointerPath) Init() {
|
||||||
|
p.mx = make(map[value.Pointer]value.Pointer)
|
||||||
|
p.my = make(map[value.Pointer]value.Pointer)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Push indicates intent to descend into pointers vx and vy where
|
||||||
|
// visited reports whether either has been seen before. If visited before,
|
||||||
|
// equal reports whether both pointers were encountered together.
|
||||||
|
// Pop must be called if and only if the pointers were never visited.
|
||||||
|
//
|
||||||
|
// The pointers vx and vy must be a reflect.Ptr, reflect.Slice, or reflect.Map
|
||||||
|
// and be non-nil.
|
||||||
|
func (p pointerPath) Push(vx, vy reflect.Value) (equal, visited bool) {
|
||||||
|
px := value.PointerOf(vx)
|
||||||
|
py := value.PointerOf(vy)
|
||||||
|
_, ok1 := p.mx[px]
|
||||||
|
_, ok2 := p.my[py]
|
||||||
|
if ok1 || ok2 {
|
||||||
|
equal = p.mx[px] == py && p.my[py] == px // Pointers paired together
|
||||||
|
return equal, true
|
||||||
|
}
|
||||||
|
p.mx[px] = py
|
||||||
|
p.my[py] = px
|
||||||
|
return false, false
|
||||||
|
}
|
||||||
|
|
||||||
|
// Pop ascends from pointers vx and vy.
|
||||||
|
func (p pointerPath) Pop(vx, vy reflect.Value) {
|
||||||
|
delete(p.mx, value.PointerOf(vx))
|
||||||
|
delete(p.my, value.PointerOf(vy))
|
||||||
|
}
|
||||||
|
|
||||||
|
// isExported reports whether the identifier is exported.
|
||||||
|
func isExported(id string) bool {
|
||||||
|
r, _ := utf8.DecodeRuneInString(id)
|
||||||
|
return unicode.IsUpper(r)
|
||||||
|
}
|
||||||
54
vendor/github.com/google/go-cmp/cmp/report.go
generated
vendored
Normal file
54
vendor/github.com/google/go-cmp/cmp/report.go
generated
vendored
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
// Copyright 2017, The Go Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a BSD-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
package cmp
|
||||||
|
|
||||||
|
// defaultReporter implements the reporter interface.
|
||||||
|
//
|
||||||
|
// As Equal serially calls the PushStep, Report, and PopStep methods, the
|
||||||
|
// defaultReporter constructs a tree-based representation of the compared value
|
||||||
|
// and the result of each comparison (see valueNode).
|
||||||
|
//
|
||||||
|
// When the String method is called, the FormatDiff method transforms the
|
||||||
|
// valueNode tree into a textNode tree, which is a tree-based representation
|
||||||
|
// of the textual output (see textNode).
|
||||||
|
//
|
||||||
|
// Lastly, the textNode.String method produces the final report as a string.
|
||||||
|
type defaultReporter struct {
|
||||||
|
root *valueNode
|
||||||
|
curr *valueNode
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *defaultReporter) PushStep(ps PathStep) {
|
||||||
|
r.curr = r.curr.PushStep(ps)
|
||||||
|
if r.root == nil {
|
||||||
|
r.root = r.curr
|
||||||
|
}
|
||||||
|
}
|
||||||
|
func (r *defaultReporter) Report(rs Result) {
|
||||||
|
r.curr.Report(rs)
|
||||||
|
}
|
||||||
|
func (r *defaultReporter) PopStep() {
|
||||||
|
r.curr = r.curr.PopStep()
|
||||||
|
}
|
||||||
|
|
||||||
|
// String provides a full report of the differences detected as a structured
|
||||||
|
// literal in pseudo-Go syntax. String may only be called after the entire tree
|
||||||
|
// has been traversed.
|
||||||
|
func (r *defaultReporter) String() string {
|
||||||
|
assert(r.root != nil && r.curr == nil)
|
||||||
|
if r.root.NumDiff == 0 {
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
ptrs := new(pointerReferences)
|
||||||
|
text := formatOptions{}.FormatDiff(r.root, ptrs)
|
||||||
|
resolveReferences(text)
|
||||||
|
return text.String()
|
||||||
|
}
|
||||||
|
|
||||||
|
func assert(ok bool) {
|
||||||
|
if !ok {
|
||||||
|
panic("assertion failure")
|
||||||
|
}
|
||||||
|
}
|
||||||
433
vendor/github.com/google/go-cmp/cmp/report_compare.go
generated
vendored
Normal file
433
vendor/github.com/google/go-cmp/cmp/report_compare.go
generated
vendored
Normal file
@@ -0,0 +1,433 @@
|
|||||||
|
// Copyright 2019, The Go Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a BSD-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
package cmp
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"reflect"
|
||||||
|
)
|
||||||
|
|
||||||
|
// numContextRecords is the number of surrounding equal records to print.
|
||||||
|
const numContextRecords = 2
|
||||||
|
|
||||||
|
type diffMode byte
|
||||||
|
|
||||||
|
const (
|
||||||
|
diffUnknown diffMode = 0
|
||||||
|
diffIdentical diffMode = ' '
|
||||||
|
diffRemoved diffMode = '-'
|
||||||
|
diffInserted diffMode = '+'
|
||||||
|
)
|
||||||
|
|
||||||
|
type typeMode int
|
||||||
|
|
||||||
|
const (
|
||||||
|
// emitType always prints the type.
|
||||||
|
emitType typeMode = iota
|
||||||
|
// elideType never prints the type.
|
||||||
|
elideType
|
||||||
|
// autoType prints the type only for composite kinds
|
||||||
|
// (i.e., structs, slices, arrays, and maps).
|
||||||
|
autoType
|
||||||
|
)
|
||||||
|
|
||||||
|
type formatOptions struct {
|
||||||
|
// DiffMode controls the output mode of FormatDiff.
|
||||||
|
//
|
||||||
|
// If diffUnknown, then produce a diff of the x and y values.
|
||||||
|
// If diffIdentical, then emit values as if they were equal.
|
||||||
|
// If diffRemoved, then only emit x values (ignoring y values).
|
||||||
|
// If diffInserted, then only emit y values (ignoring x values).
|
||||||
|
DiffMode diffMode
|
||||||
|
|
||||||
|
// TypeMode controls whether to print the type for the current node.
|
||||||
|
//
|
||||||
|
// As a general rule of thumb, we always print the type of the next node
|
||||||
|
// after an interface, and always elide the type of the next node after
|
||||||
|
// a slice or map node.
|
||||||
|
TypeMode typeMode
|
||||||
|
|
||||||
|
// formatValueOptions are options specific to printing reflect.Values.
|
||||||
|
formatValueOptions
|
||||||
|
}
|
||||||
|
|
||||||
|
func (opts formatOptions) WithDiffMode(d diffMode) formatOptions {
|
||||||
|
opts.DiffMode = d
|
||||||
|
return opts
|
||||||
|
}
|
||||||
|
func (opts formatOptions) WithTypeMode(t typeMode) formatOptions {
|
||||||
|
opts.TypeMode = t
|
||||||
|
return opts
|
||||||
|
}
|
||||||
|
func (opts formatOptions) WithVerbosity(level int) formatOptions {
|
||||||
|
opts.VerbosityLevel = level
|
||||||
|
opts.LimitVerbosity = true
|
||||||
|
return opts
|
||||||
|
}
|
||||||
|
func (opts formatOptions) verbosity() uint {
|
||||||
|
switch {
|
||||||
|
case opts.VerbosityLevel < 0:
|
||||||
|
return 0
|
||||||
|
case opts.VerbosityLevel > 16:
|
||||||
|
return 16 // some reasonable maximum to avoid shift overflow
|
||||||
|
default:
|
||||||
|
return uint(opts.VerbosityLevel)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const maxVerbosityPreset = 6
|
||||||
|
|
||||||
|
// verbosityPreset modifies the verbosity settings given an index
|
||||||
|
// between 0 and maxVerbosityPreset, inclusive.
|
||||||
|
func verbosityPreset(opts formatOptions, i int) formatOptions {
|
||||||
|
opts.VerbosityLevel = int(opts.verbosity()) + 2*i
|
||||||
|
if i > 0 {
|
||||||
|
opts.AvoidStringer = true
|
||||||
|
}
|
||||||
|
if i >= maxVerbosityPreset {
|
||||||
|
opts.PrintAddresses = true
|
||||||
|
opts.QualifiedNames = true
|
||||||
|
}
|
||||||
|
return opts
|
||||||
|
}
|
||||||
|
|
||||||
|
// FormatDiff converts a valueNode tree into a textNode tree, where the later
|
||||||
|
// is a textual representation of the differences detected in the former.
|
||||||
|
func (opts formatOptions) FormatDiff(v *valueNode, ptrs *pointerReferences) (out textNode) {
|
||||||
|
if opts.DiffMode == diffIdentical {
|
||||||
|
opts = opts.WithVerbosity(1)
|
||||||
|
} else if opts.verbosity() < 3 {
|
||||||
|
opts = opts.WithVerbosity(3)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check whether we have specialized formatting for this node.
|
||||||
|
// This is not necessary, but helpful for producing more readable outputs.
|
||||||
|
if opts.CanFormatDiffSlice(v) {
|
||||||
|
return opts.FormatDiffSlice(v)
|
||||||
|
}
|
||||||
|
|
||||||
|
var parentKind reflect.Kind
|
||||||
|
if v.parent != nil && v.parent.TransformerName == "" {
|
||||||
|
parentKind = v.parent.Type.Kind()
|
||||||
|
}
|
||||||
|
|
||||||
|
// For leaf nodes, format the value based on the reflect.Values alone.
|
||||||
|
// As a special case, treat equal []byte as a leaf nodes.
|
||||||
|
isBytes := v.Type.Kind() == reflect.Slice && v.Type.Elem() == byteType
|
||||||
|
isEqualBytes := isBytes && v.NumDiff+v.NumIgnored+v.NumTransformed == 0
|
||||||
|
if v.MaxDepth == 0 || isEqualBytes {
|
||||||
|
switch opts.DiffMode {
|
||||||
|
case diffUnknown, diffIdentical:
|
||||||
|
// Format Equal.
|
||||||
|
if v.NumDiff == 0 {
|
||||||
|
outx := opts.FormatValue(v.ValueX, parentKind, ptrs)
|
||||||
|
outy := opts.FormatValue(v.ValueY, parentKind, ptrs)
|
||||||
|
if v.NumIgnored > 0 && v.NumSame == 0 {
|
||||||
|
return textEllipsis
|
||||||
|
} else if outx.Len() < outy.Len() {
|
||||||
|
return outx
|
||||||
|
} else {
|
||||||
|
return outy
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Format unequal.
|
||||||
|
assert(opts.DiffMode == diffUnknown)
|
||||||
|
var list textList
|
||||||
|
outx := opts.WithTypeMode(elideType).FormatValue(v.ValueX, parentKind, ptrs)
|
||||||
|
outy := opts.WithTypeMode(elideType).FormatValue(v.ValueY, parentKind, ptrs)
|
||||||
|
for i := 0; i <= maxVerbosityPreset && outx != nil && outy != nil && outx.Equal(outy); i++ {
|
||||||
|
opts2 := verbosityPreset(opts, i).WithTypeMode(elideType)
|
||||||
|
outx = opts2.FormatValue(v.ValueX, parentKind, ptrs)
|
||||||
|
outy = opts2.FormatValue(v.ValueY, parentKind, ptrs)
|
||||||
|
}
|
||||||
|
if outx != nil {
|
||||||
|
list = append(list, textRecord{Diff: '-', Value: outx})
|
||||||
|
}
|
||||||
|
if outy != nil {
|
||||||
|
list = append(list, textRecord{Diff: '+', Value: outy})
|
||||||
|
}
|
||||||
|
return opts.WithTypeMode(emitType).FormatType(v.Type, list)
|
||||||
|
case diffRemoved:
|
||||||
|
return opts.FormatValue(v.ValueX, parentKind, ptrs)
|
||||||
|
case diffInserted:
|
||||||
|
return opts.FormatValue(v.ValueY, parentKind, ptrs)
|
||||||
|
default:
|
||||||
|
panic("invalid diff mode")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Register slice element to support cycle detection.
|
||||||
|
if parentKind == reflect.Slice {
|
||||||
|
ptrRefs := ptrs.PushPair(v.ValueX, v.ValueY, opts.DiffMode, true)
|
||||||
|
defer ptrs.Pop()
|
||||||
|
defer func() { out = wrapTrunkReferences(ptrRefs, out) }()
|
||||||
|
}
|
||||||
|
|
||||||
|
// Descend into the child value node.
|
||||||
|
if v.TransformerName != "" {
|
||||||
|
out := opts.WithTypeMode(emitType).FormatDiff(v.Value, ptrs)
|
||||||
|
out = &textWrap{Prefix: "Inverse(" + v.TransformerName + ", ", Value: out, Suffix: ")"}
|
||||||
|
return opts.FormatType(v.Type, out)
|
||||||
|
} else {
|
||||||
|
switch k := v.Type.Kind(); k {
|
||||||
|
case reflect.Struct, reflect.Array, reflect.Slice:
|
||||||
|
out = opts.formatDiffList(v.Records, k, ptrs)
|
||||||
|
out = opts.FormatType(v.Type, out)
|
||||||
|
case reflect.Map:
|
||||||
|
// Register map to support cycle detection.
|
||||||
|
ptrRefs := ptrs.PushPair(v.ValueX, v.ValueY, opts.DiffMode, false)
|
||||||
|
defer ptrs.Pop()
|
||||||
|
|
||||||
|
out = opts.formatDiffList(v.Records, k, ptrs)
|
||||||
|
out = wrapTrunkReferences(ptrRefs, out)
|
||||||
|
out = opts.FormatType(v.Type, out)
|
||||||
|
case reflect.Ptr:
|
||||||
|
// Register pointer to support cycle detection.
|
||||||
|
ptrRefs := ptrs.PushPair(v.ValueX, v.ValueY, opts.DiffMode, false)
|
||||||
|
defer ptrs.Pop()
|
||||||
|
|
||||||
|
out = opts.FormatDiff(v.Value, ptrs)
|
||||||
|
out = wrapTrunkReferences(ptrRefs, out)
|
||||||
|
out = &textWrap{Prefix: "&", Value: out}
|
||||||
|
case reflect.Interface:
|
||||||
|
out = opts.WithTypeMode(emitType).FormatDiff(v.Value, ptrs)
|
||||||
|
default:
|
||||||
|
panic(fmt.Sprintf("%v cannot have children", k))
|
||||||
|
}
|
||||||
|
return out
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (opts formatOptions) formatDiffList(recs []reportRecord, k reflect.Kind, ptrs *pointerReferences) textNode {
|
||||||
|
// Derive record name based on the data structure kind.
|
||||||
|
var name string
|
||||||
|
var formatKey func(reflect.Value) string
|
||||||
|
switch k {
|
||||||
|
case reflect.Struct:
|
||||||
|
name = "field"
|
||||||
|
opts = opts.WithTypeMode(autoType)
|
||||||
|
formatKey = func(v reflect.Value) string { return v.String() }
|
||||||
|
case reflect.Slice, reflect.Array:
|
||||||
|
name = "element"
|
||||||
|
opts = opts.WithTypeMode(elideType)
|
||||||
|
formatKey = func(reflect.Value) string { return "" }
|
||||||
|
case reflect.Map:
|
||||||
|
name = "entry"
|
||||||
|
opts = opts.WithTypeMode(elideType)
|
||||||
|
formatKey = func(v reflect.Value) string { return formatMapKey(v, false, ptrs) }
|
||||||
|
}
|
||||||
|
|
||||||
|
maxLen := -1
|
||||||
|
if opts.LimitVerbosity {
|
||||||
|
if opts.DiffMode == diffIdentical {
|
||||||
|
maxLen = ((1 << opts.verbosity()) >> 1) << 2 // 0, 4, 8, 16, 32, etc...
|
||||||
|
} else {
|
||||||
|
maxLen = (1 << opts.verbosity()) << 1 // 2, 4, 8, 16, 32, 64, etc...
|
||||||
|
}
|
||||||
|
opts.VerbosityLevel--
|
||||||
|
}
|
||||||
|
|
||||||
|
// Handle unification.
|
||||||
|
switch opts.DiffMode {
|
||||||
|
case diffIdentical, diffRemoved, diffInserted:
|
||||||
|
var list textList
|
||||||
|
var deferredEllipsis bool // Add final "..." to indicate records were dropped
|
||||||
|
for _, r := range recs {
|
||||||
|
if len(list) == maxLen {
|
||||||
|
deferredEllipsis = true
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
// Elide struct fields that are zero value.
|
||||||
|
if k == reflect.Struct {
|
||||||
|
var isZero bool
|
||||||
|
switch opts.DiffMode {
|
||||||
|
case diffIdentical:
|
||||||
|
isZero = r.Value.ValueX.IsZero() || r.Value.ValueY.IsZero()
|
||||||
|
case diffRemoved:
|
||||||
|
isZero = r.Value.ValueX.IsZero()
|
||||||
|
case diffInserted:
|
||||||
|
isZero = r.Value.ValueY.IsZero()
|
||||||
|
}
|
||||||
|
if isZero {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Elide ignored nodes.
|
||||||
|
if r.Value.NumIgnored > 0 && r.Value.NumSame+r.Value.NumDiff == 0 {
|
||||||
|
deferredEllipsis = !(k == reflect.Slice || k == reflect.Array)
|
||||||
|
if !deferredEllipsis {
|
||||||
|
list.AppendEllipsis(diffStats{})
|
||||||
|
}
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if out := opts.FormatDiff(r.Value, ptrs); out != nil {
|
||||||
|
list = append(list, textRecord{Key: formatKey(r.Key), Value: out})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if deferredEllipsis {
|
||||||
|
list.AppendEllipsis(diffStats{})
|
||||||
|
}
|
||||||
|
return &textWrap{Prefix: "{", Value: list, Suffix: "}"}
|
||||||
|
case diffUnknown:
|
||||||
|
default:
|
||||||
|
panic("invalid diff mode")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Handle differencing.
|
||||||
|
var numDiffs int
|
||||||
|
var list textList
|
||||||
|
var keys []reflect.Value // invariant: len(list) == len(keys)
|
||||||
|
groups := coalesceAdjacentRecords(name, recs)
|
||||||
|
maxGroup := diffStats{Name: name}
|
||||||
|
for i, ds := range groups {
|
||||||
|
if maxLen >= 0 && numDiffs >= maxLen {
|
||||||
|
maxGroup = maxGroup.Append(ds)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
// Handle equal records.
|
||||||
|
if ds.NumDiff() == 0 {
|
||||||
|
// Compute the number of leading and trailing records to print.
|
||||||
|
var numLo, numHi int
|
||||||
|
numEqual := ds.NumIgnored + ds.NumIdentical
|
||||||
|
for numLo < numContextRecords && numLo+numHi < numEqual && i != 0 {
|
||||||
|
if r := recs[numLo].Value; r.NumIgnored > 0 && r.NumSame+r.NumDiff == 0 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
numLo++
|
||||||
|
}
|
||||||
|
for numHi < numContextRecords && numLo+numHi < numEqual && i != len(groups)-1 {
|
||||||
|
if r := recs[numEqual-numHi-1].Value; r.NumIgnored > 0 && r.NumSame+r.NumDiff == 0 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
numHi++
|
||||||
|
}
|
||||||
|
if numEqual-(numLo+numHi) == 1 && ds.NumIgnored == 0 {
|
||||||
|
numHi++ // Avoid pointless coalescing of a single equal record
|
||||||
|
}
|
||||||
|
|
||||||
|
// Format the equal values.
|
||||||
|
for _, r := range recs[:numLo] {
|
||||||
|
out := opts.WithDiffMode(diffIdentical).FormatDiff(r.Value, ptrs)
|
||||||
|
list = append(list, textRecord{Key: formatKey(r.Key), Value: out})
|
||||||
|
keys = append(keys, r.Key)
|
||||||
|
}
|
||||||
|
if numEqual > numLo+numHi {
|
||||||
|
ds.NumIdentical -= numLo + numHi
|
||||||
|
list.AppendEllipsis(ds)
|
||||||
|
for len(keys) < len(list) {
|
||||||
|
keys = append(keys, reflect.Value{})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for _, r := range recs[numEqual-numHi : numEqual] {
|
||||||
|
out := opts.WithDiffMode(diffIdentical).FormatDiff(r.Value, ptrs)
|
||||||
|
list = append(list, textRecord{Key: formatKey(r.Key), Value: out})
|
||||||
|
keys = append(keys, r.Key)
|
||||||
|
}
|
||||||
|
recs = recs[numEqual:]
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
// Handle unequal records.
|
||||||
|
for _, r := range recs[:ds.NumDiff()] {
|
||||||
|
switch {
|
||||||
|
case opts.CanFormatDiffSlice(r.Value):
|
||||||
|
out := opts.FormatDiffSlice(r.Value)
|
||||||
|
list = append(list, textRecord{Key: formatKey(r.Key), Value: out})
|
||||||
|
keys = append(keys, r.Key)
|
||||||
|
case r.Value.NumChildren == r.Value.MaxDepth:
|
||||||
|
outx := opts.WithDiffMode(diffRemoved).FormatDiff(r.Value, ptrs)
|
||||||
|
outy := opts.WithDiffMode(diffInserted).FormatDiff(r.Value, ptrs)
|
||||||
|
for i := 0; i <= maxVerbosityPreset && outx != nil && outy != nil && outx.Equal(outy); i++ {
|
||||||
|
opts2 := verbosityPreset(opts, i)
|
||||||
|
outx = opts2.WithDiffMode(diffRemoved).FormatDiff(r.Value, ptrs)
|
||||||
|
outy = opts2.WithDiffMode(diffInserted).FormatDiff(r.Value, ptrs)
|
||||||
|
}
|
||||||
|
if outx != nil {
|
||||||
|
list = append(list, textRecord{Diff: diffRemoved, Key: formatKey(r.Key), Value: outx})
|
||||||
|
keys = append(keys, r.Key)
|
||||||
|
}
|
||||||
|
if outy != nil {
|
||||||
|
list = append(list, textRecord{Diff: diffInserted, Key: formatKey(r.Key), Value: outy})
|
||||||
|
keys = append(keys, r.Key)
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
out := opts.FormatDiff(r.Value, ptrs)
|
||||||
|
list = append(list, textRecord{Key: formatKey(r.Key), Value: out})
|
||||||
|
keys = append(keys, r.Key)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
recs = recs[ds.NumDiff():]
|
||||||
|
numDiffs += ds.NumDiff()
|
||||||
|
}
|
||||||
|
if maxGroup.IsZero() {
|
||||||
|
assert(len(recs) == 0)
|
||||||
|
} else {
|
||||||
|
list.AppendEllipsis(maxGroup)
|
||||||
|
for len(keys) < len(list) {
|
||||||
|
keys = append(keys, reflect.Value{})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
assert(len(list) == len(keys))
|
||||||
|
|
||||||
|
// For maps, the default formatting logic uses fmt.Stringer which may
|
||||||
|
// produce ambiguous output. Avoid calling String to disambiguate.
|
||||||
|
if k == reflect.Map {
|
||||||
|
var ambiguous bool
|
||||||
|
seenKeys := map[string]reflect.Value{}
|
||||||
|
for i, currKey := range keys {
|
||||||
|
if currKey.IsValid() {
|
||||||
|
strKey := list[i].Key
|
||||||
|
prevKey, seen := seenKeys[strKey]
|
||||||
|
if seen && prevKey.CanInterface() && currKey.CanInterface() {
|
||||||
|
ambiguous = prevKey.Interface() != currKey.Interface()
|
||||||
|
if ambiguous {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
seenKeys[strKey] = currKey
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if ambiguous {
|
||||||
|
for i, k := range keys {
|
||||||
|
if k.IsValid() {
|
||||||
|
list[i].Key = formatMapKey(k, true, ptrs)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return &textWrap{Prefix: "{", Value: list, Suffix: "}"}
|
||||||
|
}
|
||||||
|
|
||||||
|
// coalesceAdjacentRecords coalesces the list of records into groups of
|
||||||
|
// adjacent equal, or unequal counts.
|
||||||
|
func coalesceAdjacentRecords(name string, recs []reportRecord) (groups []diffStats) {
|
||||||
|
var prevCase int // Arbitrary index into which case last occurred
|
||||||
|
lastStats := func(i int) *diffStats {
|
||||||
|
if prevCase != i {
|
||||||
|
groups = append(groups, diffStats{Name: name})
|
||||||
|
prevCase = i
|
||||||
|
}
|
||||||
|
return &groups[len(groups)-1]
|
||||||
|
}
|
||||||
|
for _, r := range recs {
|
||||||
|
switch rv := r.Value; {
|
||||||
|
case rv.NumIgnored > 0 && rv.NumSame+rv.NumDiff == 0:
|
||||||
|
lastStats(1).NumIgnored++
|
||||||
|
case rv.NumDiff == 0:
|
||||||
|
lastStats(1).NumIdentical++
|
||||||
|
case rv.NumDiff > 0 && !rv.ValueY.IsValid():
|
||||||
|
lastStats(2).NumRemoved++
|
||||||
|
case rv.NumDiff > 0 && !rv.ValueX.IsValid():
|
||||||
|
lastStats(2).NumInserted++
|
||||||
|
default:
|
||||||
|
lastStats(2).NumModified++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return groups
|
||||||
|
}
|
||||||
264
vendor/github.com/google/go-cmp/cmp/report_references.go
generated
vendored
Normal file
264
vendor/github.com/google/go-cmp/cmp/report_references.go
generated
vendored
Normal file
@@ -0,0 +1,264 @@
|
|||||||
|
// Copyright 2020, The Go Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a BSD-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
package cmp
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"reflect"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/google/go-cmp/cmp/internal/flags"
|
||||||
|
"github.com/google/go-cmp/cmp/internal/value"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
pointerDelimPrefix = "⟪"
|
||||||
|
pointerDelimSuffix = "⟫"
|
||||||
|
)
|
||||||
|
|
||||||
|
// formatPointer prints the address of the pointer.
|
||||||
|
func formatPointer(p value.Pointer, withDelims bool) string {
|
||||||
|
v := p.Uintptr()
|
||||||
|
if flags.Deterministic {
|
||||||
|
v = 0xdeadf00f // Only used for stable testing purposes
|
||||||
|
}
|
||||||
|
if withDelims {
|
||||||
|
return pointerDelimPrefix + formatHex(uint64(v)) + pointerDelimSuffix
|
||||||
|
}
|
||||||
|
return formatHex(uint64(v))
|
||||||
|
}
|
||||||
|
|
||||||
|
// pointerReferences is a stack of pointers visited so far.
|
||||||
|
type pointerReferences [][2]value.Pointer
|
||||||
|
|
||||||
|
func (ps *pointerReferences) PushPair(vx, vy reflect.Value, d diffMode, deref bool) (pp [2]value.Pointer) {
|
||||||
|
if deref && vx.IsValid() {
|
||||||
|
vx = vx.Addr()
|
||||||
|
}
|
||||||
|
if deref && vy.IsValid() {
|
||||||
|
vy = vy.Addr()
|
||||||
|
}
|
||||||
|
switch d {
|
||||||
|
case diffUnknown, diffIdentical:
|
||||||
|
pp = [2]value.Pointer{value.PointerOf(vx), value.PointerOf(vy)}
|
||||||
|
case diffRemoved:
|
||||||
|
pp = [2]value.Pointer{value.PointerOf(vx), value.Pointer{}}
|
||||||
|
case diffInserted:
|
||||||
|
pp = [2]value.Pointer{value.Pointer{}, value.PointerOf(vy)}
|
||||||
|
}
|
||||||
|
*ps = append(*ps, pp)
|
||||||
|
return pp
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ps *pointerReferences) Push(v reflect.Value) (p value.Pointer, seen bool) {
|
||||||
|
p = value.PointerOf(v)
|
||||||
|
for _, pp := range *ps {
|
||||||
|
if p == pp[0] || p == pp[1] {
|
||||||
|
return p, true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
*ps = append(*ps, [2]value.Pointer{p, p})
|
||||||
|
return p, false
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ps *pointerReferences) Pop() {
|
||||||
|
*ps = (*ps)[:len(*ps)-1]
|
||||||
|
}
|
||||||
|
|
||||||
|
// trunkReferences is metadata for a textNode indicating that the sub-tree
|
||||||
|
// represents the value for either pointer in a pair of references.
|
||||||
|
type trunkReferences struct{ pp [2]value.Pointer }
|
||||||
|
|
||||||
|
// trunkReference is metadata for a textNode indicating that the sub-tree
|
||||||
|
// represents the value for the given pointer reference.
|
||||||
|
type trunkReference struct{ p value.Pointer }
|
||||||
|
|
||||||
|
// leafReference is metadata for a textNode indicating that the value is
|
||||||
|
// truncated as it refers to another part of the tree (i.e., a trunk).
|
||||||
|
type leafReference struct{ p value.Pointer }
|
||||||
|
|
||||||
|
func wrapTrunkReferences(pp [2]value.Pointer, s textNode) textNode {
|
||||||
|
switch {
|
||||||
|
case pp[0].IsNil():
|
||||||
|
return &textWrap{Value: s, Metadata: trunkReference{pp[1]}}
|
||||||
|
case pp[1].IsNil():
|
||||||
|
return &textWrap{Value: s, Metadata: trunkReference{pp[0]}}
|
||||||
|
case pp[0] == pp[1]:
|
||||||
|
return &textWrap{Value: s, Metadata: trunkReference{pp[0]}}
|
||||||
|
default:
|
||||||
|
return &textWrap{Value: s, Metadata: trunkReferences{pp}}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
func wrapTrunkReference(p value.Pointer, printAddress bool, s textNode) textNode {
|
||||||
|
var prefix string
|
||||||
|
if printAddress {
|
||||||
|
prefix = formatPointer(p, true)
|
||||||
|
}
|
||||||
|
return &textWrap{Prefix: prefix, Value: s, Metadata: trunkReference{p}}
|
||||||
|
}
|
||||||
|
func makeLeafReference(p value.Pointer, printAddress bool) textNode {
|
||||||
|
out := &textWrap{Prefix: "(", Value: textEllipsis, Suffix: ")"}
|
||||||
|
var prefix string
|
||||||
|
if printAddress {
|
||||||
|
prefix = formatPointer(p, true)
|
||||||
|
}
|
||||||
|
return &textWrap{Prefix: prefix, Value: out, Metadata: leafReference{p}}
|
||||||
|
}
|
||||||
|
|
||||||
|
// resolveReferences walks the textNode tree searching for any leaf reference
|
||||||
|
// metadata and resolves each against the corresponding trunk references.
|
||||||
|
// Since pointer addresses in memory are not particularly readable to the user,
|
||||||
|
// it replaces each pointer value with an arbitrary and unique reference ID.
|
||||||
|
func resolveReferences(s textNode) {
|
||||||
|
var walkNodes func(textNode, func(textNode))
|
||||||
|
walkNodes = func(s textNode, f func(textNode)) {
|
||||||
|
f(s)
|
||||||
|
switch s := s.(type) {
|
||||||
|
case *textWrap:
|
||||||
|
walkNodes(s.Value, f)
|
||||||
|
case textList:
|
||||||
|
for _, r := range s {
|
||||||
|
walkNodes(r.Value, f)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Collect all trunks and leaves with reference metadata.
|
||||||
|
var trunks, leaves []*textWrap
|
||||||
|
walkNodes(s, func(s textNode) {
|
||||||
|
if s, ok := s.(*textWrap); ok {
|
||||||
|
switch s.Metadata.(type) {
|
||||||
|
case leafReference:
|
||||||
|
leaves = append(leaves, s)
|
||||||
|
case trunkReference, trunkReferences:
|
||||||
|
trunks = append(trunks, s)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
// No leaf references to resolve.
|
||||||
|
if len(leaves) == 0 {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Collect the set of all leaf references to resolve.
|
||||||
|
leafPtrs := make(map[value.Pointer]bool)
|
||||||
|
for _, leaf := range leaves {
|
||||||
|
leafPtrs[leaf.Metadata.(leafReference).p] = true
|
||||||
|
}
|
||||||
|
|
||||||
|
// Collect the set of trunk pointers that are always paired together.
|
||||||
|
// This allows us to assign a single ID to both pointers for brevity.
|
||||||
|
// If a pointer in a pair ever occurs by itself or as a different pair,
|
||||||
|
// then the pair is broken.
|
||||||
|
pairedTrunkPtrs := make(map[value.Pointer]value.Pointer)
|
||||||
|
unpair := func(p value.Pointer) {
|
||||||
|
if !pairedTrunkPtrs[p].IsNil() {
|
||||||
|
pairedTrunkPtrs[pairedTrunkPtrs[p]] = value.Pointer{} // invalidate other half
|
||||||
|
}
|
||||||
|
pairedTrunkPtrs[p] = value.Pointer{} // invalidate this half
|
||||||
|
}
|
||||||
|
for _, trunk := range trunks {
|
||||||
|
switch p := trunk.Metadata.(type) {
|
||||||
|
case trunkReference:
|
||||||
|
unpair(p.p) // standalone pointer cannot be part of a pair
|
||||||
|
case trunkReferences:
|
||||||
|
p0, ok0 := pairedTrunkPtrs[p.pp[0]]
|
||||||
|
p1, ok1 := pairedTrunkPtrs[p.pp[1]]
|
||||||
|
switch {
|
||||||
|
case !ok0 && !ok1:
|
||||||
|
// Register the newly seen pair.
|
||||||
|
pairedTrunkPtrs[p.pp[0]] = p.pp[1]
|
||||||
|
pairedTrunkPtrs[p.pp[1]] = p.pp[0]
|
||||||
|
case ok0 && ok1 && p0 == p.pp[1] && p1 == p.pp[0]:
|
||||||
|
// Exact pair already seen; do nothing.
|
||||||
|
default:
|
||||||
|
// Pair conflicts with some other pair; break all pairs.
|
||||||
|
unpair(p.pp[0])
|
||||||
|
unpair(p.pp[1])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Correlate each pointer referenced by leaves to a unique identifier,
|
||||||
|
// and print the IDs for each trunk that matches those pointers.
|
||||||
|
var nextID uint
|
||||||
|
ptrIDs := make(map[value.Pointer]uint)
|
||||||
|
newID := func() uint {
|
||||||
|
id := nextID
|
||||||
|
nextID++
|
||||||
|
return id
|
||||||
|
}
|
||||||
|
for _, trunk := range trunks {
|
||||||
|
switch p := trunk.Metadata.(type) {
|
||||||
|
case trunkReference:
|
||||||
|
if print := leafPtrs[p.p]; print {
|
||||||
|
id, ok := ptrIDs[p.p]
|
||||||
|
if !ok {
|
||||||
|
id = newID()
|
||||||
|
ptrIDs[p.p] = id
|
||||||
|
}
|
||||||
|
trunk.Prefix = updateReferencePrefix(trunk.Prefix, formatReference(id))
|
||||||
|
}
|
||||||
|
case trunkReferences:
|
||||||
|
print0 := leafPtrs[p.pp[0]]
|
||||||
|
print1 := leafPtrs[p.pp[1]]
|
||||||
|
if print0 || print1 {
|
||||||
|
id0, ok0 := ptrIDs[p.pp[0]]
|
||||||
|
id1, ok1 := ptrIDs[p.pp[1]]
|
||||||
|
isPair := pairedTrunkPtrs[p.pp[0]] == p.pp[1] && pairedTrunkPtrs[p.pp[1]] == p.pp[0]
|
||||||
|
if isPair {
|
||||||
|
var id uint
|
||||||
|
assert(ok0 == ok1) // must be seen together or not at all
|
||||||
|
if ok0 {
|
||||||
|
assert(id0 == id1) // must have the same ID
|
||||||
|
id = id0
|
||||||
|
} else {
|
||||||
|
id = newID()
|
||||||
|
ptrIDs[p.pp[0]] = id
|
||||||
|
ptrIDs[p.pp[1]] = id
|
||||||
|
}
|
||||||
|
trunk.Prefix = updateReferencePrefix(trunk.Prefix, formatReference(id))
|
||||||
|
} else {
|
||||||
|
if print0 && !ok0 {
|
||||||
|
id0 = newID()
|
||||||
|
ptrIDs[p.pp[0]] = id0
|
||||||
|
}
|
||||||
|
if print1 && !ok1 {
|
||||||
|
id1 = newID()
|
||||||
|
ptrIDs[p.pp[1]] = id1
|
||||||
|
}
|
||||||
|
switch {
|
||||||
|
case print0 && print1:
|
||||||
|
trunk.Prefix = updateReferencePrefix(trunk.Prefix, formatReference(id0)+","+formatReference(id1))
|
||||||
|
case print0:
|
||||||
|
trunk.Prefix = updateReferencePrefix(trunk.Prefix, formatReference(id0))
|
||||||
|
case print1:
|
||||||
|
trunk.Prefix = updateReferencePrefix(trunk.Prefix, formatReference(id1))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Update all leaf references with the unique identifier.
|
||||||
|
for _, leaf := range leaves {
|
||||||
|
if id, ok := ptrIDs[leaf.Metadata.(leafReference).p]; ok {
|
||||||
|
leaf.Prefix = updateReferencePrefix(leaf.Prefix, formatReference(id))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func formatReference(id uint) string {
|
||||||
|
return fmt.Sprintf("ref#%d", id)
|
||||||
|
}
|
||||||
|
|
||||||
|
func updateReferencePrefix(prefix, ref string) string {
|
||||||
|
if prefix == "" {
|
||||||
|
return pointerDelimPrefix + ref + pointerDelimSuffix
|
||||||
|
}
|
||||||
|
suffix := strings.TrimPrefix(prefix, pointerDelimPrefix)
|
||||||
|
return pointerDelimPrefix + ref + ": " + suffix
|
||||||
|
}
|
||||||
414
vendor/github.com/google/go-cmp/cmp/report_reflect.go
generated
vendored
Normal file
414
vendor/github.com/google/go-cmp/cmp/report_reflect.go
generated
vendored
Normal file
@@ -0,0 +1,414 @@
|
|||||||
|
// Copyright 2019, The Go Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a BSD-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
package cmp
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"fmt"
|
||||||
|
"reflect"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
"unicode"
|
||||||
|
"unicode/utf8"
|
||||||
|
|
||||||
|
"github.com/google/go-cmp/cmp/internal/value"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
anyType = reflect.TypeOf((*interface{})(nil)).Elem()
|
||||||
|
stringType = reflect.TypeOf((*string)(nil)).Elem()
|
||||||
|
bytesType = reflect.TypeOf((*[]byte)(nil)).Elem()
|
||||||
|
byteType = reflect.TypeOf((*byte)(nil)).Elem()
|
||||||
|
)
|
||||||
|
|
||||||
|
type formatValueOptions struct {
|
||||||
|
// AvoidStringer controls whether to avoid calling custom stringer
|
||||||
|
// methods like error.Error or fmt.Stringer.String.
|
||||||
|
AvoidStringer bool
|
||||||
|
|
||||||
|
// PrintAddresses controls whether to print the address of all pointers,
|
||||||
|
// slice elements, and maps.
|
||||||
|
PrintAddresses bool
|
||||||
|
|
||||||
|
// QualifiedNames controls whether FormatType uses the fully qualified name
|
||||||
|
// (including the full package path as opposed to just the package name).
|
||||||
|
QualifiedNames bool
|
||||||
|
|
||||||
|
// VerbosityLevel controls the amount of output to produce.
|
||||||
|
// A higher value produces more output. A value of zero or lower produces
|
||||||
|
// no output (represented using an ellipsis).
|
||||||
|
// If LimitVerbosity is false, then the level is treated as infinite.
|
||||||
|
VerbosityLevel int
|
||||||
|
|
||||||
|
// LimitVerbosity specifies that formatting should respect VerbosityLevel.
|
||||||
|
LimitVerbosity bool
|
||||||
|
}
|
||||||
|
|
||||||
|
// FormatType prints the type as if it were wrapping s.
|
||||||
|
// This may return s as-is depending on the current type and TypeMode mode.
|
||||||
|
func (opts formatOptions) FormatType(t reflect.Type, s textNode) textNode {
|
||||||
|
// Check whether to emit the type or not.
|
||||||
|
switch opts.TypeMode {
|
||||||
|
case autoType:
|
||||||
|
switch t.Kind() {
|
||||||
|
case reflect.Struct, reflect.Slice, reflect.Array, reflect.Map:
|
||||||
|
if s.Equal(textNil) {
|
||||||
|
return s
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
return s
|
||||||
|
}
|
||||||
|
if opts.DiffMode == diffIdentical {
|
||||||
|
return s // elide type for identical nodes
|
||||||
|
}
|
||||||
|
case elideType:
|
||||||
|
return s
|
||||||
|
}
|
||||||
|
|
||||||
|
// Determine the type label, applying special handling for unnamed types.
|
||||||
|
typeName := value.TypeString(t, opts.QualifiedNames)
|
||||||
|
if t.Name() == "" {
|
||||||
|
// According to Go grammar, certain type literals contain symbols that
|
||||||
|
// do not strongly bind to the next lexicographical token (e.g., *T).
|
||||||
|
switch t.Kind() {
|
||||||
|
case reflect.Chan, reflect.Func, reflect.Ptr:
|
||||||
|
typeName = "(" + typeName + ")"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return &textWrap{Prefix: typeName, Value: wrapParens(s)}
|
||||||
|
}
|
||||||
|
|
||||||
|
// wrapParens wraps s with a set of parenthesis, but avoids it if the
|
||||||
|
// wrapped node itself is already surrounded by a pair of parenthesis or braces.
|
||||||
|
// It handles unwrapping one level of pointer-reference nodes.
|
||||||
|
func wrapParens(s textNode) textNode {
|
||||||
|
var refNode *textWrap
|
||||||
|
if s2, ok := s.(*textWrap); ok {
|
||||||
|
// Unwrap a single pointer reference node.
|
||||||
|
switch s2.Metadata.(type) {
|
||||||
|
case leafReference, trunkReference, trunkReferences:
|
||||||
|
refNode = s2
|
||||||
|
if s3, ok := refNode.Value.(*textWrap); ok {
|
||||||
|
s2 = s3
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Already has delimiters that make parenthesis unnecessary.
|
||||||
|
hasParens := strings.HasPrefix(s2.Prefix, "(") && strings.HasSuffix(s2.Suffix, ")")
|
||||||
|
hasBraces := strings.HasPrefix(s2.Prefix, "{") && strings.HasSuffix(s2.Suffix, "}")
|
||||||
|
if hasParens || hasBraces {
|
||||||
|
return s
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if refNode != nil {
|
||||||
|
refNode.Value = &textWrap{Prefix: "(", Value: refNode.Value, Suffix: ")"}
|
||||||
|
return s
|
||||||
|
}
|
||||||
|
return &textWrap{Prefix: "(", Value: s, Suffix: ")"}
|
||||||
|
}
|
||||||
|
|
||||||
|
// FormatValue prints the reflect.Value, taking extra care to avoid descending
|
||||||
|
// into pointers already in ptrs. As pointers are visited, ptrs is also updated.
|
||||||
|
func (opts formatOptions) FormatValue(v reflect.Value, parentKind reflect.Kind, ptrs *pointerReferences) (out textNode) {
|
||||||
|
if !v.IsValid() {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
t := v.Type()
|
||||||
|
|
||||||
|
// Check slice element for cycles.
|
||||||
|
if parentKind == reflect.Slice {
|
||||||
|
ptrRef, visited := ptrs.Push(v.Addr())
|
||||||
|
if visited {
|
||||||
|
return makeLeafReference(ptrRef, false)
|
||||||
|
}
|
||||||
|
defer ptrs.Pop()
|
||||||
|
defer func() { out = wrapTrunkReference(ptrRef, false, out) }()
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check whether there is an Error or String method to call.
|
||||||
|
if !opts.AvoidStringer && v.CanInterface() {
|
||||||
|
// Avoid calling Error or String methods on nil receivers since many
|
||||||
|
// implementations crash when doing so.
|
||||||
|
if (t.Kind() != reflect.Ptr && t.Kind() != reflect.Interface) || !v.IsNil() {
|
||||||
|
var prefix, strVal string
|
||||||
|
func() {
|
||||||
|
// Swallow and ignore any panics from String or Error.
|
||||||
|
defer func() { recover() }()
|
||||||
|
switch v := v.Interface().(type) {
|
||||||
|
case error:
|
||||||
|
strVal = v.Error()
|
||||||
|
prefix = "e"
|
||||||
|
case fmt.Stringer:
|
||||||
|
strVal = v.String()
|
||||||
|
prefix = "s"
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
if prefix != "" {
|
||||||
|
return opts.formatString(prefix, strVal)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check whether to explicitly wrap the result with the type.
|
||||||
|
var skipType bool
|
||||||
|
defer func() {
|
||||||
|
if !skipType {
|
||||||
|
out = opts.FormatType(t, out)
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
switch t.Kind() {
|
||||||
|
case reflect.Bool:
|
||||||
|
return textLine(fmt.Sprint(v.Bool()))
|
||||||
|
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
|
||||||
|
return textLine(fmt.Sprint(v.Int()))
|
||||||
|
case reflect.Uint, reflect.Uint16, reflect.Uint32, reflect.Uint64:
|
||||||
|
return textLine(fmt.Sprint(v.Uint()))
|
||||||
|
case reflect.Uint8:
|
||||||
|
if parentKind == reflect.Slice || parentKind == reflect.Array {
|
||||||
|
return textLine(formatHex(v.Uint()))
|
||||||
|
}
|
||||||
|
return textLine(fmt.Sprint(v.Uint()))
|
||||||
|
case reflect.Uintptr:
|
||||||
|
return textLine(formatHex(v.Uint()))
|
||||||
|
case reflect.Float32, reflect.Float64:
|
||||||
|
return textLine(fmt.Sprint(v.Float()))
|
||||||
|
case reflect.Complex64, reflect.Complex128:
|
||||||
|
return textLine(fmt.Sprint(v.Complex()))
|
||||||
|
case reflect.String:
|
||||||
|
return opts.formatString("", v.String())
|
||||||
|
case reflect.UnsafePointer, reflect.Chan, reflect.Func:
|
||||||
|
return textLine(formatPointer(value.PointerOf(v), true))
|
||||||
|
case reflect.Struct:
|
||||||
|
var list textList
|
||||||
|
v := makeAddressable(v) // needed for retrieveUnexportedField
|
||||||
|
maxLen := v.NumField()
|
||||||
|
if opts.LimitVerbosity {
|
||||||
|
maxLen = ((1 << opts.verbosity()) >> 1) << 2 // 0, 4, 8, 16, 32, etc...
|
||||||
|
opts.VerbosityLevel--
|
||||||
|
}
|
||||||
|
for i := 0; i < v.NumField(); i++ {
|
||||||
|
vv := v.Field(i)
|
||||||
|
if vv.IsZero() {
|
||||||
|
continue // Elide fields with zero values
|
||||||
|
}
|
||||||
|
if len(list) == maxLen {
|
||||||
|
list.AppendEllipsis(diffStats{})
|
||||||
|
break
|
||||||
|
}
|
||||||
|
sf := t.Field(i)
|
||||||
|
if supportExporters && !isExported(sf.Name) {
|
||||||
|
vv = retrieveUnexportedField(v, sf, true)
|
||||||
|
}
|
||||||
|
s := opts.WithTypeMode(autoType).FormatValue(vv, t.Kind(), ptrs)
|
||||||
|
list = append(list, textRecord{Key: sf.Name, Value: s})
|
||||||
|
}
|
||||||
|
return &textWrap{Prefix: "{", Value: list, Suffix: "}"}
|
||||||
|
case reflect.Slice:
|
||||||
|
if v.IsNil() {
|
||||||
|
return textNil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check whether this is a []byte of text data.
|
||||||
|
if t.Elem() == byteType {
|
||||||
|
b := v.Bytes()
|
||||||
|
isPrintSpace := func(r rune) bool { return unicode.IsPrint(r) || unicode.IsSpace(r) }
|
||||||
|
if len(b) > 0 && utf8.Valid(b) && len(bytes.TrimFunc(b, isPrintSpace)) == 0 {
|
||||||
|
out = opts.formatString("", string(b))
|
||||||
|
skipType = true
|
||||||
|
return opts.FormatType(t, out)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fallthrough
|
||||||
|
case reflect.Array:
|
||||||
|
maxLen := v.Len()
|
||||||
|
if opts.LimitVerbosity {
|
||||||
|
maxLen = ((1 << opts.verbosity()) >> 1) << 2 // 0, 4, 8, 16, 32, etc...
|
||||||
|
opts.VerbosityLevel--
|
||||||
|
}
|
||||||
|
var list textList
|
||||||
|
for i := 0; i < v.Len(); i++ {
|
||||||
|
if len(list) == maxLen {
|
||||||
|
list.AppendEllipsis(diffStats{})
|
||||||
|
break
|
||||||
|
}
|
||||||
|
s := opts.WithTypeMode(elideType).FormatValue(v.Index(i), t.Kind(), ptrs)
|
||||||
|
list = append(list, textRecord{Value: s})
|
||||||
|
}
|
||||||
|
|
||||||
|
out = &textWrap{Prefix: "{", Value: list, Suffix: "}"}
|
||||||
|
if t.Kind() == reflect.Slice && opts.PrintAddresses {
|
||||||
|
header := fmt.Sprintf("ptr:%v, len:%d, cap:%d", formatPointer(value.PointerOf(v), false), v.Len(), v.Cap())
|
||||||
|
out = &textWrap{Prefix: pointerDelimPrefix + header + pointerDelimSuffix, Value: out}
|
||||||
|
}
|
||||||
|
return out
|
||||||
|
case reflect.Map:
|
||||||
|
if v.IsNil() {
|
||||||
|
return textNil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check pointer for cycles.
|
||||||
|
ptrRef, visited := ptrs.Push(v)
|
||||||
|
if visited {
|
||||||
|
return makeLeafReference(ptrRef, opts.PrintAddresses)
|
||||||
|
}
|
||||||
|
defer ptrs.Pop()
|
||||||
|
|
||||||
|
maxLen := v.Len()
|
||||||
|
if opts.LimitVerbosity {
|
||||||
|
maxLen = ((1 << opts.verbosity()) >> 1) << 2 // 0, 4, 8, 16, 32, etc...
|
||||||
|
opts.VerbosityLevel--
|
||||||
|
}
|
||||||
|
var list textList
|
||||||
|
for _, k := range value.SortKeys(v.MapKeys()) {
|
||||||
|
if len(list) == maxLen {
|
||||||
|
list.AppendEllipsis(diffStats{})
|
||||||
|
break
|
||||||
|
}
|
||||||
|
sk := formatMapKey(k, false, ptrs)
|
||||||
|
sv := opts.WithTypeMode(elideType).FormatValue(v.MapIndex(k), t.Kind(), ptrs)
|
||||||
|
list = append(list, textRecord{Key: sk, Value: sv})
|
||||||
|
}
|
||||||
|
|
||||||
|
out = &textWrap{Prefix: "{", Value: list, Suffix: "}"}
|
||||||
|
out = wrapTrunkReference(ptrRef, opts.PrintAddresses, out)
|
||||||
|
return out
|
||||||
|
case reflect.Ptr:
|
||||||
|
if v.IsNil() {
|
||||||
|
return textNil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check pointer for cycles.
|
||||||
|
ptrRef, visited := ptrs.Push(v)
|
||||||
|
if visited {
|
||||||
|
out = makeLeafReference(ptrRef, opts.PrintAddresses)
|
||||||
|
return &textWrap{Prefix: "&", Value: out}
|
||||||
|
}
|
||||||
|
defer ptrs.Pop()
|
||||||
|
|
||||||
|
// Skip the name only if this is an unnamed pointer type.
|
||||||
|
// Otherwise taking the address of a value does not reproduce
|
||||||
|
// the named pointer type.
|
||||||
|
if v.Type().Name() == "" {
|
||||||
|
skipType = true // Let the underlying value print the type instead
|
||||||
|
}
|
||||||
|
out = opts.FormatValue(v.Elem(), t.Kind(), ptrs)
|
||||||
|
out = wrapTrunkReference(ptrRef, opts.PrintAddresses, out)
|
||||||
|
out = &textWrap{Prefix: "&", Value: out}
|
||||||
|
return out
|
||||||
|
case reflect.Interface:
|
||||||
|
if v.IsNil() {
|
||||||
|
return textNil
|
||||||
|
}
|
||||||
|
// Interfaces accept different concrete types,
|
||||||
|
// so configure the underlying value to explicitly print the type.
|
||||||
|
return opts.WithTypeMode(emitType).FormatValue(v.Elem(), t.Kind(), ptrs)
|
||||||
|
default:
|
||||||
|
panic(fmt.Sprintf("%v kind not handled", v.Kind()))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (opts formatOptions) formatString(prefix, s string) textNode {
|
||||||
|
maxLen := len(s)
|
||||||
|
maxLines := strings.Count(s, "\n") + 1
|
||||||
|
if opts.LimitVerbosity {
|
||||||
|
maxLen = (1 << opts.verbosity()) << 5 // 32, 64, 128, 256, etc...
|
||||||
|
maxLines = (1 << opts.verbosity()) << 2 // 4, 8, 16, 32, 64, etc...
|
||||||
|
}
|
||||||
|
|
||||||
|
// For multiline strings, use the triple-quote syntax,
|
||||||
|
// but only use it when printing removed or inserted nodes since
|
||||||
|
// we only want the extra verbosity for those cases.
|
||||||
|
lines := strings.Split(strings.TrimSuffix(s, "\n"), "\n")
|
||||||
|
isTripleQuoted := len(lines) >= 4 && (opts.DiffMode == '-' || opts.DiffMode == '+')
|
||||||
|
for i := 0; i < len(lines) && isTripleQuoted; i++ {
|
||||||
|
lines[i] = strings.TrimPrefix(strings.TrimSuffix(lines[i], "\r"), "\r") // trim leading/trailing carriage returns for legacy Windows endline support
|
||||||
|
isPrintable := func(r rune) bool {
|
||||||
|
return unicode.IsPrint(r) || r == '\t' // specially treat tab as printable
|
||||||
|
}
|
||||||
|
line := lines[i]
|
||||||
|
isTripleQuoted = !strings.HasPrefix(strings.TrimPrefix(line, prefix), `"""`) && !strings.HasPrefix(line, "...") && strings.TrimFunc(line, isPrintable) == "" && len(line) <= maxLen
|
||||||
|
}
|
||||||
|
if isTripleQuoted {
|
||||||
|
var list textList
|
||||||
|
list = append(list, textRecord{Diff: opts.DiffMode, Value: textLine(prefix + `"""`), ElideComma: true})
|
||||||
|
for i, line := range lines {
|
||||||
|
if numElided := len(lines) - i; i == maxLines-1 && numElided > 1 {
|
||||||
|
comment := commentString(fmt.Sprintf("%d elided lines", numElided))
|
||||||
|
list = append(list, textRecord{Diff: opts.DiffMode, Value: textEllipsis, ElideComma: true, Comment: comment})
|
||||||
|
break
|
||||||
|
}
|
||||||
|
list = append(list, textRecord{Diff: opts.DiffMode, Value: textLine(line), ElideComma: true})
|
||||||
|
}
|
||||||
|
list = append(list, textRecord{Diff: opts.DiffMode, Value: textLine(prefix + `"""`), ElideComma: true})
|
||||||
|
return &textWrap{Prefix: "(", Value: list, Suffix: ")"}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Format the string as a single-line quoted string.
|
||||||
|
if len(s) > maxLen+len(textEllipsis) {
|
||||||
|
return textLine(prefix + formatString(s[:maxLen]) + string(textEllipsis))
|
||||||
|
}
|
||||||
|
return textLine(prefix + formatString(s))
|
||||||
|
}
|
||||||
|
|
||||||
|
// formatMapKey formats v as if it were a map key.
|
||||||
|
// The result is guaranteed to be a single line.
|
||||||
|
func formatMapKey(v reflect.Value, disambiguate bool, ptrs *pointerReferences) string {
|
||||||
|
var opts formatOptions
|
||||||
|
opts.DiffMode = diffIdentical
|
||||||
|
opts.TypeMode = elideType
|
||||||
|
opts.PrintAddresses = disambiguate
|
||||||
|
opts.AvoidStringer = disambiguate
|
||||||
|
opts.QualifiedNames = disambiguate
|
||||||
|
opts.VerbosityLevel = maxVerbosityPreset
|
||||||
|
opts.LimitVerbosity = true
|
||||||
|
s := opts.FormatValue(v, reflect.Map, ptrs).String()
|
||||||
|
return strings.TrimSpace(s)
|
||||||
|
}
|
||||||
|
|
||||||
|
// formatString prints s as a double-quoted or backtick-quoted string.
|
||||||
|
func formatString(s string) string {
|
||||||
|
// Use quoted string if it the same length as a raw string literal.
|
||||||
|
// Otherwise, attempt to use the raw string form.
|
||||||
|
qs := strconv.Quote(s)
|
||||||
|
if len(qs) == 1+len(s)+1 {
|
||||||
|
return qs
|
||||||
|
}
|
||||||
|
|
||||||
|
// Disallow newlines to ensure output is a single line.
|
||||||
|
// Only allow printable runes for readability purposes.
|
||||||
|
rawInvalid := func(r rune) bool {
|
||||||
|
return r == '`' || r == '\n' || !(unicode.IsPrint(r) || r == '\t')
|
||||||
|
}
|
||||||
|
if utf8.ValidString(s) && strings.IndexFunc(s, rawInvalid) < 0 {
|
||||||
|
return "`" + s + "`"
|
||||||
|
}
|
||||||
|
return qs
|
||||||
|
}
|
||||||
|
|
||||||
|
// formatHex prints u as a hexadecimal integer in Go notation.
|
||||||
|
func formatHex(u uint64) string {
|
||||||
|
var f string
|
||||||
|
switch {
|
||||||
|
case u <= 0xff:
|
||||||
|
f = "0x%02x"
|
||||||
|
case u <= 0xffff:
|
||||||
|
f = "0x%04x"
|
||||||
|
case u <= 0xffffff:
|
||||||
|
f = "0x%06x"
|
||||||
|
case u <= 0xffffffff:
|
||||||
|
f = "0x%08x"
|
||||||
|
case u <= 0xffffffffff:
|
||||||
|
f = "0x%010x"
|
||||||
|
case u <= 0xffffffffffff:
|
||||||
|
f = "0x%012x"
|
||||||
|
case u <= 0xffffffffffffff:
|
||||||
|
f = "0x%014x"
|
||||||
|
case u <= 0xffffffffffffffff:
|
||||||
|
f = "0x%016x"
|
||||||
|
}
|
||||||
|
return fmt.Sprintf(f, u)
|
||||||
|
}
|
||||||
614
vendor/github.com/google/go-cmp/cmp/report_slices.go
generated
vendored
Normal file
614
vendor/github.com/google/go-cmp/cmp/report_slices.go
generated
vendored
Normal file
@@ -0,0 +1,614 @@
|
|||||||
|
// Copyright 2019, The Go Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a BSD-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
package cmp
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"fmt"
|
||||||
|
"math"
|
||||||
|
"reflect"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
"unicode"
|
||||||
|
"unicode/utf8"
|
||||||
|
|
||||||
|
"github.com/google/go-cmp/cmp/internal/diff"
|
||||||
|
)
|
||||||
|
|
||||||
|
// CanFormatDiffSlice reports whether we support custom formatting for nodes
|
||||||
|
// that are slices of primitive kinds or strings.
|
||||||
|
func (opts formatOptions) CanFormatDiffSlice(v *valueNode) bool {
|
||||||
|
switch {
|
||||||
|
case opts.DiffMode != diffUnknown:
|
||||||
|
return false // Must be formatting in diff mode
|
||||||
|
case v.NumDiff == 0:
|
||||||
|
return false // No differences detected
|
||||||
|
case !v.ValueX.IsValid() || !v.ValueY.IsValid():
|
||||||
|
return false // Both values must be valid
|
||||||
|
case v.NumIgnored > 0:
|
||||||
|
return false // Some ignore option was used
|
||||||
|
case v.NumTransformed > 0:
|
||||||
|
return false // Some transform option was used
|
||||||
|
case v.NumCompared > 1:
|
||||||
|
return false // More than one comparison was used
|
||||||
|
case v.NumCompared == 1 && v.Type.Name() != "":
|
||||||
|
// The need for cmp to check applicability of options on every element
|
||||||
|
// in a slice is a significant performance detriment for large []byte.
|
||||||
|
// The workaround is to specify Comparer(bytes.Equal),
|
||||||
|
// which enables cmp to compare []byte more efficiently.
|
||||||
|
// If they differ, we still want to provide batched diffing.
|
||||||
|
// The logic disallows named types since they tend to have their own
|
||||||
|
// String method, with nicer formatting than what this provides.
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check whether this is an interface with the same concrete types.
|
||||||
|
t := v.Type
|
||||||
|
vx, vy := v.ValueX, v.ValueY
|
||||||
|
if t.Kind() == reflect.Interface && !vx.IsNil() && !vy.IsNil() && vx.Elem().Type() == vy.Elem().Type() {
|
||||||
|
vx, vy = vx.Elem(), vy.Elem()
|
||||||
|
t = vx.Type()
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check whether we provide specialized diffing for this type.
|
||||||
|
switch t.Kind() {
|
||||||
|
case reflect.String:
|
||||||
|
case reflect.Array, reflect.Slice:
|
||||||
|
// Only slices of primitive types have specialized handling.
|
||||||
|
switch t.Elem().Kind() {
|
||||||
|
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64,
|
||||||
|
reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr,
|
||||||
|
reflect.Bool, reflect.Float32, reflect.Float64, reflect.Complex64, reflect.Complex128:
|
||||||
|
default:
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
// Both slice values have to be non-empty.
|
||||||
|
if t.Kind() == reflect.Slice && (vx.Len() == 0 || vy.Len() == 0) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
// If a sufficient number of elements already differ,
|
||||||
|
// use specialized formatting even if length requirement is not met.
|
||||||
|
if v.NumDiff > v.NumSame {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
// Use specialized string diffing for longer slices or strings.
|
||||||
|
const minLength = 32
|
||||||
|
return vx.Len() >= minLength && vy.Len() >= minLength
|
||||||
|
}
|
||||||
|
|
||||||
|
// FormatDiffSlice prints a diff for the slices (or strings) represented by v.
|
||||||
|
// This provides custom-tailored logic to make printing of differences in
|
||||||
|
// textual strings and slices of primitive kinds more readable.
|
||||||
|
func (opts formatOptions) FormatDiffSlice(v *valueNode) textNode {
|
||||||
|
assert(opts.DiffMode == diffUnknown)
|
||||||
|
t, vx, vy := v.Type, v.ValueX, v.ValueY
|
||||||
|
if t.Kind() == reflect.Interface {
|
||||||
|
vx, vy = vx.Elem(), vy.Elem()
|
||||||
|
t = vx.Type()
|
||||||
|
opts = opts.WithTypeMode(emitType)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Auto-detect the type of the data.
|
||||||
|
var sx, sy string
|
||||||
|
var ssx, ssy []string
|
||||||
|
var isString, isMostlyText, isPureLinedText, isBinary bool
|
||||||
|
switch {
|
||||||
|
case t.Kind() == reflect.String:
|
||||||
|
sx, sy = vx.String(), vy.String()
|
||||||
|
isString = true
|
||||||
|
case t.Kind() == reflect.Slice && t.Elem() == byteType:
|
||||||
|
sx, sy = string(vx.Bytes()), string(vy.Bytes())
|
||||||
|
isString = true
|
||||||
|
case t.Kind() == reflect.Array:
|
||||||
|
// Arrays need to be addressable for slice operations to work.
|
||||||
|
vx2, vy2 := reflect.New(t).Elem(), reflect.New(t).Elem()
|
||||||
|
vx2.Set(vx)
|
||||||
|
vy2.Set(vy)
|
||||||
|
vx, vy = vx2, vy2
|
||||||
|
}
|
||||||
|
if isString {
|
||||||
|
var numTotalRunes, numValidRunes, numLines, lastLineIdx, maxLineLen int
|
||||||
|
for i, r := range sx + sy {
|
||||||
|
numTotalRunes++
|
||||||
|
if (unicode.IsPrint(r) || unicode.IsSpace(r)) && r != utf8.RuneError {
|
||||||
|
numValidRunes++
|
||||||
|
}
|
||||||
|
if r == '\n' {
|
||||||
|
if maxLineLen < i-lastLineIdx {
|
||||||
|
maxLineLen = i - lastLineIdx
|
||||||
|
}
|
||||||
|
lastLineIdx = i + 1
|
||||||
|
numLines++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
isPureText := numValidRunes == numTotalRunes
|
||||||
|
isMostlyText = float64(numValidRunes) > math.Floor(0.90*float64(numTotalRunes))
|
||||||
|
isPureLinedText = isPureText && numLines >= 4 && maxLineLen <= 1024
|
||||||
|
isBinary = !isMostlyText
|
||||||
|
|
||||||
|
// Avoid diffing by lines if it produces a significantly more complex
|
||||||
|
// edit script than diffing by bytes.
|
||||||
|
if isPureLinedText {
|
||||||
|
ssx = strings.Split(sx, "\n")
|
||||||
|
ssy = strings.Split(sy, "\n")
|
||||||
|
esLines := diff.Difference(len(ssx), len(ssy), func(ix, iy int) diff.Result {
|
||||||
|
return diff.BoolResult(ssx[ix] == ssy[iy])
|
||||||
|
})
|
||||||
|
esBytes := diff.Difference(len(sx), len(sy), func(ix, iy int) diff.Result {
|
||||||
|
return diff.BoolResult(sx[ix] == sy[iy])
|
||||||
|
})
|
||||||
|
efficiencyLines := float64(esLines.Dist()) / float64(len(esLines))
|
||||||
|
efficiencyBytes := float64(esBytes.Dist()) / float64(len(esBytes))
|
||||||
|
quotedLength := len(strconv.Quote(sx + sy))
|
||||||
|
unquotedLength := len(sx) + len(sy)
|
||||||
|
escapeExpansionRatio := float64(quotedLength) / float64(unquotedLength)
|
||||||
|
isPureLinedText = efficiencyLines < 4*efficiencyBytes || escapeExpansionRatio > 1.1
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Format the string into printable records.
|
||||||
|
var list textList
|
||||||
|
var delim string
|
||||||
|
switch {
|
||||||
|
// If the text appears to be multi-lined text,
|
||||||
|
// then perform differencing across individual lines.
|
||||||
|
case isPureLinedText:
|
||||||
|
list = opts.formatDiffSlice(
|
||||||
|
reflect.ValueOf(ssx), reflect.ValueOf(ssy), 1, "line",
|
||||||
|
func(v reflect.Value, d diffMode) textRecord {
|
||||||
|
s := formatString(v.Index(0).String())
|
||||||
|
return textRecord{Diff: d, Value: textLine(s)}
|
||||||
|
},
|
||||||
|
)
|
||||||
|
delim = "\n"
|
||||||
|
|
||||||
|
// If possible, use a custom triple-quote (""") syntax for printing
|
||||||
|
// differences in a string literal. This format is more readable,
|
||||||
|
// but has edge-cases where differences are visually indistinguishable.
|
||||||
|
// This format is avoided under the following conditions:
|
||||||
|
// - A line starts with `"""`
|
||||||
|
// - A line starts with "..."
|
||||||
|
// - A line contains non-printable characters
|
||||||
|
// - Adjacent different lines differ only by whitespace
|
||||||
|
//
|
||||||
|
// For example:
|
||||||
|
//
|
||||||
|
// """
|
||||||
|
// ... // 3 identical lines
|
||||||
|
// foo
|
||||||
|
// bar
|
||||||
|
// - baz
|
||||||
|
// + BAZ
|
||||||
|
// """
|
||||||
|
isTripleQuoted := true
|
||||||
|
prevRemoveLines := map[string]bool{}
|
||||||
|
prevInsertLines := map[string]bool{}
|
||||||
|
var list2 textList
|
||||||
|
list2 = append(list2, textRecord{Value: textLine(`"""`), ElideComma: true})
|
||||||
|
for _, r := range list {
|
||||||
|
if !r.Value.Equal(textEllipsis) {
|
||||||
|
line, _ := strconv.Unquote(string(r.Value.(textLine)))
|
||||||
|
line = strings.TrimPrefix(strings.TrimSuffix(line, "\r"), "\r") // trim leading/trailing carriage returns for legacy Windows endline support
|
||||||
|
normLine := strings.Map(func(r rune) rune {
|
||||||
|
if unicode.IsSpace(r) {
|
||||||
|
return -1 // drop whitespace to avoid visually indistinguishable output
|
||||||
|
}
|
||||||
|
return r
|
||||||
|
}, line)
|
||||||
|
isPrintable := func(r rune) bool {
|
||||||
|
return unicode.IsPrint(r) || r == '\t' // specially treat tab as printable
|
||||||
|
}
|
||||||
|
isTripleQuoted = !strings.HasPrefix(line, `"""`) && !strings.HasPrefix(line, "...") && strings.TrimFunc(line, isPrintable) == ""
|
||||||
|
switch r.Diff {
|
||||||
|
case diffRemoved:
|
||||||
|
isTripleQuoted = isTripleQuoted && !prevInsertLines[normLine]
|
||||||
|
prevRemoveLines[normLine] = true
|
||||||
|
case diffInserted:
|
||||||
|
isTripleQuoted = isTripleQuoted && !prevRemoveLines[normLine]
|
||||||
|
prevInsertLines[normLine] = true
|
||||||
|
}
|
||||||
|
if !isTripleQuoted {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
r.Value = textLine(line)
|
||||||
|
r.ElideComma = true
|
||||||
|
}
|
||||||
|
if !(r.Diff == diffRemoved || r.Diff == diffInserted) { // start a new non-adjacent difference group
|
||||||
|
prevRemoveLines = map[string]bool{}
|
||||||
|
prevInsertLines = map[string]bool{}
|
||||||
|
}
|
||||||
|
list2 = append(list2, r)
|
||||||
|
}
|
||||||
|
if r := list2[len(list2)-1]; r.Diff == diffIdentical && len(r.Value.(textLine)) == 0 {
|
||||||
|
list2 = list2[:len(list2)-1] // elide single empty line at the end
|
||||||
|
}
|
||||||
|
list2 = append(list2, textRecord{Value: textLine(`"""`), ElideComma: true})
|
||||||
|
if isTripleQuoted {
|
||||||
|
var out textNode = &textWrap{Prefix: "(", Value: list2, Suffix: ")"}
|
||||||
|
switch t.Kind() {
|
||||||
|
case reflect.String:
|
||||||
|
if t != stringType {
|
||||||
|
out = opts.FormatType(t, out)
|
||||||
|
}
|
||||||
|
case reflect.Slice:
|
||||||
|
// Always emit type for slices since the triple-quote syntax
|
||||||
|
// looks like a string (not a slice).
|
||||||
|
opts = opts.WithTypeMode(emitType)
|
||||||
|
out = opts.FormatType(t, out)
|
||||||
|
}
|
||||||
|
return out
|
||||||
|
}
|
||||||
|
|
||||||
|
// If the text appears to be single-lined text,
|
||||||
|
// then perform differencing in approximately fixed-sized chunks.
|
||||||
|
// The output is printed as quoted strings.
|
||||||
|
case isMostlyText:
|
||||||
|
list = opts.formatDiffSlice(
|
||||||
|
reflect.ValueOf(sx), reflect.ValueOf(sy), 64, "byte",
|
||||||
|
func(v reflect.Value, d diffMode) textRecord {
|
||||||
|
s := formatString(v.String())
|
||||||
|
return textRecord{Diff: d, Value: textLine(s)}
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
// If the text appears to be binary data,
|
||||||
|
// then perform differencing in approximately fixed-sized chunks.
|
||||||
|
// The output is inspired by hexdump.
|
||||||
|
case isBinary:
|
||||||
|
list = opts.formatDiffSlice(
|
||||||
|
reflect.ValueOf(sx), reflect.ValueOf(sy), 16, "byte",
|
||||||
|
func(v reflect.Value, d diffMode) textRecord {
|
||||||
|
var ss []string
|
||||||
|
for i := 0; i < v.Len(); i++ {
|
||||||
|
ss = append(ss, formatHex(v.Index(i).Uint()))
|
||||||
|
}
|
||||||
|
s := strings.Join(ss, ", ")
|
||||||
|
comment := commentString(fmt.Sprintf("%c|%v|", d, formatASCII(v.String())))
|
||||||
|
return textRecord{Diff: d, Value: textLine(s), Comment: comment}
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
// For all other slices of primitive types,
|
||||||
|
// then perform differencing in approximately fixed-sized chunks.
|
||||||
|
// The size of each chunk depends on the width of the element kind.
|
||||||
|
default:
|
||||||
|
var chunkSize int
|
||||||
|
if t.Elem().Kind() == reflect.Bool {
|
||||||
|
chunkSize = 16
|
||||||
|
} else {
|
||||||
|
switch t.Elem().Bits() {
|
||||||
|
case 8:
|
||||||
|
chunkSize = 16
|
||||||
|
case 16:
|
||||||
|
chunkSize = 12
|
||||||
|
case 32:
|
||||||
|
chunkSize = 8
|
||||||
|
default:
|
||||||
|
chunkSize = 8
|
||||||
|
}
|
||||||
|
}
|
||||||
|
list = opts.formatDiffSlice(
|
||||||
|
vx, vy, chunkSize, t.Elem().Kind().String(),
|
||||||
|
func(v reflect.Value, d diffMode) textRecord {
|
||||||
|
var ss []string
|
||||||
|
for i := 0; i < v.Len(); i++ {
|
||||||
|
switch t.Elem().Kind() {
|
||||||
|
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
|
||||||
|
ss = append(ss, fmt.Sprint(v.Index(i).Int()))
|
||||||
|
case reflect.Uint, reflect.Uint16, reflect.Uint32, reflect.Uint64:
|
||||||
|
ss = append(ss, fmt.Sprint(v.Index(i).Uint()))
|
||||||
|
case reflect.Uint8, reflect.Uintptr:
|
||||||
|
ss = append(ss, formatHex(v.Index(i).Uint()))
|
||||||
|
case reflect.Bool, reflect.Float32, reflect.Float64, reflect.Complex64, reflect.Complex128:
|
||||||
|
ss = append(ss, fmt.Sprint(v.Index(i).Interface()))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
s := strings.Join(ss, ", ")
|
||||||
|
return textRecord{Diff: d, Value: textLine(s)}
|
||||||
|
},
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Wrap the output with appropriate type information.
|
||||||
|
var out textNode = &textWrap{Prefix: "{", Value: list, Suffix: "}"}
|
||||||
|
if !isMostlyText {
|
||||||
|
// The "{...}" byte-sequence literal is not valid Go syntax for strings.
|
||||||
|
// Emit the type for extra clarity (e.g. "string{...}").
|
||||||
|
if t.Kind() == reflect.String {
|
||||||
|
opts = opts.WithTypeMode(emitType)
|
||||||
|
}
|
||||||
|
return opts.FormatType(t, out)
|
||||||
|
}
|
||||||
|
switch t.Kind() {
|
||||||
|
case reflect.String:
|
||||||
|
out = &textWrap{Prefix: "strings.Join(", Value: out, Suffix: fmt.Sprintf(", %q)", delim)}
|
||||||
|
if t != stringType {
|
||||||
|
out = opts.FormatType(t, out)
|
||||||
|
}
|
||||||
|
case reflect.Slice:
|
||||||
|
out = &textWrap{Prefix: "bytes.Join(", Value: out, Suffix: fmt.Sprintf(", %q)", delim)}
|
||||||
|
if t != bytesType {
|
||||||
|
out = opts.FormatType(t, out)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return out
|
||||||
|
}
|
||||||
|
|
||||||
|
// formatASCII formats s as an ASCII string.
|
||||||
|
// This is useful for printing binary strings in a semi-legible way.
|
||||||
|
func formatASCII(s string) string {
|
||||||
|
b := bytes.Repeat([]byte{'.'}, len(s))
|
||||||
|
for i := 0; i < len(s); i++ {
|
||||||
|
if ' ' <= s[i] && s[i] <= '~' {
|
||||||
|
b[i] = s[i]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return string(b)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (opts formatOptions) formatDiffSlice(
|
||||||
|
vx, vy reflect.Value, chunkSize int, name string,
|
||||||
|
makeRec func(reflect.Value, diffMode) textRecord,
|
||||||
|
) (list textList) {
|
||||||
|
eq := func(ix, iy int) bool {
|
||||||
|
return vx.Index(ix).Interface() == vy.Index(iy).Interface()
|
||||||
|
}
|
||||||
|
es := diff.Difference(vx.Len(), vy.Len(), func(ix, iy int) diff.Result {
|
||||||
|
return diff.BoolResult(eq(ix, iy))
|
||||||
|
})
|
||||||
|
|
||||||
|
appendChunks := func(v reflect.Value, d diffMode) int {
|
||||||
|
n0 := v.Len()
|
||||||
|
for v.Len() > 0 {
|
||||||
|
n := chunkSize
|
||||||
|
if n > v.Len() {
|
||||||
|
n = v.Len()
|
||||||
|
}
|
||||||
|
list = append(list, makeRec(v.Slice(0, n), d))
|
||||||
|
v = v.Slice(n, v.Len())
|
||||||
|
}
|
||||||
|
return n0 - v.Len()
|
||||||
|
}
|
||||||
|
|
||||||
|
var numDiffs int
|
||||||
|
maxLen := -1
|
||||||
|
if opts.LimitVerbosity {
|
||||||
|
maxLen = (1 << opts.verbosity()) << 2 // 4, 8, 16, 32, 64, etc...
|
||||||
|
opts.VerbosityLevel--
|
||||||
|
}
|
||||||
|
|
||||||
|
groups := coalesceAdjacentEdits(name, es)
|
||||||
|
groups = coalesceInterveningIdentical(groups, chunkSize/4)
|
||||||
|
groups = cleanupSurroundingIdentical(groups, eq)
|
||||||
|
maxGroup := diffStats{Name: name}
|
||||||
|
for i, ds := range groups {
|
||||||
|
if maxLen >= 0 && numDiffs >= maxLen {
|
||||||
|
maxGroup = maxGroup.Append(ds)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
// Print equal.
|
||||||
|
if ds.NumDiff() == 0 {
|
||||||
|
// Compute the number of leading and trailing equal bytes to print.
|
||||||
|
var numLo, numHi int
|
||||||
|
numEqual := ds.NumIgnored + ds.NumIdentical
|
||||||
|
for numLo < chunkSize*numContextRecords && numLo+numHi < numEqual && i != 0 {
|
||||||
|
numLo++
|
||||||
|
}
|
||||||
|
for numHi < chunkSize*numContextRecords && numLo+numHi < numEqual && i != len(groups)-1 {
|
||||||
|
numHi++
|
||||||
|
}
|
||||||
|
if numEqual-(numLo+numHi) <= chunkSize && ds.NumIgnored == 0 {
|
||||||
|
numHi = numEqual - numLo // Avoid pointless coalescing of single equal row
|
||||||
|
}
|
||||||
|
|
||||||
|
// Print the equal bytes.
|
||||||
|
appendChunks(vx.Slice(0, numLo), diffIdentical)
|
||||||
|
if numEqual > numLo+numHi {
|
||||||
|
ds.NumIdentical -= numLo + numHi
|
||||||
|
list.AppendEllipsis(ds)
|
||||||
|
}
|
||||||
|
appendChunks(vx.Slice(numEqual-numHi, numEqual), diffIdentical)
|
||||||
|
vx = vx.Slice(numEqual, vx.Len())
|
||||||
|
vy = vy.Slice(numEqual, vy.Len())
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
// Print unequal.
|
||||||
|
len0 := len(list)
|
||||||
|
nx := appendChunks(vx.Slice(0, ds.NumIdentical+ds.NumRemoved+ds.NumModified), diffRemoved)
|
||||||
|
vx = vx.Slice(nx, vx.Len())
|
||||||
|
ny := appendChunks(vy.Slice(0, ds.NumIdentical+ds.NumInserted+ds.NumModified), diffInserted)
|
||||||
|
vy = vy.Slice(ny, vy.Len())
|
||||||
|
numDiffs += len(list) - len0
|
||||||
|
}
|
||||||
|
if maxGroup.IsZero() {
|
||||||
|
assert(vx.Len() == 0 && vy.Len() == 0)
|
||||||
|
} else {
|
||||||
|
list.AppendEllipsis(maxGroup)
|
||||||
|
}
|
||||||
|
return list
|
||||||
|
}
|
||||||
|
|
||||||
|
// coalesceAdjacentEdits coalesces the list of edits into groups of adjacent
|
||||||
|
// equal or unequal counts.
|
||||||
|
//
|
||||||
|
// Example:
|
||||||
|
//
|
||||||
|
// Input: "..XXY...Y"
|
||||||
|
// Output: [
|
||||||
|
// {NumIdentical: 2},
|
||||||
|
// {NumRemoved: 2, NumInserted 1},
|
||||||
|
// {NumIdentical: 3},
|
||||||
|
// {NumInserted: 1},
|
||||||
|
// ]
|
||||||
|
func coalesceAdjacentEdits(name string, es diff.EditScript) (groups []diffStats) {
|
||||||
|
var prevMode byte
|
||||||
|
lastStats := func(mode byte) *diffStats {
|
||||||
|
if prevMode != mode {
|
||||||
|
groups = append(groups, diffStats{Name: name})
|
||||||
|
prevMode = mode
|
||||||
|
}
|
||||||
|
return &groups[len(groups)-1]
|
||||||
|
}
|
||||||
|
for _, e := range es {
|
||||||
|
switch e {
|
||||||
|
case diff.Identity:
|
||||||
|
lastStats('=').NumIdentical++
|
||||||
|
case diff.UniqueX:
|
||||||
|
lastStats('!').NumRemoved++
|
||||||
|
case diff.UniqueY:
|
||||||
|
lastStats('!').NumInserted++
|
||||||
|
case diff.Modified:
|
||||||
|
lastStats('!').NumModified++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return groups
|
||||||
|
}
|
||||||
|
|
||||||
|
// coalesceInterveningIdentical coalesces sufficiently short (<= windowSize)
|
||||||
|
// equal groups into adjacent unequal groups that currently result in a
|
||||||
|
// dual inserted/removed printout. This acts as a high-pass filter to smooth
|
||||||
|
// out high-frequency changes within the windowSize.
|
||||||
|
//
|
||||||
|
// Example:
|
||||||
|
//
|
||||||
|
// WindowSize: 16,
|
||||||
|
// Input: [
|
||||||
|
// {NumIdentical: 61}, // group 0
|
||||||
|
// {NumRemoved: 3, NumInserted: 1}, // group 1
|
||||||
|
// {NumIdentical: 6}, // ├── coalesce
|
||||||
|
// {NumInserted: 2}, // ├── coalesce
|
||||||
|
// {NumIdentical: 1}, // ├── coalesce
|
||||||
|
// {NumRemoved: 9}, // └── coalesce
|
||||||
|
// {NumIdentical: 64}, // group 2
|
||||||
|
// {NumRemoved: 3, NumInserted: 1}, // group 3
|
||||||
|
// {NumIdentical: 6}, // ├── coalesce
|
||||||
|
// {NumInserted: 2}, // ├── coalesce
|
||||||
|
// {NumIdentical: 1}, // ├── coalesce
|
||||||
|
// {NumRemoved: 7}, // ├── coalesce
|
||||||
|
// {NumIdentical: 1}, // ├── coalesce
|
||||||
|
// {NumRemoved: 2}, // └── coalesce
|
||||||
|
// {NumIdentical: 63}, // group 4
|
||||||
|
// ]
|
||||||
|
// Output: [
|
||||||
|
// {NumIdentical: 61},
|
||||||
|
// {NumIdentical: 7, NumRemoved: 12, NumInserted: 3},
|
||||||
|
// {NumIdentical: 64},
|
||||||
|
// {NumIdentical: 8, NumRemoved: 12, NumInserted: 3},
|
||||||
|
// {NumIdentical: 63},
|
||||||
|
// ]
|
||||||
|
func coalesceInterveningIdentical(groups []diffStats, windowSize int) []diffStats {
|
||||||
|
groups, groupsOrig := groups[:0], groups
|
||||||
|
for i, ds := range groupsOrig {
|
||||||
|
if len(groups) >= 2 && ds.NumDiff() > 0 {
|
||||||
|
prev := &groups[len(groups)-2] // Unequal group
|
||||||
|
curr := &groups[len(groups)-1] // Equal group
|
||||||
|
next := &groupsOrig[i] // Unequal group
|
||||||
|
hadX, hadY := prev.NumRemoved > 0, prev.NumInserted > 0
|
||||||
|
hasX, hasY := next.NumRemoved > 0, next.NumInserted > 0
|
||||||
|
if ((hadX || hasX) && (hadY || hasY)) && curr.NumIdentical <= windowSize {
|
||||||
|
*prev = prev.Append(*curr).Append(*next)
|
||||||
|
groups = groups[:len(groups)-1] // Truncate off equal group
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
groups = append(groups, ds)
|
||||||
|
}
|
||||||
|
return groups
|
||||||
|
}
|
||||||
|
|
||||||
|
// cleanupSurroundingIdentical scans through all unequal groups, and
|
||||||
|
// moves any leading sequence of equal elements to the preceding equal group and
|
||||||
|
// moves and trailing sequence of equal elements to the succeeding equal group.
|
||||||
|
//
|
||||||
|
// This is necessary since coalesceInterveningIdentical may coalesce edit groups
|
||||||
|
// together such that leading/trailing spans of equal elements becomes possible.
|
||||||
|
// Note that this can occur even with an optimal diffing algorithm.
|
||||||
|
//
|
||||||
|
// Example:
|
||||||
|
//
|
||||||
|
// Input: [
|
||||||
|
// {NumIdentical: 61},
|
||||||
|
// {NumIdentical: 1 , NumRemoved: 11, NumInserted: 2}, // assume 3 leading identical elements
|
||||||
|
// {NumIdentical: 67},
|
||||||
|
// {NumIdentical: 7, NumRemoved: 12, NumInserted: 3}, // assume 10 trailing identical elements
|
||||||
|
// {NumIdentical: 54},
|
||||||
|
// ]
|
||||||
|
// Output: [
|
||||||
|
// {NumIdentical: 64}, // incremented by 3
|
||||||
|
// {NumRemoved: 9},
|
||||||
|
// {NumIdentical: 67},
|
||||||
|
// {NumRemoved: 9},
|
||||||
|
// {NumIdentical: 64}, // incremented by 10
|
||||||
|
// ]
|
||||||
|
func cleanupSurroundingIdentical(groups []diffStats, eq func(i, j int) bool) []diffStats {
|
||||||
|
var ix, iy int // indexes into sequence x and y
|
||||||
|
for i, ds := range groups {
|
||||||
|
// Handle equal group.
|
||||||
|
if ds.NumDiff() == 0 {
|
||||||
|
ix += ds.NumIdentical
|
||||||
|
iy += ds.NumIdentical
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
// Handle unequal group.
|
||||||
|
nx := ds.NumIdentical + ds.NumRemoved + ds.NumModified
|
||||||
|
ny := ds.NumIdentical + ds.NumInserted + ds.NumModified
|
||||||
|
var numLeadingIdentical, numTrailingIdentical int
|
||||||
|
for j := 0; j < nx && j < ny && eq(ix+j, iy+j); j++ {
|
||||||
|
numLeadingIdentical++
|
||||||
|
}
|
||||||
|
for j := 0; j < nx && j < ny && eq(ix+nx-1-j, iy+ny-1-j); j++ {
|
||||||
|
numTrailingIdentical++
|
||||||
|
}
|
||||||
|
if numIdentical := numLeadingIdentical + numTrailingIdentical; numIdentical > 0 {
|
||||||
|
if numLeadingIdentical > 0 {
|
||||||
|
// Remove leading identical span from this group and
|
||||||
|
// insert it into the preceding group.
|
||||||
|
if i-1 >= 0 {
|
||||||
|
groups[i-1].NumIdentical += numLeadingIdentical
|
||||||
|
} else {
|
||||||
|
// No preceding group exists, so prepend a new group,
|
||||||
|
// but do so after we finish iterating over all groups.
|
||||||
|
defer func() {
|
||||||
|
groups = append([]diffStats{{Name: groups[0].Name, NumIdentical: numLeadingIdentical}}, groups...)
|
||||||
|
}()
|
||||||
|
}
|
||||||
|
// Increment indexes since the preceding group would have handled this.
|
||||||
|
ix += numLeadingIdentical
|
||||||
|
iy += numLeadingIdentical
|
||||||
|
}
|
||||||
|
if numTrailingIdentical > 0 {
|
||||||
|
// Remove trailing identical span from this group and
|
||||||
|
// insert it into the succeeding group.
|
||||||
|
if i+1 < len(groups) {
|
||||||
|
groups[i+1].NumIdentical += numTrailingIdentical
|
||||||
|
} else {
|
||||||
|
// No succeeding group exists, so append a new group,
|
||||||
|
// but do so after we finish iterating over all groups.
|
||||||
|
defer func() {
|
||||||
|
groups = append(groups, diffStats{Name: groups[len(groups)-1].Name, NumIdentical: numTrailingIdentical})
|
||||||
|
}()
|
||||||
|
}
|
||||||
|
// Do not increment indexes since the succeeding group will handle this.
|
||||||
|
}
|
||||||
|
|
||||||
|
// Update this group since some identical elements were removed.
|
||||||
|
nx -= numIdentical
|
||||||
|
ny -= numIdentical
|
||||||
|
groups[i] = diffStats{Name: ds.Name, NumRemoved: nx, NumInserted: ny}
|
||||||
|
}
|
||||||
|
ix += nx
|
||||||
|
iy += ny
|
||||||
|
}
|
||||||
|
return groups
|
||||||
|
}
|
||||||
432
vendor/github.com/google/go-cmp/cmp/report_text.go
generated
vendored
Normal file
432
vendor/github.com/google/go-cmp/cmp/report_text.go
generated
vendored
Normal file
@@ -0,0 +1,432 @@
|
|||||||
|
// Copyright 2019, The Go Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a BSD-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
package cmp
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"fmt"
|
||||||
|
"math/rand"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
"unicode/utf8"
|
||||||
|
|
||||||
|
"github.com/google/go-cmp/cmp/internal/flags"
|
||||||
|
)
|
||||||
|
|
||||||
|
var randBool = rand.New(rand.NewSource(time.Now().Unix())).Intn(2) == 0
|
||||||
|
|
||||||
|
const maxColumnLength = 80
|
||||||
|
|
||||||
|
type indentMode int
|
||||||
|
|
||||||
|
func (n indentMode) appendIndent(b []byte, d diffMode) []byte {
|
||||||
|
// The output of Diff is documented as being unstable to provide future
|
||||||
|
// flexibility in changing the output for more humanly readable reports.
|
||||||
|
// This logic intentionally introduces instability to the exact output
|
||||||
|
// so that users can detect accidental reliance on stability early on,
|
||||||
|
// rather than much later when an actual change to the format occurs.
|
||||||
|
if flags.Deterministic || randBool {
|
||||||
|
// Use regular spaces (U+0020).
|
||||||
|
switch d {
|
||||||
|
case diffUnknown, diffIdentical:
|
||||||
|
b = append(b, " "...)
|
||||||
|
case diffRemoved:
|
||||||
|
b = append(b, "- "...)
|
||||||
|
case diffInserted:
|
||||||
|
b = append(b, "+ "...)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Use non-breaking spaces (U+00a0).
|
||||||
|
switch d {
|
||||||
|
case diffUnknown, diffIdentical:
|
||||||
|
b = append(b, " "...)
|
||||||
|
case diffRemoved:
|
||||||
|
b = append(b, "- "...)
|
||||||
|
case diffInserted:
|
||||||
|
b = append(b, "+ "...)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return repeatCount(n).appendChar(b, '\t')
|
||||||
|
}
|
||||||
|
|
||||||
|
type repeatCount int
|
||||||
|
|
||||||
|
func (n repeatCount) appendChar(b []byte, c byte) []byte {
|
||||||
|
for ; n > 0; n-- {
|
||||||
|
b = append(b, c)
|
||||||
|
}
|
||||||
|
return b
|
||||||
|
}
|
||||||
|
|
||||||
|
// textNode is a simplified tree-based representation of structured text.
|
||||||
|
// Possible node types are textWrap, textList, or textLine.
|
||||||
|
type textNode interface {
|
||||||
|
// Len reports the length in bytes of a single-line version of the tree.
|
||||||
|
// Nested textRecord.Diff and textRecord.Comment fields are ignored.
|
||||||
|
Len() int
|
||||||
|
// Equal reports whether the two trees are structurally identical.
|
||||||
|
// Nested textRecord.Diff and textRecord.Comment fields are compared.
|
||||||
|
Equal(textNode) bool
|
||||||
|
// String returns the string representation of the text tree.
|
||||||
|
// It is not guaranteed that len(x.String()) == x.Len(),
|
||||||
|
// nor that x.String() == y.String() implies that x.Equal(y).
|
||||||
|
String() string
|
||||||
|
|
||||||
|
// formatCompactTo formats the contents of the tree as a single-line string
|
||||||
|
// to the provided buffer. Any nested textRecord.Diff and textRecord.Comment
|
||||||
|
// fields are ignored.
|
||||||
|
//
|
||||||
|
// However, not all nodes in the tree should be collapsed as a single-line.
|
||||||
|
// If a node can be collapsed as a single-line, it is replaced by a textLine
|
||||||
|
// node. Since the top-level node cannot replace itself, this also returns
|
||||||
|
// the current node itself.
|
||||||
|
//
|
||||||
|
// This does not mutate the receiver.
|
||||||
|
formatCompactTo([]byte, diffMode) ([]byte, textNode)
|
||||||
|
// formatExpandedTo formats the contents of the tree as a multi-line string
|
||||||
|
// to the provided buffer. In order for column alignment to operate well,
|
||||||
|
// formatCompactTo must be called before calling formatExpandedTo.
|
||||||
|
formatExpandedTo([]byte, diffMode, indentMode) []byte
|
||||||
|
}
|
||||||
|
|
||||||
|
// textWrap is a wrapper that concatenates a prefix and/or a suffix
|
||||||
|
// to the underlying node.
|
||||||
|
type textWrap struct {
|
||||||
|
Prefix string // e.g., "bytes.Buffer{"
|
||||||
|
Value textNode // textWrap | textList | textLine
|
||||||
|
Suffix string // e.g., "}"
|
||||||
|
Metadata interface{} // arbitrary metadata; has no effect on formatting
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *textWrap) Len() int {
|
||||||
|
return len(s.Prefix) + s.Value.Len() + len(s.Suffix)
|
||||||
|
}
|
||||||
|
func (s1 *textWrap) Equal(s2 textNode) bool {
|
||||||
|
if s2, ok := s2.(*textWrap); ok {
|
||||||
|
return s1.Prefix == s2.Prefix && s1.Value.Equal(s2.Value) && s1.Suffix == s2.Suffix
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
func (s *textWrap) String() string {
|
||||||
|
var d diffMode
|
||||||
|
var n indentMode
|
||||||
|
_, s2 := s.formatCompactTo(nil, d)
|
||||||
|
b := n.appendIndent(nil, d) // Leading indent
|
||||||
|
b = s2.formatExpandedTo(b, d, n) // Main body
|
||||||
|
b = append(b, '\n') // Trailing newline
|
||||||
|
return string(b)
|
||||||
|
}
|
||||||
|
func (s *textWrap) formatCompactTo(b []byte, d diffMode) ([]byte, textNode) {
|
||||||
|
n0 := len(b) // Original buffer length
|
||||||
|
b = append(b, s.Prefix...)
|
||||||
|
b, s.Value = s.Value.formatCompactTo(b, d)
|
||||||
|
b = append(b, s.Suffix...)
|
||||||
|
if _, ok := s.Value.(textLine); ok {
|
||||||
|
return b, textLine(b[n0:])
|
||||||
|
}
|
||||||
|
return b, s
|
||||||
|
}
|
||||||
|
func (s *textWrap) formatExpandedTo(b []byte, d diffMode, n indentMode) []byte {
|
||||||
|
b = append(b, s.Prefix...)
|
||||||
|
b = s.Value.formatExpandedTo(b, d, n)
|
||||||
|
b = append(b, s.Suffix...)
|
||||||
|
return b
|
||||||
|
}
|
||||||
|
|
||||||
|
// textList is a comma-separated list of textWrap or textLine nodes.
|
||||||
|
// The list may be formatted as multi-lines or single-line at the discretion
|
||||||
|
// of the textList.formatCompactTo method.
|
||||||
|
type textList []textRecord
|
||||||
|
type textRecord struct {
|
||||||
|
Diff diffMode // e.g., 0 or '-' or '+'
|
||||||
|
Key string // e.g., "MyField"
|
||||||
|
Value textNode // textWrap | textLine
|
||||||
|
ElideComma bool // avoid trailing comma
|
||||||
|
Comment fmt.Stringer // e.g., "6 identical fields"
|
||||||
|
}
|
||||||
|
|
||||||
|
// AppendEllipsis appends a new ellipsis node to the list if none already
|
||||||
|
// exists at the end. If cs is non-zero it coalesces the statistics with the
|
||||||
|
// previous diffStats.
|
||||||
|
func (s *textList) AppendEllipsis(ds diffStats) {
|
||||||
|
hasStats := !ds.IsZero()
|
||||||
|
if len(*s) == 0 || !(*s)[len(*s)-1].Value.Equal(textEllipsis) {
|
||||||
|
if hasStats {
|
||||||
|
*s = append(*s, textRecord{Value: textEllipsis, ElideComma: true, Comment: ds})
|
||||||
|
} else {
|
||||||
|
*s = append(*s, textRecord{Value: textEllipsis, ElideComma: true})
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if hasStats {
|
||||||
|
(*s)[len(*s)-1].Comment = (*s)[len(*s)-1].Comment.(diffStats).Append(ds)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s textList) Len() (n int) {
|
||||||
|
for i, r := range s {
|
||||||
|
n += len(r.Key)
|
||||||
|
if r.Key != "" {
|
||||||
|
n += len(": ")
|
||||||
|
}
|
||||||
|
n += r.Value.Len()
|
||||||
|
if i < len(s)-1 {
|
||||||
|
n += len(", ")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return n
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s1 textList) Equal(s2 textNode) bool {
|
||||||
|
if s2, ok := s2.(textList); ok {
|
||||||
|
if len(s1) != len(s2) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
for i := range s1 {
|
||||||
|
r1, r2 := s1[i], s2[i]
|
||||||
|
if !(r1.Diff == r2.Diff && r1.Key == r2.Key && r1.Value.Equal(r2.Value) && r1.Comment == r2.Comment) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s textList) String() string {
|
||||||
|
return (&textWrap{Prefix: "{", Value: s, Suffix: "}"}).String()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s textList) formatCompactTo(b []byte, d diffMode) ([]byte, textNode) {
|
||||||
|
s = append(textList(nil), s...) // Avoid mutating original
|
||||||
|
|
||||||
|
// Determine whether we can collapse this list as a single line.
|
||||||
|
n0 := len(b) // Original buffer length
|
||||||
|
var multiLine bool
|
||||||
|
for i, r := range s {
|
||||||
|
if r.Diff == diffInserted || r.Diff == diffRemoved {
|
||||||
|
multiLine = true
|
||||||
|
}
|
||||||
|
b = append(b, r.Key...)
|
||||||
|
if r.Key != "" {
|
||||||
|
b = append(b, ": "...)
|
||||||
|
}
|
||||||
|
b, s[i].Value = r.Value.formatCompactTo(b, d|r.Diff)
|
||||||
|
if _, ok := s[i].Value.(textLine); !ok {
|
||||||
|
multiLine = true
|
||||||
|
}
|
||||||
|
if r.Comment != nil {
|
||||||
|
multiLine = true
|
||||||
|
}
|
||||||
|
if i < len(s)-1 {
|
||||||
|
b = append(b, ", "...)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Force multi-lined output when printing a removed/inserted node that
|
||||||
|
// is sufficiently long.
|
||||||
|
if (d == diffInserted || d == diffRemoved) && len(b[n0:]) > maxColumnLength {
|
||||||
|
multiLine = true
|
||||||
|
}
|
||||||
|
if !multiLine {
|
||||||
|
return b, textLine(b[n0:])
|
||||||
|
}
|
||||||
|
return b, s
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s textList) formatExpandedTo(b []byte, d diffMode, n indentMode) []byte {
|
||||||
|
alignKeyLens := s.alignLens(
|
||||||
|
func(r textRecord) bool {
|
||||||
|
_, isLine := r.Value.(textLine)
|
||||||
|
return r.Key == "" || !isLine
|
||||||
|
},
|
||||||
|
func(r textRecord) int { return utf8.RuneCountInString(r.Key) },
|
||||||
|
)
|
||||||
|
alignValueLens := s.alignLens(
|
||||||
|
func(r textRecord) bool {
|
||||||
|
_, isLine := r.Value.(textLine)
|
||||||
|
return !isLine || r.Value.Equal(textEllipsis) || r.Comment == nil
|
||||||
|
},
|
||||||
|
func(r textRecord) int { return utf8.RuneCount(r.Value.(textLine)) },
|
||||||
|
)
|
||||||
|
|
||||||
|
// Format lists of simple lists in a batched form.
|
||||||
|
// If the list is sequence of only textLine values,
|
||||||
|
// then batch multiple values on a single line.
|
||||||
|
var isSimple bool
|
||||||
|
for _, r := range s {
|
||||||
|
_, isLine := r.Value.(textLine)
|
||||||
|
isSimple = r.Diff == 0 && r.Key == "" && isLine && r.Comment == nil
|
||||||
|
if !isSimple {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if isSimple {
|
||||||
|
n++
|
||||||
|
var batch []byte
|
||||||
|
emitBatch := func() {
|
||||||
|
if len(batch) > 0 {
|
||||||
|
b = n.appendIndent(append(b, '\n'), d)
|
||||||
|
b = append(b, bytes.TrimRight(batch, " ")...)
|
||||||
|
batch = batch[:0]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for _, r := range s {
|
||||||
|
line := r.Value.(textLine)
|
||||||
|
if len(batch)+len(line)+len(", ") > maxColumnLength {
|
||||||
|
emitBatch()
|
||||||
|
}
|
||||||
|
batch = append(batch, line...)
|
||||||
|
batch = append(batch, ", "...)
|
||||||
|
}
|
||||||
|
emitBatch()
|
||||||
|
n--
|
||||||
|
return n.appendIndent(append(b, '\n'), d)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Format the list as a multi-lined output.
|
||||||
|
n++
|
||||||
|
for i, r := range s {
|
||||||
|
b = n.appendIndent(append(b, '\n'), d|r.Diff)
|
||||||
|
if r.Key != "" {
|
||||||
|
b = append(b, r.Key+": "...)
|
||||||
|
}
|
||||||
|
b = alignKeyLens[i].appendChar(b, ' ')
|
||||||
|
|
||||||
|
b = r.Value.formatExpandedTo(b, d|r.Diff, n)
|
||||||
|
if !r.ElideComma {
|
||||||
|
b = append(b, ',')
|
||||||
|
}
|
||||||
|
b = alignValueLens[i].appendChar(b, ' ')
|
||||||
|
|
||||||
|
if r.Comment != nil {
|
||||||
|
b = append(b, " // "+r.Comment.String()...)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
n--
|
||||||
|
|
||||||
|
return n.appendIndent(append(b, '\n'), d)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s textList) alignLens(
|
||||||
|
skipFunc func(textRecord) bool,
|
||||||
|
lenFunc func(textRecord) int,
|
||||||
|
) []repeatCount {
|
||||||
|
var startIdx, endIdx, maxLen int
|
||||||
|
lens := make([]repeatCount, len(s))
|
||||||
|
for i, r := range s {
|
||||||
|
if skipFunc(r) {
|
||||||
|
for j := startIdx; j < endIdx && j < len(s); j++ {
|
||||||
|
lens[j] = repeatCount(maxLen - lenFunc(s[j]))
|
||||||
|
}
|
||||||
|
startIdx, endIdx, maxLen = i+1, i+1, 0
|
||||||
|
} else {
|
||||||
|
if maxLen < lenFunc(r) {
|
||||||
|
maxLen = lenFunc(r)
|
||||||
|
}
|
||||||
|
endIdx = i + 1
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for j := startIdx; j < endIdx && j < len(s); j++ {
|
||||||
|
lens[j] = repeatCount(maxLen - lenFunc(s[j]))
|
||||||
|
}
|
||||||
|
return lens
|
||||||
|
}
|
||||||
|
|
||||||
|
// textLine is a single-line segment of text and is always a leaf node
|
||||||
|
// in the textNode tree.
|
||||||
|
type textLine []byte
|
||||||
|
|
||||||
|
var (
|
||||||
|
textNil = textLine("nil")
|
||||||
|
textEllipsis = textLine("...")
|
||||||
|
)
|
||||||
|
|
||||||
|
func (s textLine) Len() int {
|
||||||
|
return len(s)
|
||||||
|
}
|
||||||
|
func (s1 textLine) Equal(s2 textNode) bool {
|
||||||
|
if s2, ok := s2.(textLine); ok {
|
||||||
|
return bytes.Equal([]byte(s1), []byte(s2))
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
func (s textLine) String() string {
|
||||||
|
return string(s)
|
||||||
|
}
|
||||||
|
func (s textLine) formatCompactTo(b []byte, d diffMode) ([]byte, textNode) {
|
||||||
|
return append(b, s...), s
|
||||||
|
}
|
||||||
|
func (s textLine) formatExpandedTo(b []byte, _ diffMode, _ indentMode) []byte {
|
||||||
|
return append(b, s...)
|
||||||
|
}
|
||||||
|
|
||||||
|
type diffStats struct {
|
||||||
|
Name string
|
||||||
|
NumIgnored int
|
||||||
|
NumIdentical int
|
||||||
|
NumRemoved int
|
||||||
|
NumInserted int
|
||||||
|
NumModified int
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s diffStats) IsZero() bool {
|
||||||
|
s.Name = ""
|
||||||
|
return s == diffStats{}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s diffStats) NumDiff() int {
|
||||||
|
return s.NumRemoved + s.NumInserted + s.NumModified
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s diffStats) Append(ds diffStats) diffStats {
|
||||||
|
assert(s.Name == ds.Name)
|
||||||
|
s.NumIgnored += ds.NumIgnored
|
||||||
|
s.NumIdentical += ds.NumIdentical
|
||||||
|
s.NumRemoved += ds.NumRemoved
|
||||||
|
s.NumInserted += ds.NumInserted
|
||||||
|
s.NumModified += ds.NumModified
|
||||||
|
return s
|
||||||
|
}
|
||||||
|
|
||||||
|
// String prints a humanly-readable summary of coalesced records.
|
||||||
|
//
|
||||||
|
// Example:
|
||||||
|
//
|
||||||
|
// diffStats{Name: "Field", NumIgnored: 5}.String() => "5 ignored fields"
|
||||||
|
func (s diffStats) String() string {
|
||||||
|
var ss []string
|
||||||
|
var sum int
|
||||||
|
labels := [...]string{"ignored", "identical", "removed", "inserted", "modified"}
|
||||||
|
counts := [...]int{s.NumIgnored, s.NumIdentical, s.NumRemoved, s.NumInserted, s.NumModified}
|
||||||
|
for i, n := range counts {
|
||||||
|
if n > 0 {
|
||||||
|
ss = append(ss, fmt.Sprintf("%d %v", n, labels[i]))
|
||||||
|
}
|
||||||
|
sum += n
|
||||||
|
}
|
||||||
|
|
||||||
|
// Pluralize the name (adjusting for some obscure English grammar rules).
|
||||||
|
name := s.Name
|
||||||
|
if sum > 1 {
|
||||||
|
name += "s"
|
||||||
|
if strings.HasSuffix(name, "ys") {
|
||||||
|
name = name[:len(name)-2] + "ies" // e.g., "entrys" => "entries"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Format the list according to English grammar (with Oxford comma).
|
||||||
|
switch n := len(ss); n {
|
||||||
|
case 0:
|
||||||
|
return ""
|
||||||
|
case 1, 2:
|
||||||
|
return strings.Join(ss, " and ") + " " + name
|
||||||
|
default:
|
||||||
|
return strings.Join(ss[:n-1], ", ") + ", and " + ss[n-1] + " " + name
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
type commentString string
|
||||||
|
|
||||||
|
func (s commentString) String() string { return string(s) }
|
||||||
121
vendor/github.com/google/go-cmp/cmp/report_value.go
generated
vendored
Normal file
121
vendor/github.com/google/go-cmp/cmp/report_value.go
generated
vendored
Normal file
@@ -0,0 +1,121 @@
|
|||||||
|
// Copyright 2019, The Go Authors. All rights reserved.
|
||||||
|
// Use of this source code is governed by a BSD-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
package cmp
|
||||||
|
|
||||||
|
import "reflect"
|
||||||
|
|
||||||
|
// valueNode represents a single node within a report, which is a
|
||||||
|
// structured representation of the value tree, containing information
|
||||||
|
// regarding which nodes are equal or not.
|
||||||
|
type valueNode struct {
|
||||||
|
parent *valueNode
|
||||||
|
|
||||||
|
Type reflect.Type
|
||||||
|
ValueX reflect.Value
|
||||||
|
ValueY reflect.Value
|
||||||
|
|
||||||
|
// NumSame is the number of leaf nodes that are equal.
|
||||||
|
// All descendants are equal only if NumDiff is 0.
|
||||||
|
NumSame int
|
||||||
|
// NumDiff is the number of leaf nodes that are not equal.
|
||||||
|
NumDiff int
|
||||||
|
// NumIgnored is the number of leaf nodes that are ignored.
|
||||||
|
NumIgnored int
|
||||||
|
// NumCompared is the number of leaf nodes that were compared
|
||||||
|
// using an Equal method or Comparer function.
|
||||||
|
NumCompared int
|
||||||
|
// NumTransformed is the number of non-leaf nodes that were transformed.
|
||||||
|
NumTransformed int
|
||||||
|
// NumChildren is the number of transitive descendants of this node.
|
||||||
|
// This counts from zero; thus, leaf nodes have no descendants.
|
||||||
|
NumChildren int
|
||||||
|
// MaxDepth is the maximum depth of the tree. This counts from zero;
|
||||||
|
// thus, leaf nodes have a depth of zero.
|
||||||
|
MaxDepth int
|
||||||
|
|
||||||
|
// Records is a list of struct fields, slice elements, or map entries.
|
||||||
|
Records []reportRecord // If populated, implies Value is not populated
|
||||||
|
|
||||||
|
// Value is the result of a transformation, pointer indirect, of
|
||||||
|
// type assertion.
|
||||||
|
Value *valueNode // If populated, implies Records is not populated
|
||||||
|
|
||||||
|
// TransformerName is the name of the transformer.
|
||||||
|
TransformerName string // If non-empty, implies Value is populated
|
||||||
|
}
|
||||||
|
type reportRecord struct {
|
||||||
|
Key reflect.Value // Invalid for slice element
|
||||||
|
Value *valueNode
|
||||||
|
}
|
||||||
|
|
||||||
|
func (parent *valueNode) PushStep(ps PathStep) (child *valueNode) {
|
||||||
|
vx, vy := ps.Values()
|
||||||
|
child = &valueNode{parent: parent, Type: ps.Type(), ValueX: vx, ValueY: vy}
|
||||||
|
switch s := ps.(type) {
|
||||||
|
case StructField:
|
||||||
|
assert(parent.Value == nil)
|
||||||
|
parent.Records = append(parent.Records, reportRecord{Key: reflect.ValueOf(s.Name()), Value: child})
|
||||||
|
case SliceIndex:
|
||||||
|
assert(parent.Value == nil)
|
||||||
|
parent.Records = append(parent.Records, reportRecord{Value: child})
|
||||||
|
case MapIndex:
|
||||||
|
assert(parent.Value == nil)
|
||||||
|
parent.Records = append(parent.Records, reportRecord{Key: s.Key(), Value: child})
|
||||||
|
case Indirect:
|
||||||
|
assert(parent.Value == nil && parent.Records == nil)
|
||||||
|
parent.Value = child
|
||||||
|
case TypeAssertion:
|
||||||
|
assert(parent.Value == nil && parent.Records == nil)
|
||||||
|
parent.Value = child
|
||||||
|
case Transform:
|
||||||
|
assert(parent.Value == nil && parent.Records == nil)
|
||||||
|
parent.Value = child
|
||||||
|
parent.TransformerName = s.Name()
|
||||||
|
parent.NumTransformed++
|
||||||
|
default:
|
||||||
|
assert(parent == nil) // Must be the root step
|
||||||
|
}
|
||||||
|
return child
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *valueNode) Report(rs Result) {
|
||||||
|
assert(r.MaxDepth == 0) // May only be called on leaf nodes
|
||||||
|
|
||||||
|
if rs.ByIgnore() {
|
||||||
|
r.NumIgnored++
|
||||||
|
} else {
|
||||||
|
if rs.Equal() {
|
||||||
|
r.NumSame++
|
||||||
|
} else {
|
||||||
|
r.NumDiff++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
assert(r.NumSame+r.NumDiff+r.NumIgnored == 1)
|
||||||
|
|
||||||
|
if rs.ByMethod() {
|
||||||
|
r.NumCompared++
|
||||||
|
}
|
||||||
|
if rs.ByFunc() {
|
||||||
|
r.NumCompared++
|
||||||
|
}
|
||||||
|
assert(r.NumCompared <= 1)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (child *valueNode) PopStep() (parent *valueNode) {
|
||||||
|
if child.parent == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
parent = child.parent
|
||||||
|
parent.NumSame += child.NumSame
|
||||||
|
parent.NumDiff += child.NumDiff
|
||||||
|
parent.NumIgnored += child.NumIgnored
|
||||||
|
parent.NumCompared += child.NumCompared
|
||||||
|
parent.NumTransformed += child.NumTransformed
|
||||||
|
parent.NumChildren += child.NumChildren + 1
|
||||||
|
if parent.MaxDepth < child.MaxDepth+1 {
|
||||||
|
parent.MaxDepth = child.MaxDepth + 1
|
||||||
|
}
|
||||||
|
return parent
|
||||||
|
}
|
||||||
7
vendor/github.com/onsi/gomega/.gitignore
generated
vendored
Normal file
7
vendor/github.com/onsi/gomega/.gitignore
generated
vendored
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
.DS_Store
|
||||||
|
*.test
|
||||||
|
.
|
||||||
|
.idea
|
||||||
|
gomega.iml
|
||||||
|
TODO.md
|
||||||
|
.vscode
|
||||||
583
vendor/github.com/onsi/gomega/CHANGELOG.md
generated
vendored
Normal file
583
vendor/github.com/onsi/gomega/CHANGELOG.md
generated
vendored
Normal file
@@ -0,0 +1,583 @@
|
|||||||
|
## 1.27.6
|
||||||
|
|
||||||
|
### Fixes
|
||||||
|
- Allow collections matchers to work correctly when expected has nil elements [60e7cf3]
|
||||||
|
|
||||||
|
### Maintenance
|
||||||
|
- updates MatchError godoc comment to also accept a Gomega matcher (#654) [67b869d]
|
||||||
|
|
||||||
|
## 1.27.5
|
||||||
|
|
||||||
|
### Maintenance
|
||||||
|
- Bump github.com/onsi/ginkgo/v2 from 2.9.1 to 2.9.2 (#653) [a215021]
|
||||||
|
- Bump github.com/go-task/slim-sprig (#652) [a26fed8]
|
||||||
|
|
||||||
|
## 1.27.4
|
||||||
|
|
||||||
|
### Fixes
|
||||||
|
- improve error formatting and remove duplication of error message in Eventually/Consistently [854f075]
|
||||||
|
|
||||||
|
### Maintenance
|
||||||
|
- Bump github.com/onsi/ginkgo/v2 from 2.9.0 to 2.9.1 (#650) [ccebd9b]
|
||||||
|
|
||||||
|
## 1.27.3
|
||||||
|
|
||||||
|
### Fixes
|
||||||
|
- format.Object now always includes err.Error() when passed an error [86d97ef]
|
||||||
|
- Fix HaveExactElements to work inside ContainElement or other collection matchers (#648) [636757e]
|
||||||
|
|
||||||
|
### Maintenance
|
||||||
|
- Bump github.com/golang/protobuf from 1.5.2 to 1.5.3 (#649) [cc16689]
|
||||||
|
- Bump github.com/onsi/ginkgo/v2 from 2.8.4 to 2.9.0 (#646) [e783366]
|
||||||
|
|
||||||
|
## 1.27.2
|
||||||
|
|
||||||
|
### Fixes
|
||||||
|
- improve poll progress message when polling a consistently that has been passing [28a319b]
|
||||||
|
|
||||||
|
### Maintenance
|
||||||
|
- bump ginkgo
|
||||||
|
- remove tools.go hack as Ginkgo 2.8.2 automatically pulls in the cli dependencies [81443b3]
|
||||||
|
|
||||||
|
## 1.27.1
|
||||||
|
|
||||||
|
### Maintenance
|
||||||
|
|
||||||
|
- Bump golang.org/x/net from 0.6.0 to 0.7.0 (#640) [bc686cd]
|
||||||
|
|
||||||
|
## 1.27.0
|
||||||
|
|
||||||
|
### Features
|
||||||
|
- Add HaveExactElements matcher (#634) [9d50783]
|
||||||
|
- update Gomega docs to discuss GinkgoHelper() [be32774]
|
||||||
|
|
||||||
|
### Maintenance
|
||||||
|
- Bump github.com/onsi/ginkgo/v2 from 2.8.0 to 2.8.1 (#639) [296a68b]
|
||||||
|
- Bump golang.org/x/net from 0.5.0 to 0.6.0 (#638) [c2b098b]
|
||||||
|
- Bump github-pages from 227 to 228 in /docs (#636) [a9069ab]
|
||||||
|
- test: update matrix for Go 1.20 (#635) [6bd25c8]
|
||||||
|
- Bump github.com/onsi/ginkgo/v2 from 2.7.0 to 2.8.0 (#631) [5445f8b]
|
||||||
|
- Bump webrick from 1.7.0 to 1.8.1 in /docs (#630) [03e93bb]
|
||||||
|
- codeql: add ruby language (#626) [63c7d21]
|
||||||
|
- dependabot: add bundler package-ecosystem for docs (#625) [d92f963]
|
||||||
|
|
||||||
|
## 1.26.0
|
||||||
|
|
||||||
|
### Features
|
||||||
|
- When a polled function returns an error, keep track of the actual and report on the matcher state of the last non-errored actual [21f3090]
|
||||||
|
- improve eventually failure message output [c530fb3]
|
||||||
|
|
||||||
|
### Fixes
|
||||||
|
- fix several documentation spelling issues [e2eff1f]
|
||||||
|
|
||||||
|
|
||||||
|
## 1.25.0
|
||||||
|
|
||||||
|
### Features
|
||||||
|
- add `MustPassRepeatedly(int)` to asyncAssertion (#619) [4509f72]
|
||||||
|
- compare unwrapped errors using DeepEqual (#617) [aaeaa5d]
|
||||||
|
|
||||||
|
### Maintenance
|
||||||
|
- Bump golang.org/x/net from 0.4.0 to 0.5.0 (#614) [c7cfea4]
|
||||||
|
- Bump github.com/onsi/ginkgo/v2 from 2.6.1 to 2.7.0 (#615) [71b8adb]
|
||||||
|
- Docs: Fix typo "MUltiple" -> "Multiple" (#616) [9351dda]
|
||||||
|
- clean up go.sum [cd1dc1d]
|
||||||
|
|
||||||
|
## 1.24.2
|
||||||
|
|
||||||
|
### Fixes
|
||||||
|
- Correctly handle assertion failure panics for eventually/consistnetly "g Gomega"s in a goroutine [78f1660]
|
||||||
|
- docs:Fix typo "you an" -> "you can" (#607) [3187c1f]
|
||||||
|
- fixes issue #600 (#606) [808d192]
|
||||||
|
|
||||||
|
### Maintenance
|
||||||
|
- Bump golang.org/x/net from 0.2.0 to 0.4.0 (#611) [6ebc0bf]
|
||||||
|
- Bump nokogiri from 1.13.9 to 1.13.10 in /docs (#612) [258cfc8]
|
||||||
|
- Bump github.com/onsi/ginkgo/v2 from 2.5.0 to 2.5.1 (#609) [e6c3eb9]
|
||||||
|
|
||||||
|
## 1.24.1
|
||||||
|
|
||||||
|
### Fixes
|
||||||
|
- maintain backward compatibility for Eventually and Consisntetly's signatures [4c7df5e]
|
||||||
|
- fix small typo (#601) [ea0ebe6]
|
||||||
|
|
||||||
|
### Maintenance
|
||||||
|
- Bump golang.org/x/net from 0.1.0 to 0.2.0 (#603) [1ba8372]
|
||||||
|
- Bump github.com/onsi/ginkgo/v2 from 2.4.0 to 2.5.0 (#602) [f9426cb]
|
||||||
|
- fix label-filter in test.yml [d795db6]
|
||||||
|
- stop running flakey tests and rely on external network dependencies in CI [7133290]
|
||||||
|
|
||||||
|
## 1.24.0
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
Introducting [gcustom](https://onsi.github.io/gomega/#gcustom-a-convenient-mechanism-for-buildling-custom-matchers) - a convenient mechanism for building custom matchers.
|
||||||
|
|
||||||
|
This is an RC release for `gcustom`. The external API may be tweaked in response to feedback however it is expected to remain mostly stable.
|
||||||
|
|
||||||
|
### Maintenance
|
||||||
|
|
||||||
|
- Update BeComparableTo documentation [756eaa0]
|
||||||
|
|
||||||
|
## 1.23.0
|
||||||
|
|
||||||
|
### Features
|
||||||
|
- Custom formatting on a per-type basis can be provided using `format.RegisterCustomFormatter()` -- see the docs [here](https://onsi.github.io/gomega/#adjusting-output)
|
||||||
|
|
||||||
|
- Substantial improvement have been made to `StopTrying()`:
|
||||||
|
- Users can now use `StopTrying().Wrap(err)` to wrap errors and `StopTrying().Attach(description, object)` to attach arbitrary objects to the `StopTrying()` error
|
||||||
|
- `StopTrying()` is now always interpreted as a failure. If you are an early adopter of `StopTrying()` you may need to change your code as the prior version would match against the returned value even if `StopTrying()` was returned. Going forward the `StopTrying()` api should remain stable.
|
||||||
|
- `StopTrying()` and `StopTrying().Now()` can both be used in matchers - not just polled functions.
|
||||||
|
|
||||||
|
- `TryAgainAfter(duration)` is used like `StopTrying()` but instructs `Eventually` and `Consistently` that the poll should be tried again after the specified duration. This allows you to dynamically adjust the polling duration.
|
||||||
|
|
||||||
|
- `ctx` can now be passed-in as the first argument to `Eventually` and `Consistently`.
|
||||||
|
|
||||||
|
## Maintenance
|
||||||
|
|
||||||
|
- Bump github.com/onsi/ginkgo/v2 from 2.3.0 to 2.3.1 (#597) [afed901]
|
||||||
|
- Bump nokogiri from 1.13.8 to 1.13.9 in /docs (#599) [7c691b3]
|
||||||
|
- Bump github.com/google/go-cmp from 0.5.8 to 0.5.9 (#587) [ff22665]
|
||||||
|
|
||||||
|
## 1.22.1
|
||||||
|
|
||||||
|
## Fixes
|
||||||
|
- When passed a context and no explicit timeout, Eventually will only timeout when the context is cancelled [e5105cf]
|
||||||
|
- Allow StopTrying() to be wrapped [bf3cba9]
|
||||||
|
|
||||||
|
## Maintenance
|
||||||
|
- bump to ginkgo v2.3.0 [c5d5c39]
|
||||||
|
|
||||||
|
## 1.22.0
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
Several improvements have been made to `Eventually` and `Consistently` in this and the most recent releases:
|
||||||
|
|
||||||
|
- Eventually and Consistently can take a context.Context [65c01bc]
|
||||||
|
This enables integration with Ginkgo 2.3.0's interruptible nodes and node timeouts.
|
||||||
|
- Eventually and Consistently that are passed a SpecContext can provide reports when an interrupt occurs [0d063c9]
|
||||||
|
- Eventually/Consistently will forward an attached context to functions that ask for one [e2091c5]
|
||||||
|
- Eventually/Consistently supports passing arguments to functions via WithArguments() [a2dc7c3]
|
||||||
|
- Eventually and Consistently can now be stopped early with StopTrying(message) and StopTrying(message).Now() [52976bb]
|
||||||
|
|
||||||
|
These improvements are all documented in [Gomega's docs](https://onsi.github.io/gomega/#making-asynchronous-assertions)
|
||||||
|
|
||||||
|
## Fixes
|
||||||
|
|
||||||
|
## Maintenance
|
||||||
|
|
||||||
|
## 1.21.1
|
||||||
|
|
||||||
|
### Features
|
||||||
|
- Eventually and Consistently that are passed a SpecContext can provide reports when an interrupt occurs [0d063c9]
|
||||||
|
|
||||||
|
## 1.21.0
|
||||||
|
|
||||||
|
### Features
|
||||||
|
- Eventually and Consistently can take a context.Context [65c01bc]
|
||||||
|
This enables integration with Ginkgo 2.3.0's interruptible nodes and node timeouts.
|
||||||
|
- Introduces Eventually.Within.ProbeEvery with tests and documentation (#591) [f633800]
|
||||||
|
- New BeKeyOf matcher with documentation and unit tests (#590) [fb586b3]
|
||||||
|
|
||||||
|
## Fixes
|
||||||
|
- Cover the entire gmeasure suite with leak detection [8c54344]
|
||||||
|
- Fix gmeasure leak [119d4ce]
|
||||||
|
- Ignore new Ginkgo ProgressSignal goroutine in gleak [ba548e2]
|
||||||
|
|
||||||
|
## Maintenance
|
||||||
|
|
||||||
|
- Fixes crashes on newer Ruby 3 installations by upgrading github-pages gem dependency (#596) [12469a0]
|
||||||
|
|
||||||
|
|
||||||
|
## 1.20.2
|
||||||
|
|
||||||
|
## Fixes
|
||||||
|
- label specs that rely on remote access; bump timeout on short-circuit test to make it less flaky [35eeadf]
|
||||||
|
- gexec: allow more headroom for SIGABRT-related unit tests (#581) [5b78f40]
|
||||||
|
- Enable reading from a closed gbytes.Buffer (#575) [061fd26]
|
||||||
|
|
||||||
|
## Maintenance
|
||||||
|
- Bump github.com/onsi/ginkgo/v2 from 2.1.5 to 2.1.6 (#583) [55d895b]
|
||||||
|
- Bump github.com/onsi/ginkgo/v2 from 2.1.4 to 2.1.5 (#582) [346de7c]
|
||||||
|
|
||||||
|
## 1.20.1
|
||||||
|
|
||||||
|
## Fixes
|
||||||
|
- fix false positive gleaks when using ginkgo -p (#577) [cb46517]
|
||||||
|
- Fix typos in gomega_dsl.go (#569) [5f71ed2]
|
||||||
|
- don't panic on Eventually(nil), fixing #555 (#567) [9d1186f]
|
||||||
|
- vet optional description args in assertions, fixing #560 (#566) [8e37808]
|
||||||
|
|
||||||
|
## Maintenance
|
||||||
|
- test: add new Go 1.19 to test matrix (#571) [40d7efe]
|
||||||
|
- Bump tzinfo from 1.2.9 to 1.2.10 in /docs (#564) [5f26371]
|
||||||
|
|
||||||
|
## 1.20.0
|
||||||
|
|
||||||
|
## Features
|
||||||
|
- New [`gleak`](https://onsi.github.io/gomega/#codegleakcode-finding-leaked-goroutines) experimental goroutine leak detection package! (#538) [85ba7bc]
|
||||||
|
- New `BeComparableTo` matcher(#546) that uses `gocmp` to make comparisons [e77ea75]
|
||||||
|
- New `HaveExistingField` matcher (#553) [fd130e1]
|
||||||
|
- Document how to wrap Gomega (#539) [56714a4]
|
||||||
|
|
||||||
|
## Fixes
|
||||||
|
- Support pointer receivers in HaveField; fixes #543 (#544) [8dab36e]
|
||||||
|
|
||||||
|
## Maintenance
|
||||||
|
- Bump various dependencies:
|
||||||
|
- Upgrade to yaml.v3 (#556) [f5a83b1]
|
||||||
|
- Bump github/codeql-action from 1 to 2 (#549) [52f5adf]
|
||||||
|
- Bump github.com/google/go-cmp from 0.5.7 to 0.5.8 (#551) [5f3942d]
|
||||||
|
- Bump nokogiri from 1.13.4 to 1.13.6 in /docs (#554) [eb4b4c2]
|
||||||
|
- Use latest ginkgo (#535) [1c29028]
|
||||||
|
- Bump nokogiri from 1.13.3 to 1.13.4 in /docs (#541) [1ce84d5]
|
||||||
|
- Bump actions/setup-go from 2 to 3 (#540) [755485e]
|
||||||
|
- Bump nokogiri from 1.12.5 to 1.13.3 in /docs (#522) [4fbb0dc]
|
||||||
|
- Bump actions/checkout from 2 to 3 (#526) [ac49202]
|
||||||
|
|
||||||
|
## 1.19.0
|
||||||
|
|
||||||
|
## Features
|
||||||
|
- New [`HaveEach`](https://onsi.github.io/gomega/#haveeachelement-interface) matcher to ensure that each and every element in an `array`, `slice`, or `map` satisfies the passed in matcher. (#523) [9fc2ae2] (#524) [c8ba582]
|
||||||
|
- Users can now wrap the `Gomega` interface to implement custom behavior on each assertion. (#521) [1f2e714]
|
||||||
|
- [`ContainElement`](https://onsi.github.io/gomega/#containelementelement-interface) now accepts an additional pointer argument. Elements that satisfy the matcher are stored in the pointer enabling developers to easily add subsequent, more detailed, assertions against the matching element. (#527) [1a4e27f]
|
||||||
|
|
||||||
|
## Fixes
|
||||||
|
- update RELEASING instructions to match ginkgo [0917cde]
|
||||||
|
- Bump github.com/onsi/ginkgo/v2 from 2.0.0 to 2.1.3 (#519) [49ab4b0]
|
||||||
|
- Fix CVE-2021-38561 (#534) [f1b4456]
|
||||||
|
- Fix max number of samples in experiments on non-64-bit systems. (#528) [1c84497]
|
||||||
|
- Remove dependency on ginkgo v1.16.4 (#530) [4dea8d5]
|
||||||
|
- Fix for Go 1.18 (#532) [56d2a29]
|
||||||
|
- Document precendence of timeouts (#533) [b607941]
|
||||||
|
|
||||||
|
## 1.18.1
|
||||||
|
|
||||||
|
## Fixes
|
||||||
|
- Add pointer support to HaveField matcher (#495) [79e41a3]
|
||||||
|
|
||||||
|
## 1.18.0
|
||||||
|
|
||||||
|
## Features
|
||||||
|
- Docs now live on the master branch in the docs folder which will make for easier PRs. The docs also use Ginkgo 2.0's new docs html/css/js. [2570272]
|
||||||
|
- New HaveValue matcher can handle actuals that are either values (in which case they are passed on unscathed) or pointers (in which case they are indirected). [Docs here.](https://onsi.github.io/gomega/#working-with-values) (#485) [bdc087c]
|
||||||
|
- Gmeasure has been declared GA [360db9d]
|
||||||
|
|
||||||
|
## Fixes
|
||||||
|
- Gomega now uses ioutil for Go 1.15 and lower (#492) - official support is only for the most recent two major versions of Go but this will unblock users who need to stay on older unsupported versions of Go. [c29c1c0]
|
||||||
|
|
||||||
|
## Maintenace
|
||||||
|
- Remove Travis workflow (#491) [72e6040]
|
||||||
|
- Upgrade to Ginkgo 2.0.0 GA [f383637]
|
||||||
|
- chore: fix description of HaveField matcher (#487) [2b4b2c0]
|
||||||
|
- use tools.go to ensure Ginkgo cli dependencies are included [f58a52b]
|
||||||
|
- remove dockerfile and simplify github actions to match ginkgo's actions [3f8160d]
|
||||||
|
|
||||||
|
## 1.17.0
|
||||||
|
|
||||||
|
### Features
|
||||||
|
- Add HaveField matcher [3a26311]
|
||||||
|
- add Error() assertions on the final error value of multi-return values (#480) [2f96943]
|
||||||
|
- separate out offsets and timeouts (#478) [18a4723]
|
||||||
|
- fix transformation error reporting (#479) [e001fab]
|
||||||
|
- allow transform functions to report errors (#472) [bf93408]
|
||||||
|
|
||||||
|
### Fixes
|
||||||
|
Stop using deprecated ioutil package (#467) [07f405d]
|
||||||
|
|
||||||
|
## 1.16.0
|
||||||
|
|
||||||
|
### Features
|
||||||
|
- feat: HaveHTTPStatus multiple expected values (#465) [aa69f1b]
|
||||||
|
- feat: HaveHTTPHeaderWithValue() matcher (#463) [dd83a96]
|
||||||
|
- feat: HaveHTTPBody matcher (#462) [504e1f2]
|
||||||
|
- feat: formatter for HTTP responses (#461) [e5b3157]
|
||||||
|
|
||||||
|
## 1.15.0
|
||||||
|
|
||||||
|
### Fixes
|
||||||
|
The previous version (1.14.0) introduced a change to allow `Eventually` and `Consistently` to support functions that make assertions. This was accomplished by overriding the global fail handler when running the callbacks passed to `Eventually/Consistently` in order to capture any resulting errors. Issue #457 uncovered a flaw with this approach: when multiple `Eventually`s are running concurrently they race when overriding the singleton global fail handler.
|
||||||
|
|
||||||
|
1.15.0 resolves this by requiring users who want to make assertions in `Eventually/Consistently` call backs to explicitly pass in a function that takes a `Gomega` as an argument. The passed-in `Gomega` instance can be used to make assertions. Any failures will cause `Eventually` to retry the callback. This cleaner interface avoids the issue of swapping out globals but comes at the cost of changing the contract introduced in v1.14.0. As such 1.15.0 introduces a breaking change with respect to 1.14.0 - however we expect that adoption of this feature in 1.14.0 remains limited.
|
||||||
|
|
||||||
|
In addition, 1.15.0 cleans up some of Gomega's internals. Most users shouldn't notice any differences stemming from the refactoring that was made.
|
||||||
|
|
||||||
|
## 1.14.0
|
||||||
|
|
||||||
|
### Features
|
||||||
|
- gmeasure.SamplingConfig now suppers a MinSamplingInterval [e94dbca]
|
||||||
|
- Eventually and Consistently support functions that make assertions [2f04e6e]
|
||||||
|
- Eventually and Consistently now allow their passed-in functions to make assertions.
|
||||||
|
These assertions must pass or the function is considered to have failed and is retried.
|
||||||
|
- Eventually and Consistently can now take functions with no return values. These implicitly return nil
|
||||||
|
if they contain no failed assertion. Otherwise they return an error wrapping the first assertion failure. This allows
|
||||||
|
these functions to be used with the Succeed() matcher.
|
||||||
|
- Introduce InterceptGomegaFailure - an analogue to InterceptGomegaFailures - that captures the first assertion failure
|
||||||
|
and halts execution in its passed-in callback.
|
||||||
|
|
||||||
|
### Fixes
|
||||||
|
- Call Verify GHTTPWithGomega receiver funcs (#454) [496e6fd]
|
||||||
|
- Build a binary with an expected name (#446) [7356360]
|
||||||
|
|
||||||
|
## 1.13.0
|
||||||
|
|
||||||
|
### Features
|
||||||
|
- gmeasure provides BETA support for benchmarking (#447) [8f2dfbf]
|
||||||
|
- Set consistently and eventually defaults on init (#443) [12eb778]
|
||||||
|
|
||||||
|
## 1.12.0
|
||||||
|
|
||||||
|
### Features
|
||||||
|
- Add Satisfy() matcher (#437) [c548f31]
|
||||||
|
- tweak truncation message [3360b8c]
|
||||||
|
- Add format.GomegaStringer (#427) [cc80b6f]
|
||||||
|
- Add Clear() method to gbytes.Buffer [c3c0920]
|
||||||
|
|
||||||
|
### Fixes
|
||||||
|
- Fix error message in BeNumericallyMatcher (#432) [09c074a]
|
||||||
|
- Bump github.com/onsi/ginkgo from 1.12.1 to 1.16.2 (#442) [e5f6ea0]
|
||||||
|
- Bump github.com/golang/protobuf from 1.4.3 to 1.5.2 (#431) [adae3bf]
|
||||||
|
- Bump golang.org/x/net (#441) [3275b35]
|
||||||
|
|
||||||
|
## 1.11.0
|
||||||
|
|
||||||
|
### Features
|
||||||
|
- feature: add index to gstruct element func (#419) [334e00d]
|
||||||
|
- feat(gexec) Add CompileTest functions. Close #410 (#411) [47c613f]
|
||||||
|
|
||||||
|
### Fixes
|
||||||
|
- Check more carefully for nils in WithTransform (#423) [3c60a15]
|
||||||
|
- fix: typo in Makefile [b82522a]
|
||||||
|
- Allow WithTransform function to accept a nil value (#422) [b75d2f2]
|
||||||
|
- fix: print value type for interface{} containers (#409) [f08e2dc]
|
||||||
|
- fix(BeElementOf): consistently flatten expected values [1fa9468]
|
||||||
|
|
||||||
|
## 1.10.5
|
||||||
|
|
||||||
|
### Fixes
|
||||||
|
- fix: collections matchers should display type of expectation (#408) [6b4eb5a]
|
||||||
|
- fix(ContainElements): consistently flatten expected values [073b880]
|
||||||
|
- fix(ConsistOf): consistently flatten expected values [7266efe]
|
||||||
|
|
||||||
|
## 1.10.4
|
||||||
|
|
||||||
|
### Fixes
|
||||||
|
- update golang net library to more recent version without vulnerability (#406) [817a8b9]
|
||||||
|
- Correct spelling: alloted -> allotted (#403) [0bae715]
|
||||||
|
- fix a panic in MessageWithDiff with long message (#402) [ea06b9b]
|
||||||
|
|
||||||
|
## 1.10.3
|
||||||
|
|
||||||
|
### Fixes
|
||||||
|
- updates golang/x/net to fix vulnerability detected by snyk (#394) [c479356]
|
||||||
|
|
||||||
|
## 1.10.2
|
||||||
|
|
||||||
|
### Fixes
|
||||||
|
- Add ExpectWithOffset, EventuallyWithOffset and ConsistentlyWithOffset to WithT (#391) [990941a]
|
||||||
|
|
||||||
|
## 1.10.1
|
||||||
|
|
||||||
|
### Fixes
|
||||||
|
- Update dependencies (#389) [9f5eecd]
|
||||||
|
|
||||||
|
## 1.10.0
|
||||||
|
|
||||||
|
### Features
|
||||||
|
- Add HaveHTTPStatusMatcher (#378) [f335c94]
|
||||||
|
- Changed matcher for content-type in VerifyJSONRepresenting (#377) [6024f5b]
|
||||||
|
- Make ghttp usable with x-unit style tests (#376) [c0be499]
|
||||||
|
- Implement PanicWith matcher (#381) [f8032b4]
|
||||||
|
|
||||||
|
## 1.9.0
|
||||||
|
|
||||||
|
### Features
|
||||||
|
- Add ContainElements matcher (#370) [2f57380]
|
||||||
|
- Output missing and extra elements in ConsistOf failure message [a31eda7]
|
||||||
|
- Document method LargestMatching [7c5a280]
|
||||||
|
|
||||||
|
## 1.8.1
|
||||||
|
|
||||||
|
### Fixes
|
||||||
|
- Fix unexpected MatchError() behaviour (#375) [8ae7b2f]
|
||||||
|
|
||||||
|
## 1.8.0
|
||||||
|
|
||||||
|
### Features
|
||||||
|
- Allow optional description to be lazily evaluated function (#364) [bf64010]
|
||||||
|
- Support wrapped errors (#359) [0a981cb]
|
||||||
|
|
||||||
|
## 1.7.1
|
||||||
|
|
||||||
|
### Fixes
|
||||||
|
- Bump go-yaml version to cover fixed ddos heuristic (#362) [95e431e]
|
||||||
|
|
||||||
|
## 1.7.0
|
||||||
|
|
||||||
|
### Features
|
||||||
|
- export format property variables (#347) [642e5ba]
|
||||||
|
|
||||||
|
### Fixes
|
||||||
|
- minor fix in the documentation of ExpectWithOffset (#358) [beea727]
|
||||||
|
|
||||||
|
## 1.6.0
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
- Display special chars on error [41e1b26]
|
||||||
|
- Add BeElementOf matcher [6a48b48]
|
||||||
|
|
||||||
|
### Fixes
|
||||||
|
|
||||||
|
- Remove duplication in XML matcher tests [cc1a6cb]
|
||||||
|
- Remove unnecessary conversions (#357) [7bf756a]
|
||||||
|
- Fixed import order (#353) [2e3b965]
|
||||||
|
- Added missing error handling in test (#355) [c98d3eb]
|
||||||
|
- Simplify code (#356) [0001ed9]
|
||||||
|
- Simplify code (#354) [0d9100e]
|
||||||
|
- Fixed typos (#352) [3f647c4]
|
||||||
|
- Add failure message tests to BeElementOf matcher [efe19c3]
|
||||||
|
- Update go-testcov untested sections [37ee382]
|
||||||
|
- Mark all uncovered files so go-testcov ./... works [53b150e]
|
||||||
|
- Reenable gotip in travis [5c249dc]
|
||||||
|
- Fix the typo of comment (#345) [f0e010e]
|
||||||
|
- Optimize contain_element_matcher [abeb93d]
|
||||||
|
|
||||||
|
|
||||||
|
## 1.5.0
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
- Added MatchKeys matchers [8b909fc]
|
||||||
|
|
||||||
|
### Fixes and Minor Improvements
|
||||||
|
|
||||||
|
- Add type aliases to remove stuttering [03b0461]
|
||||||
|
- Don't run session_test.go on windows (#324) [5533ce8]
|
||||||
|
|
||||||
|
## 1.4.3
|
||||||
|
|
||||||
|
### Fixes:
|
||||||
|
|
||||||
|
- ensure file name and line numbers are correctly reported for XUnit [6fff58f]
|
||||||
|
- Fixed matcher for content-type (#305) [69d9b43]
|
||||||
|
|
||||||
|
## 1.4.2
|
||||||
|
|
||||||
|
### Fixes:
|
||||||
|
|
||||||
|
- Add go.mod and go.sum files to define the gomega go module [f3de367, a085d30]
|
||||||
|
- Work around go vet issue with Go v1.11 (#300) [40dd6ad]
|
||||||
|
- Better output when using with go XUnit-style tests, fixes #255 (#297) [29a4b97]
|
||||||
|
- Fix MatchJSON fail to parse json.RawMessage (#298) [ae19f1b]
|
||||||
|
- show threshold in failure message of BeNumericallyMatcher (#293) [4bbecc8]
|
||||||
|
|
||||||
|
## 1.4.1
|
||||||
|
|
||||||
|
### Fixes:
|
||||||
|
|
||||||
|
- Update documentation formatting and examples (#289) [9be8410]
|
||||||
|
- allow 'Receive' matcher to be used with concrete types (#286) [41673fd]
|
||||||
|
- Fix data race in ghttp server (#283) [7ac6b01]
|
||||||
|
- Travis badge should only show master [cc102ab]
|
||||||
|
|
||||||
|
## 1.4.0
|
||||||
|
|
||||||
|
### Features
|
||||||
|
- Make string pretty diff user configurable (#273) [eb112ce, 649b44d]
|
||||||
|
|
||||||
|
### Fixes
|
||||||
|
- Use httputil.DumpRequest to pretty-print unhandled requests (#278) [a4ff0fc, b7d1a52]
|
||||||
|
- fix typo floa32 > float32 (#272) [041ae3b, 6e33911]
|
||||||
|
- Fix link to documentation on adding your own matchers (#270) [bb2c830, fcebc62]
|
||||||
|
- Use setters and getters to avoid race condition (#262) [13057c3, a9c79f1]
|
||||||
|
- Avoid sending a signal if the process is not alive (#259) [b8043e5, 4fc1762]
|
||||||
|
- Improve message from AssignableToTypeOf when expected value is nil (#281) [9c1fb20]
|
||||||
|
|
||||||
|
## 1.3.0
|
||||||
|
|
||||||
|
Improvements:
|
||||||
|
|
||||||
|
- The `Equal` matcher matches byte slices more performantly.
|
||||||
|
- Improved how `MatchError` matches error strings.
|
||||||
|
- `MatchXML` ignores the order of xml node attributes.
|
||||||
|
- Improve support for XUnit style golang tests. ([#254](https://github.com/onsi/gomega/issues/254))
|
||||||
|
|
||||||
|
Bug Fixes:
|
||||||
|
|
||||||
|
- Diff generation now handles multi-byte sequences correctly.
|
||||||
|
- Multiple goroutines can now call `gexec.Build` concurrently.
|
||||||
|
|
||||||
|
## 1.2.0
|
||||||
|
|
||||||
|
Improvements:
|
||||||
|
|
||||||
|
- Added `BeSent` which attempts to send a value down a channel and fails if the attempt blocks. Can be paired with `Eventually` to safely send a value down a channel with a timeout.
|
||||||
|
- `Ω`, `Expect`, `Eventually`, and `Consistently` now immediately `panic` if there is no registered fail handler. This is always a mistake that can hide failing tests.
|
||||||
|
- `Receive()` no longer errors when passed a closed channel, it's perfectly fine to attempt to read from a closed channel so Ω(c).Should(Receive()) always fails and Ω(c).ShoudlNot(Receive()) always passes with a closed channel.
|
||||||
|
- Added `HavePrefix` and `HaveSuffix` matchers.
|
||||||
|
- `ghttp` can now handle concurrent requests.
|
||||||
|
- Added `Succeed` which allows one to write `Ω(MyFunction()).Should(Succeed())`.
|
||||||
|
- Improved `ghttp`'s behavior around failing assertions and panics:
|
||||||
|
- If a registered handler makes a failing assertion `ghttp` will return `500`.
|
||||||
|
- If a registered handler panics, `ghttp` will return `500` *and* fail the test. This is new behavior that may cause existing code to break. This code is almost certainly incorrect and creating a false positive.
|
||||||
|
- `ghttp` servers can take an `io.Writer`. `ghttp` will write a line to the writer when each request arrives.
|
||||||
|
- Added `WithTransform` matcher to allow munging input data before feeding into the relevant matcher
|
||||||
|
- Added boolean `And`, `Or`, and `Not` matchers to allow creating composite matchers
|
||||||
|
- Added `gbytes.TimeoutCloser`, `gbytes.TimeoutReader`, and `gbytes.TimeoutWriter` - these are convenience wrappers that timeout if the underlying Closer/Reader/Writer does not return within the alloted time.
|
||||||
|
- Added `gbytes.BufferReader` - this constructs a `gbytes.Buffer` that asynchronously reads the passed-in `io.Reader` into its buffer.
|
||||||
|
|
||||||
|
Bug Fixes:
|
||||||
|
- gexec: `session.Wait` now uses `EventuallyWithOffset` to get the right line number in the failure.
|
||||||
|
- `ContainElement` no longer bails if a passed-in matcher errors.
|
||||||
|
|
||||||
|
## 1.0 (8/2/2014)
|
||||||
|
|
||||||
|
No changes. Dropping "beta" from the version number.
|
||||||
|
|
||||||
|
## 1.0.0-beta (7/8/2014)
|
||||||
|
Breaking Changes:
|
||||||
|
|
||||||
|
- Changed OmegaMatcher interface. Instead of having `Match` return failure messages, two new methods `FailureMessage` and `NegatedFailureMessage` are called instead.
|
||||||
|
- Moved and renamed OmegaFailHandler to types.GomegaFailHandler and OmegaMatcher to types.GomegaMatcher. Any references to OmegaMatcher in any custom matchers will need to be changed to point to types.GomegaMatcher
|
||||||
|
|
||||||
|
New Test-Support Features:
|
||||||
|
|
||||||
|
- `ghttp`: supports testing http clients
|
||||||
|
- Provides a flexible fake http server
|
||||||
|
- Provides a collection of chainable http handlers that perform assertions.
|
||||||
|
- `gbytes`: supports making ordered assertions against streams of data
|
||||||
|
- Provides a `gbytes.Buffer`
|
||||||
|
- Provides a `Say` matcher to perform ordered assertions against output data
|
||||||
|
- `gexec`: supports testing external processes
|
||||||
|
- Provides support for building Go binaries
|
||||||
|
- Wraps and starts `exec.Cmd` commands
|
||||||
|
- Makes it easy to assert against stdout and stderr
|
||||||
|
- Makes it easy to send signals and wait for processes to exit
|
||||||
|
- Provides an `Exit` matcher to assert against exit code.
|
||||||
|
|
||||||
|
DSL Changes:
|
||||||
|
|
||||||
|
- `Eventually` and `Consistently` can accept `time.Duration` interval and polling inputs.
|
||||||
|
- The default timeouts for `Eventually` and `Consistently` are now configurable.
|
||||||
|
|
||||||
|
New Matchers:
|
||||||
|
|
||||||
|
- `ConsistOf`: order-independent assertion against the elements of an array/slice or keys of a map.
|
||||||
|
- `BeTemporally`: like `BeNumerically` but for `time.Time`
|
||||||
|
- `HaveKeyWithValue`: asserts a map has a given key with the given value.
|
||||||
|
|
||||||
|
Updated Matchers:
|
||||||
|
|
||||||
|
- `Receive` matcher can take a matcher as an argument and passes only if the channel under test receives an objet that satisfies the passed-in matcher.
|
||||||
|
- Matchers that implement `MatchMayChangeInTheFuture(actual interface{}) bool` can inform `Eventually` and/or `Consistently` when a match has no chance of changing status in the future. For example, `Receive` returns `false` when a channel is closed.
|
||||||
|
|
||||||
|
Misc:
|
||||||
|
|
||||||
|
- Start using semantic versioning
|
||||||
|
- Start maintaining changelog
|
||||||
|
|
||||||
|
Major refactor:
|
||||||
|
|
||||||
|
- Pull out Gomega's internal to `internal`
|
||||||
14
vendor/github.com/onsi/gomega/CONTRIBUTING.md
generated
vendored
Normal file
14
vendor/github.com/onsi/gomega/CONTRIBUTING.md
generated
vendored
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
# Contributing to Gomega
|
||||||
|
|
||||||
|
Your contributions to Gomega are essential for its long-term maintenance and improvement. To make a contribution:
|
||||||
|
|
||||||
|
- Please **open an issue first** - describe what problem you are trying to solve and give the community a forum for input and feedback ahead of investing time in writing code!
|
||||||
|
- Ensure adequate test coverage:
|
||||||
|
- Make sure to add appropriate unit tests
|
||||||
|
- Please run all tests locally (`ginkgo -r -p`) and make sure they go green before submitting the PR
|
||||||
|
- Please run following linter locally `go vet ./...` and make sure output does not contain any warnings
|
||||||
|
- Update the documentation. In addition to standard `godoc` comments Gomega has extensive documentation on the `gh-pages` branch. If relevant, please submit a docs PR to that branch alongside your code PR.
|
||||||
|
|
||||||
|
If you're a committer, check out RELEASING.md to learn how to cut a release.
|
||||||
|
|
||||||
|
Thanks for supporting Gomega!
|
||||||
20
vendor/github.com/onsi/gomega/LICENSE
generated
vendored
Normal file
20
vendor/github.com/onsi/gomega/LICENSE
generated
vendored
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
Copyright (c) 2013-2014 Onsi Fakhouri
|
||||||
|
|
||||||
|
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.
|
||||||
21
vendor/github.com/onsi/gomega/README.md
generated
vendored
Normal file
21
vendor/github.com/onsi/gomega/README.md
generated
vendored
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|

|
||||||
|
|
||||||
|
[](https://github.com/onsi/gomega/actions/workflows/test.yml)
|
||||||
|
|
||||||
|
Jump straight to the [docs](http://onsi.github.io/gomega/) to learn about Gomega, including a list of [all available matchers](http://onsi.github.io/gomega/#provided-matchers).
|
||||||
|
|
||||||
|
If you have a question, comment, bug report, feature request, etc. please open a GitHub issue.
|
||||||
|
|
||||||
|
## [Ginkgo](http://github.com/onsi/ginkgo): a BDD Testing Framework for Golang
|
||||||
|
|
||||||
|
Learn more about Ginkgo [here](http://onsi.github.io/ginkgo/)
|
||||||
|
|
||||||
|
## Community Matchers
|
||||||
|
|
||||||
|
A collection of community matchers is available on the [wiki](https://github.com/onsi/gomega/wiki).
|
||||||
|
|
||||||
|
## License
|
||||||
|
|
||||||
|
Gomega is MIT-Licensed
|
||||||
|
|
||||||
|
The `ConsistOf` matcher uses [goraph](https://github.com/amitkgupta/goraph) which is embedded in the source to simplify distribution. goraph has an MIT license.
|
||||||
23
vendor/github.com/onsi/gomega/RELEASING.md
generated
vendored
Normal file
23
vendor/github.com/onsi/gomega/RELEASING.md
generated
vendored
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
A Gomega release is a tagged sha and a GitHub release. To cut a release:
|
||||||
|
|
||||||
|
1. Ensure CHANGELOG.md is up to date.
|
||||||
|
- Use
|
||||||
|
```bash
|
||||||
|
LAST_VERSION=$(git tag --sort=version:refname | tail -n1)
|
||||||
|
CHANGES=$(git log --pretty=format:'- %s [%h]' HEAD...$LAST_VERSION)
|
||||||
|
echo -e "## NEXT\n\n$CHANGES\n\n### Features\n\n### Fixes\n\n### Maintenance\n\n$(cat CHANGELOG.md)" > CHANGELOG.md
|
||||||
|
```
|
||||||
|
to update the changelog
|
||||||
|
- Categorize the changes into
|
||||||
|
- Breaking Changes (requires a major version)
|
||||||
|
- New Features (minor version)
|
||||||
|
- Fixes (fix version)
|
||||||
|
- Maintenance (which in general should not be mentioned in `CHANGELOG.md` as they have no user impact)
|
||||||
|
1. Update GOMEGA_VERSION in `gomega_dsl.go`
|
||||||
|
1. Commit, push, and release:
|
||||||
|
```
|
||||||
|
git commit -m "vM.m.p"
|
||||||
|
git push
|
||||||
|
gh release create "vM.m.p"
|
||||||
|
git fetch --tags origin master
|
||||||
|
```
|
||||||
506
vendor/github.com/onsi/gomega/format/format.go
generated
vendored
Normal file
506
vendor/github.com/onsi/gomega/format/format.go
generated
vendored
Normal file
@@ -0,0 +1,506 @@
|
|||||||
|
/*
|
||||||
|
Gomega's format package pretty-prints objects. It explores input objects recursively and generates formatted, indented output with type information.
|
||||||
|
*/
|
||||||
|
|
||||||
|
// untested sections: 4
|
||||||
|
|
||||||
|
package format
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
"reflect"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Use MaxDepth to set the maximum recursion depth when printing deeply nested objects
|
||||||
|
var MaxDepth = uint(10)
|
||||||
|
|
||||||
|
// MaxLength of the string representation of an object.
|
||||||
|
// If MaxLength is set to 0, the Object will not be truncated.
|
||||||
|
var MaxLength = 4000
|
||||||
|
|
||||||
|
/*
|
||||||
|
By default, all objects (even those that implement fmt.Stringer and fmt.GoStringer) are recursively inspected to generate output.
|
||||||
|
|
||||||
|
Set UseStringerRepresentation = true to use GoString (for fmt.GoStringers) or String (for fmt.Stringer) instead.
|
||||||
|
|
||||||
|
Note that GoString and String don't always have all the information you need to understand why a test failed!
|
||||||
|
*/
|
||||||
|
var UseStringerRepresentation = false
|
||||||
|
|
||||||
|
/*
|
||||||
|
Print the content of context objects. By default it will be suppressed.
|
||||||
|
|
||||||
|
Set PrintContextObjects = true to enable printing of the context internals.
|
||||||
|
*/
|
||||||
|
var PrintContextObjects = false
|
||||||
|
|
||||||
|
// TruncatedDiff choose if we should display a truncated pretty diff or not
|
||||||
|
var TruncatedDiff = true
|
||||||
|
|
||||||
|
// TruncateThreshold (default 50) specifies the maximum length string to print in string comparison assertion error
|
||||||
|
// messages.
|
||||||
|
var TruncateThreshold uint = 50
|
||||||
|
|
||||||
|
// CharactersAroundMismatchToInclude (default 5) specifies how many contextual characters should be printed before and
|
||||||
|
// after the first diff location in a truncated string assertion error message.
|
||||||
|
var CharactersAroundMismatchToInclude uint = 5
|
||||||
|
|
||||||
|
var contextType = reflect.TypeOf((*context.Context)(nil)).Elem()
|
||||||
|
var timeType = reflect.TypeOf(time.Time{})
|
||||||
|
|
||||||
|
// The default indentation string emitted by the format package
|
||||||
|
var Indent = " "
|
||||||
|
|
||||||
|
var longFormThreshold = 20
|
||||||
|
|
||||||
|
// GomegaStringer allows for custom formating of objects for gomega.
|
||||||
|
type GomegaStringer interface {
|
||||||
|
// GomegaString will be used to custom format an object.
|
||||||
|
// It does not follow UseStringerRepresentation value and will always be called regardless.
|
||||||
|
// It also ignores the MaxLength value.
|
||||||
|
GomegaString() string
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
CustomFormatters can be registered with Gomega via RegisterCustomFormatter()
|
||||||
|
Any value to be rendered by Gomega is passed to each registered CustomFormatters.
|
||||||
|
The CustomFormatter signals that it will handle formatting the value by returning (formatted-string, true)
|
||||||
|
If the CustomFormatter does not want to handle the object it should return ("", false)
|
||||||
|
|
||||||
|
Strings returned by CustomFormatters are not truncated
|
||||||
|
*/
|
||||||
|
type CustomFormatter func(value interface{}) (string, bool)
|
||||||
|
type CustomFormatterKey uint
|
||||||
|
|
||||||
|
var customFormatterKey CustomFormatterKey = 1
|
||||||
|
|
||||||
|
type customFormatterKeyPair struct {
|
||||||
|
CustomFormatter
|
||||||
|
CustomFormatterKey
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
RegisterCustomFormatter registers a CustomFormatter and returns a CustomFormatterKey
|
||||||
|
|
||||||
|
You can call UnregisterCustomFormatter with the returned key to unregister the associated CustomFormatter
|
||||||
|
*/
|
||||||
|
func RegisterCustomFormatter(customFormatter CustomFormatter) CustomFormatterKey {
|
||||||
|
key := customFormatterKey
|
||||||
|
customFormatterKey += 1
|
||||||
|
customFormatters = append(customFormatters, customFormatterKeyPair{customFormatter, key})
|
||||||
|
return key
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
UnregisterCustomFormatter unregisters a previously registered CustomFormatter. You should pass in the key returned by RegisterCustomFormatter
|
||||||
|
*/
|
||||||
|
func UnregisterCustomFormatter(key CustomFormatterKey) {
|
||||||
|
formatters := []customFormatterKeyPair{}
|
||||||
|
for _, f := range customFormatters {
|
||||||
|
if f.CustomFormatterKey == key {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
formatters = append(formatters, f)
|
||||||
|
}
|
||||||
|
customFormatters = formatters
|
||||||
|
}
|
||||||
|
|
||||||
|
var customFormatters = []customFormatterKeyPair{}
|
||||||
|
|
||||||
|
/*
|
||||||
|
Generates a formatted matcher success/failure message of the form:
|
||||||
|
|
||||||
|
Expected
|
||||||
|
<pretty printed actual>
|
||||||
|
<message>
|
||||||
|
<pretty printed expected>
|
||||||
|
|
||||||
|
If expected is omitted, then the message looks like:
|
||||||
|
|
||||||
|
Expected
|
||||||
|
<pretty printed actual>
|
||||||
|
<message>
|
||||||
|
*/
|
||||||
|
func Message(actual interface{}, message string, expected ...interface{}) string {
|
||||||
|
if len(expected) == 0 {
|
||||||
|
return fmt.Sprintf("Expected\n%s\n%s", Object(actual, 1), message)
|
||||||
|
}
|
||||||
|
return fmt.Sprintf("Expected\n%s\n%s\n%s", Object(actual, 1), message, Object(expected[0], 1))
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
|
||||||
|
Generates a nicely formatted matcher success / failure message
|
||||||
|
|
||||||
|
Much like Message(...), but it attempts to pretty print diffs in strings
|
||||||
|
|
||||||
|
Expected
|
||||||
|
<string>: "...aaaaabaaaaa..."
|
||||||
|
to equal |
|
||||||
|
<string>: "...aaaaazaaaaa..."
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
func MessageWithDiff(actual, message, expected string) string {
|
||||||
|
if TruncatedDiff && len(actual) >= int(TruncateThreshold) && len(expected) >= int(TruncateThreshold) {
|
||||||
|
diffPoint := findFirstMismatch(actual, expected)
|
||||||
|
formattedActual := truncateAndFormat(actual, diffPoint)
|
||||||
|
formattedExpected := truncateAndFormat(expected, diffPoint)
|
||||||
|
|
||||||
|
spacesBeforeFormattedMismatch := findFirstMismatch(formattedActual, formattedExpected)
|
||||||
|
|
||||||
|
tabLength := 4
|
||||||
|
spaceFromMessageToActual := tabLength + len("<string>: ") - len(message)
|
||||||
|
|
||||||
|
paddingCount := spaceFromMessageToActual + spacesBeforeFormattedMismatch
|
||||||
|
if paddingCount < 0 {
|
||||||
|
return Message(formattedActual, message, formattedExpected)
|
||||||
|
}
|
||||||
|
|
||||||
|
padding := strings.Repeat(" ", paddingCount) + "|"
|
||||||
|
return Message(formattedActual, message+padding, formattedExpected)
|
||||||
|
}
|
||||||
|
|
||||||
|
actual = escapedWithGoSyntax(actual)
|
||||||
|
expected = escapedWithGoSyntax(expected)
|
||||||
|
|
||||||
|
return Message(actual, message, expected)
|
||||||
|
}
|
||||||
|
|
||||||
|
func escapedWithGoSyntax(str string) string {
|
||||||
|
withQuotes := fmt.Sprintf("%q", str)
|
||||||
|
return withQuotes[1 : len(withQuotes)-1]
|
||||||
|
}
|
||||||
|
|
||||||
|
func truncateAndFormat(str string, index int) string {
|
||||||
|
leftPadding := `...`
|
||||||
|
rightPadding := `...`
|
||||||
|
|
||||||
|
start := index - int(CharactersAroundMismatchToInclude)
|
||||||
|
if start < 0 {
|
||||||
|
start = 0
|
||||||
|
leftPadding = ""
|
||||||
|
}
|
||||||
|
|
||||||
|
// slice index must include the mis-matched character
|
||||||
|
lengthOfMismatchedCharacter := 1
|
||||||
|
end := index + int(CharactersAroundMismatchToInclude) + lengthOfMismatchedCharacter
|
||||||
|
if end > len(str) {
|
||||||
|
end = len(str)
|
||||||
|
rightPadding = ""
|
||||||
|
|
||||||
|
}
|
||||||
|
return fmt.Sprintf("\"%s\"", leftPadding+str[start:end]+rightPadding)
|
||||||
|
}
|
||||||
|
|
||||||
|
func findFirstMismatch(a, b string) int {
|
||||||
|
aSlice := strings.Split(a, "")
|
||||||
|
bSlice := strings.Split(b, "")
|
||||||
|
|
||||||
|
for index, str := range aSlice {
|
||||||
|
if index > len(bSlice)-1 {
|
||||||
|
return index
|
||||||
|
}
|
||||||
|
if str != bSlice[index] {
|
||||||
|
return index
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(b) > len(a) {
|
||||||
|
return len(a) + 1
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
const truncateHelpText = `
|
||||||
|
Gomega truncated this representation as it exceeds 'format.MaxLength'.
|
||||||
|
Consider having the object provide a custom 'GomegaStringer' representation
|
||||||
|
or adjust the parameters in Gomega's 'format' package.
|
||||||
|
|
||||||
|
Learn more here: https://onsi.github.io/gomega/#adjusting-output
|
||||||
|
`
|
||||||
|
|
||||||
|
func truncateLongStrings(s string) string {
|
||||||
|
if MaxLength > 0 && len(s) > MaxLength {
|
||||||
|
var sb strings.Builder
|
||||||
|
for i, r := range s {
|
||||||
|
if i < MaxLength {
|
||||||
|
sb.WriteRune(r)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
sb.WriteString("...\n")
|
||||||
|
sb.WriteString(truncateHelpText)
|
||||||
|
|
||||||
|
return sb.String()
|
||||||
|
}
|
||||||
|
return s
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
Pretty prints the passed in object at the passed in indentation level.
|
||||||
|
|
||||||
|
Object recurses into deeply nested objects emitting pretty-printed representations of their components.
|
||||||
|
|
||||||
|
Modify format.MaxDepth to control how deep the recursion is allowed to go
|
||||||
|
Set format.UseStringerRepresentation to true to return object.GoString() or object.String() when available instead of
|
||||||
|
recursing into the object.
|
||||||
|
|
||||||
|
Set PrintContextObjects to true to print the content of objects implementing context.Context
|
||||||
|
*/
|
||||||
|
func Object(object interface{}, indentation uint) string {
|
||||||
|
indent := strings.Repeat(Indent, int(indentation))
|
||||||
|
value := reflect.ValueOf(object)
|
||||||
|
commonRepresentation := ""
|
||||||
|
if err, ok := object.(error); ok {
|
||||||
|
commonRepresentation += "\n" + IndentString(err.Error(), indentation) + "\n" + indent
|
||||||
|
}
|
||||||
|
return fmt.Sprintf("%s<%s>: %s%s", indent, formatType(value), commonRepresentation, formatValue(value, indentation))
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
IndentString takes a string and indents each line by the specified amount.
|
||||||
|
*/
|
||||||
|
func IndentString(s string, indentation uint) string {
|
||||||
|
return indentString(s, indentation, true)
|
||||||
|
}
|
||||||
|
|
||||||
|
func indentString(s string, indentation uint, indentFirstLine bool) string {
|
||||||
|
result := &strings.Builder{}
|
||||||
|
components := strings.Split(s, "\n")
|
||||||
|
indent := strings.Repeat(Indent, int(indentation))
|
||||||
|
for i, component := range components {
|
||||||
|
if i > 0 || indentFirstLine {
|
||||||
|
result.WriteString(indent)
|
||||||
|
}
|
||||||
|
result.WriteString(component)
|
||||||
|
if i < len(components)-1 {
|
||||||
|
result.WriteString("\n")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return result.String()
|
||||||
|
}
|
||||||
|
|
||||||
|
func formatType(v reflect.Value) string {
|
||||||
|
switch v.Kind() {
|
||||||
|
case reflect.Invalid:
|
||||||
|
return "nil"
|
||||||
|
case reflect.Chan:
|
||||||
|
return fmt.Sprintf("%s | len:%d, cap:%d", v.Type(), v.Len(), v.Cap())
|
||||||
|
case reflect.Ptr:
|
||||||
|
return fmt.Sprintf("%s | 0x%x", v.Type(), v.Pointer())
|
||||||
|
case reflect.Slice:
|
||||||
|
return fmt.Sprintf("%s | len:%d, cap:%d", v.Type(), v.Len(), v.Cap())
|
||||||
|
case reflect.Map:
|
||||||
|
return fmt.Sprintf("%s | len:%d", v.Type(), v.Len())
|
||||||
|
default:
|
||||||
|
return fmt.Sprintf("%s", v.Type())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func formatValue(value reflect.Value, indentation uint) string {
|
||||||
|
if indentation > MaxDepth {
|
||||||
|
return "..."
|
||||||
|
}
|
||||||
|
|
||||||
|
if isNilValue(value) {
|
||||||
|
return "nil"
|
||||||
|
}
|
||||||
|
|
||||||
|
if value.CanInterface() {
|
||||||
|
obj := value.Interface()
|
||||||
|
|
||||||
|
// if a CustomFormatter handles this values, we'll go with that
|
||||||
|
for _, customFormatter := range customFormatters {
|
||||||
|
formatted, handled := customFormatter.CustomFormatter(obj)
|
||||||
|
// do not truncate a user-provided CustomFormatter()
|
||||||
|
if handled {
|
||||||
|
return indentString(formatted, indentation+1, false)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// GomegaStringer will take precedence to other representations and disregards UseStringerRepresentation
|
||||||
|
if x, ok := obj.(GomegaStringer); ok {
|
||||||
|
// do not truncate a user-defined GomegaString() value
|
||||||
|
return indentString(x.GomegaString(), indentation+1, false)
|
||||||
|
}
|
||||||
|
|
||||||
|
if UseStringerRepresentation {
|
||||||
|
switch x := obj.(type) {
|
||||||
|
case fmt.GoStringer:
|
||||||
|
return indentString(truncateLongStrings(x.GoString()), indentation+1, false)
|
||||||
|
case fmt.Stringer:
|
||||||
|
return indentString(truncateLongStrings(x.String()), indentation+1, false)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if !PrintContextObjects {
|
||||||
|
if value.Type().Implements(contextType) && indentation > 1 {
|
||||||
|
return "<suppressed context>"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
switch value.Kind() {
|
||||||
|
case reflect.Bool:
|
||||||
|
return fmt.Sprintf("%v", value.Bool())
|
||||||
|
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
|
||||||
|
return fmt.Sprintf("%v", value.Int())
|
||||||
|
case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64:
|
||||||
|
return fmt.Sprintf("%v", value.Uint())
|
||||||
|
case reflect.Uintptr:
|
||||||
|
return fmt.Sprintf("0x%x", value.Uint())
|
||||||
|
case reflect.Float32, reflect.Float64:
|
||||||
|
return fmt.Sprintf("%v", value.Float())
|
||||||
|
case reflect.Complex64, reflect.Complex128:
|
||||||
|
return fmt.Sprintf("%v", value.Complex())
|
||||||
|
case reflect.Chan:
|
||||||
|
return fmt.Sprintf("0x%x", value.Pointer())
|
||||||
|
case reflect.Func:
|
||||||
|
return fmt.Sprintf("0x%x", value.Pointer())
|
||||||
|
case reflect.Ptr:
|
||||||
|
return formatValue(value.Elem(), indentation)
|
||||||
|
case reflect.Slice:
|
||||||
|
return truncateLongStrings(formatSlice(value, indentation))
|
||||||
|
case reflect.String:
|
||||||
|
return truncateLongStrings(formatString(value.String(), indentation))
|
||||||
|
case reflect.Array:
|
||||||
|
return truncateLongStrings(formatSlice(value, indentation))
|
||||||
|
case reflect.Map:
|
||||||
|
return truncateLongStrings(formatMap(value, indentation))
|
||||||
|
case reflect.Struct:
|
||||||
|
if value.Type() == timeType && value.CanInterface() {
|
||||||
|
t, _ := value.Interface().(time.Time)
|
||||||
|
return t.Format(time.RFC3339Nano)
|
||||||
|
}
|
||||||
|
return truncateLongStrings(formatStruct(value, indentation))
|
||||||
|
case reflect.Interface:
|
||||||
|
return formatInterface(value, indentation)
|
||||||
|
default:
|
||||||
|
if value.CanInterface() {
|
||||||
|
return truncateLongStrings(fmt.Sprintf("%#v", value.Interface()))
|
||||||
|
}
|
||||||
|
return truncateLongStrings(fmt.Sprintf("%#v", value))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func formatString(object interface{}, indentation uint) string {
|
||||||
|
if indentation == 1 {
|
||||||
|
s := fmt.Sprintf("%s", object)
|
||||||
|
components := strings.Split(s, "\n")
|
||||||
|
result := ""
|
||||||
|
for i, component := range components {
|
||||||
|
if i == 0 {
|
||||||
|
result += component
|
||||||
|
} else {
|
||||||
|
result += Indent + component
|
||||||
|
}
|
||||||
|
if i < len(components)-1 {
|
||||||
|
result += "\n"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return result
|
||||||
|
} else {
|
||||||
|
return fmt.Sprintf("%q", object)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func formatSlice(v reflect.Value, indentation uint) string {
|
||||||
|
if v.Kind() == reflect.Slice && v.Type().Elem().Kind() == reflect.Uint8 && isPrintableString(string(v.Bytes())) {
|
||||||
|
return formatString(v.Bytes(), indentation)
|
||||||
|
}
|
||||||
|
|
||||||
|
l := v.Len()
|
||||||
|
result := make([]string, l)
|
||||||
|
longest := 0
|
||||||
|
for i := 0; i < l; i++ {
|
||||||
|
result[i] = formatValue(v.Index(i), indentation+1)
|
||||||
|
if len(result[i]) > longest {
|
||||||
|
longest = len(result[i])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if longest > longFormThreshold {
|
||||||
|
indenter := strings.Repeat(Indent, int(indentation))
|
||||||
|
return fmt.Sprintf("[\n%s%s,\n%s]", indenter+Indent, strings.Join(result, ",\n"+indenter+Indent), indenter)
|
||||||
|
}
|
||||||
|
return fmt.Sprintf("[%s]", strings.Join(result, ", "))
|
||||||
|
}
|
||||||
|
|
||||||
|
func formatMap(v reflect.Value, indentation uint) string {
|
||||||
|
l := v.Len()
|
||||||
|
result := make([]string, l)
|
||||||
|
|
||||||
|
longest := 0
|
||||||
|
for i, key := range v.MapKeys() {
|
||||||
|
value := v.MapIndex(key)
|
||||||
|
result[i] = fmt.Sprintf("%s: %s", formatValue(key, indentation+1), formatValue(value, indentation+1))
|
||||||
|
if len(result[i]) > longest {
|
||||||
|
longest = len(result[i])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if longest > longFormThreshold {
|
||||||
|
indenter := strings.Repeat(Indent, int(indentation))
|
||||||
|
return fmt.Sprintf("{\n%s%s,\n%s}", indenter+Indent, strings.Join(result, ",\n"+indenter+Indent), indenter)
|
||||||
|
}
|
||||||
|
return fmt.Sprintf("{%s}", strings.Join(result, ", "))
|
||||||
|
}
|
||||||
|
|
||||||
|
func formatStruct(v reflect.Value, indentation uint) string {
|
||||||
|
t := v.Type()
|
||||||
|
|
||||||
|
l := v.NumField()
|
||||||
|
result := []string{}
|
||||||
|
longest := 0
|
||||||
|
for i := 0; i < l; i++ {
|
||||||
|
structField := t.Field(i)
|
||||||
|
fieldEntry := v.Field(i)
|
||||||
|
representation := fmt.Sprintf("%s: %s", structField.Name, formatValue(fieldEntry, indentation+1))
|
||||||
|
result = append(result, representation)
|
||||||
|
if len(representation) > longest {
|
||||||
|
longest = len(representation)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if longest > longFormThreshold {
|
||||||
|
indenter := strings.Repeat(Indent, int(indentation))
|
||||||
|
return fmt.Sprintf("{\n%s%s,\n%s}", indenter+Indent, strings.Join(result, ",\n"+indenter+Indent), indenter)
|
||||||
|
}
|
||||||
|
return fmt.Sprintf("{%s}", strings.Join(result, ", "))
|
||||||
|
}
|
||||||
|
|
||||||
|
func formatInterface(v reflect.Value, indentation uint) string {
|
||||||
|
return fmt.Sprintf("<%s>%s", formatType(v.Elem()), formatValue(v.Elem(), indentation))
|
||||||
|
}
|
||||||
|
|
||||||
|
func isNilValue(a reflect.Value) bool {
|
||||||
|
switch a.Kind() {
|
||||||
|
case reflect.Invalid:
|
||||||
|
return true
|
||||||
|
case reflect.Chan, reflect.Func, reflect.Interface, reflect.Map, reflect.Ptr, reflect.Slice:
|
||||||
|
return a.IsNil()
|
||||||
|
}
|
||||||
|
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
Returns true when the string is entirely made of printable runes, false otherwise.
|
||||||
|
*/
|
||||||
|
func isPrintableString(str string) bool {
|
||||||
|
for _, runeValue := range str {
|
||||||
|
if !strconv.IsPrint(runeValue) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}
|
||||||
537
vendor/github.com/onsi/gomega/gomega_dsl.go
generated
vendored
Normal file
537
vendor/github.com/onsi/gomega/gomega_dsl.go
generated
vendored
Normal file
@@ -0,0 +1,537 @@
|
|||||||
|
/*
|
||||||
|
Gomega is the Ginkgo BDD-style testing framework's preferred matcher library.
|
||||||
|
|
||||||
|
The godoc documentation describes Gomega's API. More comprehensive documentation (with examples!) is available at http://onsi.github.io/gomega/
|
||||||
|
|
||||||
|
Gomega on Github: http://github.com/onsi/gomega
|
||||||
|
|
||||||
|
Learn more about Ginkgo online: http://onsi.github.io/ginkgo
|
||||||
|
|
||||||
|
Ginkgo on Github: http://github.com/onsi/ginkgo
|
||||||
|
|
||||||
|
Gomega is MIT-Licensed
|
||||||
|
*/
|
||||||
|
package gomega
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/onsi/gomega/internal"
|
||||||
|
"github.com/onsi/gomega/types"
|
||||||
|
)
|
||||||
|
|
||||||
|
const GOMEGA_VERSION = "1.27.6"
|
||||||
|
|
||||||
|
const nilGomegaPanic = `You are trying to make an assertion, but haven't registered Gomega's fail handler.
|
||||||
|
If you're using Ginkgo then you probably forgot to put your assertion in an It().
|
||||||
|
Alternatively, you may have forgotten to register a fail handler with RegisterFailHandler() or RegisterTestingT().
|
||||||
|
Depending on your vendoring solution you may be inadvertently importing gomega and subpackages (e.g. ghhtp, gexec,...) from different locations.
|
||||||
|
`
|
||||||
|
|
||||||
|
// Gomega describes the essential Gomega DSL. This interface allows libraries
|
||||||
|
// to abstract between the standard package-level function implementations
|
||||||
|
// and alternatives like *WithT.
|
||||||
|
//
|
||||||
|
// The types in the top-level DSL have gotten a bit messy due to earlier deprecations that avoid stuttering
|
||||||
|
// and due to an accidental use of a concrete type (*WithT) in an earlier release.
|
||||||
|
//
|
||||||
|
// As of 1.15 both the WithT and Ginkgo variants of Gomega are implemented by the same underlying object
|
||||||
|
// however one (the Ginkgo variant) is exported as an interface (types.Gomega) whereas the other (the withT variant)
|
||||||
|
// is shared as a concrete type (*WithT, which is aliased to *internal.Gomega). 1.15 did not clean this mess up to ensure
|
||||||
|
// that declarations of *WithT in existing code are not broken by the upgrade to 1.15.
|
||||||
|
type Gomega = types.Gomega
|
||||||
|
|
||||||
|
// DefaultGomega supplies the standard package-level implementation
|
||||||
|
var Default = Gomega(internal.NewGomega(internal.FetchDefaultDurationBundle()))
|
||||||
|
|
||||||
|
// NewGomega returns an instance of Gomega wired into the passed-in fail handler.
|
||||||
|
// You generally don't need to use this when using Ginkgo - RegisterFailHandler will wire up the global gomega
|
||||||
|
// However creating a NewGomega with a custom fail handler can be useful in contexts where you want to use Gomega's
|
||||||
|
// rich ecosystem of matchers without causing a test to fail. For example, to aggregate a series of potential failures
|
||||||
|
// or for use in a non-test setting.
|
||||||
|
func NewGomega(fail types.GomegaFailHandler) Gomega {
|
||||||
|
return internal.NewGomega(internalGomega(Default).DurationBundle).ConfigureWithFailHandler(fail)
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithT wraps a *testing.T and provides `Expect`, `Eventually`, and `Consistently` methods. This allows you to leverage
|
||||||
|
// Gomega's rich ecosystem of matchers in standard `testing` test suites.
|
||||||
|
//
|
||||||
|
// Use `NewWithT` to instantiate a `WithT`
|
||||||
|
//
|
||||||
|
// As of 1.15 both the WithT and Ginkgo variants of Gomega are implemented by the same underlying object
|
||||||
|
// however one (the Ginkgo variant) is exported as an interface (types.Gomega) whereas the other (the withT variant)
|
||||||
|
// is shared as a concrete type (*WithT, which is aliased to *internal.Gomega). 1.15 did not clean this mess up to ensure
|
||||||
|
// that declarations of *WithT in existing code are not broken by the upgrade to 1.15.
|
||||||
|
type WithT = internal.Gomega
|
||||||
|
|
||||||
|
// GomegaWithT is deprecated in favor of gomega.WithT, which does not stutter.
|
||||||
|
type GomegaWithT = WithT
|
||||||
|
|
||||||
|
// inner is an interface that allows users to provide a wrapper around Default. The wrapper
|
||||||
|
// must implement the inner interface and return either the original Default or the result of
|
||||||
|
// a call to NewGomega().
|
||||||
|
type inner interface {
|
||||||
|
Inner() Gomega
|
||||||
|
}
|
||||||
|
|
||||||
|
func internalGomega(g Gomega) *internal.Gomega {
|
||||||
|
if v, ok := g.(inner); ok {
|
||||||
|
return v.Inner().(*internal.Gomega)
|
||||||
|
}
|
||||||
|
return g.(*internal.Gomega)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewWithT takes a *testing.T and returns a `gomega.WithT` allowing you to use `Expect`, `Eventually`, and `Consistently` along with
|
||||||
|
// Gomega's rich ecosystem of matchers in standard `testing` test suits.
|
||||||
|
//
|
||||||
|
// func TestFarmHasCow(t *testing.T) {
|
||||||
|
// g := gomega.NewWithT(t)
|
||||||
|
//
|
||||||
|
// f := farm.New([]string{"Cow", "Horse"})
|
||||||
|
// g.Expect(f.HasCow()).To(BeTrue(), "Farm should have cow")
|
||||||
|
// }
|
||||||
|
func NewWithT(t types.GomegaTestingT) *WithT {
|
||||||
|
return internal.NewGomega(internalGomega(Default).DurationBundle).ConfigureWithT(t)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewGomegaWithT is deprecated in favor of gomega.NewWithT, which does not stutter.
|
||||||
|
var NewGomegaWithT = NewWithT
|
||||||
|
|
||||||
|
// RegisterFailHandler connects Ginkgo to Gomega. When a matcher fails
|
||||||
|
// the fail handler passed into RegisterFailHandler is called.
|
||||||
|
func RegisterFailHandler(fail types.GomegaFailHandler) {
|
||||||
|
internalGomega(Default).ConfigureWithFailHandler(fail)
|
||||||
|
}
|
||||||
|
|
||||||
|
// RegisterFailHandlerWithT is deprecated and will be removed in a future release.
|
||||||
|
// users should use RegisterFailHandler, or RegisterTestingT
|
||||||
|
func RegisterFailHandlerWithT(_ types.GomegaTestingT, fail types.GomegaFailHandler) {
|
||||||
|
fmt.Println("RegisterFailHandlerWithT is deprecated. Please use RegisterFailHandler or RegisterTestingT instead.")
|
||||||
|
internalGomega(Default).ConfigureWithFailHandler(fail)
|
||||||
|
}
|
||||||
|
|
||||||
|
// RegisterTestingT connects Gomega to Golang's XUnit style
|
||||||
|
// Testing.T tests. It is now deprecated and you should use NewWithT() instead to get a fresh instance of Gomega for each test.
|
||||||
|
func RegisterTestingT(t types.GomegaTestingT) {
|
||||||
|
internalGomega(Default).ConfigureWithT(t)
|
||||||
|
}
|
||||||
|
|
||||||
|
// InterceptGomegaFailures runs a given callback and returns an array of
|
||||||
|
// failure messages generated by any Gomega assertions within the callback.
|
||||||
|
// Execution continues after the first failure allowing users to collect all failures
|
||||||
|
// in the callback.
|
||||||
|
//
|
||||||
|
// This is most useful when testing custom matchers, but can also be used to check
|
||||||
|
// on a value using a Gomega assertion without causing a test failure.
|
||||||
|
func InterceptGomegaFailures(f func()) []string {
|
||||||
|
originalHandler := internalGomega(Default).Fail
|
||||||
|
failures := []string{}
|
||||||
|
internalGomega(Default).Fail = func(message string, callerSkip ...int) {
|
||||||
|
failures = append(failures, message)
|
||||||
|
}
|
||||||
|
defer func() {
|
||||||
|
internalGomega(Default).Fail = originalHandler
|
||||||
|
}()
|
||||||
|
f()
|
||||||
|
return failures
|
||||||
|
}
|
||||||
|
|
||||||
|
// InterceptGomegaFailure runs a given callback and returns the first
|
||||||
|
// failure message generated by any Gomega assertions within the callback, wrapped in an error.
|
||||||
|
//
|
||||||
|
// The callback ceases execution as soon as the first failed assertion occurs, however Gomega
|
||||||
|
// does not register a failure with the FailHandler registered via RegisterFailHandler - it is up
|
||||||
|
// to the user to decide what to do with the returned error
|
||||||
|
func InterceptGomegaFailure(f func()) (err error) {
|
||||||
|
originalHandler := internalGomega(Default).Fail
|
||||||
|
internalGomega(Default).Fail = func(message string, callerSkip ...int) {
|
||||||
|
err = errors.New(message)
|
||||||
|
panic("stop execution")
|
||||||
|
}
|
||||||
|
|
||||||
|
defer func() {
|
||||||
|
internalGomega(Default).Fail = originalHandler
|
||||||
|
if e := recover(); e != nil {
|
||||||
|
if err == nil {
|
||||||
|
panic(e)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
f()
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
func ensureDefaultGomegaIsConfigured() {
|
||||||
|
if !internalGomega(Default).IsConfigured() {
|
||||||
|
panic(nilGomegaPanic)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ω wraps an actual value allowing assertions to be made on it:
|
||||||
|
//
|
||||||
|
// Ω("foo").Should(Equal("foo"))
|
||||||
|
//
|
||||||
|
// If Ω is passed more than one argument it will pass the *first* argument to the matcher.
|
||||||
|
// All subsequent arguments will be required to be nil/zero.
|
||||||
|
//
|
||||||
|
// This is convenient if you want to make an assertion on a method/function that returns
|
||||||
|
// a value and an error - a common patter in Go.
|
||||||
|
//
|
||||||
|
// For example, given a function with signature:
|
||||||
|
//
|
||||||
|
// func MyAmazingThing() (int, error)
|
||||||
|
//
|
||||||
|
// Then:
|
||||||
|
//
|
||||||
|
// Ω(MyAmazingThing()).Should(Equal(3))
|
||||||
|
//
|
||||||
|
// Will succeed only if `MyAmazingThing()` returns `(3, nil)`
|
||||||
|
//
|
||||||
|
// Ω and Expect are identical
|
||||||
|
func Ω(actual interface{}, extra ...interface{}) Assertion {
|
||||||
|
ensureDefaultGomegaIsConfigured()
|
||||||
|
return Default.Ω(actual, extra...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Expect wraps an actual value allowing assertions to be made on it:
|
||||||
|
//
|
||||||
|
// Expect("foo").To(Equal("foo"))
|
||||||
|
//
|
||||||
|
// If Expect is passed more than one argument it will pass the *first* argument to the matcher.
|
||||||
|
// All subsequent arguments will be required to be nil/zero.
|
||||||
|
//
|
||||||
|
// This is convenient if you want to make an assertion on a method/function that returns
|
||||||
|
// a value and an error - a common pattern in Go.
|
||||||
|
//
|
||||||
|
// For example, given a function with signature:
|
||||||
|
//
|
||||||
|
// func MyAmazingThing() (int, error)
|
||||||
|
//
|
||||||
|
// Then:
|
||||||
|
//
|
||||||
|
// Expect(MyAmazingThing()).Should(Equal(3))
|
||||||
|
//
|
||||||
|
// Will succeed only if `MyAmazingThing()` returns `(3, nil)`
|
||||||
|
//
|
||||||
|
// Expect and Ω are identical
|
||||||
|
func Expect(actual interface{}, extra ...interface{}) Assertion {
|
||||||
|
ensureDefaultGomegaIsConfigured()
|
||||||
|
return Default.Expect(actual, extra...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// ExpectWithOffset wraps an actual value allowing assertions to be made on it:
|
||||||
|
//
|
||||||
|
// ExpectWithOffset(1, "foo").To(Equal("foo"))
|
||||||
|
//
|
||||||
|
// Unlike `Expect` and `Ω`, `ExpectWithOffset` takes an additional integer argument
|
||||||
|
// that is used to modify the call-stack offset when computing line numbers. It is
|
||||||
|
// the same as `Expect(...).WithOffset`.
|
||||||
|
//
|
||||||
|
// This is most useful in helper functions that make assertions. If you want Gomega's
|
||||||
|
// error message to refer to the calling line in the test (as opposed to the line in the helper function)
|
||||||
|
// set the first argument of `ExpectWithOffset` appropriately.
|
||||||
|
func ExpectWithOffset(offset int, actual interface{}, extra ...interface{}) Assertion {
|
||||||
|
ensureDefaultGomegaIsConfigured()
|
||||||
|
return Default.ExpectWithOffset(offset, actual, extra...)
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
Eventually enables making assertions on asynchronous behavior.
|
||||||
|
|
||||||
|
Eventually checks that an assertion *eventually* passes. Eventually blocks when called and attempts an assertion periodically until it passes or a timeout occurs. Both the timeout and polling interval are configurable as optional arguments.
|
||||||
|
The first optional argument is the timeout (which defaults to 1s), the second is the polling interval (which defaults to 10ms). Both intervals can be specified as time.Duration, parsable duration strings or floats/integers (in which case they are interpreted as seconds). In addition an optional context.Context can be passed in - Eventually will keep trying until either the timeout epxires or the context is cancelled, whichever comes first.
|
||||||
|
|
||||||
|
Eventually works with any Gomega compatible matcher and supports making assertions against three categories of actual value:
|
||||||
|
|
||||||
|
**Category 1: Making Eventually assertions on values**
|
||||||
|
|
||||||
|
There are several examples of values that can change over time. These can be passed in to Eventually and will be passed to the matcher repeatedly until a match occurs. For example:
|
||||||
|
|
||||||
|
c := make(chan bool)
|
||||||
|
go DoStuff(c)
|
||||||
|
Eventually(c, "50ms").Should(BeClosed())
|
||||||
|
|
||||||
|
will poll the channel repeatedly until it is closed. In this example `Eventually` will block until either the specified timeout of 50ms has elapsed or the channel is closed, whichever comes first.
|
||||||
|
|
||||||
|
Several Gomega libraries allow you to use Eventually in this way. For example, the gomega/gexec package allows you to block until a *gexec.Session exits successfully via:
|
||||||
|
|
||||||
|
Eventually(session).Should(gexec.Exit(0))
|
||||||
|
|
||||||
|
And the gomega/gbytes package allows you to monitor a streaming *gbytes.Buffer until a given string is seen:
|
||||||
|
|
||||||
|
Eventually(buffer).Should(gbytes.Say("hello there"))
|
||||||
|
|
||||||
|
In these examples, both `session` and `buffer` are designed to be thread-safe when polled by the `Exit` and `Say` matchers. This is not true in general of most raw values, so while it is tempting to do something like:
|
||||||
|
|
||||||
|
// THIS IS NOT THREAD-SAFE
|
||||||
|
var s *string
|
||||||
|
go mutateStringEventually(s)
|
||||||
|
Eventually(s).Should(Equal("I've changed"))
|
||||||
|
|
||||||
|
this will trigger Go's race detector as the goroutine polling via Eventually will race over the value of s with the goroutine mutating the string. For cases like this you can use channels or introduce your own locking around s by passing Eventually a function.
|
||||||
|
|
||||||
|
**Category 2: Make Eventually assertions on functions**
|
||||||
|
|
||||||
|
Eventually can be passed functions that **return at least one value**. When configured this way, Eventually will poll the function repeatedly and pass the first returned value to the matcher.
|
||||||
|
|
||||||
|
For example:
|
||||||
|
|
||||||
|
Eventually(func() int {
|
||||||
|
return client.FetchCount()
|
||||||
|
}).Should(BeNumerically(">=", 17))
|
||||||
|
|
||||||
|
will repeatedly poll client.FetchCount until the BeNumerically matcher is satisfied. (Note that this example could have been written as Eventually(client.FetchCount).Should(BeNumerically(">=", 17)))
|
||||||
|
|
||||||
|
If multiple values are returned by the function, Eventually will pass the first value to the matcher and require that all others are zero-valued. This allows you to pass Eventually a function that returns a value and an error - a common pattern in Go.
|
||||||
|
|
||||||
|
For example, consider a method that returns a value and an error:
|
||||||
|
|
||||||
|
func FetchFromDB() (string, error)
|
||||||
|
|
||||||
|
Then
|
||||||
|
|
||||||
|
Eventually(FetchFromDB).Should(Equal("got it"))
|
||||||
|
|
||||||
|
will pass only if and when the returned error is nil *and* the returned string satisfies the matcher.
|
||||||
|
|
||||||
|
Eventually can also accept functions that take arguments, however you must provide those arguments using .WithArguments(). For example, consider a function that takes a user-id and makes a network request to fetch a full name:
|
||||||
|
|
||||||
|
func FetchFullName(userId int) (string, error)
|
||||||
|
|
||||||
|
You can poll this function like so:
|
||||||
|
|
||||||
|
Eventually(FetchFullName).WithArguments(1138).Should(Equal("Wookie"))
|
||||||
|
|
||||||
|
It is important to note that the function passed into Eventually is invoked *synchronously* when polled. Eventually does not (in fact, it cannot) kill the function if it takes longer to return than Eventually's configured timeout. A common practice here is to use a context. Here's an example that combines Ginkgo's spec timeout support with Eventually:
|
||||||
|
|
||||||
|
It("fetches the correct count", func(ctx SpecContext) {
|
||||||
|
Eventually(ctx, func() int {
|
||||||
|
return client.FetchCount(ctx, "/users")
|
||||||
|
}).Should(BeNumerically(">=", 17))
|
||||||
|
}, SpecTimeout(time.Second))
|
||||||
|
|
||||||
|
you an also use Eventually().WithContext(ctx) to pass in the context. Passed-in contexts play nicely with paseed-in arguments as long as the context appears first. You can rewrite the above example as:
|
||||||
|
|
||||||
|
It("fetches the correct count", func(ctx SpecContext) {
|
||||||
|
Eventually(client.FetchCount).WithContext(ctx).WithArguments("/users").Should(BeNumerically(">=", 17))
|
||||||
|
}, SpecTimeout(time.Second))
|
||||||
|
|
||||||
|
Either way the context passd to Eventually is also passed to the underlying funciton. Now, when Ginkgo cancels the context both the FetchCount client and Gomega will be informed and can exit.
|
||||||
|
|
||||||
|
**Category 3: Making assertions _in_ the function passed into Eventually**
|
||||||
|
|
||||||
|
When testing complex systems it can be valuable to assert that a _set_ of assertions passes Eventually. Eventually supports this by accepting functions that take a single Gomega argument and return zero or more values.
|
||||||
|
|
||||||
|
Here's an example that makes some assertions and returns a value and error:
|
||||||
|
|
||||||
|
Eventually(func(g Gomega) (Widget, error) {
|
||||||
|
ids, err := client.FetchIDs()
|
||||||
|
g.Expect(err).NotTo(HaveOccurred())
|
||||||
|
g.Expect(ids).To(ContainElement(1138))
|
||||||
|
return client.FetchWidget(1138)
|
||||||
|
}).Should(Equal(expectedWidget))
|
||||||
|
|
||||||
|
will pass only if all the assertions in the polled function pass and the return value satisfied the matcher.
|
||||||
|
|
||||||
|
Eventually also supports a special case polling function that takes a single Gomega argument and returns no values. Eventually assumes such a function is making assertions and is designed to work with the Succeed matcher to validate that all assertions have passed.
|
||||||
|
For example:
|
||||||
|
|
||||||
|
Eventually(func(g Gomega) {
|
||||||
|
model, err := client.Find(1138)
|
||||||
|
g.Expect(err).NotTo(HaveOccurred())
|
||||||
|
g.Expect(model.Reticulate()).To(Succeed())
|
||||||
|
g.Expect(model.IsReticulated()).To(BeTrue())
|
||||||
|
g.Expect(model.Save()).To(Succeed())
|
||||||
|
}).Should(Succeed())
|
||||||
|
|
||||||
|
will rerun the function until all assertions pass.
|
||||||
|
|
||||||
|
You can also pass additional arugments to functions that take a Gomega. The only rule is that the Gomega argument must be first. If you also want to pass the context attached to Eventually you must ensure that is the second argument. For example:
|
||||||
|
|
||||||
|
Eventually(func(g Gomega, ctx context.Context, path string, expected ...string){
|
||||||
|
tok, err := client.GetToken(ctx)
|
||||||
|
g.Expect(err).NotTo(HaveOccurred())
|
||||||
|
|
||||||
|
elements, err := client.Fetch(ctx, tok, path)
|
||||||
|
g.Expect(err).NotTo(HaveOccurred())
|
||||||
|
g.Expect(elements).To(ConsistOf(expected))
|
||||||
|
}).WithContext(ctx).WithArguments("/names", "Joe", "Jane", "Sam").Should(Succeed())
|
||||||
|
|
||||||
|
You can ensure that you get a number of consecutive successful tries before succeeding using `MustPassRepeatedly(int)`. For Example:
|
||||||
|
|
||||||
|
int count := 0
|
||||||
|
Eventually(func() bool {
|
||||||
|
count++
|
||||||
|
return count > 2
|
||||||
|
}).MustPassRepeatedly(2).Should(BeTrue())
|
||||||
|
// Because we had to wait for 2 calls that returned true
|
||||||
|
Expect(count).To(Equal(3))
|
||||||
|
|
||||||
|
Finally, in addition to passing timeouts and a context to Eventually you can be more explicit with Eventually's chaining configuration methods:
|
||||||
|
|
||||||
|
Eventually(..., "1s", "2s", ctx).Should(...)
|
||||||
|
|
||||||
|
is equivalent to
|
||||||
|
|
||||||
|
Eventually(...).WithTimeout(time.Second).WithPolling(2*time.Second).WithContext(ctx).Should(...)
|
||||||
|
*/
|
||||||
|
func Eventually(actualOrCtx interface{}, args ...interface{}) AsyncAssertion {
|
||||||
|
ensureDefaultGomegaIsConfigured()
|
||||||
|
return Default.Eventually(actualOrCtx, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// EventuallyWithOffset operates like Eventually but takes an additional
|
||||||
|
// initial argument to indicate an offset in the call stack. This is useful when building helper
|
||||||
|
// functions that contain matchers. To learn more, read about `ExpectWithOffset`.
|
||||||
|
//
|
||||||
|
// `EventuallyWithOffset` is the same as `Eventually(...).WithOffset`.
|
||||||
|
//
|
||||||
|
// `EventuallyWithOffset` specifying a timeout interval (and an optional polling interval) are
|
||||||
|
// the same as `Eventually(...).WithOffset(...).WithTimeout` or
|
||||||
|
// `Eventually(...).WithOffset(...).WithTimeout(...).WithPolling`.
|
||||||
|
func EventuallyWithOffset(offset int, actualOrCtx interface{}, args ...interface{}) AsyncAssertion {
|
||||||
|
ensureDefaultGomegaIsConfigured()
|
||||||
|
return Default.EventuallyWithOffset(offset, actualOrCtx, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
Consistently, like Eventually, enables making assertions on asynchronous behavior.
|
||||||
|
|
||||||
|
Consistently blocks when called for a specified duration. During that duration Consistently repeatedly polls its matcher and ensures that it is satisfied. If the matcher is consistently satisfied, then Consistently will pass. Otherwise Consistently will fail.
|
||||||
|
|
||||||
|
Both the total waiting duration and the polling interval are configurable as optional arguments. The first optional argument is the duration that Consistently will run for (defaults to 100ms), and the second argument is the polling interval (defaults to 10ms). As with Eventually, these intervals can be passed in as time.Duration, parsable duration strings or an integer or float number of seconds. You can also pass in an optional context.Context - Consistently will exit early (with a failure) if the context is cancelled before the waiting duration expires.
|
||||||
|
|
||||||
|
Consistently accepts the same three categories of actual as Eventually, check the Eventually docs to learn more.
|
||||||
|
|
||||||
|
Consistently is useful in cases where you want to assert that something *does not happen* for a period of time. For example, you may want to assert that a goroutine does *not* send data down a channel. In this case you could write:
|
||||||
|
|
||||||
|
Consistently(channel, "200ms").ShouldNot(Receive())
|
||||||
|
|
||||||
|
This will block for 200 milliseconds and repeatedly check the channel and ensure nothing has been received.
|
||||||
|
*/
|
||||||
|
func Consistently(actualOrCtx interface{}, args ...interface{}) AsyncAssertion {
|
||||||
|
ensureDefaultGomegaIsConfigured()
|
||||||
|
return Default.Consistently(actualOrCtx, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// ConsistentlyWithOffset operates like Consistently but takes an additional
|
||||||
|
// initial argument to indicate an offset in the call stack. This is useful when building helper
|
||||||
|
// functions that contain matchers. To learn more, read about `ExpectWithOffset`.
|
||||||
|
//
|
||||||
|
// `ConsistentlyWithOffset` is the same as `Consistently(...).WithOffset` and
|
||||||
|
// optional `WithTimeout` and `WithPolling`.
|
||||||
|
func ConsistentlyWithOffset(offset int, actualOrCtx interface{}, args ...interface{}) AsyncAssertion {
|
||||||
|
ensureDefaultGomegaIsConfigured()
|
||||||
|
return Default.ConsistentlyWithOffset(offset, actualOrCtx, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
StopTrying can be used to signal to Eventually and Consistentlythat they should abort and stop trying. This always results in a failure of the assertion - and the failure message is the content of the StopTrying signal.
|
||||||
|
|
||||||
|
You can send the StopTrying signal by either returning StopTrying("message") as an error from your passed-in function _or_ by calling StopTrying("message").Now() to trigger a panic and end execution.
|
||||||
|
|
||||||
|
You can also wrap StopTrying around an error with `StopTrying("message").Wrap(err)` and can attach additional objects via `StopTrying("message").Attach("description", object). When rendered, the signal will include the wrapped error and any attached objects rendered using Gomega's default formatting.
|
||||||
|
|
||||||
|
Here are a couple of examples. This is how you might use StopTrying() as an error to signal that Eventually should stop:
|
||||||
|
|
||||||
|
playerIndex, numPlayers := 0, 11
|
||||||
|
Eventually(func() (string, error) {
|
||||||
|
if playerIndex == numPlayers {
|
||||||
|
return "", StopTrying("no more players left")
|
||||||
|
}
|
||||||
|
name := client.FetchPlayer(playerIndex)
|
||||||
|
playerIndex += 1
|
||||||
|
return name, nil
|
||||||
|
}).Should(Equal("Patrick Mahomes"))
|
||||||
|
|
||||||
|
And here's an example where `StopTrying().Now()` is called to halt execution immediately:
|
||||||
|
|
||||||
|
Eventually(func() []string {
|
||||||
|
names, err := client.FetchAllPlayers()
|
||||||
|
if err == client.IRRECOVERABLE_ERROR {
|
||||||
|
StopTrying("Irrecoverable error occurred").Wrap(err).Now()
|
||||||
|
}
|
||||||
|
return names
|
||||||
|
}).Should(ContainElement("Patrick Mahomes"))
|
||||||
|
*/
|
||||||
|
var StopTrying = internal.StopTrying
|
||||||
|
|
||||||
|
/*
|
||||||
|
TryAgainAfter(<duration>) allows you to adjust the polling interval for the _next_ iteration of `Eventually` or `Consistently`. Like `StopTrying` you can either return `TryAgainAfter` as an error or trigger it immedieately with `.Now()`
|
||||||
|
|
||||||
|
When `TryAgainAfter(<duration>` is triggered `Eventually` and `Consistently` will wait for that duration. If a timeout occurs before the next poll is triggered both `Eventually` and `Consistently` will always fail with the content of the TryAgainAfter message. As with StopTrying you can `.Wrap()` and error and `.Attach()` additional objects to `TryAgainAfter`.
|
||||||
|
*/
|
||||||
|
var TryAgainAfter = internal.TryAgainAfter
|
||||||
|
|
||||||
|
/*
|
||||||
|
PollingSignalError is the error returned by StopTrying() and TryAgainAfter()
|
||||||
|
*/
|
||||||
|
type PollingSignalError = internal.PollingSignalError
|
||||||
|
|
||||||
|
// SetDefaultEventuallyTimeout sets the default timeout duration for Eventually. Eventually will repeatedly poll your condition until it succeeds, or until this timeout elapses.
|
||||||
|
func SetDefaultEventuallyTimeout(t time.Duration) {
|
||||||
|
Default.SetDefaultEventuallyTimeout(t)
|
||||||
|
}
|
||||||
|
|
||||||
|
// SetDefaultEventuallyPollingInterval sets the default polling interval for Eventually.
|
||||||
|
func SetDefaultEventuallyPollingInterval(t time.Duration) {
|
||||||
|
Default.SetDefaultEventuallyPollingInterval(t)
|
||||||
|
}
|
||||||
|
|
||||||
|
// SetDefaultConsistentlyDuration sets the default duration for Consistently. Consistently will verify that your condition is satisfied for this long.
|
||||||
|
func SetDefaultConsistentlyDuration(t time.Duration) {
|
||||||
|
Default.SetDefaultConsistentlyDuration(t)
|
||||||
|
}
|
||||||
|
|
||||||
|
// SetDefaultConsistentlyPollingInterval sets the default polling interval for Consistently.
|
||||||
|
func SetDefaultConsistentlyPollingInterval(t time.Duration) {
|
||||||
|
Default.SetDefaultConsistentlyPollingInterval(t)
|
||||||
|
}
|
||||||
|
|
||||||
|
// AsyncAssertion is returned by Eventually and Consistently and polls the actual value passed into Eventually against
|
||||||
|
// the matcher passed to the Should and ShouldNot methods.
|
||||||
|
//
|
||||||
|
// Both Should and ShouldNot take a variadic optionalDescription argument.
|
||||||
|
// This argument allows you to make your failure messages more descriptive.
|
||||||
|
// If a single argument of type `func() string` is passed, this function will be lazily evaluated if a failure occurs
|
||||||
|
// and the returned string is used to annotate the failure message.
|
||||||
|
// Otherwise, this argument is passed on to fmt.Sprintf() and then used to annotate the failure message.
|
||||||
|
//
|
||||||
|
// Both Should and ShouldNot return a boolean that is true if the assertion passed and false if it failed.
|
||||||
|
//
|
||||||
|
// Example:
|
||||||
|
//
|
||||||
|
// Eventually(myChannel).Should(Receive(), "Something should have come down the pipe.")
|
||||||
|
// Consistently(myChannel).ShouldNot(Receive(), func() string { return "Nothing should have come down the pipe." })
|
||||||
|
type AsyncAssertion = types.AsyncAssertion
|
||||||
|
|
||||||
|
// GomegaAsyncAssertion is deprecated in favor of AsyncAssertion, which does not stutter.
|
||||||
|
type GomegaAsyncAssertion = types.AsyncAssertion
|
||||||
|
|
||||||
|
// Assertion is returned by Ω and Expect and compares the actual value to the matcher
|
||||||
|
// passed to the Should/ShouldNot and To/ToNot/NotTo methods.
|
||||||
|
//
|
||||||
|
// Typically Should/ShouldNot are used with Ω and To/ToNot/NotTo are used with Expect
|
||||||
|
// though this is not enforced.
|
||||||
|
//
|
||||||
|
// All methods take a variadic optionalDescription argument.
|
||||||
|
// This argument allows you to make your failure messages more descriptive.
|
||||||
|
// If a single argument of type `func() string` is passed, this function will be lazily evaluated if a failure occurs
|
||||||
|
// and the returned string is used to annotate the failure message.
|
||||||
|
// Otherwise, this argument is passed on to fmt.Sprintf() and then used to annotate the failure message.
|
||||||
|
//
|
||||||
|
// All methods return a bool that is true if the assertion passed and false if it failed.
|
||||||
|
//
|
||||||
|
// Example:
|
||||||
|
//
|
||||||
|
// Ω(farm.HasCow()).Should(BeTrue(), "Farm %v should have a cow", farm)
|
||||||
|
type Assertion = types.Assertion
|
||||||
|
|
||||||
|
// GomegaAssertion is deprecated in favor of Assertion, which does not stutter.
|
||||||
|
type GomegaAssertion = types.Assertion
|
||||||
|
|
||||||
|
// OmegaMatcher is deprecated in favor of the better-named and better-organized types.GomegaMatcher but sticks around to support existing code that uses it
|
||||||
|
type OmegaMatcher = types.GomegaMatcher
|
||||||
161
vendor/github.com/onsi/gomega/internal/assertion.go
generated
vendored
Normal file
161
vendor/github.com/onsi/gomega/internal/assertion.go
generated
vendored
Normal file
@@ -0,0 +1,161 @@
|
|||||||
|
package internal
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"reflect"
|
||||||
|
|
||||||
|
"github.com/onsi/gomega/format"
|
||||||
|
"github.com/onsi/gomega/types"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Assertion struct {
|
||||||
|
actuals []interface{} // actual value plus all extra values
|
||||||
|
actualIndex int // value to pass to the matcher
|
||||||
|
vet vetinari // the vet to call before calling Gomega matcher
|
||||||
|
offset int
|
||||||
|
g *Gomega
|
||||||
|
}
|
||||||
|
|
||||||
|
// ...obligatory discworld reference, as "vetineer" doesn't sound ... quite right.
|
||||||
|
type vetinari func(assertion *Assertion, optionalDescription ...interface{}) bool
|
||||||
|
|
||||||
|
func NewAssertion(actualInput interface{}, g *Gomega, offset int, extra ...interface{}) *Assertion {
|
||||||
|
return &Assertion{
|
||||||
|
actuals: append([]interface{}{actualInput}, extra...),
|
||||||
|
actualIndex: 0,
|
||||||
|
vet: (*Assertion).vetActuals,
|
||||||
|
offset: offset,
|
||||||
|
g: g,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (assertion *Assertion) WithOffset(offset int) types.Assertion {
|
||||||
|
assertion.offset = offset
|
||||||
|
return assertion
|
||||||
|
}
|
||||||
|
|
||||||
|
func (assertion *Assertion) Error() types.Assertion {
|
||||||
|
return &Assertion{
|
||||||
|
actuals: assertion.actuals,
|
||||||
|
actualIndex: len(assertion.actuals) - 1,
|
||||||
|
vet: (*Assertion).vetError,
|
||||||
|
offset: assertion.offset,
|
||||||
|
g: assertion.g,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (assertion *Assertion) Should(matcher types.GomegaMatcher, optionalDescription ...interface{}) bool {
|
||||||
|
assertion.g.THelper()
|
||||||
|
vetOptionalDescription("Assertion", optionalDescription...)
|
||||||
|
return assertion.vet(assertion, optionalDescription...) && assertion.match(matcher, true, optionalDescription...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (assertion *Assertion) ShouldNot(matcher types.GomegaMatcher, optionalDescription ...interface{}) bool {
|
||||||
|
assertion.g.THelper()
|
||||||
|
vetOptionalDescription("Assertion", optionalDescription...)
|
||||||
|
return assertion.vet(assertion, optionalDescription...) && assertion.match(matcher, false, optionalDescription...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (assertion *Assertion) To(matcher types.GomegaMatcher, optionalDescription ...interface{}) bool {
|
||||||
|
assertion.g.THelper()
|
||||||
|
vetOptionalDescription("Assertion", optionalDescription...)
|
||||||
|
return assertion.vet(assertion, optionalDescription...) && assertion.match(matcher, true, optionalDescription...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (assertion *Assertion) ToNot(matcher types.GomegaMatcher, optionalDescription ...interface{}) bool {
|
||||||
|
assertion.g.THelper()
|
||||||
|
vetOptionalDescription("Assertion", optionalDescription...)
|
||||||
|
return assertion.vet(assertion, optionalDescription...) && assertion.match(matcher, false, optionalDescription...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (assertion *Assertion) NotTo(matcher types.GomegaMatcher, optionalDescription ...interface{}) bool {
|
||||||
|
assertion.g.THelper()
|
||||||
|
vetOptionalDescription("Assertion", optionalDescription...)
|
||||||
|
return assertion.vet(assertion, optionalDescription...) && assertion.match(matcher, false, optionalDescription...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (assertion *Assertion) buildDescription(optionalDescription ...interface{}) string {
|
||||||
|
switch len(optionalDescription) {
|
||||||
|
case 0:
|
||||||
|
return ""
|
||||||
|
case 1:
|
||||||
|
if describe, ok := optionalDescription[0].(func() string); ok {
|
||||||
|
return describe() + "\n"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return fmt.Sprintf(optionalDescription[0].(string), optionalDescription[1:]...) + "\n"
|
||||||
|
}
|
||||||
|
|
||||||
|
func (assertion *Assertion) match(matcher types.GomegaMatcher, desiredMatch bool, optionalDescription ...interface{}) bool {
|
||||||
|
actualInput := assertion.actuals[assertion.actualIndex]
|
||||||
|
matches, err := matcher.Match(actualInput)
|
||||||
|
assertion.g.THelper()
|
||||||
|
if err != nil {
|
||||||
|
description := assertion.buildDescription(optionalDescription...)
|
||||||
|
assertion.g.Fail(description+err.Error(), 2+assertion.offset)
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
if matches != desiredMatch {
|
||||||
|
var message string
|
||||||
|
if desiredMatch {
|
||||||
|
message = matcher.FailureMessage(actualInput)
|
||||||
|
} else {
|
||||||
|
message = matcher.NegatedFailureMessage(actualInput)
|
||||||
|
}
|
||||||
|
description := assertion.buildDescription(optionalDescription...)
|
||||||
|
assertion.g.Fail(description+message, 2+assertion.offset)
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
// vetActuals vets the actual values, with the (optional) exception of a
|
||||||
|
// specific value, such as the first value in case non-error assertions, or the
|
||||||
|
// last value in case of Error()-based assertions.
|
||||||
|
func (assertion *Assertion) vetActuals(optionalDescription ...interface{}) bool {
|
||||||
|
success, message := vetActuals(assertion.actuals, assertion.actualIndex)
|
||||||
|
if success {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
description := assertion.buildDescription(optionalDescription...)
|
||||||
|
assertion.g.THelper()
|
||||||
|
assertion.g.Fail(description+message, 2+assertion.offset)
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
// vetError vets the actual values, except for the final error value, in case
|
||||||
|
// the final error value is non-zero. Otherwise, it doesn't vet the actual
|
||||||
|
// values, as these are allowed to take on any values unless there is a non-zero
|
||||||
|
// error value.
|
||||||
|
func (assertion *Assertion) vetError(optionalDescription ...interface{}) bool {
|
||||||
|
if err := assertion.actuals[assertion.actualIndex]; err != nil {
|
||||||
|
// Go error result idiom: all other actual values must be zero values.
|
||||||
|
return assertion.vetActuals(optionalDescription...)
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
// vetActuals vets a slice of actual values, optionally skipping a particular
|
||||||
|
// value slice element, such as the first or last value slice element.
|
||||||
|
func vetActuals(actuals []interface{}, skipIndex int) (bool, string) {
|
||||||
|
for i, actual := range actuals {
|
||||||
|
if i == skipIndex {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if actual != nil {
|
||||||
|
zeroValue := reflect.Zero(reflect.TypeOf(actual)).Interface()
|
||||||
|
if !reflect.DeepEqual(zeroValue, actual) {
|
||||||
|
var message string
|
||||||
|
if err, ok := actual.(error); ok {
|
||||||
|
message = fmt.Sprintf("Unexpected error: %s\n%s", err, format.Object(err, 1))
|
||||||
|
} else {
|
||||||
|
message = fmt.Sprintf("Unexpected non-nil/non-zero argument at index %d:\n\t<%T>: %#v", i, actual, actual)
|
||||||
|
}
|
||||||
|
return false, message
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true, ""
|
||||||
|
}
|
||||||
571
vendor/github.com/onsi/gomega/internal/async_assertion.go
generated
vendored
Normal file
571
vendor/github.com/onsi/gomega/internal/async_assertion.go
generated
vendored
Normal file
@@ -0,0 +1,571 @@
|
|||||||
|
package internal
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"reflect"
|
||||||
|
"runtime"
|
||||||
|
"sync"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/onsi/gomega/format"
|
||||||
|
"github.com/onsi/gomega/types"
|
||||||
|
)
|
||||||
|
|
||||||
|
var errInterface = reflect.TypeOf((*error)(nil)).Elem()
|
||||||
|
var gomegaType = reflect.TypeOf((*types.Gomega)(nil)).Elem()
|
||||||
|
var contextType = reflect.TypeOf(new(context.Context)).Elem()
|
||||||
|
|
||||||
|
type formattedGomegaError interface {
|
||||||
|
FormattedGomegaError() string
|
||||||
|
}
|
||||||
|
|
||||||
|
type asyncPolledActualError struct {
|
||||||
|
message string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (err *asyncPolledActualError) Error() string {
|
||||||
|
return err.message
|
||||||
|
}
|
||||||
|
|
||||||
|
func (err *asyncPolledActualError) FormattedGomegaError() string {
|
||||||
|
return err.message
|
||||||
|
}
|
||||||
|
|
||||||
|
type contextWithAttachProgressReporter interface {
|
||||||
|
AttachProgressReporter(func() string) func()
|
||||||
|
}
|
||||||
|
|
||||||
|
type asyncGomegaHaltExecutionError struct{}
|
||||||
|
|
||||||
|
func (a asyncGomegaHaltExecutionError) GinkgoRecoverShouldIgnoreThisPanic() {}
|
||||||
|
func (a asyncGomegaHaltExecutionError) Error() string {
|
||||||
|
return `An assertion has failed in a goroutine. You should call
|
||||||
|
|
||||||
|
defer GinkgoRecover()
|
||||||
|
|
||||||
|
at the top of the goroutine that caused this panic. This will allow Ginkgo and Gomega to correctly capture and manage this panic.`
|
||||||
|
}
|
||||||
|
|
||||||
|
type AsyncAssertionType uint
|
||||||
|
|
||||||
|
const (
|
||||||
|
AsyncAssertionTypeEventually AsyncAssertionType = iota
|
||||||
|
AsyncAssertionTypeConsistently
|
||||||
|
)
|
||||||
|
|
||||||
|
func (at AsyncAssertionType) String() string {
|
||||||
|
switch at {
|
||||||
|
case AsyncAssertionTypeEventually:
|
||||||
|
return "Eventually"
|
||||||
|
case AsyncAssertionTypeConsistently:
|
||||||
|
return "Consistently"
|
||||||
|
}
|
||||||
|
return "INVALID ASYNC ASSERTION TYPE"
|
||||||
|
}
|
||||||
|
|
||||||
|
type AsyncAssertion struct {
|
||||||
|
asyncType AsyncAssertionType
|
||||||
|
|
||||||
|
actualIsFunc bool
|
||||||
|
actual interface{}
|
||||||
|
argsToForward []interface{}
|
||||||
|
|
||||||
|
timeoutInterval time.Duration
|
||||||
|
pollingInterval time.Duration
|
||||||
|
mustPassRepeatedly int
|
||||||
|
ctx context.Context
|
||||||
|
offset int
|
||||||
|
g *Gomega
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewAsyncAssertion(asyncType AsyncAssertionType, actualInput interface{}, g *Gomega, timeoutInterval time.Duration, pollingInterval time.Duration, mustPassRepeatedly int, ctx context.Context, offset int) *AsyncAssertion {
|
||||||
|
out := &AsyncAssertion{
|
||||||
|
asyncType: asyncType,
|
||||||
|
timeoutInterval: timeoutInterval,
|
||||||
|
pollingInterval: pollingInterval,
|
||||||
|
mustPassRepeatedly: mustPassRepeatedly,
|
||||||
|
offset: offset,
|
||||||
|
ctx: ctx,
|
||||||
|
g: g,
|
||||||
|
}
|
||||||
|
|
||||||
|
out.actual = actualInput
|
||||||
|
if actualInput != nil && reflect.TypeOf(actualInput).Kind() == reflect.Func {
|
||||||
|
out.actualIsFunc = true
|
||||||
|
}
|
||||||
|
|
||||||
|
return out
|
||||||
|
}
|
||||||
|
|
||||||
|
func (assertion *AsyncAssertion) WithOffset(offset int) types.AsyncAssertion {
|
||||||
|
assertion.offset = offset
|
||||||
|
return assertion
|
||||||
|
}
|
||||||
|
|
||||||
|
func (assertion *AsyncAssertion) WithTimeout(interval time.Duration) types.AsyncAssertion {
|
||||||
|
assertion.timeoutInterval = interval
|
||||||
|
return assertion
|
||||||
|
}
|
||||||
|
|
||||||
|
func (assertion *AsyncAssertion) WithPolling(interval time.Duration) types.AsyncAssertion {
|
||||||
|
assertion.pollingInterval = interval
|
||||||
|
return assertion
|
||||||
|
}
|
||||||
|
|
||||||
|
func (assertion *AsyncAssertion) Within(timeout time.Duration) types.AsyncAssertion {
|
||||||
|
assertion.timeoutInterval = timeout
|
||||||
|
return assertion
|
||||||
|
}
|
||||||
|
|
||||||
|
func (assertion *AsyncAssertion) ProbeEvery(interval time.Duration) types.AsyncAssertion {
|
||||||
|
assertion.pollingInterval = interval
|
||||||
|
return assertion
|
||||||
|
}
|
||||||
|
|
||||||
|
func (assertion *AsyncAssertion) WithContext(ctx context.Context) types.AsyncAssertion {
|
||||||
|
assertion.ctx = ctx
|
||||||
|
return assertion
|
||||||
|
}
|
||||||
|
|
||||||
|
func (assertion *AsyncAssertion) WithArguments(argsToForward ...interface{}) types.AsyncAssertion {
|
||||||
|
assertion.argsToForward = argsToForward
|
||||||
|
return assertion
|
||||||
|
}
|
||||||
|
|
||||||
|
func (assertion *AsyncAssertion) MustPassRepeatedly(count int) types.AsyncAssertion {
|
||||||
|
assertion.mustPassRepeatedly = count
|
||||||
|
return assertion
|
||||||
|
}
|
||||||
|
|
||||||
|
func (assertion *AsyncAssertion) Should(matcher types.GomegaMatcher, optionalDescription ...interface{}) bool {
|
||||||
|
assertion.g.THelper()
|
||||||
|
vetOptionalDescription("Asynchronous assertion", optionalDescription...)
|
||||||
|
return assertion.match(matcher, true, optionalDescription...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (assertion *AsyncAssertion) ShouldNot(matcher types.GomegaMatcher, optionalDescription ...interface{}) bool {
|
||||||
|
assertion.g.THelper()
|
||||||
|
vetOptionalDescription("Asynchronous assertion", optionalDescription...)
|
||||||
|
return assertion.match(matcher, false, optionalDescription...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (assertion *AsyncAssertion) buildDescription(optionalDescription ...interface{}) string {
|
||||||
|
switch len(optionalDescription) {
|
||||||
|
case 0:
|
||||||
|
return ""
|
||||||
|
case 1:
|
||||||
|
if describe, ok := optionalDescription[0].(func() string); ok {
|
||||||
|
return describe() + "\n"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return fmt.Sprintf(optionalDescription[0].(string), optionalDescription[1:]...) + "\n"
|
||||||
|
}
|
||||||
|
|
||||||
|
func (assertion *AsyncAssertion) processReturnValues(values []reflect.Value) (interface{}, error) {
|
||||||
|
if len(values) == 0 {
|
||||||
|
return nil, &asyncPolledActualError{
|
||||||
|
message: fmt.Sprintf("The function passed to %s did not return any values", assertion.asyncType),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
actual := values[0].Interface()
|
||||||
|
if _, ok := AsPollingSignalError(actual); ok {
|
||||||
|
return actual, actual.(error)
|
||||||
|
}
|
||||||
|
|
||||||
|
var err error
|
||||||
|
for i, extraValue := range values[1:] {
|
||||||
|
extra := extraValue.Interface()
|
||||||
|
if extra == nil {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if _, ok := AsPollingSignalError(extra); ok {
|
||||||
|
return actual, extra.(error)
|
||||||
|
}
|
||||||
|
extraType := reflect.TypeOf(extra)
|
||||||
|
zero := reflect.Zero(extraType).Interface()
|
||||||
|
if reflect.DeepEqual(extra, zero) {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if i == len(values)-2 && extraType.Implements(errInterface) {
|
||||||
|
err = extra.(error)
|
||||||
|
}
|
||||||
|
if err == nil {
|
||||||
|
err = &asyncPolledActualError{
|
||||||
|
message: fmt.Sprintf("The function passed to %s had an unexpected non-nil/non-zero return value at index %d:\n%s", assertion.asyncType, i+1, format.Object(extra, 1)),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return actual, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (assertion *AsyncAssertion) invalidFunctionError(t reflect.Type) error {
|
||||||
|
return fmt.Errorf(`The function passed to %s had an invalid signature of %s. Functions passed to %s must either:
|
||||||
|
|
||||||
|
(a) have return values or
|
||||||
|
(b) take a Gomega interface as their first argument and use that Gomega instance to make assertions.
|
||||||
|
|
||||||
|
You can learn more at https://onsi.github.io/gomega/#eventually
|
||||||
|
`, assertion.asyncType, t, assertion.asyncType)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (assertion *AsyncAssertion) noConfiguredContextForFunctionError() error {
|
||||||
|
return fmt.Errorf(`The function passed to %s requested a context.Context, but no context has been provided. Please pass one in using %s().WithContext().
|
||||||
|
|
||||||
|
You can learn more at https://onsi.github.io/gomega/#eventually
|
||||||
|
`, assertion.asyncType, assertion.asyncType)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (assertion *AsyncAssertion) argumentMismatchError(t reflect.Type, numProvided int) error {
|
||||||
|
have := "have"
|
||||||
|
if numProvided == 1 {
|
||||||
|
have = "has"
|
||||||
|
}
|
||||||
|
return fmt.Errorf(`The function passed to %s has signature %s takes %d arguments but %d %s been provided. Please use %s().WithArguments() to pass the corect set of arguments.
|
||||||
|
|
||||||
|
You can learn more at https://onsi.github.io/gomega/#eventually
|
||||||
|
`, assertion.asyncType, t, t.NumIn(), numProvided, have, assertion.asyncType)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (assertion *AsyncAssertion) invalidMustPassRepeatedlyError(reason string) error {
|
||||||
|
return fmt.Errorf(`Invalid use of MustPassRepeatedly with %s %s
|
||||||
|
|
||||||
|
You can learn more at https://onsi.github.io/gomega/#eventually
|
||||||
|
`, assertion.asyncType, reason)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (assertion *AsyncAssertion) buildActualPoller() (func() (interface{}, error), error) {
|
||||||
|
if !assertion.actualIsFunc {
|
||||||
|
return func() (interface{}, error) { return assertion.actual, nil }, nil
|
||||||
|
}
|
||||||
|
actualValue := reflect.ValueOf(assertion.actual)
|
||||||
|
actualType := reflect.TypeOf(assertion.actual)
|
||||||
|
numIn, numOut, isVariadic := actualType.NumIn(), actualType.NumOut(), actualType.IsVariadic()
|
||||||
|
|
||||||
|
if numIn == 0 && numOut == 0 {
|
||||||
|
return nil, assertion.invalidFunctionError(actualType)
|
||||||
|
}
|
||||||
|
takesGomega, takesContext := false, false
|
||||||
|
if numIn > 0 {
|
||||||
|
takesGomega, takesContext = actualType.In(0).Implements(gomegaType), actualType.In(0).Implements(contextType)
|
||||||
|
}
|
||||||
|
if takesGomega && numIn > 1 && actualType.In(1).Implements(contextType) {
|
||||||
|
takesContext = true
|
||||||
|
}
|
||||||
|
if takesContext && len(assertion.argsToForward) > 0 && reflect.TypeOf(assertion.argsToForward[0]).Implements(contextType) {
|
||||||
|
takesContext = false
|
||||||
|
}
|
||||||
|
if !takesGomega && numOut == 0 {
|
||||||
|
return nil, assertion.invalidFunctionError(actualType)
|
||||||
|
}
|
||||||
|
if takesContext && assertion.ctx == nil {
|
||||||
|
return nil, assertion.noConfiguredContextForFunctionError()
|
||||||
|
}
|
||||||
|
|
||||||
|
var assertionFailure error
|
||||||
|
inValues := []reflect.Value{}
|
||||||
|
if takesGomega {
|
||||||
|
inValues = append(inValues, reflect.ValueOf(NewGomega(assertion.g.DurationBundle).ConfigureWithFailHandler(func(message string, callerSkip ...int) {
|
||||||
|
skip := 0
|
||||||
|
if len(callerSkip) > 0 {
|
||||||
|
skip = callerSkip[0]
|
||||||
|
}
|
||||||
|
_, file, line, _ := runtime.Caller(skip + 1)
|
||||||
|
assertionFailure = &asyncPolledActualError{
|
||||||
|
message: fmt.Sprintf("The function passed to %s failed at %s:%d with:\n%s", assertion.asyncType, file, line, message),
|
||||||
|
}
|
||||||
|
// we throw an asyncGomegaHaltExecutionError so that defer GinkgoRecover() can catch this error if the user makes an assertion in a goroutine
|
||||||
|
panic(asyncGomegaHaltExecutionError{})
|
||||||
|
})))
|
||||||
|
}
|
||||||
|
if takesContext {
|
||||||
|
inValues = append(inValues, reflect.ValueOf(assertion.ctx))
|
||||||
|
}
|
||||||
|
for _, arg := range assertion.argsToForward {
|
||||||
|
inValues = append(inValues, reflect.ValueOf(arg))
|
||||||
|
}
|
||||||
|
|
||||||
|
if !isVariadic && numIn != len(inValues) {
|
||||||
|
return nil, assertion.argumentMismatchError(actualType, len(inValues))
|
||||||
|
} else if isVariadic && len(inValues) < numIn-1 {
|
||||||
|
return nil, assertion.argumentMismatchError(actualType, len(inValues))
|
||||||
|
}
|
||||||
|
|
||||||
|
if assertion.mustPassRepeatedly != 1 && assertion.asyncType != AsyncAssertionTypeEventually {
|
||||||
|
return nil, assertion.invalidMustPassRepeatedlyError("it can only be used with Eventually")
|
||||||
|
}
|
||||||
|
if assertion.mustPassRepeatedly < 1 {
|
||||||
|
return nil, assertion.invalidMustPassRepeatedlyError("parameter can't be < 1")
|
||||||
|
}
|
||||||
|
|
||||||
|
return func() (actual interface{}, err error) {
|
||||||
|
var values []reflect.Value
|
||||||
|
assertionFailure = nil
|
||||||
|
defer func() {
|
||||||
|
if numOut == 0 && takesGomega {
|
||||||
|
actual = assertionFailure
|
||||||
|
} else {
|
||||||
|
actual, err = assertion.processReturnValues(values)
|
||||||
|
_, isAsyncError := AsPollingSignalError(err)
|
||||||
|
if assertionFailure != nil && !isAsyncError {
|
||||||
|
err = assertionFailure
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if e := recover(); e != nil {
|
||||||
|
if _, isAsyncError := AsPollingSignalError(e); isAsyncError {
|
||||||
|
err = e.(error)
|
||||||
|
} else if assertionFailure == nil {
|
||||||
|
panic(e)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
values = actualValue.Call(inValues)
|
||||||
|
return
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (assertion *AsyncAssertion) afterTimeout() <-chan time.Time {
|
||||||
|
if assertion.timeoutInterval >= 0 {
|
||||||
|
return time.After(assertion.timeoutInterval)
|
||||||
|
}
|
||||||
|
|
||||||
|
if assertion.asyncType == AsyncAssertionTypeConsistently {
|
||||||
|
return time.After(assertion.g.DurationBundle.ConsistentlyDuration)
|
||||||
|
} else {
|
||||||
|
if assertion.ctx == nil {
|
||||||
|
return time.After(assertion.g.DurationBundle.EventuallyTimeout)
|
||||||
|
} else {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (assertion *AsyncAssertion) afterPolling() <-chan time.Time {
|
||||||
|
if assertion.pollingInterval >= 0 {
|
||||||
|
return time.After(assertion.pollingInterval)
|
||||||
|
}
|
||||||
|
if assertion.asyncType == AsyncAssertionTypeConsistently {
|
||||||
|
return time.After(assertion.g.DurationBundle.ConsistentlyPollingInterval)
|
||||||
|
} else {
|
||||||
|
return time.After(assertion.g.DurationBundle.EventuallyPollingInterval)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (assertion *AsyncAssertion) matcherSaysStopTrying(matcher types.GomegaMatcher, value interface{}) bool {
|
||||||
|
if assertion.actualIsFunc || types.MatchMayChangeInTheFuture(matcher, value) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
func (assertion *AsyncAssertion) pollMatcher(matcher types.GomegaMatcher, value interface{}) (matches bool, err error) {
|
||||||
|
defer func() {
|
||||||
|
if e := recover(); e != nil {
|
||||||
|
if _, isAsyncError := AsPollingSignalError(e); isAsyncError {
|
||||||
|
err = e.(error)
|
||||||
|
} else {
|
||||||
|
panic(e)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
matches, err = matcher.Match(value)
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (assertion *AsyncAssertion) match(matcher types.GomegaMatcher, desiredMatch bool, optionalDescription ...interface{}) bool {
|
||||||
|
timer := time.Now()
|
||||||
|
timeout := assertion.afterTimeout()
|
||||||
|
lock := sync.Mutex{}
|
||||||
|
|
||||||
|
var matches, hasLastValidActual bool
|
||||||
|
var actual, lastValidActual interface{}
|
||||||
|
var actualErr, matcherErr error
|
||||||
|
var oracleMatcherSaysStop bool
|
||||||
|
|
||||||
|
assertion.g.THelper()
|
||||||
|
|
||||||
|
pollActual, buildActualPollerErr := assertion.buildActualPoller()
|
||||||
|
if buildActualPollerErr != nil {
|
||||||
|
assertion.g.Fail(buildActualPollerErr.Error(), 2+assertion.offset)
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
actual, actualErr = pollActual()
|
||||||
|
if actualErr == nil {
|
||||||
|
lastValidActual = actual
|
||||||
|
hasLastValidActual = true
|
||||||
|
oracleMatcherSaysStop = assertion.matcherSaysStopTrying(matcher, actual)
|
||||||
|
matches, matcherErr = assertion.pollMatcher(matcher, actual)
|
||||||
|
}
|
||||||
|
|
||||||
|
renderError := func(preamble string, err error) string {
|
||||||
|
message := ""
|
||||||
|
if pollingSignalErr, ok := AsPollingSignalError(err); ok {
|
||||||
|
message = err.Error()
|
||||||
|
for _, attachment := range pollingSignalErr.Attachments {
|
||||||
|
message += fmt.Sprintf("\n%s:\n", attachment.Description)
|
||||||
|
message += format.Object(attachment.Object, 1)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
message = preamble + "\n" + format.Object(err, 1)
|
||||||
|
}
|
||||||
|
return message
|
||||||
|
}
|
||||||
|
|
||||||
|
messageGenerator := func() string {
|
||||||
|
// can be called out of band by Ginkgo if the user requests a progress report
|
||||||
|
lock.Lock()
|
||||||
|
defer lock.Unlock()
|
||||||
|
message := ""
|
||||||
|
|
||||||
|
if actualErr == nil {
|
||||||
|
if matcherErr == nil {
|
||||||
|
if desiredMatch != matches {
|
||||||
|
if desiredMatch {
|
||||||
|
message += matcher.FailureMessage(actual)
|
||||||
|
} else {
|
||||||
|
message += matcher.NegatedFailureMessage(actual)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if assertion.asyncType == AsyncAssertionTypeConsistently {
|
||||||
|
message += "There is no failure as the matcher passed to Consistently has not yet failed"
|
||||||
|
} else {
|
||||||
|
message += "There is no failure as the matcher passed to Eventually succeeded on its most recent iteration"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
var fgErr formattedGomegaError
|
||||||
|
if errors.As(actualErr, &fgErr) {
|
||||||
|
message += fgErr.FormattedGomegaError() + "\n"
|
||||||
|
} else {
|
||||||
|
message += renderError(fmt.Sprintf("The matcher passed to %s returned the following error:", assertion.asyncType), matcherErr)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
var fgErr formattedGomegaError
|
||||||
|
if errors.As(actualErr, &fgErr) {
|
||||||
|
message += fgErr.FormattedGomegaError() + "\n"
|
||||||
|
} else {
|
||||||
|
message += renderError(fmt.Sprintf("The function passed to %s returned the following error:", assertion.asyncType), actualErr)
|
||||||
|
}
|
||||||
|
if hasLastValidActual {
|
||||||
|
message += fmt.Sprintf("\nAt one point, however, the function did return successfully.\nYet, %s failed because", assertion.asyncType)
|
||||||
|
_, e := matcher.Match(lastValidActual)
|
||||||
|
if e != nil {
|
||||||
|
message += renderError(" the matcher returned the following error:", e)
|
||||||
|
} else {
|
||||||
|
message += " the matcher was not satisfied:\n"
|
||||||
|
if desiredMatch {
|
||||||
|
message += matcher.FailureMessage(lastValidActual)
|
||||||
|
} else {
|
||||||
|
message += matcher.NegatedFailureMessage(lastValidActual)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
description := assertion.buildDescription(optionalDescription...)
|
||||||
|
return fmt.Sprintf("%s%s", description, message)
|
||||||
|
}
|
||||||
|
|
||||||
|
fail := func(preamble string) {
|
||||||
|
assertion.g.THelper()
|
||||||
|
assertion.g.Fail(fmt.Sprintf("%s after %.3fs.\n%s", preamble, time.Since(timer).Seconds(), messageGenerator()), 3+assertion.offset)
|
||||||
|
}
|
||||||
|
|
||||||
|
var contextDone <-chan struct{}
|
||||||
|
if assertion.ctx != nil {
|
||||||
|
contextDone = assertion.ctx.Done()
|
||||||
|
if v, ok := assertion.ctx.Value("GINKGO_SPEC_CONTEXT").(contextWithAttachProgressReporter); ok {
|
||||||
|
detach := v.AttachProgressReporter(messageGenerator)
|
||||||
|
defer detach()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Used to count the number of times in a row a step passed
|
||||||
|
passedRepeatedlyCount := 0
|
||||||
|
for {
|
||||||
|
var nextPoll <-chan time.Time = nil
|
||||||
|
var isTryAgainAfterError = false
|
||||||
|
|
||||||
|
for _, err := range []error{actualErr, matcherErr} {
|
||||||
|
if pollingSignalErr, ok := AsPollingSignalError(err); ok {
|
||||||
|
if pollingSignalErr.IsStopTrying() {
|
||||||
|
fail("Told to stop trying")
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
if pollingSignalErr.IsTryAgainAfter() {
|
||||||
|
nextPoll = time.After(pollingSignalErr.TryAgainDuration())
|
||||||
|
isTryAgainAfterError = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if actualErr == nil && matcherErr == nil && matches == desiredMatch {
|
||||||
|
if assertion.asyncType == AsyncAssertionTypeEventually {
|
||||||
|
passedRepeatedlyCount += 1
|
||||||
|
if passedRepeatedlyCount == assertion.mustPassRepeatedly {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else if !isTryAgainAfterError {
|
||||||
|
if assertion.asyncType == AsyncAssertionTypeConsistently {
|
||||||
|
fail("Failed")
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
// Reset the consecutive pass count
|
||||||
|
passedRepeatedlyCount = 0
|
||||||
|
}
|
||||||
|
|
||||||
|
if oracleMatcherSaysStop {
|
||||||
|
if assertion.asyncType == AsyncAssertionTypeEventually {
|
||||||
|
fail("No future change is possible. Bailing out early")
|
||||||
|
return false
|
||||||
|
} else {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if nextPoll == nil {
|
||||||
|
nextPoll = assertion.afterPolling()
|
||||||
|
}
|
||||||
|
|
||||||
|
select {
|
||||||
|
case <-nextPoll:
|
||||||
|
a, e := pollActual()
|
||||||
|
lock.Lock()
|
||||||
|
actual, actualErr = a, e
|
||||||
|
lock.Unlock()
|
||||||
|
if actualErr == nil {
|
||||||
|
lock.Lock()
|
||||||
|
lastValidActual = actual
|
||||||
|
hasLastValidActual = true
|
||||||
|
lock.Unlock()
|
||||||
|
oracleMatcherSaysStop = assertion.matcherSaysStopTrying(matcher, actual)
|
||||||
|
m, e := assertion.pollMatcher(matcher, actual)
|
||||||
|
lock.Lock()
|
||||||
|
matches, matcherErr = m, e
|
||||||
|
lock.Unlock()
|
||||||
|
}
|
||||||
|
case <-contextDone:
|
||||||
|
fail("Context was cancelled")
|
||||||
|
return false
|
||||||
|
case <-timeout:
|
||||||
|
if assertion.asyncType == AsyncAssertionTypeEventually {
|
||||||
|
fail("Timed out")
|
||||||
|
return false
|
||||||
|
} else {
|
||||||
|
if isTryAgainAfterError {
|
||||||
|
fail("Timed out while waiting on TryAgainAfter")
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
71
vendor/github.com/onsi/gomega/internal/duration_bundle.go
generated
vendored
Normal file
71
vendor/github.com/onsi/gomega/internal/duration_bundle.go
generated
vendored
Normal file
@@ -0,0 +1,71 @@
|
|||||||
|
package internal
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
"reflect"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
type DurationBundle struct {
|
||||||
|
EventuallyTimeout time.Duration
|
||||||
|
EventuallyPollingInterval time.Duration
|
||||||
|
ConsistentlyDuration time.Duration
|
||||||
|
ConsistentlyPollingInterval time.Duration
|
||||||
|
}
|
||||||
|
|
||||||
|
const (
|
||||||
|
EventuallyTimeoutEnvVarName = "GOMEGA_DEFAULT_EVENTUALLY_TIMEOUT"
|
||||||
|
EventuallyPollingIntervalEnvVarName = "GOMEGA_DEFAULT_EVENTUALLY_POLLING_INTERVAL"
|
||||||
|
|
||||||
|
ConsistentlyDurationEnvVarName = "GOMEGA_DEFAULT_CONSISTENTLY_DURATION"
|
||||||
|
ConsistentlyPollingIntervalEnvVarName = "GOMEGA_DEFAULT_CONSISTENTLY_POLLING_INTERVAL"
|
||||||
|
)
|
||||||
|
|
||||||
|
func FetchDefaultDurationBundle() DurationBundle {
|
||||||
|
return DurationBundle{
|
||||||
|
EventuallyTimeout: durationFromEnv(EventuallyTimeoutEnvVarName, time.Second),
|
||||||
|
EventuallyPollingInterval: durationFromEnv(EventuallyPollingIntervalEnvVarName, 10*time.Millisecond),
|
||||||
|
|
||||||
|
ConsistentlyDuration: durationFromEnv(ConsistentlyDurationEnvVarName, 100*time.Millisecond),
|
||||||
|
ConsistentlyPollingInterval: durationFromEnv(ConsistentlyPollingIntervalEnvVarName, 10*time.Millisecond),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func durationFromEnv(key string, defaultDuration time.Duration) time.Duration {
|
||||||
|
value := os.Getenv(key)
|
||||||
|
if value == "" {
|
||||||
|
return defaultDuration
|
||||||
|
}
|
||||||
|
duration, err := time.ParseDuration(value)
|
||||||
|
if err != nil {
|
||||||
|
panic(fmt.Sprintf("Expected a duration when using %s! Parse error %v", key, err))
|
||||||
|
}
|
||||||
|
return duration
|
||||||
|
}
|
||||||
|
|
||||||
|
func toDuration(input interface{}) (time.Duration, error) {
|
||||||
|
duration, ok := input.(time.Duration)
|
||||||
|
if ok {
|
||||||
|
return duration, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
value := reflect.ValueOf(input)
|
||||||
|
kind := reflect.TypeOf(input).Kind()
|
||||||
|
|
||||||
|
if reflect.Int <= kind && kind <= reflect.Int64 {
|
||||||
|
return time.Duration(value.Int()) * time.Second, nil
|
||||||
|
} else if reflect.Uint <= kind && kind <= reflect.Uint64 {
|
||||||
|
return time.Duration(value.Uint()) * time.Second, nil
|
||||||
|
} else if reflect.Float32 <= kind && kind <= reflect.Float64 {
|
||||||
|
return time.Duration(value.Float() * float64(time.Second)), nil
|
||||||
|
} else if reflect.String == kind {
|
||||||
|
duration, err := time.ParseDuration(value.String())
|
||||||
|
if err != nil {
|
||||||
|
return 0, fmt.Errorf("%#v is not a valid parsable duration string: %w", input, err)
|
||||||
|
}
|
||||||
|
return duration, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0, fmt.Errorf("%#v is not a valid interval. Must be a time.Duration, a parsable duration string, or a number.", input)
|
||||||
|
}
|
||||||
129
vendor/github.com/onsi/gomega/internal/gomega.go
generated
vendored
Normal file
129
vendor/github.com/onsi/gomega/internal/gomega.go
generated
vendored
Normal file
@@ -0,0 +1,129 @@
|
|||||||
|
package internal
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/onsi/gomega/types"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Gomega struct {
|
||||||
|
Fail types.GomegaFailHandler
|
||||||
|
THelper func()
|
||||||
|
DurationBundle DurationBundle
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewGomega(bundle DurationBundle) *Gomega {
|
||||||
|
return &Gomega{
|
||||||
|
Fail: nil,
|
||||||
|
THelper: nil,
|
||||||
|
DurationBundle: bundle,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (g *Gomega) IsConfigured() bool {
|
||||||
|
return g.Fail != nil && g.THelper != nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (g *Gomega) ConfigureWithFailHandler(fail types.GomegaFailHandler) *Gomega {
|
||||||
|
g.Fail = fail
|
||||||
|
g.THelper = func() {}
|
||||||
|
return g
|
||||||
|
}
|
||||||
|
|
||||||
|
func (g *Gomega) ConfigureWithT(t types.GomegaTestingT) *Gomega {
|
||||||
|
g.Fail = func(message string, _ ...int) {
|
||||||
|
t.Helper()
|
||||||
|
t.Fatalf("\n%s", message)
|
||||||
|
}
|
||||||
|
g.THelper = t.Helper
|
||||||
|
return g
|
||||||
|
}
|
||||||
|
|
||||||
|
func (g *Gomega) Ω(actual interface{}, extra ...interface{}) types.Assertion {
|
||||||
|
return g.ExpectWithOffset(0, actual, extra...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (g *Gomega) Expect(actual interface{}, extra ...interface{}) types.Assertion {
|
||||||
|
return g.ExpectWithOffset(0, actual, extra...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (g *Gomega) ExpectWithOffset(offset int, actual interface{}, extra ...interface{}) types.Assertion {
|
||||||
|
return NewAssertion(actual, g, offset, extra...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (g *Gomega) Eventually(actualOrCtx interface{}, args ...interface{}) types.AsyncAssertion {
|
||||||
|
return g.makeAsyncAssertion(AsyncAssertionTypeEventually, 0, actualOrCtx, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (g *Gomega) EventuallyWithOffset(offset int, actualOrCtx interface{}, args ...interface{}) types.AsyncAssertion {
|
||||||
|
return g.makeAsyncAssertion(AsyncAssertionTypeEventually, offset, actualOrCtx, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (g *Gomega) Consistently(actualOrCtx interface{}, args ...interface{}) types.AsyncAssertion {
|
||||||
|
return g.makeAsyncAssertion(AsyncAssertionTypeConsistently, 0, actualOrCtx, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (g *Gomega) ConsistentlyWithOffset(offset int, actualOrCtx interface{}, args ...interface{}) types.AsyncAssertion {
|
||||||
|
return g.makeAsyncAssertion(AsyncAssertionTypeConsistently, offset, actualOrCtx, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (g *Gomega) makeAsyncAssertion(asyncAssertionType AsyncAssertionType, offset int, actualOrCtx interface{}, args ...interface{}) types.AsyncAssertion {
|
||||||
|
baseOffset := 3
|
||||||
|
timeoutInterval := -time.Duration(1)
|
||||||
|
pollingInterval := -time.Duration(1)
|
||||||
|
intervals := []interface{}{}
|
||||||
|
var ctx context.Context
|
||||||
|
|
||||||
|
actual := actualOrCtx
|
||||||
|
startingIndex := 0
|
||||||
|
if _, isCtx := actualOrCtx.(context.Context); isCtx && len(args) > 0 {
|
||||||
|
// the first argument is a context, we should accept it as the context _only if_ it is **not** the only argumnent **and** the second argument is not a parseable duration
|
||||||
|
// this is due to an unfortunate ambiguity in early version of Gomega in which multi-type durations are allowed after the actual
|
||||||
|
if _, err := toDuration(args[0]); err != nil {
|
||||||
|
ctx = actualOrCtx.(context.Context)
|
||||||
|
actual = args[0]
|
||||||
|
startingIndex = 1
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, arg := range args[startingIndex:] {
|
||||||
|
switch v := arg.(type) {
|
||||||
|
case context.Context:
|
||||||
|
ctx = v
|
||||||
|
default:
|
||||||
|
intervals = append(intervals, arg)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
var err error
|
||||||
|
if len(intervals) > 0 {
|
||||||
|
timeoutInterval, err = toDuration(intervals[0])
|
||||||
|
if err != nil {
|
||||||
|
g.Fail(err.Error(), offset+baseOffset)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if len(intervals) > 1 {
|
||||||
|
pollingInterval, err = toDuration(intervals[1])
|
||||||
|
if err != nil {
|
||||||
|
g.Fail(err.Error(), offset+baseOffset)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return NewAsyncAssertion(asyncAssertionType, actual, g, timeoutInterval, pollingInterval, 1, ctx, offset)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (g *Gomega) SetDefaultEventuallyTimeout(t time.Duration) {
|
||||||
|
g.DurationBundle.EventuallyTimeout = t
|
||||||
|
}
|
||||||
|
|
||||||
|
func (g *Gomega) SetDefaultEventuallyPollingInterval(t time.Duration) {
|
||||||
|
g.DurationBundle.EventuallyPollingInterval = t
|
||||||
|
}
|
||||||
|
|
||||||
|
func (g *Gomega) SetDefaultConsistentlyDuration(t time.Duration) {
|
||||||
|
g.DurationBundle.ConsistentlyDuration = t
|
||||||
|
}
|
||||||
|
|
||||||
|
func (g *Gomega) SetDefaultConsistentlyPollingInterval(t time.Duration) {
|
||||||
|
g.DurationBundle.ConsistentlyPollingInterval = t
|
||||||
|
}
|
||||||
48
vendor/github.com/onsi/gomega/internal/gutil/post_ioutil.go
generated
vendored
Normal file
48
vendor/github.com/onsi/gomega/internal/gutil/post_ioutil.go
generated
vendored
Normal file
@@ -0,0 +1,48 @@
|
|||||||
|
//go:build go1.16
|
||||||
|
// +build go1.16
|
||||||
|
|
||||||
|
// Package gutil is a replacement for ioutil, which should not be used in new
|
||||||
|
// code as of Go 1.16. With Go 1.16 and higher, this implementation
|
||||||
|
// uses the ioutil replacement functions in "io" and "os" with some
|
||||||
|
// Gomega specifics. This means that we should not get deprecation warnings
|
||||||
|
// for ioutil when they are added.
|
||||||
|
package gutil
|
||||||
|
|
||||||
|
import (
|
||||||
|
"io"
|
||||||
|
"os"
|
||||||
|
)
|
||||||
|
|
||||||
|
func NopCloser(r io.Reader) io.ReadCloser {
|
||||||
|
return io.NopCloser(r)
|
||||||
|
}
|
||||||
|
|
||||||
|
func ReadAll(r io.Reader) ([]byte, error) {
|
||||||
|
return io.ReadAll(r)
|
||||||
|
}
|
||||||
|
|
||||||
|
func ReadDir(dirname string) ([]string, error) {
|
||||||
|
entries, err := os.ReadDir(dirname)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
var names []string
|
||||||
|
for _, entry := range entries {
|
||||||
|
names = append(names, entry.Name())
|
||||||
|
}
|
||||||
|
|
||||||
|
return names, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func ReadFile(filename string) ([]byte, error) {
|
||||||
|
return os.ReadFile(filename)
|
||||||
|
}
|
||||||
|
|
||||||
|
func MkdirTemp(dir, pattern string) (string, error) {
|
||||||
|
return os.MkdirTemp(dir, pattern)
|
||||||
|
}
|
||||||
|
|
||||||
|
func WriteFile(filename string, data []byte) error {
|
||||||
|
return os.WriteFile(filename, data, 0644)
|
||||||
|
}
|
||||||
47
vendor/github.com/onsi/gomega/internal/gutil/using_ioutil.go
generated
vendored
Normal file
47
vendor/github.com/onsi/gomega/internal/gutil/using_ioutil.go
generated
vendored
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
//go:build !go1.16
|
||||||
|
// +build !go1.16
|
||||||
|
|
||||||
|
// Package gutil is a replacement for ioutil, which should not be used in new
|
||||||
|
// code as of Go 1.16. With Go 1.15 and lower, this implementation
|
||||||
|
// uses the ioutil functions, meaning that although Gomega is not officially
|
||||||
|
// supported on these versions, it is still likely to work.
|
||||||
|
package gutil
|
||||||
|
|
||||||
|
import (
|
||||||
|
"io"
|
||||||
|
"io/ioutil"
|
||||||
|
)
|
||||||
|
|
||||||
|
func NopCloser(r io.Reader) io.ReadCloser {
|
||||||
|
return ioutil.NopCloser(r)
|
||||||
|
}
|
||||||
|
|
||||||
|
func ReadAll(r io.Reader) ([]byte, error) {
|
||||||
|
return ioutil.ReadAll(r)
|
||||||
|
}
|
||||||
|
|
||||||
|
func ReadDir(dirname string) ([]string, error) {
|
||||||
|
files, err := ioutil.ReadDir(dirname)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
var names []string
|
||||||
|
for _, file := range files {
|
||||||
|
names = append(names, file.Name())
|
||||||
|
}
|
||||||
|
|
||||||
|
return names, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func ReadFile(filename string) ([]byte, error) {
|
||||||
|
return ioutil.ReadFile(filename)
|
||||||
|
}
|
||||||
|
|
||||||
|
func MkdirTemp(dir, pattern string) (string, error) {
|
||||||
|
return ioutil.TempDir(dir, pattern)
|
||||||
|
}
|
||||||
|
|
||||||
|
func WriteFile(filename string, data []byte) error {
|
||||||
|
return ioutil.WriteFile(filename, data, 0644)
|
||||||
|
}
|
||||||
106
vendor/github.com/onsi/gomega/internal/polling_signal_error.go
generated
vendored
Normal file
106
vendor/github.com/onsi/gomega/internal/polling_signal_error.go
generated
vendored
Normal file
@@ -0,0 +1,106 @@
|
|||||||
|
package internal
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
type PollingSignalErrorType int
|
||||||
|
|
||||||
|
const (
|
||||||
|
PollingSignalErrorTypeStopTrying PollingSignalErrorType = iota
|
||||||
|
PollingSignalErrorTypeTryAgainAfter
|
||||||
|
)
|
||||||
|
|
||||||
|
type PollingSignalError interface {
|
||||||
|
error
|
||||||
|
Wrap(err error) PollingSignalError
|
||||||
|
Attach(description string, obj any) PollingSignalError
|
||||||
|
Now()
|
||||||
|
}
|
||||||
|
|
||||||
|
var StopTrying = func(message string) PollingSignalError {
|
||||||
|
return &PollingSignalErrorImpl{
|
||||||
|
message: message,
|
||||||
|
pollingSignalErrorType: PollingSignalErrorTypeStopTrying,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var TryAgainAfter = func(duration time.Duration) PollingSignalError {
|
||||||
|
return &PollingSignalErrorImpl{
|
||||||
|
message: fmt.Sprintf("told to try again after %s", duration),
|
||||||
|
duration: duration,
|
||||||
|
pollingSignalErrorType: PollingSignalErrorTypeTryAgainAfter,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
type PollingSignalErrorAttachment struct {
|
||||||
|
Description string
|
||||||
|
Object any
|
||||||
|
}
|
||||||
|
|
||||||
|
type PollingSignalErrorImpl struct {
|
||||||
|
message string
|
||||||
|
wrappedErr error
|
||||||
|
pollingSignalErrorType PollingSignalErrorType
|
||||||
|
duration time.Duration
|
||||||
|
Attachments []PollingSignalErrorAttachment
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *PollingSignalErrorImpl) Wrap(err error) PollingSignalError {
|
||||||
|
s.wrappedErr = err
|
||||||
|
return s
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *PollingSignalErrorImpl) Attach(description string, obj any) PollingSignalError {
|
||||||
|
s.Attachments = append(s.Attachments, PollingSignalErrorAttachment{description, obj})
|
||||||
|
return s
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *PollingSignalErrorImpl) Error() string {
|
||||||
|
if s.wrappedErr == nil {
|
||||||
|
return s.message
|
||||||
|
} else {
|
||||||
|
return s.message + ": " + s.wrappedErr.Error()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *PollingSignalErrorImpl) Unwrap() error {
|
||||||
|
if s == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return s.wrappedErr
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *PollingSignalErrorImpl) Now() {
|
||||||
|
panic(s)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *PollingSignalErrorImpl) IsStopTrying() bool {
|
||||||
|
return s.pollingSignalErrorType == PollingSignalErrorTypeStopTrying
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *PollingSignalErrorImpl) IsTryAgainAfter() bool {
|
||||||
|
return s.pollingSignalErrorType == PollingSignalErrorTypeTryAgainAfter
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *PollingSignalErrorImpl) TryAgainDuration() time.Duration {
|
||||||
|
return s.duration
|
||||||
|
}
|
||||||
|
|
||||||
|
func AsPollingSignalError(actual interface{}) (*PollingSignalErrorImpl, bool) {
|
||||||
|
if actual == nil {
|
||||||
|
return nil, false
|
||||||
|
}
|
||||||
|
if actualErr, ok := actual.(error); ok {
|
||||||
|
var target *PollingSignalErrorImpl
|
||||||
|
if errors.As(actualErr, &target) {
|
||||||
|
return target, true
|
||||||
|
} else {
|
||||||
|
return nil, false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil, false
|
||||||
|
}
|
||||||
22
vendor/github.com/onsi/gomega/internal/vetoptdesc.go
generated
vendored
Normal file
22
vendor/github.com/onsi/gomega/internal/vetoptdesc.go
generated
vendored
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
package internal
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
"github.com/onsi/gomega/types"
|
||||||
|
)
|
||||||
|
|
||||||
|
// vetOptionalDescription vets the optional description args: if it finds any
|
||||||
|
// Gomega matcher at the beginning it panics. This allows for rendering Gomega
|
||||||
|
// matchers as part of an optional Description, as long as they're not in the
|
||||||
|
// first slot.
|
||||||
|
func vetOptionalDescription(assertion string, optionalDescription ...interface{}) {
|
||||||
|
if len(optionalDescription) == 0 {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if _, isGomegaMatcher := optionalDescription[0].(types.GomegaMatcher); isGomegaMatcher {
|
||||||
|
panic(fmt.Sprintf("%s has a GomegaMatcher as the first element of optionalDescription.\n\t"+
|
||||||
|
"Do you mean to use And/Or/SatisfyAll/SatisfyAny to combine multiple matchers?",
|
||||||
|
assertion))
|
||||||
|
}
|
||||||
|
}
|
||||||
658
vendor/github.com/onsi/gomega/matchers.go
generated
vendored
Normal file
658
vendor/github.com/onsi/gomega/matchers.go
generated
vendored
Normal file
@@ -0,0 +1,658 @@
|
|||||||
|
package gomega
|
||||||
|
|
||||||
|
import (
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/google/go-cmp/cmp"
|
||||||
|
"github.com/onsi/gomega/matchers"
|
||||||
|
"github.com/onsi/gomega/types"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Equal uses reflect.DeepEqual to compare actual with expected. Equal is strict about
|
||||||
|
// types when performing comparisons.
|
||||||
|
// It is an error for both actual and expected to be nil. Use BeNil() instead.
|
||||||
|
func Equal(expected interface{}) types.GomegaMatcher {
|
||||||
|
return &matchers.EqualMatcher{
|
||||||
|
Expected: expected,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// BeEquivalentTo is more lax than Equal, allowing equality between different types.
|
||||||
|
// This is done by converting actual to have the type of expected before
|
||||||
|
// attempting equality with reflect.DeepEqual.
|
||||||
|
// It is an error for actual and expected to be nil. Use BeNil() instead.
|
||||||
|
func BeEquivalentTo(expected interface{}) types.GomegaMatcher {
|
||||||
|
return &matchers.BeEquivalentToMatcher{
|
||||||
|
Expected: expected,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// BeComparableTo uses gocmp.Equal from github.com/google/go-cmp (instead of reflect.DeepEqual) to perform a deep comparison.
|
||||||
|
// You can pass cmp.Option as options.
|
||||||
|
// It is an error for actual and expected to be nil. Use BeNil() instead.
|
||||||
|
func BeComparableTo(expected interface{}, opts ...cmp.Option) types.GomegaMatcher {
|
||||||
|
return &matchers.BeComparableToMatcher{
|
||||||
|
Expected: expected,
|
||||||
|
Options: opts,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// BeIdenticalTo uses the == operator to compare actual with expected.
|
||||||
|
// BeIdenticalTo is strict about types when performing comparisons.
|
||||||
|
// It is an error for both actual and expected to be nil. Use BeNil() instead.
|
||||||
|
func BeIdenticalTo(expected interface{}) types.GomegaMatcher {
|
||||||
|
return &matchers.BeIdenticalToMatcher{
|
||||||
|
Expected: expected,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// BeNil succeeds if actual is nil
|
||||||
|
func BeNil() types.GomegaMatcher {
|
||||||
|
return &matchers.BeNilMatcher{}
|
||||||
|
}
|
||||||
|
|
||||||
|
// BeTrue succeeds if actual is true
|
||||||
|
func BeTrue() types.GomegaMatcher {
|
||||||
|
return &matchers.BeTrueMatcher{}
|
||||||
|
}
|
||||||
|
|
||||||
|
// BeFalse succeeds if actual is false
|
||||||
|
func BeFalse() types.GomegaMatcher {
|
||||||
|
return &matchers.BeFalseMatcher{}
|
||||||
|
}
|
||||||
|
|
||||||
|
// HaveOccurred succeeds if actual is a non-nil error
|
||||||
|
// The typical Go error checking pattern looks like:
|
||||||
|
//
|
||||||
|
// err := SomethingThatMightFail()
|
||||||
|
// Expect(err).ShouldNot(HaveOccurred())
|
||||||
|
func HaveOccurred() types.GomegaMatcher {
|
||||||
|
return &matchers.HaveOccurredMatcher{}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Succeed passes if actual is a nil error
|
||||||
|
// Succeed is intended to be used with functions that return a single error value. Instead of
|
||||||
|
//
|
||||||
|
// err := SomethingThatMightFail()
|
||||||
|
// Expect(err).ShouldNot(HaveOccurred())
|
||||||
|
//
|
||||||
|
// You can write:
|
||||||
|
//
|
||||||
|
// Expect(SomethingThatMightFail()).Should(Succeed())
|
||||||
|
//
|
||||||
|
// It is a mistake to use Succeed with a function that has multiple return values. Gomega's Ω and Expect
|
||||||
|
// functions automatically trigger failure if any return values after the first return value are non-zero/non-nil.
|
||||||
|
// This means that Ω(MultiReturnFunc()).ShouldNot(Succeed()) can never pass.
|
||||||
|
func Succeed() types.GomegaMatcher {
|
||||||
|
return &matchers.SucceedMatcher{}
|
||||||
|
}
|
||||||
|
|
||||||
|
// MatchError succeeds if actual is a non-nil error that matches the passed in
|
||||||
|
// string, error, or matcher.
|
||||||
|
//
|
||||||
|
// These are valid use-cases:
|
||||||
|
//
|
||||||
|
// Expect(err).Should(MatchError("an error")) //asserts that err.Error() == "an error"
|
||||||
|
// Expect(err).Should(MatchError(SomeError)) //asserts that err == SomeError (via reflect.DeepEqual)
|
||||||
|
// Expect(err).Should(MatchError(ContainsSubstring("sprocket not found"))) // asserts that edrr.Error() contains substring "sprocket not found"
|
||||||
|
//
|
||||||
|
// It is an error for err to be nil or an object that does not implement the
|
||||||
|
// Error interface
|
||||||
|
func MatchError(expected interface{}) types.GomegaMatcher {
|
||||||
|
return &matchers.MatchErrorMatcher{
|
||||||
|
Expected: expected,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// BeClosed succeeds if actual is a closed channel.
|
||||||
|
// It is an error to pass a non-channel to BeClosed, it is also an error to pass nil
|
||||||
|
//
|
||||||
|
// In order to check whether or not the channel is closed, Gomega must try to read from the channel
|
||||||
|
// (even in the `ShouldNot(BeClosed())` case). You should keep this in mind if you wish to make subsequent assertions about
|
||||||
|
// values coming down the channel.
|
||||||
|
//
|
||||||
|
// Also, if you are testing that a *buffered* channel is closed you must first read all values out of the channel before
|
||||||
|
// asserting that it is closed (it is not possible to detect that a buffered-channel has been closed until all its buffered values are read).
|
||||||
|
//
|
||||||
|
// Finally, as a corollary: it is an error to check whether or not a send-only channel is closed.
|
||||||
|
func BeClosed() types.GomegaMatcher {
|
||||||
|
return &matchers.BeClosedMatcher{}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Receive succeeds if there is a value to be received on actual.
|
||||||
|
// Actual must be a channel (and cannot be a send-only channel) -- anything else is an error.
|
||||||
|
//
|
||||||
|
// Receive returns immediately and never blocks:
|
||||||
|
//
|
||||||
|
// - If there is nothing on the channel `c` then Expect(c).Should(Receive()) will fail and Ω(c).ShouldNot(Receive()) will pass.
|
||||||
|
//
|
||||||
|
// - If the channel `c` is closed then Expect(c).Should(Receive()) will fail and Ω(c).ShouldNot(Receive()) will pass.
|
||||||
|
//
|
||||||
|
// - If there is something on the channel `c` ready to be read, then Expect(c).Should(Receive()) will pass and Ω(c).ShouldNot(Receive()) will fail.
|
||||||
|
//
|
||||||
|
// If you have a go-routine running in the background that will write to channel `c` you can:
|
||||||
|
//
|
||||||
|
// Eventually(c).Should(Receive())
|
||||||
|
//
|
||||||
|
// This will timeout if nothing gets sent to `c` (you can modify the timeout interval as you normally do with `Eventually`)
|
||||||
|
//
|
||||||
|
// A similar use-case is to assert that no go-routine writes to a channel (for a period of time). You can do this with `Consistently`:
|
||||||
|
//
|
||||||
|
// Consistently(c).ShouldNot(Receive())
|
||||||
|
//
|
||||||
|
// You can pass `Receive` a matcher. If you do so, it will match the received object against the matcher. For example:
|
||||||
|
//
|
||||||
|
// Expect(c).Should(Receive(Equal("foo")))
|
||||||
|
//
|
||||||
|
// When given a matcher, `Receive` will always fail if there is nothing to be received on the channel.
|
||||||
|
//
|
||||||
|
// Passing Receive a matcher is especially useful when paired with Eventually:
|
||||||
|
//
|
||||||
|
// Eventually(c).Should(Receive(ContainSubstring("bar")))
|
||||||
|
//
|
||||||
|
// will repeatedly attempt to pull values out of `c` until a value matching "bar" is received.
|
||||||
|
//
|
||||||
|
// Finally, if you want to have a reference to the value *sent* to the channel you can pass the `Receive` matcher a pointer to a variable of the appropriate type:
|
||||||
|
//
|
||||||
|
// var myThing thing
|
||||||
|
// Eventually(thingChan).Should(Receive(&myThing))
|
||||||
|
// Expect(myThing.Sprocket).Should(Equal("foo"))
|
||||||
|
// Expect(myThing.IsValid()).Should(BeTrue())
|
||||||
|
func Receive(args ...interface{}) types.GomegaMatcher {
|
||||||
|
var arg interface{}
|
||||||
|
if len(args) > 0 {
|
||||||
|
arg = args[0]
|
||||||
|
}
|
||||||
|
|
||||||
|
return &matchers.ReceiveMatcher{
|
||||||
|
Arg: arg,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// BeSent succeeds if a value can be sent to actual.
|
||||||
|
// Actual must be a channel (and cannot be a receive-only channel) that can sent the type of the value passed into BeSent -- anything else is an error.
|
||||||
|
// In addition, actual must not be closed.
|
||||||
|
//
|
||||||
|
// BeSent never blocks:
|
||||||
|
//
|
||||||
|
// - If the channel `c` is not ready to receive then Expect(c).Should(BeSent("foo")) will fail immediately
|
||||||
|
// - If the channel `c` is eventually ready to receive then Eventually(c).Should(BeSent("foo")) will succeed.. presuming the channel becomes ready to receive before Eventually's timeout
|
||||||
|
// - If the channel `c` is closed then Expect(c).Should(BeSent("foo")) and Ω(c).ShouldNot(BeSent("foo")) will both fail immediately
|
||||||
|
//
|
||||||
|
// Of course, the value is actually sent to the channel. The point of `BeSent` is less to make an assertion about the availability of the channel (which is typically an implementation detail that your test should not be concerned with).
|
||||||
|
// Rather, the point of `BeSent` is to make it possible to easily and expressively write tests that can timeout on blocked channel sends.
|
||||||
|
func BeSent(arg interface{}) types.GomegaMatcher {
|
||||||
|
return &matchers.BeSentMatcher{
|
||||||
|
Arg: arg,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// MatchRegexp succeeds if actual is a string or stringer that matches the
|
||||||
|
// passed-in regexp. Optional arguments can be provided to construct a regexp
|
||||||
|
// via fmt.Sprintf().
|
||||||
|
func MatchRegexp(regexp string, args ...interface{}) types.GomegaMatcher {
|
||||||
|
return &matchers.MatchRegexpMatcher{
|
||||||
|
Regexp: regexp,
|
||||||
|
Args: args,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ContainSubstring succeeds if actual is a string or stringer that contains the
|
||||||
|
// passed-in substring. Optional arguments can be provided to construct the substring
|
||||||
|
// via fmt.Sprintf().
|
||||||
|
func ContainSubstring(substr string, args ...interface{}) types.GomegaMatcher {
|
||||||
|
return &matchers.ContainSubstringMatcher{
|
||||||
|
Substr: substr,
|
||||||
|
Args: args,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// HavePrefix succeeds if actual is a string or stringer that contains the
|
||||||
|
// passed-in string as a prefix. Optional arguments can be provided to construct
|
||||||
|
// via fmt.Sprintf().
|
||||||
|
func HavePrefix(prefix string, args ...interface{}) types.GomegaMatcher {
|
||||||
|
return &matchers.HavePrefixMatcher{
|
||||||
|
Prefix: prefix,
|
||||||
|
Args: args,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// HaveSuffix succeeds if actual is a string or stringer that contains the
|
||||||
|
// passed-in string as a suffix. Optional arguments can be provided to construct
|
||||||
|
// via fmt.Sprintf().
|
||||||
|
func HaveSuffix(suffix string, args ...interface{}) types.GomegaMatcher {
|
||||||
|
return &matchers.HaveSuffixMatcher{
|
||||||
|
Suffix: suffix,
|
||||||
|
Args: args,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// MatchJSON succeeds if actual is a string or stringer of JSON that matches
|
||||||
|
// the expected JSON. The JSONs are decoded and the resulting objects are compared via
|
||||||
|
// reflect.DeepEqual so things like key-ordering and whitespace shouldn't matter.
|
||||||
|
func MatchJSON(json interface{}) types.GomegaMatcher {
|
||||||
|
return &matchers.MatchJSONMatcher{
|
||||||
|
JSONToMatch: json,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// MatchXML succeeds if actual is a string or stringer of XML that matches
|
||||||
|
// the expected XML. The XMLs are decoded and the resulting objects are compared via
|
||||||
|
// reflect.DeepEqual so things like whitespaces shouldn't matter.
|
||||||
|
func MatchXML(xml interface{}) types.GomegaMatcher {
|
||||||
|
return &matchers.MatchXMLMatcher{
|
||||||
|
XMLToMatch: xml,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// MatchYAML succeeds if actual is a string or stringer of YAML that matches
|
||||||
|
// the expected YAML. The YAML's are decoded and the resulting objects are compared via
|
||||||
|
// reflect.DeepEqual so things like key-ordering and whitespace shouldn't matter.
|
||||||
|
func MatchYAML(yaml interface{}) types.GomegaMatcher {
|
||||||
|
return &matchers.MatchYAMLMatcher{
|
||||||
|
YAMLToMatch: yaml,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// BeEmpty succeeds if actual is empty. Actual must be of type string, array, map, chan, or slice.
|
||||||
|
func BeEmpty() types.GomegaMatcher {
|
||||||
|
return &matchers.BeEmptyMatcher{}
|
||||||
|
}
|
||||||
|
|
||||||
|
// HaveLen succeeds if actual has the passed-in length. Actual must be of type string, array, map, chan, or slice.
|
||||||
|
func HaveLen(count int) types.GomegaMatcher {
|
||||||
|
return &matchers.HaveLenMatcher{
|
||||||
|
Count: count,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// HaveCap succeeds if actual has the passed-in capacity. Actual must be of type array, chan, or slice.
|
||||||
|
func HaveCap(count int) types.GomegaMatcher {
|
||||||
|
return &matchers.HaveCapMatcher{
|
||||||
|
Count: count,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// BeZero succeeds if actual is the zero value for its type or if actual is nil.
|
||||||
|
func BeZero() types.GomegaMatcher {
|
||||||
|
return &matchers.BeZeroMatcher{}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ContainElement succeeds if actual contains the passed in element. By default
|
||||||
|
// ContainElement() uses Equal() to perform the match, however a matcher can be
|
||||||
|
// passed in instead:
|
||||||
|
//
|
||||||
|
// Expect([]string{"Foo", "FooBar"}).Should(ContainElement(ContainSubstring("Bar")))
|
||||||
|
//
|
||||||
|
// Actual must be an array, slice or map. For maps, ContainElement searches
|
||||||
|
// through the map's values.
|
||||||
|
//
|
||||||
|
// If you want to have a copy of the matching element(s) found you can pass a
|
||||||
|
// pointer to a variable of the appropriate type. If the variable isn't a slice
|
||||||
|
// or map, then exactly one match will be expected and returned. If the variable
|
||||||
|
// is a slice or map, then at least one match is expected and all matches will be
|
||||||
|
// stored in the variable.
|
||||||
|
//
|
||||||
|
// var findings []string
|
||||||
|
// Expect([]string{"Foo", "FooBar"}).Should(ContainElement(ContainSubString("Bar", &findings)))
|
||||||
|
func ContainElement(element interface{}, result ...interface{}) types.GomegaMatcher {
|
||||||
|
return &matchers.ContainElementMatcher{
|
||||||
|
Element: element,
|
||||||
|
Result: result,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// BeElementOf succeeds if actual is contained in the passed in elements.
|
||||||
|
// BeElementOf() always uses Equal() to perform the match.
|
||||||
|
// When the passed in elements are comprised of a single element that is either an Array or Slice, BeElementOf() behaves
|
||||||
|
// as the reverse of ContainElement() that operates with Equal() to perform the match.
|
||||||
|
//
|
||||||
|
// Expect(2).Should(BeElementOf([]int{1, 2}))
|
||||||
|
// Expect(2).Should(BeElementOf([2]int{1, 2}))
|
||||||
|
//
|
||||||
|
// Otherwise, BeElementOf() provides a syntactic sugar for Or(Equal(_), Equal(_), ...):
|
||||||
|
//
|
||||||
|
// Expect(2).Should(BeElementOf(1, 2))
|
||||||
|
//
|
||||||
|
// Actual must be typed.
|
||||||
|
func BeElementOf(elements ...interface{}) types.GomegaMatcher {
|
||||||
|
return &matchers.BeElementOfMatcher{
|
||||||
|
Elements: elements,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// BeKeyOf succeeds if actual is contained in the keys of the passed in map.
|
||||||
|
// BeKeyOf() always uses Equal() to perform the match between actual and the map keys.
|
||||||
|
//
|
||||||
|
// Expect("foo").Should(BeKeyOf(map[string]bool{"foo": true, "bar": false}))
|
||||||
|
func BeKeyOf(element interface{}) types.GomegaMatcher {
|
||||||
|
return &matchers.BeKeyOfMatcher{
|
||||||
|
Map: element,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ConsistOf succeeds if actual contains precisely the elements passed into the matcher. The ordering of the elements does not matter.
|
||||||
|
// By default ConsistOf() uses Equal() to match the elements, however custom matchers can be passed in instead. Here are some examples:
|
||||||
|
//
|
||||||
|
// Expect([]string{"Foo", "FooBar"}).Should(ConsistOf("FooBar", "Foo"))
|
||||||
|
// Expect([]string{"Foo", "FooBar"}).Should(ConsistOf(ContainSubstring("Bar"), "Foo"))
|
||||||
|
// Expect([]string{"Foo", "FooBar"}).Should(ConsistOf(ContainSubstring("Foo"), ContainSubstring("Foo")))
|
||||||
|
//
|
||||||
|
// Actual must be an array, slice or map. For maps, ConsistOf matches against the map's values.
|
||||||
|
//
|
||||||
|
// You typically pass variadic arguments to ConsistOf (as in the examples above). However, if you need to pass in a slice you can provided that it
|
||||||
|
// is the only element passed in to ConsistOf:
|
||||||
|
//
|
||||||
|
// Expect([]string{"Foo", "FooBar"}).Should(ConsistOf([]string{"FooBar", "Foo"}))
|
||||||
|
//
|
||||||
|
// Note that Go's type system does not allow you to write this as ConsistOf([]string{"FooBar", "Foo"}...) as []string and []interface{} are different types - hence the need for this special rule.
|
||||||
|
func ConsistOf(elements ...interface{}) types.GomegaMatcher {
|
||||||
|
return &matchers.ConsistOfMatcher{
|
||||||
|
Elements: elements,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// HaveExactElemets succeeds if actual contains elements that precisely match the elemets passed into the matcher. The ordering of the elements does matter.
|
||||||
|
// By default HaveExactElements() uses Equal() to match the elements, however custom matchers can be passed in instead. Here are some examples:
|
||||||
|
//
|
||||||
|
// Expect([]string{"Foo", "FooBar"}).Should(HaveExactElements("Foo", "FooBar"))
|
||||||
|
// Expect([]string{"Foo", "FooBar"}).Should(HaveExactElements("Foo", ContainSubstring("Bar")))
|
||||||
|
// Expect([]string{"Foo", "FooBar"}).Should(HaveExactElements(ContainSubstring("Foo"), ContainSubstring("Foo")))
|
||||||
|
//
|
||||||
|
// Actual must be an array or slice.
|
||||||
|
func HaveExactElements(elements ...interface{}) types.GomegaMatcher {
|
||||||
|
return &matchers.HaveExactElementsMatcher{
|
||||||
|
Elements: elements,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ContainElements succeeds if actual contains the passed in elements. The ordering of the elements does not matter.
|
||||||
|
// By default ContainElements() uses Equal() to match the elements, however custom matchers can be passed in instead. Here are some examples:
|
||||||
|
//
|
||||||
|
// Expect([]string{"Foo", "FooBar"}).Should(ContainElements("FooBar"))
|
||||||
|
// Expect([]string{"Foo", "FooBar"}).Should(ContainElements(ContainSubstring("Bar"), "Foo"))
|
||||||
|
//
|
||||||
|
// Actual must be an array, slice or map.
|
||||||
|
// For maps, ContainElements searches through the map's values.
|
||||||
|
func ContainElements(elements ...interface{}) types.GomegaMatcher {
|
||||||
|
return &matchers.ContainElementsMatcher{
|
||||||
|
Elements: elements,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// HaveEach succeeds if actual solely contains elements that match the passed in element.
|
||||||
|
// Please note that if actual is empty, HaveEach always will succeed.
|
||||||
|
// By default HaveEach() uses Equal() to perform the match, however a
|
||||||
|
// matcher can be passed in instead:
|
||||||
|
//
|
||||||
|
// Expect([]string{"Foo", "FooBar"}).Should(HaveEach(ContainSubstring("Foo")))
|
||||||
|
//
|
||||||
|
// Actual must be an array, slice or map.
|
||||||
|
// For maps, HaveEach searches through the map's values.
|
||||||
|
func HaveEach(element interface{}) types.GomegaMatcher {
|
||||||
|
return &matchers.HaveEachMatcher{
|
||||||
|
Element: element,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// HaveKey succeeds if actual is a map with the passed in key.
|
||||||
|
// By default HaveKey uses Equal() to perform the match, however a
|
||||||
|
// matcher can be passed in instead:
|
||||||
|
//
|
||||||
|
// Expect(map[string]string{"Foo": "Bar", "BazFoo": "Duck"}).Should(HaveKey(MatchRegexp(`.+Foo$`)))
|
||||||
|
func HaveKey(key interface{}) types.GomegaMatcher {
|
||||||
|
return &matchers.HaveKeyMatcher{
|
||||||
|
Key: key,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// HaveKeyWithValue succeeds if actual is a map with the passed in key and value.
|
||||||
|
// By default HaveKeyWithValue uses Equal() to perform the match, however a
|
||||||
|
// matcher can be passed in instead:
|
||||||
|
//
|
||||||
|
// Expect(map[string]string{"Foo": "Bar", "BazFoo": "Duck"}).Should(HaveKeyWithValue("Foo", "Bar"))
|
||||||
|
// Expect(map[string]string{"Foo": "Bar", "BazFoo": "Duck"}).Should(HaveKeyWithValue(MatchRegexp(`.+Foo$`), "Bar"))
|
||||||
|
func HaveKeyWithValue(key interface{}, value interface{}) types.GomegaMatcher {
|
||||||
|
return &matchers.HaveKeyWithValueMatcher{
|
||||||
|
Key: key,
|
||||||
|
Value: value,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// HaveField succeeds if actual is a struct and the value at the passed in field
|
||||||
|
// matches the passed in matcher. By default HaveField used Equal() to perform the match,
|
||||||
|
// however a matcher can be passed in in stead.
|
||||||
|
//
|
||||||
|
// The field must be a string that resolves to the name of a field in the struct. Structs can be traversed
|
||||||
|
// using the '.' delimiter. If the field ends with '()' a method named field is assumed to exist on the struct and is invoked.
|
||||||
|
// Such methods must take no arguments and return a single value:
|
||||||
|
//
|
||||||
|
// type Book struct {
|
||||||
|
// Title string
|
||||||
|
// Author Person
|
||||||
|
// }
|
||||||
|
// type Person struct {
|
||||||
|
// FirstName string
|
||||||
|
// LastName string
|
||||||
|
// DOB time.Time
|
||||||
|
// }
|
||||||
|
// Expect(book).To(HaveField("Title", "Les Miserables"))
|
||||||
|
// Expect(book).To(HaveField("Title", ContainSubstring("Les"))
|
||||||
|
// Expect(book).To(HaveField("Author.FirstName", Equal("Victor"))
|
||||||
|
// Expect(book).To(HaveField("Author.DOB.Year()", BeNumerically("<", 1900))
|
||||||
|
func HaveField(field string, expected interface{}) types.GomegaMatcher {
|
||||||
|
return &matchers.HaveFieldMatcher{
|
||||||
|
Field: field,
|
||||||
|
Expected: expected,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// HaveExistingField succeeds if actual is a struct and the specified field
|
||||||
|
// exists.
|
||||||
|
//
|
||||||
|
// HaveExistingField can be combined with HaveField in order to cover use cases
|
||||||
|
// with optional fields. HaveField alone would trigger an error in such situations.
|
||||||
|
//
|
||||||
|
// Expect(MrHarmless).NotTo(And(HaveExistingField("Title"), HaveField("Title", "Supervillain")))
|
||||||
|
func HaveExistingField(field string) types.GomegaMatcher {
|
||||||
|
return &matchers.HaveExistingFieldMatcher{
|
||||||
|
Field: field,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// HaveValue applies the given matcher to the value of actual, optionally and
|
||||||
|
// repeatedly dereferencing pointers or taking the concrete value of interfaces.
|
||||||
|
// Thus, the matcher will always be applied to non-pointer and non-interface
|
||||||
|
// values only. HaveValue will fail with an error if a pointer or interface is
|
||||||
|
// nil. It will also fail for more than 31 pointer or interface dereferences to
|
||||||
|
// guard against mistakenly applying it to arbitrarily deep linked pointers.
|
||||||
|
//
|
||||||
|
// HaveValue differs from gstruct.PointTo in that it does not expect actual to
|
||||||
|
// be a pointer (as gstruct.PointTo does) but instead also accepts non-pointer
|
||||||
|
// and even interface values.
|
||||||
|
//
|
||||||
|
// actual := 42
|
||||||
|
// Expect(actual).To(HaveValue(42))
|
||||||
|
// Expect(&actual).To(HaveValue(42))
|
||||||
|
func HaveValue(matcher types.GomegaMatcher) types.GomegaMatcher {
|
||||||
|
return &matchers.HaveValueMatcher{
|
||||||
|
Matcher: matcher,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// BeNumerically performs numerical assertions in a type-agnostic way.
|
||||||
|
// Actual and expected should be numbers, though the specific type of
|
||||||
|
// number is irrelevant (float32, float64, uint8, etc...).
|
||||||
|
//
|
||||||
|
// There are six, self-explanatory, supported comparators:
|
||||||
|
//
|
||||||
|
// Expect(1.0).Should(BeNumerically("==", 1))
|
||||||
|
// Expect(1.0).Should(BeNumerically("~", 0.999, 0.01))
|
||||||
|
// Expect(1.0).Should(BeNumerically(">", 0.9))
|
||||||
|
// Expect(1.0).Should(BeNumerically(">=", 1.0))
|
||||||
|
// Expect(1.0).Should(BeNumerically("<", 3))
|
||||||
|
// Expect(1.0).Should(BeNumerically("<=", 1.0))
|
||||||
|
func BeNumerically(comparator string, compareTo ...interface{}) types.GomegaMatcher {
|
||||||
|
return &matchers.BeNumericallyMatcher{
|
||||||
|
Comparator: comparator,
|
||||||
|
CompareTo: compareTo,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// BeTemporally compares time.Time's like BeNumerically
|
||||||
|
// Actual and expected must be time.Time. The comparators are the same as for BeNumerically
|
||||||
|
//
|
||||||
|
// Expect(time.Now()).Should(BeTemporally(">", time.Time{}))
|
||||||
|
// Expect(time.Now()).Should(BeTemporally("~", time.Now(), time.Second))
|
||||||
|
func BeTemporally(comparator string, compareTo time.Time, threshold ...time.Duration) types.GomegaMatcher {
|
||||||
|
return &matchers.BeTemporallyMatcher{
|
||||||
|
Comparator: comparator,
|
||||||
|
CompareTo: compareTo,
|
||||||
|
Threshold: threshold,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// BeAssignableToTypeOf succeeds if actual is assignable to the type of expected.
|
||||||
|
// It will return an error when one of the values is nil.
|
||||||
|
//
|
||||||
|
// Expect(0).Should(BeAssignableToTypeOf(0)) // Same values
|
||||||
|
// Expect(5).Should(BeAssignableToTypeOf(-1)) // different values same type
|
||||||
|
// Expect("foo").Should(BeAssignableToTypeOf("bar")) // different values same type
|
||||||
|
// Expect(struct{ Foo string }{}).Should(BeAssignableToTypeOf(struct{ Foo string }{}))
|
||||||
|
func BeAssignableToTypeOf(expected interface{}) types.GomegaMatcher {
|
||||||
|
return &matchers.AssignableToTypeOfMatcher{
|
||||||
|
Expected: expected,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Panic succeeds if actual is a function that, when invoked, panics.
|
||||||
|
// Actual must be a function that takes no arguments and returns no results.
|
||||||
|
func Panic() types.GomegaMatcher {
|
||||||
|
return &matchers.PanicMatcher{}
|
||||||
|
}
|
||||||
|
|
||||||
|
// PanicWith succeeds if actual is a function that, when invoked, panics with a specific value.
|
||||||
|
// Actual must be a function that takes no arguments and returns no results.
|
||||||
|
//
|
||||||
|
// By default PanicWith uses Equal() to perform the match, however a
|
||||||
|
// matcher can be passed in instead:
|
||||||
|
//
|
||||||
|
// Expect(fn).Should(PanicWith(MatchRegexp(`.+Foo$`)))
|
||||||
|
func PanicWith(expected interface{}) types.GomegaMatcher {
|
||||||
|
return &matchers.PanicMatcher{Expected: expected}
|
||||||
|
}
|
||||||
|
|
||||||
|
// BeAnExistingFile succeeds if a file exists.
|
||||||
|
// Actual must be a string representing the abs path to the file being checked.
|
||||||
|
func BeAnExistingFile() types.GomegaMatcher {
|
||||||
|
return &matchers.BeAnExistingFileMatcher{}
|
||||||
|
}
|
||||||
|
|
||||||
|
// BeARegularFile succeeds if a file exists and is a regular file.
|
||||||
|
// Actual must be a string representing the abs path to the file being checked.
|
||||||
|
func BeARegularFile() types.GomegaMatcher {
|
||||||
|
return &matchers.BeARegularFileMatcher{}
|
||||||
|
}
|
||||||
|
|
||||||
|
// BeADirectory succeeds if a file exists and is a directory.
|
||||||
|
// Actual must be a string representing the abs path to the file being checked.
|
||||||
|
func BeADirectory() types.GomegaMatcher {
|
||||||
|
return &matchers.BeADirectoryMatcher{}
|
||||||
|
}
|
||||||
|
|
||||||
|
// HaveHTTPStatus succeeds if the Status or StatusCode field of an HTTP response matches.
|
||||||
|
// Actual must be either a *http.Response or *httptest.ResponseRecorder.
|
||||||
|
// Expected must be either an int or a string.
|
||||||
|
//
|
||||||
|
// Expect(resp).Should(HaveHTTPStatus(http.StatusOK)) // asserts that resp.StatusCode == 200
|
||||||
|
// Expect(resp).Should(HaveHTTPStatus("404 Not Found")) // asserts that resp.Status == "404 Not Found"
|
||||||
|
// Expect(resp).Should(HaveHTTPStatus(http.StatusOK, http.StatusNoContent)) // asserts that resp.StatusCode == 200 || resp.StatusCode == 204
|
||||||
|
func HaveHTTPStatus(expected ...interface{}) types.GomegaMatcher {
|
||||||
|
return &matchers.HaveHTTPStatusMatcher{Expected: expected}
|
||||||
|
}
|
||||||
|
|
||||||
|
// HaveHTTPHeaderWithValue succeeds if the header is found and the value matches.
|
||||||
|
// Actual must be either a *http.Response or *httptest.ResponseRecorder.
|
||||||
|
// Expected must be a string header name, followed by a header value which
|
||||||
|
// can be a string, or another matcher.
|
||||||
|
func HaveHTTPHeaderWithValue(header string, value interface{}) types.GomegaMatcher {
|
||||||
|
return &matchers.HaveHTTPHeaderWithValueMatcher{
|
||||||
|
Header: header,
|
||||||
|
Value: value,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// HaveHTTPBody matches if the body matches.
|
||||||
|
// Actual must be either a *http.Response or *httptest.ResponseRecorder.
|
||||||
|
// Expected must be either a string, []byte, or other matcher
|
||||||
|
func HaveHTTPBody(expected interface{}) types.GomegaMatcher {
|
||||||
|
return &matchers.HaveHTTPBodyMatcher{Expected: expected}
|
||||||
|
}
|
||||||
|
|
||||||
|
// And succeeds only if all of the given matchers succeed.
|
||||||
|
// The matchers are tried in order, and will fail-fast if one doesn't succeed.
|
||||||
|
//
|
||||||
|
// Expect("hi").To(And(HaveLen(2), Equal("hi"))
|
||||||
|
//
|
||||||
|
// And(), Or(), Not() and WithTransform() allow matchers to be composed into complex expressions.
|
||||||
|
func And(ms ...types.GomegaMatcher) types.GomegaMatcher {
|
||||||
|
return &matchers.AndMatcher{Matchers: ms}
|
||||||
|
}
|
||||||
|
|
||||||
|
// SatisfyAll is an alias for And().
|
||||||
|
//
|
||||||
|
// Expect("hi").Should(SatisfyAll(HaveLen(2), Equal("hi")))
|
||||||
|
func SatisfyAll(matchers ...types.GomegaMatcher) types.GomegaMatcher {
|
||||||
|
return And(matchers...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Or succeeds if any of the given matchers succeed.
|
||||||
|
// The matchers are tried in order and will return immediately upon the first successful match.
|
||||||
|
//
|
||||||
|
// Expect("hi").To(Or(HaveLen(3), HaveLen(2))
|
||||||
|
//
|
||||||
|
// And(), Or(), Not() and WithTransform() allow matchers to be composed into complex expressions.
|
||||||
|
func Or(ms ...types.GomegaMatcher) types.GomegaMatcher {
|
||||||
|
return &matchers.OrMatcher{Matchers: ms}
|
||||||
|
}
|
||||||
|
|
||||||
|
// SatisfyAny is an alias for Or().
|
||||||
|
//
|
||||||
|
// Expect("hi").SatisfyAny(Or(HaveLen(3), HaveLen(2))
|
||||||
|
func SatisfyAny(matchers ...types.GomegaMatcher) types.GomegaMatcher {
|
||||||
|
return Or(matchers...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Not negates the given matcher; it succeeds if the given matcher fails.
|
||||||
|
//
|
||||||
|
// Expect(1).To(Not(Equal(2))
|
||||||
|
//
|
||||||
|
// And(), Or(), Not() and WithTransform() allow matchers to be composed into complex expressions.
|
||||||
|
func Not(matcher types.GomegaMatcher) types.GomegaMatcher {
|
||||||
|
return &matchers.NotMatcher{Matcher: matcher}
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithTransform applies the `transform` to the actual value and matches it against `matcher`.
|
||||||
|
// The given transform must be either a function of one parameter that returns one value or a
|
||||||
|
// function of one parameter that returns two values, where the second value must be of the
|
||||||
|
// error type.
|
||||||
|
//
|
||||||
|
// var plus1 = func(i int) int { return i + 1 }
|
||||||
|
// Expect(1).To(WithTransform(plus1, Equal(2))
|
||||||
|
//
|
||||||
|
// var failingplus1 = func(i int) (int, error) { return 42, "this does not compute" }
|
||||||
|
// Expect(1).To(WithTransform(failingplus1, Equal(2)))
|
||||||
|
//
|
||||||
|
// And(), Or(), Not() and WithTransform() allow matchers to be composed into complex expressions.
|
||||||
|
func WithTransform(transform interface{}, matcher types.GomegaMatcher) types.GomegaMatcher {
|
||||||
|
return matchers.NewWithTransformMatcher(transform, matcher)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Satisfy matches the actual value against the `predicate` function.
|
||||||
|
// The given predicate must be a function of one paramter that returns bool.
|
||||||
|
//
|
||||||
|
// var isEven = func(i int) bool { return i%2 == 0 }
|
||||||
|
// Expect(2).To(Satisfy(isEven))
|
||||||
|
func Satisfy(predicate interface{}) types.GomegaMatcher {
|
||||||
|
return matchers.NewSatisfyMatcher(predicate)
|
||||||
|
}
|
||||||
62
vendor/github.com/onsi/gomega/matchers/and.go
generated
vendored
Normal file
62
vendor/github.com/onsi/gomega/matchers/and.go
generated
vendored
Normal file
@@ -0,0 +1,62 @@
|
|||||||
|
package matchers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
"github.com/onsi/gomega/format"
|
||||||
|
"github.com/onsi/gomega/types"
|
||||||
|
)
|
||||||
|
|
||||||
|
type AndMatcher struct {
|
||||||
|
Matchers []types.GomegaMatcher
|
||||||
|
|
||||||
|
// state
|
||||||
|
firstFailedMatcher types.GomegaMatcher
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *AndMatcher) Match(actual interface{}) (success bool, err error) {
|
||||||
|
m.firstFailedMatcher = nil
|
||||||
|
for _, matcher := range m.Matchers {
|
||||||
|
success, err := matcher.Match(actual)
|
||||||
|
if !success || err != nil {
|
||||||
|
m.firstFailedMatcher = matcher
|
||||||
|
return false, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *AndMatcher) FailureMessage(actual interface{}) (message string) {
|
||||||
|
return m.firstFailedMatcher.FailureMessage(actual)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *AndMatcher) NegatedFailureMessage(actual interface{}) (message string) {
|
||||||
|
// not the most beautiful list of matchers, but not bad either...
|
||||||
|
return format.Message(actual, fmt.Sprintf("To not satisfy all of these matchers: %s", m.Matchers))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *AndMatcher) MatchMayChangeInTheFuture(actual interface{}) bool {
|
||||||
|
/*
|
||||||
|
Example with 3 matchers: A, B, C
|
||||||
|
|
||||||
|
Match evaluates them: T, F, <?> => F
|
||||||
|
So match is currently F, what should MatchMayChangeInTheFuture() return?
|
||||||
|
Seems like it only depends on B, since currently B MUST change to allow the result to become T
|
||||||
|
|
||||||
|
Match eval: T, T, T => T
|
||||||
|
So match is currently T, what should MatchMayChangeInTheFuture() return?
|
||||||
|
Seems to depend on ANY of them being able to change to F.
|
||||||
|
*/
|
||||||
|
|
||||||
|
if m.firstFailedMatcher == nil {
|
||||||
|
// so all matchers succeeded.. Any one of them changing would change the result.
|
||||||
|
for _, matcher := range m.Matchers {
|
||||||
|
if types.MatchMayChangeInTheFuture(matcher, actual) {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false // none of were going to change
|
||||||
|
}
|
||||||
|
// one of the matchers failed.. it must be able to change in order to affect the result
|
||||||
|
return types.MatchMayChangeInTheFuture(m.firstFailedMatcher, actual)
|
||||||
|
}
|
||||||
37
vendor/github.com/onsi/gomega/matchers/assignable_to_type_of_matcher.go
generated
vendored
Normal file
37
vendor/github.com/onsi/gomega/matchers/assignable_to_type_of_matcher.go
generated
vendored
Normal file
@@ -0,0 +1,37 @@
|
|||||||
|
// untested sections: 2
|
||||||
|
|
||||||
|
package matchers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"reflect"
|
||||||
|
|
||||||
|
"github.com/onsi/gomega/format"
|
||||||
|
)
|
||||||
|
|
||||||
|
type AssignableToTypeOfMatcher struct {
|
||||||
|
Expected interface{}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *AssignableToTypeOfMatcher) Match(actual interface{}) (success bool, err error) {
|
||||||
|
if actual == nil && matcher.Expected == nil {
|
||||||
|
return false, fmt.Errorf("Refusing to compare <nil> to <nil>.\nBe explicit and use BeNil() instead. This is to avoid mistakes where both sides of an assertion are erroneously uninitialized.")
|
||||||
|
} else if matcher.Expected == nil {
|
||||||
|
return false, fmt.Errorf("Refusing to compare type to <nil>.\nBe explicit and use BeNil() instead. This is to avoid mistakes where both sides of an assertion are erroneously uninitialized.")
|
||||||
|
} else if actual == nil {
|
||||||
|
return false, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
actualType := reflect.TypeOf(actual)
|
||||||
|
expectedType := reflect.TypeOf(matcher.Expected)
|
||||||
|
|
||||||
|
return actualType.AssignableTo(expectedType), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *AssignableToTypeOfMatcher) FailureMessage(actual interface{}) string {
|
||||||
|
return format.Message(actual, fmt.Sprintf("to be assignable to the type: %T", matcher.Expected))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *AssignableToTypeOfMatcher) NegatedFailureMessage(actual interface{}) string {
|
||||||
|
return format.Message(actual, fmt.Sprintf("not to be assignable to the type: %T", matcher.Expected))
|
||||||
|
}
|
||||||
14
vendor/github.com/onsi/gomega/matchers/attributes_slice.go
generated
vendored
Normal file
14
vendor/github.com/onsi/gomega/matchers/attributes_slice.go
generated
vendored
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
package matchers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/xml"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
type attributesSlice []xml.Attr
|
||||||
|
|
||||||
|
func (attrs attributesSlice) Len() int { return len(attrs) }
|
||||||
|
func (attrs attributesSlice) Less(i, j int) bool {
|
||||||
|
return strings.Compare(attrs[i].Name.Local, attrs[j].Name.Local) == -1
|
||||||
|
}
|
||||||
|
func (attrs attributesSlice) Swap(i, j int) { attrs[i], attrs[j] = attrs[j], attrs[i] }
|
||||||
56
vendor/github.com/onsi/gomega/matchers/be_a_directory.go
generated
vendored
Normal file
56
vendor/github.com/onsi/gomega/matchers/be_a_directory.go
generated
vendored
Normal file
@@ -0,0 +1,56 @@
|
|||||||
|
// untested sections: 5
|
||||||
|
|
||||||
|
package matchers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
|
||||||
|
"github.com/onsi/gomega/format"
|
||||||
|
)
|
||||||
|
|
||||||
|
type notADirectoryError struct {
|
||||||
|
os.FileInfo
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t notADirectoryError) Error() string {
|
||||||
|
fileInfo := os.FileInfo(t)
|
||||||
|
switch {
|
||||||
|
case fileInfo.Mode().IsRegular():
|
||||||
|
return "file is a regular file"
|
||||||
|
default:
|
||||||
|
return fmt.Sprintf("file mode is: %s", fileInfo.Mode().String())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
type BeADirectoryMatcher struct {
|
||||||
|
expected interface{}
|
||||||
|
err error
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeADirectoryMatcher) Match(actual interface{}) (success bool, err error) {
|
||||||
|
actualFilename, ok := actual.(string)
|
||||||
|
if !ok {
|
||||||
|
return false, fmt.Errorf("BeADirectoryMatcher matcher expects a file path")
|
||||||
|
}
|
||||||
|
|
||||||
|
fileInfo, err := os.Stat(actualFilename)
|
||||||
|
if err != nil {
|
||||||
|
matcher.err = err
|
||||||
|
return false, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if !fileInfo.Mode().IsDir() {
|
||||||
|
matcher.err = notADirectoryError{fileInfo}
|
||||||
|
return false, nil
|
||||||
|
}
|
||||||
|
return true, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeADirectoryMatcher) FailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, fmt.Sprintf("to be a directory: %s", matcher.err))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeADirectoryMatcher) NegatedFailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, fmt.Sprintf("not be a directory"))
|
||||||
|
}
|
||||||
56
vendor/github.com/onsi/gomega/matchers/be_a_regular_file.go
generated
vendored
Normal file
56
vendor/github.com/onsi/gomega/matchers/be_a_regular_file.go
generated
vendored
Normal file
@@ -0,0 +1,56 @@
|
|||||||
|
// untested sections: 5
|
||||||
|
|
||||||
|
package matchers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
|
||||||
|
"github.com/onsi/gomega/format"
|
||||||
|
)
|
||||||
|
|
||||||
|
type notARegularFileError struct {
|
||||||
|
os.FileInfo
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t notARegularFileError) Error() string {
|
||||||
|
fileInfo := os.FileInfo(t)
|
||||||
|
switch {
|
||||||
|
case fileInfo.IsDir():
|
||||||
|
return "file is a directory"
|
||||||
|
default:
|
||||||
|
return fmt.Sprintf("file mode is: %s", fileInfo.Mode().String())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
type BeARegularFileMatcher struct {
|
||||||
|
expected interface{}
|
||||||
|
err error
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeARegularFileMatcher) Match(actual interface{}) (success bool, err error) {
|
||||||
|
actualFilename, ok := actual.(string)
|
||||||
|
if !ok {
|
||||||
|
return false, fmt.Errorf("BeARegularFileMatcher matcher expects a file path")
|
||||||
|
}
|
||||||
|
|
||||||
|
fileInfo, err := os.Stat(actualFilename)
|
||||||
|
if err != nil {
|
||||||
|
matcher.err = err
|
||||||
|
return false, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if !fileInfo.Mode().IsRegular() {
|
||||||
|
matcher.err = notARegularFileError{fileInfo}
|
||||||
|
return false, nil
|
||||||
|
}
|
||||||
|
return true, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeARegularFileMatcher) FailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, fmt.Sprintf("to be a regular file: %s", matcher.err))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeARegularFileMatcher) NegatedFailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, fmt.Sprintf("not be a regular file"))
|
||||||
|
}
|
||||||
40
vendor/github.com/onsi/gomega/matchers/be_an_existing_file.go
generated
vendored
Normal file
40
vendor/github.com/onsi/gomega/matchers/be_an_existing_file.go
generated
vendored
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
// untested sections: 3
|
||||||
|
|
||||||
|
package matchers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
|
||||||
|
"github.com/onsi/gomega/format"
|
||||||
|
)
|
||||||
|
|
||||||
|
type BeAnExistingFileMatcher struct {
|
||||||
|
expected interface{}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeAnExistingFileMatcher) Match(actual interface{}) (success bool, err error) {
|
||||||
|
actualFilename, ok := actual.(string)
|
||||||
|
if !ok {
|
||||||
|
return false, fmt.Errorf("BeAnExistingFileMatcher matcher expects a file path")
|
||||||
|
}
|
||||||
|
|
||||||
|
if _, err = os.Stat(actualFilename); err != nil {
|
||||||
|
switch {
|
||||||
|
case os.IsNotExist(err):
|
||||||
|
return false, nil
|
||||||
|
default:
|
||||||
|
return false, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeAnExistingFileMatcher) FailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, fmt.Sprintf("to exist"))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeAnExistingFileMatcher) NegatedFailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, fmt.Sprintf("not to exist"))
|
||||||
|
}
|
||||||
48
vendor/github.com/onsi/gomega/matchers/be_closed_matcher.go
generated
vendored
Normal file
48
vendor/github.com/onsi/gomega/matchers/be_closed_matcher.go
generated
vendored
Normal file
@@ -0,0 +1,48 @@
|
|||||||
|
// untested sections: 2
|
||||||
|
|
||||||
|
package matchers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"reflect"
|
||||||
|
|
||||||
|
"github.com/onsi/gomega/format"
|
||||||
|
)
|
||||||
|
|
||||||
|
type BeClosedMatcher struct {
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeClosedMatcher) Match(actual interface{}) (success bool, err error) {
|
||||||
|
if !isChan(actual) {
|
||||||
|
return false, fmt.Errorf("BeClosed matcher expects a channel. Got:\n%s", format.Object(actual, 1))
|
||||||
|
}
|
||||||
|
|
||||||
|
channelType := reflect.TypeOf(actual)
|
||||||
|
channelValue := reflect.ValueOf(actual)
|
||||||
|
|
||||||
|
if channelType.ChanDir() == reflect.SendDir {
|
||||||
|
return false, fmt.Errorf("BeClosed matcher cannot determine if a send-only channel is closed or open. Got:\n%s", format.Object(actual, 1))
|
||||||
|
}
|
||||||
|
|
||||||
|
winnerIndex, _, open := reflect.Select([]reflect.SelectCase{
|
||||||
|
{Dir: reflect.SelectRecv, Chan: channelValue},
|
||||||
|
{Dir: reflect.SelectDefault},
|
||||||
|
})
|
||||||
|
|
||||||
|
var closed bool
|
||||||
|
if winnerIndex == 0 {
|
||||||
|
closed = !open
|
||||||
|
} else if winnerIndex == 1 {
|
||||||
|
closed = false
|
||||||
|
}
|
||||||
|
|
||||||
|
return closed, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeClosedMatcher) FailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, "to be closed")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeClosedMatcher) NegatedFailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, "to be open")
|
||||||
|
}
|
||||||
49
vendor/github.com/onsi/gomega/matchers/be_comparable_to_matcher.go
generated
vendored
Normal file
49
vendor/github.com/onsi/gomega/matchers/be_comparable_to_matcher.go
generated
vendored
Normal file
@@ -0,0 +1,49 @@
|
|||||||
|
package matchers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
"github.com/google/go-cmp/cmp"
|
||||||
|
"github.com/onsi/gomega/format"
|
||||||
|
)
|
||||||
|
|
||||||
|
type BeComparableToMatcher struct {
|
||||||
|
Expected interface{}
|
||||||
|
Options cmp.Options
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeComparableToMatcher) Match(actual interface{}) (success bool, matchErr error) {
|
||||||
|
if actual == nil && matcher.Expected == nil {
|
||||||
|
return false, fmt.Errorf("Refusing to compare <nil> to <nil>.\nBe explicit and use BeNil() instead. This is to avoid mistakes where both sides of an assertion are erroneously uninitialized.")
|
||||||
|
}
|
||||||
|
// Shortcut for byte slices.
|
||||||
|
// Comparing long byte slices with reflect.DeepEqual is very slow,
|
||||||
|
// so use bytes.Equal if actual and expected are both byte slices.
|
||||||
|
if actualByteSlice, ok := actual.([]byte); ok {
|
||||||
|
if expectedByteSlice, ok := matcher.Expected.([]byte); ok {
|
||||||
|
return bytes.Equal(actualByteSlice, expectedByteSlice), nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
defer func() {
|
||||||
|
if r := recover(); r != nil {
|
||||||
|
success = false
|
||||||
|
if err, ok := r.(error); ok {
|
||||||
|
matchErr = err
|
||||||
|
} else if errMsg, ok := r.(string); ok {
|
||||||
|
matchErr = fmt.Errorf(errMsg)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
return cmp.Equal(actual, matcher.Expected, matcher.Options...), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeComparableToMatcher) FailureMessage(actual interface{}) (message string) {
|
||||||
|
return cmp.Diff(matcher.Expected, actual, matcher.Options)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeComparableToMatcher) NegatedFailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, "not to equal", matcher.Expected)
|
||||||
|
}
|
||||||
43
vendor/github.com/onsi/gomega/matchers/be_element_of_matcher.go
generated
vendored
Normal file
43
vendor/github.com/onsi/gomega/matchers/be_element_of_matcher.go
generated
vendored
Normal file
@@ -0,0 +1,43 @@
|
|||||||
|
// untested sections: 1
|
||||||
|
|
||||||
|
package matchers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"reflect"
|
||||||
|
|
||||||
|
"github.com/onsi/gomega/format"
|
||||||
|
)
|
||||||
|
|
||||||
|
type BeElementOfMatcher struct {
|
||||||
|
Elements []interface{}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeElementOfMatcher) Match(actual interface{}) (success bool, err error) {
|
||||||
|
if reflect.TypeOf(actual) == nil {
|
||||||
|
return false, fmt.Errorf("BeElement matcher expects actual to be typed")
|
||||||
|
}
|
||||||
|
|
||||||
|
var lastError error
|
||||||
|
for _, m := range flatten(matcher.Elements) {
|
||||||
|
matcher := &EqualMatcher{Expected: m}
|
||||||
|
success, err := matcher.Match(actual)
|
||||||
|
if err != nil {
|
||||||
|
lastError = err
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if success {
|
||||||
|
return true, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false, lastError
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeElementOfMatcher) FailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, "to be an element of", presentable(matcher.Elements))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeElementOfMatcher) NegatedFailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, "not to be an element of", presentable(matcher.Elements))
|
||||||
|
}
|
||||||
29
vendor/github.com/onsi/gomega/matchers/be_empty_matcher.go
generated
vendored
Normal file
29
vendor/github.com/onsi/gomega/matchers/be_empty_matcher.go
generated
vendored
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
// untested sections: 2
|
||||||
|
|
||||||
|
package matchers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
"github.com/onsi/gomega/format"
|
||||||
|
)
|
||||||
|
|
||||||
|
type BeEmptyMatcher struct {
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeEmptyMatcher) Match(actual interface{}) (success bool, err error) {
|
||||||
|
length, ok := lengthOf(actual)
|
||||||
|
if !ok {
|
||||||
|
return false, fmt.Errorf("BeEmpty matcher expects a string/array/map/channel/slice. Got:\n%s", format.Object(actual, 1))
|
||||||
|
}
|
||||||
|
|
||||||
|
return length == 0, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeEmptyMatcher) FailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, "to be empty")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeEmptyMatcher) NegatedFailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, "not to be empty")
|
||||||
|
}
|
||||||
36
vendor/github.com/onsi/gomega/matchers/be_equivalent_to_matcher.go
generated
vendored
Normal file
36
vendor/github.com/onsi/gomega/matchers/be_equivalent_to_matcher.go
generated
vendored
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
// untested sections: 2
|
||||||
|
|
||||||
|
package matchers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"reflect"
|
||||||
|
|
||||||
|
"github.com/onsi/gomega/format"
|
||||||
|
)
|
||||||
|
|
||||||
|
type BeEquivalentToMatcher struct {
|
||||||
|
Expected interface{}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeEquivalentToMatcher) Match(actual interface{}) (success bool, err error) {
|
||||||
|
if actual == nil && matcher.Expected == nil {
|
||||||
|
return false, fmt.Errorf("Both actual and expected must not be nil.")
|
||||||
|
}
|
||||||
|
|
||||||
|
convertedActual := actual
|
||||||
|
|
||||||
|
if actual != nil && matcher.Expected != nil && reflect.TypeOf(actual).ConvertibleTo(reflect.TypeOf(matcher.Expected)) {
|
||||||
|
convertedActual = reflect.ValueOf(actual).Convert(reflect.TypeOf(matcher.Expected)).Interface()
|
||||||
|
}
|
||||||
|
|
||||||
|
return reflect.DeepEqual(convertedActual, matcher.Expected), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeEquivalentToMatcher) FailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, "to be equivalent to", matcher.Expected)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeEquivalentToMatcher) NegatedFailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, "not to be equivalent to", matcher.Expected)
|
||||||
|
}
|
||||||
28
vendor/github.com/onsi/gomega/matchers/be_false_matcher.go
generated
vendored
Normal file
28
vendor/github.com/onsi/gomega/matchers/be_false_matcher.go
generated
vendored
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
// untested sections: 2
|
||||||
|
|
||||||
|
package matchers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
"github.com/onsi/gomega/format"
|
||||||
|
)
|
||||||
|
|
||||||
|
type BeFalseMatcher struct {
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeFalseMatcher) Match(actual interface{}) (success bool, err error) {
|
||||||
|
if !isBool(actual) {
|
||||||
|
return false, fmt.Errorf("Expected a boolean. Got:\n%s", format.Object(actual, 1))
|
||||||
|
}
|
||||||
|
|
||||||
|
return actual == false, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeFalseMatcher) FailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, "to be false")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeFalseMatcher) NegatedFailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, "not to be false")
|
||||||
|
}
|
||||||
39
vendor/github.com/onsi/gomega/matchers/be_identical_to.go
generated
vendored
Normal file
39
vendor/github.com/onsi/gomega/matchers/be_identical_to.go
generated
vendored
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
// untested sections: 2
|
||||||
|
|
||||||
|
package matchers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"runtime"
|
||||||
|
|
||||||
|
"github.com/onsi/gomega/format"
|
||||||
|
)
|
||||||
|
|
||||||
|
type BeIdenticalToMatcher struct {
|
||||||
|
Expected interface{}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeIdenticalToMatcher) Match(actual interface{}) (success bool, matchErr error) {
|
||||||
|
if actual == nil && matcher.Expected == nil {
|
||||||
|
return false, fmt.Errorf("Refusing to compare <nil> to <nil>.\nBe explicit and use BeNil() instead. This is to avoid mistakes where both sides of an assertion are erroneously uninitialized.")
|
||||||
|
}
|
||||||
|
|
||||||
|
defer func() {
|
||||||
|
if r := recover(); r != nil {
|
||||||
|
if _, ok := r.(runtime.Error); ok {
|
||||||
|
success = false
|
||||||
|
matchErr = nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
return actual == matcher.Expected, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeIdenticalToMatcher) FailureMessage(actual interface{}) string {
|
||||||
|
return format.Message(actual, "to be identical to", matcher.Expected)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeIdenticalToMatcher) NegatedFailureMessage(actual interface{}) string {
|
||||||
|
return format.Message(actual, "not to be identical to", matcher.Expected)
|
||||||
|
}
|
||||||
45
vendor/github.com/onsi/gomega/matchers/be_key_of_matcher.go
generated
vendored
Normal file
45
vendor/github.com/onsi/gomega/matchers/be_key_of_matcher.go
generated
vendored
Normal file
@@ -0,0 +1,45 @@
|
|||||||
|
package matchers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"reflect"
|
||||||
|
|
||||||
|
"github.com/onsi/gomega/format"
|
||||||
|
)
|
||||||
|
|
||||||
|
type BeKeyOfMatcher struct {
|
||||||
|
Map interface{}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeKeyOfMatcher) Match(actual interface{}) (success bool, err error) {
|
||||||
|
if !isMap(matcher.Map) {
|
||||||
|
return false, fmt.Errorf("BeKeyOf matcher needs expected to be a map type")
|
||||||
|
}
|
||||||
|
|
||||||
|
if reflect.TypeOf(actual) == nil {
|
||||||
|
return false, fmt.Errorf("BeKeyOf matcher expects actual to be typed")
|
||||||
|
}
|
||||||
|
|
||||||
|
var lastError error
|
||||||
|
for _, key := range reflect.ValueOf(matcher.Map).MapKeys() {
|
||||||
|
matcher := &EqualMatcher{Expected: key.Interface()}
|
||||||
|
success, err := matcher.Match(actual)
|
||||||
|
if err != nil {
|
||||||
|
lastError = err
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if success {
|
||||||
|
return true, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false, lastError
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeKeyOfMatcher) FailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, "to be a key of", presentable(valuesOf(matcher.Map)))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeKeyOfMatcher) NegatedFailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, "not to be a key of", presentable(valuesOf(matcher.Map)))
|
||||||
|
}
|
||||||
20
vendor/github.com/onsi/gomega/matchers/be_nil_matcher.go
generated
vendored
Normal file
20
vendor/github.com/onsi/gomega/matchers/be_nil_matcher.go
generated
vendored
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
// untested sections: 2
|
||||||
|
|
||||||
|
package matchers
|
||||||
|
|
||||||
|
import "github.com/onsi/gomega/format"
|
||||||
|
|
||||||
|
type BeNilMatcher struct {
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeNilMatcher) Match(actual interface{}) (success bool, err error) {
|
||||||
|
return isNil(actual), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeNilMatcher) FailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, "to be nil")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeNilMatcher) NegatedFailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, "not to be nil")
|
||||||
|
}
|
||||||
134
vendor/github.com/onsi/gomega/matchers/be_numerically_matcher.go
generated
vendored
Normal file
134
vendor/github.com/onsi/gomega/matchers/be_numerically_matcher.go
generated
vendored
Normal file
@@ -0,0 +1,134 @@
|
|||||||
|
// untested sections: 4
|
||||||
|
|
||||||
|
package matchers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"math"
|
||||||
|
|
||||||
|
"github.com/onsi/gomega/format"
|
||||||
|
)
|
||||||
|
|
||||||
|
type BeNumericallyMatcher struct {
|
||||||
|
Comparator string
|
||||||
|
CompareTo []interface{}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeNumericallyMatcher) FailureMessage(actual interface{}) (message string) {
|
||||||
|
return matcher.FormatFailureMessage(actual, false)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeNumericallyMatcher) NegatedFailureMessage(actual interface{}) (message string) {
|
||||||
|
return matcher.FormatFailureMessage(actual, true)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeNumericallyMatcher) FormatFailureMessage(actual interface{}, negated bool) (message string) {
|
||||||
|
if len(matcher.CompareTo) == 1 {
|
||||||
|
message = fmt.Sprintf("to be %s", matcher.Comparator)
|
||||||
|
} else {
|
||||||
|
message = fmt.Sprintf("to be within %v of %s", matcher.CompareTo[1], matcher.Comparator)
|
||||||
|
}
|
||||||
|
if negated {
|
||||||
|
message = "not " + message
|
||||||
|
}
|
||||||
|
return format.Message(actual, message, matcher.CompareTo[0])
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeNumericallyMatcher) Match(actual interface{}) (success bool, err error) {
|
||||||
|
if len(matcher.CompareTo) == 0 || len(matcher.CompareTo) > 2 {
|
||||||
|
return false, fmt.Errorf("BeNumerically requires 1 or 2 CompareTo arguments. Got:\n%s", format.Object(matcher.CompareTo, 1))
|
||||||
|
}
|
||||||
|
if !isNumber(actual) {
|
||||||
|
return false, fmt.Errorf("Expected a number. Got:\n%s", format.Object(actual, 1))
|
||||||
|
}
|
||||||
|
if !isNumber(matcher.CompareTo[0]) {
|
||||||
|
return false, fmt.Errorf("Expected a number. Got:\n%s", format.Object(matcher.CompareTo[0], 1))
|
||||||
|
}
|
||||||
|
if len(matcher.CompareTo) == 2 && !isNumber(matcher.CompareTo[1]) {
|
||||||
|
return false, fmt.Errorf("Expected a number. Got:\n%s", format.Object(matcher.CompareTo[1], 1))
|
||||||
|
}
|
||||||
|
|
||||||
|
switch matcher.Comparator {
|
||||||
|
case "==", "~", ">", ">=", "<", "<=":
|
||||||
|
default:
|
||||||
|
return false, fmt.Errorf("Unknown comparator: %s", matcher.Comparator)
|
||||||
|
}
|
||||||
|
|
||||||
|
if isFloat(actual) || isFloat(matcher.CompareTo[0]) {
|
||||||
|
var secondOperand float64 = 1e-8
|
||||||
|
if len(matcher.CompareTo) == 2 {
|
||||||
|
secondOperand = toFloat(matcher.CompareTo[1])
|
||||||
|
}
|
||||||
|
success = matcher.matchFloats(toFloat(actual), toFloat(matcher.CompareTo[0]), secondOperand)
|
||||||
|
} else if isInteger(actual) {
|
||||||
|
var secondOperand int64 = 0
|
||||||
|
if len(matcher.CompareTo) == 2 {
|
||||||
|
secondOperand = toInteger(matcher.CompareTo[1])
|
||||||
|
}
|
||||||
|
success = matcher.matchIntegers(toInteger(actual), toInteger(matcher.CompareTo[0]), secondOperand)
|
||||||
|
} else if isUnsignedInteger(actual) {
|
||||||
|
var secondOperand uint64 = 0
|
||||||
|
if len(matcher.CompareTo) == 2 {
|
||||||
|
secondOperand = toUnsignedInteger(matcher.CompareTo[1])
|
||||||
|
}
|
||||||
|
success = matcher.matchUnsignedIntegers(toUnsignedInteger(actual), toUnsignedInteger(matcher.CompareTo[0]), secondOperand)
|
||||||
|
} else {
|
||||||
|
return false, fmt.Errorf("Failed to compare:\n%s\n%s:\n%s", format.Object(actual, 1), matcher.Comparator, format.Object(matcher.CompareTo[0], 1))
|
||||||
|
}
|
||||||
|
|
||||||
|
return success, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeNumericallyMatcher) matchIntegers(actual, compareTo, threshold int64) (success bool) {
|
||||||
|
switch matcher.Comparator {
|
||||||
|
case "==", "~":
|
||||||
|
diff := actual - compareTo
|
||||||
|
return -threshold <= diff && diff <= threshold
|
||||||
|
case ">":
|
||||||
|
return (actual > compareTo)
|
||||||
|
case ">=":
|
||||||
|
return (actual >= compareTo)
|
||||||
|
case "<":
|
||||||
|
return (actual < compareTo)
|
||||||
|
case "<=":
|
||||||
|
return (actual <= compareTo)
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeNumericallyMatcher) matchUnsignedIntegers(actual, compareTo, threshold uint64) (success bool) {
|
||||||
|
switch matcher.Comparator {
|
||||||
|
case "==", "~":
|
||||||
|
if actual < compareTo {
|
||||||
|
actual, compareTo = compareTo, actual
|
||||||
|
}
|
||||||
|
return actual-compareTo <= threshold
|
||||||
|
case ">":
|
||||||
|
return (actual > compareTo)
|
||||||
|
case ">=":
|
||||||
|
return (actual >= compareTo)
|
||||||
|
case "<":
|
||||||
|
return (actual < compareTo)
|
||||||
|
case "<=":
|
||||||
|
return (actual <= compareTo)
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeNumericallyMatcher) matchFloats(actual, compareTo, threshold float64) (success bool) {
|
||||||
|
switch matcher.Comparator {
|
||||||
|
case "~":
|
||||||
|
return math.Abs(actual-compareTo) <= threshold
|
||||||
|
case "==":
|
||||||
|
return (actual == compareTo)
|
||||||
|
case ">":
|
||||||
|
return (actual > compareTo)
|
||||||
|
case ">=":
|
||||||
|
return (actual >= compareTo)
|
||||||
|
case "<":
|
||||||
|
return (actual < compareTo)
|
||||||
|
case "<=":
|
||||||
|
return (actual <= compareTo)
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
73
vendor/github.com/onsi/gomega/matchers/be_sent_matcher.go
generated
vendored
Normal file
73
vendor/github.com/onsi/gomega/matchers/be_sent_matcher.go
generated
vendored
Normal file
@@ -0,0 +1,73 @@
|
|||||||
|
// untested sections: 3
|
||||||
|
|
||||||
|
package matchers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"reflect"
|
||||||
|
|
||||||
|
"github.com/onsi/gomega/format"
|
||||||
|
)
|
||||||
|
|
||||||
|
type BeSentMatcher struct {
|
||||||
|
Arg interface{}
|
||||||
|
channelClosed bool
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeSentMatcher) Match(actual interface{}) (success bool, err error) {
|
||||||
|
if !isChan(actual) {
|
||||||
|
return false, fmt.Errorf("BeSent expects a channel. Got:\n%s", format.Object(actual, 1))
|
||||||
|
}
|
||||||
|
|
||||||
|
channelType := reflect.TypeOf(actual)
|
||||||
|
channelValue := reflect.ValueOf(actual)
|
||||||
|
|
||||||
|
if channelType.ChanDir() == reflect.RecvDir {
|
||||||
|
return false, fmt.Errorf("BeSent matcher cannot be passed a receive-only channel. Got:\n%s", format.Object(actual, 1))
|
||||||
|
}
|
||||||
|
|
||||||
|
argType := reflect.TypeOf(matcher.Arg)
|
||||||
|
assignable := argType.AssignableTo(channelType.Elem())
|
||||||
|
|
||||||
|
if !assignable {
|
||||||
|
return false, fmt.Errorf("Cannot pass:\n%s to the channel:\n%s\nThe types don't match.", format.Object(matcher.Arg, 1), format.Object(actual, 1))
|
||||||
|
}
|
||||||
|
|
||||||
|
argValue := reflect.ValueOf(matcher.Arg)
|
||||||
|
|
||||||
|
defer func() {
|
||||||
|
if e := recover(); e != nil {
|
||||||
|
success = false
|
||||||
|
err = fmt.Errorf("Cannot send to a closed channel")
|
||||||
|
matcher.channelClosed = true
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
winnerIndex, _, _ := reflect.Select([]reflect.SelectCase{
|
||||||
|
{Dir: reflect.SelectSend, Chan: channelValue, Send: argValue},
|
||||||
|
{Dir: reflect.SelectDefault},
|
||||||
|
})
|
||||||
|
|
||||||
|
var didSend bool
|
||||||
|
if winnerIndex == 0 {
|
||||||
|
didSend = true
|
||||||
|
}
|
||||||
|
|
||||||
|
return didSend, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeSentMatcher) FailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, "to send:", matcher.Arg)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeSentMatcher) NegatedFailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, "not to send:", matcher.Arg)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeSentMatcher) MatchMayChangeInTheFuture(actual interface{}) bool {
|
||||||
|
if !isChan(actual) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
return !matcher.channelClosed
|
||||||
|
}
|
||||||
68
vendor/github.com/onsi/gomega/matchers/be_temporally_matcher.go
generated
vendored
Normal file
68
vendor/github.com/onsi/gomega/matchers/be_temporally_matcher.go
generated
vendored
Normal file
@@ -0,0 +1,68 @@
|
|||||||
|
// untested sections: 3
|
||||||
|
|
||||||
|
package matchers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/onsi/gomega/format"
|
||||||
|
)
|
||||||
|
|
||||||
|
type BeTemporallyMatcher struct {
|
||||||
|
Comparator string
|
||||||
|
CompareTo time.Time
|
||||||
|
Threshold []time.Duration
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeTemporallyMatcher) FailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, fmt.Sprintf("to be %s", matcher.Comparator), matcher.CompareTo)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeTemporallyMatcher) NegatedFailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, fmt.Sprintf("not to be %s", matcher.Comparator), matcher.CompareTo)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeTemporallyMatcher) Match(actual interface{}) (bool, error) {
|
||||||
|
// predicate to test for time.Time type
|
||||||
|
isTime := func(t interface{}) bool {
|
||||||
|
_, ok := t.(time.Time)
|
||||||
|
return ok
|
||||||
|
}
|
||||||
|
|
||||||
|
if !isTime(actual) {
|
||||||
|
return false, fmt.Errorf("Expected a time.Time. Got:\n%s", format.Object(actual, 1))
|
||||||
|
}
|
||||||
|
|
||||||
|
switch matcher.Comparator {
|
||||||
|
case "==", "~", ">", ">=", "<", "<=":
|
||||||
|
default:
|
||||||
|
return false, fmt.Errorf("Unknown comparator: %s", matcher.Comparator)
|
||||||
|
}
|
||||||
|
|
||||||
|
var threshold = time.Millisecond
|
||||||
|
if len(matcher.Threshold) == 1 {
|
||||||
|
threshold = matcher.Threshold[0]
|
||||||
|
}
|
||||||
|
|
||||||
|
return matcher.matchTimes(actual.(time.Time), matcher.CompareTo, threshold), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeTemporallyMatcher) matchTimes(actual, compareTo time.Time, threshold time.Duration) (success bool) {
|
||||||
|
switch matcher.Comparator {
|
||||||
|
case "==":
|
||||||
|
return actual.Equal(compareTo)
|
||||||
|
case "~":
|
||||||
|
diff := actual.Sub(compareTo)
|
||||||
|
return -threshold <= diff && diff <= threshold
|
||||||
|
case ">":
|
||||||
|
return actual.After(compareTo)
|
||||||
|
case ">=":
|
||||||
|
return !actual.Before(compareTo)
|
||||||
|
case "<":
|
||||||
|
return actual.Before(compareTo)
|
||||||
|
case "<=":
|
||||||
|
return !actual.After(compareTo)
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
28
vendor/github.com/onsi/gomega/matchers/be_true_matcher.go
generated
vendored
Normal file
28
vendor/github.com/onsi/gomega/matchers/be_true_matcher.go
generated
vendored
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
// untested sections: 2
|
||||||
|
|
||||||
|
package matchers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
"github.com/onsi/gomega/format"
|
||||||
|
)
|
||||||
|
|
||||||
|
type BeTrueMatcher struct {
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeTrueMatcher) Match(actual interface{}) (success bool, err error) {
|
||||||
|
if !isBool(actual) {
|
||||||
|
return false, fmt.Errorf("Expected a boolean. Got:\n%s", format.Object(actual, 1))
|
||||||
|
}
|
||||||
|
|
||||||
|
return actual.(bool), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeTrueMatcher) FailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, "to be true")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeTrueMatcher) NegatedFailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, "not to be true")
|
||||||
|
}
|
||||||
28
vendor/github.com/onsi/gomega/matchers/be_zero_matcher.go
generated
vendored
Normal file
28
vendor/github.com/onsi/gomega/matchers/be_zero_matcher.go
generated
vendored
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
package matchers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"reflect"
|
||||||
|
|
||||||
|
"github.com/onsi/gomega/format"
|
||||||
|
)
|
||||||
|
|
||||||
|
type BeZeroMatcher struct {
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeZeroMatcher) Match(actual interface{}) (success bool, err error) {
|
||||||
|
if actual == nil {
|
||||||
|
return true, nil
|
||||||
|
}
|
||||||
|
zeroValue := reflect.Zero(reflect.TypeOf(actual)).Interface()
|
||||||
|
|
||||||
|
return reflect.DeepEqual(zeroValue, actual), nil
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeZeroMatcher) FailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, "to be zero-valued")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *BeZeroMatcher) NegatedFailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, "not to be zero-valued")
|
||||||
|
}
|
||||||
153
vendor/github.com/onsi/gomega/matchers/consist_of.go
generated
vendored
Normal file
153
vendor/github.com/onsi/gomega/matchers/consist_of.go
generated
vendored
Normal file
@@ -0,0 +1,153 @@
|
|||||||
|
// untested sections: 3
|
||||||
|
|
||||||
|
package matchers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"reflect"
|
||||||
|
|
||||||
|
"github.com/onsi/gomega/format"
|
||||||
|
"github.com/onsi/gomega/matchers/support/goraph/bipartitegraph"
|
||||||
|
)
|
||||||
|
|
||||||
|
type ConsistOfMatcher struct {
|
||||||
|
Elements []interface{}
|
||||||
|
missingElements []interface{}
|
||||||
|
extraElements []interface{}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *ConsistOfMatcher) Match(actual interface{}) (success bool, err error) {
|
||||||
|
if !isArrayOrSlice(actual) && !isMap(actual) {
|
||||||
|
return false, fmt.Errorf("ConsistOf matcher expects an array/slice/map. Got:\n%s", format.Object(actual, 1))
|
||||||
|
}
|
||||||
|
|
||||||
|
matchers := matchers(matcher.Elements)
|
||||||
|
values := valuesOf(actual)
|
||||||
|
|
||||||
|
bipartiteGraph, err := bipartitegraph.NewBipartiteGraph(values, matchers, neighbours)
|
||||||
|
if err != nil {
|
||||||
|
return false, err
|
||||||
|
}
|
||||||
|
|
||||||
|
edges := bipartiteGraph.LargestMatching()
|
||||||
|
if len(edges) == len(values) && len(edges) == len(matchers) {
|
||||||
|
return true, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var missingMatchers []interface{}
|
||||||
|
matcher.extraElements, missingMatchers = bipartiteGraph.FreeLeftRight(edges)
|
||||||
|
matcher.missingElements = equalMatchersToElements(missingMatchers)
|
||||||
|
|
||||||
|
return false, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func neighbours(value, matcher interface{}) (bool, error) {
|
||||||
|
match, err := matcher.(omegaMatcher).Match(value)
|
||||||
|
return match && err == nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func equalMatchersToElements(matchers []interface{}) (elements []interface{}) {
|
||||||
|
for _, matcher := range matchers {
|
||||||
|
if equalMatcher, ok := matcher.(*EqualMatcher); ok {
|
||||||
|
elements = append(elements, equalMatcher.Expected)
|
||||||
|
} else if _, ok := matcher.(*BeNilMatcher); ok {
|
||||||
|
elements = append(elements, nil)
|
||||||
|
} else {
|
||||||
|
elements = append(elements, matcher)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func flatten(elems []interface{}) []interface{} {
|
||||||
|
if len(elems) != 1 || !isArrayOrSlice(elems[0]) {
|
||||||
|
return elems
|
||||||
|
}
|
||||||
|
|
||||||
|
value := reflect.ValueOf(elems[0])
|
||||||
|
flattened := make([]interface{}, value.Len())
|
||||||
|
for i := 0; i < value.Len(); i++ {
|
||||||
|
flattened[i] = value.Index(i).Interface()
|
||||||
|
}
|
||||||
|
return flattened
|
||||||
|
}
|
||||||
|
|
||||||
|
func matchers(expectedElems []interface{}) (matchers []interface{}) {
|
||||||
|
for _, e := range flatten(expectedElems) {
|
||||||
|
if e == nil {
|
||||||
|
matchers = append(matchers, &BeNilMatcher{})
|
||||||
|
} else if matcher, isMatcher := e.(omegaMatcher); isMatcher {
|
||||||
|
matchers = append(matchers, matcher)
|
||||||
|
} else {
|
||||||
|
matchers = append(matchers, &EqualMatcher{Expected: e})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func presentable(elems []interface{}) interface{} {
|
||||||
|
elems = flatten(elems)
|
||||||
|
|
||||||
|
if len(elems) == 0 {
|
||||||
|
return []interface{}{}
|
||||||
|
}
|
||||||
|
|
||||||
|
sv := reflect.ValueOf(elems)
|
||||||
|
firstEl := sv.Index(0)
|
||||||
|
if firstEl.IsNil() {
|
||||||
|
return elems
|
||||||
|
}
|
||||||
|
tt := firstEl.Elem().Type()
|
||||||
|
for i := 1; i < sv.Len(); i++ {
|
||||||
|
el := sv.Index(i)
|
||||||
|
if el.IsNil() || (sv.Index(i).Elem().Type() != tt) {
|
||||||
|
return elems
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ss := reflect.MakeSlice(reflect.SliceOf(tt), sv.Len(), sv.Len())
|
||||||
|
for i := 0; i < sv.Len(); i++ {
|
||||||
|
ss.Index(i).Set(sv.Index(i).Elem())
|
||||||
|
}
|
||||||
|
|
||||||
|
return ss.Interface()
|
||||||
|
}
|
||||||
|
|
||||||
|
func valuesOf(actual interface{}) []interface{} {
|
||||||
|
value := reflect.ValueOf(actual)
|
||||||
|
values := []interface{}{}
|
||||||
|
if isMap(actual) {
|
||||||
|
keys := value.MapKeys()
|
||||||
|
for i := 0; i < value.Len(); i++ {
|
||||||
|
values = append(values, value.MapIndex(keys[i]).Interface())
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
for i := 0; i < value.Len(); i++ {
|
||||||
|
values = append(values, value.Index(i).Interface())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return values
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *ConsistOfMatcher) FailureMessage(actual interface{}) (message string) {
|
||||||
|
message = format.Message(actual, "to consist of", presentable(matcher.Elements))
|
||||||
|
message = appendMissingElements(message, matcher.missingElements)
|
||||||
|
if len(matcher.extraElements) > 0 {
|
||||||
|
message = fmt.Sprintf("%s\nthe extra elements were\n%s", message,
|
||||||
|
format.Object(presentable(matcher.extraElements), 1))
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func appendMissingElements(message string, missingElements []interface{}) string {
|
||||||
|
if len(missingElements) == 0 {
|
||||||
|
return message
|
||||||
|
}
|
||||||
|
return fmt.Sprintf("%s\nthe missing elements were\n%s", message,
|
||||||
|
format.Object(presentable(missingElements), 1))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *ConsistOfMatcher) NegatedFailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, "not to consist of", presentable(matcher.Elements))
|
||||||
|
}
|
||||||
174
vendor/github.com/onsi/gomega/matchers/contain_element_matcher.go
generated
vendored
Normal file
174
vendor/github.com/onsi/gomega/matchers/contain_element_matcher.go
generated
vendored
Normal file
@@ -0,0 +1,174 @@
|
|||||||
|
// untested sections: 2
|
||||||
|
|
||||||
|
package matchers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"reflect"
|
||||||
|
|
||||||
|
"github.com/onsi/gomega/format"
|
||||||
|
)
|
||||||
|
|
||||||
|
type ContainElementMatcher struct {
|
||||||
|
Element interface{}
|
||||||
|
Result []interface{}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *ContainElementMatcher) Match(actual interface{}) (success bool, err error) {
|
||||||
|
if !isArrayOrSlice(actual) && !isMap(actual) {
|
||||||
|
return false, fmt.Errorf("ContainElement matcher expects an array/slice/map. Got:\n%s", format.Object(actual, 1))
|
||||||
|
}
|
||||||
|
|
||||||
|
var actualT reflect.Type
|
||||||
|
var result reflect.Value
|
||||||
|
switch l := len(matcher.Result); {
|
||||||
|
case l > 1:
|
||||||
|
return false, errors.New("ContainElement matcher expects at most a single optional pointer to store its findings at")
|
||||||
|
case l == 1:
|
||||||
|
if reflect.ValueOf(matcher.Result[0]).Kind() != reflect.Ptr {
|
||||||
|
return false, fmt.Errorf("ContainElement matcher expects a non-nil pointer to store its findings at. Got\n%s",
|
||||||
|
format.Object(matcher.Result[0], 1))
|
||||||
|
}
|
||||||
|
actualT = reflect.TypeOf(actual)
|
||||||
|
resultReference := matcher.Result[0]
|
||||||
|
result = reflect.ValueOf(resultReference).Elem() // what ResultReference points to, to stash away our findings
|
||||||
|
switch result.Kind() {
|
||||||
|
case reflect.Array:
|
||||||
|
return false, fmt.Errorf("ContainElement cannot return findings. Need *%s, got *%s",
|
||||||
|
reflect.SliceOf(actualT.Elem()).String(), result.Type().String())
|
||||||
|
case reflect.Slice:
|
||||||
|
if !isArrayOrSlice(actual) {
|
||||||
|
return false, fmt.Errorf("ContainElement cannot return findings. Need *%s, got *%s",
|
||||||
|
reflect.MapOf(actualT.Key(), actualT.Elem()).String(), result.Type().String())
|
||||||
|
}
|
||||||
|
if !actualT.Elem().AssignableTo(result.Type().Elem()) {
|
||||||
|
return false, fmt.Errorf("ContainElement cannot return findings. Need *%s, got *%s",
|
||||||
|
actualT.String(), result.Type().String())
|
||||||
|
}
|
||||||
|
case reflect.Map:
|
||||||
|
if !isMap(actual) {
|
||||||
|
return false, fmt.Errorf("ContainElement cannot return findings. Need *%s, got *%s",
|
||||||
|
actualT.String(), result.Type().String())
|
||||||
|
}
|
||||||
|
if !actualT.AssignableTo(result.Type()) {
|
||||||
|
return false, fmt.Errorf("ContainElement cannot return findings. Need *%s, got *%s",
|
||||||
|
actualT.String(), result.Type().String())
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
if !actualT.Elem().AssignableTo(result.Type()) {
|
||||||
|
return false, fmt.Errorf("ContainElement cannot return findings. Need *%s, got *%s",
|
||||||
|
actualT.Elem().String(), result.Type().String())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
elemMatcher, elementIsMatcher := matcher.Element.(omegaMatcher)
|
||||||
|
if !elementIsMatcher {
|
||||||
|
elemMatcher = &EqualMatcher{Expected: matcher.Element}
|
||||||
|
}
|
||||||
|
|
||||||
|
value := reflect.ValueOf(actual)
|
||||||
|
var valueAt func(int) interface{}
|
||||||
|
|
||||||
|
var getFindings func() reflect.Value
|
||||||
|
var foundAt func(int)
|
||||||
|
|
||||||
|
if isMap(actual) {
|
||||||
|
keys := value.MapKeys()
|
||||||
|
valueAt = func(i int) interface{} {
|
||||||
|
return value.MapIndex(keys[i]).Interface()
|
||||||
|
}
|
||||||
|
if result.Kind() != reflect.Invalid {
|
||||||
|
fm := reflect.MakeMap(actualT)
|
||||||
|
getFindings = func() reflect.Value {
|
||||||
|
return fm
|
||||||
|
}
|
||||||
|
foundAt = func(i int) {
|
||||||
|
fm.SetMapIndex(keys[i], value.MapIndex(keys[i]))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
valueAt = func(i int) interface{} {
|
||||||
|
return value.Index(i).Interface()
|
||||||
|
}
|
||||||
|
if result.Kind() != reflect.Invalid {
|
||||||
|
var f reflect.Value
|
||||||
|
if result.Kind() == reflect.Slice {
|
||||||
|
f = reflect.MakeSlice(result.Type(), 0, 0)
|
||||||
|
} else {
|
||||||
|
f = reflect.MakeSlice(reflect.SliceOf(result.Type()), 0, 0)
|
||||||
|
}
|
||||||
|
getFindings = func() reflect.Value {
|
||||||
|
return f
|
||||||
|
}
|
||||||
|
foundAt = func(i int) {
|
||||||
|
f = reflect.Append(f, value.Index(i))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var lastError error
|
||||||
|
for i := 0; i < value.Len(); i++ {
|
||||||
|
elem := valueAt(i)
|
||||||
|
success, err := elemMatcher.Match(elem)
|
||||||
|
if err != nil {
|
||||||
|
lastError = err
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if success {
|
||||||
|
if result.Kind() == reflect.Invalid {
|
||||||
|
return true, nil
|
||||||
|
}
|
||||||
|
foundAt(i)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// when the expectation isn't interested in the findings except for success
|
||||||
|
// or non-success, then we're done here and return the last matcher error
|
||||||
|
// seen, if any, as well as non-success.
|
||||||
|
if result.Kind() == reflect.Invalid {
|
||||||
|
return false, lastError
|
||||||
|
}
|
||||||
|
|
||||||
|
// pick up any findings the test is interested in as it specified a non-nil
|
||||||
|
// result reference. However, the expection always is that there are at
|
||||||
|
// least one or multiple findings. So, if a result is expected, but we had
|
||||||
|
// no findings, then this is an error.
|
||||||
|
findings := getFindings()
|
||||||
|
if findings.Len() == 0 {
|
||||||
|
return false, lastError
|
||||||
|
}
|
||||||
|
|
||||||
|
// there's just a single finding and the result is neither a slice nor a map
|
||||||
|
// (so it's a scalar): pick the one and only finding and return it in the
|
||||||
|
// place the reference points to.
|
||||||
|
if findings.Len() == 1 && !isArrayOrSlice(result.Interface()) && !isMap(result.Interface()) {
|
||||||
|
if isMap(actual) {
|
||||||
|
miter := findings.MapRange()
|
||||||
|
miter.Next()
|
||||||
|
result.Set(miter.Value())
|
||||||
|
} else {
|
||||||
|
result.Set(findings.Index(0))
|
||||||
|
}
|
||||||
|
return true, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// at least one or even multiple findings and a the result references a
|
||||||
|
// slice or a map, so all we need to do is to store our findings where the
|
||||||
|
// reference points to.
|
||||||
|
if !findings.Type().AssignableTo(result.Type()) {
|
||||||
|
return false, fmt.Errorf("ContainElement cannot return multiple findings. Need *%s, got *%s",
|
||||||
|
findings.Type().String(), result.Type().String())
|
||||||
|
}
|
||||||
|
result.Set(findings)
|
||||||
|
return true, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *ContainElementMatcher) FailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, "to contain element matching", matcher.Element)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *ContainElementMatcher) NegatedFailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, "not to contain element matching", matcher.Element)
|
||||||
|
}
|
||||||
44
vendor/github.com/onsi/gomega/matchers/contain_elements_matcher.go
generated
vendored
Normal file
44
vendor/github.com/onsi/gomega/matchers/contain_elements_matcher.go
generated
vendored
Normal file
@@ -0,0 +1,44 @@
|
|||||||
|
package matchers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
"github.com/onsi/gomega/format"
|
||||||
|
"github.com/onsi/gomega/matchers/support/goraph/bipartitegraph"
|
||||||
|
)
|
||||||
|
|
||||||
|
type ContainElementsMatcher struct {
|
||||||
|
Elements []interface{}
|
||||||
|
missingElements []interface{}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *ContainElementsMatcher) Match(actual interface{}) (success bool, err error) {
|
||||||
|
if !isArrayOrSlice(actual) && !isMap(actual) {
|
||||||
|
return false, fmt.Errorf("ContainElements matcher expects an array/slice/map. Got:\n%s", format.Object(actual, 1))
|
||||||
|
}
|
||||||
|
|
||||||
|
matchers := matchers(matcher.Elements)
|
||||||
|
bipartiteGraph, err := bipartitegraph.NewBipartiteGraph(valuesOf(actual), matchers, neighbours)
|
||||||
|
if err != nil {
|
||||||
|
return false, err
|
||||||
|
}
|
||||||
|
|
||||||
|
edges := bipartiteGraph.LargestMatching()
|
||||||
|
if len(edges) == len(matchers) {
|
||||||
|
return true, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
_, missingMatchers := bipartiteGraph.FreeLeftRight(edges)
|
||||||
|
matcher.missingElements = equalMatchersToElements(missingMatchers)
|
||||||
|
|
||||||
|
return false, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *ContainElementsMatcher) FailureMessage(actual interface{}) (message string) {
|
||||||
|
message = format.Message(actual, "to contain elements", presentable(matcher.Elements))
|
||||||
|
return appendMissingElements(message, matcher.missingElements)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *ContainElementsMatcher) NegatedFailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, "not to contain elements", presentable(matcher.Elements))
|
||||||
|
}
|
||||||
40
vendor/github.com/onsi/gomega/matchers/contain_substring_matcher.go
generated
vendored
Normal file
40
vendor/github.com/onsi/gomega/matchers/contain_substring_matcher.go
generated
vendored
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
// untested sections: 2
|
||||||
|
|
||||||
|
package matchers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/onsi/gomega/format"
|
||||||
|
)
|
||||||
|
|
||||||
|
type ContainSubstringMatcher struct {
|
||||||
|
Substr string
|
||||||
|
Args []interface{}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *ContainSubstringMatcher) Match(actual interface{}) (success bool, err error) {
|
||||||
|
actualString, ok := toString(actual)
|
||||||
|
if !ok {
|
||||||
|
return false, fmt.Errorf("ContainSubstring matcher requires a string or stringer. Got:\n%s", format.Object(actual, 1))
|
||||||
|
}
|
||||||
|
|
||||||
|
return strings.Contains(actualString, matcher.stringToMatch()), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *ContainSubstringMatcher) stringToMatch() string {
|
||||||
|
stringToMatch := matcher.Substr
|
||||||
|
if len(matcher.Args) > 0 {
|
||||||
|
stringToMatch = fmt.Sprintf(matcher.Substr, matcher.Args...)
|
||||||
|
}
|
||||||
|
return stringToMatch
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *ContainSubstringMatcher) FailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, "to contain substring", matcher.stringToMatch())
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *ContainSubstringMatcher) NegatedFailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, "not to contain substring", matcher.stringToMatch())
|
||||||
|
}
|
||||||
42
vendor/github.com/onsi/gomega/matchers/equal_matcher.go
generated
vendored
Normal file
42
vendor/github.com/onsi/gomega/matchers/equal_matcher.go
generated
vendored
Normal file
@@ -0,0 +1,42 @@
|
|||||||
|
package matchers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"fmt"
|
||||||
|
"reflect"
|
||||||
|
|
||||||
|
"github.com/onsi/gomega/format"
|
||||||
|
)
|
||||||
|
|
||||||
|
type EqualMatcher struct {
|
||||||
|
Expected interface{}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *EqualMatcher) Match(actual interface{}) (success bool, err error) {
|
||||||
|
if actual == nil && matcher.Expected == nil {
|
||||||
|
return false, fmt.Errorf("Refusing to compare <nil> to <nil>.\nBe explicit and use BeNil() instead. This is to avoid mistakes where both sides of an assertion are erroneously uninitialized.")
|
||||||
|
}
|
||||||
|
// Shortcut for byte slices.
|
||||||
|
// Comparing long byte slices with reflect.DeepEqual is very slow,
|
||||||
|
// so use bytes.Equal if actual and expected are both byte slices.
|
||||||
|
if actualByteSlice, ok := actual.([]byte); ok {
|
||||||
|
if expectedByteSlice, ok := matcher.Expected.([]byte); ok {
|
||||||
|
return bytes.Equal(actualByteSlice, expectedByteSlice), nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return reflect.DeepEqual(actual, matcher.Expected), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *EqualMatcher) FailureMessage(actual interface{}) (message string) {
|
||||||
|
actualString, actualOK := actual.(string)
|
||||||
|
expectedString, expectedOK := matcher.Expected.(string)
|
||||||
|
if actualOK && expectedOK {
|
||||||
|
return format.MessageWithDiff(actualString, "to equal", expectedString)
|
||||||
|
}
|
||||||
|
|
||||||
|
return format.Message(actual, "to equal", matcher.Expected)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *EqualMatcher) NegatedFailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, "not to equal", matcher.Expected)
|
||||||
|
}
|
||||||
30
vendor/github.com/onsi/gomega/matchers/have_cap_matcher.go
generated
vendored
Normal file
30
vendor/github.com/onsi/gomega/matchers/have_cap_matcher.go
generated
vendored
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
// untested sections: 2
|
||||||
|
|
||||||
|
package matchers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
"github.com/onsi/gomega/format"
|
||||||
|
)
|
||||||
|
|
||||||
|
type HaveCapMatcher struct {
|
||||||
|
Count int
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *HaveCapMatcher) Match(actual interface{}) (success bool, err error) {
|
||||||
|
length, ok := capOf(actual)
|
||||||
|
if !ok {
|
||||||
|
return false, fmt.Errorf("HaveCap matcher expects a array/channel/slice. Got:\n%s", format.Object(actual, 1))
|
||||||
|
}
|
||||||
|
|
||||||
|
return length == matcher.Count, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *HaveCapMatcher) FailureMessage(actual interface{}) (message string) {
|
||||||
|
return fmt.Sprintf("Expected\n%s\nto have capacity %d", format.Object(actual, 1), matcher.Count)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *HaveCapMatcher) NegatedFailureMessage(actual interface{}) (message string) {
|
||||||
|
return fmt.Sprintf("Expected\n%s\nnot to have capacity %d", format.Object(actual, 1), matcher.Count)
|
||||||
|
}
|
||||||
65
vendor/github.com/onsi/gomega/matchers/have_each_matcher.go
generated
vendored
Normal file
65
vendor/github.com/onsi/gomega/matchers/have_each_matcher.go
generated
vendored
Normal file
@@ -0,0 +1,65 @@
|
|||||||
|
package matchers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"reflect"
|
||||||
|
|
||||||
|
"github.com/onsi/gomega/format"
|
||||||
|
)
|
||||||
|
|
||||||
|
type HaveEachMatcher struct {
|
||||||
|
Element interface{}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *HaveEachMatcher) Match(actual interface{}) (success bool, err error) {
|
||||||
|
if !isArrayOrSlice(actual) && !isMap(actual) {
|
||||||
|
return false, fmt.Errorf("HaveEach matcher expects an array/slice/map. Got:\n%s",
|
||||||
|
format.Object(actual, 1))
|
||||||
|
}
|
||||||
|
|
||||||
|
elemMatcher, elementIsMatcher := matcher.Element.(omegaMatcher)
|
||||||
|
if !elementIsMatcher {
|
||||||
|
elemMatcher = &EqualMatcher{Expected: matcher.Element}
|
||||||
|
}
|
||||||
|
|
||||||
|
value := reflect.ValueOf(actual)
|
||||||
|
if value.Len() == 0 {
|
||||||
|
return false, fmt.Errorf("HaveEach matcher expects a non-empty array/slice/map. Got:\n%s",
|
||||||
|
format.Object(actual, 1))
|
||||||
|
}
|
||||||
|
|
||||||
|
var valueAt func(int) interface{}
|
||||||
|
if isMap(actual) {
|
||||||
|
keys := value.MapKeys()
|
||||||
|
valueAt = func(i int) interface{} {
|
||||||
|
return value.MapIndex(keys[i]).Interface()
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
valueAt = func(i int) interface{} {
|
||||||
|
return value.Index(i).Interface()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// if there are no elements, then HaveEach will match.
|
||||||
|
for i := 0; i < value.Len(); i++ {
|
||||||
|
success, err := elemMatcher.Match(valueAt(i))
|
||||||
|
if err != nil {
|
||||||
|
return false, err
|
||||||
|
}
|
||||||
|
if !success {
|
||||||
|
return false, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// FailureMessage returns a suitable failure message.
|
||||||
|
func (matcher *HaveEachMatcher) FailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, "to contain element matching", matcher.Element)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NegatedFailureMessage returns a suitable negated failure message.
|
||||||
|
func (matcher *HaveEachMatcher) NegatedFailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, "not to contain element matching", matcher.Element)
|
||||||
|
}
|
||||||
83
vendor/github.com/onsi/gomega/matchers/have_exact_elements.go
generated
vendored
Normal file
83
vendor/github.com/onsi/gomega/matchers/have_exact_elements.go
generated
vendored
Normal file
@@ -0,0 +1,83 @@
|
|||||||
|
package matchers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
"github.com/onsi/gomega/format"
|
||||||
|
)
|
||||||
|
|
||||||
|
type mismatchFailure struct {
|
||||||
|
failure string
|
||||||
|
index int
|
||||||
|
}
|
||||||
|
|
||||||
|
type HaveExactElementsMatcher struct {
|
||||||
|
Elements []interface{}
|
||||||
|
mismatchFailures []mismatchFailure
|
||||||
|
missingIndex int
|
||||||
|
extraIndex int
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *HaveExactElementsMatcher) Match(actual interface{}) (success bool, err error) {
|
||||||
|
matcher.resetState()
|
||||||
|
|
||||||
|
if isMap(actual) {
|
||||||
|
return false, fmt.Errorf("error")
|
||||||
|
}
|
||||||
|
|
||||||
|
matchers := matchers(matcher.Elements)
|
||||||
|
values := valuesOf(actual)
|
||||||
|
|
||||||
|
lenMatchers := len(matchers)
|
||||||
|
lenValues := len(values)
|
||||||
|
|
||||||
|
for i := 0; i < lenMatchers || i < lenValues; i++ {
|
||||||
|
if i >= lenMatchers {
|
||||||
|
matcher.extraIndex = i
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if i >= lenValues {
|
||||||
|
matcher.missingIndex = i
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
elemMatcher := matchers[i].(omegaMatcher)
|
||||||
|
match, err := elemMatcher.Match(values[i])
|
||||||
|
if err != nil || !match {
|
||||||
|
matcher.mismatchFailures = append(matcher.mismatchFailures, mismatchFailure{
|
||||||
|
index: i,
|
||||||
|
failure: elemMatcher.FailureMessage(values[i]),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return matcher.missingIndex+matcher.extraIndex+len(matcher.mismatchFailures) == 0, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *HaveExactElementsMatcher) FailureMessage(actual interface{}) (message string) {
|
||||||
|
message = format.Message(actual, "to have exact elements with", presentable(matcher.Elements))
|
||||||
|
if matcher.missingIndex > 0 {
|
||||||
|
message = fmt.Sprintf("%s\nthe missing elements start from index %d", message, matcher.missingIndex)
|
||||||
|
}
|
||||||
|
if matcher.extraIndex > 0 {
|
||||||
|
message = fmt.Sprintf("%s\nthe extra elements start from index %d", message, matcher.extraIndex)
|
||||||
|
}
|
||||||
|
if len(matcher.mismatchFailures) != 0 {
|
||||||
|
message = fmt.Sprintf("%s\nthe mismatch indexes were:", message)
|
||||||
|
}
|
||||||
|
for _, mismatch := range matcher.mismatchFailures {
|
||||||
|
message = fmt.Sprintf("%s\n%d: %s", message, mismatch.index, mismatch.failure)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *HaveExactElementsMatcher) NegatedFailureMessage(actual interface{}) (message string) {
|
||||||
|
return format.Message(actual, "not to contain elements", presentable(matcher.Elements))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *HaveExactElementsMatcher) resetState() {
|
||||||
|
matcher.mismatchFailures = nil
|
||||||
|
matcher.missingIndex = 0
|
||||||
|
matcher.extraIndex = 0
|
||||||
|
}
|
||||||
36
vendor/github.com/onsi/gomega/matchers/have_existing_field_matcher.go
generated
vendored
Normal file
36
vendor/github.com/onsi/gomega/matchers/have_existing_field_matcher.go
generated
vendored
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
package matchers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
"github.com/onsi/gomega/format"
|
||||||
|
)
|
||||||
|
|
||||||
|
type HaveExistingFieldMatcher struct {
|
||||||
|
Field string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *HaveExistingFieldMatcher) Match(actual interface{}) (success bool, err error) {
|
||||||
|
// we don't care about the field's actual value, just about any error in
|
||||||
|
// trying to find the field (or method).
|
||||||
|
_, err = extractField(actual, matcher.Field, "HaveExistingField")
|
||||||
|
if err == nil {
|
||||||
|
return true, nil
|
||||||
|
}
|
||||||
|
var mferr missingFieldError
|
||||||
|
if errors.As(err, &mferr) {
|
||||||
|
// missing field errors aren't errors in this context, but instead
|
||||||
|
// unsuccessful matches.
|
||||||
|
return false, nil
|
||||||
|
}
|
||||||
|
return false, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *HaveExistingFieldMatcher) FailureMessage(actual interface{}) (message string) {
|
||||||
|
return fmt.Sprintf("Expected\n%s\nto have field '%s'", format.Object(actual, 1), matcher.Field)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *HaveExistingFieldMatcher) NegatedFailureMessage(actual interface{}) (message string) {
|
||||||
|
return fmt.Sprintf("Expected\n%s\nnot to have field '%s'", format.Object(actual, 1), matcher.Field)
|
||||||
|
}
|
||||||
99
vendor/github.com/onsi/gomega/matchers/have_field.go
generated
vendored
Normal file
99
vendor/github.com/onsi/gomega/matchers/have_field.go
generated
vendored
Normal file
@@ -0,0 +1,99 @@
|
|||||||
|
package matchers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"reflect"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/onsi/gomega/format"
|
||||||
|
)
|
||||||
|
|
||||||
|
// missingFieldError represents a missing field extraction error that
|
||||||
|
// HaveExistingFieldMatcher can ignore, as opposed to other, sever field
|
||||||
|
// extraction errors, such as nil pointers, et cetera.
|
||||||
|
type missingFieldError string
|
||||||
|
|
||||||
|
func (e missingFieldError) Error() string {
|
||||||
|
return string(e)
|
||||||
|
}
|
||||||
|
|
||||||
|
func extractField(actual interface{}, field string, matchername string) (interface{}, error) {
|
||||||
|
fields := strings.SplitN(field, ".", 2)
|
||||||
|
actualValue := reflect.ValueOf(actual)
|
||||||
|
|
||||||
|
if actualValue.Kind() == reflect.Ptr {
|
||||||
|
actualValue = actualValue.Elem()
|
||||||
|
}
|
||||||
|
if actualValue == (reflect.Value{}) {
|
||||||
|
return nil, fmt.Errorf("%s encountered nil while dereferencing a pointer of type %T.", matchername, actual)
|
||||||
|
}
|
||||||
|
|
||||||
|
if actualValue.Kind() != reflect.Struct {
|
||||||
|
return nil, fmt.Errorf("%s encountered:\n%s\nWhich is not a struct.", matchername, format.Object(actual, 1))
|
||||||
|
}
|
||||||
|
|
||||||
|
var extractedValue reflect.Value
|
||||||
|
|
||||||
|
if strings.HasSuffix(fields[0], "()") {
|
||||||
|
extractedValue = actualValue.MethodByName(strings.TrimSuffix(fields[0], "()"))
|
||||||
|
if extractedValue == (reflect.Value{}) && actualValue.CanAddr() {
|
||||||
|
extractedValue = actualValue.Addr().MethodByName(strings.TrimSuffix(fields[0], "()"))
|
||||||
|
}
|
||||||
|
if extractedValue == (reflect.Value{}) {
|
||||||
|
return nil, missingFieldError(fmt.Sprintf("%s could not find method named '%s' in struct of type %T.", matchername, fields[0], actual))
|
||||||
|
}
|
||||||
|
t := extractedValue.Type()
|
||||||
|
if t.NumIn() != 0 || t.NumOut() != 1 {
|
||||||
|
return nil, fmt.Errorf("%s found an invalid method named '%s' in struct of type %T.\nMethods must take no arguments and return exactly one value.", matchername, fields[0], actual)
|
||||||
|
}
|
||||||
|
extractedValue = extractedValue.Call([]reflect.Value{})[0]
|
||||||
|
} else {
|
||||||
|
extractedValue = actualValue.FieldByName(fields[0])
|
||||||
|
if extractedValue == (reflect.Value{}) {
|
||||||
|
return nil, missingFieldError(fmt.Sprintf("%s could not find field named '%s' in struct:\n%s", matchername, fields[0], format.Object(actual, 1)))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(fields) == 1 {
|
||||||
|
return extractedValue.Interface(), nil
|
||||||
|
} else {
|
||||||
|
return extractField(extractedValue.Interface(), fields[1], matchername)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
type HaveFieldMatcher struct {
|
||||||
|
Field string
|
||||||
|
Expected interface{}
|
||||||
|
|
||||||
|
extractedField interface{}
|
||||||
|
expectedMatcher omegaMatcher
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *HaveFieldMatcher) Match(actual interface{}) (success bool, err error) {
|
||||||
|
matcher.extractedField, err = extractField(actual, matcher.Field, "HaveField")
|
||||||
|
if err != nil {
|
||||||
|
return false, err
|
||||||
|
}
|
||||||
|
|
||||||
|
var isMatcher bool
|
||||||
|
matcher.expectedMatcher, isMatcher = matcher.Expected.(omegaMatcher)
|
||||||
|
if !isMatcher {
|
||||||
|
matcher.expectedMatcher = &EqualMatcher{Expected: matcher.Expected}
|
||||||
|
}
|
||||||
|
|
||||||
|
return matcher.expectedMatcher.Match(matcher.extractedField)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *HaveFieldMatcher) FailureMessage(actual interface{}) (message string) {
|
||||||
|
message = fmt.Sprintf("Value for field '%s' failed to satisfy matcher.\n", matcher.Field)
|
||||||
|
message += matcher.expectedMatcher.FailureMessage(matcher.extractedField)
|
||||||
|
|
||||||
|
return message
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *HaveFieldMatcher) NegatedFailureMessage(actual interface{}) (message string) {
|
||||||
|
message = fmt.Sprintf("Value for field '%s' satisfied matcher, but should not have.\n", matcher.Field)
|
||||||
|
message += matcher.expectedMatcher.NegatedFailureMessage(matcher.extractedField)
|
||||||
|
|
||||||
|
return message
|
||||||
|
}
|
||||||
101
vendor/github.com/onsi/gomega/matchers/have_http_body_matcher.go
generated
vendored
Normal file
101
vendor/github.com/onsi/gomega/matchers/have_http_body_matcher.go
generated
vendored
Normal file
@@ -0,0 +1,101 @@
|
|||||||
|
package matchers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"net/http"
|
||||||
|
"net/http/httptest"
|
||||||
|
|
||||||
|
"github.com/onsi/gomega/format"
|
||||||
|
"github.com/onsi/gomega/internal/gutil"
|
||||||
|
"github.com/onsi/gomega/types"
|
||||||
|
)
|
||||||
|
|
||||||
|
type HaveHTTPBodyMatcher struct {
|
||||||
|
Expected interface{}
|
||||||
|
cachedBody []byte
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *HaveHTTPBodyMatcher) Match(actual interface{}) (bool, error) {
|
||||||
|
body, err := matcher.body(actual)
|
||||||
|
if err != nil {
|
||||||
|
return false, err
|
||||||
|
}
|
||||||
|
|
||||||
|
switch e := matcher.Expected.(type) {
|
||||||
|
case string:
|
||||||
|
return (&EqualMatcher{Expected: e}).Match(string(body))
|
||||||
|
case []byte:
|
||||||
|
return (&EqualMatcher{Expected: e}).Match(body)
|
||||||
|
case types.GomegaMatcher:
|
||||||
|
return e.Match(body)
|
||||||
|
default:
|
||||||
|
return false, fmt.Errorf("HaveHTTPBody matcher expects string, []byte, or GomegaMatcher. Got:\n%s", format.Object(matcher.Expected, 1))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *HaveHTTPBodyMatcher) FailureMessage(actual interface{}) (message string) {
|
||||||
|
body, err := matcher.body(actual)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Sprintf("failed to read body: %s", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
switch e := matcher.Expected.(type) {
|
||||||
|
case string:
|
||||||
|
return (&EqualMatcher{Expected: e}).FailureMessage(string(body))
|
||||||
|
case []byte:
|
||||||
|
return (&EqualMatcher{Expected: e}).FailureMessage(body)
|
||||||
|
case types.GomegaMatcher:
|
||||||
|
return e.FailureMessage(body)
|
||||||
|
default:
|
||||||
|
return fmt.Sprintf("HaveHTTPBody matcher expects string, []byte, or GomegaMatcher. Got:\n%s", format.Object(matcher.Expected, 1))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *HaveHTTPBodyMatcher) NegatedFailureMessage(actual interface{}) (message string) {
|
||||||
|
body, err := matcher.body(actual)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Sprintf("failed to read body: %s", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
switch e := matcher.Expected.(type) {
|
||||||
|
case string:
|
||||||
|
return (&EqualMatcher{Expected: e}).NegatedFailureMessage(string(body))
|
||||||
|
case []byte:
|
||||||
|
return (&EqualMatcher{Expected: e}).NegatedFailureMessage(body)
|
||||||
|
case types.GomegaMatcher:
|
||||||
|
return e.NegatedFailureMessage(body)
|
||||||
|
default:
|
||||||
|
return fmt.Sprintf("HaveHTTPBody matcher expects string, []byte, or GomegaMatcher. Got:\n%s", format.Object(matcher.Expected, 1))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// body returns the body. It is cached because once we read it in Match()
|
||||||
|
// the Reader is closed and it is not readable again in FailureMessage()
|
||||||
|
// or NegatedFailureMessage()
|
||||||
|
func (matcher *HaveHTTPBodyMatcher) body(actual interface{}) ([]byte, error) {
|
||||||
|
if matcher.cachedBody != nil {
|
||||||
|
return matcher.cachedBody, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
body := func(a *http.Response) ([]byte, error) {
|
||||||
|
if a.Body != nil {
|
||||||
|
defer a.Body.Close()
|
||||||
|
var err error
|
||||||
|
matcher.cachedBody, err = gutil.ReadAll(a.Body)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("error reading response body: %w", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return matcher.cachedBody, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
switch a := actual.(type) {
|
||||||
|
case *http.Response:
|
||||||
|
return body(a)
|
||||||
|
case *httptest.ResponseRecorder:
|
||||||
|
return body(a.Result())
|
||||||
|
default:
|
||||||
|
return nil, fmt.Errorf("HaveHTTPBody matcher expects *http.Response or *httptest.ResponseRecorder. Got:\n%s", format.Object(actual, 1))
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
81
vendor/github.com/onsi/gomega/matchers/have_http_header_with_value_matcher.go
generated
vendored
Normal file
81
vendor/github.com/onsi/gomega/matchers/have_http_header_with_value_matcher.go
generated
vendored
Normal file
@@ -0,0 +1,81 @@
|
|||||||
|
package matchers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"net/http"
|
||||||
|
"net/http/httptest"
|
||||||
|
|
||||||
|
"github.com/onsi/gomega/format"
|
||||||
|
"github.com/onsi/gomega/types"
|
||||||
|
)
|
||||||
|
|
||||||
|
type HaveHTTPHeaderWithValueMatcher struct {
|
||||||
|
Header string
|
||||||
|
Value interface{}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *HaveHTTPHeaderWithValueMatcher) Match(actual interface{}) (success bool, err error) {
|
||||||
|
headerValue, err := matcher.extractHeader(actual)
|
||||||
|
if err != nil {
|
||||||
|
return false, err
|
||||||
|
}
|
||||||
|
|
||||||
|
headerMatcher, err := matcher.getSubMatcher()
|
||||||
|
if err != nil {
|
||||||
|
return false, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return headerMatcher.Match(headerValue)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *HaveHTTPHeaderWithValueMatcher) FailureMessage(actual interface{}) string {
|
||||||
|
headerValue, err := matcher.extractHeader(actual)
|
||||||
|
if err != nil {
|
||||||
|
panic(err) // protected by Match()
|
||||||
|
}
|
||||||
|
|
||||||
|
headerMatcher, err := matcher.getSubMatcher()
|
||||||
|
if err != nil {
|
||||||
|
panic(err) // protected by Match()
|
||||||
|
}
|
||||||
|
|
||||||
|
diff := format.IndentString(headerMatcher.FailureMessage(headerValue), 1)
|
||||||
|
return fmt.Sprintf("HTTP header %q:\n%s", matcher.Header, diff)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *HaveHTTPHeaderWithValueMatcher) NegatedFailureMessage(actual interface{}) (message string) {
|
||||||
|
headerValue, err := matcher.extractHeader(actual)
|
||||||
|
if err != nil {
|
||||||
|
panic(err) // protected by Match()
|
||||||
|
}
|
||||||
|
|
||||||
|
headerMatcher, err := matcher.getSubMatcher()
|
||||||
|
if err != nil {
|
||||||
|
panic(err) // protected by Match()
|
||||||
|
}
|
||||||
|
|
||||||
|
diff := format.IndentString(headerMatcher.NegatedFailureMessage(headerValue), 1)
|
||||||
|
return fmt.Sprintf("HTTP header %q:\n%s", matcher.Header, diff)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *HaveHTTPHeaderWithValueMatcher) getSubMatcher() (types.GomegaMatcher, error) {
|
||||||
|
switch m := matcher.Value.(type) {
|
||||||
|
case string:
|
||||||
|
return &EqualMatcher{Expected: matcher.Value}, nil
|
||||||
|
case types.GomegaMatcher:
|
||||||
|
return m, nil
|
||||||
|
default:
|
||||||
|
return nil, fmt.Errorf("HaveHTTPHeaderWithValue matcher must be passed a string or a GomegaMatcher. Got:\n%s", format.Object(matcher.Value, 1))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *HaveHTTPHeaderWithValueMatcher) extractHeader(actual interface{}) (string, error) {
|
||||||
|
switch r := actual.(type) {
|
||||||
|
case *http.Response:
|
||||||
|
return r.Header.Get(matcher.Header), nil
|
||||||
|
case *httptest.ResponseRecorder:
|
||||||
|
return r.Result().Header.Get(matcher.Header), nil
|
||||||
|
default:
|
||||||
|
return "", fmt.Errorf("HaveHTTPHeaderWithValue matcher expects *http.Response or *httptest.ResponseRecorder. Got:\n%s", format.Object(actual, 1))
|
||||||
|
}
|
||||||
|
}
|
||||||
96
vendor/github.com/onsi/gomega/matchers/have_http_status_matcher.go
generated
vendored
Normal file
96
vendor/github.com/onsi/gomega/matchers/have_http_status_matcher.go
generated
vendored
Normal file
@@ -0,0 +1,96 @@
|
|||||||
|
package matchers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"net/http"
|
||||||
|
"net/http/httptest"
|
||||||
|
"reflect"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/onsi/gomega/format"
|
||||||
|
"github.com/onsi/gomega/internal/gutil"
|
||||||
|
)
|
||||||
|
|
||||||
|
type HaveHTTPStatusMatcher struct {
|
||||||
|
Expected []interface{}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *HaveHTTPStatusMatcher) Match(actual interface{}) (success bool, err error) {
|
||||||
|
var resp *http.Response
|
||||||
|
switch a := actual.(type) {
|
||||||
|
case *http.Response:
|
||||||
|
resp = a
|
||||||
|
case *httptest.ResponseRecorder:
|
||||||
|
resp = a.Result()
|
||||||
|
default:
|
||||||
|
return false, fmt.Errorf("HaveHTTPStatus matcher expects *http.Response or *httptest.ResponseRecorder. Got:\n%s", format.Object(actual, 1))
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(matcher.Expected) == 0 {
|
||||||
|
return false, fmt.Errorf("HaveHTTPStatus matcher must be passed an int or a string. Got nothing")
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, expected := range matcher.Expected {
|
||||||
|
switch e := expected.(type) {
|
||||||
|
case int:
|
||||||
|
if resp.StatusCode == e {
|
||||||
|
return true, nil
|
||||||
|
}
|
||||||
|
case string:
|
||||||
|
if resp.Status == e {
|
||||||
|
return true, nil
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
return false, fmt.Errorf("HaveHTTPStatus matcher must be passed int or string types. Got:\n%s", format.Object(expected, 1))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *HaveHTTPStatusMatcher) FailureMessage(actual interface{}) (message string) {
|
||||||
|
return fmt.Sprintf("Expected\n%s\n%s\n%s", formatHttpResponse(actual), "to have HTTP status", matcher.expectedString())
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *HaveHTTPStatusMatcher) NegatedFailureMessage(actual interface{}) (message string) {
|
||||||
|
return fmt.Sprintf("Expected\n%s\n%s\n%s", formatHttpResponse(actual), "not to have HTTP status", matcher.expectedString())
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *HaveHTTPStatusMatcher) expectedString() string {
|
||||||
|
var lines []string
|
||||||
|
for _, expected := range matcher.Expected {
|
||||||
|
lines = append(lines, format.Object(expected, 1))
|
||||||
|
}
|
||||||
|
return strings.Join(lines, "\n")
|
||||||
|
}
|
||||||
|
|
||||||
|
func formatHttpResponse(input interface{}) string {
|
||||||
|
var resp *http.Response
|
||||||
|
switch r := input.(type) {
|
||||||
|
case *http.Response:
|
||||||
|
resp = r
|
||||||
|
case *httptest.ResponseRecorder:
|
||||||
|
resp = r.Result()
|
||||||
|
default:
|
||||||
|
return "cannot format invalid HTTP response"
|
||||||
|
}
|
||||||
|
|
||||||
|
body := "<nil>"
|
||||||
|
if resp.Body != nil {
|
||||||
|
defer resp.Body.Close()
|
||||||
|
data, err := gutil.ReadAll(resp.Body)
|
||||||
|
if err != nil {
|
||||||
|
data = []byte("<error reading body>")
|
||||||
|
}
|
||||||
|
body = format.Object(string(data), 0)
|
||||||
|
}
|
||||||
|
|
||||||
|
var s strings.Builder
|
||||||
|
s.WriteString(fmt.Sprintf("%s<%s>: {\n", format.Indent, reflect.TypeOf(input)))
|
||||||
|
s.WriteString(fmt.Sprintf("%s%sStatus: %s\n", format.Indent, format.Indent, format.Object(resp.Status, 0)))
|
||||||
|
s.WriteString(fmt.Sprintf("%s%sStatusCode: %s\n", format.Indent, format.Indent, format.Object(resp.StatusCode, 0)))
|
||||||
|
s.WriteString(fmt.Sprintf("%s%sBody: %s\n", format.Indent, format.Indent, body))
|
||||||
|
s.WriteString(fmt.Sprintf("%s}", format.Indent))
|
||||||
|
|
||||||
|
return s.String()
|
||||||
|
}
|
||||||
56
vendor/github.com/onsi/gomega/matchers/have_key_matcher.go
generated
vendored
Normal file
56
vendor/github.com/onsi/gomega/matchers/have_key_matcher.go
generated
vendored
Normal file
@@ -0,0 +1,56 @@
|
|||||||
|
// untested sections: 6
|
||||||
|
|
||||||
|
package matchers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"reflect"
|
||||||
|
|
||||||
|
"github.com/onsi/gomega/format"
|
||||||
|
)
|
||||||
|
|
||||||
|
type HaveKeyMatcher struct {
|
||||||
|
Key interface{}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *HaveKeyMatcher) Match(actual interface{}) (success bool, err error) {
|
||||||
|
if !isMap(actual) {
|
||||||
|
return false, fmt.Errorf("HaveKey matcher expects a map. Got:%s", format.Object(actual, 1))
|
||||||
|
}
|
||||||
|
|
||||||
|
keyMatcher, keyIsMatcher := matcher.Key.(omegaMatcher)
|
||||||
|
if !keyIsMatcher {
|
||||||
|
keyMatcher = &EqualMatcher{Expected: matcher.Key}
|
||||||
|
}
|
||||||
|
|
||||||
|
keys := reflect.ValueOf(actual).MapKeys()
|
||||||
|
for i := 0; i < len(keys); i++ {
|
||||||
|
success, err := keyMatcher.Match(keys[i].Interface())
|
||||||
|
if err != nil {
|
||||||
|
return false, fmt.Errorf("HaveKey's key matcher failed with:\n%s%s", format.Indent, err.Error())
|
||||||
|
}
|
||||||
|
if success {
|
||||||
|
return true, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *HaveKeyMatcher) FailureMessage(actual interface{}) (message string) {
|
||||||
|
switch matcher.Key.(type) {
|
||||||
|
case omegaMatcher:
|
||||||
|
return format.Message(actual, "to have key matching", matcher.Key)
|
||||||
|
default:
|
||||||
|
return format.Message(actual, "to have key", matcher.Key)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *HaveKeyMatcher) NegatedFailureMessage(actual interface{}) (message string) {
|
||||||
|
switch matcher.Key.(type) {
|
||||||
|
case omegaMatcher:
|
||||||
|
return format.Message(actual, "not to have key matching", matcher.Key)
|
||||||
|
default:
|
||||||
|
return format.Message(actual, "not to have key", matcher.Key)
|
||||||
|
}
|
||||||
|
}
|
||||||
76
vendor/github.com/onsi/gomega/matchers/have_key_with_value_matcher.go
generated
vendored
Normal file
76
vendor/github.com/onsi/gomega/matchers/have_key_with_value_matcher.go
generated
vendored
Normal file
@@ -0,0 +1,76 @@
|
|||||||
|
// untested sections:10
|
||||||
|
|
||||||
|
package matchers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"reflect"
|
||||||
|
|
||||||
|
"github.com/onsi/gomega/format"
|
||||||
|
)
|
||||||
|
|
||||||
|
type HaveKeyWithValueMatcher struct {
|
||||||
|
Key interface{}
|
||||||
|
Value interface{}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *HaveKeyWithValueMatcher) Match(actual interface{}) (success bool, err error) {
|
||||||
|
if !isMap(actual) {
|
||||||
|
return false, fmt.Errorf("HaveKeyWithValue matcher expects a map. Got:%s", format.Object(actual, 1))
|
||||||
|
}
|
||||||
|
|
||||||
|
keyMatcher, keyIsMatcher := matcher.Key.(omegaMatcher)
|
||||||
|
if !keyIsMatcher {
|
||||||
|
keyMatcher = &EqualMatcher{Expected: matcher.Key}
|
||||||
|
}
|
||||||
|
|
||||||
|
valueMatcher, valueIsMatcher := matcher.Value.(omegaMatcher)
|
||||||
|
if !valueIsMatcher {
|
||||||
|
valueMatcher = &EqualMatcher{Expected: matcher.Value}
|
||||||
|
}
|
||||||
|
|
||||||
|
keys := reflect.ValueOf(actual).MapKeys()
|
||||||
|
for i := 0; i < len(keys); i++ {
|
||||||
|
success, err := keyMatcher.Match(keys[i].Interface())
|
||||||
|
if err != nil {
|
||||||
|
return false, fmt.Errorf("HaveKeyWithValue's key matcher failed with:\n%s%s", format.Indent, err.Error())
|
||||||
|
}
|
||||||
|
if success {
|
||||||
|
actualValue := reflect.ValueOf(actual).MapIndex(keys[i])
|
||||||
|
success, err := valueMatcher.Match(actualValue.Interface())
|
||||||
|
if err != nil {
|
||||||
|
return false, fmt.Errorf("HaveKeyWithValue's value matcher failed with:\n%s%s", format.Indent, err.Error())
|
||||||
|
}
|
||||||
|
return success, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *HaveKeyWithValueMatcher) FailureMessage(actual interface{}) (message string) {
|
||||||
|
str := "to have {key: value}"
|
||||||
|
if _, ok := matcher.Key.(omegaMatcher); ok {
|
||||||
|
str += " matching"
|
||||||
|
} else if _, ok := matcher.Value.(omegaMatcher); ok {
|
||||||
|
str += " matching"
|
||||||
|
}
|
||||||
|
|
||||||
|
expect := make(map[interface{}]interface{}, 1)
|
||||||
|
expect[matcher.Key] = matcher.Value
|
||||||
|
return format.Message(actual, str, expect)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (matcher *HaveKeyWithValueMatcher) NegatedFailureMessage(actual interface{}) (message string) {
|
||||||
|
kStr := "not to have key"
|
||||||
|
if _, ok := matcher.Key.(omegaMatcher); ok {
|
||||||
|
kStr = "not to have key matching"
|
||||||
|
}
|
||||||
|
|
||||||
|
vStr := "or that key's value not be"
|
||||||
|
if _, ok := matcher.Value.(omegaMatcher); ok {
|
||||||
|
vStr = "or to have that key's value not matching"
|
||||||
|
}
|
||||||
|
|
||||||
|
return format.Message(actual, kStr, matcher.Key, vStr, matcher.Value)
|
||||||
|
}
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user