diff --git a/EcoSonar-API/dataBase/lighthouseRepository.js b/EcoSonar-API/dataBase/lighthouseRepository.js index 30f9ae7..3f0821a 100644 --- a/EcoSonar-API/dataBase/lighthouseRepository.js +++ b/EcoSonar-API/dataBase/lighthouseRepository.js @@ -180,7 +180,8 @@ const LighthouseRepository = function () { speedIndex: 1, totalBlockingTime: 1, interactive: 1, - dateLighthouseAnalysis: 1 + dateLighthouseAnalysis: 1, + mobile: 1 } ) .sort({ dateLighthouseAnalysis: 1 }) diff --git a/EcoSonar-API/dataBase/models/lighthouses.js b/EcoSonar-API/dataBase/models/lighthouses.js index 9a2b290..1de8fe0 100644 --- a/EcoSonar-API/dataBase/models/lighthouses.js +++ b/EcoSonar-API/dataBase/models/lighthouses.js @@ -1,7 +1,7 @@ const mongoose = require('mongoose') const Schema = mongoose.Schema -const lighthouseSchema = new Schema({ +const reportSchema = { idLighthouseAnalysis: { type: String, required: true, @@ -119,6 +119,14 @@ const lighthouseSchema = new Schema({ required: true } } +} + +const lighthouseSchema = new Schema({ + ...reportSchema, + mobile: { + type: reportSchema, + required: false + } }) const lighthouse = mongoose.model('lighthouses', lighthouseSchema) diff --git a/EcoSonar-API/jest.config.js b/EcoSonar-API/jest.config.js new file mode 100644 index 0000000..272807e --- /dev/null +++ b/EcoSonar-API/jest.config.js @@ -0,0 +1,198 @@ +/** + * For a detailed explanation regarding each configuration property, visit: + * https://jestjs.io/docs/configuration + */ + +/** @type {import('jest').Config} */ +const config = { + // All imported modules in your tests should be mocked automatically + // automock: false, + + // Stop running tests after `n` failures + // bail: 0, + + // The directory where Jest should store its cached dependency information + // cacheDirectory: "/tmp/jest_rs", + + // Automatically clear mock calls, instances, contexts and results before every test + clearMocks: true, + + // Indicates whether the coverage information should be collected while executing the test + // collectCoverage: false, + + // An array of glob patterns indicating a set of files for which coverage information should be collected + // collectCoverageFrom: undefined, + + // The directory where Jest should output its coverage files + // coverageDirectory: undefined, + + // An array of regexp pattern strings used to skip coverage collection + // coveragePathIgnorePatterns: [ + // "/node_modules/" + // ], + + // Indicates which provider should be used to instrument code for coverage + coverageProvider: "v8", + + // A list of reporter names that Jest uses when writing coverage reports + // coverageReporters: [ + // "json", + // "text", + // "lcov", + // "clover" + // ], + + // An object that configures minimum threshold enforcement for coverage results + // coverageThreshold: undefined, + + // A path to a custom dependency extractor + // dependencyExtractor: undefined, + + // Make calling deprecated APIs throw helpful error messages + // errorOnDeprecated: false, + + // The default configuration for fake timers + // fakeTimers: { + // "enableGlobally": false + // }, + + // Force coverage collection from ignored files using an array of glob patterns + // forceCoverageMatch: [], + + // A path to a module which exports an async function that is triggered once before all test suites + // globalSetup: undefined, + + // A path to a module which exports an async function that is triggered once after all test suites + // globalTeardown: undefined, + + // A set of global variables that need to be available in all test environments + // globals: {}, + + // The maximum amount of workers used to run your tests. Can be specified as % or a number. E.g. maxWorkers: 10% will use 10% of your CPU amount + 1 as the maximum worker number. maxWorkers: 2 will use a maximum of 2 workers. + // maxWorkers: "50%", + + // An array of directory names to be searched recursively up from the requiring module's location + // moduleDirectories: [ + // "node_modules" + // ], + + // An array of file extensions your modules use + // moduleFileExtensions: [ + // "js", + // "mjs", + // "cjs", + // "jsx", + // "ts", + // "tsx", + // "json", + // "node" + // ], + + // A map from regular expressions to module names or to arrays of module names that allow to stub out resources with a single module + // moduleNameMapper: {}, + + // An array of regexp pattern strings, matched against all module paths before considered 'visible' to the module loader + // modulePathIgnorePatterns: [], + + // Activates notifications for test results + // notify: false, + + // An enum that specifies notification mode. Requires { notify: true } + // notifyMode: "failure-change", + + // A preset that is used as a base for Jest's configuration + // preset: undefined, + + // Run tests from one or more projects + // projects: undefined, + + // Use this configuration option to add custom reporters to Jest + // reporters: undefined, + + // Automatically reset mock state before every test + // resetMocks: false, + + // Reset the module registry before running each individual test + // resetModules: false, + + // A path to a custom resolver + // resolver: undefined, + + // Automatically restore mock state and implementation before every test + // restoreMocks: false, + + // The root directory that Jest should scan for tests and modules within + // rootDir: undefined, + + // A list of paths to directories that Jest should use to search for files in + // roots: [ + // "" + // ], + + // Allows you to use a custom runner instead of Jest's default test runner + // runner: "jest-runner", + + // The paths to modules that run some code to configure or set up the testing environment before each test + // setupFiles: [], + + // A list of paths to modules that run some code to configure or set up the testing framework before each test + // setupFilesAfterEnv: [], + + // The number of seconds after which a test is considered as slow and reported as such in the results. + // slowTestThreshold: 5, + + // A list of paths to snapshot serializer modules Jest should use for snapshot testing + // snapshotSerializers: [], + + // The test environment that will be used for testing + // testEnvironment: "jest-environment-node", + + // Options that will be passed to the testEnvironment + // testEnvironmentOptions: {}, + + // Adds a location field to test results + // testLocationInResults: false, + + // The glob patterns Jest uses to detect test files + // testMatch: [ + // "**/__tests__/**/*.[jt]s?(x)", + // "**/?(*.)+(spec|test).[tj]s?(x)" + // ], + + // An array of regexp pattern strings that are matched against all test paths, matched tests are skipped + // testPathIgnorePatterns: [ + // "/node_modules/" + // ], + + // The regexp pattern or array of patterns that Jest uses to detect test files + // testRegex: [], + + // This option allows the use of a custom results processor + // testResultsProcessor: undefined, + + // This option allows use of a custom test runner + // testRunner: "jest-circus/runner", + + // A map from regular expressions to paths to transformers + // transform: undefined, + + // An array of regexp pattern strings that are matched against all source file paths, matched files will skip transformation + // transformIgnorePatterns: [ + // "/node_modules/", + // "\\.pnp\\.[^\\/]+$" + // ], + + // An array of regexp pattern strings that are matched against all modules before the module loader will automatically return a mock for them + // unmockedModulePathPatterns: undefined, + + // Indicates whether each individual test should be reported during the run + // verbose: undefined, + + // An array of regexp patterns that are matched against all source file paths before re-running tests in watch mode + // watchPathIgnorePatterns: [], + + // Whether to use watchman for file crawling + // watchman: true, +}; + +module.exports = config; diff --git a/EcoSonar-API/package.json b/EcoSonar-API/package.json index e75d895..1086280 100644 --- a/EcoSonar-API/package.json +++ b/EcoSonar-API/package.json @@ -9,7 +9,7 @@ "main": "app.js", "scripts": { "postinstall": "node builder.js", - "test": "echo \"Error: no test specified\" && exit 1", + "test": "jest", "build": "node builder.js", "start": "nodemon server.js", "lint": "eslint --fix --ext .js ." @@ -45,6 +45,7 @@ "eslint-plugin-n": "^15.6.1", "eslint-plugin-node": "^11.1.0", "eslint-plugin-promise": "^6.1.1", + "jest": "^29.7.0", "nodemon": "^3.0.1" } } diff --git a/EcoSonar-API/routes/app.js b/EcoSonar-API/routes/app.js index 671524b..bd7d606 100644 --- a/EcoSonar-API/routes/app.js +++ b/EcoSonar-API/routes/app.js @@ -2,7 +2,7 @@ const express = require('express') const cors = require('cors') const dotenv = require('dotenv') const urlConfigurationService = require('../services/urlConfigurationService') -const analysisService = require('../services/analysisService') +const { analysisService } = require('../services/analysisService') const retrieveAnalysisService = require('../services/retrieveAnalysisService') const retrieveBestPracticesService = require('../services/retrieveBestPracticesService') const crawlerService = require('../services/crawler/crawlerService') diff --git a/EcoSonar-API/services/analysisService.js b/EcoSonar-API/services/analysisService.js index 16e167d..960c378 100644 --- a/EcoSonar-API/services/analysisService.js +++ b/EcoSonar-API/services/analysisService.js @@ -12,68 +12,69 @@ const w3cRepository = require('../dataBase/w3cRepository') const formatW3cBestPractices = require('./format/formatW3cBestPractices') const formatW3cAnalysis = require('./format/formatW3cAnalysis') -class AnalysisService {} - -/** - * Insert a new analysis into database - * @param {string} projectName - * @param {boolean} autoscroll is used to enable autoscrolling for each tab opened during analysis - */ -AnalysisService.prototype.insert = async function (projectName, autoscroll) { - const allowExternalAPI = process.env.ECOSONAR_ENV_ALLOW_EXTERNAL_API - let urlProjectList = [] - let reports = [] - let systemError = false +class AnalysisService { + + /** + * Insert a new analysis into database + * @param {string} projectName + * @param {boolean} autoscroll is used to enable autoscrolling for each tab opened during analysis + */ + async insert (projectName, autoscroll) { + const allowExternalAPI = process.env.ECOSONAR_ENV_ALLOW_EXTERNAL_API + let urlProjectList = [] + let reports = [] + let systemError = false - try { - urlProjectList = await urlsProjectRepository.findAll(projectName, true) - } catch (error) { - console.log('GREENIT INSERT - can not retrieved urls from project') - systemError = true - } - - if (systemError || urlProjectList.length === 0) { - console.log('GREENIT INSERT - project has no url to do the audit. Audit stopped') - } else { - reports = await launchAuditsToUrlList(urlProjectList, autoscroll, projectName, allowExternalAPI) - const reportsFormatted = formatAuditsToBeSaved(reports, urlProjectList) - - greenItRepository - .insertAll(reportsFormatted.greenitAnalysisFormatted) - .then(() => { - console.log('GREENIT INSERT - analysis has been insert') - }) - .catch(() => { - console.log('GREENIT INSERT - greenit insertion failed') - }) - - lighthouseRepository - .insertAll(reportsFormatted.analysisLighthouseFormatted) - .then(() => { - console.log('LIGHTHOUSE INSERT - analysis has been insert') - }) - .catch(() => { - console.log('LIGHTHOUSE INSERT - lighthouse insertion failed') - }) - - if (allowExternalAPI === 'true') { - w3cRepository.insertAll(reportsFormatted.w3cAnalysisFormatted) - .then(() => { - console.log('W3C INSERT - analysis has been insert') - }) - .catch(() => { - console.log('W3C INSERT - w3c insertion failed') - }) + try { + urlProjectList = await urlsProjectRepository.findAll(projectName, true) + } catch (error) { + console.log('GREENIT INSERT - can not retrieved urls from project') + systemError = true } - bestPracticesRepository - .insertBestPractices(reportsFormatted.bestPracticesFormatted) - .then(() => { - console.log('BEST PRACTICES INSERT - best practices have been inserted') - }) - .catch(() => { - console.log('BEST PRACTICES INSERT : best practice insertion failed') - }) + if (systemError || urlProjectList.length === 0) { + console.log('GREENIT INSERT - project has no url to do the audit. Audit stopped') + } else { + reports = await launchAuditsToUrlList(urlProjectList, autoscroll, projectName, allowExternalAPI) + const reportsFormatted = formatAuditsToBeSaved(reports, urlProjectList) + + greenItRepository + .insertAll(reportsFormatted.greenitAnalysisFormatted) + .then(() => { + console.log('GREENIT INSERT - analysis has been insert') + }) + .catch(() => { + console.log('GREENIT INSERT - greenit insertion failed') + }) + + lighthouseRepository + .insertAll(reportsFormatted.analysisLighthouseFormatted) + .then(() => { + console.log('LIGHTHOUSE INSERT - analysis has been insert') + }) + .catch(() => { + console.log('LIGHTHOUSE INSERT - lighthouse insertion failed') + }) + + if (allowExternalAPI === 'true') { + w3cRepository.insertAll(reportsFormatted.w3cAnalysisFormatted) + .then(() => { + console.log('W3C INSERT - analysis has been insert') + }) + .catch(() => { + console.log('W3C INSERT - w3c insertion failed') + }) + } + + bestPracticesRepository + .insertBestPractices(reportsFormatted.bestPracticesFormatted) + .then(() => { + console.log('BEST PRACTICES INSERT - best practices have been inserted') + }) + .catch(() => { + console.log('BEST PRACTICES INSERT : best practice insertion failed') + }) + } } } @@ -109,6 +110,25 @@ async function launchAuditsToUrlList (urlProjectList, autoscroll, projectName, a } } +function createLighthouseAudit(lighthouseReport, urlProjectList, date) { + const formattedLighthouseMetrics = formatLighthouseMetrics.formatLighthouseMetrics(lighthouseReport) + const urlProjectAudited = urlProjectList.filter((urlProject) => urlProject.urlName === lighthouseReport.url) + const lighthouseAudit = { + idLighthouseAnalysis: uniqid(), + idUrlLighthouse: urlProjectAudited[0].idKey, + dateLighthouseAnalysis: date, + performance: formattedLighthouseMetrics.performance, + accessibility: formattedLighthouseMetrics.accessibility, + largestContentfulPaint: formattedLighthouseMetrics.largestContentfulPaint, + cumulativeLayoutShift: formattedLighthouseMetrics.cumulativeLayoutShift, + firstContentfulPaint: formattedLighthouseMetrics.firstContentfulPaint, + speedIndex: formattedLighthouseMetrics.speedIndex, + totalBlockingTime: formattedLighthouseMetrics.totalBlockingTime, + interactive: formattedLighthouseMetrics.interactive + } + return lighthouseAudit; +} + function formatAuditsToBeSaved (reports, urlProjectList) { const greenitAnalysisFormatted = [] const analysisLighthouseFormatted = [] @@ -140,22 +160,9 @@ function formatAuditsToBeSaved (reports, urlProjectList) { // Format Lighthouse Analysis for (const lighthouseReport of reports.reportsLighthouse) { if (lighthouseReport?.runtimeError === undefined) { - const formattedLighthouseMetrics = formatLighthouseMetrics.formatLighthouseMetrics(lighthouseReport) - const urlProjectAudited = urlProjectList.filter((urlProject) => urlProject.urlName === lighthouseReport.url) - const lighthouseAudit = { - idLighthouseAnalysis: uniqid(), - idUrlLighthouse: urlProjectAudited[0].idKey, - dateLighthouseAnalysis: date, - performance: formattedLighthouseMetrics.performance, - accessibility: formattedLighthouseMetrics.accessibility, - largestContentfulPaint: formattedLighthouseMetrics.largestContentfulPaint, - cumulativeLayoutShift: formattedLighthouseMetrics.cumulativeLayoutShift, - firstContentfulPaint: formattedLighthouseMetrics.firstContentfulPaint, - speedIndex: formattedLighthouseMetrics.speedIndex, - totalBlockingTime: formattedLighthouseMetrics.totalBlockingTime, - interactive: formattedLighthouseMetrics.interactive - } - analysisLighthouseFormatted.push(lighthouseAudit) + const lighthouseAudit = createLighthouseAudit(lighthouseReport, urlProjectList, date); + const lighthouseAuditMobile = createLighthouseAudit(lighthouseReport.mobile, urlProjectList, date); + analysisLighthouseFormatted.push({ ...lighthouseAudit, mobile: lighthouseAuditMobile }) lighthousePerformanceBestPractices.push(formatLighthouseBestPractices.formatPerformance(lighthouseReport)) lighthouseAccessibilityBestPractices.push(formatLighthouseBestPractices.formatAccessibility(lighthouseReport)) } @@ -209,4 +216,4 @@ function formatAuditsToBeSaved (reports, urlProjectList) { } const analysisService = new AnalysisService() -module.exports = analysisService +module.exports = { analysisService, formatAuditsToBeSaved } diff --git a/EcoSonar-API/services/analysisService.test.js b/EcoSonar-API/services/analysisService.test.js new file mode 100644 index 0000000..00159a5 --- /dev/null +++ b/EcoSonar-API/services/analysisService.test.js @@ -0,0 +1,141 @@ +const { formatAuditsToBeSaved } = require('./analysisService'); +const uniqid = require('uniqid') +const lighthouseDesktopJson = require('../test/lighthouse-report-desktop-test.json') +const lighthouseMobileJson = require('../test/lighthouse-report-mobile-test.json') + +jest.mock('uniqid'); +describe('analysis service', () => { + const dateMock = 123456789 + const uniqIdMock = 123456 + + beforeEach(() => { + jest.spyOn(Date, 'now').mockReturnValue(dateMock) + }) + + afterEach(() => { + jest.clearAllMocks(); + }) + + describe('formatAuditsToBeSaved', () => { + + it('should format lighthouse audit with mobile', () => { + uniqid.mockReturnValue(uniqIdMock) + + const reports = { + reportsLighthouse: [ + { + ...lighthouseDesktopJson, + url: 'http://example.com', + mobile: { + ...lighthouseMobileJson, + url: 'http://example.com', + } + } + ], + reportsGreenit: [ + ], + reportsW3c:[] + }; + + const urlProjectList = [ + { + idKey: 'idKey', + urlName: 'http://example.com' + } + ]; + + const formattedAudits = formatAuditsToBeSaved(reports, urlProjectList) + + expect(formattedAudits.analysisLighthouseFormatted).toStrictEqual([ + { + accessibility: { + complianceLevel: "A", + score: 100 + }, + cumulativeLayoutShift: { + complianceLevel: "A", + displayValue: 0.006, + score: 100 + }, + dateLighthouseAnalysis: dateMock, + firstContentfulPaint: { + complianceLevel: "A", + displayValue: 0.3, + score: 100 + }, + idLighthouseAnalysis: uniqIdMock, + idUrlLighthouse: "idKey", + interactive: { + complianceLevel: "A", + displayValue: 2.8, + score: 84 + }, + largestContentfulPaint: { + complianceLevel: "E", + displayValue: 2.9, + score: 37 + }, + performance: { + complianceLevel: "C", + score: 67 + }, + speedIndex: { + complianceLevel: "A", + displayValue: 1, + score: 97 + }, + totalBlockingTime: { + complianceLevel: "D", + displayValue: 350, + score: 50 + }, + mobile: { + accessibility: { + complianceLevel: "G", + score: 50 + }, + cumulativeLayoutShift: { + complianceLevel: "A", + displayValue: 0.006, + score: 100 + }, + dateLighthouseAnalysis: dateMock, + firstContentfulPaint: { + complianceLevel: "A", + displayValue: 0.3, + score: 100 + }, + idLighthouseAnalysis: uniqIdMock, + idUrlLighthouse: "idKey", + interactive: { + complianceLevel: "A", + displayValue: 2.8, + score: 84 + }, + largestContentfulPaint: { + complianceLevel: "E", + displayValue: 2.9, + score: 37 + }, + performance: { + complianceLevel: "F", + score: 20 + }, + speedIndex: { + complianceLevel: "A", + displayValue: 1, + score: 97 + }, + totalBlockingTime: { + complianceLevel: "D", + displayValue: 350, + score: 50 + } + } + } + ]) + + }) + + }) +}) \ No newline at end of file diff --git a/EcoSonar-API/services/format/formatLighthouseAnalysis.js b/EcoSonar-API/services/format/formatLighthouseAnalysis.js index dd3ff65..e523f23 100644 --- a/EcoSonar-API/services/format/formatLighthouseAnalysis.js +++ b/EcoSonar-API/services/format/formatLighthouseAnalysis.js @@ -1,5 +1,188 @@ const formatCompliance = require('./formatCompliance') -class FormatLighthouseAnalysis {} +class FormatLighthouseAnalysis { + + lighthouseAnalysisFormattedDeployments(res) { + let j = 0 + const deployments = [] + + let formattedMetrics + + try { + while (j < res.length) { + formattedMetrics = { + performanceScore: res[j].performance.score, + accessibilityScore: res[j].accessibility.score, + dateAnalysis: res[j].dateLighthouseAnalysis, + largestContentfulPaint: res[j].largestContentfulPaint.score, + cumulativeLayoutShift: res[j].cumulativeLayoutShift.score, + firstContentfulPaint: res[j].firstContentfulPaint.score, + speedIndex: res[j].speedIndex.score, + totalBlockingTime: res[j].totalBlockingTime.score, + interactive: res[j].interactive.score, + mobile: { + performanceScore: res[j].mobile.performance.score, + accessibilityScore: res[j].mobile.accessibility.score, + dateAnalysis: res[j].mobile.dateLighthouseAnalysis, + largestContentfulPaint: res[j].mobile.largestContentfulPaint.score, + cumulativeLayoutShift: res[j].mobile.cumulativeLayoutShift.score, + firstContentfulPaint: res[j].mobile.firstContentfulPaint.score, + speedIndex: res[j].mobile.speedIndex.score, + totalBlockingTime: res[j].mobile.totalBlockingTime.score, + interactive: res[j].mobile.interactive.score + } + } + + deployments[j] = formattedMetrics + + j++ + } + return this.formatDeploymentsForGraphs(deployments) + } catch (err) { + console.log(err) + console.log('LIGHTHOUSE - error during the formatting of project analysis') + } + } + + lighthouseProjectLastAnalysisFormatted(res) { + let analysisDesktop, analysisMobile + let j = 0 + let count, dateAnalysis, desktop, mobile + + try { + while (j < res.length) { + count = 0 + dateAnalysis = res[j].dateLighthouseAnalysis + + // Creating clones + desktop = createClone(res[j]) + mobile = createClone(res[j].mobile) + + while (j < res.length && dateAnalysis.getTime() === res[j].dateLighthouseAnalysis.getTime()) { + desktop.performance.score += res[j].performance.score + desktop.accessibility.score += res[j].accessibility.score + desktop.largestContentfulPaint.displayValue += res[j].largestContentfulPaint.displayValue + desktop.largestContentfulPaint.score += res[j].largestContentfulPaint.score + desktop.cumulativeLayoutShift.displayValue += res[j].cumulativeLayoutShift.displayValue + desktop.cumulativeLayoutShift.score += res[j].cumulativeLayoutShift.score + desktop.firstContentfulPaint.displayValue += res[j].firstContentfulPaint.displayValue + desktop.firstContentfulPaint.score += res[j].firstContentfulPaint.score + desktop.speedIndex.displayValue += res[j].speedIndex.displayValue + desktop.speedIndex.score += res[j].speedIndex.score + desktop.totalBlockingTime.displayValue += res[j].totalBlockingTime.displayValue + desktop.totalBlockingTime.score += res[j].totalBlockingTime.score + desktop.interactive.displayValue += res[j].interactive.displayValue + desktop.interactive.score += res[j].interactive.score + + mobile.performance.score += res[j].mobile.performance.score + mobile.accessibility.score += res[j].mobile.accessibility.score + mobile.largestContentfulPaint.displayValue += res[j].mobile.largestContentfulPaint.displayValue + mobile.largestContentfulPaint.score += res[j].mobile.largestContentfulPaint.score + mobile.cumulativeLayoutShift.displayValue += res[j].mobile.cumulativeLayoutShift.displayValue + mobile.cumulativeLayoutShift.score += res[j].mobile.cumulativeLayoutShift.score + mobile.firstContentfulPaint.displayValue += res[j].mobile.firstContentfulPaint.displayValue + mobile.firstContentfulPaint.score += res[j].mobile.firstContentfulPaint.score + mobile.speedIndex.displayValue += res[j].mobile.speedIndex.displayValue + mobile.speedIndex.score += res[j].mobile.speedIndex.score + mobile.totalBlockingTime.displayValue += res[j].mobile.totalBlockingTime.displayValue + mobile.totalBlockingTime.score += res[j].mobile.totalBlockingTime.score + mobile.interactive.displayValue += res[j].mobile.interactive.displayValue + mobile.interactive.score += res[j].mobile.interactive.score + + count++ + j++ + } + + desktop.largestContentfulPaint.score = calculateAverageScore(desktop.largestContentfulPaint.score) + desktop.cumulativeLayoutShift.score = calculateAverageScore(desktop.cumulativeLayoutShift.score) + desktop.firstContentfulPaint.score = calculateAverageScore(desktop.firstContentfulPaint.score) + desktop.speedIndex.score = calculateAverageScore(desktop.speedIndex.score) + desktop.totalBlockingTime.score = calculateAverageScore(desktop.totalBlockingTime.score) + desktop.interactive.score = calculateAverageScore(desktop.interactive.score) + desktop.performance.score = calculateAverageScore(desktop.performance.score) + desktop.accessibility.score = calculateAverageScore(desktop.accessibility.score) + + mobile.largestContentfulPaint.score = calculateAverageScore(mobile.largestContentfulPaint.score) + mobile.cumulativeLayoutShift.score = calculateAverageScore(mobile.cumulativeLayoutShift.score) + mobile.firstContentfulPaint.score = calculateAverageScore(mobile.firstContentfulPaint.score) + mobile.speedIndex.score = calculateAverageScore(mobile.speedIndex.score) + mobile.totalBlockingTime.score = calculateAverageScore(mobile.totalBlockingTime.score) + mobile.interactive.score = calculateAverageScore(mobile.interactive.score) + mobile.performance.score = calculateAverageScore(mobile.performance.score) + mobile.accessibility.score = calculateAverageScore(mobile.accessibility.score) + + analysisDesktop = createAnalysis(desktop) + analysisMobile = createAnalysis(mobile) + + } + return { ...analysisDesktop, mobile: analysisMobile } + } catch (err) { + console.log(err) + console.log('LIGHTHOUSE - error during the formatting of project analysis') + } + + function createClone(report) { + return { + performance: { ...report.performance, score: 0 }, + accessibility: { ...report.accessibility, score: 0 }, + largestContentfulPaint: { ...report.largestContentfulPaint, score: 0, displayValue: 0 }, + cumulativeLayoutShift: { ...report.cumulativeLayoutShift, score: 0, displayValue: 0 }, + firstContentfulPaint: { ...report.firstContentfulPaint, score: 0, displayValue: 0 }, + speedIndex: { ...report.speedIndex, score: 0, displayValue: 0 }, + totalBlockingTime: { ...report.totalBlockingTime, score: 0, displayValue: 0 }, + interactive: { ...report.interactive, score: 0, displayValue: 0 }, + } + } + + function createAnalysis(rawReport) { + return { + performance: { + displayValue: Math.trunc(parseFloat(rawReport.performance.score)), + complianceLevel: formatCompliance.getEcodesignGrade(rawReport.performance.score) + }, + accessibility: { + displayValue: Math.trunc(parseFloat(rawReport.accessibility.score)), + complianceLevel: formatCompliance.getAccessibilityGrade(rawReport.accessibility.score) + }, + dateAnalysis, + largestContentfulPaint: { + displayValue: calculateAverageScore(rawReport.largestContentfulPaint.displayValue, 1) + ' s', + score: rawReport.largestContentfulPaint.score, + complianceLevel: formatCompliance.getEcodesignGrade(rawReport.largestContentfulPaint.score) + }, + cumulativeLayoutShift: { + displayValue: calculateAverageScore(rawReport.cumulativeLayoutShift.displayValue, 3), + score: rawReport.cumulativeLayoutShift.score, + complianceLevel: formatCompliance.getEcodesignGrade(rawReport.cumulativeLayoutShift.score) + }, + firstContentfulPaint: { + displayValue: calculateAverageScore(rawReport.firstContentfulPaint.displayValue, 1) + ' s', + score: rawReport.firstContentfulPaint.score, + complianceLevel: formatCompliance.getEcodesignGrade(rawReport.firstContentfulPaint.score) + }, + speedIndex: { + displayValue: calculateAverageScore(rawReport.speedIndex.displayValue, 1) + ' s', + score: rawReport.speedIndex.score, + complianceLevel: formatCompliance.getEcodesignGrade(rawReport.speedIndex.score) + }, + totalBlockingTime: { + displayValue: calculateAverageScore(rawReport.totalBlockingTime.displayValue, 0) + ' ms', + score: rawReport.totalBlockingTime.score, + complianceLevel: formatCompliance.getEcodesignGrade(rawReport.totalBlockingTime.score) + }, + interactive: { + displayValue: calculateAverageScore(rawReport.interactive.displayValue, 1) + ' s', + score: rawReport.interactive.score, + complianceLevel: formatCompliance.getEcodesignGrade(rawReport.interactive.score) + } + } + } + + function calculateAverageScore (score, toFixParam) { + return (score / count).toFixed(toFixParam) + } + } + +} FormatLighthouseAnalysis.prototype.lighthouseUrlAnalysisFormatted = function (analysis) { let formattedAnalysis @@ -46,146 +229,6 @@ FormatLighthouseAnalysis.prototype.lighthouseUrlAnalysisFormatted = function (an return formattedAnalysis } -FormatLighthouseAnalysis.prototype.lighthouseProjectLastAnalysisFormatted = function (res) { - let analysis - let j = 0 - let count, performance, dateAnalysis, accessibility, largestContentfulPaint, cumulativeLayoutShift, firstContentfulPaint, speedIndex, totalBlockingTime, interactive - - try { - while (j < res.length) { - count = 0 - dateAnalysis = res[j].dateLighthouseAnalysis - - // Creating clones - performance = JSON.parse(JSON.stringify(res[j].performance)) - accessibility = JSON.parse(JSON.stringify(res[j].accessibility)) - largestContentfulPaint = JSON.parse(JSON.stringify(res[j].largestContentfulPaint)) - cumulativeLayoutShift = JSON.parse(JSON.stringify(res[j].cumulativeLayoutShift)) - firstContentfulPaint = JSON.parse(JSON.stringify(res[j].firstContentfulPaint)) - speedIndex = JSON.parse(JSON.stringify(res[j].speedIndex)) - totalBlockingTime = JSON.parse(JSON.stringify(res[j].totalBlockingTime)) - interactive = JSON.parse(JSON.stringify(res[j].interactive)) - - performance.score = 0 - accessibility.score = 0 - largestContentfulPaint.displayValue = 0 - largestContentfulPaint.score = 0 - cumulativeLayoutShift.displayValue = 0 - cumulativeLayoutShift.score = 0 - firstContentfulPaint.displayValue = 0 - firstContentfulPaint.score = 0 - speedIndex.displayValue = 0 - speedIndex.score = 0 - totalBlockingTime.displayValue = 0 - totalBlockingTime.score = 0 - interactive.displayValue = 0 - interactive.score = 0 - - while (j < res.length && dateAnalysis.getTime() === res[j].dateLighthouseAnalysis.getTime()) { - performance.score += res[j].performance.score - accessibility.score += res[j].accessibility.score - largestContentfulPaint.displayValue += res[j].largestContentfulPaint.displayValue - largestContentfulPaint.score += res[j].largestContentfulPaint.score - cumulativeLayoutShift.displayValue += res[j].cumulativeLayoutShift.displayValue - cumulativeLayoutShift.score += res[j].cumulativeLayoutShift.score - firstContentfulPaint.displayValue += res[j].firstContentfulPaint.displayValue - firstContentfulPaint.score += res[j].firstContentfulPaint.score - speedIndex.displayValue += res[j].speedIndex.displayValue - speedIndex.score += res[j].speedIndex.score - totalBlockingTime.displayValue += res[j].totalBlockingTime.displayValue - totalBlockingTime.score += res[j].totalBlockingTime.score - interactive.displayValue += res[j].interactive.displayValue - interactive.score += res[j].interactive.score - count++ - j++ - } - - largestContentfulPaint.score = calculateAverageScore(largestContentfulPaint.score) - cumulativeLayoutShift.score = calculateAverageScore(cumulativeLayoutShift.score) - firstContentfulPaint.score = calculateAverageScore(firstContentfulPaint.score) - speedIndex.score = calculateAverageScore(speedIndex.score) - totalBlockingTime.score = calculateAverageScore(totalBlockingTime.score) - interactive.score = calculateAverageScore(interactive.score) - performance.score = calculateAverageScore(performance.score) - accessibility.score = calculateAverageScore(accessibility.score) - - analysis = { - performance: { displayValue: Math.trunc(parseFloat(performance.score)), complianceLevel: formatCompliance.getEcodesignGrade(performance.score) }, - accessibility: { displayValue: Math.trunc(parseFloat(accessibility.score)), complianceLevel: formatCompliance.getAccessibilityGrade(accessibility.score) }, - dateAnalysis, - largestContentfulPaint: { - displayValue: calculateAverageScore(largestContentfulPaint.displayValue, 1) + ' s', - score: largestContentfulPaint.score, - complianceLevel: formatCompliance.getEcodesignGrade(largestContentfulPaint.score) - }, - cumulativeLayoutShift: { - displayValue: calculateAverageScore(cumulativeLayoutShift.displayValue, 3), - score: cumulativeLayoutShift.score, - complianceLevel: formatCompliance.getEcodesignGrade(cumulativeLayoutShift.score) - }, - firstContentfulPaint: { - displayValue: calculateAverageScore(firstContentfulPaint.displayValue, 1) + ' s', - score: firstContentfulPaint.score, - complianceLevel: formatCompliance.getEcodesignGrade(firstContentfulPaint.score) - }, - speedIndex: { - displayValue: calculateAverageScore(speedIndex.displayValue, 1) + ' s', - score: speedIndex.score, - complianceLevel: formatCompliance.getEcodesignGrade(speedIndex.score) - }, - totalBlockingTime: { - displayValue: calculateAverageScore(totalBlockingTime.displayValue, 0) + ' ms', - score: totalBlockingTime.score, - complianceLevel: formatCompliance.getEcodesignGrade(totalBlockingTime.score) - }, - interactive: { - displayValue: calculateAverageScore(interactive.displayValue, 1) + ' s', - score: interactive.score, - complianceLevel: formatCompliance.getEcodesignGrade(interactive.score) - } - } - } - return analysis - } catch (err) { - console.log(err) - console.log('LIGHTHOUSE - error during the formatting of project analysis') - } - function calculateAverageScore (score, toFixParam) { - return (score / count).toFixed(toFixParam) - } -} - -FormatLighthouseAnalysis.prototype.lighthouseAnalysisFormattedDeployments = function (res) { - let j = 0 - const deployments = [] - - let formattedMetrics - - try { - while (j < res.length) { - formattedMetrics = { - performanceScore: res[j].performance.score, - accessibilityScore: res[j].accessibility.score, - dateAnalysis: res[j].dateLighthouseAnalysis, - largestContentfulPaint: res[j].largestContentfulPaint.score, - cumulativeLayoutShift: res[j].cumulativeLayoutShift.score, - firstContentfulPaint: res[j].firstContentfulPaint.score, - speedIndex: res[j].speedIndex.score, - totalBlockingTime: res[j].totalBlockingTime.score, - interactive: res[j].interactive.score - } - - deployments[j] = formattedMetrics - - j++ - } - return this.formatDeploymentsForGraphs(deployments) - } catch (err) { - console.log(err) - console.log('LIGHTHOUSE - error during the formatting of project analysis') - } -} - FormatLighthouseAnalysis.prototype.formatDeploymentsForGraphs = function (deployments) { const duplicatedDeployments = [] @@ -202,7 +245,19 @@ FormatLighthouseAnalysis.prototype.formatDeploymentsForGraphs = function (deploy speedIndex: 0, totalBlockingTime: 0, interactive: 0, - numberOfValues: 0 + numberOfValues: 0, + mobile: { + performanceScore: 0, + accessibilityScore: 0, + dateAnalysis: duplicatedValuesArray[0].dateAnalysis, + largestContentfulPaint: 0, + cumulativeLayoutShift: 0, + firstContentfulPaint: 0, + speedIndex: 0, + totalBlockingTime: 0, + interactive: 0, + numberOfValues: 0 + } } // We add up every element with the same date (only DD/MM/YYYY) in one (sumElement) @@ -216,6 +271,15 @@ FormatLighthouseAnalysis.prototype.formatDeploymentsForGraphs = function (deploy sumElement.totalBlockingTime += element.totalBlockingTime sumElement.interactive += element.interactive sumElement.numberOfValues++ + sumElement.mobile.performanceScore += element.mobile.performanceScore + sumElement.mobile.accessibilityScore += element.mobile.accessibilityScore + sumElement.mobile.largestContentfulPaint += element.mobile.largestContentfulPaint + sumElement.mobile.cumulativeLayoutShift += element.mobile.cumulativeLayoutShift + sumElement.mobile.firstContentfulPaint += element.mobile.firstContentfulPaint + sumElement.mobile.speedIndex += element.mobile.speedIndex + sumElement.mobile.totalBlockingTime += element.mobile.totalBlockingTime + sumElement.mobile.interactive += element.mobile.interactive + sumElement.mobile.numberOfValues++ }) duplicatedDeployments.push(sumElement) @@ -238,7 +302,16 @@ FormatLighthouseAnalysis.prototype.formatDeploymentsForGraphs = function (deploy i.speedIndex = Math.round(i.speedIndex / i.numberOfValues) i.totalBlockingTime = Math.round(i.totalBlockingTime / i.numberOfValues) i.interactive = Math.round(i.interactive / i.numberOfValues) + i.mobile.performanceScore = Math.round(i.mobile.performanceScore / i.mobile.numberOfValues) + i.mobile.accessibilityScore = Math.round(i.mobile.accessibilityScore / i.mobile.numberOfValues) + i.mobile.largestContentfulPaint = Math.round(i.mobile.largestContentfulPaint / i.mobile.numberOfValues) + i.mobile.cumulativeLayoutShift = Math.round(i.mobile.cumulativeLayoutShift / i.mobile.numberOfValues) + i.mobile.firstContentfulPaint = Math.round(i.mobile.firstContentfulPaint / i.mobile.numberOfValues) + i.mobile.speedIndex = Math.round(i.mobile.speedIndex / i.mobile.numberOfValues) + i.mobile.totalBlockingTime = Math.round(i.mobile.totalBlockingTime / i.mobile.numberOfValues) + i.mobile.interactive = Math.round(i.mobile.interactive / i.mobile.numberOfValues) delete i.numberOfValues + delete i.mobile.numberOfValues } function compareFullDate (firstDate, secondDate) { diff --git a/EcoSonar-API/services/format/formatLighthouseAnalysis.test.js b/EcoSonar-API/services/format/formatLighthouseAnalysis.test.js new file mode 100644 index 0000000..c581df7 --- /dev/null +++ b/EcoSonar-API/services/format/formatLighthouseAnalysis.test.js @@ -0,0 +1,134 @@ +const formatLighthouseAnalysis = require('./formatLighthouseAnalysis') +const lighthouseReportsDto = require('../../test/lighthouse-report-dto-test.json') +describe('format lighthouse analysis', () => { + + afterEach(() => { + jest.clearAllMocks(); + }) + + describe('lighthouseAnalysisFormattedDeployments', () => { + + it('should present lighthouse report', () => { + lighthouseReportsDto.deployments[0].dateLighthouseAnalysis = new Date(lighthouseReportsDto.deployments[0].dateLighthouseAnalysis) + lighthouseReportsDto.deployments[0].mobile.dateLighthouseAnalysis = new Date(lighthouseReportsDto.deployments[0].mobile.dateLighthouseAnalysis) + const presentedReports = formatLighthouseAnalysis.lighthouseAnalysisFormattedDeployments(lighthouseReportsDto.deployments) + expect(presentedReports).toStrictEqual([ + { + accessibilityScore: 94, + cumulativeLayoutShift: 1, + dateAnalysis: new Date('2024-02-02T10:57:24.843Z'), + firstContentfulPaint: 100, + interactive: 18, + largestContentfulPaint: 0, + performanceScore: 15, + speedIndex: 33, + totalBlockingTime: 0, + mobile: { + accessibilityScore: 94, + cumulativeLayoutShift: 1, + dateAnalysis: new Date('2024-02-02T10:57:24.843Z'), + firstContentfulPaint: 100, + interactive: 19, + largestContentfulPaint: 0, + performanceScore: 16, + speedIndex: 37, + totalBlockingTime: 0 + } + } + ]) + }) + + }) + + describe('lighthouseProjectLastAnalysisFormatted', () => { + + it('should present lighthouse last analysis', () => { + lighthouseReportsDto.lastAnalysis[0].dateLighthouseAnalysis = new Date(lighthouseReportsDto.lastAnalysis[0].dateLighthouseAnalysis) + lighthouseReportsDto.lastAnalysis[0].mobile.dateLighthouseAnalysis = new Date(lighthouseReportsDto.lastAnalysis[0].mobile.dateLighthouseAnalysis) + const presentedReports = formatLighthouseAnalysis.lighthouseProjectLastAnalysisFormatted(lighthouseReportsDto.lastAnalysis) + expect(presentedReports).toStrictEqual({ + accessibility: { + complianceLevel: "B", + displayValue: 94 + }, + cumulativeLayoutShift: { + complianceLevel: "G", + displayValue: "1.162", + score: "1" + }, + dateAnalysis: new Date('2024-02-02T10:57:24.843Z'), + firstContentfulPaint: { + complianceLevel: "A", + displayValue: "0.8 s", + score: "100" + }, + interactive: { + complianceLevel: "F", + displayValue: "11.5 s", + score: "18" + }, + largestContentfulPaint: { + complianceLevel: "G", + displayValue: "11.2 s", + score: "0" + }, + performance: { + complianceLevel: "F", + displayValue: 15 + }, + speedIndex: { + complianceLevel: "E", + displayValue: "6.9 s", + score: "33" + }, + totalBlockingTime: { + complianceLevel: "G", + displayValue: "10330 ms", + score: "0" + }, + mobile: { + accessibility: { + complianceLevel: "B", + displayValue: 94 + }, + cumulativeLayoutShift: { + complianceLevel: "G", + displayValue: "1.198", + score: "1" + }, + dateAnalysis: new Date('2024-02-02T10:57:24.843Z'), + firstContentfulPaint: { + complianceLevel: "A", + displayValue: "0.8 s", + score: "100" + }, + interactive: { + complianceLevel: "F", + displayValue: "11.4 s", + score: "19" + }, + largestContentfulPaint: { + complianceLevel: "G", + displayValue: "11.1 s", + score: "0" + }, + performance: { + complianceLevel: "F", + displayValue: 16 + }, + speedIndex: { + complianceLevel: "E", + displayValue: "6.6 s", + score: "37" + }, + totalBlockingTime: { + complianceLevel: "G", + displayValue: "10260 ms", + score: "0" + } + } + }) + }) + + }) +}) \ No newline at end of file diff --git a/EcoSonar-API/services/lighthouse/lighthouse.js b/EcoSonar-API/services/lighthouse/lighthouse.js index 36dfb5c..5345a21 100644 --- a/EcoSonar-API/services/lighthouse/lighthouse.js +++ b/EcoSonar-API/services/lighthouse/lighthouse.js @@ -7,66 +7,91 @@ const viewPortParams = require('../../utils/viewportParams') module.exports = { lighthouseAnalysis: async function (urlList, projectName) { - const browserArgs = [ + const browserArgsDesktop = [ '--no-sandbox', // can't run inside docker without '--disable-setuid-sandbox', // but security issues '--ignore-certificate-errors', '--window-size=1920,1080', '--start-maximized' ] + const configDesktop = { ...config } + const results = await lighthouseAnalysisWithSpecifiedConfig(urlList, projectName, browserArgsDesktop, configDesktop) + const browserArgsMobile = [ + '--no-sandbox', // can't run inside docker without + '--disable-setuid-sandbox', // but security issues + '--ignore-certificate-errors', + '--window-size=430,932', + '--start-maximized' + ] + const configMobile = { ...config } + configMobile.screenEmulation.width = 430 + configMobile.screenEmulation.height = 932 - const proxyConfiguration = await authenticationService.useProxyIfNeeded(projectName) - if (proxyConfiguration) { - browserArgs.push(proxyConfiguration) - } - - // start browser - const browser = await puppeteer.launch({ - headless: true, - args: browserArgs, - ignoreHTTPSErrors: true, - // Keep gpu horsepower in headless - ignoreDefaultArgs: [ - '--disable-gpu' - ], - defaultViewport: viewPortParams.viewPortParams + const resultsMobile = await lighthouseAnalysisWithSpecifiedConfig(urlList, projectName, browserArgsMobile, configMobile) + results.forEach((result) => { + const currentResultMobile = resultsMobile.find((resultMobile) => result.url === resultMobile.url) + if (currentResultMobile) { + result.mobile = currentResultMobile + } }) + return results + } +} + +async function lighthouseAnalysisWithSpecifiedConfig (urlList, projectName, browserArgs, config) { - const results = [] - const options = { logLevel: 'error', output: 'json', onlyCategories: ['performance', 'accessibility'], port: (new URL(browser.wsEndpoint())).port, disableStorageReset: true } - try { - let lighthouseResults - let userJourney - const loginSucceeded = await authenticationService.loginIfNeeded(browser, projectName) - if (loginSucceeded) { - for (const [index, url] of urlList.entries()) { - try { - console.log('Lighthouse Analysis launched for url ' + url) - await userJourneyService.getUserFlow(projectName, url) + const proxyConfiguration = await authenticationService.useProxyIfNeeded(projectName) + if (proxyConfiguration) { + browserArgs.push(proxyConfiguration) + } + + // start browser + const browser = await puppeteer.launch({ + headless: true, + args: browserArgs, + ignoreHTTPSErrors: true, + // Keep gpu horsepower in headless + ignoreDefaultArgs: [ + '--disable-gpu' + ], + defaultViewport: viewPortParams.viewPortParams + }) + + const results = [] + const options = { logLevel: 'error', output: 'json', onlyCategories: ['performance', 'accessibility'], port: (new URL(browser.wsEndpoint())).port, disableStorageReset: true } + + try { + let lighthouseResults + let userJourney + const loginSucceeded = await authenticationService.loginIfNeeded(browser, projectName) + if (loginSucceeded) { + for (const [index, url] of urlList.entries()) { + try { + console.log('Lighthouse Analysis launched for url ' + url) + await userJourneyService.getUserFlow(projectName, url) .then((userflow) => { userJourney = userflow }).catch((error) => { console.log(error.message) }) - if (userJourney) { - lighthouseResults = await userJourneyService.playUserFlowLighthouse(url, browser, userJourney) - } else { - lighthouseResults = await lighthouse(url, options, config) - } - console.log('Lighthouse Analysis ended for url ' + url) - results[index] = { ...lighthouseResults.lhr, url } - } catch (error) { - console.log('LIGHTHOUSE ANALYSIS - An error occured when auditing ' + url) - console.error('\x1b[31m%s\x1b[0m', error) + if (userJourney) { + lighthouseResults = await userJourneyService.playUserFlowLighthouse(url, browser, userJourney) + } else { + lighthouseResults = await lighthouse(url, options, config) } + console.log('Lighthouse Analysis ended for url ' + url) + results[index] = { ...lighthouseResults.lhr, url } + } catch (error) { + console.log('LIGHTHOUSE ANALYSIS - An error occured when auditing ' + url) + console.error('\x1b[31m%s\x1b[0m', error) } } - } catch (error) { - console.error('\x1b[31m%s\x1b[0m', error) - } finally { - await browser.close() } - return results + } catch (error) { + console.error('\x1b[31m%s\x1b[0m', error) + } finally { + await browser.close() } -} + return results +} \ No newline at end of file diff --git a/EcoSonar-API/services/lighthouse/lighthouse.test.js b/EcoSonar-API/services/lighthouse/lighthouse.test.js new file mode 100644 index 0000000..6f81935 --- /dev/null +++ b/EcoSonar-API/services/lighthouse/lighthouse.test.js @@ -0,0 +1,75 @@ +const puppeteer = require('puppeteer'); +const lighthouse = require('lighthouse') +const authenticationService = require('../authenticationService'); +const userJourneyService = require('../userJourneyService'); +const lighthouseAnalysis = require('./lighthouse'); + +jest.mock('puppeteer'); +jest.mock('lighthouse'); +jest.mock('../authenticationService'); +jest.mock('../userJourneyService'); + +describe('lighthouseAnalysis function', () => { + beforeEach(() => { + jest.clearAllMocks(); + }); + + it('should analyze URLs with Lighthouse', async () => { + // given + puppeteer.launch.mockResolvedValue({ + close: jest.fn(), + wsEndpoint: jest.fn().mockReturnValue('ws://example.com'), + }); + authenticationService.loginIfNeeded.mockResolvedValue(true); + + userJourneyService.getUserFlow.mockResolvedValue(undefined); + userJourneyService.playUserFlowLighthouse.mockResolvedValue({ /* lighthouse results */ }); + + lighthouse.mockResolvedValueOnce({ + lhr: { + categories: [ + { + speedIndex: { + score: 50 + } + } + ] + } + }).mockResolvedValueOnce({ + lhr: { + categories: [ + { + speedIndex: { + score: 30 + } + } + ] + } + }) + + // when + const result = await lighthouseAnalysis.lighthouseAnalysis(['http://example.com'], 'projectName'); + + // then + expect(result).toStrictEqual([{ + categories: [ + { + speedIndex: { + score: 50 + } + } + ], + mobile: { + categories: [ + { + speedIndex: { + score: 30 + } + } + ], + url: 'http://example.com' + }, + url: 'http://example.com' + }]); + }); +}); \ No newline at end of file diff --git a/EcoSonar-API/services/retrieveAnalysisService.js b/EcoSonar-API/services/retrieveAnalysisService.js index 3f30af4..a4b213c 100644 --- a/EcoSonar-API/services/retrieveAnalysisService.js +++ b/EcoSonar-API/services/retrieveAnalysisService.js @@ -129,7 +129,6 @@ RetrieveAnalysisService.prototype.getProjectAnalysis = async function (projectNa if (res.deployments.length !== 0) { // deployments lighthouseAnalysisDeployments = formatLighthouseAnalysis.lighthouseAnalysisFormattedDeployments(res.deployments) - // lastAnalysis lighthouseProjectLastAnalysis = formatLighthouseAnalysis.lighthouseProjectLastAnalysisFormatted(res.lastAnalysis) } else { diff --git a/EcoSonar-API/test/lighthouse-report-desktop-test.json b/EcoSonar-API/test/lighthouse-report-desktop-test.json new file mode 100644 index 0000000..9eed8d9 --- /dev/null +++ b/EcoSonar-API/test/lighthouse-report-desktop-test.json @@ -0,0 +1,6655 @@ +{ + "lighthouseVersion": "9.3.0", + "requestedUrl": "http://localhost:3000/", + "finalUrl": "http://localhost:3000/", + "fetchTime": "2024-02-01T09:46:26.034Z", + "gatherMode": "navigation", + "runWarnings": [], + "userAgent": "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) HeadlessChrome/121.0.6167.85 Safari/537.36", + "environment": { + "networkUserAgent": "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/98.0.4695.0 Safari/537.36 Chrome-Lighthouse", + "hostUserAgent": "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) HeadlessChrome/121.0.6167.85 Safari/537.36", + "benchmarkIndex": 2259.5, + "credits": { + "axe-core": "4.3.5" + } + }, + "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](https://web.dev/is-on-https/).", + "score": 1, + "scoreDisplayMode": "binary", + "details": { + "type": "table", + "headings": [], + "items": [] + } + }, + "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://developers.google.com/web/updates/2013/12/300ms-tap-delay-gone-away). [Learn more](https://web.dev/viewport/).", + "score": 1, + "scoreDisplayMode": "binary", + "warnings": [] + }, + "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](https://web.dev/first-contentful-paint/).", + "score": 1, + "scoreDisplayMode": "numeric", + "numericValue": 250.27999999999997, + "numericUnit": "millisecond", + "displayValue": "0.3 s" + }, + "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](https://web.dev/lighthouse-largest-contentful-paint/)", + "score": 0.37, + "scoreDisplayMode": "numeric", + "numericValue": 2853.9699999999993, + "numericUnit": "millisecond", + "displayValue": "2.9 s" + }, + "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](https://web.dev/first-meaningful-paint/).", + "score": 1, + "scoreDisplayMode": "numeric", + "numericValue": 250.27999999999997, + "numericUnit": "millisecond", + "displayValue": "0.3 s" + }, + "speed-index": { + "id": "speed-index", + "title": "Speed Index", + "description": "Speed Index shows how quickly the contents of a page are visibly populated. [Learn more](https://web.dev/speed-index/).", + "score": 0.97, + "scoreDisplayMode": "numeric", + "numericValue": 1020.5271856668203, + "numericUnit": "millisecond", + "displayValue": "1.0 s" + }, + "screenshot-thumbnails": { + "id": "screenshot-thumbnails", + "title": "Screenshot Thumbnails", + "description": "This is what the load of your site looked like.", + "score": null, + "scoreDisplayMode": "informative", + "details": { + "type": "filmstrip", + "scale": 3000, + "items": [ + { + "timing": 300, + "timestamp": 6398103028, + "data": "" + }, + { + "timing": 600, + "timestamp": 6398403028, + "data": "" + }, + { + "timing": 900, + "timestamp": 6398703028, + "data": "" + }, + { + "timing": 1200, + "timestamp": 6399003028, + "data": "" + }, + { + "timing": 1500, + "timestamp": 6399303028, + "data": "" + }, + { + "timing": 1800, + "timestamp": 6399603028, + "data": "" + }, + { + "timing": 2100, + "timestamp": 6399903028, + "data": "" + }, + { + "timing": 2400, + "timestamp": 6400203028, + "data": "" + }, + { + "timing": 2700, + "timestamp": 6400503028, + "data": "" + }, + { + "timing": 3000, + "timestamp": 6400803028, + "data": "" + } + ] + } + }, + "final-screenshot": { + "id": "final-screenshot", + "title": "Final Screenshot", + "description": "The last screenshot captured of the pageload.", + "score": null, + "scoreDisplayMode": "informative", + "details": { + "type": "screenshot", + "timing": 1641, + "timestamp": 6399444012, + "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](https://web.dev/lighthouse-total-blocking-time/).", + "score": 0.5, + "scoreDisplayMode": "numeric", + "numericValue": 347.43000000000006, + "numericUnit": "millisecond", + "displayValue": "350 ms" + }, + "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](https://web.dev/lighthouse-max-potential-fid/).", + "score": 0.17, + "scoreDisplayMode": "numeric", + "numericValue": 400, + "numericUnit": "millisecond", + "displayValue": "400 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](https://web.dev/cls/).", + "score": 1, + "scoreDisplayMode": "numeric", + "numericValue": 0.005971847054303684, + "numericUnit": "unitless", + "displayValue": "0.006", + "details": { + "type": "debugdata", + "items": [ + { + "cumulativeLayoutShiftMainFrame": 0.005971847054303684, + "totalCumulativeLayoutShift": 0.005971847054303684 + } + ] + } + }, + "errors-in-console": { + "id": "errors-in-console", + "title": "No browser errors 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](https://web.dev/errors-in-console/)", + "score": 1, + "scoreDisplayMode": "binary", + "details": { + "type": "table", + "headings": [], + "items": [] + } + }, + "server-response-time": { + "id": "server-response-time", + "title": "Reduce initial server response time", + "description": "Keep the server response time for the main document short because all other requests depend on it. [Learn more](https://web.dev/time-to-first-byte/).", + "score": 0, + "scoreDisplayMode": "binary", + "numericValue": 845.798, + "numericUnit": "millisecond", + "displayValue": "Root document took 850 ms", + "details": { + "type": "opportunity", + "headings": [ + { + "key": "url", + "valueType": "url", + "label": "URL" + }, + { + "key": "responseTime", + "valueType": "timespanMs", + "label": "Time Spent" + } + ], + "items": [ + { + "url": "http://localhost:3000/", + "responseTime": 845.798 + } + ], + "overallSavingsMs": 745.798 + } + }, + "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](https://web.dev/interactive/).", + "score": 0.84, + "scoreDisplayMode": "numeric", + "numericValue": 2806.2599999999998, + "numericUnit": "millisecond", + "displayValue": "2.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](https://web.dev/user-timings/).", + "score": null, + "scoreDisplayMode": "informative", + "displayValue": "7 user timings", + "details": { + "type": "table", + "headings": [ + { + "key": "name", + "itemType": "text", + "text": "Name" + }, + { + "key": "timingType", + "itemType": "text", + "text": "Type" + }, + { + "key": "startTime", + "itemType": "ms", + "granularity": 0.01, + "text": "Start Time" + }, + { + "key": "duration", + "itemType": "ms", + "granularity": 0.01, + "text": "Duration" + } + ], + "items": [ + { + "name": "Next.js-before-hydration", + "startTime": 0, + "duration": 1475.503, + "timingType": "Measure" + }, + { + "name": "Next.js-hydration", + "startTime": 1475.503, + "duration": 105.613, + "timingType": "Measure" + }, + { + "name": "sentry-tracing-init", + "startTime": 1395.599, + "timingType": "Mark" + }, + { + "name": "beforeRender", + "startTime": 1475.503, + "timingType": "Mark" + }, + { + "name": "routeChange", + "startTime": 1579.918, + "timingType": "Mark" + }, + { + "name": "afterHydrate", + "startTime": 1581.116, + "timingType": "Mark" + }, + { + "name": "routeChange", + "startTime": 1587.205, + "timingType": "Mark" + } + ] + } + }, + "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 more](https://web.dev/critical-request-chains/).", + "score": null, + "scoreDisplayMode": "informative", + "displayValue": "1 chain found", + "details": { + "type": "criticalrequestchain", + "chains": { + "52F76661D21D971A1BA55CE39E941E82": { + "request": { + "url": "http://localhost:3000/", + "startTime": 6397.803832, + "endTime": 6398.666501, + "responseReceivedTime": 6398.665719, + "transferSize": 2483 + }, + "children": { + "11724.8": { + "request": { + "url": "http://localhost:3000/_next/static/chunks/react-refresh.js?ts=1706780787914", + "startTime": 6398.679814, + "endTime": 6398.697017, + "responseReceivedTime": 6398.692729, + "transferSize": 25050 + } + } + } + } + }, + "longestChain": { + "duration": 893.1850000008126, + "length": 2, + "transferSize": 25050 + } + } + }, + "redirects": { + "id": "redirects", + "title": "Avoid multiple page redirects", + "description": "Redirects introduce additional delays before the page can be loaded. [Learn more](https://web.dev/redirects/).", + "score": 1, + "scoreDisplayMode": "numeric", + "numericValue": 0, + "numericUnit": "millisecond", + "displayValue": "", + "details": { + "type": "opportunity", + "headings": [], + "items": [], + "overallSavingsMs": 0 + } + }, + "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](https://web.dev/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 more](https://web.dev/serve-responsive-images/).", + "score": 1, + "scoreDisplayMode": "binary", + "details": { + "type": "table", + "headings": [], + "items": [] + } + }, + "preload-fonts": { + "id": "preload-fonts", + "title": "Fonts with `font-display: optional` are preloaded", + "description": "Preload `optional` fonts so first-time visitors may use them. [Learn more](https://web.dev/preload-optional-fonts/)", + "score": null, + "scoreDisplayMode": "notApplicable" + }, + "deprecations": { + "id": "deprecations", + "title": "Avoids deprecated APIs", + "description": "Deprecated APIs will eventually be removed from the browser. [Learn more](https://web.dev/deprecations/).", + "score": 1, + "scoreDisplayMode": "binary", + "details": { + "type": "table", + "headings": [], + "items": [] + } + }, + "mainthread-work-breakdown": { + "id": "mainthread-work-breakdown", + "title": "Minimizes 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 more](https://web.dev/mainthread-work-breakdown/)", + "score": 1, + "scoreDisplayMode": "numeric", + "numericValue": 658.9429999999979, + "numericUnit": "millisecond", + "displayValue": "0.7 s", + "details": { + "type": "table", + "headings": [ + { + "key": "groupLabel", + "itemType": "text", + "text": "Category" + }, + { + "key": "duration", + "itemType": "ms", + "granularity": 1, + "text": "Time Spent" + } + ], + "items": [ + { + "group": "scriptEvaluation", + "groupLabel": "Script Evaluation", + "duration": 373.36299999999693 + }, + { + "group": "other", + "groupLabel": "Other", + "duration": 96.67700000000102 + }, + { + "group": "scriptParseCompile", + "groupLabel": "Script Parsing & Compilation", + "duration": 96.459 + }, + { + "group": "styleLayout", + "groupLabel": "Style & Layout", + "duration": 82.516 + }, + { + "group": "paintCompositeRender", + "groupLabel": "Rendering", + "duration": 5.801999999999999 + }, + { + "group": "parseHTML", + "groupLabel": "Parse HTML & CSS", + "duration": 2.986999999999999 + }, + { + "group": "garbageCollection", + "groupLabel": "Garbage Collection", + "duration": 1.1389999999999993 + } + ] + } + }, + "bootup-time": { + "id": "bootup-time", + "title": "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 more](https://web.dev/bootup-time/).", + "score": 1, + "scoreDisplayMode": "numeric", + "numericValue": 375.2089999999986, + "numericUnit": "millisecond", + "displayValue": "0.4 s", + "details": { + "type": "table", + "headings": [ + { + "key": "url", + "itemType": "url", + "text": "URL" + }, + { + "key": "total", + "granularity": 1, + "itemType": "ms", + "text": "Total CPU Time" + }, + { + "key": "scripting", + "granularity": 1, + "itemType": "ms", + "text": "Script Evaluation" + }, + { + "key": "scriptParseCompile", + "granularity": 1, + "itemType": "ms", + "text": "Script Parse" + } + ], + "items": [ + { + "url": "http://localhost:3000/_next/static/chunks/pages/_app.js?ts=1706780787914", + "total": 202.8379999999986, + "scripting": 156.7539999999986, + "scriptParseCompile": 46.084 + }, + { + "url": "http://localhost:3000/_next/static/chunks/main.js?ts=1706780787914", + "total": 181.40799999999996, + "scripting": 124.61199999999997, + "scriptParseCompile": 43.391 + }, + { + "url": "http://localhost:3000/", + "total": 102.02900000000001, + "scripting": 2.244999999999999, + "scriptParseCompile": 0.576 + }, + { + "url": "Unattributable", + "total": 57.05400000000007, + "scripting": 1.547, + "scriptParseCompile": 0 + } + ], + "summary": { + "wastedMs": 375.2089999999986 + } + } + }, + "uses-rel-preload": { + "id": "uses-rel-preload", + "title": "Preload key requests", + "description": "Consider using `` to prioritize fetching resources that are currently requested later in page load. [Learn more](https://web.dev/uses-rel-preload/).", + "score": null, + "scoreDisplayMode": "notApplicable", + "details": { + "type": "opportunity", + "headings": [], + "items": [], + "overallSavingsMs": 0 + } + }, + "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 more](https://web.dev/uses-rel-preconnect/).", + "score": 1, + "scoreDisplayMode": "numeric", + "numericValue": 0, + "numericUnit": "millisecond", + "displayValue": "", + "warnings": [], + "details": { + "type": "opportunity", + "headings": [], + "items": [], + "overallSavingsMs": 0 + } + }, + "font-display": { + "id": "font-display", + "title": "All text remains visible during webfont loads", + "description": "Leverage the font-display CSS feature to ensure text is user-visible while webfonts are loading. [Learn more](https://web.dev/font-display/).", + "score": 1, + "scoreDisplayMode": "binary", + "warnings": [], + "details": { + "type": "table", + "headings": [], + "items": [] + } + }, + "diagnostics": { + "id": "diagnostics", + "title": "Diagnostics", + "description": "Collection of useful page vitals.", + "score": null, + "scoreDisplayMode": "informative", + "details": { + "type": "debugdata", + "items": [ + { + "numRequests": 21, + "numScripts": 8, + "numStylesheets": 0, + "numFonts": 6, + "numTasks": 884, + "numTasksOver10ms": 3, + "numTasksOver25ms": 3, + "numTasksOver50ms": 2, + "numTasksOver100ms": 1, + "numTasksOver500ms": 0, + "rtt": 0.10199999999999998, + "throughput": 117042188.37154868, + "maxRtt": 0.10199999999999998, + "maxServerLatency": 5.14, + "totalByteWeight": 2402025, + "totalTaskTime": 658.9429999999998, + "mainDocumentTransferSize": 2483 + } + ] + } + }, + "network-requests": { + "id": "network-requests", + "title": "Network Requests", + "description": "Lists the network requests that were made during page load.", + "score": null, + "scoreDisplayMode": "informative", + "details": { + "type": "table", + "headings": [ + { + "key": "url", + "itemType": "url", + "text": "URL" + }, + { + "key": "protocol", + "itemType": "text", + "text": "Protocol" + }, + { + "key": "startTime", + "itemType": "ms", + "granularity": 1, + "text": "Start Time" + }, + { + "key": "endTime", + "itemType": "ms", + "granularity": 1, + "text": "End Time" + }, + { + "key": "transferSize", + "itemType": "bytes", + "displayUnit": "kb", + "granularity": 1, + "text": "Transfer Size" + }, + { + "key": "resourceSize", + "itemType": "bytes", + "displayUnit": "kb", + "granularity": 1, + "text": "Resource Size" + }, + { + "key": "statusCode", + "itemType": "text", + "text": "Status Code" + }, + { + "key": "mimeType", + "itemType": "text", + "text": "MIME Type" + }, + { + "key": "resourceType", + "itemType": "text", + "text": "Resource Type" + } + ], + "items": [ + { + "url": "http://localhost:3000/", + "protocol": "http/1.1", + "startTime": 0, + "endTime": 862.6690000000963, + "finished": true, + "transferSize": 2483, + "resourceSize": 7190, + "statusCode": 200, + "mimeType": "text/html", + "resourceType": "Document" + }, + { + "url": "http://localhost:3000/_next/static/chunks/webpack.js?ts=1706780787914", + "protocol": "http/1.1", + "startTime": 875.2880000001824, + "endTime": 890.4860000002373, + "finished": true, + "transferSize": 9638, + "resourceSize": 48521, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script" + }, + { + "url": "http://localhost:3000/_next/static/chunks/main.js?ts=1706780787914", + "protocol": "http/1.1", + "startTime": 875.583000000006, + "endTime": 1041.363000000274, + "finished": true, + "transferSize": 1112197, + "resourceSize": 4585314, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script" + }, + { + "url": "http://localhost:3000/_next/static/chunks/pages/_app.js?ts=1706780787914", + "protocol": "http/1.1", + "startTime": 881.6160000005766, + "endTime": 1023.8990000007107, + "finished": true, + "transferSize": 989804, + "resourceSize": 4273958, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script" + }, + { + "url": "http://localhost:3000/_next/static/chunks/pages/index.js?ts=1706780787914", + "protocol": "http/1.1", + "startTime": 882.1729999999661, + "endTime": 904.6750000006796, + "finished": true, + "transferSize": 47223, + "resourceSize": 296625, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script" + }, + { + "url": "http://localhost:3000/_next/static/development/_buildManifest.js?ts=1706780787914", + "protocol": "http/1.1", + "startTime": 882.5320000005377, + "endTime": 896.3940000003277, + "finished": true, + "transferSize": 633, + "resourceSize": 296, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script" + }, + { + "url": "http://localhost:3000/_next/static/development/_ssgManifest.js?ts=1706780787914", + "protocol": "http/1.1", + "startTime": 890.6810000007681, + "endTime": 899.3960000007064, + "finished": true, + "transferSize": 411, + "resourceSize": 76, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script" + }, + { + "url": "http://localhost:3000/_next/static/chunks/react-refresh.js?ts=1706780787914", + "protocol": "http/1.1", + "startTime": 875.9820000004765, + "endTime": 893.1850000008126, + "finished": true, + "transferSize": 25050, + "resourceSize": 77113, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script" + }, + { + "url": "http://localhost:3000/_next/static/development/_devMiddlewareManifest.json", + "protocol": "http/1.1", + "startTime": 1435.5080000004818, + "endTime": 1439.740000000711, + "finished": true, + "transferSize": 491, + "resourceSize": 1102, + "statusCode": 200, + "mimeType": "application/json", + "resourceType": "Fetch" + }, + { + "url": "data:image/svg+xml;charset=utf8,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 44 18'%3E%3Cp", + "protocol": "data", + "startTime": 1479.9930000008317, + "endTime": 1480.0780000005034, + "finished": true, + "transferSize": 0, + "resourceSize": 3398, + "statusCode": 200, + "mimeType": "image/svg+xml", + "resourceType": "Image" + }, + { + "url": "data:image/svg+xml;charset=utf8,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 252 180'%3E%3", + "protocol": "data", + "startTime": 1482.1799999999712, + "endTime": 1482.2780000004059, + "finished": true, + "transferSize": 0, + "resourceSize": 5348, + "statusCode": 200, + "mimeType": "image/svg+xml", + "resourceType": "Image" + }, + { + "url": "http://localhost:3000/_next/static/media/Marianne-Bold.f0ef9bad.woff2", + "protocol": "http/1.1", + "startTime": 1590.991000000031, + "endTime": 1598.1870000005074, + "finished": true, + "transferSize": 42382, + "resourceSize": 42092, + "statusCode": 200, + "mimeType": "font/woff2", + "resourceType": "Font" + }, + { + "url": "data:font/truetype;charset=utf-8;base64,d09GRgABAAAAADvoAAsAAAAAd/wAAQAAAAAAAAAAAAAAAAAAAAAAAAAAAABH", + "protocol": "data", + "startTime": 1484.3760000003385, + "endTime": 1589.9380000000747, + "finished": true, + "transferSize": 15336, + "resourceSize": 15336, + "statusCode": 200, + "mimeType": "font/truetype", + "resourceType": "Font" + }, + { + "url": "http://localhost:3000/_next/static/media/Marianne-Medium.452138fa.woff2", + "protocol": "http/1.1", + "startTime": 1591.272000000572, + "endTime": 1598.6090000005788, + "finished": true, + "transferSize": 42230, + "resourceSize": 41940, + "statusCode": 200, + "mimeType": "font/woff2", + "resourceType": "Font" + }, + { + "url": "data:font/truetype;charset=utf-8;base64,d09GRgABAAAAAAc8AAsAAAAADAgAAQAAAAAAAAAAAAAAAAAAAAAAAAAAAABH", + "protocol": "data", + "startTime": 1484.6290000004956, + "endTime": 1590.8920000001672, + "finished": true, + "transferSize": 1852, + "resourceSize": 1852, + "statusCode": 200, + "mimeType": "font/truetype", + "resourceType": "Font" + }, + { + "url": "http://localhost:3000/_next/static/media/Marianne-Regular.119b3a3e.woff2", + "protocol": "http/1.1", + "startTime": 1591.536000000815, + "endTime": 1598.7700000005134, + "finished": true, + "transferSize": 41618, + "resourceSize": 41328, + "statusCode": 200, + "mimeType": "font/woff2", + "resourceType": "Font" + }, + { + "url": "http://localhost:3000/_next/static/media/Marianne-Regular_Italic.db8a6f8b.woff2", + "protocol": "http/1.1", + "startTime": 1591.6980000001786, + "endTime": 1598.8590000006297, + "finished": true, + "transferSize": 44574, + "resourceSize": 44284, + "statusCode": 200, + "mimeType": "font/woff2", + "resourceType": "Font" + }, + { + "url": "http://localhost:3000/logo.svg", + "protocol": "http/1.1", + "startTime": 1553.9290000006076, + "endTime": 1559.573000000455, + "finished": true, + "transferSize": 2660, + "resourceSize": 5508, + "statusCode": 200, + "mimeType": "image/svg+xml", + "resourceType": "Image" + }, + { + "url": "http://localhost:3000/_next/static/media/carte-france.11cfe763.svg", + "protocol": "http/1.1", + "startTime": 1554.135000000315, + "endTime": 1559.3680000001768, + "finished": true, + "transferSize": 7159, + "resourceSize": 16521, + "statusCode": 200, + "mimeType": "image/svg+xml", + "resourceType": "Image" + }, + { + "url": "http://localhost:3000/_next/static/development/_devPagesManifest.json", + "protocol": "http/1.1", + "startTime": 1580.1060000003417, + "endTime": 1582.647000000179, + "finished": true, + "transferSize": 562, + "resourceSize": 349, + "statusCode": 200, + "mimeType": "application/json", + "resourceType": "Fetch" + }, + { + "url": "http://localhost:3000/dsfr-1.4.1.module.min.js", + "protocol": "http/1.1", + "startTime": 1632.6090000002296, + "endTime": 1636.6530000004786, + "finished": true, + "transferSize": 15722, + "resourceSize": 57644, + "statusCode": 200, + "mimeType": "application/javascript", + "resourceType": "Script" + } + ] + } + }, + "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](https://hpbn.co/primer-on-latency-and-bandwidth/).", + "score": null, + "scoreDisplayMode": "informative", + "numericValue": 0.10199999999999998, + "numericUnit": "millisecond", + "displayValue": "0 ms", + "details": { + "type": "table", + "headings": [ + { + "key": "origin", + "itemType": "text", + "text": "URL" + }, + { + "key": "rtt", + "itemType": "ms", + "granularity": 1, + "text": "Time Spent" + } + ], + "items": [ + { + "origin": "http://localhost:3000", + "rtt": 0.10199999999999998 + } + ] + } + }, + "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](https://hpbn.co/primer-on-web-performance/#analyzing-the-resource-waterfall).", + "score": null, + "scoreDisplayMode": "informative", + "numericValue": 5.14, + "numericUnit": "millisecond", + "displayValue": "10 ms", + "details": { + "type": "table", + "headings": [ + { + "key": "origin", + "itemType": "text", + "text": "URL" + }, + { + "key": "serverResponseTime", + "itemType": "ms", + "granularity": 1, + "text": "Time Spent" + } + ], + "items": [ + { + "origin": "http://localhost:3000", + "serverResponseTime": 5.14 + } + ] + } + }, + "main-thread-tasks": { + "id": "main-thread-tasks", + "title": "Tasks", + "description": "Lists the toplevel main thread tasks that executed during page load.", + "score": null, + "scoreDisplayMode": "informative", + "details": { + "type": "table", + "headings": [ + { + "key": "startTime", + "itemType": "ms", + "granularity": 1, + "text": "Start Time" + }, + { + "key": "duration", + "itemType": "ms", + "granularity": 1, + "text": "End Time" + } + ], + "items": [ + { + "duration": 5.746, + "startTime": 877.934 + }, + { + "duration": 7.104, + "startTime": 902.148 + }, + { + "duration": 400.165, + "startTime": 1057.869 + }, + { + "duration": 80.401, + "startTime": 1479.468 + }, + { + "duration": 35.214, + "startTime": 1559.909 + } + ] + } + }, + "metrics": { + "id": "metrics", + "title": "Metrics", + "description": "Collects all available metrics.", + "score": null, + "scoreDisplayMode": "informative", + "numericValue": 2806, + "numericUnit": "millisecond", + "details": { + "type": "debugdata", + "items": [ + { + "firstContentfulPaint": 250, + "firstMeaningfulPaint": 250, + "largestContentfulPaint": 2854, + "interactive": 2806, + "speedIndex": 1021, + "totalBlockingTime": 347, + "maxPotentialFID": 400, + "cumulativeLayoutShift": 0.005971847054303684, + "cumulativeLayoutShiftMainFrame": 0.005971847054303684, + "totalCumulativeLayoutShift": 0.005971847054303684, + "observedTimeOrigin": 0, + "observedTimeOriginTs": 6397803028, + "observedNavigationStart": 0, + "observedNavigationStartTs": 6397803028, + "observedFirstPaint": 1584, + "observedFirstPaintTs": 6399387404, + "observedFirstContentfulPaint": 1584, + "observedFirstContentfulPaintTs": 6399387404, + "observedFirstContentfulPaintAllFrames": 1584, + "observedFirstContentfulPaintAllFramesTs": 6399387404, + "observedFirstMeaningfulPaint": 1584, + "observedFirstMeaningfulPaintTs": 6399387404, + "observedLargestContentfulPaint": 1642, + "observedLargestContentfulPaintTs": 6399445235, + "observedLargestContentfulPaintAllFrames": 1642, + "observedLargestContentfulPaintAllFramesTs": 6399445235, + "observedTraceEnd": 3933, + "observedTraceEndTs": 6401736398, + "observedLoad": 1455, + "observedLoadTs": 6399258229, + "observedDomContentLoaded": 1450, + "observedDomContentLoadedTs": 6399252648, + "observedCumulativeLayoutShift": 0.005971847054303684, + "observedCumulativeLayoutShiftMainFrame": 0.005971847054303684, + "observedTotalCumulativeLayoutShift": 0.005971847054303684, + "observedFirstVisualChange": 1574, + "observedFirstVisualChangeTs": 6399377028, + "observedLastVisualChange": 1640, + "observedLastVisualChangeTs": 6399443028, + "observedSpeedIndex": 1588, + "observedSpeedIndexTs": 6399391014 + }, + { + "lcpInvalidated": false + } + ] + } + }, + "performance-budget": { + "id": "performance-budget", + "title": "Performance budget", + "description": "Keep the quantity and size of network requests under the targets set by the provided performance budget. [Learn more](https://developers.google.com/web/tools/lighthouse/audits/budgets).", + "score": null, + "scoreDisplayMode": "notApplicable" + }, + "timing-budget": { + "id": "timing-budget", + "title": "Timing budget", + "description": "Set a timing budget to help you keep an eye on the performance of your site. Performant sites load fast and respond to user input events quickly. [Learn more](https://developers.google.com/web/tools/lighthouse/audits/budgets).", + "score": null, + "scoreDisplayMode": "notApplicable" + }, + "resource-summary": { + "id": "resource-summary", + "title": "Keep request counts low and transfer sizes small", + "description": "To set budgets for the quantity and size of page resources, add a budget.json file. [Learn more](https://web.dev/use-lighthouse-for-performance-budgets/).", + "score": null, + "scoreDisplayMode": "informative", + "displayValue": "17 requests • 2,329 KiB", + "details": { + "type": "table", + "headings": [ + { + "key": "label", + "itemType": "text", + "text": "Resource Type" + }, + { + "key": "requestCount", + "itemType": "numeric", + "text": "Requests" + }, + { + "key": "transferSize", + "itemType": "bytes", + "text": "Transfer Size" + } + ], + "items": [ + { + "resourceType": "total", + "label": "Total", + "requestCount": 17, + "transferSize": 2384837 + }, + { + "resourceType": "script", + "label": "Script", + "requestCount": 8, + "transferSize": 2200678 + }, + { + "resourceType": "font", + "label": "Font", + "requestCount": 4, + "transferSize": 170804 + }, + { + "resourceType": "image", + "label": "Image", + "requestCount": 2, + "transferSize": 9819 + }, + { + "resourceType": "document", + "label": "Document", + "requestCount": 1, + "transferSize": 2483 + }, + { + "resourceType": "other", + "label": "Other", + "requestCount": 2, + "transferSize": 1053 + }, + { + "resourceType": "stylesheet", + "label": "Stylesheet", + "requestCount": 0, + "transferSize": 0 + }, + { + "resourceType": "media", + "label": "Media", + "requestCount": 0, + "transferSize": 0 + }, + { + "resourceType": "third-party", + "label": "Third-party", + "requestCount": 0, + "transferSize": 0 + } + ] + } + }, + "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 more](https://developers.google.com/web/fundamentals/performance/optimizing-content-efficiency/loading-third-party-javascript/).", + "score": null, + "scoreDisplayMode": "notApplicable" + }, + "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 more](https://web.dev/third-party-facades/).", + "score": null, + "scoreDisplayMode": "notApplicable" + }, + "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](https://web.dev/lighthouse-largest-contentful-paint/)", + "score": null, + "scoreDisplayMode": "informative", + "displayValue": "1 element found", + "details": { + "type": "table", + "headings": [ + { + "key": "node", + "itemType": "node", + "text": "Element" + } + ], + "items": [ + { + "node": { + "type": "node", + "lhId": "page-1-IMG", + "path": "1,HTML,1,BODY,0,DIV,1,MAIN,2,SECTION,1,DIV,1,DIV,0,IMG", + "selector": "section > div.fr-card > div.fr-card__header > img", + "boundingRect": { + "top": 520, + "bottom": 820, + "left": 68, + "right": 368, + "width": 300, + "height": 300 + }, + "snippet": "\"\"", + "nodeLabel": "section > div.fr-card > div.fr-card__header > img" + } + } + ] + } + }, + "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](https://web.dev/lcp-lazy-loading/).", + "score": 0, + "scoreDisplayMode": "binary", + "details": { + "type": "table", + "headings": [ + { + "key": "node", + "itemType": "node", + "text": "Element" + } + ], + "items": [ + { + "node": { + "type": "node", + "lhId": "page-1-IMG", + "path": "1,HTML,1,BODY,0,DIV,1,MAIN,2,SECTION,1,DIV,1,DIV,0,IMG", + "selector": "section > div.fr-card > div.fr-card__header > img", + "boundingRect": { + "top": 520, + "bottom": 820, + "left": 68, + "right": 368, + "width": 300, + "height": 300 + }, + "snippet": "\"\"", + "nodeLabel": "section > div.fr-card > div.fr-card__header > img" + } + } + ] + } + }, + "layout-shift-elements": { + "id": "layout-shift-elements", + "title": "Avoid large layout shifts", + "description": "These DOM elements contribute most to the CLS of the page.", + "score": null, + "scoreDisplayMode": "informative", + "displayValue": "5 elements found", + "details": { + "type": "table", + "headings": [ + { + "key": "node", + "itemType": "node", + "text": "Element" + }, + { + "key": "score", + "itemType": "numeric", + "granularity": 0.001, + "text": "CLS Contribution" + } + ], + "items": [ + { + "node": { + "type": "node", + "lhId": "page-3-H2", + "path": "1,HTML,1,BODY,0,DIV,1,MAIN,2,SECTION,0,H2", + "selector": "div#__next > main.fr-container > section > h2.Cartographie_titre___MSjT", + "boundingRect": { + "top": 456, + "bottom": 496, + "left": 68, + "right": 1268, + "width": 1200, + "height": 40 + }, + "snippet": "

", + "nodeLabel": "Cartographie" + }, + "score": 0.0019098926619498077 + }, + { + "node": { + "type": "node", + "lhId": "page-4-H1", + "path": "1,HTML,1,BODY,0,DIV,1,MAIN,1,DIV,0,SECTION,0,H1", + "selector": "main.fr-container > div.fr-grid-row > section.fr-col-8 > h1", + "boundingRect": { + "top": 232, + "bottom": 280, + "left": 268, + "right": 1068, + "width": 800, + "height": 48 + }, + "snippet": "

", + "nodeLabel": "Bienvenue sur Helios !" + }, + "score": 0.0016033666791677396 + }, + { + "node": { + "type": "node", + "lhId": "page-5-P", + "path": "1,HTML,1,BODY,0,DIV,1,MAIN,0,DIV,0,P", + "selector": "div#__next > main.fr-container > div.fr-alert > p", + "boundingRect": { + "top": 168, + "bottom": 192, + "left": 116, + "right": 1232, + "width": 1116, + "height": 24 + }, + "snippet": "

", + "nodeLabel": "Le site est dans sa phase pilote : il est toujours en construction et va évolue…" + }, + "score": 0.0008881000878066606 + }, + { + "node": { + "type": "node", + "lhId": "page-6-H3", + "path": "1,HTML,1,BODY,0,DIV,1,MAIN,2,SECTION,1,DIV,0,DIV,0,DIV,0,H3", + "selector": "div.fr-card > div.fr-card__body > div.fr-card__content > h3.fr-card__title", + "boundingRect": { + "top": 544, + "bottom": 572, + "left": 392, + "right": 1244, + "width": 852, + "height": 28 + }, + "snippet": "

", + "nodeLabel": "Offre de santé par région" + }, + "score": 0.0008537927566568214 + }, + { + "node": { + "type": "node", + "lhId": "page-7-DIV", + "path": "1,HTML,1,BODY,0,DIV,0,HEADER,0,DIV,0,DIV,0,DIV,1,DIV", + "selector": "div.fr-header__body > div.fr-container > div.fr-header__body-row > div.fr-header__tools", + "boundingRect": { + "top": 48, + "bottom": 80, + "left": 569, + "right": 1284, + "width": 714, + "height": 32 + }, + "snippet": "
", + "nodeLabel": "Déconnexion" + }, + "score": 0.0006743571621205494 + } + ] + } + }, + "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 more](https://web.dev/long-tasks-devtools/)", + "score": null, + "scoreDisplayMode": "informative", + "displayValue": "1 long task found", + "details": { + "type": "table", + "headings": [ + { + "key": "url", + "itemType": "url", + "text": "URL" + }, + { + "key": "startTime", + "itemType": "ms", + "granularity": 1, + "text": "Start Time" + }, + { + "key": "duration", + "itemType": "ms", + "granularity": 1, + "text": "Duration" + } + ], + "items": [ + { + "url": "http://localhost:3000/_next/static/chunks/main.js?ts=1706780787914", + "duration": 400, + "startTime": 2517.0999999999995 + } + ] + } + }, + "no-unload-listeners": { + "id": "no-unload-listeners", + "title": "Avoids `unload` event listeners", + "description": "The `unload` event does not fire reliably and listening for it can prevent browser optimizations like the Back-Forward Cache. Use `pagehide` or `visibilitychange` events instead. [Learn more](https://web.dev/bfcache/#never-use-the-unload-event)", + "score": 1, + "scoreDisplayMode": "binary" + }, + "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 more](https://web.dev/non-composited-animations)", + "score": null, + "scoreDisplayMode": "notApplicable", + "details": { + "type": "table", + "headings": [], + "items": [] + } + }, + "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 more](https://web.dev/optimize-cls/#images-without-dimensions)", + "score": 1, + "scoreDisplayMode": "binary", + "details": { + "type": "table", + "headings": [], + "items": [] + } + }, + "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](https://developers.google.com/web/tools/chrome-devtools/javascript/source-maps).", + "score": 0, + "scoreDisplayMode": "binary", + "details": { + "type": "table", + "headings": [ + { + "key": "scriptUrl", + "itemType": "url", + "subItemsHeading": { + "key": "error" + }, + "text": "URL" + }, + { + "key": "sourceMapUrl", + "itemType": "url", + "text": "Map URL" + } + ], + "items": [ + { + "scriptUrl": "http://localhost:3000/_next/static/chunks/pages/_app.js?ts=1706780787914", + "subItems": { + "type": "subitems", + "items": [ + { + "error": "Large JavaScript file is missing a source map" + } + ] + } + }, + { + "scriptUrl": "http://localhost:3000/_next/static/chunks/main.js?ts=1706780787914", + "subItems": { + "type": "subitems", + "items": [ + { + "error": "Large JavaScript file is missing a source map" + } + ] + } + }, + { + "scriptUrl": "http://localhost:3000/dsfr-1.4.1.module.min.js", + "sourceMapUrl": "http://localhost:3000/dsfr.module.min.js.map", + "subItems": { + "type": "subitems", + "items": [ + { + "error": "Error: Failed fetching source map (404)" + } + ] + } + } + ] + } + }, + "preload-lcp-image": { + "id": "preload-lcp-image", + "title": "Preload Largest Contentful Paint image", + "description": "Preload the image used by the LCP element in order to improve your LCP time. [Learn more](https://web.dev/optimize-lcp/#preload-important-resources).", + "score": 1, + "scoreDisplayMode": "numeric", + "numericValue": 0, + "numericUnit": "millisecond", + "displayValue": "", + "details": { + "type": "opportunity", + "headings": [ + { + "key": "node", + "valueType": "node", + "label": "" + }, + { + "key": "url", + "valueType": "url", + "label": "URL" + }, + { + "key": "wastedMs", + "valueType": "timespanMs", + "label": "Potential Savings" + } + ], + "items": [ + { + "node": { + "type": "node", + "lhId": "page-1-IMG", + "path": "1,HTML,1,BODY,0,DIV,1,MAIN,2,SECTION,1,DIV,1,DIV,0,IMG", + "selector": "section > div.fr-card > div.fr-card__header > img", + "boundingRect": { + "top": 520, + "bottom": 820, + "left": 68, + "right": 368, + "width": 300, + "height": 300 + }, + "snippet": "\"\"", + "nodeLabel": "section > div.fr-card > div.fr-card__header > img" + }, + "url": "http://localhost:3000/_next/static/media/carte-france.11cfe763.svg", + "wastedMs": 0 + } + ], + "overallSavingsMs": 0 + } + }, + "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 more](https://web.dev/csp-xss/)", + "score": null, + "scoreDisplayMode": "informative", + "details": { + "type": "table", + "headings": [ + { + "key": "description", + "itemType": "text", + "subItemsHeading": { + "key": "description" + }, + "text": "Description" + }, + { + "key": "directive", + "itemType": "code", + "subItemsHeading": { + "key": "directive" + }, + "text": "Directive" + }, + { + "key": "severity", + "itemType": "text", + "subItemsHeading": { + "key": "severity" + }, + "text": "Severity" + } + ], + "items": [ + { + "severity": "High", + "description": "No CSP found in enforcement mode" + } + ] + } + }, + "full-page-screenshot": { + "id": "full-page-screenshot", + "title": "Full-page screenshot", + "description": "A full-height screenshot of the final rendered page", + "score": null, + "scoreDisplayMode": "informative", + "details": { + "type": "full-page-screenshot", + "screenshot": { + "data": "", + "width": 1335, + "height": 1146 + }, + "nodes": { + "page-0-IMG": { + "top": 24, + "bottom": 104, + "left": 183, + "right": 263, + "width": 80, + "height": 80 + }, + "page-1-IMG": { + "top": 520, + "bottom": 820, + "left": 68, + "right": 368, + "width": 300, + "height": 300 + }, + "page-2-IMG": { + "top": 918, + "bottom": 1018, + "left": 242, + "right": 342, + "width": 100, + "height": 100 + }, + "page-3-H2": { + "top": 456, + "bottom": 496, + "left": 68, + "right": 1268, + "width": 1200, + "height": 40 + }, + "page-4-H1": { + "top": 232, + "bottom": 280, + "left": 268, + "right": 1068, + "width": 800, + "height": 48 + }, + "page-5-P": { + "top": 168, + "bottom": 192, + "left": 116, + "right": 1232, + "width": 1116, + "height": 24 + }, + "page-6-H3": { + "top": 544, + "bottom": 572, + "left": 392, + "right": 1244, + "width": 852, + "height": 28 + }, + "page-7-DIV": { + "top": 48, + "bottom": 80, + "left": 569, + "right": 1284, + "width": 714, + "height": 32 + }, + "4-0-META": { + "top": 0, + "bottom": 0, + "left": 0, + "right": 0, + "width": 0, + "height": 0 + }, + "4-1-META": { + "top": 0, + "bottom": 0, + "left": 0, + "right": 0, + "width": 0, + "height": 0 + }, + "4-2-META": { + "top": 0, + "bottom": 0, + "left": 0, + "right": 0, + "width": 0, + "height": 0 + }, + "4-3-SCRIPT": { + "top": 0, + "bottom": 0, + "left": 0, + "right": 0, + "width": 0, + "height": 0 + }, + "4-4-SCRIPT": { + "top": 0, + "bottom": 0, + "left": 0, + "right": 0, + "width": 0, + "height": 0 + }, + "4-5-SCRIPT": { + "top": 0, + "bottom": 0, + "left": 0, + "right": 0, + "width": 0, + "height": 0 + }, + "4-6-SCRIPT": { + "top": 0, + "bottom": 0, + "left": 0, + "right": 0, + "width": 0, + "height": 0 + }, + "4-7-SCRIPT": { + "top": 0, + "bottom": 0, + "left": 0, + "right": 0, + "width": 0, + "height": 0 + }, + "4-8-SCRIPT": { + "top": 0, + "bottom": 0, + "left": 0, + "right": 0, + "width": 0, + "height": 0 + }, + "4-9-SCRIPT": { + "top": 0, + "bottom": 0, + "left": 0, + "right": 0, + "width": 0, + "height": 0 + }, + "4-10-SCRIPT": { + "top": 0, + "bottom": 0, + "left": 0, + "right": 0, + "width": 0, + "height": 0 + }, + "4-11-SCRIPT": { + "top": 0, + "bottom": 0, + "left": 0, + "right": 0, + "width": 0, + "height": 0 + }, + "4-12-SCRIPT": { + "top": 0, + "bottom": 0, + "left": 0, + "right": 0, + "width": 0, + "height": 0 + }, + "4-13-BR": { + "top": 43, + "bottom": 62, + "left": 143, + "right": 143, + "width": 0, + "height": 19 + }, + "4-14-BODY": { + "top": 0, + "bottom": 1146, + "left": 0, + "right": 1335, + "width": 1335, + "height": 1146 + }, + "4-15-A": { + "top": 48, + "bottom": 80, + "left": 1137, + "right": 1283, + "width": 145, + "height": 32 + } + } + } + }, + "script-treemap-data": { + "id": "script-treemap-data", + "title": "Script Treemap Data", + "description": "Used for treemap app", + "score": null, + "scoreDisplayMode": "informative", + "details": { + "type": "treemap-data", + "nodes": [ + { + "name": "http://localhost:3000/", + "resourceBytes": 145 + }, + { + "name": "http://localhost:3000/_next/static/chunks/polyfills.js?ts=1706780787914", + "resourceBytes": 71 + }, + { + "name": "http://localhost:3000/_next/static/chunks/webpack.js?ts=1706780787914", + "resourceBytes": 48521, + "unusedBytes": 30474 + }, + { + "name": "http://localhost:3000/_next/static/chunks/main.js?ts=1706780787914", + "resourceBytes": 4585283, + "unusedBytes": 25579 + }, + { + "name": "http://localhost:3000/_next/static/chunks/pages/_app.js?ts=1706780787914", + "resourceBytes": 4272998, + "unusedBytes": 410856 + }, + { + "name": "http://localhost:3000/_next/static/chunks/pages/index.js?ts=1706780787914", + "resourceBytes": 296413, + "unusedBytes": 0 + }, + { + "name": "http://localhost:3000/_next/static/development/_buildManifest.js?ts=1706780787914", + "resourceBytes": 296, + "unusedBytes": 0 + }, + { + "name": "http://localhost:3000/_next/static/development/_ssgManifest.js?ts=1706780787914", + "resourceBytes": 76, + "unusedBytes": 0 + }, + { + "name": "http://localhost:3000/_next/static/chunks/react-refresh.js?ts=1706780787914", + "resourceBytes": 77113, + "unusedBytes": 0 + }, + { + "name": "http://localhost:3000/dsfr-1.4.1.module.min.js", + "resourceBytes": 57640, + "unusedBytes": 30524 + } + ] + } + }, + "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](https://web.dev/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 more](https://web.dev/aria-allowed-attr/).", + "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 more](https://web.dev/aria-name/).", + "score": null, + "scoreDisplayMode": "notApplicable" + }, + "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 more](https://web.dev/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 more](https://web.dev/aria-hidden-focus/).", + "score": null, + "scoreDisplayMode": "notApplicable" + }, + "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](https://web.dev/aria-name/).", + "score": null, + "scoreDisplayMode": "notApplicable" + }, + "aria-meter-name": { + "id": "aria-meter-name", + "title": "ARIA `meter` 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 more](https://web.dev/aria-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 more](https://web.dev/aria-name/).", + "score": null, + "scoreDisplayMode": "notApplicable" + }, + "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](https://web.dev/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](https://web.dev/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](https://web.dev/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](https://web.dev/aria-roles/).", + "score": 1, + "scoreDisplayMode": "binary", + "details": { + "type": "table", + "headings": [], + "items": [] + } + }, + "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](https://web.dev/aria-name/).", + "score": null, + "scoreDisplayMode": "notApplicable" + }, + "aria-tooltip-name": { + "id": "aria-tooltip-name", + "title": "ARIA `tooltip` 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 more](https://web.dev/aria-name/).", + "score": null, + "scoreDisplayMode": "notApplicable" + }, + "aria-treeitem-name": { + "id": "aria-treeitem-name", + "title": "ARIA `treeitem` 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 more](https://web.dev/aria-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](https://web.dev/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](https://web.dev/aria-valid-attr/).", + "score": 1, + "scoreDisplayMode": "binary", + "details": { + "type": "table", + "headings": [], + "items": [] + } + }, + "button-name": { + "id": "button-name", + "title": "Buttons 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 more](https://web.dev/button-name/).", + "score": 1, + "scoreDisplayMode": "binary", + "details": { + "type": "table", + "headings": [], + "items": [] + } + }, + "bypass": { + "id": "bypass", + "title": "The page contains a heading, skip link, or landmark region", + "description": "Adding ways to bypass repetitive content lets keyboard users navigate the page more efficiently. [Learn more](https://web.dev/bypass/).", + "score": 1, + "scoreDisplayMode": "binary", + "details": { + "type": "table", + "headings": [], + "items": [] + } + }, + "color-contrast": { + "id": "color-contrast", + "title": "Background and foreground colors have a sufficient contrast ratio", + "description": "Low-contrast text is difficult or impossible for many users to read. [Learn more](https://web.dev/color-contrast/).", + "score": 1, + "scoreDisplayMode": "binary", + "details": { + "type": "table", + "headings": [], + "items": [] + } + }, + "definition-list": { + "id": "definition-list", + "title": "`
`'s contain only properly-ordered `
` and `
` groups, `