node/lib/querystring.js

159 lines
3.8 KiB
JavaScript
Raw Normal View History

// Query String Utilities
var QueryString = exports;
2010-12-02 10:07:20 +08:00
var urlDecode = process.binding('http_parser').urlDecode;
2010-11-19 07:12:21 +08:00
2010-12-02 10:07:20 +08:00
function charCode(c) {
2010-11-19 07:12:21 +08:00
return c.charCodeAt(0);
}
// a safe fast alternative to decodeURIComponent
2010-12-02 10:07:20 +08:00
QueryString.unescapeBuffer = function(s, decodeSpaces) {
2010-11-19 07:12:21 +08:00
var out = new Buffer(s.length);
2010-12-02 10:07:20 +08:00
var state = 'CHAR'; // states: CHAR, HEX0, HEX1
2010-11-19 07:12:21 +08:00
var n, m, hexchar;
for (var inIndex = 0, outIndex = 0; inIndex <= s.length; inIndex++) {
var c = s.charCodeAt(inIndex);
switch (state) {
case 'CHAR':
switch (c) {
case charCode('%'):
n = 0;
m = 0;
state = 'HEX0';
break;
case charCode('+'):
if (decodeSpaces) c = charCode(' ');
// pass thru
default:
out[outIndex++] = c;
break;
}
break;
case 'HEX0':
state = 'HEX1';
hexchar = c;
if (charCode('0') <= c && c <= charCode('9')) {
n = c - charCode('0');
} else if (charCode('a') <= c && c <= charCode('f')) {
n = c - charCode('a') + 10;
} else if (charCode('A') <= c && c <= charCode('F')) {
n = c - charCode('A') + 10;
} else {
out[outIndex++] = charCode('%');
out[outIndex++] = c;
state = 'CHAR';
break;
}
break;
case 'HEX1':
state = 'CHAR';
if (charCode('0') <= c && c <= charCode('9')) {
m = c - charCode('0');
} else if (charCode('a') <= c && c <= charCode('f')) {
m = c - charCode('a') + 10;
} else if (charCode('A') <= c && c <= charCode('F')) {
m = c - charCode('A') + 10;
} else {
out[outIndex++] = charCode('%');
out[outIndex++] = hexchar;
out[outIndex++] = c;
break;
}
2010-12-02 10:07:20 +08:00
out[outIndex++] = 16 * n + m;
2010-11-19 07:12:21 +08:00
break;
}
}
// TODO support returning arbitrary buffers.
2010-12-02 10:07:20 +08:00
return out.slice(0, outIndex - 1);
2010-11-19 07:18:06 +08:00
};
2010-12-02 10:07:20 +08:00
QueryString.unescape = function(s, decodeSpaces) {
2010-11-19 07:18:06 +08:00
return QueryString.unescapeBuffer(s, decodeSpaces).toString();
2010-11-19 07:12:21 +08:00
};
2010-12-02 10:07:20 +08:00
QueryString.escape = function(str) {
return encodeURIComponent(str);
};
var stringifyPrimitive = function(v) {
switch (typeof v) {
2010-12-02 10:07:20 +08:00
case 'string':
return v;
2010-12-02 10:07:20 +08:00
case 'boolean':
return v ? 'true' : 'false';
2010-12-02 10:07:20 +08:00
case 'number':
return isFinite(v) ? v : '';
default:
2010-12-02 10:07:20 +08:00
return '';
}
};
2010-12-02 10:07:20 +08:00
QueryString.stringify = QueryString.encode = function(obj, sep, eq, name) {
sep = sep || '&';
eq = eq || '=';
obj = (obj === null) ? undefined : obj;
2010-04-12 04:46:24 +08:00
switch (typeof obj) {
2010-12-02 10:07:20 +08:00
case 'object':
return Object.keys(obj).map(function(k) {
if (Array.isArray(obj[k])) {
return obj[k].map(function(v) {
return QueryString.escape(stringifyPrimitive(k)) +
eq +
QueryString.escape(stringifyPrimitive(v));
}).join(sep);
} else {
2010-12-02 10:07:20 +08:00
return QueryString.escape(stringifyPrimitive(k)) +
eq +
QueryString.escape(stringifyPrimitive(obj[k]));
}
}).join(sep);
2010-04-12 04:46:24 +08:00
default:
2010-12-02 10:07:20 +08:00
if (!name) return '';
return QueryString.escape(stringifyPrimitive(name)) + eq +
QueryString.escape(stringifyPrimitive(obj));
}
};
// Parse a key=val string.
2010-12-02 10:07:20 +08:00
QueryString.parse = QueryString.decode = function(qs, sep, eq) {
sep = sep || '&';
eq = eq || '=';
var obj = {};
if (typeof qs !== 'string' || qs.length === 0) {
return obj;
}
qs.split(sep).forEach(function(kvp) {
var x = kvp.split(eq);
var k = QueryString.unescape(x[0], true);
var v = QueryString.unescape(x.slice(1).join(eq), true);
if (!(k in obj)) {
2010-12-02 10:07:20 +08:00
obj[k] = v;
} else if (!Array.isArray(obj[k])) {
2010-12-02 10:07:20 +08:00
obj[k] = [obj[k], v];
} else {
2010-12-02 10:07:20 +08:00
obj[k].push(v);
}
});
return obj;
};