diff --git a/.github/workflows/generate-lighthouse-diff.yml b/.github/workflows/generate-lighthouse-diff.yml new file mode 100644 index 0000000000..72a6bee774 --- /dev/null +++ b/.github/workflows/generate-lighthouse-diff.yml @@ -0,0 +1,86 @@ +name: Run Lighthouse Diff + +on: + pull_request: + branches: ["master"] + types: + - opened + - reopened + - synchronize + # we skip ci for draft PRs + # https://github.com/reviewdog/action-eslint/issues/29#issuecomment-985939887 + - ready_for_review + workflow_dispatch: + merge_group: + +concurrency: + group: ${{ github.workflow }}-${{ github.event.pull_request.number || github.ref }} + cancel-in-progress: true + +jobs: + generate_lighthouse_diff: + name: "Generate lighthouse diff" + runs-on: ubuntu-latest + steps: + - name: Checkout repository + uses: actions/checkout@v4 + with: + token: ${{ secrets.GITHUB_TOKEN }} + + - name: Setup Node.js + uses: actions/setup-node@v3 + with: + node-version: '20' + cache: 'yarn' + + - name: Install node_modules + uses: OffchainLabs/actions/node-modules/install@main + + - name: Build scripts + run: yarn workspace scripts build + + - name: Build the bridge + run: yarn build + env: + NEXT_PUBLIC_INFURA_KEY: ${{ secrets.NEXT_PUBLIC_INFURA_KEY }} + NEXT_PUBLIC_INFURA_KEY_ETHEREUM: https://mainnet.infura.io/v3/${{ secrets.NEXT_PUBLIC_INFURA_KEY }} + NEXT_PUBLIC_INFURA_KEY_SEPOLIA: https://sepolia.infura.io/v3/${{ secrets.NEXT_PUBLIC_INFURA_KEY }} + NEXT_PUBLIC_INFURA_KEY_ARBITRUM_ONE: https://arbitrum-mainnet.infura.io/v3/${{ secrets.NEXT_PUBLIC_INFURA_KEY }} + NEXT_PUBLIC_INFURA_KEY_BASE: https://base-mainnet.infura.io/v3/${{ secrets.NEXT_PUBLIC_INFURA_KEY }} + NEXT_PUBLIC_INFURA_KEY_ARBITRUM_SEPOLIA: https://arbitrum-sepolia.infura.io/v3/${{ secrets.NEXT_PUBLIC_INFURA_KEY }} + NEXT_PUBLIC_INFURA_KEY_BASE_SEPOLIA: https://base-sepolia.infura.io/v3/${{ secrets.NEXT_PUBLIC_INFURA_KEY }} + NEXT_PUBLIC_WALLETCONNECT_PROJECT_ID: ${{ secrets.NEXT_PUBLIC_WALLETCONNECT_PROJECT_ID }} + THE_GRAPH_NETWORK_API_KEY: ${{ secrets.THE_GRAPH_NETWORK_API_KEY }} + + - name: Start server and wait + run: yarn start & npx wait-on http://localhost:3000 + + - name: Install linux dependencies + run: | + sudo apt-get update + sudo apt-get install --no-install-recommends -y \ + fluxbox \ + xvfb + + - name: Run xvfb and fluxbox + run: | + Xvfb :0 -screen 0 1366x768x24 -listen tcp -ac & + fluxbox & + env: + DISPLAY: :0.0 + + - name: Run generateLighthouseDiff script + run: yarn workspace scripts generate-lighthouse-diff + env: + GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} + NEXT_PUBLIC_INFURA_KEY: ${{ secrets.NEXT_PUBLIC_INFURA_KEY }} + DISPLAY: :0.0 + + - name: Upload reports + uses: actions/upload-artifact@v4 + with: + name: lighthouse-reports-${{ github.sha }} + path: | + ./lhreport.html + if-no-files-found: 'ignore' + continue-on-error: true \ No newline at end of file diff --git a/packages/arb-token-bridge-ui/package.json b/packages/arb-token-bridge-ui/package.json index 98a4ee6d54..b07ca300d8 100644 --- a/packages/arb-token-bridge-ui/package.json +++ b/packages/arb-token-bridge-ui/package.json @@ -64,6 +64,7 @@ "start": "next start", "test": "vitest --config vitest.config.ts --watch", "test:ci": "vitest --config vitest.config.ts --run", + "test:perf": "ts-node ./tests/perf/switchNetwork.mts", "lint": "tsc && eslint", "lint:fix": "tsc && eslint --quiet --fix", "prettier:format": "prettier --config-precedence file-override --write \"src/**/*.{tsx,ts,scss,md,json}\"", diff --git a/packages/arb-token-bridge-ui/src/components/TransferPanel/TokenRow.tsx b/packages/arb-token-bridge-ui/src/components/TransferPanel/TokenRow.tsx index d0a5cd9358..91bb39e65d 100644 --- a/packages/arb-token-bridge-ui/src/components/TransferPanel/TokenRow.tsx +++ b/packages/arb-token-bridge-ui/src/components/TransferPanel/TokenRow.tsx @@ -341,6 +341,7 @@ export function TokenRow({ onClick={() => onTokenSelected(token)} style={{ ...style, minHeight: '84px' }} disabled={!tokenIsBridgeable} + aria-label={`Select ${token?.symbol}`} className={twMerge( 'flex w-full flex-row items-center justify-between px-4 py-3 transition duration-200 hover:bg-white/10', tokenIsBridgeable diff --git a/packages/arb-token-bridge-ui/src/components/TransferPanel/TokenSearch.tsx b/packages/arb-token-bridge-ui/src/components/TransferPanel/TokenSearch.tsx index da06258931..c70253c962 100644 --- a/packages/arb-token-bridge-ui/src/components/TransferPanel/TokenSearch.tsx +++ b/packages/arb-token-bridge-ui/src/components/TransferPanel/TokenSearch.tsx @@ -542,7 +542,6 @@ function TokensPanel({ errorMessage={errorMessage} onSubmit={addNewToken} SearchInputButton={AddButton} - dataCy="tokenSearchList" isDialog={true} > diff --git a/packages/arb-token-bridge-ui/src/components/common/NetworkSelectionContainer.tsx b/packages/arb-token-bridge-ui/src/components/common/NetworkSelectionContainer.tsx index 4c6b0e7d85..2bab60cb60 100644 --- a/packages/arb-token-bridge-ui/src/components/common/NetworkSelectionContainer.tsx +++ b/packages/arb-token-bridge-ui/src/components/common/NetworkSelectionContainer.tsx @@ -155,6 +155,9 @@ export function NetworkButton({ )} disabled={disabled} onClick={onClick} + aria-label={ + (isSource ? 'From: ' : 'To: ') + getNetworkName(selectedChainId) + } > {isSource ? 'From:' : 'To: '} {getNetworkName(selectedChainId)} diff --git a/packages/arb-token-bridge-ui/tests/support/commands.ts b/packages/arb-token-bridge-ui/tests/support/commands.ts index 3d3801e53c..5378864a18 100644 --- a/packages/arb-token-bridge-ui/tests/support/commands.ts +++ b/packages/arb-token-bridge-ui/tests/support/commands.ts @@ -163,20 +163,13 @@ export function typeAmount2( export function findSourceChainButton( chain: string ): Cypress.Chainable> { - return cy - .findByRole('button', { name: `From: ${chain}` }) - .should('be.visible') + return cy.get(`[aria-label="From: ${chain}"]`).should('be.visible') } export function findDestinationChainButton( chain: string ): Cypress.Chainable> { - return ( - cy - // - .findByRole('button', { name: `To: ${chain}` }) - .should('be.visible') - ) + return cy.get(`[aria-label="To: ${chain}"]`).should('be.visible') } export function findGasFeeSummary( diff --git a/packages/scripts/package.json b/packages/scripts/package.json index 2564914c93..d4b51eda7b 100644 --- a/packages/scripts/package.json +++ b/packages/scripts/package.json @@ -9,7 +9,8 @@ "test": "vitest", "coverage": "vitest run --coverage", "start": "node dist/index.js", - "add-orbit-chain": "node dist/scripts.cjs.js add-orbit-chain" + "add-orbit-chain": "node dist/scripts.cjs.js add-orbit-chain", + "generate-lighthouse-diff": "node dist/scripts.cjs.js generate-lighthouse-report" }, "author": "", "license": "ISC", @@ -22,7 +23,9 @@ "commander": "^12.1.0", "ethers": "^5.7.2", "file-type": "^19.6.0", + "lighthouse": "^12.6.0", "mime-types": "^2.1.35", + "puppeteer": "^24.9.0", "sharp": "0.32.6", "zod": "^3.23.8" }, diff --git a/packages/scripts/src/generateLighthouseReport/compareLighthouseReports.ts b/packages/scripts/src/generateLighthouseReport/compareLighthouseReports.ts new file mode 100644 index 0000000000..8bd0469047 --- /dev/null +++ b/packages/scripts/src/generateLighthouseReport/compareLighthouseReports.ts @@ -0,0 +1,184 @@ +import { + NavigationResult, + SnapshotResult, + TimespanResult, +} from "./parseLighthouseReports"; +import { parseToFixedNumber } from "./parseToFixedNumber"; + +export async function compareLighthouseReports({ + prevReport, + results, +}: { + prevReport: [NavigationResult, TimespanResult, SnapshotResult]; + results: [NavigationResult, TimespanResult, SnapshotResult]; +}): Promise<{ + navigationDiff: NavigationResult; + timespanDiff: TimespanResult; + snapshotDiff: SnapshotResult; +}> { + // Compare Navigation Results + const prevNavigationResult = prevReport[0]; + const navigationResult = results[0]; + const navigationDiff = { + fcp: { + numericValue: parseToFixedNumber( + prevNavigationResult.fcp.numericValue - + navigationResult.fcp.numericValue, + 3 + ), + score: parseToFixedNumber( + prevNavigationResult.fcp.score - navigationResult.fcp.score, + 2 + ), + }, + lcp: { + numericValue: parseToFixedNumber( + prevNavigationResult.lcp.numericValue - + navigationResult.lcp.numericValue, + 3 + ), + score: parseToFixedNumber( + prevNavigationResult.lcp.score - navigationResult.lcp.score, + 2 + ), + }, + tbt: { + numericValue: parseToFixedNumber( + prevNavigationResult.tbt.numericValue - + navigationResult.tbt.numericValue, + 3 + ), + score: parseToFixedNumber( + prevNavigationResult.tbt.score - navigationResult.tbt.score, + 2 + ), + }, + cls: { + numericValue: parseToFixedNumber( + prevNavigationResult.cls.numericValue - + navigationResult.cls.numericValue, + 3 + ), + score: parseToFixedNumber( + prevNavigationResult.cls.score - navigationResult.cls.score, + 2 + ), + }, + speed: { + numericValue: parseToFixedNumber( + prevNavigationResult.speed.numericValue - + navigationResult.speed.numericValue, + 3 + ), + score: parseToFixedNumber( + prevNavigationResult.speed.score - navigationResult.speed.score, + 2 + ), + }, + performance: parseToFixedNumber( + prevNavigationResult.performance - navigationResult.performance, + 2 + ), + accessibility: parseToFixedNumber( + prevNavigationResult.accessibility - navigationResult.accessibility, + 2 + ), + best_practices: parseToFixedNumber( + prevNavigationResult.best_practices - navigationResult.best_practices, + 2 + ), + seo: parseToFixedNumber(prevNavigationResult.seo - navigationResult.seo, 2), + bundle_size: { + numericValue: parseToFixedNumber( + prevNavigationResult.bundle_size.numericValue - + navigationResult.bundle_size.numericValue, + 3 + ), + score: parseToFixedNumber( + prevNavigationResult.bundle_size.score - + navigationResult.bundle_size.score, + 2 + ), + }, + } satisfies NavigationResult; + + // Compare Timespan Results + const prevTimespanResult = prevReport[1]; + const timespanResult = results[1]; + const timespanDiff = { + tbt: { + numericValue: parseToFixedNumber( + prevTimespanResult.tbt.numericValue - timespanResult.tbt.numericValue, + 3 + ), + score: parseToFixedNumber( + prevTimespanResult.tbt.score - timespanResult.tbt.score, + 2 + ), + }, + cls: { + numericValue: parseToFixedNumber( + prevTimespanResult.cls.numericValue - timespanResult.cls.numericValue, + 3 + ), + score: parseToFixedNumber( + prevTimespanResult.cls.score - timespanResult.cls.score, + 2 + ), + }, + inp: { + numericValue: parseToFixedNumber( + prevTimespanResult.inp.numericValue - timespanResult.inp.numericValue, + 3 + ), + score: parseToFixedNumber( + prevTimespanResult.inp.score - timespanResult.inp.score, + 2 + ), + }, + best_practices: parseToFixedNumber( + prevTimespanResult.best_practices - timespanResult.best_practices, + 2 + ), + longTasks: { + durationMs: parseToFixedNumber( + prevTimespanResult.longTasks.durationMs - + timespanResult.longTasks.durationMs, + 3 + ), + total: parseToFixedNumber( + prevTimespanResult.longTasks.total - timespanResult.longTasks.total, + 2 + ), + }, + performance: parseToFixedNumber( + prevTimespanResult.performance - timespanResult.performance, + 2 + ), + } satisfies TimespanResult; + + // Compare Snapshot Result + const prevSnapshotResult = prevReport[2]; + const snapshotResult = results[2]; + const snapshotDiff = { + performance: parseToFixedNumber( + prevSnapshotResult.performance - snapshotResult.performance, + 2 + ), + accessibility: parseToFixedNumber( + prevSnapshotResult.accessibility - snapshotResult.accessibility, + 2 + ), + best_practices: parseToFixedNumber( + prevSnapshotResult.best_practices - snapshotResult.best_practices, + 2 + ), + seo: parseToFixedNumber(prevSnapshotResult.seo - snapshotResult.seo, 2), + } satisfies SnapshotResult; + + return { + navigationDiff, + timespanDiff, + snapshotDiff, + }; +} diff --git a/packages/scripts/src/generateLighthouseReport/executeLighthouse.ts b/packages/scripts/src/generateLighthouseReport/executeLighthouse.ts new file mode 100644 index 0000000000..6e8f315414 --- /dev/null +++ b/packages/scripts/src/generateLighthouseReport/executeLighthouse.ts @@ -0,0 +1,84 @@ +import puppeteer from "puppeteer"; +import * as core from "@actions/core"; +import { startFlow, desktopConfig } from "lighthouse"; +import { writeFileSync } from "fs"; +import { join, resolve } from "path"; + +const workspaceRoot = resolve(process.cwd(), "../.."); +export async function executeLighthouseFlow(chromePath?: string) { + core.startGroup("Lighthouse execution"); + // Setup the browser and Lighthouse. + const browser = await puppeteer.launch({ + headless: false, + args: ["--no-sandbox"], + dumpio: true, + ...(chromePath ? { executablePath: chromePath } : {}), + }); + const page = await browser.newPage(); + await page.setViewport({ width: 1366, height: 768 }); + + const flow = await startFlow(page, { + config: desktopConfig, + }); + + page.on("console", (log) => { + core.info(`[log] ${log.text()}`); + }); + + page.on("pageError", (err) => { + core.error(`[err] ${err}`); + }); + + await flow.navigate( + "http://localhost:3000/?sourceChain=ethereum&destinationChain=arbitrum-one&tab=bridge&txHistory=0" + ); + + await flow.startTimespan(); + + // Type amount + const input = await page.waitForSelector('[aria-label="Amount input"]'); + await input?.type("2"); + await page.waitForSelector('[aria-label="Route arbitrum"]'); + + // Switch network + await page.click('[aria-label="Switch Networks"]'); + await page.waitForSelector("[aria-label='From: Arbitrum One']"); + + // Open token selection + await page.click("[aria-label='Select Token']"); + const tokenInput = await page.waitForSelector( + "[placeholder='Search by token name, symbol, or address']" + ); + await tokenInput?.type("USDC"); + const usdcButton = await page.waitForSelector('[aria-label="Select USDC.e"]'); + await usdcButton?.click(); + await page.waitForSelector("xpath///button[contains(., 'USDC.e')]"); + + // Open chain selection + await page.click("[aria-label='From: Arbitrum One']"); + const chainInput = await page.waitForSelector( + '[placeholder="Search a network name"]' + ); + await chainInput?.type("Xai"); + + const xaiRow = await page.waitForSelector('[aria-label="Switch to Xai"]'); + await xaiRow?.click(); + await page.waitForSelector("[aria-label='From: Xai']"); + + await flow.endTimespan(); + await flow.snapshot(); + + // Get the comprehensive flow report. + const report = await flow.createFlowResult(); + + writeFileSync( + join(workspaceRoot, "./lhreport.html"), + await flow.generateReport() + ); + + // Cleanup. + await browser.close(); + core.endGroup(); + + return report; +} diff --git a/packages/scripts/src/generateLighthouseReport/index.ts b/packages/scripts/src/generateLighthouseReport/index.ts new file mode 100644 index 0000000000..0630ff867a --- /dev/null +++ b/packages/scripts/src/generateLighthouseReport/index.ts @@ -0,0 +1,60 @@ +import * as core from "@actions/core"; +import { executeLighthouseFlow } from "./executeLighthouse"; +import { + NavigationResult, + parseLighthouseReports, + SnapshotResult, + TimespanResult, +} from "./parseLighthouseReports"; +import { join, resolve } from "path"; +import { config } from "../../../../package.json"; +import { postComment } from "./postComment"; +import { compareLighthouseReports } from "./compareLighthouseReports"; +import lighthouseBaseline from "./lighthouseBaseline.json"; + +const workspaceRoot = resolve(process.cwd(), "../.."); +// "node_modules/.cache/synpress/chrome/linux-128.0.6613.137/chrome-linux64/chrome" +const chromePath = join( + workspaceRoot, + config.chromePath, + `/chrome/linux-${config.chromeVersion}`, + "chrome-linux64/chrome" +); +export async function generateLighthouseReport() { + try { + // Reports need to be run sequentially + const report1 = await executeLighthouseFlow(chromePath); + const report2 = await executeLighthouseFlow(chromePath); + const report3 = await executeLighthouseFlow(chromePath); + + core.startGroup("Parse lighthouse report"); + const [parsedNavigationReport, parsedTimespanReport, parsedSnapshotReport] = + parseLighthouseReports([report1, report2, report3]); + core.endGroup(); + + core.setOutput("parsedNavigationReport", parsedNavigationReport); + core.startGroup("Compare lighthouse results"); + const diff = await compareLighthouseReports({ + prevReport: lighthouseBaseline as [ + NavigationResult, + TimespanResult, + SnapshotResult + ], + results: [ + parsedNavigationReport, + parsedTimespanReport, + parsedSnapshotReport, + ], + }); + core.endGroup(); + + await postComment({ + parsedNavigationReport, + parsedTimespanReport, + parsedSnapshotReport, + ...diff, + }); + } catch (error) { + console.log(error); + } +} diff --git a/packages/scripts/src/generateLighthouseReport/lighthouseBaseline.json b/packages/scripts/src/generateLighthouseReport/lighthouseBaseline.json new file mode 100755 index 0000000000..8342ff942c --- /dev/null +++ b/packages/scripts/src/generateLighthouseReport/lighthouseBaseline.json @@ -0,0 +1,23 @@ +[ + { + "fcp": { "numericValue": 729.705, "score": 96.67 }, + "lcp": { "numericValue": 2824.252, "score": 38 }, + "tbt": { "numericValue": 124.167, "score": 93.33 }, + "cls": { "numericValue": 0, "score": 100 }, + "speed": { "numericValue": 1274.504, "score": 90.33 }, + "performance": 81.33, + "accessibility": 90, + "best_practices": 100, + "seo": 100, + "bundle_size": { "numericValue": 2754742.333, "score": 50 } + }, + { + "performance": 49.67, + "tbt": { "numericValue": 931.928, "score": 7.33 }, + "cls": { "numericValue": 0.005, "score": 100 }, + "inp": { "numericValue": 189.333, "score": 91 }, + "best_practices": 95, + "longTasks": { "total": 16, "durationMs": 1748.595 } + }, + { "performance": 0, "best_practices": 100, "accessibility": 91, "seo": 100 } +] diff --git a/packages/scripts/src/generateLighthouseReport/parseLighthouseReports.ts b/packages/scripts/src/generateLighthouseReport/parseLighthouseReports.ts new file mode 100644 index 0000000000..0eac342504 --- /dev/null +++ b/packages/scripts/src/generateLighthouseReport/parseLighthouseReports.ts @@ -0,0 +1,319 @@ +import { FlowResult } from "lighthouse"; +import { parseToFixedNumber } from "./parseToFixedNumber"; +import { log } from "console"; + +type Metric = { + /** Number from 0 to 1 */ + score: number; + numericValue: number; +}; +export type NavigationResult = { + fcp: Metric; + lcp: Metric; + tbt: Metric; + cls: Metric; + speed: Metric; + bundle_size: Metric; + performance: number; + accessibility: number; + best_practices: number; + seo: number; +}; +export type TimespanResult = { + performance: number; + tbt: Metric; + cls: Metric; + inp: Metric; + best_practices: number; + longTasks: { + total: number; + durationMs: number; + }; +}; +export type SnapshotResult = { + performance: number; + accessibility: number; + best_practices: number; + seo: number; +}; + +function parse(result: FlowResult.Step, metricName: string): Metric { + const metric = result.lhr.audits[metricName]; + + return { + score: metric.score || 0, + numericValue: metric.numericValue || 0, + }; +} + +function generateMetric({ + key, + length, + report, +}: { + key: TKey; + length: number; + report: { + [key in TKey]: { + numericValue: number; + score: number; + }; + }; +}) { + return { + numericValue: parseToFixedNumber(report[key].numericValue / length, 3), + score: parseToFixedNumber((report[key].score / length) * 100, 2), + }; +} + +export function parseNavigationResults( + navigationReports: FlowResult.Step[] +): NavigationResult { + const mergedReports = navigationReports.reduce( + function parseNavigationResultsReduce(acc, report) { + const fcp = parse(report, "first-contentful-paint"); + const lcp = parse(report, "largest-contentful-paint"); + const tbt = parse(report, "total-blocking-time"); + const cls = parse(report, "cumulative-layout-shift"); + const bundle_size = parse(report, "total-byte-weight"); + const speed = report.lhr.audits["speed-index"]; + + return { + fcp: { + numericValue: acc.fcp.numericValue + fcp.numericValue, + score: acc.fcp.score + fcp.score, + }, + lcp: { + numericValue: acc.lcp.numericValue + lcp.numericValue, + score: acc.lcp.score + lcp.score, + }, + tbt: { + numericValue: acc.tbt.numericValue + tbt.numericValue, + score: acc.tbt.score + tbt.score, + }, + cls: { + numericValue: acc.cls.numericValue + cls.numericValue, + score: acc.cls.score + cls.score, + }, + speed: { + numericValue: acc.speed.numericValue + (speed.numericValue || 0), + score: acc.speed.score + (speed.score || 0), + }, + bundle_size: { + numericValue: acc.bundle_size.numericValue + bundle_size.numericValue, + score: acc.bundle_size.score + bundle_size.score, + }, + performance: + acc.performance + (report.lhr.categories.performance.score || 0), + accessibility: + acc.accessibility + (report.lhr.categories.accessibility.score || 0), + best_practices: + acc.best_practices + + (report.lhr.categories["best-practices"].score || 0), + seo: acc.seo + (report.lhr.categories.seo.score || 0), + }; + }, + { + fcp: { + numericValue: 0, + score: 0, + }, + lcp: { + numericValue: 0, + score: 0, + }, + tbt: { + numericValue: 0, + score: 0, + }, + cls: { + numericValue: 0, + score: 0, + }, + speed: { + numericValue: 0, + score: 0, + }, + bundle_size: { + numericValue: 0, + score: 0, + }, + performance: 0, + accessibility: 0, + best_practices: 0, + seo: 0, + } satisfies NavigationResult + ); + + const length = navigationReports.length; + return { + fcp: generateMetric({ key: "fcp", length, report: mergedReports }), + lcp: generateMetric({ key: "lcp", length, report: mergedReports }), + tbt: generateMetric({ key: "tbt", length, report: mergedReports }), + cls: generateMetric({ key: "cls", length, report: mergedReports }), + speed: generateMetric({ key: "speed", length, report: mergedReports }), + bundle_size: generateMetric({ + key: "bundle_size", + length, + report: mergedReports, + }), + performance: parseToFixedNumber( + (mergedReports.performance / length) * 100, + 2 + ), + accessibility: parseToFixedNumber( + (mergedReports.accessibility / length) * 100, + 2 + ), + best_practices: parseToFixedNumber( + (mergedReports.best_practices / length) * 100, + 2 + ), + seo: parseToFixedNumber((mergedReports.seo / length) * 100, 2), + }; +} + +export function parseTimespanResults( + timespanReports: FlowResult.Step[] +): TimespanResult { + const mergedReports = timespanReports.reduce( + function parseTimespanResultsReduce(acc, report) { + const tbt = parse(report, "total-blocking-time"); + const cls = parse(report, "cumulative-layout-shift"); + const inp = parse(report, "interaction-to-next-paint"); + const longTasks = ( + report.lhr.audits["long-tasks"].details! as unknown as { + items: { + url: string; + duration: number; + startTime: number; + }[]; + } + ).items; + + return { + performance: + acc.performance + (report.lhr.categories.performance.score || 0), + tbt: { + numericValue: acc.tbt.numericValue + tbt.numericValue, + score: acc.tbt.score + tbt.score, + }, + cls: { + numericValue: acc.cls.numericValue + cls.numericValue, + score: acc.cls.score + cls.score, + }, + inp: { + numericValue: acc.inp.numericValue + inp.numericValue, + score: acc.inp.score + inp.score, + }, + best_practices: + acc.best_practices + + (report.lhr.categories["best-practices"].score || 0), + longTasks: { + total: acc.longTasks.total + longTasks.length, + durationMs: longTasks.reduce( + (sum, task) => sum + task.duration, + acc.longTasks.durationMs + ), + }, + }; + }, + { + performance: 0, + tbt: { + numericValue: 0, + score: 0, + }, + cls: { + numericValue: 0, + score: 0, + }, + inp: { numericValue: 0, score: 0 }, + best_practices: 0, + longTasks: { + total: 0, + durationMs: 0, + }, + } satisfies TimespanResult + ); + + const length = timespanReports.length; + return { + performance: parseToFixedNumber( + (mergedReports.performance / length) * 100, + 2 + ), + tbt: generateMetric({ key: "tbt", length, report: mergedReports }), + cls: generateMetric({ key: "cls", length, report: mergedReports }), + inp: generateMetric({ key: "inp", length, report: mergedReports }), + best_practices: parseToFixedNumber( + (mergedReports.best_practices / length) * 100, + 2 + ), + longTasks: { + total: parseToFixedNumber(mergedReports.longTasks.total / length, 2), + durationMs: parseToFixedNumber( + mergedReports.longTasks.durationMs / length, + 3 + ), + }, + }; +} + +export function parseSnapshotResults( + snapshotReports: FlowResult.Step[] +): SnapshotResult { + const mergedReports = snapshotReports.reduce( + function parseSnapshotResultsReduce(acc, report) { + return { + performance: + acc.performance + (report.lhr.categories.performance.score || 0), + best_practices: + acc.best_practices + + (report.lhr.categories["best-practices"].score || 0), + accessibility: + acc.accessibility + (report.lhr.categories.accessibility.score || 0), + seo: acc.seo + (report.lhr.categories.seo.score || 0), + }; + }, + { + performance: 0, + best_practices: 0, + accessibility: 0, + seo: 0, + } satisfies SnapshotResult + ); + + const length = snapshotReports.length; + return { + performance: parseToFixedNumber( + (mergedReports.performance / length) * 100, + 2 + ), + best_practices: parseToFixedNumber( + (mergedReports.best_practices / length) * 100, + 2 + ), + accessibility: parseToFixedNumber( + (mergedReports.accessibility / length) * 100, + 2 + ), + seo: parseToFixedNumber((mergedReports.seo / length) * 100, 2), + }; +} + +export function parseLighthouseReports( + reports: FlowResult[] +): [NavigationResult, TimespanResult, SnapshotResult] { + const navigationReports = reports.map((report) => report.steps[0]); + const timespanReports = reports.map((report) => report.steps[1]); + const snapshotReports = reports.map((report) => report.steps[2]); + + console.log(parseNavigationResults(navigationReports)); + + return [ + parseNavigationResults(navigationReports), + parseTimespanResults(timespanReports), + parseSnapshotResults(snapshotReports), + ] as const; +} diff --git a/packages/scripts/src/generateLighthouseReport/parseToFixedNumber.ts b/packages/scripts/src/generateLighthouseReport/parseToFixedNumber.ts new file mode 100644 index 0000000000..c610b0a1a5 --- /dev/null +++ b/packages/scripts/src/generateLighthouseReport/parseToFixedNumber.ts @@ -0,0 +1,3 @@ +export function parseToFixedNumber(num: number, fractionDigits: number) { + return Number(num.toFixed(fractionDigits)); +} diff --git a/packages/scripts/src/generateLighthouseReport/postComment.ts b/packages/scripts/src/generateLighthouseReport/postComment.ts new file mode 100644 index 0000000000..2b76ff34cf --- /dev/null +++ b/packages/scripts/src/generateLighthouseReport/postComment.ts @@ -0,0 +1,79 @@ +import * as core from "@actions/core"; +import { getOctokit, context } from "@actions/github"; +import { + NavigationResult, + SnapshotResult, + TimespanResult, +} from "./parseLighthouseReports"; + +export async function postComment({ + parsedNavigationReport, + parsedTimespanReport, + parsedSnapshotReport, + navigationDiff, + timespanDiff, + snapshotDiff, +}: { + parsedNavigationReport: NavigationResult; + parsedTimespanReport: TimespanResult; + parsedSnapshotReport: SnapshotResult; + navigationDiff: NavigationResult; + timespanDiff: TimespanResult; + snapshotDiff: SnapshotResult; +}) { + core.startGroup("Post comment"); + const github = getOctokit(process.env.GITHUB_TOKEN || ""); + + const { data: comment } = await github.rest.issues.createComment({ + ...context.repo, + issue_number: context.issue.number, + // Identation needs to be on the same level, otherwise github doesn't format it properly + // prettier-ignore + body: ` +
+Lighthouse results + + + +Navigation: +| Name | Result | Delta | +|----------------------------|---------------------------------|-------| +| Performance | ${parsedNavigationReport.performance} | ${navigationDiff.performance} | +| Accessibility | ${parsedNavigationReport.accessibility} | ${navigationDiff.accessibility} | +| Best Practices | ${parsedNavigationReport.best_practices} | ${navigationDiff.best_practices} | +| SEO | ${parsedNavigationReport.seo} | ${navigationDiff.seo} | +| First Contentful Paint | ${parsedNavigationReport.fcp.score} (${parsedNavigationReport.fcp.numericValue} ms) | ${navigationDiff.fcp.score} (${navigationDiff.fcp.numericValue} ms) | +| Largest Contentful Paint | ${parsedNavigationReport.lcp.score} (${parsedNavigationReport.lcp.numericValue} s) | ${navigationDiff.lcp.score} (${navigationDiff.lcp.numericValue} ms) | +| Total Blocking Time | ${parsedNavigationReport.tbt.score} (${parsedNavigationReport.tbt.numericValue} ms) | ${navigationDiff.tbt.score} (${navigationDiff.tbt.numericValue} ms) | +| Cumulative Layout Shift | ${parsedNavigationReport.cls.score} (${parsedNavigationReport.cls.numericValue}) | ${navigationDiff.cls.score} (${navigationDiff.cls.numericValue} ms) | +| Speed Index | ${parsedNavigationReport.speed.score} (${parsedNavigationReport.speed.numericValue}s) | ${navigationDiff.speed.score} (${navigationDiff.speed.numericValue} ms) | +| Bundle Size | ${parsedNavigationReport.bundle_size.score} (${parsedNavigationReport.bundle_size.numericValue} byte) | ${navigationDiff.bundle_size.score} (${navigationDiff.bundle_size.numericValue} byte) | + + +Timespan: +| Name | Result | Delta | +|----------------------------|---------------------------------|-------| +| Performance | ${parsedTimespanReport.performance} |${timespanDiff.performance} | +| Total Blocking Time | ${parsedTimespanReport.tbt.score} (${parsedTimespanReport.tbt.numericValue} ms) | ${timespanDiff.tbt.score} (${timespanDiff.tbt.numericValue} ms) | +| Cumulative Layout Shift | ${parsedTimespanReport.cls.score} (${parsedTimespanReport.cls.numericValue}) | ${timespanDiff.cls.score} (${timespanDiff.cls.numericValue} ms) | +| Interaction to Next Paint | ${parsedTimespanReport.inp.score} (${parsedTimespanReport.inp.numericValue} ms) | ${timespanDiff.inp.score} (${timespanDiff.inp.numericValue} ms) | +| Best practices | ${parsedTimespanReport.best_practices} | ${timespanDiff.best_practices} | +| Long tasks | ${parsedTimespanReport.longTasks.total} (${parsedTimespanReport.longTasks.durationMs} ms) | ${timespanDiff.longTasks.total} (${timespanDiff.longTasks.durationMs} ms) | + +Snapshot: +| Name | Result | Delta | +|----------------------------|---------------------------------|-------| +| Performance | ${parsedSnapshotReport.performance} | ${snapshotDiff.performance} | +| Accessibility | ${parsedSnapshotReport.accessibility} | ${snapshotDiff.accessibility} | +| Best practices | ${parsedSnapshotReport.best_practices} | ${snapshotDiff.best_practices} | +| SEO | ${parsedSnapshotReport.seo} | ${snapshotDiff.seo} | + +
+ `, + }); + + core.info( + `Created comment id '${comment.id}' on issue '${context.issue.number}'.` + ); + core.endGroup(); +} diff --git a/packages/scripts/src/generateLighthouseReport/tests/__mocks__/output.json b/packages/scripts/src/generateLighthouseReport/tests/__mocks__/output.json new file mode 100644 index 0000000000..19b9823e5b --- /dev/null +++ b/packages/scripts/src/generateLighthouseReport/tests/__mocks__/output.json @@ -0,0 +1,26650 @@ +{ + "steps": [ + { + "lhr": { + "lighthouseVersion": "12.6.0", + "requestedUrl": "http://localhost:3000/?sourceChain=ethereum&destinationChain=arbitrum-one&tab=bridge&txHistory=0", + "mainDocumentUrl": "http://localhost:3000/?sourceChain=ethereum&destinationChain=arbitrum-one&tab=bridge&txHistory=0", + "finalDisplayedUrl": "http://localhost:3000/?sourceChain=ethereum&destinationChain=arbitrum-one&tab=bridge&txHistory=0", + "finalUrl": "http://localhost:3000/?sourceChain=ethereum&destinationChain=arbitrum-one&tab=bridge&txHistory=0", + "fetchTime": "2025-05-30T19:04:56.257Z", + "gatherMode": "navigation", + "runWarnings": [], + "userAgent": "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/128.0.0.0 Safari/537.36", + "environment": { + "networkUserAgent": "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/119.0.0.0 Safari/537.36", + "hostUserAgent": "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/128.0.0.0 Safari/537.36", + "benchmarkIndex": 3387.5, + "credits": { + "axe-core": "4.10.3" + } + }, + "audits": { + "is-on-https": { + "id": "is-on-https", + "title": "Uses HTTPS", + "description": "All sites should be protected with HTTPS, even ones that don't handle sensitive data. This includes avoiding [mixed content](https://developers.google.com/web/fundamentals/security/prevent-mixed-content/what-is-mixed-content), where some resources are loaded over HTTP despite the initial request being served over HTTPS. HTTPS prevents intruders from tampering with or passively listening in on the communications between your app and your users, and is a prerequisite for HTTP/2 and many new web platform APIs. [Learn more about HTTPS](https://developer.chrome.com/docs/lighthouse/pwa/is-on-https/).", + "score": 1, + "scoreDisplayMode": "binary", + "details": { + "type": "table", + "headings": [], + "items": [] + } + }, + "redirects-http": { + "id": "redirects-http", + "title": "Redirects HTTP traffic to HTTPS", + "description": "Make sure that you redirect all HTTP traffic to HTTPS in order to enable secure web features for all your users. [Learn more](https://developer.chrome.com/docs/lighthouse/pwa/redirects-http/).", + "score": null, + "scoreDisplayMode": "notApplicable" + }, + "viewport": { + "id": "viewport", + "title": "Has a `` tag with `width` or `initial-scale`", + "description": "A `` not only optimizes your app for mobile screen sizes, but also prevents [a 300 millisecond delay to user input](https://developer.chrome.com/blog/300ms-tap-delay-gone-away/). [Learn more about using the viewport meta tag](https://developer.chrome.com/docs/lighthouse/pwa/viewport/).", + "score": 1, + "scoreDisplayMode": "metricSavings", + "warnings": [], + "metricSavings": { + "INP": 0 + }, + "details": { + "type": "debugdata", + "viewportContent": "width=device-width, initial-scale=1" + }, + "guidanceLevel": 3 + }, + "first-contentful-paint": { + "id": "first-contentful-paint", + "title": "First Contentful Paint", + "description": "First Contentful Paint marks the time at which the first text or image is painted. [Learn more about the First Contentful Paint metric](https://developer.chrome.com/docs/lighthouse/performance/first-contentful-paint/).", + "score": 1, + "scoreDisplayMode": "numeric", + "numericValue": 442.24474999999995, + "numericUnit": "millisecond", + "displayValue": "0.4 s", + "scoringOptions": { + "p10": 934, + "median": 1600 + } + }, + "largest-contentful-paint": { + "id": "largest-contentful-paint", + "title": "Largest Contentful Paint", + "description": "Largest Contentful Paint marks the time at which the largest text or image is painted. [Learn more about the Largest Contentful Paint metric](https://developer.chrome.com/docs/lighthouse/performance/lighthouse-largest-contentful-paint/)", + "score": 0.03, + "scoreDisplayMode": "numeric", + "numericValue": 6386.565749999998, + "numericUnit": "millisecond", + "displayValue": "6.4 s", + "scoringOptions": { + "p10": 1200, + "median": 2400 + } + }, + "first-meaningful-paint": { + "id": "first-meaningful-paint", + "title": "First Meaningful Paint", + "description": "First Meaningful Paint measures when the primary content of a page is visible. [Learn more about the First Meaningful Paint metric](https://developer.chrome.com/docs/lighthouse/performance/first-meaningful-paint/).", + "score": null, + "scoreDisplayMode": "notApplicable" + }, + "speed-index": { + "id": "speed-index", + "title": "Speed Index", + "description": "Speed Index shows how quickly the contents of a page are visibly populated. [Learn more about the Speed Index metric](https://developer.chrome.com/docs/lighthouse/performance/speed-index/).", + "score": 0.01, + "scoreDisplayMode": "numeric", + "numericValue": 5786.937630172584, + "numericUnit": "millisecond", + "displayValue": "5.8 s", + "scoringOptions": { + "p10": 1311, + "median": 2300 + } + }, + "screenshot-thumbnails": { + "id": "screenshot-thumbnails", + "title": "Screenshot Thumbnails", + "description": "This is what the load of your site looked like.", + "score": 1, + "scoreDisplayMode": "informative", + "details": { + "type": "filmstrip", + "scale": 5187, + "items": [ + { + "timing": 648, + "timestamp": 31363305945, + "data": "" + }, + { + "timing": 1297, + "timestamp": 31363954320, + "data": "" + }, + { + "timing": 1945, + "timestamp": 31364602695, + "data": "" + }, + { + "timing": 2594, + "timestamp": 31365251070, + "data": "" + }, + { + "timing": 3242, + "timestamp": 31365899445, + "data": "" + }, + { + "timing": 3890, + "timestamp": 31366547820, + "data": "" + }, + { + "timing": 4539, + "timestamp": 31367196195, + "data": "" + }, + { + "timing": 5187, + "timestamp": 31367844570, + "data": "" + } + ] + } + }, + "final-screenshot": { + "id": "final-screenshot", + "title": "Final Screenshot", + "description": "The last screenshot captured of the pageload.", + "score": 1, + "scoreDisplayMode": "informative", + "details": { + "type": "screenshot", + "timing": 5188, + "timestamp": 31367845236, + "data": "" + } + }, + "total-blocking-time": { + "id": "total-blocking-time", + "title": "Total Blocking Time", + "description": "Sum of all time periods between FCP and Time to Interactive, when task length exceeded 50ms, expressed in milliseconds. [Learn more about the Total Blocking Time metric](https://developer.chrome.com/docs/lighthouse/performance/lighthouse-total-blocking-time/).", + "score": 0, + "scoreDisplayMode": "numeric", + "numericValue": 2630, + "numericUnit": "millisecond", + "displayValue": "2,630 ms", + "scoringOptions": { + "p10": 150, + "median": 350 + } + }, + "max-potential-fid": { + "id": "max-potential-fid", + "title": "Max Potential First Input Delay", + "description": "The maximum potential First Input Delay that your users could experience is the duration of the longest task. [Learn more about the Maximum Potential First Input Delay metric](https://developer.chrome.com/docs/lighthouse/performance/lighthouse-max-potential-fid/).", + "score": 0, + "scoreDisplayMode": "numeric", + "numericValue": 1196, + "numericUnit": "millisecond", + "displayValue": "1,200 ms" + }, + "cumulative-layout-shift": { + "id": "cumulative-layout-shift", + "title": "Cumulative Layout Shift", + "description": "Cumulative Layout Shift measures the movement of visible elements within the viewport. [Learn more about the Cumulative Layout Shift metric](https://web.dev/articles/cls).", + "score": 1, + "scoreDisplayMode": "numeric", + "numericValue": 0, + "numericUnit": "unitless", + "displayValue": "0", + "scoringOptions": { + "p10": 0.1, + "median": 0.25 + }, + "details": { + "type": "debugdata", + "items": [ + { + "cumulativeLayoutShiftMainFrame": 0, + "newEngineResult": { + "cumulativeLayoutShift": 0, + "cumulativeLayoutShiftMainFrame": 0 + }, + "newEngineResultDiffered": false + } + ] + } + }, + "errors-in-console": { + "id": "errors-in-console", + "title": "Browser errors were logged to the console", + "description": "Errors logged to the console indicate unresolved problems. They can come from network request failures and other browser concerns. [Learn more about this errors in console diagnostic audit](https://developer.chrome.com/docs/lighthouse/best-practices/errors-in-console/)", + "score": 0, + "scoreDisplayMode": "binary", + "details": { + "type": "table", + "headings": [ + { + "key": "sourceLocation", + "valueType": "source-location", + "label": "Source" + }, + { + "key": "description", + "valueType": "code", + "label": "Description" + } + ], + "items": [ + { + "source": "console.error", + "description": "[PostHog.js] PostHog was initialized without a token. This likely indicates a misconfiguration. Please check the first argument passed to posthog.init()", + "sourceLocation": { + "type": "source-location", + "url": "webpack-internal:///../../node_modules/next/dist/client/components/react-dev-overlay/internal/helpers/hydration-error-info.js", + "urlProvider": "network", + "line": 70, + "column": 13 + } + }, + { + "source": "network", + "description": "Failed to load resource: net::ERR_CONNECTION_REFUSED", + "sourceLocation": { + "type": "source-location", + "url": "http://127.0.0.1:8547/", + "urlProvider": "network", + "line": 0, + "column": 0 + } + }, + { + "source": "network", + "description": "Failed to load resource: net::ERR_CONNECTION_REFUSED", + "sourceLocation": { + "type": "source-location", + "url": "http://127.0.0.1:8547/", + "urlProvider": "network", + "line": 0, + "column": 0 + } + }, + { + "source": "network", + "description": "Failed to load resource: net::ERR_CONNECTION_REFUSED", + "sourceLocation": { + "type": "source-location", + "url": "http://127.0.0.1:8547/", + "urlProvider": "network", + "line": 0, + "column": 0 + } + }, + { + "source": "network", + "description": "Failed to load resource: net::ERR_CONNECTION_REFUSED", + "sourceLocation": { + "type": "source-location", + "url": "http://127.0.0.1:8547/", + "urlProvider": "network", + "line": 0, + "column": 0 + } + }, + { + "source": "console.error", + "description": "OVERMIND DEVTOOLS: Not able to connect. You are trying to connect to \"localhost:3031\", but there was no devtool there. Try the following:\n \n - Make sure you are running the latest version of the devtools, using \"npx overmind-devtools@latest\" or install latest extension for VSCode\n - Close the current tab and open a new one\n - Make sure the correct port is configured in the devtools\n ", + "sourceLocation": { + "type": "source-location", + "url": "webpack-internal:///../../node_modules/next/dist/client/components/react-dev-overlay/internal/helpers/hydration-error-info.js", + "urlProvider": "network", + "line": 70, + "column": 13 + } + }, + { + "source": "network", + "description": "WebSocket connection to 'ws://localhost:3031/?name=Bridge%20to%20Arbitrum%20One' failed: Error in connection establishment: net::ERR_CONNECTION_REFUSED", + "sourceLocation": { + "type": "source-location", + "url": "webpack-internal:///../../node_modules/overmind/es/Devtools.js", + "urlProvider": "network", + "line": 20, + "column": 22 + } + } + ] + } + }, + "server-response-time": { + "id": "server-response-time", + "title": "Initial server response time was short", + "description": "Keep the server response time for the main document short because all other requests depend on it. [Learn more about the Time to First Byte metric](https://developer.chrome.com/docs/lighthouse/performance/time-to-first-byte/).", + "score": 1, + "scoreDisplayMode": "metricSavings", + "numericValue": 24.311, + "numericUnit": "millisecond", + "displayValue": "Root document took 20 ms", + "metricSavings": { + "FCP": 0, + "LCP": 0 + }, + "details": { + "type": "opportunity", + "headings": [ + { + "key": "url", + "valueType": "url", + "label": "URL" + }, + { + "key": "responseTime", + "valueType": "timespanMs", + "label": "Time Spent" + } + ], + "items": [ + { + "url": "http://localhost:3000/?sourceChain=ethereum&destinationChain=arbitrum-one&tab=bridge&txHistory=0", + "responseTime": 24.311 + } + ], + "overallSavingsMs": 0 + }, + "guidanceLevel": 1 + }, + "interactive": { + "id": "interactive", + "title": "Time to Interactive", + "description": "Time to Interactive is the amount of time it takes for the page to become fully interactive. [Learn more about the Time to Interactive metric](https://developer.chrome.com/docs/lighthouse/performance/interactive/).", + "score": 0, + "scoreDisplayMode": "numeric", + "numericValue": 15824.922262500015, + "numericUnit": "millisecond", + "displayValue": "15.8 s" + }, + "user-timings": { + "id": "user-timings", + "title": "User Timing marks and measures", + "description": "Consider instrumenting your app with the User Timing API to measure your app's real-world performance during key user experiences. [Learn more about User Timing marks](https://developer.chrome.com/docs/lighthouse/performance/user-timings/).", + "score": 1, + "scoreDisplayMode": "informative", + "displayValue": "4 user timings", + "details": { + "type": "table", + "headings": [ + { + "key": "name", + "valueType": "text", + "label": "Name" + }, + { + "key": "timingType", + "valueType": "text", + "label": "Type" + }, + { + "key": "startTime", + "valueType": "ms", + "granularity": 0.01, + "label": "Start Time" + }, + { + "key": "duration", + "valueType": "ms", + "granularity": 0.01, + "label": "Duration" + } + ], + "items": [ + { + "name": "Next.js-before-hydration", + "startTime": 0, + "duration": 1768.372, + "timingType": "Measure" + }, + { + "name": "Next.js-hydration", + "startTime": 1768.372, + "duration": 32.067, + "timingType": "Measure" + }, + { + "name": "beforeRender", + "startTime": 1768.372, + "timingType": "Mark" + }, + { + "name": "afterHydrate", + "startTime": 1800.439, + "timingType": "Mark" + } + ] + }, + "guidanceLevel": 2 + }, + "critical-request-chains": { + "id": "critical-request-chains", + "title": "Avoid chaining critical requests", + "description": "The Critical Request Chains below show you what resources are loaded with a high priority. Consider reducing the length of chains, reducing the download size of resources, or deferring the download of unnecessary resources to improve page load. [Learn how to avoid chaining critical requests](https://developer.chrome.com/docs/lighthouse/performance/critical-request-chains/).", + "score": 1, + "scoreDisplayMode": "informative", + "displayValue": "1 chain found", + "details": { + "type": "criticalrequestchain", + "chains": { + "D7DAA8A36A4A8B8F6DA92B5D47DDD0D3": { + "request": { + "url": "http://localhost:3000/?sourceChain=ethereum&destinationChain=arbitrum-one&tab=bridge&txHistory=0", + "startTime": 31362.659238, + "endTime": 31362.684541, + "responseReceivedTime": 31362.683994000003, + "transferSize": 1685 + }, + "children": { + "27474.11": { + "request": { + "url": "http://localhost:3000/_next/static/chunks/react-refresh.js", + "startTime": 31362.690642, + "endTime": 31362.697673, + "responseReceivedTime": 31362.696745999998, + "transferSize": 25433 + } + } + } + } + }, + "longestChain": { + "duration": 38.434999998658895, + "length": 2, + "transferSize": 25433 + } + }, + "guidanceLevel": 1 + }, + "redirects": { + "id": "redirects", + "title": "Avoid multiple page redirects", + "description": "Redirects introduce additional delays before the page can be loaded. [Learn how to avoid page redirects](https://developer.chrome.com/docs/lighthouse/performance/redirects/).", + "score": 1, + "scoreDisplayMode": "metricSavings", + "numericValue": 0, + "numericUnit": "millisecond", + "displayValue": "", + "metricSavings": { + "LCP": 0, + "FCP": 0 + }, + "details": { + "type": "opportunity", + "headings": [], + "items": [], + "overallSavingsMs": 0 + }, + "guidanceLevel": 2 + }, + "image-aspect-ratio": { + "id": "image-aspect-ratio", + "title": "Displays images with correct aspect ratio", + "description": "Image display dimensions should match natural aspect ratio. [Learn more about image aspect ratio](https://developer.chrome.com/docs/lighthouse/best-practices/image-aspect-ratio/).", + "score": 1, + "scoreDisplayMode": "binary", + "details": { + "type": "table", + "headings": [], + "items": [] + } + }, + "image-size-responsive": { + "id": "image-size-responsive", + "title": "Serves images with appropriate resolution", + "description": "Image natural dimensions should be proportional to the display size and the pixel ratio to maximize image clarity. [Learn how to provide responsive images](https://web.dev/articles/serve-responsive-images).", + "score": 1, + "scoreDisplayMode": "binary", + "details": { + "type": "table", + "headings": [], + "items": [] + } + }, + "deprecations": { + "id": "deprecations", + "title": "Avoids deprecated APIs", + "description": "Deprecated APIs will eventually be removed from the browser. [Learn more about deprecated APIs](https://developer.chrome.com/docs/lighthouse/best-practices/deprecations/).", + "score": 1, + "scoreDisplayMode": "binary", + "details": { + "type": "table", + "headings": [], + "items": [] + } + }, + "third-party-cookies": { + "id": "third-party-cookies", + "title": "Avoids third-party cookies", + "description": "Third-party cookies may be blocked in some contexts. [Learn more about preparing for third-party cookie restrictions](https://privacysandbox.google.com/cookies/prepare/overview).", + "score": 1, + "scoreDisplayMode": "binary", + "details": { + "type": "table", + "headings": [], + "items": [] + } + }, + "mainthread-work-breakdown": { + "id": "mainthread-work-breakdown", + "title": "Minimize main-thread work", + "description": "Consider reducing the time spent parsing, compiling and executing JS. You may find delivering smaller JS payloads helps with this. [Learn how to minimize main-thread work](https://developer.chrome.com/docs/lighthouse/performance/mainthread-work-breakdown/)", + "score": 0, + "scoreDisplayMode": "metricSavings", + "numericValue": 4661.641000000063, + "numericUnit": "millisecond", + "displayValue": "4.7 s", + "metricSavings": { + "TBT": 2650 + }, + "details": { + "type": "table", + "headings": [ + { + "key": "groupLabel", + "valueType": "text", + "label": "Category" + }, + { + "key": "duration", + "valueType": "ms", + "granularity": 1, + "label": "Time Spent" + } + ], + "items": [ + { + "group": "scriptParseCompile", + "groupLabel": "Script Parsing & Compilation", + "duration": 3230.0540000001097 + }, + { + "group": "scriptEvaluation", + "groupLabel": "Script Evaluation", + "duration": 1194.2339999999524 + }, + { + "group": "other", + "groupLabel": "Other", + "duration": 161.93500000000057 + }, + { + "group": "garbageCollection", + "groupLabel": "Garbage Collection", + "duration": 42.045000000000016 + }, + { + "group": "styleLayout", + "groupLabel": "Style & Layout", + "duration": 27.511000000000013 + }, + { + "group": "paintCompositeRender", + "groupLabel": "Rendering", + "duration": 2.960999999999995 + }, + { + "group": "parseHTML", + "groupLabel": "Parse HTML & CSS", + "duration": 2.9009999999999994 + } + ], + "sortedBy": ["duration"] + }, + "guidanceLevel": 1 + }, + "bootup-time": { + "id": "bootup-time", + "title": "Reduce JavaScript execution time", + "description": "Consider reducing the time spent parsing, compiling, and executing JS. You may find delivering smaller JS payloads helps with this. [Learn how to reduce Javascript execution time](https://developer.chrome.com/docs/lighthouse/performance/bootup-time/).", + "score": 0, + "scoreDisplayMode": "metricSavings", + "numericValue": 4345.536000000076, + "numericUnit": "millisecond", + "displayValue": "4.3 s", + "metricSavings": { + "TBT": 2650 + }, + "details": { + "type": "table", + "headings": [ + { + "key": "url", + "valueType": "url", + "label": "URL" + }, + { + "key": "total", + "granularity": 1, + "valueType": "ms", + "label": "Total CPU Time" + }, + { + "key": "scripting", + "granularity": 1, + "valueType": "ms", + "label": "Script Evaluation" + }, + { + "key": "scriptParseCompile", + "granularity": 1, + "valueType": "ms", + "label": "Script Parse" + } + ], + "items": [ + { + "url": "http://localhost:3000/_next/static/chunks/src_components_App_App_tsx.js", + "total": 2392.5660000000803, + "scripting": 181.6330000000002, + "scriptParseCompile": 2188.1670000000795 + }, + { + "url": "http://localhost:3000/_next/static/chunks/pages/_app.js", + "total": 1169.038999999994, + "scripting": 344.98599999999425, + "scriptParseCompile": 818.1769999999999 + }, + { + "url": "http://localhost:3000/_next/static/chunks/main.js", + "total": 341.58799999999957, + "scripting": 221.23599999999956, + "scriptParseCompile": 116.766 + }, + { + "url": "http://localhost:3000/_next/static/chunks/node_modules_walletconnect_ethereum-provider_dist_index_es_js.js", + "total": 145.26000000000204, + "scripting": 99.19300000000203, + "scriptParseCompile": 37.504 + }, + { + "url": "Unattributable", + "total": 140.86500000000157, + "scripting": 9.475999999999994, + "scriptParseCompile": 0 + }, + { + "url": "http://localhost:3000/_next/static/chunks/node_modules_offchainlabs_cobalt_dist_cobalt_es_js.js", + "total": 82.45600000000012, + "scripting": 67.12700000000011, + "scriptParseCompile": 11.775000000000002 + }, + { + "url": "http://localhost:3000/_next/static/chunks/node_modules_walletconnect_modal-ui_dist_index_js.js", + "total": 75.68499999999992, + "scripting": 50.400999999999904, + "scriptParseCompile": 22.612000000000002 + }, + { + "url": "webpack-internal:///../../node_modules/scheduler/cjs/scheduler.development.js", + "total": 70.4140000000001, + "scripting": 69.2150000000001, + "scriptParseCompile": 0 + }, + { + "url": "webpack-internal:///../../node_modules/idb-keyval/dist/index.js", + "total": 60.43000000000005, + "scripting": 55.07600000000005, + "scriptParseCompile": 0 + }, + { + "url": "http://localhost:3000/_next/static/chunks/pages/index.js", + "total": 52.19200000000005, + "scripting": 37.15200000000005, + "scriptParseCompile": 15.04 + } + ], + "summary": { + "wastedMs": 4345.536000000076 + }, + "sortedBy": ["total"] + }, + "guidanceLevel": 1 + }, + "uses-rel-preconnect": { + "id": "uses-rel-preconnect", + "title": "Preconnect to required origins", + "description": "Consider adding `preconnect` or `dns-prefetch` resource hints to establish early connections to important third-party origins. [Learn how to preconnect to required origins](https://developer.chrome.com/docs/lighthouse/performance/uses-rel-preconnect/).", + "score": 0, + "scoreDisplayMode": "metricSavings", + "numericValue": 118.932, + "numericUnit": "millisecond", + "displayValue": "Est savings of 120 ms", + "warnings": [], + "metricSavings": { + "LCP": 100, + "FCP": 100 + }, + "details": { + "type": "opportunity", + "headings": [ + { + "key": "url", + "valueType": "url", + "label": "URL" + }, + { + "key": "wastedMs", + "valueType": "timespanMs", + "label": "Est Savings" + } + ], + "items": [ + { + "url": "https://fonts.googleapis.com", + "wastedMs": 118.932 + } + ], + "overallSavingsMs": 118.932, + "sortedBy": ["wastedMs"] + }, + "guidanceLevel": 3 + }, + "font-display": { + "id": "font-display", + "title": "Ensure text remains visible during webfont load", + "description": "Leverage the `font-display` CSS feature to ensure text is user-visible while webfonts are loading. [Learn more about `font-display`](https://developer.chrome.com/docs/lighthouse/performance/font-display/).", + "score": 0.5, + "scoreDisplayMode": "metricSavings", + "warnings": [], + "details": { + "type": "table", + "headings": [ + { + "key": "url", + "valueType": "url", + "label": "URL" + }, + { + "key": "wastedMs", + "valueType": "ms", + "label": "Est Savings" + } + ], + "items": [ + { + "url": "https://fonts.gstatic.com/s/roboto/v47/KFO7CnqEu92Fr1ME7kSn66aGLdTylUAMa3yUBHMdazQ.woff2", + "wastedMs": 21.74099999666214 + } + ] + }, + "guidanceLevel": 3 + }, + "diagnostics": { + "id": "diagnostics", + "title": "Diagnostics", + "description": "Collection of useful page vitals.", + "score": 1, + "scoreDisplayMode": "informative", + "details": { + "type": "debugdata", + "items": [ + { + "numRequests": 64, + "numScripts": 23, + "numStylesheets": 1, + "numFonts": 4, + "numTasks": 3336, + "numTasksOver10ms": 14, + "numTasksOver25ms": 9, + "numTasksOver50ms": 7, + "numTasksOver100ms": 4, + "numTasksOver500ms": 2, + "rtt": 0, + "throughput": 120020378.17802115, + "maxRtt": 27.57, + "maxServerLatency": 30.154, + "totalByteWeight": 17480764, + "totalTaskTime": 4661.641000000021, + "mainDocumentTransferSize": 1685 + } + ] + } + }, + "network-requests": { + "id": "network-requests", + "title": "Network Requests", + "description": "Lists the network requests that were made during page load.", + "score": 1, + "scoreDisplayMode": "informative", + "details": { + "type": "table", + "headings": [ + { + "key": "url", + "valueType": "url", + "label": "URL" + }, + { + "key": "protocol", + "valueType": "text", + "label": "Protocol" + }, + { + "key": "networkRequestTime", + "valueType": "ms", + "granularity": 1, + "label": "Network Request Time" + }, + { + "key": "networkEndTime", + "valueType": "ms", + "granularity": 1, + "label": "Network End Time" + }, + { + "key": "transferSize", + "valueType": "bytes", + "displayUnit": "kb", + "granularity": 1, + "label": "Transfer Size" + }, + { + "key": "resourceSize", + "valueType": "bytes", + "displayUnit": "kb", + "granularity": 1, + "label": "Resource Size" + }, + { + "key": "statusCode", + "valueType": "text", + "label": "Status Code" + }, + { + "key": "mimeType", + "valueType": "text", + "label": "MIME Type" + }, + { + "key": "resourceType", + "valueType": "text", + "label": "Resource Type" + } + ], + "items": [ + { + "url": "http://localhost:3000/?sourceChain=ethereum&destinationChain=arbitrum-one&tab=bridge&txHistory=0", + "sessionTargetType": "page", + "protocol": "http/1.1", + "rendererStartTime": 0, + "networkRequestTime": 0.6770000010728836, + "networkEndTime": 25.979999996721745, + "finished": true, + "transferSize": 1685, + "resourceSize": 4734, + "statusCode": 200, + "mimeType": "text/html", + "resourceType": "Document", + "priority": "VeryHigh", + "experimentalFromMainFrame": true, + "entity": "localhost" + }, + { + "url": "http://localhost:3000/_next/static/media/e1d74c7d0918b0fb-s.p.woff2", + "sessionTargetType": "page", + "protocol": "http/1.1", + "rendererStartTime": 30.281999997794628, + "networkRequestTime": 30.83500000089407, + "networkEndTime": 39.99100000038743, + "finished": true, + "transferSize": 72022, + "resourceSize": 71721, + "statusCode": 200, + "mimeType": "font/woff2", + "resourceType": "Font", + "priority": "High", + "isLinkPreload": true, + "experimentalFromMainFrame": true, + "entity": "localhost" + }, + { + "url": "http://localhost:3000/_next/static/media/47bad734875a3c19-s.p.woff2", + "sessionTargetType": "page", + "protocol": "http/1.1", + "rendererStartTime": 30.534000001847744, + "networkRequestTime": 31.096999999135733, + "networkEndTime": 40.40299999713898, + "finished": true, + "transferSize": 73529, + "resourceSize": 73228, + "statusCode": 200, + "mimeType": "font/woff2", + "resourceType": "Font", + "priority": "High", + "isLinkPreload": true, + "experimentalFromMainFrame": true, + "entity": "localhost" + }, + { + "url": "http://localhost:3000/_next/static/media/a0dda0300042e3c7-s.p.woff2", + "sessionTargetType": "page", + "protocol": "http/1.1", + "rendererStartTime": 30.682000000029802, + "networkRequestTime": 31.3260000012815, + "networkEndTime": 40.19200000166893, + "finished": true, + "transferSize": 74635, + "resourceSize": 74334, + "statusCode": 200, + "mimeType": "font/woff2", + "resourceType": "Font", + "priority": "High", + "isLinkPreload": true, + "experimentalFromMainFrame": true, + "entity": "localhost" + }, + { + "url": "http://localhost:3000/_next/static/chunks/webpack.js", + "sessionTargetType": "page", + "protocol": "http/1.1", + "rendererStartTime": 30.822999998927116, + "networkRequestTime": 36.493000000715256, + "networkEndTime": 48.94700000062585, + "finished": true, + "transferSize": 10384, + "resourceSize": 53011, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script", + "priority": "Low", + "experimentalFromMainFrame": true, + "entity": "localhost" + }, + { + "url": "http://localhost:3000/_next/static/chunks/main.js", + "sessionTargetType": "page", + "protocol": "http/1.1", + "rendererStartTime": 30.943999998271465, + "networkRequestTime": 36.82899999991059, + "networkEndTime": 158.5190000012517, + "finished": true, + "transferSize": 1177149, + "resourceSize": 5287954, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script", + "priority": "Low", + "experimentalFromMainFrame": true, + "entity": "localhost" + }, + { + "url": "http://localhost:3000/_next/static/chunks/pages/_app.js", + "sessionTargetType": "page", + "protocol": "http/1.1", + "rendererStartTime": 31.07599999755621, + "networkRequestTime": 39.418000001460314, + "networkEndTime": 383.3449999988079, + "finished": true, + "transferSize": 3979857, + "resourceSize": 18915423, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script", + "priority": "Low", + "experimentalFromMainFrame": true, + "entity": "localhost" + }, + { + "url": "http://localhost:3000/_next/static/chunks/pages/index.js", + "sessionTargetType": "page", + "protocol": "http/1.1", + "rendererStartTime": 31.195000000298023, + "networkRequestTime": 40.53700000047684, + "networkEndTime": 71.51000000163913, + "finished": true, + "transferSize": 196454, + "resourceSize": 895933, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script", + "priority": "Low", + "experimentalFromMainFrame": true, + "entity": "localhost" + }, + { + "url": "http://localhost:3000/_next/static/development/_buildManifest.js", + "sessionTargetType": "page", + "protocol": "http/1.1", + "rendererStartTime": 31.313999999314547, + "networkRequestTime": 40.64799999818206, + "networkEndTime": 49.53000000119209, + "finished": true, + "transferSize": 633, + "resourceSize": 296, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script", + "priority": "Low", + "experimentalFromMainFrame": true, + "entity": "localhost" + }, + { + "url": "http://localhost:3000/_next/static/development/_ssgManifest.js", + "sessionTargetType": "page", + "protocol": "http/1.1", + "rendererStartTime": 31.46900000050664, + "networkRequestTime": 40.76400000229478, + "networkEndTime": 50.05199999734759, + "finished": true, + "transferSize": 411, + "resourceSize": 76, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script", + "priority": "Low", + "experimentalFromMainFrame": true, + "entity": "localhost" + }, + { + "url": "http://localhost:3000/_next/static/chunks/react-refresh.js", + "sessionTargetType": "page", + "protocol": "http/1.1", + "rendererStartTime": 31.605000000447035, + "networkRequestTime": 32.08100000023842, + "networkEndTime": 39.11199999973178, + "finished": true, + "transferSize": 25433, + "resourceSize": 79107, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script", + "priority": "High", + "experimentalFromMainFrame": true, + "entity": "localhost" + }, + { + "url": "https://fonts.googleapis.com/css2?family=Roboto:wght@300;400;500;600;700", + "sessionTargetType": "page", + "protocol": "h2", + "rendererStartTime": 983.8819999992847, + "networkRequestTime": 984.3619999997318, + "networkEndTime": 1132.8929999992251, + "finished": true, + "transferSize": 2318, + "resourceSize": 27200, + "statusCode": 200, + "mimeType": "text/css", + "resourceType": "Stylesheet", + "priority": "VeryHigh", + "experimentalFromMainFrame": true, + "entity": "Google Fonts" + }, + { + "url": "http://localhost:3000/_next/static/development/_devMiddlewareManifest.json", + "sessionTargetType": "page", + "protocol": "http/1.1", + "rendererStartTime": 1744.6009999997914, + "networkRequestTime": 1745.2359999977052, + "networkEndTime": 1747.2919999994338, + "finished": true, + "transferSize": 213, + "resourceSize": 2, + "statusCode": 200, + "mimeType": "application/json", + "resourceType": "Fetch", + "priority": "High", + "experimentalFromMainFrame": true, + "entity": "localhost" + }, + { + "url": "http://localhost:3000/logo.png", + "sessionTargetType": "page", + "protocol": "http/1.1", + "rendererStartTime": 1761.6409999988973, + "networkRequestTime": 1762.0570000000298, + "networkEndTime": 1764.5969999991357, + "finished": true, + "transferSize": 22781, + "resourceSize": 22500, + "statusCode": 200, + "mimeType": "image/png", + "resourceType": "Other", + "priority": "High", + "experimentalFromMainFrame": true, + "entity": "localhost" + }, + { + "url": "http://localhost:3000/_next/image?url=%2F_next%2Fstatic%2Fmedia%2Feclipse_bottom.4304bd37.png&w=1920&q=75", + "sessionTargetType": "page", + "protocol": "http/1.1", + "rendererStartTime": 1772.6860000006855, + "networkRequestTime": 1773.0720000006258, + "networkEndTime": 1779.062000002712, + "finished": true, + "transferSize": 477810, + "resourceSize": 477384, + "statusCode": 200, + "mimeType": "image/webp", + "resourceType": "Image", + "priority": "Low", + "experimentalFromMainFrame": true, + "entity": "localhost" + }, + { + "url": "http://localhost:3000/_next/static/chunks/node_modules_offchainlabs_cobalt_dist_cobalt_es_js.js", + "sessionTargetType": "page", + "protocol": "http/1.1", + "rendererStartTime": 1793.5509999990463, + "networkRequestTime": 1794.4749999977648, + "networkEndTime": 1809.2509999983013, + "finished": true, + "transferSize": 147417, + "resourceSize": 576083, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script", + "priority": "Low", + "experimentalFromMainFrame": true, + "entity": "localhost" + }, + { + "url": "http://localhost:3000/api/status", + "sessionTargetType": "page", + "protocol": "http/1.1", + "rendererStartTime": 1812.562999997288, + "networkRequestTime": 1813.2269999980927, + "networkEndTime": 1925.8379999995232, + "finished": true, + "transferSize": 1070, + "resourceSize": 3206, + "statusCode": 200, + "mimeType": "application/json", + "resourceType": "Fetch", + "priority": "High", + "experimentalFromMainFrame": true, + "entity": "localhost" + }, + { + "url": "http://localhost:3000/api/status", + "sessionTargetType": "page", + "protocol": "http/1.1", + "rendererStartTime": 1817.8459999971092, + "networkRequestTime": 1818.5679999999702, + "networkEndTime": 1957.3759999983013, + "finished": true, + "transferSize": 1072, + "resourceSize": 3206, + "statusCode": 200, + "mimeType": "application/json", + "resourceType": "Fetch", + "priority": "High", + "experimentalFromMainFrame": true, + "entity": "localhost" + }, + { + "url": "…", + "sessionTargetType": "page", + "protocol": "data", + "rendererStartTime": 1912.9109999984503, + "networkRequestTime": 1912.9109999984503, + "networkEndTime": 1913.1499999985099, + "finished": true, + "transferSize": 0, + "resourceSize": 6129, + "statusCode": 200, + "mimeType": "image/svg+xml", + "resourceType": "Image", + "priority": "Low", + "experimentalFromMainFrame": true + }, + { + "url": "…", + "sessionTargetType": "page", + "protocol": "data", + "rendererStartTime": 1913.7819999977946, + "networkRequestTime": 1913.7819999977946, + "networkEndTime": 1913.910000000149, + "finished": true, + "transferSize": 0, + "resourceSize": 2056, + "statusCode": 200, + "mimeType": "image/svg+xml", + "resourceType": "Image", + "priority": "Low", + "experimentalFromMainFrame": true + }, + { + "url": "…", + "sessionTargetType": "page", + "protocol": "data", + "rendererStartTime": 1914.3499999977648, + "networkRequestTime": 1914.3499999977648, + "networkEndTime": 1914.4659999981523, + "finished": true, + "transferSize": 0, + "resourceSize": 1720, + "statusCode": 200, + "mimeType": "image/svg+xml", + "resourceType": "Image", + "priority": "Low", + "experimentalFromMainFrame": true + }, + { + "url": "…", + "sessionTargetType": "page", + "protocol": "data", + "rendererStartTime": 1915.2939999997616, + "networkRequestTime": 1915.2939999997616, + "networkEndTime": 1915.3699999973178, + "finished": true, + "transferSize": 0, + "resourceSize": 1611, + "statusCode": 200, + "mimeType": "image/svg+xml", + "resourceType": "Image", + "priority": "Low", + "experimentalFromMainFrame": true + }, + { + "url": "…", + "sessionTargetType": "page", + "protocol": "data", + "rendererStartTime": 1916.1579999998212, + "networkRequestTime": 1916.1579999998212, + "networkEndTime": 1916.2780000008643, + "finished": true, + "transferSize": 0, + "resourceSize": 2112, + "statusCode": 200, + "mimeType": "image/svg+xml", + "resourceType": "Image", + "priority": "Low", + "experimentalFromMainFrame": true + }, + { + "url": "…", + "sessionTargetType": "page", + "protocol": "data", + "rendererStartTime": 1917.0289999991655, + "networkRequestTime": 1917.0289999991655, + "networkEndTime": 1917.1369999982417, + "finished": true, + "transferSize": 0, + "resourceSize": 1166, + "statusCode": 200, + "mimeType": "image/svg+xml", + "resourceType": "Image", + "priority": "Low", + "experimentalFromMainFrame": true + }, + { + "url": "…", + "sessionTargetType": "page", + "protocol": "data", + "rendererStartTime": 1917.8579999990761, + "networkRequestTime": 1917.8579999990761, + "networkEndTime": 1917.9680000022054, + "finished": true, + "transferSize": 0, + "resourceSize": 1790, + "statusCode": 200, + "mimeType": "image/svg+xml", + "resourceType": "Image", + "priority": "Low", + "experimentalFromMainFrame": true + }, + { + "url": "…", + "sessionTargetType": "page", + "protocol": "data", + "rendererStartTime": 1918.7769999988377, + "networkRequestTime": 1918.7769999988377, + "networkEndTime": 1918.9079999998212, + "finished": true, + "transferSize": 0, + "resourceSize": 3274, + "statusCode": 200, + "mimeType": "image/svg+xml", + "resourceType": "Image", + "priority": "Low", + "experimentalFromMainFrame": true + }, + { + "url": "…", + "sessionTargetType": "page", + "protocol": "data", + "rendererStartTime": 1919.582999996841, + "networkRequestTime": 1919.582999996841, + "networkEndTime": 1919.6530000008643, + "finished": true, + "transferSize": 0, + "resourceSize": 1046, + "statusCode": 200, + "mimeType": "image/svg+xml", + "resourceType": "Image", + "priority": "Low", + "experimentalFromMainFrame": true + }, + { + "url": "https://fonts.gstatic.com/s/roboto/v47/KFO7CnqEu92Fr1ME7kSn66aGLdTylUAMa3yUBHMdazQ.woff2", + "sessionTargetType": "page", + "protocol": "h3", + "rendererStartTime": 1930.684999998659, + "networkRequestTime": 1935.3220000006258, + "networkEndTime": 1957.062999997288, + "finished": true, + "transferSize": 34700, + "resourceSize": 34668, + "statusCode": 200, + "mimeType": "font/woff2", + "resourceType": "Font", + "priority": "VeryHigh", + "experimentalFromMainFrame": true, + "entity": "Google Fonts" + }, + { + "url": "http://127.0.0.1:8547/", + "sessionTargetType": "page", + "protocol": "", + "rendererStartTime": 1942.6429999992251, + "networkRequestTime": 1942.6429999992251, + "networkEndTime": 1946.6990000009537, + "finished": true, + "transferSize": 0, + "resourceSize": 0, + "statusCode": -1, + "mimeType": "", + "resourceType": "Fetch", + "priority": "High", + "experimentalFromMainFrame": true, + "entity": "127.0.0.1" + }, + { + "url": "http://127.0.0.1:8547/", + "sessionTargetType": "page", + "protocol": "", + "rendererStartTime": 1945.2060000002384, + "networkRequestTime": 1945.2060000002384, + "networkEndTime": 1945.3269999995828, + "finished": true, + "transferSize": 0, + "resourceSize": 0, + "statusCode": -1, + "mimeType": "", + "resourceType": "Preflight", + "priority": "High", + "entity": "127.0.0.1" + }, + { + "url": "http://127.0.0.1:8547/", + "sessionTargetType": "page", + "protocol": "", + "rendererStartTime": 1947.9059999994934, + "networkRequestTime": 1947.9059999994934, + "networkEndTime": 1950.422999996692, + "finished": true, + "transferSize": 0, + "resourceSize": 0, + "statusCode": -1, + "mimeType": "", + "resourceType": "Fetch", + "priority": "High", + "experimentalFromMainFrame": true, + "entity": "127.0.0.1" + }, + { + "url": "http://127.0.0.1:8547/", + "sessionTargetType": "page", + "protocol": "", + "rendererStartTime": 1948.6519999988377, + "networkRequestTime": 1948.6519999988377, + "networkEndTime": 1949.7069999985397, + "finished": true, + "transferSize": 0, + "resourceSize": 0, + "statusCode": -1, + "mimeType": "", + "resourceType": "Preflight", + "priority": "High", + "entity": "127.0.0.1" + }, + { + "url": "http://127.0.0.1:8547/", + "sessionTargetType": "page", + "protocol": "", + "rendererStartTime": 1954.5719999969006, + "networkRequestTime": 1954.5719999969006, + "networkEndTime": 1957.1109999977052, + "finished": true, + "transferSize": 0, + "resourceSize": 0, + "statusCode": -1, + "mimeType": "", + "resourceType": "Fetch", + "priority": "High", + "experimentalFromMainFrame": true, + "entity": "127.0.0.1" + }, + { + "url": "http://127.0.0.1:8547/", + "sessionTargetType": "page", + "protocol": "", + "rendererStartTime": 1955.7359999977052, + "networkRequestTime": 1955.7359999977052, + "networkEndTime": 1957.030000001192, + "finished": true, + "transferSize": 0, + "resourceSize": 0, + "statusCode": -1, + "mimeType": "", + "resourceType": "Preflight", + "priority": "High", + "entity": "127.0.0.1" + }, + { + "url": "http://127.0.0.1:8547/", + "sessionTargetType": "page", + "protocol": "", + "rendererStartTime": 1957.8319999985397, + "networkRequestTime": 1957.8319999985397, + "networkEndTime": 1960.0109999999404, + "finished": true, + "transferSize": 0, + "resourceSize": 0, + "statusCode": -1, + "mimeType": "", + "resourceType": "Fetch", + "priority": "High", + "experimentalFromMainFrame": true, + "entity": "127.0.0.1" + }, + { + "url": "http://127.0.0.1:8547/", + "sessionTargetType": "page", + "protocol": "", + "rendererStartTime": 1959.605000000447, + "networkRequestTime": 1959.605000000447, + "networkEndTime": 1959.9219999983907, + "finished": true, + "transferSize": 0, + "resourceSize": 0, + "statusCode": -1, + "mimeType": "", + "resourceType": "Preflight", + "priority": "High", + "entity": "127.0.0.1" + }, + { + "url": "http://localhost:3000/_next/static/chunks/src_components_App_App_tsx.js", + "sessionTargetType": "page", + "protocol": "http/1.1", + "rendererStartTime": 1960.4829999990761, + "networkRequestTime": 1960.925000000745, + "networkEndTime": 2658.162999998778, + "finished": true, + "transferSize": 9942484, + "resourceSize": 43420453, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script", + "priority": "Low", + "experimentalFromMainFrame": true, + "entity": "localhost" + }, + { + "url": "http://localhost:3000/_next/static/chunks/node_modules_walletconnect_ethereum-provider_dist_index_es_js.js", + "sessionTargetType": "page", + "protocol": "http/1.1", + "rendererStartTime": 4986.796000000089, + "networkRequestTime": 4987.328999999911, + "networkEndTime": 5038.241999998689, + "finished": true, + "transferSize": 639168, + "resourceSize": 2332526, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script", + "priority": "Low", + "experimentalFromMainFrame": true, + "entity": "localhost" + }, + { + "url": "http://localhost:3000/_next/static/chunks/node_modules_rainbow-me_rainbowkit_dist_safeWallet-VUYZPLY4_js.js", + "sessionTargetType": "page", + "protocol": "http/1.1", + "rendererStartTime": 5140.322000000626, + "networkRequestTime": 5140.866000000387, + "networkEndTime": 5144.382999997586, + "finished": true, + "transferSize": 3266, + "resourceSize": 5674, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script", + "priority": "Low", + "experimentalFromMainFrame": true, + "entity": "localhost" + }, + { + "url": "http://localhost:3000/_next/static/chunks/node_modules_rainbow-me_rainbowkit_dist_rainbowWallet-2SR6TVBF_js.js", + "sessionTargetType": "page", + "protocol": "http/1.1", + "rendererStartTime": 5140.985999997705, + "networkRequestTime": 5141.56799999997, + "networkEndTime": 5145.585999999195, + "finished": true, + "transferSize": 4076, + "resourceSize": 11347, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script", + "priority": "Low", + "experimentalFromMainFrame": true, + "entity": "localhost" + }, + { + "url": "http://localhost:3000/_next/static/chunks/node_modules_rainbow-me_rainbowkit_dist_coinbaseWallet-WWX6LF36_js.js", + "sessionTargetType": "page", + "protocol": "http/1.1", + "rendererStartTime": 5141.629999998957, + "networkRequestTime": 5142.051999997348, + "networkEndTime": 5146.741999998689, + "finished": true, + "transferSize": 2301, + "resourceSize": 3751, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script", + "priority": "Low", + "experimentalFromMainFrame": true, + "entity": "localhost" + }, + { + "url": "http://localhost:3000/_next/static/chunks/node_modules_rainbow-me_rainbowkit_dist_metaMaskWallet-YFHEHW7V_js.js", + "sessionTargetType": "page", + "protocol": "http/1.1", + "rendererStartTime": 5142.186999998987, + "networkRequestTime": 5142.592000000179, + "networkEndTime": 5148.065000001341, + "finished": true, + "transferSize": 4980, + "resourceSize": 18353, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script", + "priority": "Low", + "experimentalFromMainFrame": true, + "entity": "localhost" + }, + { + "url": "http://localhost:3000/_next/static/chunks/node_modules_rainbow-me_rainbowkit_dist_walletConnectWallet-FNSU4KNU_js.js", + "sessionTargetType": "page", + "protocol": "http/1.1", + "rendererStartTime": 5142.726999998093, + "networkRequestTime": 5143.187000002712, + "networkEndTime": 5148.48900000006, + "finished": true, + "transferSize": 3315, + "resourceSize": 5968, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script", + "priority": "Low", + "experimentalFromMainFrame": true, + "entity": "localhost" + }, + { + "url": "http://localhost:3000/_next/static/chunks/node_modules_rainbow-me_rainbowkit_dist_wallets_walletConnectors_okxWallet-GJMKZIND_js.js", + "sessionTargetType": "page", + "protocol": "http/1.1", + "rendererStartTime": 5143.3069999963045, + "networkRequestTime": 5143.695999998599, + "networkEndTime": 5149.804000001401, + "finished": true, + "transferSize": 2437, + "resourceSize": 4565, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script", + "priority": "Low", + "experimentalFromMainFrame": true, + "entity": "localhost" + }, + { + "url": "http://localhost:3000/_next/static/chunks/node_modules_rainbow-me_rainbowkit_dist_wallets_walletConnectors_trustWallet-E2GVGE4U_js.js", + "sessionTargetType": "page", + "protocol": "http/1.1", + "rendererStartTime": 5143.853000000119, + "networkRequestTime": 5144.5289999991655, + "networkEndTime": 5150.170000001788, + "finished": true, + "transferSize": 2509, + "resourceSize": 4371, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script", + "priority": "Low", + "experimentalFromMainFrame": true, + "entity": "localhost" + }, + { + "url": "http://localhost:3000/_next/static/chunks/node_modules_rainbow-me_rainbowkit_dist_wallets_walletConnectors_rabbyWallet-FLVUU35F_js.js", + "sessionTargetType": "page", + "protocol": "http/1.1", + "rendererStartTime": 5144.51099999994, + "networkRequestTime": 5145.7289999984205, + "networkEndTime": 5151.063000001013, + "finished": true, + "transferSize": 4330, + "resourceSize": 8269, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script", + "priority": "Low", + "experimentalFromMainFrame": true, + "entity": "localhost" + }, + { + "url": "http://localhost:3000/_next/static/chunks/node_modules_rainbow-me_rainbowkit_dist_assets-NU2OP443_js.js", + "sessionTargetType": "page", + "protocol": "http/1.1", + "rendererStartTime": 5145.228000000119, + "networkRequestTime": 5146.85000000149, + "networkEndTime": 5152.309000000358, + "finished": true, + "transferSize": 12469, + "resourceSize": 45515, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script", + "priority": "Low", + "experimentalFromMainFrame": true, + "entity": "localhost" + }, + { + "url": "http://localhost:3000/_next/static/chunks/node_modules_rainbow-me_rainbowkit_dist_login-CWDTIDNK_js.js", + "sessionTargetType": "page", + "protocol": "http/1.1", + "rendererStartTime": 5145.796000000089, + "networkRequestTime": 5148.248999997973, + "networkEndTime": 5152.491000000387, + "finished": true, + "transferSize": 9697, + "resourceSize": 24218, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script", + "priority": "Low", + "experimentalFromMainFrame": true, + "entity": "localhost" + }, + { + "url": "…", + "sessionTargetType": "page", + "protocol": "data", + "rendererStartTime": 5166.4310000017285, + "networkRequestTime": 5166.4310000017285, + "networkEndTime": 5166.513000000268, + "finished": true, + "transferSize": 0, + "resourceSize": 280, + "statusCode": 200, + "mimeType": "image/svg+xml", + "resourceType": "Image", + "priority": "Low", + "experimentalFromMainFrame": true + }, + { + "url": "https://verify.walletconnect.com/8702c5ea8d157774a011deb8e1433d01", + "sessionTargetType": "iframe", + "protocol": "h3", + "rendererStartTime": 5554.9439999982715, + "networkRequestTime": 5555.662999998778, + "networkEndTime": 5660.646999999881, + "finished": true, + "transferSize": 278, + "resourceSize": 62, + "statusCode": 200, + "mimeType": "text/html", + "resourceType": "Document", + "priority": "VeryHigh", + "entity": "walletconnect.com" + }, + { + "url": "http://localhost:3000/_next/static/chunks/node_modules_coinbase_wallet-sdk_dist_index_js.js", + "sessionTargetType": "page", + "protocol": "http/1.1", + "rendererStartTime": 5554.157999999821, + "networkRequestTime": 5554.584000002593, + "networkEndTime": 5567.478000000119, + "finished": true, + "transferSize": 173361, + "resourceSize": 657128, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script", + "priority": "Low", + "experimentalFromMainFrame": true, + "entity": "localhost" + }, + { + "url": "http://localhost:3000/", + "sessionTargetType": "page", + "protocol": "http/1.1", + "rendererStartTime": 5601.936999998987, + "networkRequestTime": 5602.295000001788, + "networkEndTime": 5613.890999998897, + "finished": true, + "transferSize": 276, + "resourceSize": 0, + "statusCode": 200, + "mimeType": "text/html", + "resourceType": "Fetch", + "priority": "High", + "experimentalFromMainFrame": true, + "entity": "localhost" + }, + { + "url": "http://localhost:3000/_next/static/chunks/node_modules_walletconnect_modal_dist_index_js.js", + "sessionTargetType": "page", + "protocol": "http/1.1", + "rendererStartTime": 5633.605000000447, + "networkRequestTime": 5634.261999998242, + "networkEndTime": 5637.504999998957, + "finished": true, + "transferSize": 22370, + "resourceSize": 63923, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script", + "priority": "Low", + "experimentalFromMainFrame": true, + "entity": "localhost" + }, + { + "url": "http://localhost:3000/_next/static/chunks/node_modules_walletconnect_modal-ui_dist_index_js.js", + "sessionTargetType": "page", + "protocol": "http/1.1", + "rendererStartTime": 5643.223999999464, + "networkRequestTime": 5643.6819999963045, + "networkEndTime": 5668.773000001907, + "finished": true, + "transferSize": 334017, + "resourceSize": 1321332, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script", + "priority": "Low", + "experimentalFromMainFrame": true, + "entity": "localhost" + }, + { + "url": "https://explorer-api.walletconnect.com/w3m/v1/getDesktopListings?projectId=8702c5ea8d157774a011deb8e1433d01&sdkType=wcm&sdkVersion=js-2.6.2&page=1&entries=9&version=2", + "sessionTargetType": "page", + "protocol": "h3", + "rendererStartTime": 5745.813000001013, + "networkRequestTime": 5746.335000000894, + "networkEndTime": 6012.017999999225, + "finished": true, + "transferSize": 2226, + "resourceSize": 7613, + "statusCode": 200, + "mimeType": "application/json", + "resourceType": "Fetch", + "priority": "High", + "experimentalFromMainFrame": true, + "entity": "walletconnect.com" + }, + { + "url": "https://explorer-api.walletconnect.com/w3m/v1/getWalletImage/3d7eb880-7654-431f-ed84-a25712b45200?projectId=8702c5ea8d157774a011deb8e1433d01&sdkType=wcm&sdkVersion=js-2.6.2", + "sessionTargetType": "page", + "protocol": "h3", + "rendererStartTime": 6013.951999999583, + "networkRequestTime": 6014.236999999732, + "networkEndTime": 6220.143999997526, + "finished": true, + "transferSize": 2572, + "resourceSize": 2176, + "statusCode": 200, + "mimeType": "image/webp", + "resourceType": "Image", + "priority": "Low", + "experimentalFromMainFrame": true, + "entity": "walletconnect.com" + }, + { + "url": "https://explorer-api.walletconnect.com/w3m/v1/getWalletImage/4e7d6f52-f663-4fc1-4b88-eebe7fc72800?projectId=8702c5ea8d157774a011deb8e1433d01&sdkType=wcm&sdkVersion=js-2.6.2", + "sessionTargetType": "page", + "protocol": "h3", + "rendererStartTime": 6014.06099999696, + "networkRequestTime": 6014.401000000536, + "networkEndTime": 6061.516000002623, + "finished": true, + "transferSize": 2038, + "resourceSize": 1638, + "statusCode": 200, + "mimeType": "image/webp", + "resourceType": "Image", + "priority": "Low", + "experimentalFromMainFrame": true, + "entity": "walletconnect.com" + }, + { + "url": "https://explorer-api.walletconnect.com/w3m/v1/getWalletImage/d8e930b6-ccde-471e-ecbe-6967b1c0c400?projectId=8702c5ea8d157774a011deb8e1433d01&sdkType=wcm&sdkVersion=js-2.6.2", + "sessionTargetType": "page", + "protocol": "h3", + "rendererStartTime": 6014.12899999693, + "networkRequestTime": 6014.546999998391, + "networkEndTime": 6076.444999996573, + "finished": true, + "transferSize": 1974, + "resourceSize": 1576, + "statusCode": 200, + "mimeType": "image/webp", + "resourceType": "Image", + "priority": "Low", + "experimentalFromMainFrame": true, + "entity": "walletconnect.com" + }, + { + "url": "https://explorer-api.walletconnect.com/w3m/v1/getWalletImage/77c1d3dd-0213-400a-f9cc-bfd524c47f00?projectId=8702c5ea8d157774a011deb8e1433d01&sdkType=wcm&sdkVersion=js-2.6.2", + "sessionTargetType": "page", + "protocol": "h3", + "rendererStartTime": 6014.242000002414, + "networkRequestTime": 6014.741000000387, + "networkEndTime": 6067.776999998838, + "finished": true, + "transferSize": 1804, + "resourceSize": 1404, + "statusCode": 200, + "mimeType": "image/webp", + "resourceType": "Image", + "priority": "Low", + "experimentalFromMainFrame": true, + "entity": "walletconnect.com" + }, + { + "url": "https://explorer-api.walletconnect.com/w3m/v1/getWalletImage/c20e1cec-05e8-4ac6-a086-7ce355092400?projectId=8702c5ea8d157774a011deb8e1433d01&sdkType=wcm&sdkVersion=js-2.6.2", + "sessionTargetType": "page", + "protocol": "h3", + "rendererStartTime": 6014.335999999195, + "networkRequestTime": 6014.942000001669, + "networkEndTime": 6070.927999999374, + "finished": true, + "transferSize": 1592, + "resourceSize": 1192, + "statusCode": 200, + "mimeType": "image/webp", + "resourceType": "Image", + "priority": "Low", + "experimentalFromMainFrame": true, + "entity": "walletconnect.com" + }, + { + "url": "https://explorer-api.walletconnect.com/w3m/v1/getWalletImage/67336675-0daa-489b-6885-cb95234bc400?projectId=8702c5ea8d157774a011deb8e1433d01&sdkType=wcm&sdkVersion=js-2.6.2", + "sessionTargetType": "page", + "protocol": "h3", + "rendererStartTime": 6014.425000000745, + "networkRequestTime": 6015.127999998629, + "networkEndTime": 6071.069999996573, + "finished": true, + "transferSize": 2273, + "resourceSize": 1872, + "statusCode": 200, + "mimeType": "image/webp", + "resourceType": "Image", + "priority": "Low", + "experimentalFromMainFrame": true, + "entity": "walletconnect.com" + }, + { + "url": "https://explorer-api.walletconnect.com/w3m/v1/getWalletImage/a578bd45-b418-4111-2c56-8ddcd1417c00?projectId=8702c5ea8d157774a011deb8e1433d01&sdkType=wcm&sdkVersion=js-2.6.2", + "sessionTargetType": "page", + "protocol": "h3", + "rendererStartTime": 6014.479000002146, + "networkRequestTime": 6015.288999997079, + "networkEndTime": 6073.75, + "finished": true, + "transferSize": 1972, + "resourceSize": 1572, + "statusCode": 200, + "mimeType": "image/webp", + "resourceType": "Image", + "priority": "Low", + "experimentalFromMainFrame": true, + "entity": "walletconnect.com" + }, + { + "url": "https://explorer-api.walletconnect.com/w3m/v1/getWalletImage/7e1514ba-932d-415d-1bdb-bccb6c2cbc00?projectId=8702c5ea8d157774a011deb8e1433d01&sdkType=wcm&sdkVersion=js-2.6.2", + "sessionTargetType": "page", + "protocol": "h3", + "rendererStartTime": 6014.559999998659, + "networkRequestTime": 6015.457999996841, + "networkEndTime": 6080.824000000954, + "finished": true, + "transferSize": 1309, + "resourceSize": 910, + "statusCode": 200, + "mimeType": "image/webp", + "resourceType": "Image", + "priority": "Low", + "experimentalFromMainFrame": true, + "entity": "walletconnect.com" + }, + { + "url": "https://explorer-api.walletconnect.com/w3m/v1/getWalletImage/f5b26eef-c5e8-421a-e379-ae010b4a7400?projectId=8702c5ea8d157774a011deb8e1433d01&sdkType=wcm&sdkVersion=js-2.6.2", + "sessionTargetType": "page", + "protocol": "h3", + "rendererStartTime": 6014.614999998361, + "networkRequestTime": 6015.614999998361, + "networkEndTime": 6073.625, + "finished": true, + "transferSize": 2097, + "resourceSize": 1694, + "statusCode": 200, + "mimeType": "image/webp", + "resourceType": "Image", + "priority": "Low", + "experimentalFromMainFrame": true, + "entity": "walletconnect.com" + } + ], + "debugData": { + "type": "debugdata", + "networkStartTimeTs": 31362658561 + } + } + }, + "network-rtt": { + "id": "network-rtt", + "title": "Network Round Trip Times", + "description": "Network round trip times (RTT) have a large impact on performance. If the RTT to an origin is high, it's an indication that servers closer to the user could improve performance. [Learn more about the Round Trip Time](https://hpbn.co/primer-on-latency-and-bandwidth/).", + "score": 1, + "scoreDisplayMode": "informative", + "numericValue": 27.57, + "numericUnit": "millisecond", + "displayValue": "30 ms", + "details": { + "type": "table", + "headings": [ + { + "key": "origin", + "valueType": "text", + "label": "URL" + }, + { + "key": "rtt", + "valueType": "ms", + "granularity": 1, + "label": "Time Spent" + } + ], + "items": [ + { + "origin": "https://verify.walletconnect.com", + "rtt": 27.57 + }, + { + "origin": "https://explorer-api.walletconnect.com", + "rtt": 26.305 + }, + { + "origin": "https://fonts.googleapis.com", + "rtt": 19.466 + }, + { + "origin": "http://localhost:3000", + "rtt": 0.03675 + }, + { + "origin": "https://fonts.gstatic.com", + "rtt": 0 + } + ], + "sortedBy": ["rtt"] + } + }, + "network-server-latency": { + "id": "network-server-latency", + "title": "Server Backend Latencies", + "description": "Server latencies can impact web performance. If the server latency of an origin is high, it's an indication the server is overloaded or has poor backend performance. [Learn more about server response time](https://hpbn.co/primer-on-web-performance/#analyzing-the-resource-waterfall).", + "score": 1, + "scoreDisplayMode": "informative", + "numericValue": 30.154, + "numericUnit": "millisecond", + "displayValue": "30 ms", + "details": { + "type": "table", + "headings": [ + { + "key": "origin", + "valueType": "text", + "label": "URL" + }, + { + "key": "serverResponseTime", + "valueType": "ms", + "granularity": 1, + "label": "Time Spent" + } + ], + "items": [ + { + "origin": "https://explorer-api.walletconnect.com", + "serverResponseTime": 30.154 + }, + { + "origin": "https://fonts.googleapis.com", + "serverResponseTime": 19.48600000000001 + }, + { + "origin": "https://fonts.gstatic.com", + "serverResponseTime": 18.366 + }, + { + "origin": "https://verify.walletconnect.com", + "serverResponseTime": 13.793 + }, + { + "origin": "http://localhost:3000", + "serverResponseTime": 4.28725 + } + ], + "sortedBy": ["serverResponseTime"] + } + }, + "main-thread-tasks": { + "id": "main-thread-tasks", + "title": "Tasks", + "description": "Lists the toplevel main thread tasks that executed during page load.", + "score": 1, + "scoreDisplayMode": "informative", + "details": { + "type": "table", + "headings": [ + { + "key": "startTime", + "valueType": "ms", + "granularity": 1, + "label": "Start Time" + }, + { + "key": "duration", + "valueType": "ms", + "granularity": 1, + "label": "End Time" + } + ], + "items": [ + { + "duration": 6.119, + "startTime": 29.302 + }, + { + "duration": 6.4, + "startTime": 52.092 + }, + { + "duration": 341.588, + "startTime": 172.552 + }, + { + "duration": 9.503, + "startTime": 515.115 + }, + { + "duration": 1169.039, + "startTime": 524.635 + }, + { + "duration": 52.192, + "startTime": 1693.778 + }, + { + "duration": 6.687, + "startTime": 1767.221 + }, + { + "duration": 27.735, + "startTime": 1773.964 + }, + { + "duration": 17.067, + "startTime": 1810.716 + }, + { + "duration": 9.507, + "startTime": 1831.01 + }, + { + "duration": 82.731, + "startTime": 1840.89 + }, + { + "duration": 12.054, + "startTime": 1923.692 + }, + { + "duration": 5.147, + "startTime": 2692.1 + }, + { + "duration": 24.434, + "startTime": 2751.595 + }, + { + "duration": 2392.075, + "startTime": 2776.096 + }, + { + "duration": 5.452, + "startTime": 5168.969 + }, + { + "duration": 144.056, + "startTime": 5185.436 + }, + { + "duration": 23.188, + "startTime": 5409.964 + }, + { + "duration": 11.402, + "startTime": 5435.098 + }, + { + "duration": 8.081, + "startTime": 5448.236 + }, + { + "duration": 35.519, + "startTime": 5570.688 + }, + { + "duration": 5.66, + "startTime": 5639.285 + }, + { + "duration": 75.716, + "startTime": 5675.205 + }, + { + "duration": 6.075, + "startTime": 5750.934 + } + ] + } + }, + "metrics": { + "id": "metrics", + "title": "Metrics", + "description": "Collects all available metrics.", + "score": 1, + "scoreDisplayMode": "informative", + "numericValue": 15825, + "numericUnit": "millisecond", + "details": { + "type": "debugdata", + "items": [ + { + "firstContentfulPaint": 442, + "largestContentfulPaint": 6387, + "interactive": 15825, + "speedIndex": 5787, + "totalBlockingTime": 2630, + "maxPotentialFID": 1196, + "cumulativeLayoutShift": 0, + "cumulativeLayoutShiftMainFrame": 0, + "lcpLoadStart": 5942, + "lcpLoadEnd": 5962, + "timeToFirstByte": 124, + "observedTimeOrigin": 0, + "observedTimeOriginTs": 31362657570, + "observedNavigationStart": 0, + "observedNavigationStartTs": 31362657570, + "observedFirstPaint": 1879, + "observedFirstPaintTs": 31364536563, + "observedFirstContentfulPaint": 1907, + "observedFirstContentfulPaintTs": 31364564476, + "observedFirstContentfulPaintAllFrames": 1907, + "observedFirstContentfulPaintAllFramesTs": 31364564476, + "observedLargestContentfulPaint": 1907, + "observedLargestContentfulPaintTs": 31364564476, + "observedLargestContentfulPaintAllFrames": 1907, + "observedLargestContentfulPaintAllFramesTs": 31364564476, + "observedTraceEnd": 7051, + "observedTraceEndTs": 31369708833, + "observedLoad": 1759, + "observedLoadTs": 31364416347, + "observedDomContentLoaded": 1751, + "observedDomContentLoadedTs": 31364408638, + "observedCumulativeLayoutShift": 0, + "observedCumulativeLayoutShiftMainFrame": 0, + "observedFirstVisualChange": 1854, + "observedFirstVisualChangeTs": 31364511570, + "observedLastVisualChange": 5187, + "observedLastVisualChangeTs": 31367844570, + "observedSpeedIndex": 2191, + "observedSpeedIndexTs": 31364848523 + }, + { + "lcpInvalidated": false + } + ] + } + }, + "resource-summary": { + "id": "resource-summary", + "title": "Resources Summary", + "description": "Aggregates all network requests and groups them by type", + "score": 1, + "scoreDisplayMode": "informative", + "details": { + "type": "table", + "headings": [ + { + "key": "label", + "valueType": "text", + "label": "Resource Type" + }, + { + "key": "requestCount", + "valueType": "numeric", + "label": "Requests" + }, + { + "key": "transferSize", + "valueType": "bytes", + "label": "Transfer Size" + } + ], + "items": [ + { + "resourceType": "total", + "label": "Total", + "requestCount": 54, + "transferSize": 17480764 + }, + { + "resourceType": "script", + "label": "Script", + "requestCount": 23, + "transferSize": 16698518 + }, + { + "resourceType": "image", + "label": "Image", + "requestCount": 10, + "transferSize": 495441 + }, + { + "resourceType": "font", + "label": "Font", + "requestCount": 4, + "transferSize": 254886 + }, + { + "resourceType": "other", + "label": "Other", + "requestCount": 14, + "transferSize": 27638 + }, + { + "resourceType": "stylesheet", + "label": "Stylesheet", + "requestCount": 1, + "transferSize": 2318 + }, + { + "resourceType": "document", + "label": "Document", + "requestCount": 2, + "transferSize": 1963 + }, + { + "resourceType": "media", + "label": "Media", + "requestCount": 0, + "transferSize": 0 + }, + { + "resourceType": "third-party", + "label": "Third-party", + "requestCount": 21, + "transferSize": 57153 + } + ] + } + }, + "third-party-summary": { + "id": "third-party-summary", + "title": "Minimize third-party usage", + "description": "Third-party code can significantly impact load performance. Limit the number of redundant third-party providers and try to load third-party code after your page has primarily finished loading. [Learn how to minimize third-party impact](https://developers.google.com/web/fundamentals/performance/optimizing-content-efficiency/loading-third-party-javascript/).", + "score": 1, + "scoreDisplayMode": "informative", + "displayValue": "Third-party code blocked the main thread for 0 ms", + "metricSavings": { + "TBT": 0 + }, + "details": { + "type": "table", + "headings": [ + { + "key": "entity", + "valueType": "text", + "label": "Third-Party", + "subItemsHeading": { + "key": "url", + "valueType": "url" + } + }, + { + "key": "transferSize", + "granularity": 1, + "valueType": "bytes", + "label": "Transfer Size", + "subItemsHeading": { + "key": "transferSize" + } + }, + { + "key": "blockingTime", + "granularity": 1, + "valueType": "ms", + "label": "Main-Thread Blocking Time", + "subItemsHeading": { + "key": "blockingTime" + } + } + ], + "items": [ + { + "mainThreadTime": 1.65, + "blockingTime": 0, + "transferSize": 37018, + "tbtImpact": 0, + "entity": "Google Fonts", + "subItems": { + "type": "subitems", + "items": [ + { + "url": "https://fonts.gstatic.com/s/roboto/v47/KFO7CnqEu92Fr1ME7kSn66aGLdTylUAMa3yUBHMdazQ.woff2", + "mainThreadTime": 0, + "blockingTime": 0, + "transferSize": 34700, + "tbtImpact": 0 + }, + { + "url": "https://fonts.googleapis.com/css2?family=Roboto:wght@300;400;500;600;700", + "mainThreadTime": 1.65, + "blockingTime": 0, + "transferSize": 2318, + "tbtImpact": 0 + } + ] + } + }, + { + "mainThreadTime": 0.052000000000000005, + "blockingTime": 0, + "transferSize": 20135, + "tbtImpact": 0, + "entity": "walletconnect.com", + "subItems": { + "type": "subitems", + "items": [ + { + "url": "https://explorer-api.walletconnect.com/w3m/v1/getWalletImage/3d7eb880-7654-431f-ed84-a25712b45200?projectId=8702c5ea8d157774a011deb8e1433d01&sdkType=wcm&sdkVersion=js-2.6.2", + "mainThreadTime": 0, + "blockingTime": 0, + "transferSize": 2572, + "tbtImpact": 0 + }, + { + "url": "https://explorer-api.walletconnect.com/w3m/v1/getWalletImage/67336675-0daa-489b-6885-cb95234bc400?projectId=8702c5ea8d157774a011deb8e1433d01&sdkType=wcm&sdkVersion=js-2.6.2", + "mainThreadTime": 0, + "blockingTime": 0, + "transferSize": 2273, + "tbtImpact": 0 + }, + { + "url": "https://explorer-api.walletconnect.com/w3m/v1/getDesktopListings?projectId=8702c5ea8d157774a011deb8e1433d01&sdkType=wcm&sdkVersion=js-2.6.2&page=1&entries=9&version=2", + "mainThreadTime": 0, + "blockingTime": 0, + "transferSize": 2226, + "tbtImpact": 0 + }, + { + "url": "https://explorer-api.walletconnect.com/w3m/v1/getWalletImage/f5b26eef-c5e8-421a-e379-ae010b4a7400?projectId=8702c5ea8d157774a011deb8e1433d01&sdkType=wcm&sdkVersion=js-2.6.2", + "mainThreadTime": 0, + "blockingTime": 0, + "transferSize": 2097, + "tbtImpact": 0 + }, + { + "url": "https://explorer-api.walletconnect.com/w3m/v1/getWalletImage/4e7d6f52-f663-4fc1-4b88-eebe7fc72800?projectId=8702c5ea8d157774a011deb8e1433d01&sdkType=wcm&sdkVersion=js-2.6.2", + "mainThreadTime": 0, + "blockingTime": 0, + "transferSize": 2038, + "tbtImpact": 0 + }, + { + "url": "https://explorer-api.walletconnect.com/w3m/v1/getWalletImage/d8e930b6-ccde-471e-ecbe-6967b1c0c400?projectId=8702c5ea8d157774a011deb8e1433d01&sdkType=wcm&sdkVersion=js-2.6.2", + "mainThreadTime": 0, + "blockingTime": 0, + "transferSize": 1974, + "tbtImpact": 0 + }, + { + "url": "https://explorer-api.walletconnect.com/w3m/v1/getWalletImage/a578bd45-b418-4111-2c56-8ddcd1417c00?projectId=8702c5ea8d157774a011deb8e1433d01&sdkType=wcm&sdkVersion=js-2.6.2", + "mainThreadTime": 0, + "blockingTime": 0, + "transferSize": 1972, + "tbtImpact": 0 + }, + { + "url": "https://explorer-api.walletconnect.com/w3m/v1/getWalletImage/77c1d3dd-0213-400a-f9cc-bfd524c47f00?projectId=8702c5ea8d157774a011deb8e1433d01&sdkType=wcm&sdkVersion=js-2.6.2", + "mainThreadTime": 0, + "blockingTime": 0, + "transferSize": 1804, + "tbtImpact": 0 + }, + { + "url": "https://explorer-api.walletconnect.com/w3m/v1/getWalletImage/c20e1cec-05e8-4ac6-a086-7ce355092400?projectId=8702c5ea8d157774a011deb8e1433d01&sdkType=wcm&sdkVersion=js-2.6.2", + "mainThreadTime": 0, + "blockingTime": 0, + "transferSize": 1592, + "tbtImpact": 0 + }, + { + "url": "https://explorer-api.walletconnect.com/w3m/v1/getWalletImage/7e1514ba-932d-415d-1bdb-bccb6c2cbc00?projectId=8702c5ea8d157774a011deb8e1433d01&sdkType=wcm&sdkVersion=js-2.6.2", + "mainThreadTime": 0, + "blockingTime": 0, + "transferSize": 1309, + "tbtImpact": 0 + }, + { + "url": "https://verify.walletconnect.com/8702c5ea8d157774a011deb8e1433d01", + "mainThreadTime": 0.052000000000000005, + "blockingTime": 0, + "transferSize": 278, + "tbtImpact": 0 + } + ] + } + }, + { + "mainThreadTime": 0, + "blockingTime": 0, + "transferSize": 0, + "tbtImpact": 0, + "entity": "127.0.0.1", + "subItems": { + "type": "subitems", + "items": [ + { + "url": "http://127.0.0.1:8547/", + "mainThreadTime": 0, + "blockingTime": 0, + "transferSize": 0, + "tbtImpact": 0 + } + ] + } + } + ], + "summary": { + "wastedBytes": 57153, + "wastedMs": 0 + }, + "isEntityGrouped": true + }, + "guidanceLevel": 1 + }, + "third-party-facades": { + "id": "third-party-facades", + "title": "Lazy load third-party resources with facades", + "description": "Some third-party embeds can be lazy loaded. Consider replacing them with a facade until they are required. [Learn how to defer third-parties with a facade](https://developer.chrome.com/docs/lighthouse/performance/third-party-facades/).", + "score": null, + "scoreDisplayMode": "notApplicable", + "metricSavings": { + "TBT": 0 + }, + "guidanceLevel": 3 + }, + "largest-contentful-paint-element": { + "id": "largest-contentful-paint-element", + "title": "Largest Contentful Paint element", + "description": "This is the largest contentful element painted within the viewport. [Learn more about the Largest Contentful Paint element](https://developer.chrome.com/docs/lighthouse/performance/lighthouse-largest-contentful-paint/)", + "score": 0, + "scoreDisplayMode": "metricSavings", + "displayValue": "6,390 ms", + "metricSavings": { + "LCP": 5200 + }, + "details": { + "type": "list", + "items": [ + { + "type": "table", + "headings": [ + { + "key": "node", + "valueType": "node", + "label": "Element" + } + ], + "items": [ + { + "node": { + "type": "node", + "lhId": "page-0-IMG", + "path": "1,HTML,1,BODY,0,DIV,0,DIV,1,IMG", + "selector": "body > div#__next > div.relative > img.absolute", + "boundingRect": { + "top": 366, + "bottom": 940, + "left": 0, + "right": 1350, + "width": 1350, + "height": 574 + }, + "snippet": "\"grains\"", + "nodeLabel": "grains" + } + } + ] + }, + { + "type": "table", + "headings": [ + { + "key": "phase", + "valueType": "text", + "label": "Phase" + }, + { + "key": "percent", + "valueType": "text", + "label": "% of LCP" + }, + { + "key": "timing", + "valueType": "ms", + "label": "Timing" + } + ], + "items": [ + { + "phase": "TTFB", + "timing": 124.28725, + "percent": "2%" + }, + { + "phase": "Load Delay", + "timing": 5817.363777449029, + "percent": "91%" + }, + { + "phase": "Load Time", + "timing": 20.061570342651066, + "percent": "0%" + }, + { + "phase": "Render Delay", + "timing": 424.8531522083176, + "percent": "7%" + } + ] + } + ] + }, + "guidanceLevel": 1 + }, + "lcp-lazy-loaded": { + "id": "lcp-lazy-loaded", + "title": "Largest Contentful Paint image was lazily loaded", + "description": "Above-the-fold images that are lazily loaded render later in the page lifecycle, which can delay the largest contentful paint. [Learn more about optimal lazy loading](https://web.dev/articles/lcp-lazy-loading).", + "score": 0, + "scoreDisplayMode": "metricSavings", + "metricSavings": { + "LCP": 950 + }, + "details": { + "type": "table", + "headings": [ + { + "key": "node", + "valueType": "node", + "label": "Element" + } + ], + "items": [ + { + "node": { + "type": "node", + "lhId": "1-32-IMG", + "path": "1,HTML,1,BODY,0,DIV,0,DIV,1,IMG", + "selector": "body > div#__next > div.relative > img.absolute", + "boundingRect": { + "top": 366, + "bottom": 940, + "left": 0, + "right": 1350, + "width": 1350, + "height": 574 + }, + "snippet": "\"grains\"", + "nodeLabel": "grains" + } + } + ] + }, + "guidanceLevel": 3 + }, + "layout-shifts": { + "id": "layout-shifts", + "title": "Avoid large layout shifts", + "description": "These are the largest layout shifts observed on the page. Each table item represents a single layout shift, and shows the element that shifted the most. Below each item are possible root causes that led to the layout shift. Some of these layout shifts may not be included in the CLS metric value due to [windowing](https://web.dev/articles/cls#what_is_cls). [Learn how to improve CLS](https://web.dev/articles/optimize-cls)", + "score": null, + "scoreDisplayMode": "notApplicable", + "metricSavings": { + "CLS": 0 + }, + "details": { + "type": "table", + "headings": [], + "items": [] + }, + "guidanceLevel": 2 + }, + "long-tasks": { + "id": "long-tasks", + "title": "Avoid long main-thread tasks", + "description": "Lists the longest tasks on the main thread, useful for identifying worst contributors to input delay. [Learn how to avoid long main-thread tasks](https://web.dev/articles/optimize-long-tasks)", + "score": 1, + "scoreDisplayMode": "informative", + "displayValue": "5 long tasks found", + "metricSavings": { + "TBT": 2650 + }, + "details": { + "type": "table", + "headings": [ + { + "key": "url", + "valueType": "url", + "label": "URL" + }, + { + "key": "startTime", + "valueType": "ms", + "granularity": 1, + "label": "Start Time" + }, + { + "key": "duration", + "valueType": "ms", + "granularity": 1, + "label": "Duration" + } + ], + "items": [ + { + "url": "http://localhost:3000/_next/static/chunks/src_components_App_App_tsx.js", + "duration": 1196, + "startTime": 14034.225250000014 + }, + { + "url": "http://localhost:3000/_next/static/chunks/pages/_app.js", + "duration": 1169, + "startTime": 9261.277500000007 + }, + { + "url": "http://localhost:3000/_next/static/chunks/main.js", + "duration": 342, + "startTime": 4692.8007499999985 + }, + { + "url": "http://localhost:3000/_next/static/chunks/node_modules_walletconnect_ethereum-provider_dist_index_es_js.js", + "duration": 144, + "startTime": 15915.137250000013 + }, + { + "url": "http://localhost:3000/_next/static/chunks/node_modules_walletconnect_modal-ui_dist_index_js.js", + "duration": 76, + "startTime": 15230.225250000014 + } + ], + "sortedBy": ["duration"], + "skipSumming": ["startTime"], + "debugData": { + "type": "debugdata", + "urls": [ + "http://localhost:3000/_next/static/chunks/src_components_App_App_tsx.js", + "http://localhost:3000/_next/static/chunks/pages/_app.js", + "http://localhost:3000/_next/static/chunks/main.js", + "http://localhost:3000/_next/static/chunks/node_modules_walletconnect_ethereum-provider_dist_index_es_js.js", + "http://localhost:3000/_next/static/chunks/node_modules_walletconnect_modal-ui_dist_index_js.js" + ], + "tasks": [ + { + "urlIndex": 0, + "startTime": 14034.2, + "duration": 1196, + "other": 1196, + "scriptEvaluation": 0 + }, + { + "urlIndex": 1, + "startTime": 9261.3, + "duration": 1169, + "other": 1169, + "scriptEvaluation": 0 + }, + { + "urlIndex": 2, + "startTime": 4692.8, + "duration": 342, + "other": 342, + "scriptEvaluation": 0 + }, + { + "urlIndex": 3, + "startTime": 15915.1, + "duration": 144, + "other": 144, + "scriptEvaluation": 0 + }, + { + "urlIndex": 4, + "startTime": 15230.2, + "duration": 76, + "other": 76, + "scriptEvaluation": 0 + } + ] + } + }, + "guidanceLevel": 1 + }, + "non-composited-animations": { + "id": "non-composited-animations", + "title": "Avoid non-composited animations", + "description": "Animations which are not composited can be janky and increase CLS. [Learn how to avoid non-composited animations](https://developer.chrome.com/docs/lighthouse/performance/non-composited-animations/)", + "score": 1, + "scoreDisplayMode": "informative", + "displayValue": "4 animated elements found", + "metricSavings": { + "CLS": 0 + }, + "details": { + "type": "table", + "headings": [ + { + "key": "node", + "valueType": "node", + "subItemsHeading": { + "key": "failureReason", + "valueType": "text" + }, + "label": "Element" + }, + { + "key": null, + "valueType": "text", + "subItemsHeading": { + "key": "animation", + "valueType": "text" + }, + "label": "Name" + } + ], + "items": [ + { + "node": { + "type": "node", + "lhId": "page-1-DIV", + "path": "1,HTML,1,BODY,0,DIV,0,DIV,2,DIV,0,DIV,0,DIV,0,DIV,0,DIV,2,DIV,4,DIV,1,DIV", + "selector": "div.relative > div.z-10 > div.flex > div.flex", + "boundingRect": { + "top": 364, + "bottom": 364, + "left": 50, + "right": 237, + "width": 187, + "height": 0 + }, + "snippet": "
", + "nodeLabel": "div.relative > div.z-10 > div.flex > div.flex" + }, + "subItems": { + "type": "subitems", + "items": [ + { + "failureReason": "Unsupported CSS Property: scrollbar-color", + "animation": "scrollbar-color" + }, + { + "failureReason": "Unsupported CSS Property: scrollbar-color", + "animation": "scrollbar-color" + } + ] + } + }, + { + "node": { + "type": "node", + "lhId": "page-2-DIV", + "path": "1,HTML,1,BODY,0,DIV,0,DIV,2,DIV,0,DIV,0,DIV,0,DIV,0,DIV,2,DIV,1,DIV,1,DIV", + "selector": "div.relative > div.z-10 > div.flex > div.flex", + "boundingRect": { + "top": 199, + "bottom": 199, + "left": 50, + "right": 237, + "width": 187, + "height": 0 + }, + "snippet": "
", + "nodeLabel": "div.relative > div.z-10 > div.flex > div.flex" + }, + "subItems": { + "type": "subitems", + "items": [ + { + "failureReason": "Unsupported CSS Property: scrollbar-color", + "animation": "scrollbar-color" + }, + { + "failureReason": "Unsupported CSS Property: scrollbar-color", + "animation": "scrollbar-color" + } + ] + } + }, + { + "node": { + "type": "node", + "lhId": "page-3-DIV", + "path": "1,HTML,1,BODY,0,DIV,0,DIV,2,DIV,0,DIV,0,DIV,0,DIV,0,DIV,2,DIV,2,DIV,1,DIV", + "selector": "div.relative > div.z-10 > div.flex > div.flex", + "boundingRect": { + "top": 254, + "bottom": 254, + "left": 50, + "right": 237, + "width": 187, + "height": 0 + }, + "snippet": "
", + "nodeLabel": "div.relative > div.z-10 > div.flex > div.flex" + }, + "subItems": { + "type": "subitems", + "items": [ + { + "failureReason": "Unsupported CSS Property: scrollbar-color", + "animation": "scrollbar-color" + }, + { + "failureReason": "Unsupported CSS Property: scrollbar-color", + "animation": "scrollbar-color" + } + ] + } + }, + { + "node": { + "type": "node", + "lhId": "page-4-DIV", + "path": "1,HTML,1,BODY,0,DIV,0,DIV,2,DIV,0,DIV,0,DIV,0,DIV,0,DIV", + "selector": "div.relative > div.sticky > div.sticky > div.relative", + "boundingRect": { + "top": 0, + "bottom": 940, + "left": 0, + "right": 256, + "width": 256, + "height": 940 + }, + "snippet": "
", + "nodeLabel": "Home\nProjects\nChains\nBridge\nDevelopers\nCommunity\nGet Help\nThe Most Decentralize…" + }, + "subItems": { + "type": "subitems", + "items": [ + { + "failureReason": "Unsupported CSS Property: scrollbar-color", + "animation": "scrollbar-color" + } + ] + } + } + ] + }, + "guidanceLevel": 2 + }, + "unsized-images": { + "id": "unsized-images", + "title": "Image elements have explicit `width` and `height`", + "description": "Set an explicit width and height on image elements to reduce layout shifts and improve CLS. [Learn how to set image dimensions](https://web.dev/articles/optimize-cls#images_without_dimensions)", + "score": 1, + "scoreDisplayMode": "metricSavings", + "metricSavings": { + "CLS": 0 + }, + "details": { + "type": "table", + "headings": [], + "items": [] + }, + "guidanceLevel": 4 + }, + "valid-source-maps": { + "id": "valid-source-maps", + "title": "Missing source maps for large first-party JavaScript", + "description": "Source maps translate minified code to the original source code. This helps developers debug in production. In addition, Lighthouse is able to provide further insights. Consider deploying source maps to take advantage of these benefits. [Learn more about source maps](https://developer.chrome.com/docs/devtools/javascript/source-maps/).", + "score": 0, + "scoreDisplayMode": "binary", + "details": { + "type": "table", + "headings": [ + { + "key": "scriptUrl", + "valueType": "url", + "subItemsHeading": { + "key": "error" + }, + "label": "URL" + }, + { + "key": "sourceMapUrl", + "valueType": "url", + "label": "Map URL" + } + ], + "items": [ + { + "scriptUrl": "http://localhost:3000/_next/static/chunks/src_components_App_App_tsx.js", + "subItems": { + "type": "subitems", + "items": [ + { + "error": "Large JavaScript file is missing a source map" + } + ] + } + }, + { + "scriptUrl": "http://localhost:3000/_next/static/chunks/pages/index.js", + "subItems": { + "type": "subitems", + "items": [ + { + "error": "Large JavaScript file is missing a source map" + } + ] + } + }, + { + "scriptUrl": "http://localhost:3000/_next/static/chunks/pages/_app.js", + "subItems": { + "type": "subitems", + "items": [ + { + "error": "Large JavaScript file is missing a source map" + } + ] + } + }, + { + "scriptUrl": "http://localhost:3000/_next/static/chunks/node_modules_walletconnect_modal-ui_dist_index_js.js", + "subItems": { + "type": "subitems", + "items": [ + { + "error": "Large JavaScript file is missing a source map" + } + ] + } + }, + { + "scriptUrl": "http://localhost:3000/_next/static/chunks/node_modules_walletconnect_ethereum-provider_dist_index_es_js.js", + "subItems": { + "type": "subitems", + "items": [ + { + "error": "Large JavaScript file is missing a source map" + } + ] + } + }, + { + "scriptUrl": "http://localhost:3000/_next/static/chunks/node_modules_offchainlabs_cobalt_dist_cobalt_es_js.js", + "subItems": { + "type": "subitems", + "items": [ + { + "error": "Large JavaScript file is missing a source map" + } + ] + } + }, + { + "scriptUrl": "http://localhost:3000/_next/static/chunks/node_modules_coinbase_wallet-sdk_dist_index_js.js", + "subItems": { + "type": "subitems", + "items": [ + { + "error": "Large JavaScript file is missing a source map" + } + ] + } + }, + { + "scriptUrl": "http://localhost:3000/_next/static/chunks/main.js", + "subItems": { + "type": "subitems", + "items": [ + { + "error": "Large JavaScript file is missing a source map" + } + ] + } + } + ] + } + }, + "prioritize-lcp-image": { + "id": "prioritize-lcp-image", + "title": "Preload Largest Contentful Paint image", + "description": "If the LCP element is dynamically added to the page, you should preload the image in order to improve LCP. [Learn more about preloading LCP elements](https://web.dev/articles/optimize-lcp#optimize_when_the_resource_is_discovered).", + "score": 1, + "scoreDisplayMode": "metricSavings", + "numericValue": 0, + "numericUnit": "millisecond", + "displayValue": "", + "metricSavings": { + "LCP": 0 + }, + "details": { + "type": "opportunity", + "headings": [], + "items": [], + "overallSavingsMs": 0, + "sortedBy": ["wastedMs"], + "debugData": { + "type": "debugdata", + "initiatorPath": [ + { + "url": "http://localhost:3000/_next/image?url=%2F_next%2Fstatic%2Fmedia%2Feclipse_bottom.4304bd37.png&w=1920&q=75", + "initiatorType": "fallbackToMain" + }, + { + "url": "http://localhost:3000/?sourceChain=ethereum&destinationChain=arbitrum-one&tab=bridge&txHistory=0", + "initiatorType": "other" + } + ], + "pathLength": 2 + } + }, + "guidanceLevel": 4 + }, + "csp-xss": { + "id": "csp-xss", + "title": "Ensure CSP is effective against XSS attacks", + "description": "A strong Content Security Policy (CSP) significantly reduces the risk of cross-site scripting (XSS) attacks. [Learn how to use a CSP to prevent XSS](https://developer.chrome.com/docs/lighthouse/best-practices/csp-xss/)", + "score": 1, + "scoreDisplayMode": "informative", + "details": { + "type": "table", + "headings": [ + { + "key": "description", + "valueType": "text", + "subItemsHeading": { + "key": "description" + }, + "label": "Description" + }, + { + "key": "directive", + "valueType": "code", + "subItemsHeading": { + "key": "directive" + }, + "label": "Directive" + }, + { + "key": "severity", + "valueType": "text", + "subItemsHeading": { + "key": "severity" + }, + "label": "Severity" + } + ], + "items": [ + { + "severity": "High", + "description": "No CSP found in enforcement mode" + } + ] + } + }, + "has-hsts": { + "id": "has-hsts", + "title": "Use a strong HSTS policy", + "description": "Deployment of the HSTS header significantly reduces the risk of downgrading HTTP connections and eavesdropping attacks. A rollout in stages, starting with a low max-age is recommended. [Learn more about using a strong HSTS policy.](https://developer.chrome.com/docs/lighthouse/best-practices/has-hsts)", + "score": 1, + "scoreDisplayMode": "informative", + "details": { + "type": "table", + "headings": [ + { + "key": "description", + "valueType": "text", + "subItemsHeading": { + "key": "description" + }, + "label": "Description" + }, + { + "key": "directive", + "valueType": "code", + "subItemsHeading": { + "key": "directive" + }, + "label": "Directive" + }, + { + "key": "severity", + "valueType": "text", + "subItemsHeading": { + "key": "severity" + }, + "label": "Severity" + } + ], + "items": [ + { + "severity": "High", + "description": "No HSTS header found" + } + ] + } + }, + "origin-isolation": { + "id": "origin-isolation", + "title": "Ensure proper origin isolation with COOP", + "description": "The Cross-Origin-Opener-Policy (COOP) can be used to isolate the top-level window from other documents such as pop-ups. [Learn more about deploying the COOP header.](https://web.dev/articles/why-coop-coep#coop)", + "score": 1, + "scoreDisplayMode": "informative", + "details": { + "type": "table", + "headings": [ + { + "key": "description", + "valueType": "text", + "subItemsHeading": { + "key": "description" + }, + "label": "Description" + }, + { + "key": "directive", + "valueType": "code", + "subItemsHeading": { + "key": "directive" + }, + "label": "Directive" + }, + { + "key": "severity", + "valueType": "text", + "subItemsHeading": { + "key": "severity" + }, + "label": "Severity" + } + ], + "items": [ + { + "description": "No COOP header found", + "severity": "High" + } + ] + } + }, + "clickjacking-mitigation": { + "id": "clickjacking-mitigation", + "title": "Mitigate clickjacking with XFO or CSP", + "description": "The `X-Frame-Options` (XFO) header or the `frame-ancestors` directive in the `Content-Security-Policy` (CSP) header control where a page can be embedded. These can mitigate clickjacking attacks by blocking some or all sites from embedding the page. [Learn more about mitigating clickjacking](https://developer.chrome.com/docs/lighthouse/best-practices/clickjacking-mitigation).", + "score": 1, + "scoreDisplayMode": "informative", + "details": { + "type": "table", + "headings": [ + { + "key": "description", + "valueType": "text", + "subItemsHeading": { + "key": "description" + }, + "label": "Description" + }, + { + "key": "severity", + "valueType": "text", + "subItemsHeading": { + "key": "severity" + }, + "label": "Severity" + } + ], + "items": [ + { + "severity": "High", + "description": "No frame control policy found" + } + ] + } + }, + "script-treemap-data": { + "id": "script-treemap-data", + "title": "Script Treemap Data", + "description": "Used for treemap app", + "score": 1, + "scoreDisplayMode": "informative", + "details": { + "type": "treemap-data", + "nodes": [ + { + "name": "http://localhost:3000/_next/static/chunks/react-refresh.js", + "resourceBytes": 79107, + "encodedBytes": 25063, + "unusedBytes": 0 + }, + { + "name": "http://localhost:3000/_next/static/chunks/webpack.js", + "resourceBytes": 53011, + "encodedBytes": 10015, + "unusedBytes": 28078 + }, + { + "name": "http://localhost:3000/_next/static/chunks/main.js", + "resourceBytes": 5287916, + "encodedBytes": 1176778, + "unusedBytes": 23699 + }, + { + "name": "http://localhost:3000/_next/static/chunks/pages/_app.js", + "resourceBytes": 18915166, + "encodedBytes": 3979485, + "unusedBytes": 83488 + }, + { + "name": "http://localhost:3000/_next/static/chunks/pages/index.js", + "resourceBytes": 895933, + "encodedBytes": 196084, + "unusedBytes": 0 + }, + { + "name": "http://localhost:3000/_next/static/development/_buildManifest.js", + "resourceBytes": 296, + "encodedBytes": 296, + "unusedBytes": 0 + }, + { + "name": "http://localhost:3000/_next/static/development/_ssgManifest.js", + "resourceBytes": 76, + "encodedBytes": 76, + "unusedBytes": 0 + }, + { + "name": "http://localhost:3000/_next/static/chunks/node_modules_offchainlabs_cobalt_dist_cobalt_es_js.js", + "resourceBytes": 576077, + "encodedBytes": 147047, + "unusedBytes": 0 + }, + { + "name": "http://localhost:3000/_next/static/chunks/src_components_App_App_tsx.js", + "resourceBytes": 43336048, + "encodedBytes": 9942112, + "unusedBytes": 574631 + }, + { + "name": "http://localhost:3000/_next/static/chunks/node_modules_walletconnect_ethereum-provider_dist_index_es_js.js", + "resourceBytes": 2332522, + "encodedBytes": 638797, + "unusedBytes": 1282 + }, + { + "name": "http://localhost:3000/_next/static/chunks/node_modules_rainbow-me_rainbowkit_dist_safeWallet-VUYZPLY4_js.js", + "resourceBytes": 5674, + "encodedBytes": 2897, + "unusedBytes": 0 + }, + { + "name": "http://localhost:3000/_next/static/chunks/node_modules_rainbow-me_rainbowkit_dist_rainbowWallet-2SR6TVBF_js.js", + "resourceBytes": 11347, + "encodedBytes": 3707, + "unusedBytes": 0 + }, + { + "name": "http://localhost:3000/_next/static/chunks/node_modules_rainbow-me_rainbowkit_dist_coinbaseWallet-WWX6LF36_js.js", + "resourceBytes": 3751, + "encodedBytes": 1933, + "unusedBytes": 0 + }, + { + "name": "http://localhost:3000/_next/static/chunks/node_modules_rainbow-me_rainbowkit_dist_metaMaskWallet-YFHEHW7V_js.js", + "resourceBytes": 18353, + "encodedBytes": 4611, + "unusedBytes": 0 + }, + { + "name": "http://localhost:3000/_next/static/chunks/node_modules_rainbow-me_rainbowkit_dist_walletConnectWallet-FNSU4KNU_js.js", + "resourceBytes": 5968, + "encodedBytes": 2946, + "unusedBytes": 0 + }, + { + "name": "http://localhost:3000/_next/static/chunks/node_modules_rainbow-me_rainbowkit_dist_wallets_walletConnectors_okxWallet-GJMKZIND_js.js", + "resourceBytes": 4565, + "encodedBytes": 2068, + "unusedBytes": 0 + }, + { + "name": "http://localhost:3000/_next/static/chunks/node_modules_rainbow-me_rainbowkit_dist_wallets_walletConnectors_trustWallet-E2GVGE4U_js.js", + "resourceBytes": 4371, + "encodedBytes": 2140, + "unusedBytes": 0 + }, + { + "name": "http://localhost:3000/_next/static/chunks/node_modules_rainbow-me_rainbowkit_dist_wallets_walletConnectors_rabbyWallet-FLVUU35F_js.js", + "resourceBytes": 8269, + "encodedBytes": 3961, + "unusedBytes": 0 + }, + { + "name": "http://localhost:3000/_next/static/chunks/node_modules_rainbow-me_rainbowkit_dist_assets-NU2OP443_js.js", + "resourceBytes": 45515, + "encodedBytes": 12100, + "unusedBytes": 0 + }, + { + "name": "http://localhost:3000/_next/static/chunks/node_modules_rainbow-me_rainbowkit_dist_login-CWDTIDNK_js.js", + "resourceBytes": 24218, + "encodedBytes": 9328, + "unusedBytes": 0 + }, + { + "name": "http://localhost:3000/_next/static/chunks/node_modules_coinbase_wallet-sdk_dist_index_js.js", + "resourceBytes": 657123, + "encodedBytes": 172991, + "unusedBytes": 0 + }, + { + "name": "http://localhost:3000/_next/static/chunks/node_modules_walletconnect_modal_dist_index_js.js", + "resourceBytes": 63923, + "encodedBytes": 22001, + "unusedBytes": 0 + }, + { + "name": "http://localhost:3000/_next/static/chunks/node_modules_walletconnect_modal-ui_dist_index_js.js", + "resourceBytes": 1321329, + "encodedBytes": 333646, + "unusedBytes": 0 + } + ] + } + }, + "accesskeys": { + "id": "accesskeys", + "title": "`[accesskey]` values are unique", + "description": "Access keys let users quickly focus a part of the page. For proper navigation, each access key must be unique. [Learn more about access keys](https://dequeuniversity.com/rules/axe/4.10/accesskeys).", + "score": null, + "scoreDisplayMode": "notApplicable" + }, + "aria-allowed-attr": { + "id": "aria-allowed-attr", + "title": "`[aria-*]` attributes match their roles", + "description": "Each ARIA `role` supports a specific subset of `aria-*` attributes. Mismatching these invalidates the `aria-*` attributes. [Learn how to match ARIA attributes to their roles](https://dequeuniversity.com/rules/axe/4.10/aria-allowed-attr).", + "score": 1, + "scoreDisplayMode": "binary", + "details": { + "type": "table", + "headings": [], + "items": [] + } + }, + "aria-allowed-role": { + "id": "aria-allowed-role", + "title": "Uses ARIA roles only on compatible elements", + "description": "Many HTML elements can only be assigned certain ARIA roles. Using ARIA roles where they are not allowed can interfere with the accessibility of the web page. [Learn more about ARIA roles](https://dequeuniversity.com/rules/axe/4.10/aria-allowed-role).", + "score": 1, + "scoreDisplayMode": "binary", + "details": { + "type": "table", + "headings": [], + "items": [] + } + }, + "aria-command-name": { + "id": "aria-command-name", + "title": "`button`, `link`, and `menuitem` elements have accessible names", + "description": "When an element doesn't have an accessible name, screen readers announce it with a generic name, making it unusable for users who rely on screen readers. [Learn how to make command elements more accessible](https://dequeuniversity.com/rules/axe/4.10/aria-command-name).", + "score": null, + "scoreDisplayMode": "notApplicable" + }, + "aria-conditional-attr": { + "id": "aria-conditional-attr", + "title": "ARIA attributes are used as specified for the element's role", + "description": "Some ARIA attributes are only allowed on an element under certain conditions. [Learn more about conditional ARIA attributes](https://dequeuniversity.com/rules/axe/4.10/aria-conditional-attr).", + "score": 1, + "scoreDisplayMode": "binary", + "details": { + "type": "table", + "headings": [], + "items": [] + } + }, + "aria-deprecated-role": { + "id": "aria-deprecated-role", + "title": "Deprecated ARIA roles were not used", + "description": "Deprecated ARIA roles may not be processed correctly by assistive technology. [Learn more about deprecated ARIA roles](https://dequeuniversity.com/rules/axe/4.10/aria-deprecated-role).", + "score": 1, + "scoreDisplayMode": "binary", + "details": { + "type": "table", + "headings": [], + "items": [] + } + }, + "aria-dialog-name": { + "id": "aria-dialog-name", + "title": "Elements with `role=\"dialog\"` or `role=\"alertdialog\"` do not have accessible names.", + "description": "ARIA dialog elements without accessible names may prevent screen readers users from discerning the purpose of these elements. [Learn how to make ARIA dialog elements more accessible](https://dequeuniversity.com/rules/axe/4.10/aria-dialog-name).", + "score": 0, + "scoreDisplayMode": "binary", + "details": { + "type": "table", + "headings": [ + { + "key": "node", + "valueType": "node", + "subItemsHeading": { + "key": "relatedNode", + "valueType": "node" + }, + "label": "Failing Elements" + } + ], + "items": [ + { + "node": { + "type": "node", + "lhId": "1-0-DIV", + "path": "1,HTML,1,BODY,7,WCM-MODAL,a,#document-fragment,3,DIV", + "selector": "div#wcm-modal", + "boundingRect": { + "top": 0, + "bottom": 940, + "left": 0, + "right": 1350, + "width": 1350, + "height": 940 + }, + "snippet": "
", + "nodeLabel": "div#wcm-modal", + "explanation": "Fix any of the following:\n aria-label attribute does not exist or is empty\n aria-labelledby attribute does not exist, references elements that do not exist or references elements that are empty\n Element has no title attribute" + } + } + ], + "debugData": { + "type": "debugdata", + "impact": "serious", + "tags": ["cat.aria", "best-practice"] + } + } + }, + "aria-hidden-body": { + "id": "aria-hidden-body", + "title": "`[aria-hidden=\"true\"]` is not present on the document ``", + "description": "Assistive technologies, like screen readers, work inconsistently when `aria-hidden=\"true\"` is set on the document ``. [Learn how `aria-hidden` affects the document body](https://dequeuniversity.com/rules/axe/4.10/aria-hidden-body).", + "score": 1, + "scoreDisplayMode": "binary", + "details": { + "type": "table", + "headings": [], + "items": [] + } + }, + "aria-hidden-focus": { + "id": "aria-hidden-focus", + "title": "`[aria-hidden=\"true\"]` elements do not contain focusable descendents", + "description": "Focusable descendents within an `[aria-hidden=\"true\"]` element prevent those interactive elements from being available to users of assistive technologies like screen readers. [Learn how `aria-hidden` affects focusable elements](https://dequeuniversity.com/rules/axe/4.10/aria-hidden-focus).", + "score": 1, + "scoreDisplayMode": "binary", + "details": { + "type": "table", + "headings": [], + "items": [] + } + }, + "aria-input-field-name": { + "id": "aria-input-field-name", + "title": "ARIA input fields have accessible names", + "description": "When an input field doesn't have an accessible name, screen readers announce it with a generic name, making it unusable for users who rely on screen readers. [Learn more about input field labels](https://dequeuniversity.com/rules/axe/4.10/aria-input-field-name).", + "score": null, + "scoreDisplayMode": "notApplicable" + }, + "aria-meter-name": { + "id": "aria-meter-name", + "title": "ARIA `meter` elements have accessible names", + "description": "When a meter element doesn't have an accessible name, screen readers announce it with a generic name, making it unusable for users who rely on screen readers. [Learn how to name `meter` elements](https://dequeuniversity.com/rules/axe/4.10/aria-meter-name).", + "score": null, + "scoreDisplayMode": "notApplicable" + }, + "aria-progressbar-name": { + "id": "aria-progressbar-name", + "title": "ARIA `progressbar` elements have accessible names", + "description": "When a `progressbar` element doesn't have an accessible name, screen readers announce it with a generic name, making it unusable for users who rely on screen readers. [Learn how to label `progressbar` elements](https://dequeuniversity.com/rules/axe/4.10/aria-progressbar-name).", + "score": null, + "scoreDisplayMode": "notApplicable" + }, + "aria-prohibited-attr": { + "id": "aria-prohibited-attr", + "title": "Elements use only permitted ARIA attributes", + "description": "Using ARIA attributes in roles where they are prohibited can mean that important information is not communicated to users of assistive technologies. [Learn more about prohibited ARIA roles](https://dequeuniversity.com/rules/axe/4.10/aria-prohibited-attr).", + "score": 1, + "scoreDisplayMode": "binary", + "details": { + "type": "table", + "headings": [], + "items": [] + } + }, + "aria-required-attr": { + "id": "aria-required-attr", + "title": "`[role]`s have all required `[aria-*]` attributes", + "description": "Some ARIA roles have required attributes that describe the state of the element to screen readers. [Learn more about roles and required attributes](https://dequeuniversity.com/rules/axe/4.10/aria-required-attr).", + "score": 1, + "scoreDisplayMode": "binary", + "details": { + "type": "table", + "headings": [], + "items": [] + } + }, + "aria-required-children": { + "id": "aria-required-children", + "title": "Elements with an ARIA `[role]` that require children to contain a specific `[role]` have all required children.", + "description": "Some ARIA parent roles must contain specific child roles to perform their intended accessibility functions. [Learn more about roles and required children elements](https://dequeuniversity.com/rules/axe/4.10/aria-required-children).", + "score": null, + "scoreDisplayMode": "notApplicable" + }, + "aria-required-parent": { + "id": "aria-required-parent", + "title": "`[role]`s are contained by their required parent element", + "description": "Some ARIA child roles must be contained by specific parent roles to properly perform their intended accessibility functions. [Learn more about ARIA roles and required parent element](https://dequeuniversity.com/rules/axe/4.10/aria-required-parent).", + "score": null, + "scoreDisplayMode": "notApplicable" + }, + "aria-roles": { + "id": "aria-roles", + "title": "`[role]` values are valid", + "description": "ARIA roles must have valid values in order to perform their intended accessibility functions. [Learn more about valid ARIA roles](https://dequeuniversity.com/rules/axe/4.10/aria-roles).", + "score": 1, + "scoreDisplayMode": "binary", + "details": { + "type": "table", + "headings": [], + "items": [] + } + }, + "aria-text": { + "id": "aria-text", + "title": "Elements with the `role=text` attribute do not have focusable descendents.", + "description": "Adding `role=text` around a text node split by markup enables VoiceOver to treat it as one phrase, but the element's focusable descendents will not be announced. [Learn more about the `role=text` attribute](https://dequeuniversity.com/rules/axe/4.10/aria-text).", + "score": null, + "scoreDisplayMode": "notApplicable" + }, + "aria-toggle-field-name": { + "id": "aria-toggle-field-name", + "title": "ARIA toggle fields have accessible names", + "description": "When a toggle field doesn't have an accessible name, screen readers announce it with a generic name, making it unusable for users who rely on screen readers. [Learn more about toggle fields](https://dequeuniversity.com/rules/axe/4.10/aria-toggle-field-name).", + "score": null, + "scoreDisplayMode": "notApplicable" + }, + "aria-tooltip-name": { + "id": "aria-tooltip-name", + "title": "ARIA `tooltip` elements have accessible names", + "description": "When a tooltip element doesn't have an accessible name, screen readers announce it with a generic name, making it unusable for users who rely on screen readers. [Learn how to name `tooltip` elements](https://dequeuniversity.com/rules/axe/4.10/aria-tooltip-name).", + "score": null, + "scoreDisplayMode": "notApplicable" + }, + "aria-treeitem-name": { + "id": "aria-treeitem-name", + "title": "ARIA `treeitem` elements have accessible names", + "description": "When a `treeitem` element doesn't have an accessible name, screen readers announce it with a generic name, making it unusable for users who rely on screen readers. [Learn more about labeling `treeitem` elements](https://dequeuniversity.com/rules/axe/4.10/aria-treeitem-name).", + "score": null, + "scoreDisplayMode": "notApplicable" + }, + "aria-valid-attr-value": { + "id": "aria-valid-attr-value", + "title": "`[aria-*]` attributes have valid values", + "description": "Assistive technologies, like screen readers, can't interpret ARIA attributes with invalid values. [Learn more about valid values for ARIA attributes](https://dequeuniversity.com/rules/axe/4.10/aria-valid-attr-value).", + "score": 1, + "scoreDisplayMode": "binary", + "details": { + "type": "table", + "headings": [], + "items": [] + } + }, + "aria-valid-attr": { + "id": "aria-valid-attr", + "title": "`[aria-*]` attributes are valid and not misspelled", + "description": "Assistive technologies, like screen readers, can't interpret ARIA attributes with invalid names. [Learn more about valid ARIA attributes](https://dequeuniversity.com/rules/axe/4.10/aria-valid-attr).", + "score": 1, + "scoreDisplayMode": "binary", + "details": { + "type": "table", + "headings": [], + "items": [] + } + }, + "button-name": { + "id": "button-name", + "title": "Buttons do not have an accessible name", + "description": "When a button doesn't have an accessible name, screen readers announce it as \"button\", making it unusable for users who rely on screen readers. [Learn how to make buttons more accessible](https://dequeuniversity.com/rules/axe/4.10/button-name).", + "score": 0, + "scoreDisplayMode": "binary", + "details": { + "type": "table", + "headings": [ + { + "key": "node", + "valueType": "node", + "subItemsHeading": { + "key": "relatedNode", + "valueType": "node" + }, + "label": "Failing Elements" + } + ], + "items": [ + { + "node": { + "type": "node", + "lhId": "1-1-BUTTON", + "path": "1,HTML,1,BODY,0,DIV,0,DIV,2,DIV,0,DIV,0,DIV,0,DIV,0,DIV,0,BUTTON", + "selector": "div.sticky > div.sticky > div.relative > button.absolute", + "boundingRect": { + "top": 60, + "bottom": 92, + "left": 239, + "right": 271, + "width": 32, + "height": 32 + }, + "snippet": "