2013-06-10 11:57:33 -04:00
|
|
|
/**
|
|
|
|
* Utility functions for web applications.
|
|
|
|
*
|
|
|
|
* @author Dave Longley
|
|
|
|
*
|
|
|
|
* Copyright (c) 2010-2012 Digital Bazaar, Inc.
|
|
|
|
*/
|
|
|
|
(function() {
|
|
|
|
/* ########## Begin module implementation ########## */
|
|
|
|
function initModule(forge) {
|
|
|
|
|
|
|
|
/* Utilities API */
|
|
|
|
var util = forge.util = forge.util || {};
|
|
|
|
|
|
|
|
// define setImmediate and nextTick
|
|
|
|
if(typeof process === 'undefined' || !process.nextTick) {
|
|
|
|
if(typeof setImmediate === 'function') {
|
|
|
|
util.setImmediate = setImmediate;
|
|
|
|
util.nextTick = function(callback) {
|
|
|
|
return setImmediate(callback);
|
|
|
|
};
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
util.setImmediate = function(callback) {
|
|
|
|
setTimeout(callback, 0);
|
|
|
|
};
|
|
|
|
util.nextTick = util.setImmediate;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
util.nextTick = process.nextTick;
|
|
|
|
if(typeof setImmediate === 'function') {
|
|
|
|
util.setImmediate = setImmediate;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
util.setImmediate = util.nextTick;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-05 10:45:02 -04:00
|
|
|
// define isArray
|
|
|
|
util.isArray = Array.isArray || function(x) {
|
|
|
|
return Object.prototype.toString.call(x) === '[object Array]';
|
|
|
|
};
|
|
|
|
|
2013-06-10 11:57:33 -04:00
|
|
|
/**
|
|
|
|
* Constructor for a byte buffer.
|
|
|
|
*
|
|
|
|
* @param b the bytes to wrap (as a UTF-8 string) (optional).
|
|
|
|
*/
|
|
|
|
util.ByteBuffer = function(b) {
|
|
|
|
// the data in this buffer
|
|
|
|
this.data = b || '';
|
|
|
|
// the pointer for reading from this buffer
|
|
|
|
this.read = 0;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the number of bytes in this buffer.
|
|
|
|
*
|
|
|
|
* @return the number of bytes in this buffer.
|
|
|
|
*/
|
|
|
|
util.ByteBuffer.prototype.length = function() {
|
|
|
|
return this.data.length - this.read;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets whether or not this buffer is empty.
|
|
|
|
*
|
|
|
|
* @return true if this buffer is empty, false if not.
|
|
|
|
*/
|
|
|
|
util.ByteBuffer.prototype.isEmpty = function() {
|
2013-08-05 10:45:02 -04:00
|
|
|
return this.length() <= 0;
|
2013-06-10 11:57:33 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Puts a byte in this buffer.
|
|
|
|
*
|
|
|
|
* @param b the byte to put.
|
2013-08-05 10:45:02 -04:00
|
|
|
*
|
|
|
|
* @return this buffer.
|
2013-06-10 11:57:33 -04:00
|
|
|
*/
|
|
|
|
util.ByteBuffer.prototype.putByte = function(b) {
|
|
|
|
this.data += String.fromCharCode(b);
|
2013-08-05 10:45:02 -04:00
|
|
|
return this;
|
2013-06-10 11:57:33 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Puts a byte in this buffer N times.
|
|
|
|
*
|
|
|
|
* @param b the byte to put.
|
|
|
|
* @param n the number of bytes of value b to put.
|
2013-08-05 10:45:02 -04:00
|
|
|
*
|
|
|
|
* @return this buffer.
|
2013-06-10 11:57:33 -04:00
|
|
|
*/
|
|
|
|
util.ByteBuffer.prototype.fillWithByte = function(b, n) {
|
|
|
|
b = String.fromCharCode(b);
|
|
|
|
var d = this.data;
|
|
|
|
while(n > 0) {
|
|
|
|
if(n & 1) {
|
|
|
|
d += b;
|
|
|
|
}
|
|
|
|
n >>>= 1;
|
|
|
|
if(n > 0) {
|
|
|
|
b += b;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
this.data = d;
|
2013-08-05 10:45:02 -04:00
|
|
|
return this;
|
2013-06-10 11:57:33 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Puts bytes in this buffer.
|
|
|
|
*
|
|
|
|
* @param bytes the bytes (as a UTF-8 encoded string) to put.
|
2013-08-05 10:45:02 -04:00
|
|
|
*
|
|
|
|
* @return this buffer.
|
2013-06-10 11:57:33 -04:00
|
|
|
*/
|
|
|
|
util.ByteBuffer.prototype.putBytes = function(bytes) {
|
|
|
|
this.data += bytes;
|
2013-08-05 10:45:02 -04:00
|
|
|
return this;
|
2013-06-10 11:57:33 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Puts a UTF-16 encoded string into this buffer.
|
|
|
|
*
|
|
|
|
* @param str the string to put.
|
2013-08-05 10:45:02 -04:00
|
|
|
*
|
|
|
|
* @return this buffer.
|
2013-06-10 11:57:33 -04:00
|
|
|
*/
|
|
|
|
util.ByteBuffer.prototype.putString = function(str) {
|
|
|
|
this.data += util.encodeUtf8(str);
|
2013-08-05 10:45:02 -04:00
|
|
|
return this;
|
2013-06-10 11:57:33 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Puts a 16-bit integer in this buffer in big-endian order.
|
|
|
|
*
|
|
|
|
* @param i the 16-bit integer.
|
2013-08-05 10:45:02 -04:00
|
|
|
*
|
|
|
|
* @return this buffer.
|
2013-06-10 11:57:33 -04:00
|
|
|
*/
|
|
|
|
util.ByteBuffer.prototype.putInt16 = function(i) {
|
|
|
|
this.data +=
|
|
|
|
String.fromCharCode(i >> 8 & 0xFF) +
|
|
|
|
String.fromCharCode(i & 0xFF);
|
2013-08-05 10:45:02 -04:00
|
|
|
return this;
|
2013-06-10 11:57:33 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Puts a 24-bit integer in this buffer in big-endian order.
|
|
|
|
*
|
|
|
|
* @param i the 24-bit integer.
|
2013-08-05 10:45:02 -04:00
|
|
|
*
|
|
|
|
* @return this buffer.
|
2013-06-10 11:57:33 -04:00
|
|
|
*/
|
|
|
|
util.ByteBuffer.prototype.putInt24 = function(i) {
|
|
|
|
this.data +=
|
|
|
|
String.fromCharCode(i >> 16 & 0xFF) +
|
|
|
|
String.fromCharCode(i >> 8 & 0xFF) +
|
|
|
|
String.fromCharCode(i & 0xFF);
|
2013-08-05 10:45:02 -04:00
|
|
|
return this;
|
2013-06-10 11:57:33 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Puts a 32-bit integer in this buffer in big-endian order.
|
|
|
|
*
|
|
|
|
* @param i the 32-bit integer.
|
2013-08-05 10:45:02 -04:00
|
|
|
*
|
|
|
|
* @return this buffer.
|
2013-06-10 11:57:33 -04:00
|
|
|
*/
|
|
|
|
util.ByteBuffer.prototype.putInt32 = function(i) {
|
|
|
|
this.data +=
|
|
|
|
String.fromCharCode(i >> 24 & 0xFF) +
|
|
|
|
String.fromCharCode(i >> 16 & 0xFF) +
|
|
|
|
String.fromCharCode(i >> 8 & 0xFF) +
|
|
|
|
String.fromCharCode(i & 0xFF);
|
2013-08-05 10:45:02 -04:00
|
|
|
return this;
|
2013-06-10 11:57:33 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Puts a 16-bit integer in this buffer in little-endian order.
|
|
|
|
*
|
|
|
|
* @param i the 16-bit integer.
|
2013-08-05 10:45:02 -04:00
|
|
|
*
|
|
|
|
* @return this buffer.
|
2013-06-10 11:57:33 -04:00
|
|
|
*/
|
|
|
|
util.ByteBuffer.prototype.putInt16Le = function(i) {
|
|
|
|
this.data +=
|
|
|
|
String.fromCharCode(i & 0xFF) +
|
|
|
|
String.fromCharCode(i >> 8 & 0xFF);
|
2013-08-05 10:45:02 -04:00
|
|
|
return this;
|
2013-06-10 11:57:33 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Puts a 24-bit integer in this buffer in little-endian order.
|
|
|
|
*
|
|
|
|
* @param i the 24-bit integer.
|
2013-08-05 10:45:02 -04:00
|
|
|
*
|
|
|
|
* @return this buffer.
|
2013-06-10 11:57:33 -04:00
|
|
|
*/
|
|
|
|
util.ByteBuffer.prototype.putInt24Le = function(i) {
|
|
|
|
this.data +=
|
|
|
|
String.fromCharCode(i & 0xFF) +
|
|
|
|
String.fromCharCode(i >> 8 & 0xFF) +
|
|
|
|
String.fromCharCode(i >> 16 & 0xFF);
|
2013-08-05 10:45:02 -04:00
|
|
|
return this;
|
2013-06-10 11:57:33 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Puts a 32-bit integer in this buffer in little-endian order.
|
|
|
|
*
|
|
|
|
* @param i the 32-bit integer.
|
2013-08-05 10:45:02 -04:00
|
|
|
*
|
|
|
|
* @return this buffer.
|
2013-06-10 11:57:33 -04:00
|
|
|
*/
|
|
|
|
util.ByteBuffer.prototype.putInt32Le = function(i) {
|
|
|
|
this.data +=
|
|
|
|
String.fromCharCode(i & 0xFF) +
|
|
|
|
String.fromCharCode(i >> 8 & 0xFF) +
|
|
|
|
String.fromCharCode(i >> 16 & 0xFF) +
|
|
|
|
String.fromCharCode(i >> 24 & 0xFF);
|
2013-08-05 10:45:02 -04:00
|
|
|
return this;
|
2013-06-10 11:57:33 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Puts an n-bit integer in this buffer in big-endian order.
|
|
|
|
*
|
|
|
|
* @param i the n-bit integer.
|
|
|
|
* @param n the number of bits in the integer.
|
2013-08-05 10:45:02 -04:00
|
|
|
*
|
|
|
|
* @return this buffer.
|
2013-06-10 11:57:33 -04:00
|
|
|
*/
|
|
|
|
util.ByteBuffer.prototype.putInt = function(i, n) {
|
|
|
|
do {
|
|
|
|
n -= 8;
|
|
|
|
this.data += String.fromCharCode((i >> n) & 0xFF);
|
|
|
|
}
|
|
|
|
while(n > 0);
|
2013-08-05 10:45:02 -04:00
|
|
|
return this;
|
2013-06-10 11:57:33 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Puts the given buffer into this buffer.
|
|
|
|
*
|
|
|
|
* @param buffer the buffer to put into this one.
|
2013-08-05 10:45:02 -04:00
|
|
|
*
|
|
|
|
* @return this buffer.
|
2013-06-10 11:57:33 -04:00
|
|
|
*/
|
|
|
|
util.ByteBuffer.prototype.putBuffer = function(buffer) {
|
|
|
|
this.data += buffer.getBytes();
|
2013-08-05 10:45:02 -04:00
|
|
|
return this;
|
2013-06-10 11:57:33 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets a byte from this buffer and advances the read pointer by 1.
|
|
|
|
*
|
|
|
|
* @return the byte.
|
|
|
|
*/
|
|
|
|
util.ByteBuffer.prototype.getByte = function() {
|
|
|
|
return this.data.charCodeAt(this.read++);
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets a uint16 from this buffer in big-endian order and advances the read
|
|
|
|
* pointer by 2.
|
|
|
|
*
|
|
|
|
* @return the uint16.
|
|
|
|
*/
|
|
|
|
util.ByteBuffer.prototype.getInt16 = function() {
|
|
|
|
var rval = (
|
|
|
|
this.data.charCodeAt(this.read) << 8 ^
|
|
|
|
this.data.charCodeAt(this.read + 1));
|
|
|
|
this.read += 2;
|
|
|
|
return rval;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets a uint24 from this buffer in big-endian order and advances the read
|
|
|
|
* pointer by 3.
|
|
|
|
*
|
|
|
|
* @return the uint24.
|
|
|
|
*/
|
|
|
|
util.ByteBuffer.prototype.getInt24 = function() {
|
|
|
|
var rval = (
|
|
|
|
this.data.charCodeAt(this.read) << 16 ^
|
|
|
|
this.data.charCodeAt(this.read + 1) << 8 ^
|
|
|
|
this.data.charCodeAt(this.read + 2));
|
|
|
|
this.read += 3;
|
|
|
|
return rval;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets a uint32 from this buffer in big-endian order and advances the read
|
|
|
|
* pointer by 4.
|
|
|
|
*
|
|
|
|
* @return the word.
|
|
|
|
*/
|
|
|
|
util.ByteBuffer.prototype.getInt32 = function() {
|
|
|
|
var rval = (
|
|
|
|
this.data.charCodeAt(this.read) << 24 ^
|
|
|
|
this.data.charCodeAt(this.read + 1) << 16 ^
|
|
|
|
this.data.charCodeAt(this.read + 2) << 8 ^
|
|
|
|
this.data.charCodeAt(this.read + 3));
|
|
|
|
this.read += 4;
|
|
|
|
return rval;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets a uint16 from this buffer in little-endian order and advances the read
|
|
|
|
* pointer by 2.
|
|
|
|
*
|
|
|
|
* @return the uint16.
|
|
|
|
*/
|
|
|
|
util.ByteBuffer.prototype.getInt16Le = function() {
|
|
|
|
var rval = (
|
|
|
|
this.data.charCodeAt(this.read) ^
|
|
|
|
this.data.charCodeAt(this.read + 1) << 8);
|
|
|
|
this.read += 2;
|
|
|
|
return rval;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets a uint24 from this buffer in little-endian order and advances the read
|
|
|
|
* pointer by 3.
|
|
|
|
*
|
|
|
|
* @return the uint24.
|
|
|
|
*/
|
|
|
|
util.ByteBuffer.prototype.getInt24Le = function() {
|
|
|
|
var rval = (
|
|
|
|
this.data.charCodeAt(this.read) ^
|
|
|
|
this.data.charCodeAt(this.read + 1) << 8 ^
|
|
|
|
this.data.charCodeAt(this.read + 2) << 16);
|
|
|
|
this.read += 3;
|
|
|
|
return rval;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets a uint32 from this buffer in little-endian order and advances the read
|
|
|
|
* pointer by 4.
|
|
|
|
*
|
|
|
|
* @return the word.
|
|
|
|
*/
|
|
|
|
util.ByteBuffer.prototype.getInt32Le = function() {
|
|
|
|
var rval = (
|
|
|
|
this.data.charCodeAt(this.read) ^
|
|
|
|
this.data.charCodeAt(this.read + 1) << 8 ^
|
|
|
|
this.data.charCodeAt(this.read + 2) << 16 ^
|
|
|
|
this.data.charCodeAt(this.read + 3) << 24);
|
|
|
|
this.read += 4;
|
|
|
|
return rval;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets an n-bit integer from this buffer in big-endian order and advances the
|
|
|
|
* read pointer by n/8.
|
|
|
|
*
|
|
|
|
* @param n the number of bits in the integer.
|
|
|
|
*
|
|
|
|
* @return the integer.
|
|
|
|
*/
|
|
|
|
util.ByteBuffer.prototype.getInt = function(n) {
|
|
|
|
var rval = 0;
|
|
|
|
do {
|
2013-08-05 10:45:02 -04:00
|
|
|
rval = (rval << 8) + this.data.charCodeAt(this.read++);
|
2013-06-10 11:57:33 -04:00
|
|
|
n -= 8;
|
|
|
|
}
|
|
|
|
while(n > 0);
|
|
|
|
return rval;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Reads bytes out into a UTF-8 string and clears them from the buffer.
|
|
|
|
*
|
|
|
|
* @param count the number of bytes to read, undefined or null for all.
|
|
|
|
*
|
|
|
|
* @return a UTF-8 string of bytes.
|
|
|
|
*/
|
|
|
|
util.ByteBuffer.prototype.getBytes = function(count) {
|
|
|
|
var rval;
|
|
|
|
if(count) {
|
|
|
|
// read count bytes
|
|
|
|
count = Math.min(this.length(), count);
|
|
|
|
rval = this.data.slice(this.read, this.read + count);
|
|
|
|
this.read += count;
|
|
|
|
}
|
|
|
|
else if(count === 0) {
|
|
|
|
rval = '';
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// read all bytes, optimize to only copy when needed
|
|
|
|
rval = (this.read === 0) ? this.data : this.data.slice(this.read);
|
|
|
|
this.clear();
|
|
|
|
}
|
|
|
|
return rval;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets a UTF-8 encoded string of the bytes from this buffer without modifying
|
|
|
|
* the read pointer.
|
|
|
|
*
|
|
|
|
* @param count the number of bytes to get, omit to get all.
|
|
|
|
*
|
|
|
|
* @return a string full of UTF-8 encoded characters.
|
|
|
|
*/
|
|
|
|
util.ByteBuffer.prototype.bytes = function(count) {
|
|
|
|
return (typeof(count) === 'undefined' ?
|
|
|
|
this.data.slice(this.read) :
|
|
|
|
this.data.slice(this.read, this.read + count));
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets a byte at the given index without modifying the read pointer.
|
|
|
|
*
|
|
|
|
* @param i the byte index.
|
|
|
|
*
|
|
|
|
* @return the byte.
|
|
|
|
*/
|
|
|
|
util.ByteBuffer.prototype.at = function(i) {
|
|
|
|
return this.data.charCodeAt(this.read + i);
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Puts a byte at the given index without modifying the read pointer.
|
|
|
|
*
|
|
|
|
* @param i the byte index.
|
|
|
|
* @param b the byte to put.
|
2013-08-05 10:45:02 -04:00
|
|
|
*
|
|
|
|
* @return this buffer.
|
2013-06-10 11:57:33 -04:00
|
|
|
*/
|
|
|
|
util.ByteBuffer.prototype.setAt = function(i, b) {
|
|
|
|
this.data = this.data.substr(0, this.read + i) +
|
|
|
|
String.fromCharCode(b) +
|
|
|
|
this.data.substr(this.read + i + 1);
|
2013-08-05 10:45:02 -04:00
|
|
|
return this;
|
2013-06-10 11:57:33 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the last byte without modifying the read pointer.
|
|
|
|
*
|
|
|
|
* @return the last byte.
|
|
|
|
*/
|
|
|
|
util.ByteBuffer.prototype.last = function() {
|
|
|
|
return this.data.charCodeAt(this.data.length - 1);
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates a copy of this buffer.
|
|
|
|
*
|
|
|
|
* @return the copy.
|
|
|
|
*/
|
|
|
|
util.ByteBuffer.prototype.copy = function() {
|
|
|
|
var c = util.createBuffer(this.data);
|
|
|
|
c.read = this.read;
|
|
|
|
return c;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Compacts this buffer.
|
2013-08-05 10:45:02 -04:00
|
|
|
*
|
|
|
|
* @return this buffer.
|
2013-06-10 11:57:33 -04:00
|
|
|
*/
|
|
|
|
util.ByteBuffer.prototype.compact = function() {
|
|
|
|
if(this.read > 0) {
|
|
|
|
this.data = this.data.slice(this.read);
|
|
|
|
this.read = 0;
|
|
|
|
}
|
2013-08-05 10:45:02 -04:00
|
|
|
return this;
|
2013-06-10 11:57:33 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Clears this buffer.
|
2013-08-05 10:45:02 -04:00
|
|
|
*
|
|
|
|
* @return this buffer.
|
2013-06-10 11:57:33 -04:00
|
|
|
*/
|
|
|
|
util.ByteBuffer.prototype.clear = function() {
|
|
|
|
this.data = '';
|
|
|
|
this.read = 0;
|
2013-08-05 10:45:02 -04:00
|
|
|
return this;
|
2013-06-10 11:57:33 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Shortens this buffer by triming bytes off of the end of this buffer.
|
|
|
|
*
|
|
|
|
* @param count the number of bytes to trim off.
|
2013-08-05 10:45:02 -04:00
|
|
|
*
|
|
|
|
* @return this buffer.
|
2013-06-10 11:57:33 -04:00
|
|
|
*/
|
|
|
|
util.ByteBuffer.prototype.truncate = function(count) {
|
|
|
|
var len = Math.max(0, this.length() - count);
|
|
|
|
this.data = this.data.substr(this.read, len);
|
|
|
|
this.read = 0;
|
2013-08-05 10:45:02 -04:00
|
|
|
return this;
|
2013-06-10 11:57:33 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Converts this buffer to a hexadecimal string.
|
|
|
|
*
|
|
|
|
* @return a hexadecimal string.
|
|
|
|
*/
|
|
|
|
util.ByteBuffer.prototype.toHex = function() {
|
|
|
|
var rval = '';
|
|
|
|
for(var i = this.read; i < this.data.length; ++i) {
|
|
|
|
var b = this.data.charCodeAt(i);
|
|
|
|
if(b < 16) {
|
|
|
|
rval += '0';
|
|
|
|
}
|
|
|
|
rval += b.toString(16);
|
|
|
|
}
|
|
|
|
return rval;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Converts this buffer to a UTF-16 string (standard JavaScript string).
|
|
|
|
*
|
|
|
|
* @return a UTF-16 string.
|
|
|
|
*/
|
|
|
|
util.ByteBuffer.prototype.toString = function() {
|
|
|
|
return util.decodeUtf8(this.bytes());
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates a buffer that stores bytes. A value may be given to put into the
|
|
|
|
* buffer that is either a string of bytes or a UTF-16 string that will
|
|
|
|
* be encoded using UTF-8 (to do the latter, specify 'utf8' as the encoding).
|
|
|
|
*
|
|
|
|
* @param [input] the bytes to wrap (as a string) or a UTF-16 string to encode
|
|
|
|
* as UTF-8.
|
|
|
|
* @param [encoding] (default: 'raw', other: 'utf8').
|
|
|
|
*/
|
|
|
|
util.createBuffer = function(input, encoding) {
|
|
|
|
encoding = encoding || 'raw';
|
|
|
|
if(input !== undefined && encoding === 'utf8') {
|
|
|
|
input = util.encodeUtf8(input);
|
|
|
|
}
|
|
|
|
return new util.ByteBuffer(input);
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Fills a string with a particular value. If you want the string to be a byte
|
|
|
|
* string, pass in String.fromCharCode(theByte).
|
|
|
|
*
|
|
|
|
* @param c the character to fill the string with, use String.fromCharCode
|
|
|
|
* to fill the string with a byte value.
|
|
|
|
* @param n the number of characters of value c to fill with.
|
|
|
|
*
|
|
|
|
* @return the filled string.
|
|
|
|
*/
|
|
|
|
util.fillString = function(c, n) {
|
|
|
|
var s = '';
|
|
|
|
while(n > 0) {
|
|
|
|
if(n & 1) {
|
|
|
|
s += c;
|
|
|
|
}
|
|
|
|
n >>>= 1;
|
|
|
|
if(n > 0) {
|
|
|
|
c += c;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return s;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Performs a per byte XOR between two byte strings and returns the result as a
|
|
|
|
* string of bytes.
|
|
|
|
*
|
|
|
|
* @param s1 first string of bytes.
|
|
|
|
* @param s2 second string of bytes.
|
|
|
|
* @param n the number of bytes to XOR.
|
|
|
|
*
|
|
|
|
* @return the XOR'd result.
|
|
|
|
*/
|
|
|
|
util.xorBytes = function(s1, s2, n) {
|
|
|
|
var s3 = '';
|
|
|
|
var b = '';
|
|
|
|
var t = '';
|
|
|
|
var i = 0;
|
|
|
|
var c = 0;
|
|
|
|
for(; n > 0; --n, ++i) {
|
|
|
|
b = s1.charCodeAt(i) ^ s2.charCodeAt(i);
|
|
|
|
if(c >= 10) {
|
|
|
|
s3 += t;
|
|
|
|
t = '';
|
|
|
|
c = 0;
|
|
|
|
}
|
|
|
|
t += String.fromCharCode(b);
|
|
|
|
++c;
|
|
|
|
}
|
|
|
|
s3 += t;
|
|
|
|
return s3;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Converts a hex string into a UTF-8 string of bytes.
|
|
|
|
*
|
|
|
|
* @param hex the hexadecimal string to convert.
|
|
|
|
*
|
|
|
|
* @return the string of bytes.
|
|
|
|
*/
|
|
|
|
util.hexToBytes = function(hex) {
|
|
|
|
var rval = '';
|
|
|
|
var i = 0;
|
|
|
|
if(hex.length & 1 == 1) {
|
|
|
|
// odd number of characters, convert first character alone
|
|
|
|
i = 1;
|
|
|
|
rval += String.fromCharCode(parseInt(hex[0], 16));
|
|
|
|
}
|
|
|
|
// convert 2 characters (1 byte) at a time
|
|
|
|
for(; i < hex.length; i += 2) {
|
|
|
|
rval += String.fromCharCode(parseInt(hex.substr(i, 2), 16));
|
|
|
|
}
|
|
|
|
return rval;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Converts a UTF-8 byte string into a string of hexadecimal characters.
|
|
|
|
*
|
|
|
|
* @param bytes the byte string to convert.
|
|
|
|
*
|
|
|
|
* @return the string of hexadecimal characters.
|
|
|
|
*/
|
|
|
|
util.bytesToHex = function(bytes) {
|
|
|
|
return util.createBuffer(bytes).toHex();
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Converts an 32-bit integer to 4-big-endian byte string.
|
|
|
|
*
|
|
|
|
* @param i the integer.
|
|
|
|
*
|
|
|
|
* @return the byte string.
|
|
|
|
*/
|
|
|
|
util.int32ToBytes = function(i) {
|
|
|
|
return (
|
|
|
|
String.fromCharCode(i >> 24 & 0xFF) +
|
|
|
|
String.fromCharCode(i >> 16 & 0xFF) +
|
|
|
|
String.fromCharCode(i >> 8 & 0xFF) +
|
|
|
|
String.fromCharCode(i & 0xFF));
|
|
|
|
};
|
|
|
|
|
|
|
|
// base64 characters, reverse mapping
|
|
|
|
var _base64 =
|
|
|
|
'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=';
|
|
|
|
var _base64Idx = [
|
|
|
|
/*43 -43 = 0*/
|
|
|
|
/*'+', 1, 2, 3,'/' */
|
|
|
|
62, -1, -1, -1, 63,
|
|
|
|
|
|
|
|
/*'0','1','2','3','4','5','6','7','8','9' */
|
|
|
|
52, 53, 54, 55, 56, 57, 58, 59, 60, 61,
|
|
|
|
|
|
|
|
/*15, 16, 17,'=', 19, 20, 21 */
|
|
|
|
-1, -1, -1, 64, -1, -1, -1,
|
|
|
|
|
|
|
|
/*65 - 43 = 22*/
|
|
|
|
/*'A','B','C','D','E','F','G','H','I','J','K','L','M', */
|
|
|
|
0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12,
|
|
|
|
|
|
|
|
/*'N','O','P','Q','R','S','T','U','V','W','X','Y','Z' */
|
|
|
|
13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25,
|
|
|
|
|
|
|
|
/*91 - 43 = 48 */
|
|
|
|
/*48, 49, 50, 51, 52, 53 */
|
|
|
|
-1, -1, -1, -1, -1, -1,
|
|
|
|
|
|
|
|
/*97 - 43 = 54*/
|
|
|
|
/*'a','b','c','d','e','f','g','h','i','j','k','l','m' */
|
|
|
|
26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38,
|
|
|
|
|
|
|
|
/*'n','o','p','q','r','s','t','u','v','w','x','y','z' */
|
|
|
|
39, 40, 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51
|
|
|
|
];
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Base64 encodes a UTF-8 string of bytes.
|
|
|
|
*
|
|
|
|
* @param input the UTF-8 string of bytes to encode.
|
|
|
|
* @param maxline the maximum number of encoded bytes per line to use,
|
|
|
|
* defaults to none.
|
|
|
|
*
|
|
|
|
* @return the base64-encoded output.
|
|
|
|
*/
|
|
|
|
util.encode64 = function(input, maxline) {
|
|
|
|
var line = '';
|
|
|
|
var output = '';
|
|
|
|
var chr1, chr2, chr3;
|
|
|
|
var i = 0;
|
|
|
|
while(i < input.length) {
|
|
|
|
chr1 = input.charCodeAt(i++);
|
|
|
|
chr2 = input.charCodeAt(i++);
|
|
|
|
chr3 = input.charCodeAt(i++);
|
|
|
|
|
|
|
|
// encode 4 character group
|
|
|
|
line += _base64.charAt(chr1 >> 2);
|
|
|
|
line += _base64.charAt(((chr1 & 3) << 4) | (chr2 >> 4));
|
|
|
|
if(isNaN(chr2)) {
|
|
|
|
line += '==';
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
line += _base64.charAt(((chr2 & 15) << 2) | (chr3 >> 6));
|
|
|
|
line += isNaN(chr3) ? '=' : _base64.charAt(chr3 & 63);
|
|
|
|
}
|
|
|
|
|
|
|
|
if(maxline && line.length > maxline) {
|
|
|
|
output += line.substr(0, maxline) + '\r\n';
|
|
|
|
line = line.substr(maxline);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
output += line;
|
|
|
|
|
|
|
|
return output;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Base64 decodes a string into a UTF-8 string of bytes.
|
|
|
|
*
|
|
|
|
* @param input the base64-encoded input.
|
|
|
|
*
|
|
|
|
* @return the raw bytes.
|
|
|
|
*/
|
|
|
|
util.decode64 = function(input) {
|
|
|
|
// remove all non-base64 characters
|
|
|
|
input = input.replace(/[^A-Za-z0-9\+\/\=]/g, '');
|
|
|
|
|
|
|
|
var output = '';
|
|
|
|
var enc1, enc2, enc3, enc4;
|
|
|
|
var i = 0;
|
|
|
|
|
|
|
|
while(i < input.length) {
|
|
|
|
enc1 = _base64Idx[input.charCodeAt(i++) - 43];
|
|
|
|
enc2 = _base64Idx[input.charCodeAt(i++) - 43];
|
|
|
|
enc3 = _base64Idx[input.charCodeAt(i++) - 43];
|
|
|
|
enc4 = _base64Idx[input.charCodeAt(i++) - 43];
|
|
|
|
|
|
|
|
output += String.fromCharCode((enc1 << 2) | (enc2 >> 4));
|
|
|
|
if(enc3 !== 64) {
|
|
|
|
// decoded at least 2 bytes
|
|
|
|
output += String.fromCharCode(((enc2 & 15) << 4) | (enc3 >> 2));
|
|
|
|
if(enc4 !== 64) {
|
|
|
|
// decoded 3 bytes
|
|
|
|
output += String.fromCharCode(((enc3 & 3) << 6) | enc4);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return output;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* UTF-8 encodes the given UTF-16 encoded string (a standard JavaScript
|
|
|
|
* string). Non-ASCII characters will be encoded as multiple bytes according
|
|
|
|
* to UTF-8.
|
|
|
|
*
|
|
|
|
* @param str the string to encode.
|
|
|
|
*
|
|
|
|
* @return the UTF-8 encoded string.
|
|
|
|
*/
|
|
|
|
util.encodeUtf8 = function(str) {
|
|
|
|
return unescape(encodeURIComponent(str));
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Decodes a UTF-8 encoded string into a UTF-16 string.
|
|
|
|
*
|
|
|
|
* @param str the string to encode.
|
|
|
|
*
|
|
|
|
* @return the UTF-16 encoded string (standard JavaScript string).
|
|
|
|
*/
|
|
|
|
util.decodeUtf8 = function(str) {
|
|
|
|
return decodeURIComponent(escape(str));
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Deflates the given data using a flash interface.
|
|
|
|
*
|
|
|
|
* @param api the flash interface.
|
|
|
|
* @param bytes the data.
|
|
|
|
* @param raw true to return only raw deflate data, false to include zlib
|
|
|
|
* header and trailer.
|
|
|
|
*
|
|
|
|
* @return the deflated data as a string.
|
|
|
|
*/
|
|
|
|
util.deflate = function(api, bytes, raw) {
|
|
|
|
bytes = util.decode64(api.deflate(util.encode64(bytes)).rval);
|
|
|
|
|
|
|
|
// strip zlib header and trailer if necessary
|
|
|
|
if(raw) {
|
|
|
|
// zlib header is 2 bytes (CMF,FLG) where FLG indicates that
|
|
|
|
// there is a 4-byte DICT (alder-32) block before the data if
|
|
|
|
// its 5th bit is set
|
|
|
|
var start = 2;
|
|
|
|
var flg = bytes.charCodeAt(1);
|
|
|
|
if(flg & 0x20) {
|
|
|
|
start = 6;
|
|
|
|
}
|
|
|
|
// zlib trailer is 4 bytes of adler-32
|
|
|
|
bytes = bytes.substring(start, bytes.length - 4);
|
|
|
|
}
|
|
|
|
|
|
|
|
return bytes;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Inflates the given data using a flash interface.
|
|
|
|
*
|
|
|
|
* @param api the flash interface.
|
|
|
|
* @param bytes the data.
|
|
|
|
* @param raw true if the incoming data has no zlib header or trailer and is
|
|
|
|
* raw DEFLATE data.
|
|
|
|
*
|
|
|
|
* @return the inflated data as a string, null on error.
|
|
|
|
*/
|
|
|
|
util.inflate = function(api, bytes, raw) {
|
|
|
|
// TODO: add zlib header and trailer if necessary/possible
|
|
|
|
var rval = api.inflate(util.encode64(bytes)).rval;
|
|
|
|
return (rval === null) ? null : util.decode64(rval);
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets a storage object.
|
|
|
|
*
|
|
|
|
* @param api the storage interface.
|
|
|
|
* @param id the storage ID to use.
|
|
|
|
* @param obj the storage object, null to remove.
|
|
|
|
*/
|
|
|
|
var _setStorageObject = function(api, id, obj) {
|
|
|
|
if(!api) {
|
|
|
|
throw {
|
|
|
|
message: 'WebStorage not available.'
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
var rval;
|
|
|
|
if(obj === null) {
|
|
|
|
rval = api.removeItem(id);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// json-encode and base64-encode object
|
|
|
|
obj = util.encode64(JSON.stringify(obj));
|
|
|
|
rval = api.setItem(id, obj);
|
|
|
|
}
|
|
|
|
|
|
|
|
// handle potential flash error
|
|
|
|
if(typeof(rval) !== 'undefined' && rval.rval !== true) {
|
|
|
|
throw rval.error;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets a storage object.
|
|
|
|
*
|
|
|
|
* @param api the storage interface.
|
|
|
|
* @param id the storage ID to use.
|
|
|
|
*
|
|
|
|
* @return the storage object entry or null if none exists.
|
|
|
|
*/
|
|
|
|
var _getStorageObject = function(api, id) {
|
|
|
|
if(!api) {
|
|
|
|
throw {
|
|
|
|
message: 'WebStorage not available.'
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
// get the existing entry
|
|
|
|
var rval = api.getItem(id);
|
|
|
|
|
|
|
|
/* Note: We check api.init because we can't do (api == localStorage)
|
|
|
|
on IE because of "Class doesn't support Automation" exception. Only
|
|
|
|
the flash api has an init method so this works too, but we need a
|
|
|
|
better solution in the future. */
|
|
|
|
|
|
|
|
// flash returns item wrapped in an object, handle special case
|
|
|
|
if(api.init) {
|
|
|
|
if(rval.rval === null) {
|
|
|
|
if(rval.error) {
|
|
|
|
throw rval.error;
|
|
|
|
}
|
|
|
|
// no error, but also no item
|
|
|
|
rval = null;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
rval = rval.rval;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// handle decoding
|
|
|
|
if(rval !== null) {
|
|
|
|
// base64-decode and json-decode data
|
|
|
|
rval = JSON.parse(util.decode64(rval));
|
|
|
|
}
|
|
|
|
|
|
|
|
return rval;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Stores an item in local storage.
|
|
|
|
*
|
|
|
|
* @param api the storage interface.
|
|
|
|
* @param id the storage ID to use.
|
|
|
|
* @param key the key for the item.
|
|
|
|
* @param data the data for the item (any javascript object/primitive).
|
|
|
|
*/
|
|
|
|
var _setItem = function(api, id, key, data) {
|
|
|
|
// get storage object
|
|
|
|
var obj = _getStorageObject(api, id);
|
|
|
|
if(obj === null) {
|
|
|
|
// create a new storage object
|
|
|
|
obj = {};
|
|
|
|
}
|
|
|
|
// update key
|
|
|
|
obj[key] = data;
|
|
|
|
|
|
|
|
// set storage object
|
|
|
|
_setStorageObject(api, id, obj);
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets an item from local storage.
|
|
|
|
*
|
|
|
|
* @param api the storage interface.
|
|
|
|
* @param id the storage ID to use.
|
|
|
|
* @param key the key for the item.
|
|
|
|
*
|
|
|
|
* @return the item.
|
|
|
|
*/
|
|
|
|
var _getItem = function(api, id, key) {
|
|
|
|
// get storage object
|
|
|
|
var rval = _getStorageObject(api, id);
|
|
|
|
if(rval !== null) {
|
|
|
|
// return data at key
|
|
|
|
rval = (key in rval) ? rval[key] : null;
|
|
|
|
}
|
|
|
|
|
|
|
|
return rval;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Removes an item from local storage.
|
|
|
|
*
|
|
|
|
* @param api the storage interface.
|
|
|
|
* @param id the storage ID to use.
|
|
|
|
* @param key the key for the item.
|
|
|
|
*/
|
|
|
|
var _removeItem = function(api, id, key) {
|
|
|
|
// get storage object
|
|
|
|
var obj = _getStorageObject(api, id);
|
|
|
|
if(obj !== null && key in obj) {
|
|
|
|
// remove key
|
|
|
|
delete obj[key];
|
|
|
|
|
|
|
|
// see if entry has no keys remaining
|
|
|
|
var empty = true;
|
|
|
|
for(var prop in obj) {
|
|
|
|
empty = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if(empty) {
|
|
|
|
// remove entry entirely if no keys are left
|
|
|
|
obj = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
// set storage object
|
|
|
|
_setStorageObject(api, id, obj);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Clears the local disk storage identified by the given ID.
|
|
|
|
*
|
|
|
|
* @param api the storage interface.
|
|
|
|
* @param id the storage ID to use.
|
|
|
|
*/
|
|
|
|
var _clearItems = function(api, id) {
|
|
|
|
_setStorageObject(api, id, null);
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Calls a storage function.
|
|
|
|
*
|
|
|
|
* @param func the function to call.
|
|
|
|
* @param args the arguments for the function.
|
|
|
|
* @param location the location argument.
|
|
|
|
*
|
|
|
|
* @return the return value from the function.
|
|
|
|
*/
|
|
|
|
var _callStorageFunction = function(func, args, location) {
|
|
|
|
var rval = null;
|
|
|
|
|
|
|
|
// default storage types
|
|
|
|
if(typeof(location) === 'undefined') {
|
|
|
|
location = ['web', 'flash'];
|
|
|
|
}
|
|
|
|
|
|
|
|
// apply storage types in order of preference
|
|
|
|
var type;
|
|
|
|
var done = false;
|
|
|
|
var exception = null;
|
|
|
|
for(var idx in location) {
|
|
|
|
type = location[idx];
|
|
|
|
try {
|
|
|
|
if(type === 'flash' || type === 'both') {
|
|
|
|
if(args[0] === null) {
|
|
|
|
throw {
|
|
|
|
message: 'Flash local storage not available.'
|
|
|
|
};
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
rval = func.apply(this, args);
|
|
|
|
done = (type === 'flash');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if(type === 'web' || type === 'both') {
|
|
|
|
args[0] = localStorage;
|
|
|
|
rval = func.apply(this, args);
|
|
|
|
done = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch(ex) {
|
|
|
|
exception = ex;
|
|
|
|
}
|
|
|
|
if(done) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!done) {
|
|
|
|
throw exception;
|
|
|
|
}
|
|
|
|
|
|
|
|
return rval;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Stores an item on local disk.
|
|
|
|
*
|
|
|
|
* The available types of local storage include 'flash', 'web', and 'both'.
|
|
|
|
*
|
|
|
|
* The type 'flash' refers to flash local storage (SharedObject). In order
|
|
|
|
* to use flash local storage, the 'api' parameter must be valid. The type
|
|
|
|
* 'web' refers to WebStorage, if supported by the browser. The type 'both'
|
|
|
|
* refers to storing using both 'flash' and 'web', not just one or the
|
|
|
|
* other.
|
|
|
|
*
|
|
|
|
* The location array should list the storage types to use in order of
|
|
|
|
* preference:
|
|
|
|
*
|
|
|
|
* ['flash']: flash only storage
|
|
|
|
* ['web']: web only storage
|
|
|
|
* ['both']: try to store in both
|
|
|
|
* ['flash','web']: store in flash first, but if not available, 'web'
|
|
|
|
* ['web','flash']: store in web first, but if not available, 'flash'
|
|
|
|
*
|
|
|
|
* The location array defaults to: ['web', 'flash']
|
|
|
|
*
|
|
|
|
* @param api the flash interface, null to use only WebStorage.
|
|
|
|
* @param id the storage ID to use.
|
|
|
|
* @param key the key for the item.
|
|
|
|
* @param data the data for the item (any javascript object/primitive).
|
|
|
|
* @param location an array with the preferred types of storage to use.
|
|
|
|
*/
|
|
|
|
util.setItem = function(api, id, key, data, location) {
|
|
|
|
_callStorageFunction(_setItem, arguments, location);
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets an item on local disk.
|
|
|
|
*
|
|
|
|
* Set setItem() for details on storage types.
|
|
|
|
*
|
|
|
|
* @param api the flash interface, null to use only WebStorage.
|
|
|
|
* @param id the storage ID to use.
|
|
|
|
* @param key the key for the item.
|
|
|
|
* @param location an array with the preferred types of storage to use.
|
|
|
|
*
|
|
|
|
* @return the item.
|
|
|
|
*/
|
|
|
|
util.getItem = function(api, id, key, location) {
|
|
|
|
return _callStorageFunction(_getItem, arguments, location);
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Removes an item on local disk.
|
|
|
|
*
|
|
|
|
* Set setItem() for details on storage types.
|
|
|
|
*
|
|
|
|
* @param api the flash interface.
|
|
|
|
* @param id the storage ID to use.
|
|
|
|
* @param key the key for the item.
|
|
|
|
* @param location an array with the preferred types of storage to use.
|
|
|
|
*/
|
|
|
|
util.removeItem = function(api, id, key, location) {
|
|
|
|
_callStorageFunction(_removeItem, arguments, location);
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Clears the local disk storage identified by the given ID.
|
|
|
|
*
|
|
|
|
* Set setItem() for details on storage types.
|
|
|
|
*
|
|
|
|
* @param api the flash interface if flash is available.
|
|
|
|
* @param id the storage ID to use.
|
|
|
|
* @param location an array with the preferred types of storage to use.
|
|
|
|
*/
|
|
|
|
util.clearItems = function(api, id, location) {
|
|
|
|
_callStorageFunction(_clearItems, arguments, location);
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Parses the scheme, host, and port from an http(s) url.
|
|
|
|
*
|
|
|
|
* @param str the url string.
|
|
|
|
*
|
|
|
|
* @return the parsed url object or null if the url is invalid.
|
|
|
|
*/
|
|
|
|
util.parseUrl = function(str) {
|
|
|
|
// FIXME: this regex looks a bit broken
|
|
|
|
var regex = /^(https?):\/\/([^:&^\/]*):?(\d*)(.*)$/g;
|
|
|
|
regex.lastIndex = 0;
|
|
|
|
var m = regex.exec(str);
|
|
|
|
var url = (m === null) ? null : {
|
|
|
|
full: str,
|
|
|
|
scheme: m[1],
|
|
|
|
host: m[2],
|
|
|
|
port: m[3],
|
|
|
|
path: m[4]
|
|
|
|
};
|
|
|
|
if(url) {
|
|
|
|
url.fullHost = url.host;
|
|
|
|
if(url.port) {
|
|
|
|
if(url.port !== 80 && url.scheme === 'http') {
|
|
|
|
url.fullHost += ':' + url.port;
|
|
|
|
}
|
|
|
|
else if(url.port !== 443 && url.scheme === 'https') {
|
|
|
|
url.fullHost += ':' + url.port;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if(url.scheme === 'http') {
|
|
|
|
url.port = 80;
|
|
|
|
}
|
|
|
|
else if(url.scheme === 'https') {
|
|
|
|
url.port = 443;
|
|
|
|
}
|
|
|
|
url.full = url.scheme + '://' + url.fullHost;
|
|
|
|
}
|
|
|
|
return url;
|
|
|
|
};
|
|
|
|
|
|
|
|
/* Storage for query variables */
|
|
|
|
var _queryVariables = null;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the window location query variables. Query is parsed on the first
|
|
|
|
* call and the same object is returned on subsequent calls. The mapping
|
|
|
|
* is from keys to an array of values. Parameters without values will have
|
|
|
|
* an object key set but no value added to the value array. Values are
|
|
|
|
* unescaped.
|
|
|
|
*
|
|
|
|
* ...?k1=v1&k2=v2:
|
|
|
|
* {
|
|
|
|
* "k1": ["v1"],
|
|
|
|
* "k2": ["v2"]
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* ...?k1=v1&k1=v2:
|
|
|
|
* {
|
|
|
|
* "k1": ["v1", "v2"]
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* ...?k1=v1&k2:
|
|
|
|
* {
|
|
|
|
* "k1": ["v1"],
|
|
|
|
* "k2": []
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* ...?k1=v1&k1:
|
|
|
|
* {
|
|
|
|
* "k1": ["v1"]
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* ...?k1&k1:
|
|
|
|
* {
|
|
|
|
* "k1": []
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* @param query the query string to parse (optional, default to cached
|
|
|
|
* results from parsing window location search query).
|
|
|
|
*
|
|
|
|
* @return object mapping keys to variables.
|
|
|
|
*/
|
|
|
|
util.getQueryVariables = function(query) {
|
|
|
|
var parse = function(q) {
|
|
|
|
var rval = {};
|
|
|
|
var kvpairs = q.split('&');
|
|
|
|
for(var i = 0; i < kvpairs.length; i++) {
|
|
|
|
var pos = kvpairs[i].indexOf('=');
|
|
|
|
var key;
|
|
|
|
var val;
|
|
|
|
if(pos > 0) {
|
|
|
|
key = kvpairs[i].substring(0,pos);
|
|
|
|
val = kvpairs[i].substring(pos+1);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
key = kvpairs[i];
|
|
|
|
val = null;
|
|
|
|
}
|
|
|
|
if(!(key in rval)) {
|
|
|
|
rval[key] = [];
|
|
|
|
}
|
2013-08-05 10:45:02 -04:00
|
|
|
// disallow overriding object prototype keys
|
|
|
|
if(!(key in Object.prototype) && val !== null) {
|
2013-06-10 11:57:33 -04:00
|
|
|
rval[key].push(unescape(val));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return rval;
|
|
|
|
};
|
|
|
|
|
|
|
|
var rval;
|
|
|
|
if(typeof(query) === 'undefined') {
|
|
|
|
// set cached variables if needed
|
|
|
|
if(_queryVariables === null) {
|
|
|
|
if(typeof(window) === 'undefined') {
|
|
|
|
// no query variables available
|
|
|
|
_queryVariables = {};
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// parse window search query
|
|
|
|
_queryVariables = parse(window.location.search.substring(1));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
rval = _queryVariables;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// parse given query
|
|
|
|
rval = parse(query);
|
|
|
|
}
|
|
|
|
return rval;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Parses a fragment into a path and query. This method will take a URI
|
|
|
|
* fragment and break it up as if it were the main URI. For example:
|
|
|
|
* /bar/baz?a=1&b=2
|
|
|
|
* results in:
|
|
|
|
* {
|
|
|
|
* path: ["bar", "baz"],
|
|
|
|
* query: {"k1": ["v1"], "k2": ["v2"]}
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* @return object with a path array and query object.
|
|
|
|
*/
|
|
|
|
util.parseFragment = function(fragment) {
|
|
|
|
// default to whole fragment
|
|
|
|
var fp = fragment;
|
|
|
|
var fq = '';
|
|
|
|
// split into path and query if possible at the first '?'
|
|
|
|
var pos = fragment.indexOf('?');
|
|
|
|
if(pos > 0) {
|
|
|
|
fp = fragment.substring(0,pos);
|
|
|
|
fq = fragment.substring(pos+1);
|
|
|
|
}
|
|
|
|
// split path based on '/' and ignore first element if empty
|
|
|
|
var path = fp.split('/');
|
|
|
|
if(path.length > 0 && path[0] === '') {
|
|
|
|
path.shift();
|
|
|
|
}
|
|
|
|
// convert query into object
|
|
|
|
var query = (fq === '') ? {} : util.getQueryVariables(fq);
|
|
|
|
|
|
|
|
return {
|
|
|
|
pathString: fp,
|
|
|
|
queryString: fq,
|
|
|
|
path: path,
|
|
|
|
query: query
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Makes a request out of a URI-like request string. This is intended to
|
|
|
|
* be used where a fragment id (after a URI '#') is parsed as a URI with
|
|
|
|
* path and query parts. The string should have a path beginning and
|
|
|
|
* delimited by '/' and optional query parameters following a '?'. The
|
|
|
|
* query should be a standard URL set of key value pairs delimited by
|
|
|
|
* '&'. For backwards compatibility the initial '/' on the path is not
|
|
|
|
* required. The request object has the following API, (fully described
|
|
|
|
* in the method code):
|
|
|
|
* {
|
|
|
|
* path: <the path string part>.
|
|
|
|
* query: <the query string part>,
|
|
|
|
* getPath(i): get part or all of the split path array,
|
|
|
|
* getQuery(k, i): get part or all of a query key array,
|
|
|
|
* getQueryLast(k, _default): get last element of a query key array.
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* @return object with request parameters.
|
|
|
|
*/
|
|
|
|
util.makeRequest = function(reqString) {
|
|
|
|
var frag = util.parseFragment(reqString);
|
|
|
|
var req = {
|
|
|
|
// full path string
|
|
|
|
path: frag.pathString,
|
|
|
|
// full query string
|
|
|
|
query: frag.queryString,
|
|
|
|
/**
|
|
|
|
* Get path or element in path.
|
|
|
|
*
|
|
|
|
* @param i optional path index.
|
|
|
|
*
|
|
|
|
* @return path or part of path if i provided.
|
|
|
|
*/
|
|
|
|
getPath: function(i) {
|
|
|
|
return (typeof(i) === 'undefined') ? frag.path : frag.path[i];
|
|
|
|
},
|
|
|
|
/**
|
|
|
|
* Get query, values for a key, or value for a key index.
|
|
|
|
*
|
|
|
|
* @param k optional query key.
|
|
|
|
* @param i optional query key index.
|
|
|
|
*
|
|
|
|
* @return query, values for a key, or value for a key index.
|
|
|
|
*/
|
|
|
|
getQuery: function(k, i) {
|
|
|
|
var rval;
|
|
|
|
if(typeof(k) === 'undefined') {
|
|
|
|
rval = frag.query;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
rval = frag.query[k];
|
|
|
|
if(rval && typeof(i) !== 'undefined')
|
|
|
|
{
|
|
|
|
rval = rval[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return rval;
|
|
|
|
},
|
|
|
|
getQueryLast: function(k, _default) {
|
|
|
|
var rval;
|
|
|
|
var vals = req.getQuery(k);
|
|
|
|
if(vals) {
|
|
|
|
rval = vals[vals.length - 1];
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
rval = _default;
|
|
|
|
}
|
|
|
|
return rval;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
return req;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Makes a URI out of a path, an object with query parameters, and a
|
|
|
|
* fragment. Uses jQuery.param() internally for query string creation.
|
|
|
|
* If the path is an array, it will be joined with '/'.
|
|
|
|
*
|
|
|
|
* @param path string path or array of strings.
|
|
|
|
* @param query object with query parameters. (optional)
|
|
|
|
* @param fragment fragment string. (optional)
|
|
|
|
*
|
|
|
|
* @return string object with request parameters.
|
|
|
|
*/
|
|
|
|
util.makeLink = function(path, query, fragment) {
|
|
|
|
// join path parts if needed
|
|
|
|
path = jQuery.isArray(path) ? path.join('/') : path;
|
|
|
|
|
|
|
|
var qstr = jQuery.param(query || {});
|
|
|
|
fragment = fragment || '';
|
|
|
|
return path +
|
|
|
|
((qstr.length > 0) ? ('?' + qstr) : '') +
|
|
|
|
((fragment.length > 0) ? ('#' + fragment) : '');
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Follows a path of keys deep into an object hierarchy and set a value.
|
|
|
|
* If a key does not exist or it's value is not an object, create an
|
|
|
|
* object in it's place. This can be destructive to a object tree if
|
|
|
|
* leaf nodes are given as non-final path keys.
|
|
|
|
* Used to avoid exceptions from missing parts of the path.
|
|
|
|
*
|
|
|
|
* @param object the starting object.
|
|
|
|
* @param keys an array of string keys.
|
|
|
|
* @param value the value to set.
|
|
|
|
*/
|
|
|
|
util.setPath = function(object, keys, value) {
|
|
|
|
// need to start at an object
|
|
|
|
if(typeof(object) === 'object' && object !== null) {
|
|
|
|
var i = 0;
|
|
|
|
var len = keys.length;
|
|
|
|
while(i < len) {
|
|
|
|
var next = keys[i++];
|
|
|
|
if(i == len) {
|
|
|
|
// last
|
|
|
|
object[next] = value;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// more
|
|
|
|
var hasNext = (next in object);
|
|
|
|
if(!hasNext ||
|
|
|
|
(hasNext && typeof(object[next]) !== 'object') ||
|
|
|
|
(hasNext && object[next] === null)) {
|
|
|
|
object[next] = {};
|
|
|
|
}
|
|
|
|
object = object[next];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Follows a path of keys deep into an object hierarchy and return a value.
|
|
|
|
* If a key does not exist, create an object in it's place.
|
|
|
|
* Used to avoid exceptions from missing parts of the path.
|
|
|
|
*
|
|
|
|
* @param object the starting object.
|
|
|
|
* @param keys an array of string keys.
|
|
|
|
* @param _default value to return if path not found.
|
|
|
|
*
|
|
|
|
* @return the value at the path if found, else default if given, else
|
|
|
|
* undefined.
|
|
|
|
*/
|
|
|
|
util.getPath = function(object, keys, _default) {
|
|
|
|
var i = 0;
|
|
|
|
var len = keys.length;
|
|
|
|
var hasNext = true;
|
|
|
|
while(hasNext && i < len &&
|
|
|
|
typeof(object) === 'object' && object !== null) {
|
|
|
|
var next = keys[i++];
|
|
|
|
hasNext = next in object;
|
|
|
|
if(hasNext) {
|
|
|
|
object = object[next];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return (hasNext ? object : _default);
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Follow a path of keys deep into an object hierarchy and delete the
|
|
|
|
* last one. If a key does not exist, do nothing.
|
|
|
|
* Used to avoid exceptions from missing parts of the path.
|
|
|
|
*
|
|
|
|
* @param object the starting object.
|
|
|
|
* @param keys an array of string keys.
|
|
|
|
*/
|
|
|
|
util.deletePath = function(object, keys) {
|
|
|
|
// need to start at an object
|
|
|
|
if(typeof(object) === 'object' && object !== null) {
|
|
|
|
var i = 0;
|
|
|
|
var len = keys.length;
|
|
|
|
while(i < len) {
|
|
|
|
var next = keys[i++];
|
|
|
|
if(i == len) {
|
|
|
|
// last
|
|
|
|
delete object[next];
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// more
|
|
|
|
if(!(next in object) ||
|
|
|
|
(typeof(object[next]) !== 'object') ||
|
|
|
|
(object[next] === null)) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
object = object[next];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Check if an object is empty.
|
|
|
|
*
|
|
|
|
* Taken from:
|
|
|
|
* http://stackoverflow.com/questions/679915/how-do-i-test-for-an-empty-javascript-object-from-json/679937#679937
|
|
|
|
*
|
|
|
|
* @param object the object to check.
|
|
|
|
*/
|
|
|
|
util.isEmpty = function(obj) {
|
|
|
|
for(var prop in obj) {
|
|
|
|
if(obj.hasOwnProperty(prop)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Format with simple printf-style interpolation.
|
|
|
|
*
|
|
|
|
* %%: literal '%'
|
|
|
|
* %s,%o: convert next argument into a string.
|
|
|
|
*
|
|
|
|
* @param format the string to format.
|
|
|
|
* @param ... arguments to interpolate into the format string.
|
|
|
|
*/
|
|
|
|
util.format = function(format) {
|
|
|
|
var re = /%./g;
|
|
|
|
// current match
|
|
|
|
var match;
|
|
|
|
// current part
|
|
|
|
var part;
|
|
|
|
// current arg index
|
|
|
|
var argi = 0;
|
|
|
|
// collected parts to recombine later
|
|
|
|
var parts = [];
|
|
|
|
// last index found
|
|
|
|
var last = 0;
|
|
|
|
// loop while matches remain
|
|
|
|
while((match = re.exec(format))) {
|
|
|
|
part = format.substring(last, re.lastIndex - 2);
|
|
|
|
// don't add empty strings (ie, parts between %s%s)
|
|
|
|
if(part.length > 0) {
|
|
|
|
parts.push(part);
|
|
|
|
}
|
|
|
|
last = re.lastIndex;
|
|
|
|
// switch on % code
|
|
|
|
var code = match[0][1];
|
|
|
|
switch(code) {
|
|
|
|
case 's':
|
|
|
|
case 'o':
|
|
|
|
// check if enough arguments were given
|
|
|
|
if(argi < arguments.length) {
|
|
|
|
parts.push(arguments[argi++ + 1]);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
parts.push('<?>');
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
// FIXME: do proper formating for numbers, etc
|
|
|
|
//case 'f':
|
|
|
|
//case 'd':
|
|
|
|
case '%':
|
|
|
|
parts.push('%');
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
parts.push('<%' + code + '?>');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// add trailing part of format string
|
|
|
|
parts.push(format.substring(last));
|
|
|
|
return parts.join('');
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Formats a number.
|
|
|
|
*
|
|
|
|
* http://snipplr.com/view/5945/javascript-numberformat--ported-from-php/
|
|
|
|
*/
|
|
|
|
util.formatNumber = function(number, decimals, dec_point, thousands_sep) {
|
|
|
|
// http://kevin.vanzonneveld.net
|
|
|
|
// + original by: Jonas Raoni Soares Silva (http://www.jsfromhell.com)
|
|
|
|
// + improved by: Kevin van Zonneveld (http://kevin.vanzonneveld.net)
|
|
|
|
// + bugfix by: Michael White (http://crestidg.com)
|
|
|
|
// + bugfix by: Benjamin Lupton
|
|
|
|
// + bugfix by: Allan Jensen (http://www.winternet.no)
|
|
|
|
// + revised by: Jonas Raoni Soares Silva (http://www.jsfromhell.com)
|
|
|
|
// * example 1: number_format(1234.5678, 2, '.', '');
|
|
|
|
// * returns 1: 1234.57
|
|
|
|
|
|
|
|
var n = number, c = isNaN(decimals = Math.abs(decimals)) ? 2 : decimals;
|
|
|
|
var d = dec_point === undefined ? ',' : dec_point;
|
|
|
|
var t = thousands_sep === undefined ?
|
|
|
|
'.' : thousands_sep, s = n < 0 ? '-' : '';
|
|
|
|
var i = parseInt((n = Math.abs(+n || 0).toFixed(c)), 10) + '';
|
|
|
|
var j = (i.length > 3) ? i.length % 3 : 0;
|
|
|
|
return s + (j ? i.substr(0, j) + t : '') +
|
|
|
|
i.substr(j).replace(/(\d{3})(?=\d)/g, '$1' + t) +
|
|
|
|
(c ? d + Math.abs(n - i).toFixed(c).slice(2) : '');
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Formats a byte size.
|
|
|
|
*
|
|
|
|
* http://snipplr.com/view/5949/format-humanize-file-byte-size-presentation-in-javascript/
|
|
|
|
*/
|
|
|
|
util.formatSize = function(size) {
|
|
|
|
if(size >= 1073741824) {
|
|
|
|
size = util.formatNumber(size / 1073741824, 2, '.', '') + ' GiB';
|
|
|
|
}
|
|
|
|
else if(size >= 1048576) {
|
|
|
|
size = util.formatNumber(size / 1048576, 2, '.', '') + ' MiB';
|
|
|
|
}
|
|
|
|
else if(size >= 1024) {
|
|
|
|
size = util.formatNumber(size / 1024, 0) + ' KiB';
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
size = util.formatNumber(size, 0) + ' bytes';
|
|
|
|
}
|
|
|
|
return size;
|
|
|
|
};
|
|
|
|
|
|
|
|
} // end module implementation
|
|
|
|
|
|
|
|
/* ########## Begin module wrapper ########## */
|
|
|
|
var name = 'util';
|
|
|
|
if(typeof define !== 'function') {
|
|
|
|
// NodeJS -> AMD
|
|
|
|
if(typeof module === 'object' && module.exports) {
|
2013-08-05 10:45:02 -04:00
|
|
|
var nodeJS = true;
|
|
|
|
define = function(ids, factory) {
|
2013-06-10 11:57:33 -04:00
|
|
|
factory(require, module);
|
|
|
|
};
|
|
|
|
}
|
|
|
|
// <script>
|
|
|
|
else {
|
|
|
|
if(typeof forge === 'undefined') {
|
|
|
|
forge = {};
|
|
|
|
}
|
2013-08-05 10:45:02 -04:00
|
|
|
return initModule(forge);
|
2013-06-10 11:57:33 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
// AMD
|
2013-08-05 10:45:02 -04:00
|
|
|
var deps;
|
|
|
|
var defineFunc = function(require, module) {
|
|
|
|
module.exports = function(forge) {
|
|
|
|
var mods = deps.map(function(dep) {
|
|
|
|
return require(dep);
|
|
|
|
}).concat(initModule);
|
|
|
|
// handle circular dependencies
|
|
|
|
forge = forge || {};
|
|
|
|
forge.defined = forge.defined || {};
|
|
|
|
if(forge.defined[name]) {
|
2013-06-10 11:57:33 -04:00
|
|
|
return forge[name];
|
2013-08-05 10:45:02 -04:00
|
|
|
}
|
|
|
|
forge.defined[name] = true;
|
|
|
|
for(var i = 0; i < mods.length; ++i) {
|
|
|
|
mods[i](forge);
|
|
|
|
}
|
|
|
|
return forge[name];
|
|
|
|
};
|
|
|
|
};
|
|
|
|
var tmpDefine = define;
|
|
|
|
define = function(ids, factory) {
|
|
|
|
deps = (typeof ids === 'string') ? factory.slice(2) : ids.slice(2);
|
|
|
|
if(nodeJS) {
|
|
|
|
delete define;
|
|
|
|
return tmpDefine.apply(null, Array.prototype.slice.call(arguments, 0));
|
|
|
|
}
|
|
|
|
define = tmpDefine;
|
|
|
|
return define.apply(null, Array.prototype.slice.call(arguments, 0));
|
|
|
|
};
|
|
|
|
define(['require', 'module'], function() {
|
|
|
|
defineFunc.apply(null, Array.prototype.slice.call(arguments, 0));
|
|
|
|
});
|
2013-06-10 11:57:33 -04:00
|
|
|
})();
|