/* 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.
 */
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) {
  return getDb().then(function (db) {
    var tx = db.transaction(dbCacheTable, 'readwrite');
    var store = tx.objectStore(dbCacheTable);
    store.put({
      address: address,
      hashCode: hashCode,
      translated: translated,
      expires: Date.now() + cacheExpiration,
      format: format,
    });
    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,
        } : 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).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));
};