Skondensowałem test binarny do tej funkcji, której używam:
function getStorageTotalSize(upperLimit/*in bytes*/) {
var store = localStorage, testkey = "$_test"; // (NOTE: Test key is part of the storage!!! It should also be an even number of characters)
var test = function (_size) { try { store.removeItem(testkey); store.setItem(testkey, new Array(_size + 1).join('0')); } catch (_ex) { return false; } return true; }
var backup = {};
for (var i = 0, n = store.length; i < n; ++i) backup[store.key(i)] = store.getItem(store.key(i));
store.clear(); // (you could iterate over the items and backup first then restore later)
var low = 0, high = 1, _upperLimit = (upperLimit || 1024 * 1024 * 1024) / 2, upperTest = true;
while ((upperTest = test(high)) && high < _upperLimit) { low = high; high *= 2; }
if (!upperTest) {
var half = ~~((high - low + 1) / 2); // (~~ is a faster Math.floor())
high -= half;
while (half > 0) high += (half = ~~(half / 2)) * (test(high) ? 1 : -1);
high = testkey.length + high;
}
if (high > _upperLimit) high = _upperLimit;
store.removeItem(testkey);
for (var p in backup) store.setItem(p, backup[p]);
return high * 2; // (*2 because of Unicode storage)
}
Tworzy kopię zapasową zawartości przed testowaniem, a następnie przywraca ją.
Jak to działa: podwaja rozmiar, dopóki limit nie zostanie osiągnięty lub test się nie powiedzie. Następnie zapisuje połowę odległości między niskim a wysokim i odejmuje / dodaje za każdym razem połowę połowy (odejmuje w przypadku niepowodzenia i dodaje w przypadku sukcesu); szlifowanie do właściwej wartości.
upperLimit
Domyślnie jest to 1 GB i ogranicza jedynie ekspansję wykładniczą w górę przed rozpoczęciem wyszukiwania binarnego. Wątpię, czy trzeba to zmienić, ale zawsze myślę o przyszłości. ;)
W Chrome:
> getStorageTotalSize();
> 10485762
> 10485762/2
> 5242881
> localStorage.setItem("a", new Array(5242880).join("0")) // works
> localStorage.setItem("a", new Array(5242881).join("0")) // fails ('a' takes one spot [2 bytes])
IE11, Edge i FireFox również zgłaszają ten sam maksymalny rozmiar (10485762 bajtów).