How to write a Chrome extension to prevent bias in hiring

Kim T
Kim T
Jun 20 · 4 min read
Antibias Chrome Extension

Every person is biased, affecting the decisions they make everyday. People with a position of power can have their biases magnified resulting in huge effects across society. Technology is often considered a force for good, but can also magnify cognitive biases.

LinkedIn makes it really easy to find and review potential candidates for job positions. Although more accurate/trustworthy information helps us hire the right people, it can also help reinforce our biases. Some of my colleagues wondered whether actually removing or hiding some information would equal the playing field.

I was tasked with writing the Chrome extension, and this is a quick explainer of how it works.

Every Chrome extension has a manifest.json which holds the configuration settings. We need to give permissions for the extension (which runs in the toolbar) to communicate with page tabs and the page local storage. We also want to allow it to run on only linkedin.com urls:

“permissions”: [
“tabs”,
“storage”,
http://*.linkedin.com/*",
https://*.linkedin.com/*"
],

Next we add a background script which runs behind-the-scenes to control the storage and toolbar:

"background": {
"persistent": false,
"scripts": ["js/background.js"]
},

In background.js we listen to the chrome extension installed event and then trigger a function to update the toolbar icon:

chrome.runtime.onInstalled.addListener(function (details) {
toggleState(enabled);
});
function toggleState(state) {
var icon = chrome.browserAction;
if (state === true) {
icon.setBadgeText({text: 'on'});
icon.setBadgeBackgroundColor({color: '#0b8043'});
} else {
icon.setBadgeText({text: 'off'});
icon.setBadgeBackgroundColor({color: '#616161'});
}
}

In our package.json we also configure the popup window which shows when clicking the toolbar icon:

"browser_action": {
"default_popup": "html/popup.html",
"default_icon": {
"16": "images/16x16.png",
"32": "images/32x32.png",
"48": "images/48x48.png",
"128": "images/128x128.png"
},
"matches": ["https://*.linkedin.com/*"]
},

Next we add the content script which will run on the page when on a linkedin page:

"content_scripts": [
{
"persistent": false,
"matches": ["https://*.linkedin.com/*"],
"web_accessible_resources": ["css/*", "js/*", "images/*"],
"css" : ["css/content.css"],
"js" : ["js/content.js"],
"run_at": "document_start"
}
],

The content script loads a css file to hide/replace any images of people on the page. I decided to go with manual css selectors as it would place an uncessary load on the browser to intelligently detect which images are of people:

Hide text (except on hover) using an svg squiggle:

.antibias .profile .name a:not(:hover) {
background-image: url('');
background-position: 0px center;
background-repeat: repeat-x;
background-size: 26px;
color: transparent !important;
display: inline-block;
}

Hide image (except on hover) using a png background:

.antibias .profile .photo:not(:hover) {
background-repeat: repeat !important;
background-size: 800px;
content: url('');
background-image: url('') !important;
}

Unfortunately there were a few compromises to get this working:

  1. Had to inline images into css as LinkedIn CSP security policy was blocking them loading via url
  2. Needed to use negative css :not logic to overwrite the values when static, but use the original values when hovering
  3. Had to replace inner content with a blank image, and always use background image to get the tiling effect we wanted.

The final file content.js runs some more advanced functionality. It listens to various window events and then updates the page content:

window.addEventListener('DOMContentLoaded', function() {
updatePage();
});
window.addEventListener('load', function() {
updatePage();
});
window.onpopstate = function() {
window.setTimeout(function () {
updatePage();
}, 200);
};
window.addEventListener('scroll', function(e) {
scrolling += 1;
if (scrolling === 10) {
window.requestAnimationFrame(function() {
updatePage();
});
scrolling = 0;
}
});

Update page loops through all the css selectors and updates their background position to be unique based on their image src url:

function updatePage() {
document.body.classList.add('antibias');
selectors.forEach((selector) => {
var elements = document.querySelectorAll(selector);
[].forEach.call(elements, function(element) {
updateElement(element);
});
});
}
function updateElement(el) {
var val = getImageVal(el);
if (val && val !== '') {
if (!el.hasAttribute('data-bias')) {
el.setAttribute('data-bias', val);
}
var hash = hashCode(val);
el.style.backgroundPosition = `${hash.substring(0, 3)}px ${hash.substring(1, 4)}px`;
el.style.filter = `hue-rotate(${hash.substring(2, 5)}deg)`;
}
}
function getImageVal(el) {
return el.hasAttribute('src') ? el.getAttribute('src') : el.style.backgroundImage.slice(5, -2);
}

That’s a quick walkthrough of how the extension works, the actual code has more communication between tabs and code to store your preference in localstorage. As a result of this we were honored by Fast Company for a world-changing idea!

Fast Company World-Changing Ideas: Honorable Mention 2019

You can install the extension here:
https://chrome.google.com/webstore/detail/antibias/gfeilphhbcfaekkffklahpndiidiloln

And view the full source code here:
https://github.com/Beyond-Digital/antibias-chrome-extension

Creative Technology Concepts & Code

find, learn, share

Kim T

Written by

Kim T

Creative Technologist, coder, music producer, and bike fanatic. I find creative uses for technology.

Creative Technology Concepts & Code

find, learn, share

Welcome to a place where words matter. On Medium, smart voices and original ideas take center stage - with no ads in sight. Watch
Follow all the topics you care about, and we’ll deliver the best stories for you to your homepage and inbox. Explore
Get unlimited access to the best stories on Medium — and support writers while you’re at it. Just $5/month. Upgrade