Vue 组件测试与 Jest 错误,类型错误:无法读取未定义的属性'name'



我有一个Vue组件,模板如下:它接收一个对象作为道具。首先,它从Admin.vue中的created()的后端拉出,作为数组传递给OrderList,循环,然后作为单个命令传递给Order<router-link>中的ShowOrder组件在自己的页面上显示单个订单。

<template v-if="order ? order : error">
<div class="order-container -shadow">
<div class="order-number">
<p>{{ order.orderId }}</p>
<p>{{ order.daySelected }}</p>
<p>{{ order.timeSelected }}</p>
</div>
<div class="customer-info">
<h2>{{ order.userInfo.name }}</h2>
<p>
{{ order.userInfo.street }},
{{ order.userInfo.city }}
</p>
<p v-if="order.userInfo.apt">
{{ order.userInfo.apt }}
</p>
<p>{{ order.userInfo.phone }}</p>
...
...
<router-link :to="{ name: 'ShowOrder', params: { id: order.orderId, order: order }}"
>Show Details</router-link
>
</div> 
</template>
<script>
export default {
name: "Order",
props: {
order: {
type: Object,
default: function() {
return {};
}
}
},
data() {
return {
error: "Error"
};
}
};
</script>

我的Order.spec.js如下:

const localVue = createLocalVue();
localVue.use(VueRouter);
const router = new VueRouter();
const $route = {
path: '"/show-order/:id"'
};
shallowMount(Order, {
localVue,
router,
stubs: ["router-link", "router-view"]
});
const mockOrder = {
orderId: 123,
chocolateChip: 24,
oatmeal: 0,
campfire: 12,
daySelected: "Wed Jan 27",
timeSelected: "2:30 - 3:00",
userInfo: {
apt: "",
city: "Port ",
email: "k@gmail.com",
street: " Point Road",
phone: "(123)446-1980",
name: "Mr.Smith"
}
};
describe("Order.vue", () => {
it("renders correctly", () => {
const wrapper = shallowMount(Order, {
localVue,
propsData: {
order: mockOrder,
$route
}
});
console.log(wrapper.html());
expect(wrapper.element).toMatchSnapshot();
});
});

我收到的错误看起来像这样:

FAIL  tests/unit/Order.spec.js
● Test suite failed to run
TypeError: Cannot read property 'name' of undefined
72 |   flex-flow: column wrap;
73 |   margin-top: 1em;
> 74 |   transition: all 0.2s linear;
|                                                             ^
75 | }
76 | 
77 | .order-container:hover {

我甚至不能让快照测试工作。我在此错误中阅读了许多其他类似的问题,并尝试在模板中的v-if中等待组件加载之前的数据。这是怎么回事?如果这是一个对象错误,为什么在控制台有一个指针指向我的作用域css,而不是order.userInfo.name?

我也试过从Admin.vuecreated()更改为mounted()

Order.vue中,我尝试添加对象以使其基于这些Vue文档响应。

data() {
return {
error: "Error",
orderItem: {}
};
},
mounted() {
this.orderItem = Object.assign({}, this.order);
console.log(this.orderItem, "line 65");
}

您调用了两次shallowMount():一次是在没有mockOrder的测试之外(这个调用是不必要的),另一次是在使用mockOrder的测试内部。

// Order.spec.js
shallowMount(Order, { localVue }) // ❌ no props, but no need for this call
describe("Order.vue", () => {
shallowMount(Order, {
localVue,
propsData: {
order: mockOrder
}
})
})

第一个调用导致错误,因为它的orderprop默认为空对象,所以order.userInfo将是undefined,导致order.userInfo.name的错误。

解决方案是删除对shallowMount()的不必要的第一次调用。

最新更新