首页 > 解决方案 > 集合时如何测试json api

问题描述

我读到最好在 Hibernate 关系中使用 Set 而不是 List 。

我创建了两个与一对多相关的实体:

@Entity
public class Product {

    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private Long id;

    @Column(nullable = false, unique = true)
    private String name;

    @ManyToOne
    @JoinColumn(name = "company_id", nullable = false)
    private Company company;
}

@Entity
public class Company {

    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private Long id;

    @Column(nullable = false, unique = true)
    private String name;

    @LazyCollection(LazyCollectionOption.TRUE)
    @OneToMany(mappedBy = "company", cascade = CascadeType.ALL)
    private Set<Product> products;
}

关于@OneToMany 集合集合私有集合产品;

然后我尝试测试返回结果:

@RunWith(SpringRunner.class)
@SpringBootTest
@Transactional
public class CompanyControllerTest {

    private static final String API_COMPANY = "/api/company/";

    @Autowired
    private WebApplicationContext context;

    private MockMvc mockMvc;

    @Before
    public void setup() {
        mockMvc = MockMvcBuilders
                .webAppContextSetup(context)
                .build();
    }

    @Test
    public void getById() throws Exception {
        int id = 1;

        this.mockMvc.perform(get(API_COMPANY + id))
                .andExpect(status().isOk())
                .andExpect(content().contentType(APPLICATION_JSON_UTF8))
                .andExpect(jsonPath("id", is(1)))
                .andExpect(jsonPath("$.name", is("Google")))
                .andExpect(jsonPath("$.products", hasSize(2)))
                .andExpect(jsonPath("$.products[0].id", is(1)))
                .andExpect(jsonPath("$.products[0].name", is("search engine")))
                .andExpect(jsonPath("$.products[0].company").doesNotExist())
                .andExpect(jsonPath("$.products[1].id", is(2)))
                .andExpect(jsonPath("$.products[1].name", is("adv.")))
                .andExpect(jsonPath("$.products[1].company").doesNotExist());
    }
}

但问题是产品列表是不断变化的,因为我使用的是 Set。事实证明,测试要么通过,要么失败,因为产品的顺序发生了变化。

我的问题是如何在使用 Set 时测试结果。

标签: javaspringspring-test-mvc

解决方案


您可以使用[*]获取所有元素,并为Matchers.containsInAnyOrder(T...)提供您要检查的所有元素。

像这样的东西:

 this.mockMvc.perform(get(API_COMPANY + id))
            .andExpect(status().isOk())
            .andExpect(content().contentType(APPLICATION_JSON_UTF8))
            .andExpect(jsonPath("id", is(1)))
            .andExpect(jsonPath("$.name", is("Google")))
            .andExpect(jsonPath("$.products", hasSize(2)))
            .andExpect(jsonPath("$.products[*].id", Matchers.containsInAnyOrder(1, 2)))
            .andExpect(jsonPath("$.products[*].name", Matchers.containsInAnyOrder("search engine", "adv.")))
            .andExpect(jsonPath("$.products[*].company").doesNotExist());

推荐阅读