Files
espocrm/client/src/cache.js
Yuri Kuznetsov febbfa2126 ref
2023-06-09 18:35:39 +03:00

223 lines
5.5 KiB
JavaScript

/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2023 Yurii Kuznietsov, Taras Machyshyn, Oleksii Avramenko
* Website: https://www.espocrm.com
*
* EspoCRM is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* EspoCRM is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
*
* The interactive user interfaces in modified source and object code versions
* of this program must display Appropriate Legal Notices, as required under
* Section 5 of the GNU General Public License version 3.
*
* In accordance with Section 7(b) of the GNU General Public License version 3,
* these Appropriate Legal Notices must retain the display of the "EspoCRM" word.
************************************************************************/
/** @module cache */
/**
* Cache for source and resource files.
*/
class Cache {
/**
* @param {Number} [cacheTimestamp] A cache timestamp.
*/
constructor(cacheTimestamp) {
this.basePrefix = this.prefix;
if (cacheTimestamp) {
this.prefix = this.basePrefix + '-' + cacheTimestamp;
}
if (!this.get('app', 'timestamp')) {
this.storeTimestamp();
}
}
/** @private */
prefix = 'cache'
/**
* Handle actuality. Clears cache if not actual.
*
* @param {Number} cacheTimestamp A cache timestamp.
*/
handleActuality(cacheTimestamp) {
let storedTimestamp = this.getCacheTimestamp();
if (storedTimestamp) {
if (storedTimestamp !== cacheTimestamp) {
this.clear();
this.set('app', 'cacheTimestamp', cacheTimestamp);
this.storeTimestamp();
}
return;
}
this.clear();
this.set('app', 'cacheTimestamp', cacheTimestamp);
this.storeTimestamp();
}
/**
* Get a cache timestamp.
*
* @returns {number}
*/
getCacheTimestamp() {
return parseInt(this.get('app', 'cacheTimestamp') || 0);
}
/**
* @todo Revise whether is needed.
*/
storeTimestamp() {
let frontendCacheTimestamp = Date.now();
this.set('app', 'timestamp', frontendCacheTimestamp);
}
/**
* @private
* @param {string} type
* @returns {string}
*/
composeFullPrefix(type) {
return this.prefix + '-' + type;
}
/**
* @private
* @param {string} type
* @param {string} name
* @returns {string}
*/
composeKey(type, name) {
return this.composeFullPrefix(type) + '-' + name;
}
/**
* @private
* @param {string} type
*/
checkType(type) {
if (typeof type === 'undefined' && toString.call(type) !== '[object String]') {
throw new TypeError("Bad type \"" + type + "\" passed to Cache().");
}
}
/**
* Get a stored value.
*
* @param {string} type A type/category.
* @param {string} name A name.
* @returns {string|null} Null if no stored value.
*/
get(type, name) {
this.checkType(type);
let key = this.composeKey(type, name);
let stored;
try {
stored = localStorage.getItem(key);
}
catch (error) {
console.error(error);
return null;
}
if (stored) {
let result = stored;
if (stored.length > 9 && stored.substring(0, 9) === '__JSON__:') {
let jsonString = stored.slice(9);
try {
result = JSON.parse(jsonString);
}
catch (error) {
result = stored;
}
}
return result;
}
return null;
}
/**
* Store a value.
*
* @param {string} type A type/category.
* @param {string} name A name.
* @param {any} value A value.
*/
set(type, name, value) {
this.checkType(type);
let key = this.composeKey(type, name);
if (value instanceof Object || Array.isArray(value)) {
value = '__JSON__:' + JSON.stringify(value);
}
try {
localStorage.setItem(key, value);
}
catch (error) {
console.log('Local storage limit exceeded.');
}
}
/**
* Clear a stored value.
*
* @param {string} [type] A type/category.
* @param {string} [name] A name.
*/
clear(type, name) {
let reText;
if (typeof type !== 'undefined') {
if (typeof name === 'undefined') {
reText = '^' + this.composeFullPrefix(type);
}
else {
reText = '^' + this.composeKey(type, name);
}
}
else {
reText = '^' + this.basePrefix + '-';
}
let re = new RegExp(reText);
for (let i in localStorage) {
if (re.test(i)) {
delete localStorage[i];
}
}
}
}
export default Cache;