# Instructions - Following Playwright test failed. - Explain why, be concise, respect Playwright best practices. - Provide a snippet of code with the fix, if possible. # Test info - Name: src/manager/tests/e2e/insurance/insurance-delete-booking-plans.spec.ts >> rejects deleting all booking plans via the row action @bug - Location: src/manager/tests/e2e/insurance/insurance-delete-booking-plans.spec.ts:39:6 # Error details ``` Error: expect(locator).toHaveCountGreaterThan(0) Locator: locator('.v-snackbar').filter({ hasText: 'At least one booking plan is required' }) Expected: > 0 Received: 0 ``` # Page snapshot ```yaml - generic [active] [ref=e1]: - generic [ref=e6]: - navigation [ref=e7]: - generic [ref=e9]: - link [ref=e10] [cursor=pointer]: - /url: / - generic [ref=e11]: - generic [ref=e12]: KINNOVIS - generic [ref=e13]: - generic [ref=e14]: Manager - button [ref=e15] [cursor=pointer]: - img [ref=e16] - list [ref=e19]: - generic [ref=e21]: Dashboard - link "Tasks" [ref=e22] [cursor=pointer]: - /url: /tasks - img [ref=e25] - generic [ref=e28]: Tasks - link "Facility maps" [ref=e29] [cursor=pointer]: - /url: /facility-map - generic [ref=e31]: 󰧾 - generic [ref=e33]: Facility maps - link "Analytics" [ref=e34] [cursor=pointer]: - /url: /dashboard - generic [ref=e36]: 󱖶 - generic [ref=e38]: Analytics - generic [ref=e40]: Sales - link "Bookings" [ref=e41] [cursor=pointer]: - /url: /bookings - generic [ref=e43]: 󰇡 - generic [ref=e45]: Bookings - link "Customers" [ref=e46] [cursor=pointer]: - /url: /customers - generic [ref=e48]: 󰀏 - generic [ref=e50]: Customers - link "Invoices" [ref=e51] [cursor=pointer]: - /url: /invoices - generic [ref=e53]: 󰷉 - generic [ref=e55]: Invoices - link "Credit notes" [ref=e56] [cursor=pointer]: - /url: /credit-notes - img [ref=e59] - generic [ref=e62]: Credit notes - link "Units" [ref=e63] [cursor=pointer]: - /url: /units - generic [ref=e65]: 󰍀 - generic [ref=e67]: Units - generic [ref=e69]: Site management - link "Locations" [ref=e70] [cursor=pointer]: - /url: /locations - generic [ref=e72]: 󰟙 - generic [ref=e74]: Locations - link "Unit types" [ref=e75] [cursor=pointer]: - /url: /unit-types - generic [ref=e77]: 󰆧 - generic [ref=e79]: Unit types - link "Protection Plans" [ref=e80] [cursor=pointer]: - /url: /insurances - generic [ref=e82]: 󰳌 - generic [ref=e84]: Protection Plans - link "Deposits" [ref=e85] [cursor=pointer]: - /url: /deposits - generic [ref=e87]: 󱙆 - generic [ref=e89]: Deposits - link "Products" [ref=e90] [cursor=pointer]: - /url: /products - generic [ref=e92]: 󰄑 - generic [ref=e94]: Products - link "Discounts" [ref=e95] [cursor=pointer]: - /url: /discounts - generic [ref=e97]: 󰓼 - generic [ref=e99]: Discounts - generic [ref=e100]: - option "Emails" [ref=e101] [cursor=pointer]: - generic [ref=e103]: 󰻨 - generic [ref=e105]: Emails - generic [ref=e109]: 󰅀 - text: 󱡰 󰁥 - generic [ref=e111]: Admin - link "Integrations" [ref=e112] [cursor=pointer]: - /url: /connected-apps - generic [ref=e114]: 󱘖 - generic [ref=e116]: Integrations - link "User & Roles" [ref=e117] [cursor=pointer]: - /url: /users - generic [ref=e119]: 󰭘 - generic [ref=e121]: User & Roles - generic [ref=e122]: - option "Booking Portal" [ref=e123] [cursor=pointer]: - generic [ref=e125]: 󱃁 - generic [ref=e127]: Booking Portal - generic [ref=e131]: 󰅀 - text: 󰖟 󰟙 - generic [ref=e132]: - option "JaneAI" [ref=e133] [cursor=pointer]: - generic [ref=e135]: 󱙺 - generic [ref=e137]: JaneAI - generic [ref=e141]: 󰅀 - text: 󱜹 - generic [ref=e143]: Feedback - link "Voting Portal" [ref=e144] [cursor=pointer]: - /url: /voting-portal - generic [ref=e146]: 󰔔 - generic [ref=e148]: Voting Portal - generic [ref=e150]: - button "AS Andreas Stadler andreas.stadler@storeroom.at" [ref=e153] [cursor=pointer]: - generic [ref=e156]: AS - generic [ref=e157]: - generic [ref=e158]: Andreas Stadler - generic "andreas.stadler@storeroom.at" [ref=e159] - generic [ref=e160]: 󰇙 - generic: - text: 󰗊 󰅀 - text: 󰷖 󰍃 - button [ref=e161] [cursor=pointer]: - generic [ref=e163]: 󰋽 - main [ref=e164]: - generic [ref=e166]: - generic [ref=e168]: - generic [ref=e169]: - link "Protection Plan" [ref=e170] [cursor=pointer]: - /url: /insurances - generic [ref=e171]: Protection Plan - generic [ref=e172]: 󰅂 - generic [ref=e173]: Licensed Insurance JNmpjN - button "Actions" [ref=e175] [cursor=pointer]: - generic [ref=e176]: - text: Actions - generic [ref=e177]: 󱨉 - generic [ref=e179]: - generic [ref=e180]: - generic [ref=e182]: - generic [ref=e184]: - heading "General" [level=3] [ref=e186] - link [ref=e188] [cursor=pointer]: - /url: /insurances/1406417555/edit/general - generic [ref=e190]: 󰲶 - table [ref=e193]: - rowgroup [ref=e194]: - row "Name Licensed Insurance JNmpjN" [ref=e195]: - rowheader "Name" [ref=e196] - cell "Licensed Insurance JNmpjN" [ref=e197]: - generic "Licensed Insurance JNmpjN" [ref=e199] - row "Location Vienna South" [ref=e200]: - rowheader "Location" [ref=e201] - cell "Vienna South" [ref=e202]: - generic "Vienna South" [ref=e204] - generic [ref=e205]: - generic [ref=e206]: - generic [ref=e208]: - heading "Booking portal" [level=3] [ref=e210] - link [ref=e212] [cursor=pointer]: - /url: /insurances/1406417555/edit/storefront - generic [ref=e214]: 󰲶 - table [ref=e217]: - rowgroup [ref=e218]: - row "Short description -" [ref=e219]: - rowheader "Short description" [ref=e220] - cell "-" [ref=e221]: - generic [ref=e223]: "-" - generic [ref=e224]: - generic [ref=e226]: - heading "Taxes" [level=3] [ref=e228] - link [ref=e230] [cursor=pointer]: - /url: /insurances/1406417555/edit/tax - generic [ref=e232]: 󰲶 - table [ref=e235]: - rowgroup [ref=e236]: - row "B2C 20%" [ref=e237]: - rowheader "B2C" [ref=e238] - cell "20%" [ref=e239]: - generic "20%" [ref=e241] - row "B2B 20%" [ref=e242]: - rowheader "B2B" [ref=e243] - cell "20%" [ref=e244]: - generic "20%" [ref=e246] - generic [ref=e248]: - generic [ref=e250]: - heading "Booking plan" [level=3] [ref=e252] - link [ref=e254] [cursor=pointer]: - /url: /insurances/1406417555/booking-plan - generic [ref=e256]: 󰐙 - generic [ref=e259]: - generic [ref=e261]: - generic [ref=e264]: - button "Edit price" [disabled] [ref=e268]: - img [ref=e270] - generic [ref=e276]: Edit price - button "Publish" [disabled] [ref=e279]: - generic [ref=e281]: 󰛐 - generic [ref=e282]: Publish - button "Unpublish" [disabled] [ref=e285]: - generic [ref=e287]: 󰛑 - generic [ref=e288]: Unpublish - button "Delete" [disabled] [ref=e292]: - generic [ref=e294]: 󰩺 - generic [ref=e295]: Delete - table [ref=e297]: - rowgroup [ref=e298]: - row "Billing period Discount Price / period (excl. VAT) Period price (excl. VAT) Status Active Bookings Action" [ref=e299]: - columnheader "Billing period" [ref=e300]: - button "Billing period" [ref=e301] [cursor=pointer]: - generic [ref=e303]: Billing period - generic [ref=e306]: 󰁝 - columnheader "Discount" [ref=e307]: - button "Discount" [ref=e308] [cursor=pointer]: - generic [ref=e310]: Discount - generic [ref=e313]: 󰁝 - columnheader "Price / period (excl. VAT)" [ref=e314]: - button "Price / period (excl. VAT)" [ref=e315] [cursor=pointer]: - generic [ref=e317]: Price / period (excl. VAT) - generic [ref=e320]: 󰁝 - columnheader "Period price (excl. VAT)" [ref=e321]: - button "Period price (excl. VAT)" [ref=e322] [cursor=pointer]: - generic [ref=e324]: Period price (excl. VAT) - generic [ref=e327]: 󰁝 - columnheader "Status" [ref=e328]: - generic [ref=e331]: Status - columnheader "Active Bookings" [ref=e332]: - button "Active Bookings" [ref=e333] [cursor=pointer]: - generic [ref=e335]: Active Bookings - generic [ref=e338]: 󰁝 - columnheader "Action" [ref=e339]: - generic [ref=e342]: Action - row [ref=e343]: - columnheader [ref=e344] - rowgroup [ref=e345]: - row "No data yet" [ref=e346]: - cell "No data yet" [ref=e347]: - generic [ref=e348]: No data yet - generic [ref=e351]: - generic [ref=e352]: - generic [ref=e353]: "Items per page:" - combobox [ref=e356]: - generic [ref=e358] [cursor=pointer]: - generic [ref=e360]: "5" - combobox "Items per page:": "5" - generic [ref=e362]: 󰍝 - generic [ref=e363]: 0-0 of 0 - generic [ref=e364]: - button [disabled]: - generic: - generic: 󰘀 - button [disabled]: - generic: - generic: 󰅁 - button [disabled]: - generic: - generic: 󰅂 - button [disabled]: - generic: - generic: 󰘁 - generic: - tooltip - tooltip - tooltip - tooltip - tooltip - tooltip - tooltip - tooltip - tooltip - tooltip - tooltip - tooltip - tooltip - tooltip - tooltip - tooltip - tooltip - tooltip - tooltip - tooltip - tooltip - tooltip - tooltip - tooltip - tooltip - tooltip - tooltip - tooltip - tooltip - tooltip - tooltip - tooltip - tooltip - tooltip - tooltip - tooltip - tooltip - tooltip ``` # Test source ```ts 1 | import { 2 | getDefaultInsuranceCreateWeeklyBookingPlans, 3 | getInsuranceCreateData, 4 | } from '@/manager/modules/insurance/insurance-factories'; 5 | import { test } from '@/manager/modules/insurance/insurance-fixtures'; 6 | import { applyInsuranceDeleteBookingPlans } from '@/manager/modules/insurance/insurance-mappers'; 7 | import { insuranceDeleteBookingPlansHappyTestCases } from '@/manager/modules/insurance/test-cases/insurance-delete-booking-plans-happy-test-cases'; 8 | import { verifyBookingPlansTable } from '@/manager/modules/ui/booking-plan/booking-plan-assertions'; 9 | import { expect } from '@/shared/utils/matchers'; 10 | 11 | for (const tc of insuranceDeleteBookingPlansHappyTestCases) { 12 | test(tc.description, async ({ createInsurance, insuranceDetailsPage }) => { 13 | const insurance = await createInsurance(getInsuranceCreateData(tc.initialOptions)); 14 | const newInsurance = applyInsuranceDeleteBookingPlans(insurance, tc.changes); 15 | 16 | await insuranceDetailsPage.goto(insurance.id); 17 | 18 | await test.step('select booking plans to delete', () => 19 | insuranceDetailsPage.bookingPlanCard.dataTable.rowSelector.select( 20 | tc.changes.indexes.map((index) => insurance.bookingPlans.length - (index + 1)) 21 | )); 22 | 23 | await test.step('delete booking plans', async () => { 24 | const dialog = await insuranceDetailsPage.bookingPlanCard.openBookingPlansDeleteDialog(); 25 | await dialog.delete(); 26 | }); 27 | 28 | await test.step('verify booking plans on details page', async () => { 29 | await verifyBookingPlansTable( 30 | insuranceDetailsPage.bookingPlanCard.dataTable, 31 | newInsurance.bookingPlans, 32 | newInsurance.location.currency, 33 | 'insurance' 34 | ); 35 | }); 36 | }); 37 | } 38 | 39 | test.fail( 40 | 'rejects deleting all booking plans via the row action @bug', 41 | async ({ createInsurance, insuranceDetailsPage }) => { 42 | const insurance = await createInsurance( 43 | getInsuranceCreateData({ bookingPlans: getDefaultInsuranceCreateWeeklyBookingPlans() }) 44 | ); 45 | 46 | await insuranceDetailsPage.goto(insurance.id); 47 | 48 | const dialog = await test.step('attempt to delete all booking plans via row action', async () => { 49 | await insuranceDetailsPage.bookingPlanCard.dataTable.rowSelector.selectAll(); 50 | const deleteDialog = await insuranceDetailsPage.bookingPlanCard.openBookingPlansDeleteDialog(); 51 | await deleteDialog.submit(); 52 | return deleteDialog; 53 | }); 54 | 55 | await test.step('verify error snackbar', async () => { > 56 | await expect(dialog.errorSnackbar).toHaveCountGreaterThan(0); | ^ Error: expect(locator).toHaveCountGreaterThan(0) 57 | }); 58 | } 59 | ); 60 | ```