如何在jsonpath中过滤时获取第一个元素?

2022-09-03 08:04:07

所以我正在研究下面的json:

{
   "id": "",
   "owner": "some dude",
   "metaData": {
      "request": {
         "ref": null,
         "contacts":[
            {
               "email": null,
               "name": null,
               "contactType": "R"
            },
            {
               "email": null,
               "name": "Dante",
               "contactType": "S"
            }
         ]
      }
   }
}

我想检索的联系人具有类型,并且只有返回的第一个。nameS

将 jsonpath 与此路径一起使用时,始终返回字符串数组,因为筛选操作始终将结果作为数组返回。"$..contacts[?(@.contactType == 'S')].name"

所以我试过了,但没有运气。这些路径返回空结果。"$..contacts[?(@.contactType == 'S')].name[0]""$..contacts[?(@.contactType == 'S')][0].name"

所以我的问题是,在jsonpath中使用过滤器时,有没有办法只获取第一个元素。我目前正在使用jayway jsonpath v2.2.0。


答案 1

我发现的最快的解决方案只是将其与列表进行比较,因此在您的场景中如下所示:

.andExpect(jsonPath("$..contacts[?(@.contactType == 'S')].name", equalTo(Arrays.asList("Dante"))))

(当然,假设您正在尝试在测试中比较某些结果)


答案 2

如果你从 spring-test 中将 jsonpath 与 MockMvc 类一起使用,那么你可以编写以下虚拟匹配器:

import org.hamcrest.BaseMatcher;
import org.hamcrest.Description;
import org.hamcrest.Matcher;
import org.hamcrest.core.IsEqual;

import net.minidev.json.JSONArray;

public class FirstMatcher<T> extends BaseMatcher<T> {

    private final Matcher<?> matcher;

    public static <T> FirstMatcher<T> matcher(Matcher<T> matcher) {
        return new FirstMatcher<T>(matcher);
    }

    public static FirstMatcher<Object> value(Object value) {
        return new FirstMatcher<Object>(value);
    }

    public FirstMatcher(Matcher<T> matcher) {
        this.matcher = matcher;
    }

    public FirstMatcher(Object value) {
        this.matcher = new IsEqual<Object>(value); 
    }

    @Override
    public void describeTo(Description description) {
        description.appendText("first matcher");
    }

    @Override
    public boolean matches(Object item) {
        if (!(item instanceof JSONArray)) {
            return false;
        }

        JSONArray array = (JSONArray)item;
        if (array.isEmpty()) {
            return false;
        }

        Object obj = array.get(0);
        return matcher.matches(obj);
    }

}

并使用以下方式:

mockMvc.
    perform(get(url).
            accept(MediaType.APPLICATION_JSON).accept(MediaType.TEXT_PLAIN).accept(MediaType.ALL)).
    andExpect(status().isOk()).
    andExpect(content().contentType(MediaType.APPLICATION_JSON_UTF8)).
    andExpect(jsonPath("$.properties[?(@.propertyName == 'name1')].description").value(FirstMatcher.matcher(IsNull.nullValue()))).
    andExpect(jsonPath("$.properties[?(@.propertyName == 'name2')].required").value(FirstMatcher.value(true)));

P.S. 由于 net.minidev.json.JSONArray 子类 java.util.List,因此可以转换为 List 甚至 Iterable,而不是 net.minidev.json.JSONArray。:)


推荐