Decidi reiterar meu comentário à resposta de Pumbaa80 como uma resposta separada para que seja mais fácil reutilizá-la como uma biblioteca.
Peguei o código do Pumbaa80, refinei um pouco, adicionei testes e publiquei como um módulo npm aqui:
https://www.npmjs.com/package/mock-local-storage .
Aqui está um código-fonte:
https://github.com/letsrock-today/mock-local-storage/blob/master/src/mock-localstorage.js
Alguns testes:
https://github.com/letsrock-today/mock-local-storage/blob/master/test/mock-localstorage.js
O módulo cria localStorage e sessionStorage simulados no objeto global (janela ou global, qual deles é definido).
Em meus outros testes de projeto, eu o solicitei com o mocha da seguinte forma: mocha -r mock-local-storage
para disponibilizar definições globais para todo o código em teste.
Basicamente, o código se parece com o seguinte:
(function (glob) {
function createStorage() {
let s = {},
noopCallback = () => {},
_itemInsertionCallback = noopCallback;
Object.defineProperty(s, 'setItem', {
get: () => {
return (k, v) => {
k = k + '';
_itemInsertionCallback(s.length);
s[k] = v + '';
};
}
});
Object.defineProperty(s, 'getItem', {
// ...
});
Object.defineProperty(s, 'removeItem', {
// ...
});
Object.defineProperty(s, 'clear', {
// ...
});
Object.defineProperty(s, 'length', {
get: () => {
return Object.keys(s).length;
}
});
Object.defineProperty(s, "key", {
// ...
});
Object.defineProperty(s, 'itemInsertionCallback', {
get: () => {
return _itemInsertionCallback;
},
set: v => {
if (!v || typeof v != 'function') {
v = noopCallback;
}
_itemInsertionCallback = v;
}
});
return s;
}
glob.localStorage = createStorage();
glob.sessionStorage = createStorage();
}(typeof window !== 'undefined' ? window : global));
Observe que todos os métodos adicionados por meio de Object.defineProperty
para que não sejam iterados, acessados ou removidos como itens regulares e não contam em comprimento. Também adicionei uma maneira de registrar o retorno de chamada que é chamado quando um item está prestes a ser colocado no objeto. Este retorno de chamada pode ser usado para emular o erro de cota excedida em testes.
Profit!