# 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-create.spec.ts >> rejects an insurance with a duplicate name @bug - Location: src/manager/tests/e2e/insurance/insurance-create.spec.ts:71:6 # Error details ``` Error: expect(locator).toHaveCountGreaterThan(0) Locator: locator('.v-dialog:has([data-test-id="insurance-create-dialog"]):has(.v-overlay__content:not(.dialog-bottom-transition-enter-active))').locator('.v-input--error') Expected: > 0 Received: 0 ``` # Page snapshot ```yaml - generic [active] [ref=e1]: - generic [ref=e6]: - banner [ref=e7]: - generic [ref=e8]: - button [ref=e9] [cursor=pointer]: - generic [ref=e11]: 󰍜 - link [ref=e12] [cursor=pointer]: - /url: / - button "AS" [ref=e15] [cursor=pointer]: - generic [ref=e19]: AS - generic: - text: 󰗊 󰅀 - text: 󰷖 󰍃 - navigation [ref=e20]: - list [ref=e22]: - generic [ref=e24]: Dashboard - link "Tasks" [ref=e25] [cursor=pointer]: - /url: /tasks - img [ref=e28] - generic [ref=e31]: Tasks - link "Facility maps" [ref=e32] [cursor=pointer]: - /url: /facility-map - generic [ref=e34]: 󰧾 - generic [ref=e36]: Facility maps - link "Analytics" [ref=e37] [cursor=pointer]: - /url: /dashboard - generic [ref=e39]: 󱖶 - generic [ref=e41]: Analytics - generic [ref=e43]: Sales - link "Bookings" [ref=e44] [cursor=pointer]: - /url: /bookings - generic [ref=e46]: 󰇡 - generic [ref=e48]: Bookings - link "Customers" [ref=e49] [cursor=pointer]: - /url: /customers - generic [ref=e51]: 󰀏 - generic [ref=e53]: Customers - link "Invoices" [ref=e54] [cursor=pointer]: - /url: /invoices - generic [ref=e56]: 󰷉 - generic [ref=e58]: Invoices - link "Credit notes" [ref=e59] [cursor=pointer]: - /url: /credit-notes - img [ref=e62] - generic [ref=e65]: Credit notes - link "Units" [ref=e66] [cursor=pointer]: - /url: /units - generic [ref=e68]: 󰍀 - generic [ref=e70]: Units - generic [ref=e72]: Site management - link "Locations" [ref=e73] [cursor=pointer]: - /url: /locations - generic [ref=e75]: 󰟙 - generic [ref=e77]: Locations - link "Unit types" [ref=e78] [cursor=pointer]: - /url: /unit-types - generic [ref=e80]: 󰆧 - generic [ref=e82]: Unit types - link "Protection Plans" [ref=e83] [cursor=pointer]: - /url: /insurances - generic [ref=e85]: 󰳌 - generic [ref=e87]: Protection Plans - link "Deposits" [ref=e88] [cursor=pointer]: - /url: /deposits - generic [ref=e90]: 󱙆 - generic [ref=e92]: Deposits - link "Products" [ref=e93] [cursor=pointer]: - /url: /products - generic [ref=e95]: 󰄑 - generic [ref=e97]: Products - link "Discounts" [ref=e98] [cursor=pointer]: - /url: /discounts - generic [ref=e100]: 󰓼 - generic [ref=e102]: Discounts - generic [ref=e103]: - option "Emails" [ref=e104] [cursor=pointer]: - generic [ref=e106]: 󰻨 - generic [ref=e108]: Emails - generic [ref=e112]: 󰅀 - text: 󱡰 󰁥 - generic [ref=e114]: Admin - link "Integrations" [ref=e115] [cursor=pointer]: - /url: /connected-apps - generic [ref=e117]: 󱘖 - generic [ref=e119]: Integrations - link "User & Roles" [ref=e120] [cursor=pointer]: - /url: /users - generic [ref=e122]: 󰭘 - generic [ref=e124]: User & Roles - generic [ref=e125]: - option "Booking Portal" [ref=e126] [cursor=pointer]: - generic [ref=e128]: 󱃁 - generic [ref=e130]: Booking Portal - generic [ref=e134]: 󰅀 - text: 󰖟 󰟙 - generic [ref=e135]: - option "JaneAI" [ref=e136] [cursor=pointer]: - generic [ref=e138]: 󱙺 - generic [ref=e140]: JaneAI - generic [ref=e144]: 󰅀 - text: 󱜹 - generic [ref=e146]: Feedback - link "Voting Portal" [ref=e147] [cursor=pointer]: - /url: /voting-portal - generic [ref=e149]: 󰔔 - generic [ref=e151]: Voting Portal - generic [ref=e153]: - generic [ref=e154]: Kinnovis GmbH © 2026 - generic [ref=e155]: v2026.05.04 - main [ref=e156]: - generic [ref=e158]: - generic [ref=e160]: - generic [ref=e161]: - link "Protection Plan" [ref=e162] [cursor=pointer]: - /url: /insurances - generic [ref=e163]: 󰁍 - generic [ref=e164]: Protection Plan - heading "Handcrafted Insurance DjBedy" [level=1] [ref=e168]: - generic [ref=e169]: Handcrafted Insurance DjBedy - button "Actions" [ref=e171] [cursor=pointer]: - generic [ref=e172]: - text: Actions - generic [ref=e173]: 󱨉 - generic [ref=e175]: - generic [ref=e176]: - generic [ref=e178]: - generic [ref=e180]: - heading "General" [level=3] [ref=e182] - link [ref=e184] [cursor=pointer]: - /url: /insurances/1661166278/edit/general - generic [ref=e186]: 󰲶 - table [ref=e189]: - rowgroup [ref=e190]: - row "Name Handcrafted Insurance DjBedy" [ref=e191]: - rowheader "Name" [ref=e192] - cell "Handcrafted Insurance DjBedy" [ref=e193]: - generic "Handcrafted Insurance DjBedy" [ref=e195] - row "Location Vienna South" [ref=e196]: - rowheader "Location" [ref=e197] - cell "Vienna South" [ref=e198]: - generic "Vienna South" [ref=e200] - generic [ref=e201]: - generic [ref=e202]: - generic [ref=e204]: - heading "Booking portal" [level=3] [ref=e206] - link [ref=e208] [cursor=pointer]: - /url: /insurances/1661166278/edit/storefront - generic [ref=e210]: 󰲶 - table [ref=e213]: - rowgroup [ref=e214]: - row "Short description -" [ref=e215]: - rowheader "Short description" [ref=e216] - cell "-" [ref=e217]: - generic [ref=e219]: "-" - generic [ref=e220]: - generic [ref=e222]: - heading "Taxes" [level=3] [ref=e224] - link [ref=e226] [cursor=pointer]: - /url: /insurances/1661166278/edit/tax - generic [ref=e228]: 󰲶 - table [ref=e231]: - rowgroup [ref=e232]: - row "B2C 20%" [ref=e233]: - rowheader "B2C" [ref=e234] - cell "20%" [ref=e235]: - generic "20%" [ref=e237] - row "B2B 20%" [ref=e238]: - rowheader "B2B" [ref=e239] - cell "20%" [ref=e240]: - generic "20%" [ref=e242] - generic [ref=e244]: - generic [ref=e246]: - heading "Booking plan" [level=3] [ref=e248] - link [ref=e250] [cursor=pointer]: - /url: /insurances/1661166278/booking-plan - generic [ref=e252]: 󰐙 - generic [ref=e255]: - generic [ref=e257]: - generic [ref=e260]: - button "Edit price" [disabled] [ref=e264]: - img [ref=e266] - generic [ref=e272]: Edit price - button "Publish" [disabled] [ref=e275]: - generic [ref=e277]: 󰛐 - generic [ref=e278]: Publish - button "Unpublish" [disabled] [ref=e281]: - generic [ref=e283]: 󰛑 - generic [ref=e284]: Unpublish - button "Delete" [disabled] [ref=e288]: - generic [ref=e290]: 󰩺 - generic [ref=e291]: Delete - table [ref=e293]: - rowgroup [ref=e294]: - row "Billing period Discount Price / period (excl. VAT) Period price (excl. VAT) Status Active Bookings Action" [ref=e295]: - columnheader [ref=e296]: - generic [ref=e297]: - checkbox [ref=e298] - generic [ref=e300] [cursor=pointer]: 󰄱 - columnheader "Billing period" [ref=e301]: - button "Billing period" [ref=e302] [cursor=pointer]: - generic [ref=e304]: Billing period - generic [ref=e307]: 󰁝 - columnheader "Discount" [ref=e308]: - button "Discount" [ref=e309] [cursor=pointer]: - generic [ref=e311]: Discount - generic [ref=e314]: 󰁝 - columnheader "Price / period (excl. VAT)" [ref=e315]: - button "Price / period (excl. VAT)" [ref=e316] [cursor=pointer]: - generic [ref=e318]: Price / period (excl. VAT) - generic [ref=e321]: 󰁝 - columnheader "Period price (excl. VAT)" [ref=e322]: - button "Period price (excl. VAT)" [ref=e323] [cursor=pointer]: - generic [ref=e325]: Period price (excl. VAT) - generic [ref=e328]: 󰁝 - columnheader "Status" [ref=e329]: - generic [ref=e332]: Status - columnheader "Active Bookings" [ref=e333]: - button "Active Bookings" [ref=e334] [cursor=pointer]: - generic [ref=e336]: Active Bookings - generic [ref=e339]: 󰁝 - columnheader "Action" [ref=e340]: - generic [ref=e343]: Action - row [ref=e344]: - columnheader [ref=e345] - rowgroup [ref=e346]: - row "1 month 0% €150.00 €150.00 Published 0" [ref=e347]: - cell [ref=e348]: - generic [ref=e349]: - checkbox [ref=e350] - generic [ref=e352] [cursor=pointer]: 󰄱 - cell "1 month" [ref=e353]: - generic [ref=e354]: 1 month - cell "0%" [ref=e355]: - generic [ref=e356]: 0% - cell "€150.00" [ref=e357]: - generic [ref=e358]: €150.00 - cell "€150.00" [ref=e359]: - generic [ref=e360]: €150.00 - cell "Published" [ref=e361]: - generic [ref=e365]: Published - cell "0" [ref=e366]: - generic [ref=e367]: "0" - cell [ref=e368]: - button [ref=e371] [cursor=pointer]: - generic [ref=e373]: 󰇘 - generic: 󰲶 󰩺 - generic [ref=e376]: - generic [ref=e377]: - generic [ref=e378]: "Items per page:" - combobox [ref=e381]: - generic [ref=e383] [cursor=pointer]: - generic [ref=e385]: "5" - combobox "Items per page:": "5" - generic [ref=e387]: 󰍝 - generic [ref=e388]: 1-1 of 1 - generic [ref=e389]: - 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 ``` # Test source ```ts 1 | import { getInsuranceCreateData } from '@/manager/modules/insurance/insurance-factories'; 2 | import { test } from '@/manager/modules/insurance/insurance-fixtures'; 3 | import { insuranceCreateBookingPlanUnhappyTestCases } from '@/manager/modules/insurance/test-cases/insurance-create-booking-plan-unhappy-test-cases'; 4 | import { insuranceCreateHappyTestCases } from '@/manager/modules/insurance/test-cases/insurance-create-happy-test-cases'; 5 | import { insuranceCreateUnhappyTestCases } from '@/manager/modules/insurance/test-cases/insurance-create-unhappy-test-cases'; 6 | import { verifyBookingPlansTable } from '@/manager/modules/ui/booking-plan/booking-plan-assertions'; 7 | import { expectDataTableTextColumnToHaveText } from '@/manager/modules/ui/data-table/data-table-assertions'; 8 | import { insuranceTableColumnTestIds } from '@/manager/modules/ui/data-table/data/data-table-column-test-ids'; 9 | import { verifyTaxCard } from '@/manager/modules/ui/tax/tax-assertions'; 10 | import { expectSingleLocatorToHaveText } from '@/manager/shared/utils/expect-utils'; 11 | import { BookingPlan } from '@/shared/types/booking-plan-types'; 12 | import { expect } from '@/shared/utils/matchers'; 13 | 14 | for (const tc of insuranceCreateHappyTestCases) { 15 | test(tc.description, async ({ insuranceCreateDialog, trackInsuranceForTeardown }) => { 16 | const data = getInsuranceCreateData(tc.options); 17 | const shortDescription = data.shortDescription?.[data.location.customerDefaultLanguage.id]; 18 | 19 | await insuranceCreateDialog.goto(); 20 | 21 | const detailsPage = await test.step('create insurance', () => insuranceCreateDialog.create(data)); 22 | 23 | trackInsuranceForTeardown(detailsPage.getInsuranceId()); 24 | 25 | await test.step('verify new insurance on details page', async () => { 26 | await expectSingleLocatorToHaveText(detailsPage.title, data.name); 27 | await expectSingleLocatorToHaveText(detailsPage.generalCard.name, data.name); 28 | await expectSingleLocatorToHaveText(detailsPage.generalCard.location, data.location.name); 29 | await expectSingleLocatorToHaveText(detailsPage.bookingPortalCard.shortDescription, shortDescription); 30 | await verifyTaxCard(detailsPage.taxCard, data.taxes); 31 | await verifyBookingPlansTable( 32 | detailsPage.bookingPlanCard.dataTable, 33 | data.bookingPlans, 34 | data.location.currency, 35 | 'insurance' 36 | ); 37 | }); 38 | 39 | await test.step('verify new insurance on list page', async () => { 40 | const listPage = await detailsPage.returnToInsuranceListPage(); 41 | 42 | await listPage.searchTextField.fill(data.name); 43 | 44 | await expect(listPage.dataTable.getRows()).toHaveCount(1); 45 | await expect(listPage.dataTable.getRowColumn(0, insuranceTableColumnTestIds.id)).not.toBeEmpty(); 46 | await expectDataTableTextColumnToHaveText(listPage.dataTable, insuranceTableColumnTestIds.name, data.name); 47 | await expectDataTableTextColumnToHaveText( 48 | listPage.dataTable, 49 | insuranceTableColumnTestIds.location, 50 | data.location.name 51 | ); 52 | }); 53 | }); 54 | } 55 | 56 | test('rejects an insurance without all required fields', async ({ insuranceCreateDialog }) => { 57 | const data = getInsuranceCreateData({ bookingPlans: [] }); 58 | 59 | await insuranceCreateDialog.goto(); 60 | 61 | await test.step('submit invalid insurance create form', async () => { 62 | await insuranceCreateDialog.fill(data); 63 | await insuranceCreateDialog.submit(); 64 | }); 65 | 66 | await test.step('verify errors on form', async () => { 67 | await expect(insuranceCreateDialog.errors).toHaveCountGreaterThan(0); 68 | }); 69 | }); 70 | 71 | test.fail('rejects an insurance with a duplicate name @bug', async ({ createInsurance, insuranceCreateDialog }) => { 72 | const existing = await createInsurance(getInsuranceCreateData()); 73 | const data = getInsuranceCreateData({ name: existing.name }); 74 | 75 | await insuranceCreateDialog.goto(); 76 | 77 | await test.step('submit insurance create form with duplicate name', async () => { 78 | await insuranceCreateDialog.fill(data); 79 | await insuranceCreateDialog.submit(); 80 | }); 81 | 82 | await test.step('verify errors on form', async () => { > 83 | await expect(insuranceCreateDialog.errors).toHaveCountGreaterThan(0); | ^ Error: expect(locator).toHaveCountGreaterThan(0) 84 | }); 85 | }); 86 | 87 | for (const tc of insuranceCreateUnhappyTestCases) { 88 | test(tc.description, async ({ insuranceCreateDialog }) => { 89 | const data = getInsuranceCreateData(tc.options); 90 | 91 | await insuranceCreateDialog.goto(); 92 | 93 | await test.step('submit invalid insurance create form', async () => { 94 | await insuranceCreateDialog.fill(data); 95 | await insuranceCreateDialog.submit(); 96 | }); 97 | 98 | await test.step('verify errors on form', async () => { 99 | await expect(insuranceCreateDialog.errors).toHaveCountGreaterThan(0); 100 | }); 101 | }); 102 | } 103 | 104 | for (const tc of insuranceCreateBookingPlanUnhappyTestCases) { 105 | test(tc.description, async ({ insuranceCreateDialog }) => { 106 | await insuranceCreateDialog.goto(); 107 | 108 | await test.step('verify invalid booking plans cannot be added', async () => { 109 | for (const plan of tc.bookingPlans) { 110 | const bookingPlanAddDialog = await insuranceCreateDialog.bookingPlanFieldSet.openBookingPlanAddDialog(); 111 | await bookingPlanAddDialog.fill(plan); 112 | await bookingPlanAddDialog.submit(); 113 | await expect(bookingPlanAddDialog.errors.or(bookingPlanAddDialog.notUniqueSnackbar)).toHaveCountGreaterThan(0); 114 | await bookingPlanAddDialog.cancel(); 115 | await bookingPlanAddDialog.main.waitFor({ state: 'hidden' }); 116 | } 117 | }); 118 | 119 | await test.step('verify booking plans table is empty', async () => { 120 | await expect(insuranceCreateDialog.bookingPlanFieldSet.dataTable.getRows()).toHaveCount(0); 121 | }); 122 | }); 123 | } 124 | 125 | test('rejects an insurance with two booking plans sharing period and amount', async ({ insuranceCreateDialog }) => { 126 | const plan: BookingPlan = { 127 | period: 'weekly', 128 | amount: 1, 129 | priceExclVat: 50, 130 | discount: undefined, 131 | publish: true, 132 | }; 133 | 134 | await insuranceCreateDialog.goto(); 135 | 136 | await test.step('add two duplicate booking plans', () => 137 | insuranceCreateDialog.bookingPlanFieldSet.fill([plan, plan])); 138 | 139 | await test.step('verify errors on form', async () => { 140 | await expect(insuranceCreateDialog.errors).toHaveCountGreaterThan(0); 141 | }); 142 | }); 143 | ```