diff --git a/tests/fulfillment.spec.js b/tests/fulfillment.spec.js index a8f61ecc..b237bffb 100644 --- a/tests/fulfillment.spec.js +++ b/tests/fulfillment.spec.js @@ -1,83 +1,220 @@ -const { test, expect, chromium} = require('@playwright/test'); -const fs = require('fs'); - -// Create a writable stream for logging -const logFile = fs.createWriteStream('playwright-logs.txt', { flags: 'a' }); -const errorFile = fs.createWriteStream('playwright-errors.txt', { flags: 'a' }); -const networkLogFile = fs.createWriteStream('network-logs.txt', { flags: 'a' }); -const consoleLogFile = fs.createWriteStream('console-logs.txt', { flags: 'a' }); - -// Redirect console.log and console.error to the log files -const originalLog = console.log; -const originalError = console.error; - -console.log = (...args) => { - logFile.write(args.join(' ') + '\n'); - originalLog.apply(console, args); -}; - -console.error = (...args) => { - errorFile.write(args.join(' ') + '\n'); - originalError.apply(console, args); -}; - -test ('Fulfillment', async () => { +const { test, expect, chromium } = require('@playwright/test'); +test('Fulfillment', async () => { test.setTimeout(8000000); - - //Set up the environment - const browser = await chromium.launch(); - const context = await browser.newContext({headless: false, viewport: { width: 1380, height: 700 }}); - await context.tracing.start({ screenshots: true, snapshots: true }); // launch the browser in a visible window and set the visible portion of the web page within the browser window. + const browser = await chromium.launch({headless:false}); + const context = await browser.newContext({recordVideo: {dir: 'videos/'}}, {viewport: {width:1350, height:650}}); const page = await context.newPage(); - page.on('request', request => { - networkLogFile.write(`Request: ${request.method()} ${request.url()}\n`); - if (request.postData()) { - networkLogFile.write(`Post Data: ${request.postData()}\n`); - } - }); - - page.on('response', response => { - networkLogFile.write(`Response: ${response.status()} ${response.url()}\n`); - response.text().then(text => { - networkLogFile.write(`Response Body: ${text}\n`); - }).catch(error => { - networkLogFile.write(`Error reading response body: ${error}\n`); - }); - }); - - page.on('console', msg => { - consoleLogFile.write(`Console Log: ${msg.type()} ${msg.text()}\n`); - for (let i = 0; i < msg.args().length; ++i) - consoleLogFile.write(`Console Log Arg[${i}]: ${msg.args()[i]}\n`); - }); - // try-catch block for error handling - try { - - // Go to URL and verify the heading + // Function to login to launchpad + async function launchpadLogin() { await page.goto("https://launchpad.hotwax.io"); await page.waitForTimeout(2000); const loginButton = '.ion-color.ion-color-danger.md.button.button-outline.ion-activatable.ion-focusable'; if ((await page.isVisible(loginButton)) && (await page.isEnabled(loginButton))) await page.click(loginButton); await page.waitForTimeout(2000); - await page.waitForSelector('.flex', { visible: true}); - await page.fill('input[name="instanceUrl"]', 'demo-oms'); - const instanceEnteredValue = await page.$eval('input[name="instanceUrl"]', instance => instance.value); + await page.fill('input[name="instanceUrl"]', 'ucg-uat'); + await page.waitForTimeout(2000); + await page.keyboard.press('Enter'); + await page.waitForTimeout(2000); + //const username1 = process.env.USERNAME; + //const password1 = process.env.PASSWORD; + await page.fill('input[name="username"]', 'hotwax.user'); + await page.fill('input[name="password"]', 'hotwax@786'); await page.waitForTimeout(2000); await page.keyboard.press('Enter'); + await page.waitForSelector('#app > ion-app > ion-router-outlet > div:nth-child(1) > ion-content > main', {visible:true, timeout:10000}); + } + + // Function to get user details + async function getUserDetails(selectors) { + return { + username: await page.textContent(selectors.usernameElement), + instance: await page.textContent(selectors.instanceElement), + appVersion: await page.textContent(selectors.appVersionElement), + buildDateTime: await page.textContent(selectors.buildDateTimeElement), + }; + } + + // Function to check complete orderflow for max 3 facilities + async function orderFlowStart() { + const cancelButton = 'button:has-text("Cancel")'; + if ((await page.isVisible(cancelButton)) && (await page.isEnabled(cancelButton))) await page.click(cancelButton); + const crossButton = '.md.button.button-clear.in-toolbar.in-buttons.button-has-icon-only.ion-activatable.ion-focusable'; + if ((await page.isVisible(crossButton)) && (await page.isEnabled(crossButton))) await page.click(crossButton); + await page.waitForTimeout(2000); + await page.locator('ion-label:has-text("Open")').click(); + try { + await page.waitForTimeout(2000); + if (await page.isVisible('.md.chip-outline.ion-activatable')) { + await orderDetails(); + await checkOrderFilters(); + //await pickPackShipOrder(); + const result = await pickPackShipOrder(); + if (result === false) { + console.log("Failed to pick, pack, or ship order."); + return false; + } + } + return true; + } catch (e) { + const timestamp = new Date().toISOString().replace(/:/g, '-'); // Get current timestamp + await page.screenshot({ path: `error-screenshot-${timestamp}.png` }); + console.log('Error:', e); + return false; + } + } + + async function orderDetails() { + const metadata = await page.textContent('#view-size-selector > div > div.results > ion-card:nth-child(2) > div.order-header > div.order-metadata'); + const orderPrimaryInfo = await page.textContent('#view-size-selector > div > div.results > ion-card:nth-child(2) > div.order-header > div.order-primary-info'); + const orderId = await page.textContent('#view-size-selector > div > div.results > ion-card:nth-child(2) > div.order-header > div.order-tags > ion-chip > ion-label'); + await page.locator('.md.chip-outline.ion-activatable').first().click(); + await page.waitForSelector('.popover-viewport', {visible:true, timeout:10000}); + await page.waitForTimeout(1000); + await page.locator('ion-item:has-text("View details")').click(); + await page.waitForSelector('.ion-page.can-go-back', {visible:true, timeout:10000}); + expect(await page.textContent('#main-content > div.ion-page.can-go-back > ion-content > div > div.order-header > div.order-metadata > ion-badge')).toBe("Open"); + expect(await page.textContent('#view-size-selector > div > div.results > ion-card:nth-child(2) > div.order-header > div.order-metadata')).toBe(metadata); + expect(await page.textContent('#main-content > div.ion-page.can-go-back > ion-content > div > ion-card > div.order-header > div.order-primary-info')).toBe(orderPrimaryInfo); + expect(await page.textContent('#main-content > div.ion-page.can-go-back > ion-content > div > div.order-header > div.order-primary-info')).toBe(orderId); + await page.waitForTimeout(2000); + await page.click('.md.button.back-button-has-icon-only.in-toolbar.ion-activatable.ion-focusable.show-back-button'); await page.waitForTimeout(2000); - await page.fill('input[name="username"]',''); - const usernameEnteredValue = await page.$eval('input[name="username"]', username => username.value); - await page.locator('input[name="password"]').fill(''); + } + + // Function to apply order filters and validate the number of orders being fetched + async function checkOrderFilters() { + let previousOrderCount = 0; + let previousLastOrderItemsCount = 0; + let filterOptionText = 0; + const orderCount = await page.locator('#main-content > div > ion-header > ion-toolbar > ion-title'); + const numberOfOrders = await orderCount.evaluate(element => element.childNodes[0].textContent.split(' ')[0]); + const totalNumberOfOrders = await orderCount.evaluate(element => element.childNodes[0].textContent.split(' ')[2]); + + if (Number(totalNumberOfOrders) >= 11) { + await page.locator('ion-buttons.buttons-last-slot.sc-ion-buttons-md-h.sc-ion-buttons-md-s.md ion-menu-button').click(); + await page.waitForSelector('div.ion-page div.menu-inner', {visible:true, timeout:10000}); + const matchOrderNumber = await page.$eval('.radio-checked', element => element.textContent.split(' ')[0]); + expect(Number(numberOfOrders)).toBe(Number(matchOrderNumber)); + await page.waitForTimeout(2000); + const orderFilterOptions = await page.$$('.md.in-item.radio-justify-start.radio-alignment-center.radio-label-placement-end:not(.radio-checked)'); + const randomOption = orderFilterOptions[Math.floor(Math.random() * orderFilterOptions.length)]; + filterOptionText = await randomOption.textContent(); + await randomOption.click(); + await page.click('.ion-color.ion-color-light.md.item'); + await page.waitForTimeout(2000); + } else { + await page.click('.ion-color.ion-color-light.md.item'); + await page.waitForTimeout(2000); + } + + while (true) { + const orderCards = await page.$$('.md.order'); + const currentOrderCount = orderCards.length; + if (currentOrderCount === previousOrderCount) break; + const lastOrderCard = orderCards[currentOrderCount - 1]; + await lastOrderCard.scrollIntoViewIfNeeded(); + await page.waitForTimeout(2000); + previousOrderCount = currentOrderCount; + while (true) { + const lastOrderItems = await lastOrderCard.$$('.order-item'); + const lastOrderItemsCount = lastOrderItems.length; + if (lastOrderItemsCount === previousLastOrderItemsCount) break; + const lastOrderItem = lastOrderItems[lastOrderItemsCount - 1]; + await lastOrderItem.scrollIntoViewIfNeeded(); + await page.waitForTimeout(2000); + previousLastOrderItemsCount = lastOrderItemsCount; + } + } + const orderCards = await page.$$('.md.chip-outline.ion-activatable'); + console.log(`Total order cards loaded: ${orderCards.length}`); + expect (filterOptionText).toContain(orderCards.length.toString()); + } + + // Function to pick, pack and ship the order + async function pickPackShipOrder() { + await page.locator('.md.chip-outline.ion-activatable').first().click(); + await page.waitForSelector('.popover-viewport', {visible:true, timeout:10000}); + await page.locator('ion-item:has-text("Pick order")').click(); + await page.waitForTimeout(4000); + await page.locator('ion-item:has-text("Copy ID")').click(); + await page.waitForTimeout(4000); + const pickerList = await page.$$('.item.md.item-lines-default.item-fill-none.item-has-interactive-control.in-list.ion-activatable.ion-focusable.item-label'); + if (pickerList.length == 0) console.log("No picker found in picklist"); + const randomPicker = pickerList[Math.floor(Math.random() * pickerList.length)]; + await randomPicker.click(); await page.waitForTimeout(2000); + const ionFabButton = 'ion-modal > div > ion-fab > ion-fab-button'; + if (await page.isDisabled(ionFabButton)) { + console.log('Picklist not saved'); + } else { + await page.click(ionFabButton); + } + await newTab(); + await page.waitForTimeout(2000); + await page.locator('ion-label:has-text("In Progress")').click(); + await page.waitForTimeout(2000); + await page.locator('.searchbar-input.sc-ion-searchbar-md').click(); + await page.keyboard.press('Control+V'); await page.keyboard.press('Enter'); + await page.waitForTimeout(4000); + await packShipButton(); + await page.waitForSelector('.alert-wrapper.ion-overlay-wrapper.sc-ion-alert-md', {visible:true, timeout:10000}); + await page.waitForTimeout(2000); + await page.locator('.alert-tappable.alert-checkbox.alert-checkbox-button.ion-focusable.sc-ion-alert-md').nth(0).click(); + await page.waitForTimeout(1000); + await page.locator('.alert-tappable.alert-checkbox.alert-checkbox-button.ion-focusable.sc-ion-alert-md').nth(1).click(); + await page.waitForTimeout(2000); + await page.locator('.alert-button.ion-focusable.ion-activatable.alert-button-role-confirm.sc-ion-alert-md').click(); + await newTab(); await page.waitForTimeout(2000); - await page.waitForSelector('#app > ion-app > ion-router-outlet > div:nth-child(1) > ion-content > main', { visible: true}); - const appCard = await page.locator('ion-card:has-text("Fulfillment")'); - await appCard.locator('.ion-color.ion-color-medium.md.button.button-clear.in-buttons.button-has-icon-only.ion-activatable.ion-focusable').nth(1).click(); - await page.waitForSelector('.split-pane-side.md.menu-type-overlay.menu-side-start.menu-pane-visible.menu-enabled', { visible: true}); + await page.locator('ion-label:has-text("Completed")').click(); + await page.waitForTimeout(3000); + await page.locator('.searchbar-input.sc-ion-searchbar-md').click(); + await page.keyboard.press('Control+V'); + await page.keyboard.press('Enter'); + await page.waitForTimeout(3000); + return await packShipButton(); + } + + async function newTab() { + const newTabPopup = page.waitForEvent('popup'); + const newTab = await newTabPopup; + await newTab.waitForLoadState(); + await page.waitForTimeout(4000); + await newTab.close(); + } + + async function packShipButton() { + if (await page.isVisible('.md.button.button-solid.ion-activatable.ion-focusable')) { + await page.click('.md.button.button-solid.ion-activatable.ion-focusable'); + console.log('pack button clicked'); + await page.waitForTimeout(3000); + return true; + } else { + console.log('order not found'); + // await orderFlowStart(); + return false; + } + } + + // Function to change the facility after each orderflow or incase no orders are present in open state for a facility + async function changeFacility() { await page.locator('ion-label:has-text("Settings")').click(); - await page.waitForSelector('.user-profile'); + await page.waitForTimeout(2000); + await page.locator('#main-content > div > ion-content > section:nth-child(3) > ion-card:nth-child(3) > ion-item').click(); + await page.waitForSelector('.popover-viewport', {visible:true, timeout:10000}); + await page.waitForTimeout(2000); + } + + try { + await launchpadLogin(); + await page.locator('ion-card:has-text("Fulfillment")').locator('.ion-color.ion-color-medium.md.button.button-clear.in-buttons.button-has-icon-only.ion-activatable.ion-focusable').nth(1).click(); + await page.waitForSelector('.split-pane-side.md.menu-type-overlay.menu-side-start.menu-pane-visible.menu-enabled', {visible:true, timeout:10000}); + await page.waitForTimeout(2000); + await page.locator('ion-label:has-text("Settings")').click(); + await page.waitForSelector('.user-profile', {visible:true, timeout:10000}); + await page.waitForTimeout(2000); const selectors = { usernameElement: 'div.user-profile ion-card-subtitle', @@ -85,139 +222,65 @@ test ('Fulfillment', async () => { appVersionElement: '.section-header div p.overline:nth-child(2)', buildDateTimeElement: '.section-header div p.overline:nth-child(1)' }; - const username = await page.locator(selectors.usernameElement).evaluate(element => element.textContent.trim()); - const instance = await page.locator(selectors.instanceElement).evaluate(element => element.textContent.trim()); - const appVersion = await page.locator(selectors.appVersionElement).evaluate(element => element.textContent.trim()); - const buildDateTime= await page.locator(selectors.buildDateTimeElement).evaluate(element => element.textContent.trim()); - if (instance && username && appVersion && buildDateTime) { - await expect(username).toContain(usernameEnteredValue); - await expect(instance).toContain(instanceEnteredValue); - } + + const { username, instance, appVersion, buildDateTime } = await getUserDetails(selectors); + expect(username).toContain('hotwax.user'); + expect(instance).toContain('ucg-uat'); + await page.locator('#main-content > div > ion-content > section:nth-child(3) > ion-card:nth-child(3) > ion-item').click(); - await page.waitForSelector('.popover-viewport', { visible: true}); + await page.waitForSelector('.popover-viewport', {visible:true, timeout:10000}); let dropdownItems = await page.$$('.select-interface-option.md.sc-ion-select-popover-md.item.item-lines-default.item-fill-none.item-has-interactive-control.ion-activatable.ion-focusable'); - console.log(dropdownItems.length, 'facilities found'); - for (let i = 0; i < dropdownItems.length; i++) { - await dropdownItems[i].evaluate(element => element.scrollIntoView()); - await page.waitForTimeout(2000); + console.log(`${dropdownItems.length} facilities found`); + + let facilityRunCount = 0; + for (let i = 10; i < dropdownItems.length && facilityRunCount < 3; i++) { + await dropdownItems[i].scrollIntoViewIfNeeded(); + const facilityName = await dropdownItems[i].textContent(); await dropdownItems[i].click(); - await page.waitForTimeout(2000); - await page.locator('ion-label:has-text("Open")').click(); await page.waitForTimeout(3000); - if (await page.isVisible('.md.chip-outline.ion-activatable')){ - await page.waitForTimeout(2000); - let orderCards = await page.$$('.md.chip-outline.ion-activatable'); - const orderCount = await page.locator('#main-content > div > ion-header > ion-toolbar > ion-title'); - const numberOfOrders = await orderCount.evaluate(element => { - const getOrderValue = element.childNodes[0].textContent; - const getOrderValue1 = getOrderValue.split(' ')[0]; - return getOrderValue1;}); - await expect (orderCards.length).toBe(Number(numberOfOrders)); - console.log(numberOfOrders, 'orders are visible'); - await page.locator('ion-buttons.buttons-last-slot.sc-ion-buttons-md-h.sc-ion-buttons-md-s.md ion-menu-button').click(); - await page.waitForSelector('div.ion-page div.menu-inner'); - const matchOrderNumber = await page.$eval('.radio-checked', element => element.textContent); - //const matchOrderNumber1 = matchOrderNumber.split(' ')[0]; - await expect (matchOrderNumber).toContain(numberOfOrders.toString()); - await page.waitForTimeout(2000); - const orderFilterOptions = await page.$$('.md.in-item.radio-justify-start.radio-alignment-center.radio-label-placement-end'); - let randomIndex = Math.floor(Math.random() * orderFilterOptions.length); - const randomOption = orderFilterOptions[randomIndex]; - const filterOptionText = await randomOption.textContent(); - await randomOption.click(); - if (await page.isVisible('.ion-color.ion-color-light.md.item')) await page.click('.ion-color.ion-color-light.md.item'); - orderCards = await page.$$('.md.chip-outline.ion-activatable'); - await expect (filterOptionText).toContain(orderCards.length.toString()); - await page.waitForTimeout(2000); - - while (await page.isVisible('.md.chip-outline.ion-activatable')){ - await page.locator('.md.chip-outline.ion-activatable').nth(0).click(); - await page.waitForTimeout(2000); - await page.locator('ion-item:has-text("Pick order")').click(); - await page.waitForTimeout(4000); - await page.locator('ion-item:has-text("Copy ID")').click(); - await page.waitForTimeout(4000); - const pickerList = await page.$$('.item.md.item-lines-default.item-fill-none.item-has-interactive-control.in-list.ion-activatable.ion-focusable.item-label'); - randomIndex = Math.floor(Math.random() * pickerList.length); - const randomPicker = pickerList[randomIndex]; - await randomPicker.click(); - await page.waitForTimeout(2000); - await page.waitForSelector('ion-modal'); - const ionFabButton = await page.$('ion-modal > div > ion-fab > ion-fab-button'); - if (ionFabButton) { - await ionFabButton.click(); - } else { - console.log('Picklist not saved'); - } - let newTabPopup = page.waitForEvent('popup'); - let newTab = await newTabPopup; - await newTab.waitForLoadState(); - await page.waitForTimeout(4000); - await newTab.close(); - await page.waitForTimeout(2000); - await page.locator('ion-label:has-text("In Progress")').click(); - await page.waitForTimeout(2000); - await page.locator('.searchbar-input.sc-ion-searchbar-md').click(); - await page.keyboard.press('Control+V'); - await page.keyboard.press('Enter'); - await page.waitForTimeout(2000); - await page.locator('.md.button.button-solid.ion-activatable.ion-focusable').nth(0).click(); - await page.waitForSelector('.sc-ion-alert-md-h.sc-ion-alert-md-s.md'); - await page.waitForTimeout(2000); - const packOptions = await page.locator('.alert-checkbox-label.sc-ion-alert-md'); - const count = await packOptions.count(); - for (let j = 0; j < count; j++) { - await packOptions.nth(j).click(); - await page.waitForTimeout(2000); + const matchFacility = await page.locator('ion-menu ion-header ion-title').textContent(); + if (matchFacility === facilityName) { + let flowCompleted = false; + while(!flowCompleted) { + flowCompleted = await orderFlowStart(); + } + if (flowCompleted) { + if (!(await page.isVisible('.md.chip-outline.ion-activatable'))) { + await page.waitForSelector('.buttons-last-slot.sc-ion-buttons-md-h.sc-ion-buttons-md-s.md', {visible:true, timeout:10000}); + const button1 = '.ion-color.ion-color-danger.md.button.button-clear.in-toolbar.in-buttons.button-disabled.ion-activatable.ion-focusable'; + const button2 = '.md.button.menu-button-disabled.in-toolbar.ion-activatable.ion-focusable'; + if ((!await page.isEnabled(button1)) && (!await page.isEnabled(button2))){ + if (i==dropdownItems.length-1) { + break; + } + } + else { + await changeFacility(); + } } - await page.waitForTimeout(2000); - await page.locator('.alert-button.ion-focusable.ion-activatable.alert-button-role-confirm.sc-ion-alert-md').click(); - newTabPopup = page.waitForEvent('popup'); - newTab = await newTabPopup; - await newTab.waitForLoadState(); - await page.waitForTimeout(4000); - await newTab.close(); - await page.waitForTimeout(2000); - await page.locator('ion-label:has-text("Completed")').click(); - await page.waitForTimeout(3000); - await page.locator('.searchbar-input.sc-ion-searchbar-md').click(); - await page.keyboard.press('Control+V'); - await page.keyboard.press('Enter'); - await page.waitForTimeout(3000); - await page.locator('.md.button.button-solid.ion-activatable.ion-focusable').nth(0).click(); - await page.waitForTimeout(2000); - await page.locator('ion-label:has-text("Open")').click(); - await page.waitForTimeout(2000); + else { + facilityRunCount++; + if (facilityRunCount==3) break; + await changeFacility(); + } } } - if (i < dropdownItems.length-1) { - console.log('change facility'); - await page.locator('ion-label:has-text("Settings")').click(); - await page.waitForTimeout(2000); - await page.locator('#main-content > div > ion-content > section:nth-child(3) > ion-card:nth-child(3) > ion-item').click(); - await page.waitForTimeout(2000); - dropdownItems = await page.$$('.select-interface-option.md.sc-ion-select-popover-md.item.item-lines-default.item-fill-none.item-has-interactive-control.ion-activatable.ion-focusable'); + else { + console.log('Facility names do not match'); + await changeFacility(); } + dropdownItems = await page.$$('.select-interface-option.md.sc-ion-select-popover-md.item.item-lines-default.item-fill-none.item-has-interactive-control.ion-activatable.ion-focusable'); } - await page.waitForTimeout(5000); - console.log(appVersion); + + console.log(appVersion); console.log(buildDateTime); - // await page.click(".button-solid.ion-activatable.ion-focusable"); await page.waitForTimeout(3000); - } - catch (error) { - // Log the error for debugging - console.error('Error:', error); - } - finally { - await context.tracing.stop({ path: 'trace.zip' }); + } catch (error) { + const timestamp = new Date().toISOString().replace(/:/g, '-'); + await page.screenshot({path: `error-screenshot-${timestamp}.png`}); + console.error('Error:', error); + } finally { await page.close(); await browser.close(); - logFile.end(); - errorFile.end(); - networkLogFile.end(); - consoleLogFile.end(); } }); - -