/* This Source Code Form is subject to the terms of the Mozilla Public
 * License, v. 2.0. If a copy of the MPL was not distributed with this
 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */

var util = {
  // Compare the contents of two ArrayBuffer(View)s
  memcmp: function util_memcmp(x, y) {
    if (!x || !y) { return false; }

    var xb = new Uint8Array(x);
    var yb = new Uint8Array(y);
    if (x.byteLength !== y.byteLength) { return false; }

    for (var i=0; i<xb.byteLength; ++i) {
      if (xb[i] !== yb[i]) {
        return false;
      }
    }
    return true;
  },

  // Convert an ArrayBufferView to a hex string
  abv2hex: function util_abv2hex(abv) {
    var b = new Uint8Array(abv);
    var hex = "";
    for (var i=0; i <b.length; ++i) {
      var zeropad = (b[i] < 0x10) ? "0" : "";
      hex += zeropad + b[i].toString(16);
    }
    return hex;
  },

  // Convert a hex string to an ArrayBufferView
  hex2abv: function util_hex2abv(hex) {
    if (hex.length % 2 !== 0) {
      hex = "0" + hex;
    }

    var abv = new Uint8Array(hex.length / 2);
    for (var i=0; i<abv.length; ++i) {
      abv[i] = parseInt(hex.substr(2*i, 2), 16);
    }
    return abv;
  },

  clone: function (obj) {
    return new Promise(resolve => {
      let {port1, port2} = new MessageChannel();

      // Wait for the cloned object to arrive.
      port1.onmessage = msg => resolve(msg.data);

      // Clone the object.
      port2.postMessage(obj);
    });
  },

  cloneExportCompareKeys: function (key) {
    return util.clone(key).then(clone => {
      var exports = [];

      if (key instanceof CryptoKey) {
        exports.push(crypto.subtle.exportKey("raw", key));
        exports.push(crypto.subtle.exportKey("raw", clone));
      } else {
        exports.push(crypto.subtle.exportKey("spki", key.publicKey));
        exports.push(crypto.subtle.exportKey("spki", clone.publicKey));
        exports.push(crypto.subtle.exportKey("pkcs8", key.privateKey));
        exports.push(crypto.subtle.exportKey("pkcs8", clone.privateKey));
      }

      return Promise.all(exports).then(pairs => {
        for (var i = 0; i < pairs.length; i += 2) {
          if (!util.memcmp(pairs[i], pairs[i + 1])) {
            throw new Error("keys don't match");
          }
        }

        return clone;
      });
    });
  }
};

function exists(x) {
  return (x !== undefined);
}

function hasFields(object, fields) {
  return fields
          .map(x => exists(object[x]))
          .reduce((x,y) => (x && y));
}

function hasKeyFields(x) {
  return hasFields(x, ["algorithm", "extractable", "type", "usages"]);
}

function hasBaseJwkFields(x) {
  return hasFields(x, ["kty", "alg", "ext", "key_ops"]);
}

function shallowArrayEquals(x, y) {
  if (x.length != y.length) {
    return false;
  }

  for (i in x) {
    if (x[i] != y[i]) {
      return false;
    }
  }

  return true;
}