# 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-plan.spec.ts >> rejects deleting the only booking plan via the inline action @bug - Location: src/manager/tests/e2e/insurance/insurance-delete-booking-plan.spec.ts:36: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]: Manager - button [ref=e15] [cursor=pointer]: - img [ref=e16] - list [ref=e19]: - generic [ref=e22]: Dashboard - link "Tasks" [ref=e23] [cursor=pointer]: - /url: /tasks - img [ref=e26] - generic [ref=e29]: Tasks - link "Facility maps" [ref=e30] [cursor=pointer]: - /url: /facility-map - generic [ref=e32]: 󰧾 - generic [ref=e34]: Facility maps - link "Analytics" [ref=e35] [cursor=pointer]: - /url: /dashboard - generic [ref=e37]: 󱖶 - generic [ref=e39]: Analytics - generic [ref=e41]: Sales - link "Bookings" [ref=e42] [cursor=pointer]: - /url: /bookings - generic [ref=e44]: 󰇡 - generic [ref=e46]: Bookings - link "Customers" [ref=e47] [cursor=pointer]: - /url: /customers - generic [ref=e49]: 󰀏 - generic [ref=e51]: Customers - link "Invoices" [ref=e52] [cursor=pointer]: - /url: /invoices - generic [ref=e54]: 󰷉 - generic [ref=e56]: Invoices - link "Credit notes" [ref=e57] [cursor=pointer]: - /url: /credit-notes - img [ref=e60] - generic [ref=e63]: Credit notes - link "Units" [ref=e64] [cursor=pointer]: - /url: /units - generic [ref=e66]: 󰍀 - generic [ref=e68]: Units - generic [ref=e70]: Site management - link "Locations" [ref=e71] [cursor=pointer]: - /url: /locations - generic [ref=e73]: 󰟙 - generic [ref=e75]: Locations - link "Unit types" [ref=e76] [cursor=pointer]: - /url: /unit-types - generic [ref=e78]: 󰆧 - generic [ref=e80]: Unit types - link "Protection Plans" [ref=e81] [cursor=pointer]: - /url: /insurances - generic [ref=e83]: 󰳌 - generic [ref=e85]: Protection Plans - link "Deposits" [ref=e86] [cursor=pointer]: - /url: /deposits - generic [ref=e88]: 󱙆 - generic [ref=e90]: Deposits - link "Products" [ref=e91] [cursor=pointer]: - /url: /products - generic [ref=e93]: 󰄑 - generic [ref=e95]: Products - link "Discounts" [ref=e96] [cursor=pointer]: - /url: /discounts - generic [ref=e98]: 󰓼 - generic [ref=e100]: Discounts - generic [ref=e101]: - option "Emails" [ref=e102] [cursor=pointer]: - generic [ref=e104]: 󰻨 - generic [ref=e106]: Emails - generic [ref=e110]: 󰅀 - text: 󱡰 󰁥 - generic [ref=e112]: Admin - link "Integrations" [ref=e113] [cursor=pointer]: - /url: /connected-apps - generic [ref=e115]: 󱘖 - generic [ref=e117]: Integrations - link "User & Roles" [ref=e118] [cursor=pointer]: - /url: /users - generic [ref=e120]: 󰭘 - generic [ref=e122]: User & Roles - generic [ref=e123]: - option "Booking Portal" [ref=e124] [cursor=pointer]: - generic [ref=e126]: 󱃁 - generic [ref=e128]: Booking Portal - generic [ref=e132]: 󰅀 - text: 󰖟 󰟙 - generic [ref=e133]: - option "JaneAI" [ref=e134] [cursor=pointer]: - generic [ref=e136]: 󱙺 - generic [ref=e138]: JaneAI - generic [ref=e142]: 󰅀 - text: 󱜹 - generic [ref=e144]: Feedback - link "Voting Portal" [ref=e145] [cursor=pointer]: - /url: /voting-portal - generic [ref=e147]: 󰔔 - generic [ref=e149]: Voting Portal - generic [ref=e152]: - button "AS Andreas Stadler andreas.stadler@storeroom.at" [ref=e155] [cursor=pointer]: - generic [ref=e158]: AS - generic [ref=e159]: - generic [ref=e160]: Andreas Stadler - generic "andreas.stadler@storeroom.at" [ref=e161] - generic [ref=e162]: 󰇙 - generic: - text: 󰗊 󰅀 - text: 󰷖 󰍃 - button [ref=e163] [cursor=pointer]: - generic [ref=e165]: 󰋽 - main [ref=e166]: - generic [ref=e168]: - generic [ref=e170]: - generic [ref=e171]: - link "Protection Plan" [ref=e172] [cursor=pointer]: - /url: /insurances - generic [ref=e173]: Protection Plan - generic [ref=e174]: 󰅂 - generic [ref=e175]: Practical Insurance RX8yTJ - button "Actions" [ref=e177] [cursor=pointer]: - generic [ref=e178]: - text: Actions - generic [ref=e179]: 󱨉 - generic [ref=e181]: - generic [ref=e182]: - generic [ref=e184]: - generic [ref=e186]: - heading "General" [level=3] [ref=e188] - link [ref=e190] [cursor=pointer]: - /url: /insurances/951124457/edit/general - generic [ref=e192]: 󰲶 - table [ref=e195]: - rowgroup [ref=e196]: - row "Name Practical Insurance RX8yTJ" [ref=e197]: - rowheader "Name" [ref=e198] - cell "Practical Insurance RX8yTJ" [ref=e199]: - generic "Practical Insurance RX8yTJ" [ref=e201] - row "Location Vienna South" [ref=e202]: - rowheader "Location" [ref=e203] - cell "Vienna South" [ref=e204]: - generic "Vienna South" [ref=e206] - generic [ref=e207]: - generic [ref=e208]: - generic [ref=e210]: - heading "Booking portal" [level=3] [ref=e212] - link [ref=e214] [cursor=pointer]: - /url: /insurances/951124457/edit/storefront - generic [ref=e216]: 󰲶 - table [ref=e219]: - rowgroup [ref=e220]: - row "Short description -" [ref=e221]: - rowheader "Short description" [ref=e222] - cell "-" [ref=e223]: - generic [ref=e225]: "-" - generic [ref=e226]: - generic [ref=e228]: - heading "Taxes" [level=3] [ref=e230] - link [ref=e232] [cursor=pointer]: - /url: /insurances/951124457/edit/tax - generic [ref=e234]: 󰲶 - table [ref=e237]: - rowgroup [ref=e238]: - row "B2C 20%" [ref=e239]: - rowheader "B2C" [ref=e240] - cell "20%" [ref=e241]: - generic "20%" [ref=e243] - row "B2B 20%" [ref=e244]: - rowheader "B2B" [ref=e245] - cell "20%" [ref=e246]: - generic "20%" [ref=e248] - generic [ref=e250]: - generic [ref=e252]: - heading "Booking plan" [level=3] [ref=e254] - link [ref=e256] [cursor=pointer]: - /url: /insurances/951124457/booking-plan - generic [ref=e258]: 󰐙 - generic [ref=e261]: - generic [ref=e263]: - generic [ref=e266]: - button "Edit price" [disabled] [ref=e270]: - img [ref=e272] - generic [ref=e278]: Edit price - button "Publish" [disabled] [ref=e281]: - generic [ref=e283]: 󰛐 - generic [ref=e284]: Publish - button "Unpublish" [disabled] [ref=e287]: - generic [ref=e289]: 󰛑 - generic [ref=e290]: Unpublish - button "Delete" [disabled] [ref=e294]: - generic [ref=e296]: 󰩺 - generic [ref=e297]: Delete - table [ref=e299]: - rowgroup [ref=e300]: - row "Billing period Discount Price / period (excl. VAT) Period price (excl. VAT) Status Active Bookings Action" [ref=e301]: - columnheader "Billing period" [ref=e302]: - button "Billing period" [ref=e303] [cursor=pointer]: - generic [ref=e305]: Billing period - generic [ref=e308]: 󰁝 - columnheader "Discount" [ref=e309]: - button "Discount" [ref=e310] [cursor=pointer]: - generic [ref=e312]: Discount - generic [ref=e315]: 󰁝 - columnheader "Price / period (excl. VAT)" [ref=e316]: - button "Price / period (excl. VAT)" [ref=e317] [cursor=pointer]: - generic [ref=e319]: Price / period (excl. VAT) - generic [ref=e322]: 󰁝 - columnheader "Period price (excl. VAT)" [ref=e323]: - button "Period price (excl. VAT)" [ref=e324] [cursor=pointer]: - generic [ref=e326]: Period price (excl. VAT) - generic [ref=e329]: 󰁝 - columnheader "Status" [ref=e330]: - generic [ref=e333]: Status - columnheader "Active Bookings" [ref=e334]: - button "Active Bookings" [ref=e335] [cursor=pointer]: - generic [ref=e337]: Active Bookings - generic [ref=e340]: 󰁝 - columnheader "Action" [ref=e341]: - generic [ref=e344]: Action - row [ref=e345]: - columnheader [ref=e346] - rowgroup [ref=e347]: - row "No data yet" [ref=e348]: - cell "No data yet" [ref=e349]: - generic [ref=e350]: No data yet - generic [ref=e353]: - generic [ref=e354]: - generic [ref=e355]: "Items per page:" - combobox [ref=e358]: - generic [ref=e360] [cursor=pointer]: - generic [ref=e362]: "5" - combobox "Items per page:": "5" - generic [ref=e364]: 󰍝 - generic [ref=e365]: 0-0 of 0 - generic [ref=e366]: - 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 { verifyBookingPlansTable } from '@/manager/modules/ui/booking-plan/booking-plan-assertions'; 8 | import { expect } from '@/shared/utils/matchers'; 9 | 10 | test('deletes a single booking plan via the inline action', async ({ createInsurance, insuranceDetailsPage }) => { 11 | const targetIndex = 0; 12 | const insurance = await createInsurance( 13 | getInsuranceCreateData({ bookingPlans: getDefaultInsuranceCreateWeeklyBookingPlans() }) 14 | ); 15 | const newInsurance = applyInsuranceDeleteBookingPlans(insurance, { indexes: [targetIndex] }); 16 | 17 | await insuranceDetailsPage.goto(insurance.id); 18 | 19 | await test.step('delete booking plan', async () => { 20 | const dialog = await insuranceDetailsPage.bookingPlanCard.openBookingPlanDeleteDialog( 21 | insurance.bookingPlans.length - (targetIndex + 1) 22 | ); 23 | await dialog.delete(); 24 | }); 25 | 26 | await test.step('verify booking plans on details page', async () => { 27 | await verifyBookingPlansTable( 28 | insuranceDetailsPage.bookingPlanCard.dataTable, 29 | newInsurance.bookingPlans, 30 | newInsurance.location.currency, 31 | 'insurance' 32 | ); 33 | }); 34 | }); 35 | 36 | test.fail( 37 | 'rejects deleting the only booking plan via the inline action @bug', 38 | async ({ createInsurance, insuranceDetailsPage }) => { 39 | const insurance = await createInsurance(getInsuranceCreateData()); 40 | 41 | await insuranceDetailsPage.goto(insurance.id); 42 | 43 | const dialog = await test.step('attempt to delete the only booking plan via inline action', async () => { 44 | const deleteDialog = await insuranceDetailsPage.bookingPlanCard.openBookingPlanDeleteDialog(0); 45 | await deleteDialog.submit(); 46 | return deleteDialog; 47 | }); 48 | 49 | await test.step('verify error snackbar', async () => { > 50 | await expect(dialog.errorSnackbar).toHaveCountGreaterThan(0); | ^ Error: expect(locator).toHaveCountGreaterThan(0) 51 | }); 52 | } 53 | ); 54 | ```