diff --git a/.devcontainer/devcontainer.json b/.devcontainer/devcontainer.json index 1459e9ee..3d068995 100644 --- a/.devcontainer/devcontainer.json +++ b/.devcontainer/devcontainer.json @@ -33,7 +33,8 @@ "esbenp.prettier-vscode", "humao.rest-client", "GitHub.copilot", - "runem.lit-plugin" + "runem.lit-plugin", + "ms-playwright.playwright" ] } }, @@ -42,7 +43,7 @@ "forwardPorts": [3000, 3001, 5173], // Use 'postCreateCommand' to run commands after the container is created. - "postCreateCommand": "npm install", + "postCreateCommand": "./.devcontainer/postCreateCommand.sh", // Set minimal host requirements for the container. "hostRequirements": { diff --git a/.devcontainer/postCreateCommand.sh b/.devcontainer/postCreateCommand.sh new file mode 100755 index 00000000..a50a9893 --- /dev/null +++ b/.devcontainer/postCreateCommand.sh @@ -0,0 +1,7 @@ +#!/bin/bash + +echo "Installing dependencies..." +npm install + +echo "Installing playwright browsers..." +npx playwright install --with-deps diff --git a/.github/workflows/playwright.yml b/.github/workflows/playwright.yml new file mode 100644 index 00000000..a1d52131 --- /dev/null +++ b/.github/workflows/playwright.yml @@ -0,0 +1,27 @@ +name: Playwright Tests +on: + push: + branches: [main] + pull_request: + branches: [main] +jobs: + test: + timeout-minutes: 60 + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v3 + - uses: actions/setup-node@v3 + with: + node-version: 18 + - name: Install dependencies + run: npm ci + - name: Install Playwright Browsers + run: npm run install:playwright + - name: Run Playwright tests + run: npm run test:playwright + - uses: actions/upload-artifact@v3 + if: failure() + with: + name: playwright-report + path: playwright-report/ + retention-days: 30 diff --git a/.gitignore b/.gitignore index 84bfa743..0bc8ef4c 100644 --- a/.gitignore +++ b/.gitignore @@ -20,3 +20,6 @@ dist/ # misc TODO +/test-results/ +/playwright-report/ +/playwright/.cache/ diff --git a/package-lock.json b/package-lock.json index ddf7c504..e94c91f6 100644 --- a/package-lock.json +++ b/package-lock.json @@ -12,7 +12,9 @@ "packages/*" ], "devDependencies": { + "@playwright/test": "^1.39.0", "@tapjs/nock": "^3.1.13", + "@types/node": "^18.15.3", "concurrently": "^8.2.1", "eslint-config-shared": "^1.0.0", "lint-staged": "^14.0.1", @@ -2794,6 +2796,21 @@ "node": ">=14" } }, + "node_modules/@playwright/test": { + "version": "1.39.0", + "resolved": "https://registry.npmjs.org/@playwright/test/-/test-1.39.0.tgz", + "integrity": "sha512-3u1iFqgzl7zr004bGPYiN/5EZpRUSFddQBra8Rqll5N0/vfpqlP9I9EXqAoGacuAbX6c9Ulg/Cjqglp5VkK6UQ==", + "dev": true, + "dependencies": { + "playwright": "1.39.0" + }, + "bin": { + "playwright": "cli.js" + }, + "engines": { + "node": ">=16" + } + }, "node_modules/@react-spring/animated": { "version": "9.7.3", "resolved": "https://registry.npmjs.org/@react-spring/animated/-/animated-9.7.3.tgz", @@ -3611,12 +3628,9 @@ } }, "node_modules/@types/node": { - "version": "20.8.6", - "resolved": "https://registry.npmjs.org/@types/node/-/node-20.8.6.tgz", - "integrity": "sha512-eWO4K2Ji70QzKUqRy6oyJWUeB7+g2cRagT3T/nxYibYcT4y2BDL8lqolRXjTHmkZCdJfIPaY73KbJAZmcryxTQ==", - "dependencies": { - "undici-types": "~5.25.1" - } + "version": "18.15.3", + "resolved": "https://registry.npmjs.org/@types/node/-/node-18.15.3.tgz", + "integrity": "sha512-p6ua9zBxz5otCmbpb5D3U4B5Nanw6Pk3PPyX05xnxbB/fRv71N7CPmORg7uAD5P70T0xmx1pzAx/FUfa5X+3cw==" }, "node_modules/@types/node-fetch": { "version": "2.6.6", @@ -10604,6 +10618,50 @@ "node": ">=8" } }, + "node_modules/playwright": { + "version": "1.39.0", + "resolved": "https://registry.npmjs.org/playwright/-/playwright-1.39.0.tgz", + "integrity": "sha512-naE5QT11uC/Oiq0BwZ50gDmy8c8WLPRTEWuSSFVG2egBka/1qMoSqYQcROMT9zLwJ86oPofcTH2jBY/5wWOgIw==", + "devOptional": true, + "dependencies": { + "playwright-core": "1.39.0" + }, + "bin": { + "playwright": "cli.js" + }, + "engines": { + "node": ">=16" + }, + "optionalDependencies": { + "fsevents": "2.3.2" + } + }, + "node_modules/playwright-core": { + "version": "1.39.0", + "resolved": "https://registry.npmjs.org/playwright-core/-/playwright-core-1.39.0.tgz", + "integrity": "sha512-+k4pdZgs1qiM+OUkSjx96YiKsXsmb59evFoqv8SKO067qBA+Z2s/dCzJij/ZhdQcs2zlTAgRKfeiiLm8PQ2qvw==", + "devOptional": true, + "bin": { + "playwright-core": "cli.js" + }, + "engines": { + "node": ">=16" + } + }, + "node_modules/playwright/node_modules/fsevents": { + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz", + "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==", + "dev": true, + "hasInstallScript": true, + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": "^8.16.0 || ^10.6.0 || >=11.0.0" + } + }, "node_modules/pluralize": { "version": "8.0.0", "resolved": "https://registry.npmjs.org/pluralize/-/pluralize-8.0.0.tgz", @@ -12782,11 +12840,6 @@ "url": "https://github.com/sponsors/ljharb" } }, - "node_modules/undici-types": { - "version": "5.25.3", - "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-5.25.3.tgz", - "integrity": "sha512-Ga1jfYwRn7+cP9v8auvEXN1rX3sWqlayd4HP7OKk4mZWylEmu3KzXDUGrQUN6Ol7qo1gPvB2e5gX6udnyEPgdA==" - }, "node_modules/unique-filename": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/unique-filename/-/unique-filename-3.0.0.tgz", diff --git a/package.json b/package.json index ce15601e..cd563ab3 100644 --- a/package.json +++ b/package.json @@ -12,6 +12,7 @@ "start:search": "npm run dev --workspace=search", "start:indexer": "npm run dev --workspace=indexer", "test": "npm run test -ws --if-present", + "test:playwright": "npx playwright test", "build": "npm run build -ws --if-present", "clean": "npm run clean -ws --if-present", "docker:build": "npm run docker:build -ws --if-present", @@ -19,7 +20,8 @@ "format:check": "prettier --check .", "lint": "eslint .", "lint:fix": "eslint --fix .", - "prepare": "simple-git-hooks || echo 'simple-git-hooks install skipped'" + "prepare": "simple-git-hooks || echo 'simple-git-hooks install skipped'", + "install:playwright": "npx playwright install --with-deps" }, "keywords": [], "author": "Microsoft", @@ -28,7 +30,9 @@ "packages/*" ], "devDependencies": { + "@playwright/test": "^1.39.0", "@tapjs/nock": "^3.1.13", + "@types/node": "^18.15.3", "concurrently": "^8.2.1", "eslint-config-shared": "^1.0.0", "lint-staged": "^14.0.1", diff --git a/packages/chat-component/src/main.ts b/packages/chat-component/src/main.ts index 7e9b01d4..e4fedacf 100644 --- a/packages/chat-component/src/main.ts +++ b/packages/chat-component/src/main.ts @@ -368,6 +368,7 @@ export class ChatComponent extends LitElement { ${citation.ref}. ${citation.text} ${globalConfig.RESET_CHAT_BUTTON_TITLE} @@ -435,6 +437,7 @@ export class ChatComponent extends LitElement { diff --git a/playwright.config.ts b/playwright.config.ts new file mode 100644 index 00000000..d5c610b0 --- /dev/null +++ b/playwright.config.ts @@ -0,0 +1,82 @@ +import { defineConfig, devices } from '@playwright/test'; + +/** + * Read environment variables from file. + * https://github.com/motdotla/dotenv + */ +// require('dotenv').config(); + +const useLocalServer = true; + +/** + * See https://playwright.dev/docs/test-configuration. + */ +export default defineConfig({ + testDir: './tests', + testMatch: '*.spec.ts', + /* Run tests in files in parallel */ + fullyParallel: true, + /* Fail the build on CI if you accidentally left test.only in the source code. */ + forbidOnly: !!process.env.CI, + /* Retry on CI only */ + retries: process.env.CI ? 2 : 0, + /* Opt out of parallel tests on CI. */ + workers: process.env.CI ? 1 : undefined, + /* Reporter to use. See https://playwright.dev/docs/test-reporters */ + reporter: 'html', + /* Shared settings for all the projects below. See https://playwright.dev/docs/api/class-testoptions. */ + use: { + /* Base URL to use in actions like `await page.goto('/')`. */ + baseURL: useLocalServer ? 'http://localhost:5173' : process.env.WEBAPP_URI, + + /* Collect trace when retrying the failed test. See https://playwright.dev/docs/trace-viewer */ + trace: 'on-first-retry', + }, + + /* Configure projects for major browsers */ + projects: [ + { + name: 'chromium', + use: { ...devices['Desktop Chrome'] }, + }, + + { + name: 'firefox', + use: { ...devices['Desktop Firefox'] }, + }, + + { + name: 'webkit', + use: { ...devices['Desktop Safari'] }, + }, + + /* Test against mobile viewports. */ + // { + // name: 'Mobile Chrome', + // use: { ...devices['Pixel 5'] }, + // }, + // { + // name: 'Mobile Safari', + // use: { ...devices['iPhone 12'] }, + // }, + + /* Test against branded browsers. */ + // { + // name: 'Microsoft Edge', + // use: { ...devices['Desktop Edge'], channel: 'msedge' }, + // }, + // { + // name: 'Google Chrome', + // use: { ...devices['Desktop Chrome'], channel: 'chrome' }, + // }, + ], + + /* Run your local dev server before starting the tests */ + webServer: useLocalServer + ? { + command: 'npm run start:webapp', + url: 'http://localhost:5173', + reuseExistingServer: !process.env.CI, + } + : undefined, +}); diff --git a/tests/e2e/hars/default-chat-response-nostream.har b/tests/e2e/hars/default-chat-response-nostream.har new file mode 100644 index 00000000..e5bf2165 --- /dev/null +++ b/tests/e2e/hars/default-chat-response-nostream.har @@ -0,0 +1,75 @@ +{ + "log": { + "version": "1.2", + "creator": { + "name": "Playwright", + "version": "1.39.0" + }, + "browser": { + "name": "chromium", + "version": "119.0.6045.9" + }, + "entries": [ + { + "startedDateTime": "2023-10-25T00:22:30.306Z", + "time": 260.475, + "request": { + "method": "POST", + "url": "http://localhost:5173/chat", + "httpVersion": "HTTP/2.0", + "cookies": [], + "headers": [ + { "name": ":authority", "value": "http://localhost:5173" }, + { "name": ":method", "value": "POST" }, + { "name": ":path", "value": "/chat" }, + { "name": ":scheme", "value": "https" }, + { "name": "accept", "value": "*/*" }, + { "name": "accept-encoding", "value": "gzip, deflate, br" }, + { "name": "accept-language", "value": "en-US" }, + { "name": "content-length", "value": "438" }, + { "name": "content-type", "value": "application/json" }, + { "name": "origin", "value": "http://localhost:5173" }, + { "name": "sec-fetch-dest", "value": "empty" }, + { "name": "sec-fetch-mode", "value": "cors" }, + { "name": "sec-fetch-site", "value": "cross-site" }, + { + "name": "user-agent", + "value": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/119.0.6045.9 Safari/537.36" + } + ], + "queryString": [], + "headersSize": -1, + "bodySize": -1, + "postData": { + "mimeType": "application/json", + "text": "{\"messages\":[{\"content\":\"How to search and book rentals?\",\"role\":\"user\"}],\"context\":{\"retrieval_mode\":\"hybrid\",\"semantic_ranker\":true,\"semantic_captions\":false,\"suggest_followup_questions\":true,\"retrievalMode\":\"hybrid\",\"top\":3,\"useSemanticRanker\":true,\"useSemanticCaptions\":false,\"excludeCategory\":\"\",\"promptTemplate\":\"\",\"promptTemplatePrefix\":\"\",\"promptTemplateSuffix\":\"\",\"suggestFollowupQuestions\":true,\"approach\":\"rrr\"},\"stream\":false}", + "params": [] + } + }, + "response": { + "status": 200, + "statusText": "", + "httpVersion": "HTTP/2.0", + "cookies": [], + "headers": [ + { "name": "access-control-allow-origin", "value": "http://localhost:5173.net" }, + { "name": "content-length", "value": "10389" }, + { "name": "content-type", "value": "application/json; charset=utf-8" }, + { "name": "date", "value": "Wed, 25 Oct 2023 00:22:36 GMT" }, + { "name": "vary", "value": "Origin" } + ], + "content": { + "size": -1, + "mimeType": "application/json; charset=utf-8", + "text": "{\"choices\":[{\"index\":0,\"message\":{\"content\":\"To search and book rentals on Contoso Real Estate, follow these steps:\\n\\n1. Search for Rentals: Enter your destination, check-in and check-out dates, and the number of guests. Apply filters such as price range, property type, and amenities to narrow down your options. [support.md]\\n2. View Listing Details: Click on a listing to view detailed information, including photos, property description, reviews, and host information. [support.md]\\n3. Make a Booking: Click the \\\"Book Now\\\" button on the listing page. Review the booking details, including the total cost and house rules. Confirm your booking by providing payment information. Once the host accepts your booking, you'll receive a confirmation. [support.md]\\n4. Payment: Contoso Real Estate handles the payment process securely. You'll only be charged once your booking is confirmed. [support.md]\\n5. Communication: You can communicate with the host through our messaging system for any questions or special requests. [support.md]\\n\\n<>\\n<>\\n<>\",\"role\":\"assistant\",\"context\":{\"data_points\":[\"support.md: [How to Search and Book Rentals](#how-to-search-and-book-rentals) 2. [How to Cancel a Confirmed Booking](#how-to-cancel-a-confirmed-booking) 3. [How to Contact Customer Support](#how-to-contact-customer-support) 4. [How to Report a Problem with a Listing](#how-to-report-a-problem-with-a-listing) 5. [How to Report a Problem with a Guest or Host](#how-to-report-a-problem-with-a-guest-or-host) 6. [How to Report a Safety Issue](#how-to-report-a-safety-issue) 7. [How to Report a Payment or Refund Issue](#how-to-report-a-payment-or-refund-issue) 8. [Guest or Host Rule Violations](#guest-or-host-rule-violations) 9. [Compensation for Damage](#compensation-for-damage) 10. [Payment Error Resolution](#payment-error-resolution) --- ## 1. How to Search and Book Rentals Contoso Real Estate offers a wide range of rental properties for your stay. Here's how to search and book rentals: 1. **Search for Rentals:** - Enter your destination, check-in and check-out dates, and the number of guests. - Apply filters such as price range, property type, and amenities to narrow down your options.\",\"support.md: **Search for Rentals:** - Enter your destination, check-in and check-out dates, and the number of guests. - Apply filters such as price range, property type, and amenities to narrow down your options. - Browse through the listings to find the perfect place for your stay. 2. **View Listing Details:** - Click on a listing to view detailed information, including photos, property description, reviews, and host information. 3. **Make a Booking:** - Click the \\\"Book Now\\\" button on the listing page. - Review the booking details, including the total cost and house rules. - Confirm your booking by providing payment information. - Once the host accepts your booking, you'll receive a confirmation. 4. **Payment:** - Contoso Real Estate handles the payment process securely. You'll only be charged once your booking is confirmed. 5. **Communication:** - You can communicate with the host through our messaging system for any questions or special requests. --- ## 2. How to Cancel a Confirmed Booking Life happens, and sometimes you need to cancel a confirmed booking. Here's how to do it: 1.\",\"terms-of-service.md: We aim to create a trusted community where hosts can share their spaces and guests can enjoy memorable experiences. ## 2. How to Search and Book Rentals ### 2.1 Creating an Account To search for and book rentals, users must create a Contoso Real Estate account. You must provide accurate and complete information during the registration process. Users are responsible for maintaining the confidentiality of their account credentials. ### 2.2 Browsing and Searching - Users can browse available rentals without an account. - Use our search filters to narrow down rental options based on your preferences. - Click on a rental listing to view details, including pricing, availability, and host information. ### 2.3 Booking a Rental - To book a rental, click the \\\"Book Now\\\" button on the listing page. - Review the booking details, including dates and pricing, before confirming the reservation. - Payment is processed securely through our platform. Guests will receive a booking confirmation email. ### 2.4 Guest Verification - Guests may be required to complete identity verification steps.\"],\"thoughts\":\"Search query:
Search query: \\\"how to search and book rentals\\\"

Conversations:
system: Below is a history of the conversation so far, and a new question asked by the user that needs to be answered by searching in a knowledge base about terms of service, privacy policy, and questions about support requests.
Generate a search query based on the conversation and the new question.
Do not include cited source filenames and document names e.g info.txt or doc.pdf in the search query terms.
Do not include any text inside [] or <<>> in the search query terms.
Do not include any special characters like '+'.
If the question is not in English, translate the question to English before generating the search query.
If you cannot generate a search query, return just the number 0.


assistant: Refund policy

user: can I get refunded if cannot travel?

assistant: Show support for payment errors

user: What happens if a payment error occurs?

user: Generate search query for: How to search and book rentals?

system: Assistant helps the Consto Real Estate company customers with support questions regarding terms of service, privacy policy, and questions about support requests. Be brief in your answers.
Answer ONLY with the facts listed in the list of sources below. If there isn't enough information below, say you don't know. Do not generate answers that don't use the sources below. If asking a clarifying question to the user would help, ask the question.
For tabular information return it as an html table. Do not return markdown format. If the question is not in English, answer in the language used in the question.
Each source has a name followed by colon and the actual information, always include the source name for each fact you use in the response. Use square brackets to reference the source, for example: [info1.txt]. Don't combine sources, list each source separately, for example: [info1.txt][info2.pdf].
Generate 3 very brief follow-up questions that the user would likely ask next.
Enclose the follow-up questions in double angle brackets. Example:
<>
<>
<>

Do no repeat questions that have already been asked.
Make sure the last question ends with \\\">>\\\".



user: How to search and book rentals?

Sources:
support.md: [How to Search and Book Rentals](#how-to-search-and-book-rentals) 2. [How to Cancel a Confirmed Booking](#how-to-cancel-a-confirmed-booking) 3. [How to Contact Customer Support](#how-to-contact-customer-support) 4. [How to Report a Problem with a Listing](#how-to-report-a-problem-with-a-listing) 5. [How to Report a Problem with a Guest or Host](#how-to-report-a-problem-with-a-guest-or-host) 6. [How to Report a Safety Issue](#how-to-report-a-safety-issue) 7. [How to Report a Payment or Refund Issue](#how-to-report-a-payment-or-refund-issue) 8. [Guest or Host Rule Violations](#guest-or-host-rule-violations) 9. [Compensation for Damage](#compensation-for-damage) 10. [Payment Error Resolution](#payment-error-resolution) --- ## 1. How to Search and Book Rentals Contoso Real Estate offers a wide range of rental properties for your stay. Here's how to search and book rentals: 1. **Search for Rentals:** - Enter your destination, check-in and check-out dates, and the number of guests. - Apply filters such as price range, property type, and amenities to narrow down your options.
support.md: **Search for Rentals:** - Enter your destination, check-in and check-out dates, and the number of guests. - Apply filters such as price range, property type, and amenities to narrow down your options. - Browse through the listings to find the perfect place for your stay. 2. **View Listing Details:** - Click on a listing to view detailed information, including photos, property description, reviews, and host information. 3. **Make a Booking:** - Click the \\\"Book Now\\\" button on the listing page. - Review the booking details, including the total cost and house rules. - Confirm your booking by providing payment information. - Once the host accepts your booking, you'll receive a confirmation. 4. **Payment:** - Contoso Real Estate handles the payment process securely. You'll only be charged once your booking is confirmed. 5. **Communication:** - You can communicate with the host through our messaging system for any questions or special requests. --- ## 2. How to Cancel a Confirmed Booking Life happens, and sometimes you need to cancel a confirmed booking. Here's how to do it: 1.
terms-of-service.md: We aim to create a trusted community where hosts can share their spaces and guests can enjoy memorable experiences. ## 2. How to Search and Book Rentals ### 2.1 Creating an Account To search for and book rentals, users must create a Contoso Real Estate account. You must provide accurate and complete information during the registration process. Users are responsible for maintaining the confidentiality of their account credentials. ### 2.2 Browsing and Searching - Users can browse available rentals without an account. - Use our search filters to narrow down rental options based on your preferences. - Click on a rental listing to view details, including pricing, availability, and host information. ### 2.3 Booking a Rental - To book a rental, click the \\\"Book Now\\\" button on the listing page. - Review the booking details, including dates and pricing, before confirming the reservation. - Payment is processed securely through our platform. Guests will receive a booking confirmation email. ### 2.4 Guest Verification - Guests may be required to complete identity verification steps.\"}}}],\"object\":\"chat.completion\"}" + }, + "headersSize": -1, + "bodySize": -1, + "redirectURL": "" + }, + "cache": {}, + "timings": { "send": -1, "wait": -1, "receive": 260.475 } + } + ] + } +} diff --git a/tests/e2e/hars/default-chat-response-stream.har b/tests/e2e/hars/default-chat-response-stream.har new file mode 100644 index 00000000..48cffc47 --- /dev/null +++ b/tests/e2e/hars/default-chat-response-stream.har @@ -0,0 +1,75 @@ +{ + "log": { + "version": "1.2", + "creator": { + "name": "Playwright", + "version": "1.39.0" + }, + "browser": { + "name": "chromium", + "version": "119.0.6045.9" + }, + "entries": [ + { + "startedDateTime": "2023-10-25T00:27:07.983Z", + "time": 4348.057, + "request": { + "method": "POST", + "url": "http://localhost:5173/chat", + "httpVersion": "HTTP/2.0", + "cookies": [], + "headers": [ + { "name": ":authority", "value": "http://localhost:5173" }, + { "name": ":method", "value": "POST" }, + { "name": ":path", "value": "/chat" }, + { "name": ":scheme", "value": "http" }, + { "name": "accept", "value": "*/*" }, + { "name": "accept-encoding", "value": "gzip, deflate, br" }, + { "name": "accept-language", "value": "en-US" }, + { "name": "content-length", "value": "437" }, + { "name": "content-type", "value": "application/json" }, + { "name": "origin", "value": "http://localhost:5173" }, + { "name": "sec-fetch-dest", "value": "empty" }, + { "name": "sec-fetch-mode", "value": "cors" }, + { "name": "sec-fetch-site", "value": "cross-site" }, + { + "name": "user-agent", + "value": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/119.0.6045.9 Safari/537.36" + } + ], + "queryString": [], + "headersSize": -1, + "bodySize": -1, + "postData": { + "mimeType": "application/json", + "text": "{\"messages\":[{\"content\":\"How to search and book rentals?\",\"role\":\"user\"}],\"context\":{\"retrieval_mode\":\"hybrid\",\"semantic_ranker\":true,\"semantic_captions\":false,\"suggest_followup_questions\":true,\"retrievalMode\":\"hybrid\",\"top\":3,\"useSemanticRanker\":true,\"useSemanticCaptions\":false,\"excludeCategory\":\"\",\"promptTemplate\":\"\",\"promptTemplatePrefix\":\"\",\"promptTemplateSuffix\":\"\",\"suggestFollowupQuestions\":true,\"approach\":\"rrr\"},\"stream\":true}", + "params": [] + } + }, + "response": { + "status": 200, + "statusText": "", + "httpVersion": "HTTP/2.0", + "cookies": [], + "headers": [ + { "name": "access-control-allow-origin", "value": "http://localhost:5173" }, + { "name": "content-type", "value": "application/x-ndjson" }, + { "name": "date", "value": "Wed, 25 Oct 2023 00:27:10 GMT" }, + { "name": "vary", "value": "Origin" } + ], + "content": { + "size": -1, + "mimeType": "application/x-ndjson", + "text": "{"choices":[{"index":0,"delta":{"content":"","role":"assistant","context":{"data_points":["support.md:  [How to Search and Book Rentals](#how-to-search-and-book-rentals) 2. [How to Cancel a Confirmed Booking](#how-to-cancel-a-confirmed-booking) 3. [How to Contact Customer Support](#how-to-contact-customer-support) 4. [How to Report a Problem with a Listing](#how-to-report-a-problem-with-a-listing) 5. [How to Report a Problem with a Guest or Host](#how-to-report-a-problem-with-a-guest-or-host) 6. [How to Report a Safety Issue](#how-to-report-a-safety-issue) 7. [How to Report a Payment or Refund Issue](#how-to-report-a-payment-or-refund-issue) 8. [Guest or Host Rule Violations](#guest-or-host-rule-violations) 9. [Compensation for Damage](#compensation-for-damage) 10. [Payment Error Resolution](#payment-error-resolution) --- ## 1. How to Search and Book Rentals Contoso Real Estate offers a wide range of rental properties for your stay. Here's how to search and book rentals: 1. **Search for Rentals:**    - Enter your destination, check-in and check-out dates, and the number of guests.    - Apply filters such as price range, property type, and amenities to narrow down your options.","terms-of-service.md:  We aim to create a trusted community where hosts can share their spaces and guests can enjoy memorable experiences. ## 2. How to Search and Book Rentals ### 2.1 Creating an Account To search for and book rentals, users must create a Contoso Real Estate account. You must provide accurate and complete information during the registration process. Users are responsible for maintaining the confidentiality of their account credentials. ### 2.2 Browsing and Searching - Users can browse available rentals without an account. - Use our search filters to narrow down rental options based on your preferences. - Click on a rental listing to view details, including pricing, availability, and host information. ### 2.3 Booking a Rental - To book a rental, click the \"Book Now\" button on the listing page. - Review the booking details, including dates and pricing, before confirming the reservation. - Payment is processed securely through our platform. Guests will receive a booking confirmation email. ### 2.4 Guest Verification - Guests may be required to complete identity verification steps.","support.md:  **Search for Rentals:**    - Enter your destination, check-in and check-out dates, and the number of guests.    - Apply filters such as price range, property type, and amenities to narrow down your options.    - Browse through the listings to find the perfect place for your stay. 2. **View Listing Details:**    - Click on a listing to view detailed information, including photos, property description, reviews, and host information. 3. **Make a Booking:**    - Click the \"Book Now\" button on the listing page.    - Review the booking details, including the total cost and house rules.    - Confirm your booking by providing payment information.    - Once the host accepts your booking, you'll receive a confirmation. 4. **Payment:**    - Contoso Real Estate handles the payment process securely. You'll only be charged once your booking is confirmed. 5. **Communication:**    - You can communicate with the host through our messaging system for any questions or special requests. --- ## 2. How to Cancel a Confirmed Booking Life happens, and sometimes you need to cancel a confirmed booking. Here's how to do it: 1."],"thoughts":"Search query:<br>Search and book rentals process<br><br>Conversations:<br>system: Below is a history of the conversation so far, and a new question asked by the user that needs to be answered by searching in a knowledge base about terms of service, privacy policy, and questions about support requests.<br>Generate a search query based on the conversation and the new question.<br>Do not include cited source filenames and document names e.g info.txt or doc.pdf in the search query terms.<br>Do not include any text inside [] or <<>> in the search query terms.<br>Do not include any special characters like '+'.<br>If the question is not in English, translate the question to English before generating the search query.<br>If you cannot generate a search query, return just the number 0.<br><br><br>user: What happens if a payment error occurs?<br><br>assistant: Show support for payment errors<br><br>user: can I get refunded if cannot travel?<br><br>assistant: Refund policy<br><br>user: Generate search query for: How to search and book rentals?<br><br>system: Assistant helps the Consto Real Estate company customers with support questions regarding terms of service, privacy policy, and questions about support requests. Be brief in your answers.<br>Answer ONLY with the facts listed in the list of sources below. If there isn't enough information below, say you don't know. Do not generate answers that don't use the sources below. If asking a clarifying question to the user would help, ask the question.<br>For tabular information return it as an html table. Do not return markdown format. If the question is not in English, answer in the language used in the question.<br>Each source has a name followed by colon and the actual information, always include the source name for each fact you use in the response. Use square brackets to reference the source, for example: [info1.txt]. Don't combine sources, list each source separately, for example: [info1.txt][info2.pdf].<br>Generate 3 very brief follow-up questions that the user would likely ask next.<br>Enclose the follow-up questions in double angle brackets. Example:<br><<Am I allowed to invite friends for a party?>><br><<How can I ask for a refund?>><br><<What If I break something?>><br><br>Do no repeat questions that have already been asked.<br>Make sure the last question ends with \">>\".<br><br><br><br>user: How to search and book rentals?<br><br>Sources:<br>support.md:  [How to Search and Book Rentals](#how-to-search-and-book-rentals) 2. [How to Cancel a Confirmed Booking](#how-to-cancel-a-confirmed-booking) 3. [How to Contact Customer Support](#how-to-contact-customer-support) 4. [How to Report a Problem with a Listing](#how-to-report-a-problem-with-a-listing) 5. [How to Report a Problem with a Guest or Host](#how-to-report-a-problem-with-a-guest-or-host) 6. [How to Report a Safety Issue](#how-to-report-a-safety-issue) 7. [How to Report a Payment or Refund Issue](#how-to-report-a-payment-or-refund-issue) 8. [Guest or Host Rule Violations](#guest-or-host-rule-violations) 9. [Compensation for Damage](#compensation-for-damage) 10. [Payment Error Resolution](#payment-error-resolution) --- ## 1. How to Search and Book Rentals Contoso Real Estate offers a wide range of rental properties for your stay. Here's how to search and book rentals: 1. **Search for Rentals:**    - Enter your destination, check-in and check-out dates, and the number of guests.    - Apply filters such as price range, property type, and amenities to narrow down your options.<br>terms-of-service.md:  We aim to create a trusted community where hosts can share their spaces and guests can enjoy memorable experiences. ## 2. How to Search and Book Rentals ### 2.1 Creating an Account To search for and book rentals, users must create a Contoso Real Estate account. You must provide accurate and complete information during the registration process. Users are responsible for maintaining the confidentiality of their account credentials. ### 2.2 Browsing and Searching - Users can browse available rentals without an account. - Use our search filters to narrow down rental options based on your preferences. - Click on a rental listing to view details, including pricing, availability, and host information. ### 2.3 Booking a Rental - To book a rental, click the \"Book Now\" button on the listing page. - Review the booking details, including dates and pricing, before confirming the reservation. - Payment is processed securely through our platform. Guests will receive a booking confirmation email. ### 2.4 Guest Verification - Guests may be required to complete identity verification steps.<br>support.md:  **Search for Rentals:**    - Enter your destination, check-in and check-out dates, and the number of guests.    - Apply filters such as price range, property type, and amenities to narrow down your options.    - Browse through the listings to find the perfect place for your stay. 2. **View Listing Details:**    - Click on a listing to view detailed information, including photos, property description, reviews, and host information. 3. **Make a Booking:**    - Click the \"Book Now\" button on the listing page.    - Review the booking details, including the total cost and house rules.    - Confirm your booking by providing payment information.    - Once the host accepts your booking, you'll receive a confirmation. 4. **Payment:**    - Contoso Real Estate handles the payment process securely. You'll only be charged once your booking is confirmed. 5. **Communication:**    - You can communicate with the host through our messaging system for any questions or special requests. --- ## 2. How to Cancel a Confirmed Booking Life happens, and sometimes you need to cancel a confirmed booking. Here's how to do it: 1."}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":"To","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" search","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" and","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" book","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" rentals","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" with","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" Cont","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":"oso","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" Real","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" Estate","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":",","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" follow","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" these","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" steps","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":":\n","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":"1","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":".","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" Search","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" for","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" Rentals","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":":\n","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":"  ","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" -","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" Enter","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" your","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" destination","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":",","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" check","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":"-in","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" and","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" check","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":"-out","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" dates","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":",","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" and","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" the","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" number","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" of","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" guests","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":".\n","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":"  ","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" -","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" Apply","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" filters","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" such","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" as","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" price","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" range","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":",","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" property","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" type","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":",","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" and","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" amenities","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" to","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" narrow","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" down","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" your","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" options","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":".\n","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":"2","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":".","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" View","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" Listing","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" Details","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":":\n","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":"  ","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" -","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" Click","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" on","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" a","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" listing","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" to","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" view","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" detailed","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" information","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":",","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" including","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" photos","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":",","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" property","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" description","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":",","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" reviews","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":",","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" and","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" host","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" information","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":".\n","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":"3","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":".","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" Make","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" a","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" Booking","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":":\n","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":"  ","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" -","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" Click","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" the","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" \"","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":"Book","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" Now","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":"\"","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" button","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" on","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" the","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" listing","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" page","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":".\n","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":"  ","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" -","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" Review","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" the","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" booking","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" details","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":",","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" including","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" the","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" total","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" cost","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" and","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" house","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" rules","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":".\n","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":"  ","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" -","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" Confirm","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" your","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" booking","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" by","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" providing","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" payment","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" information","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":".\n","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":"  ","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" -","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" Once","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" the","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" host","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" accepts","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" your","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" booking","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":",","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" you","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":"'ll","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" receive","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" a","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" confirmation","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":".\n","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":"4","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":".","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" Payment","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":":\n","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":"  ","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" -","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" Cont","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":"oso","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" Real","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" Estate","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" handles","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" the","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" payment","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" process","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" securely","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":".","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" You","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":"'ll","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" only","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" be","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" charged","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" once","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" your","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" booking","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" is","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" confirmed","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":".\n","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":"5","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":".","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" Communication","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":":\n","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":"  ","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" -","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" You","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" can","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" communicate","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" with","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" the","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" host","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" through","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" our","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" messaging","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" system","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" for","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" any","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" questions","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" or","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" special","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" requests","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":".","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" [","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":"support","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":".md","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":"]\n\n","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":"<<","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":"How","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" can","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" I","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" cancel","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" a","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" confirmed","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" booking","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":"?>","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":">\n","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":"<<","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":"What","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" payment","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" methods","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" are","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" accepted","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":"?>","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":">\n","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":"<<","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":"Can","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" I","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" contact","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" the","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" host","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" before","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":" booking","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":"?","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":">>","role":"assistant","context":{}},"finish_reason":null}],"object":"chat.completion.chunk"}
{"choices":[{"index":0,"delta":{"content":"","role":"assistant","context":{}},"finish_reason":"stop"}],"object":"chat.completion.chunk"}
", + "encoding": "base64" + }, + "headersSize": -1, + "bodySize": -1, + "redirectURL": "" + }, + "cache": {}, + "timings": { "send": -1, "wait": -1, "receive": 4348.057 } + } + ] + } +} diff --git a/tests/e2e/webapp.spec.ts b/tests/e2e/webapp.spec.ts new file mode 100644 index 00000000..0b23b067 --- /dev/null +++ b/tests/e2e/webapp.spec.ts @@ -0,0 +1,173 @@ +import { test, expect } from '@playwright/test'; + +test.describe('default', () => { + test('chat interation', async ({ page }) => { + await page.goto('/'); + + const defaultQuestions = page.getByTestId('default-question'); + + // expect there to be at least 3 default question buttons on page load + await test.step('Get default questions', async () => { + await expect(defaultQuestions).toHaveCount(3); + }); + + const chatInput = page.getByTestId('question-input'); + const firstQuestionButton = defaultQuestions.nth(0); + const firstQuestionText = ((await firstQuestionButton.textContent()) ?? '').replace('Ask now', '').trim(); + + // should not have any text at the start + await test.step('Use default question', async () => { + await expect(chatInput).toHaveValue(''); + + await firstQuestionButton.click(); + await expect(chatInput).toHaveValue(firstQuestionText); + }); + + const userMessage = page.locator('.chat__txt.user-message'); + + // Set to replay the response for a local route (will not be used for the official) + await page.routeFromHAR('./tests/e2e/hars/default-chat-response-stream.har', { + url: '/chat', + update: false, + }); + + const showThoughtProcess = page.getByTestId('chat-show-thought-process'); + await test.step('Get answer', async () => { + await expect(showThoughtProcess).not.toBeVisible(); + + await page.getByTestId('submit-question-button').click(); + + // wait for the thought process button to be enabled. + await expect(showThoughtProcess).toBeEnabled({ timeout: 30_000 }); + + await expect(userMessage).toHaveCount(1); + await expect(userMessage.nth(0)).toHaveText(firstQuestionText); + + await expect(defaultQuestions).toHaveCount(0); + }); + + // make sure the response is formatted as list items + await test.step('response formatting', async () => { + await expect(page.locator('.items__listItem--step')).not.toHaveCount(0); + }); + + await test.step('Reset chat', async () => { + await page.getByTestId('chat-reset-button').click(); + await expect(userMessage).toHaveCount(0); + await expect(defaultQuestions).toHaveCount(3); + }); + }); + + test('waiting for response', async ({ page }) => { + await page.goto('/'); + await page.getByTestId('default-question').nth(0).click(); + + await page.route('/chat', (route) => + route.fulfill({ + status: -1, + }), + ); + + await expect(page.locator('.loading-skeleton')).not.toBeVisible(); + await page.getByTestId('submit-question-button').click(); + await expect(page.locator('.loading-skeleton')).toBeVisible(); + await expect(page.getByTestId('question-input')).not.toBeEnabled(); + }); + + test('show error on failure', async ({ page }) => { + await page.goto('/'); + await page.getByTestId('default-question').nth(0).click(); + + await page.route('/chat', (route) => route.abort()); + await page.route('**/chat', (route) => route.abort()); + + await page.getByTestId('submit-question-button').click(); + await expect(page.locator('.chat__txt.error')).toBeVisible(); + }); +}); + +test.describe('generate answer', () => { + test.beforeEach(async ({ page }) => { + await page.goto('/'); + await page.getByTestId('default-question').nth(0).click(); + + await page.routeFromHAR('./tests/e2e/hars/default-chat-response-stream.har', { + url: '/chat', + update: false, + }); + + await page.getByTestId('submit-question-button').click(); + // wait for the thought process button to be enabled. + await expect(page.getByTestId('chat-show-thought-process')).toBeEnabled({ timeout: 30_000 }); + }); + + test('show thought process', async ({ page }) => { + const showThoughtProcess = page.getByTestId('chat-show-thought-process'); + const thoughtProcessAside = page.getByTestId('aside-thought-process'); + + await test.step('show/hide aside', async () => { + await expect(thoughtProcessAside).not.toBeVisible(); + await showThoughtProcess.click(); + await expect(thoughtProcessAside).toBeVisible(); + + await page.getByTestId('chat-hide-thought-process').click(); + await expect(thoughtProcessAside).not.toBeVisible(); + }); + + await test.step('Reset chat', async () => { + await showThoughtProcess.click(); + await expect(thoughtProcessAside).toBeVisible(); + await page.getByTestId('chat-reset-button').click(); + await expect(thoughtProcessAside).not.toBeVisible(); + }); + }); + + test('citation', async ({ page }) => { + const citations = page.getByTestId('citation'); + await expect(citations).toHaveCount(1); + + await expect(citations.nth(0)).toBeEnabled(); + await expect(citations.nth(0)).toContainText('support.md'); + expect(await citations.nth(0).getAttribute('href')).toContain('/content/support.md'); + }); +}); + +test.describe('developer settings', () => { + test('default settings', async ({ page }) => { + await page.goto('/'); + await page.getByTestId('button__developer-settings').click(); + + await expect(page.getByLabel('Override prompt template')).toBeVisible(); + await expect(page.getByLabel('Retrieve this many search results:')).toBeVisible(); + await expect(page.locator('label').filter({ hasText: 'Use semantic ranker' }).locator('i')).toBeChecked(); + await expect(page.locator('label').filter({ hasText: 'contextual summaries' }).locator('i')).not.toBeChecked(); + await expect(page.locator('label').filter({ hasText: 'follow-up questions' }).locator('i')).toBeChecked(); + await expect(page.locator('label').filter({ hasText: 'Stream chat' }).locator('i')).toBeChecked(); + await expect(page.getByLabel('Retrieval mode')).toContainText('Vectors + Text (Hybrid)'); + }); + + test('handle no stream parsing', async ({ page }) => { + await page.goto('/'); + await page.getByTestId('default-question').nth(0).click(); + + await page.routeFromHAR('./tests/e2e/hars/default-chat-response-nostream.har', { + url: '/chat', + update: false, + }); + + await page.getByTestId('button__developer-settings').click(); + const streamSetting = page.locator('label').filter({ hasText: 'Stream chat' }).locator('i'); + + await expect(streamSetting).toBeChecked(); + await streamSetting.click(); + await expect(streamSetting).not.toBeChecked(); + + await page.locator('button').filter({ hasText: 'Close' }).click(); + + await page.getByTestId('submit-question-button').click(); + // wait for the thought process button to be enabled. + await expect(page.getByTestId('chat-show-thought-process')).toBeEnabled({ timeout: 30_000 }); + + await expect(page.locator('.items__listItem--step')).not.toHaveCount(0); + }); +});