a31d142253
When source is already cached and you reload, the source map is lost which makes debugging async functions difficult.
153 lines
4.1 KiB
JavaScript
153 lines
4.1 KiB
JavaScript
/* Copyright 2017 Mozilla Foundation
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
/* eslint-disable no-var */
|
|
|
|
var babel = require("plugin-babel");
|
|
|
|
var cacheExpiration = 60 /* min */ * 60 * 1000;
|
|
var dbVersion = 1;
|
|
var dbName = "babelcache";
|
|
var dbCacheTable = "translated";
|
|
var dbPromise;
|
|
|
|
function getDb() {
|
|
if (!dbPromise) {
|
|
dbPromise = new Promise(function (resolve, reject) {
|
|
var request = indexedDB.open(dbName, dbVersion);
|
|
request.onupgradeneeded = function () {
|
|
var db = request.result;
|
|
db.createObjectStore(dbCacheTable, { keyPath: "address" });
|
|
};
|
|
request.onsuccess = function () {
|
|
var db = request.result;
|
|
resolve(db);
|
|
};
|
|
request.onerror = function () {
|
|
console.warn("getDb: " + request.error);
|
|
reject(request.error);
|
|
};
|
|
});
|
|
}
|
|
return dbPromise;
|
|
}
|
|
|
|
function storeCache(address, hashCode, translated, format, sourceMap) {
|
|
return getDb().then(function (db) {
|
|
var tx = db.transaction(dbCacheTable, "readwrite");
|
|
var store = tx.objectStore(dbCacheTable);
|
|
store.put({
|
|
address,
|
|
hashCode,
|
|
translated,
|
|
expires: Date.now() + cacheExpiration,
|
|
format,
|
|
sourceMap,
|
|
});
|
|
return new Promise(function (resolve, reject) {
|
|
tx.oncomplete = function () {
|
|
resolve();
|
|
};
|
|
tx.onerror = function () {
|
|
resolve();
|
|
};
|
|
});
|
|
});
|
|
}
|
|
|
|
function loadCache(address, hashCode) {
|
|
return getDb().then(function (db) {
|
|
var tx = db.transaction(dbCacheTable, "readonly");
|
|
var store = tx.objectStore(dbCacheTable);
|
|
var getAddress = store.get(address);
|
|
return new Promise(function (resolve, reject) {
|
|
tx.oncomplete = function () {
|
|
var found = getAddress.result;
|
|
var isValid =
|
|
found && found.hashCode === hashCode && Date.now() < found.expires;
|
|
resolve(
|
|
isValid
|
|
? {
|
|
translated: found.translated,
|
|
format: found.format,
|
|
sourceMap: found.sourceMap,
|
|
}
|
|
: null
|
|
);
|
|
};
|
|
tx.onerror = function () {
|
|
resolve(null);
|
|
};
|
|
});
|
|
});
|
|
}
|
|
|
|
var encoder = new TextEncoder("utf-8");
|
|
function sha256(str) {
|
|
var buffer = encoder.encode(str);
|
|
return crypto.subtle.digest("SHA-256", buffer).then(function (hash) {
|
|
var data = new Int32Array(hash);
|
|
return (
|
|
data[0].toString(36) +
|
|
"-" +
|
|
data[1].toString(36) +
|
|
"-" +
|
|
data[2].toString(36) +
|
|
"-" +
|
|
data[3].toString(36)
|
|
);
|
|
});
|
|
}
|
|
|
|
exports.translate = function (load, opt) {
|
|
var savedHashCode, babelTranslateError;
|
|
return sha256(load.source)
|
|
.then(function (hashCode) {
|
|
savedHashCode = hashCode;
|
|
return loadCache(load.address, hashCode);
|
|
})
|
|
.then(
|
|
function (cache) {
|
|
if (cache) {
|
|
load.metadata.format = cache.format;
|
|
return cache.translated;
|
|
}
|
|
return babel.translate.call(this, load, opt).then(
|
|
function (translated) {
|
|
return storeCache(
|
|
load.address,
|
|
savedHashCode,
|
|
translated,
|
|
load.metadata.format,
|
|
load.metadata.sourceMap
|
|
).then(function () {
|
|
return translated;
|
|
});
|
|
},
|
|
function (reason) {
|
|
throw (babelTranslateError = reason);
|
|
}
|
|
);
|
|
}.bind(this)
|
|
)
|
|
.catch(
|
|
function (reason) {
|
|
if (babelTranslateError) {
|
|
throw babelTranslateError;
|
|
}
|
|
return babel.translate.call(this, load, opt);
|
|
}.bind(this)
|
|
);
|
|
};
|