Skip to content

Commit

Permalink
fix: avoid memory issues by doing better cleanup (#4059)
Browse files Browse the repository at this point in the history
* fix: ensure better cleanup after t

* 🤖 Automated formatting fixes

* Fix tests

* Resolve feedback
  • Loading branch information
WilcoFiers committed Jun 22, 2023
1 parent 8d135dd commit 16c5cfa
Show file tree
Hide file tree
Showing 11 changed files with 69 additions and 39 deletions.
5 changes: 4 additions & 1 deletion build/tasks/esbuild.js
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,10 @@ module.exports = function (grunt) {
bundle: true
})
.then(done)
.catch(done);
.catch(e => {
grunt.fail.fatal(e);
done();
});
});
});
}
Expand Down
2 changes: 1 addition & 1 deletion lib/checks/navigation/region-evaluate.js
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,6 @@ import * as standards from '../../commons/standards';
import matches from '../../commons/matches';
import cache from '../../core/base/cache';

const landmarkRoles = standards.getAriaRolesByType('landmark');
const implicitAriaLiveRoles = ['alert', 'log', 'status'];

export default function regionEvaluate(node, options, virtualNode) {
Expand Down Expand Up @@ -92,6 +91,7 @@ function isRegion(virtualNode, options) {
const node = virtualNode.actualNode;
const role = getRole(virtualNode);
const ariaLive = (node.getAttribute('aria-live') || '').toLowerCase().trim();
const landmarkRoles = standards.getAriaRolesByType('landmark');

// Ignore content inside of aria-live
if (
Expand Down
11 changes: 8 additions & 3 deletions lib/commons/color/incomplete-data.js
Original file line number Diff line number Diff line change
@@ -1,9 +1,12 @@
import cache from '../../core/base/cache';

const cacheKey = 'color.incompleteData';

/**
* API for handling incomplete color data
* @namespace axe.commons.color.incompleteData
* @inner
*/
let data = {};
const incompleteData = {
/**
* Store incomplete data by key with a string value
Expand All @@ -17,6 +20,7 @@ const incompleteData = {
if (typeof key !== 'string') {
throw new Error('Incomplete data: key must be a string');
}
const data = cache.get(cacheKey, () => ({}));
if (reason) {
data[key] = reason;
}
Expand All @@ -31,7 +35,8 @@ const incompleteData = {
* @return {String} String for reason we couldn't tell
*/
get: function (key) {
return data[key];
const data = cache.get(cacheKey);
return data?.[key];
},
/**
* Clear incomplete data on demand
Expand All @@ -40,7 +45,7 @@ const incompleteData = {
* @instance
*/
clear: function () {
data = {};
cache.set(cacheKey, {});
}
};

Expand Down
20 changes: 13 additions & 7 deletions lib/commons/standards/implicit-html-roles.js
Original file line number Diff line number Diff line change
Expand Up @@ -10,13 +10,19 @@ import isRowHeader from '../table/is-row-header';
import sanitize from '../text/sanitize';
import isFocusable from '../dom/is-focusable';
import { closest } from '../../core/utils';
import cache from '../../core/base/cache';
import getExplicitRole from '../aria/get-explicit-role';

const sectioningElementSelector =
getElementsByContentType('sectioning')
.map(nodeName => `${nodeName}:not([role])`)
.join(', ') +
' , main:not([role]), [role=article], [role=complementary], [role=main], [role=navigation], [role=region]';
const getSectioningElementSelector = () => {
return cache.get('sectioningElementSelector', () => {
return (
getElementsByContentType('sectioning')
.map(nodeName => `${nodeName}:not([role])`)
.join(', ') +
' , main:not([role]), [role=article], [role=complementary], [role=main], [role=navigation], [role=region]'
);
});
};

// sectioning elements only have an accessible name if the
// aria-label, aria-labelledby, or title attribute has valid
Expand Down Expand Up @@ -64,7 +70,7 @@ const implicitHtmlRoles = {
fieldset: 'group',
figure: 'figure',
footer: vNode => {
const sectioningElement = closest(vNode, sectioningElementSelector);
const sectioningElement = closest(vNode, getSectioningElementSelector());

return !sectioningElement ? 'contentinfo' : null;
},
Expand All @@ -78,7 +84,7 @@ const implicitHtmlRoles = {
h5: 'heading',
h6: 'heading',
header: vNode => {
const sectioningElement = closest(vNode, sectioningElementSelector);
const sectioningElement = closest(vNode, getSectioningElementSelector());

return !sectioningElement ? 'banner' : null;
},
Expand Down
7 changes: 1 addition & 6 deletions lib/core/base/virtual-node/virtual-node.js
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,6 @@ import { isXHTML, validInputTypes } from '../../utils';
import { isFocusable, getTabbableElements } from '../../../commons/dom';
import cache from '../cache';

let isXHTMLGlobal;
let nodeIndex = 0;

class VirtualNode extends AbstractVirtualNode {
Expand All @@ -27,11 +26,7 @@ class VirtualNode extends AbstractVirtualNode {

this._isHidden = null; // will be populated by axe.utils.isHidden
this._cache = {};

if (typeof isXHTMLGlobal === 'undefined') {
isXHTMLGlobal = isXHTML(node.ownerDocument);
}
this._isXHTML = isXHTMLGlobal;
this._isXHTML = isXHTML(node.ownerDocument);

// we will normalize the type prop for inputs by looking strictly
// at the attribute and not what the browser resolves the type
Expand Down
1 change: 1 addition & 0 deletions lib/core/utils/frame-messenger/message-id.js
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
import { v4 as createUuid } from '../uuid';

// No cache, so that this can persist across axe.run calls
const messageIds = [];

export function createMessageId() {
Expand Down
7 changes: 3 additions & 4 deletions lib/core/utils/frame-messenger/post-message.js
Original file line number Diff line number Diff line change
Expand Up @@ -15,10 +15,6 @@ import { createMessageId } from './message-id';
* @return {Boolean} true if the message was sent
*/
export function postMessage(win, data, sendToParent, replyHandler) {
if (typeof replyHandler === 'function') {
storeReplyHandler(data.channelId, replyHandler, sendToParent);
}

// Prevent messaging to an inappropriate window
sendToParent ? assertIsParentWindow(win) : assertIsFrameWindow(win);
if (data.message instanceof Error && !sendToParent) {
Expand All @@ -37,6 +33,9 @@ export function postMessage(win, data, sendToParent, replyHandler) {
return false;
}

if (typeof replyHandler === 'function') {
storeReplyHandler(data.channelId, replyHandler, sendToParent);
}
// There is no way to know the origin of `win`, so we'll try them all.
allowedOrigins.forEach(origin => {
try {
Expand Down
5 changes: 1 addition & 4 deletions lib/core/utils/get-selector.js
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,6 @@ import matchesSelector from './element-matches';
import isXHTML from './is-xhtml';
import getShadowSelector from './get-shadow-selector';

let xhtml;
const ignoredAttributes = [
'class',
'style',
Expand Down Expand Up @@ -238,9 +237,7 @@ function getElmId(elm) {
* @return {String|Array<String>} Base CSS selector for the node
*/
function getBaseSelector(elm) {
if (typeof xhtml === 'undefined') {
xhtml = isXHTML(document);
}
const xhtml = isXHTML(document);
return escapeSelector(xhtml ? elm.localName : elm.nodeName.toLowerCase());
}

Expand Down
8 changes: 5 additions & 3 deletions lib/core/utils/is-xhtml.js
Original file line number Diff line number Diff line change
@@ -1,15 +1,17 @@
import memoize from './memoize';

/**
* Determines if a document node is XHTML
* @method isXHTML
* @memberof axe.utils
* @param {Node} doc a document node
* @return {Boolean}
*/
function isXHTML(doc) {
if (!doc.createElement) {
const isXHTML = memoize(doc => {
if (!doc?.createElement) {
return false;
}
return doc.createElement('A').localName === 'A';
}
});

export default isXHTML;
24 changes: 14 additions & 10 deletions lib/core/utils/query-selector-all-filter.js
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
import cache from '../base/cache';
import { matchesExpression, convertSelector } from './matches';
import { getNodesMatchingExpression } from './selector-cache';

Expand All @@ -19,17 +20,20 @@ function createLocalVariables(
return retVal;
}

/**
* Allocating new objects in createLocalVariables is quite expensive given
* that matchExpressions is in the hot path.
*
* Keep track of previously allocated objects to avoid useless allocations
* and garbage collection. This is intentionally shared between calls of
* matchExpressions.
*/
const recycledLocalVariables = [];

function matchExpressions(domTree, expressions, filter) {
/**
* Allocating new objects in createLocalVariables is quite expensive given
* that matchExpressions is in the hot path.
*
* Keep track of previously allocated objects to avoid useless allocations
* and garbage collection. This is intentionally shared between calls of
* matchExpressions.
*/
const recycledLocalVariables = cache.get(
'qsa.recycledLocalVariables',
() => []
);

const stack = [];
const vNodes = Array.isArray(domTree) ? domTree : [domTree];
let currentLevel = createLocalVariables(
Expand Down
18 changes: 18 additions & 0 deletions test/checks/navigation/region.js
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@ describe('region', function () {
afterEach(function () {
fixture.innerHTML = '';
checkContext.reset();
axe.reset();
});

it('should return true when content is inside the region', function () {
Expand All @@ -28,6 +29,23 @@ describe('region', function () {
assert.isTrue(checkEvaluate.apply(checkContext, checkArgs));
});

it('should return true when a region role is added to standards', () => {
axe.configure({
standards: {
ariaRoles: {
feed: {
type: 'landmark'
}
}
}
});
var checkArgs = checkSetup(
'<div role="feed" id="target">This is random content.</div>' +
'<div role="main"><h1 id="mainheader">Introduction</h1></div>'
);
assert.isTrue(checkEvaluate.apply(checkContext, checkArgs));
});

it('should return false when img content is outside the region', function () {
var checkArgs = checkSetup(
'<img id="target" src=""><div role="main"><h1 id="mainheader" tabindex="0">Introduction</h1></div>'
Expand Down

0 comments on commit 16c5cfa

Please sign in to comment.