如何比较两个 JSON 请求?

Ant*_*unt 14 json go

短篇小说:如何比较两个 JSON 块?下面的代码出错了。

var j, j2 interface{}
b := []byte(srv.req)
if err := json.Unmarshal(b, j); err !=nil{
    t.Errorf("err %v, req %s", err, b)
    return
}
d := json.NewDecoder(r.Body)
if err := d.Decode(j2); err !=nil{
    t.Error(err)
    return
}
if !reflect.DeepEqual(j2, j){
    t.Errorf("j %v, j2 %v", j, j2)
    return
}
Run Code Online (Sandbox Code Playgroud)

长话短说:我正在做一些 E2E 测试,其中一部分我需要将请求的 JSON 正文与收到的 JSON 进行比较。为此,我尝试将预期的和收到的 json 解组到一个空接口(以避免任何类型错误),但我收到一个错误: json: Unmarshal(nil). 我猜 encoding/json 不喜欢空接口,所以问题是如何比较两个 JSON 块?字符串比较容易出错,所以我试图避免这种情况。

Hig*_*ead 15

这里的派对超级迟到。

golang 中有一个流行的测试包叫做 require github.com/stretchr/testify/require,它会为你做这件事。

func TestJsonEquality(t *testing.t) { 
  expected := `{"a": 1, "b": 2} `
  actual := ` {"b":   2, "a":   1}`
  require.JSONEq(t, expected, actual)
}
Run Code Online (Sandbox Code Playgroud)

GoDocs:https ://godoc.org/github.com/stretchr/testify/require#JSONEqf

  • 当您有 JSON 对象列表时,这不起作用。如果列表中的项目无序,则相等性为 false。 (2认同)

two*_*two 12

您需要将指针传递给DecodeUnmarshal。我用和建立了一个可运行的样本,两者都返回。您可以通过使用或包装您的预期内容来将请求正文与您的静态预期内容进行比较(就像您在问题中尝试做的那样)。这是代码:func JSONEqual(a, b io.Reader)JSONBytesEqual(a, b []byte)(bool, error)bytes.NewBufferstrings.NewReader

package main

import (
    "encoding/json"
    "fmt"
    "io"
    "reflect"
)

// JSONEqual compares the JSON from two Readers.
func JSONEqual(a, b io.Reader) (bool, error) {
    var j, j2 interface{}
    d := json.NewDecoder(a)
    if err := d.Decode(&j); err != nil {
        return false, err
    }
    d = json.NewDecoder(b)
    if err := d.Decode(&j2); err != nil {
        return false, err
    }
    return reflect.DeepEqual(j2, j), nil
}

// JSONBytesEqual compares the JSON in two byte slices.
func JSONBytesEqual(a, b []byte) (bool, error) {
    var j, j2 interface{}
    if err := json.Unmarshal(a, &j); err != nil {
        return false, err
    }
    if err := json.Unmarshal(b, &j2); err != nil {
        return false, err
    }
    return reflect.DeepEqual(j2, j), nil
}

func main() {
    a := []byte(`{"x": ["y",42]}`)
    b := []byte(`{"x":                  ["y",  42]}`)
    c := []byte(`{"z": ["y", "42"]}`)
    empty := []byte{}
    bad := []byte(`{this? this is a test.}`)

    eq, err := JSONBytesEqual(a, b)
    fmt.Println("a=b\t", eq, "with error", err)
    eq, err = JSONBytesEqual(a, c)
    fmt.Println("a=c\t", eq, "with error", err)
    eq, err = JSONBytesEqual(a, empty)
    fmt.Println("a=empty\t", eq, "with error", err)
    eq, err = JSONBytesEqual(a, bad)
    fmt.Println("a=bad\t", eq, "with error", err)
}
Run Code Online (Sandbox Code Playgroud)

它输出:

a=b  true with error <nil>
a=c  false with error <nil>
a=empty  false with error EOF
a=bad    false with error invalid character 't' looking for beginning of object key string
Run Code Online (Sandbox Code Playgroud)