123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176 |
- /* Copyright (c) 2009 Anton Ekblad
- Permission is hereby granted, free of charge, to any person obtaining a copy
- of this software and associated documentation files (the "Software"), to deal
- in the Software without restriction, including without limitation the rights
- to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
- copies of the Software, and to permit persons to whom the Software is
- furnished to do so, subject to the following conditions:
- The above copyright notice and this permission notice shall be included in
- all copies or substantial portions of the Software. */
- /*self.addEventListener('message', function(e) {
- self.postMessage(bdecode(e.data));
- }, false);*/
- // bencode an object
- function bencode(obj) {
- switch(btypeof(obj)) {
- case "string": return bstring(obj);
- case "number": return bint(obj);
- case "list": return blist(obj);
- case "dictionary": return bdict(obj);
- default: return null;
- }
- }
- // decode a bencoded string into a javascript object
- function bdecode(str) {
- var dec = bparse(str);
- if(dec !== null && dec[1] === "")
- return dec[0];
- return null;
- }
- // parse a bencoded string; bdecode is really just a wrapper for this one.
- // all bparse* functions return an array in the form
- // [parsed object, remaining string to parse]
- function bparse(str) {
- switch(str.charAt(0)) {
- case "d": return bparseDict(str.substr(1));
- case "l": return bparseList(str.substr(1));
- case "i": return bparseInt(str.substr(1));
- default: return bparseString(str);
- }
- }
- // parse a bencoded string
- function bparseString(str) {
- str2 = str.split(":", 1)[0];
- if(isNum(str2)) {
- len = parseInt(str2);
- return [str.substr(str2.length+1, len),
- str.substr(str2.length+1+len)];
- }
- return null;
- }
- // parse a bencoded integer
- function bparseInt(str) {
- var str2 = str.split("e", 1)[0];
- if(!isNum(str2)) {
- return null;
- }
- return [Number(str2), str.substr(str2.length+1)];
- }
- // parse a bencoded list
- function bparseList(str) {
- var p, list = [];
- while(str.charAt(0) != "e" && str.length > 0) {
- p = bparse(str);
- if(null === p)
- return null;
- list.push(p[0]);
- str = p[1];
- }
- if(str.length <= 0)
- return null;
- return [list, str.substr(1)];
- }
- // parse a bencoded dictionary
- function bparseDict(str) {
- var key, val, dict = {};
- while(str.charAt(0) != "e" && str.length > 0) {
- key = bparseString(str);
- if(null === key)
- return;
- val = bparse(key[1]);
- if(null === val)
- return null;
- dict[key[0]] = val[0];
- str = val[1];
- }
- if(str.length <= 0)
- return null;
- return [dict, str.substr(1)];
- }
- // is the given string numeric?
- function isNum(str) {
- var i, c;
- str = str.toString();
- if(str.charAt(0) == '-')
- i = 1;
- else
- i = 0;
- for(; i < str.length; i++) {
- c = str.charCodeAt(i);
- if(c < 48 || c > 57) {
- return false;
- }
- }
- return true;
- }
- // returns the bencoding type of the given object
- function btypeof(obj) {
- var type = typeof obj;
- if(type == "object") {
- if(typeof obj.length == "undefined")
- return "dictionary";
- return "list";
- }
- return type;
- }
- // bencode a string
- function bstring(str) {
- return (str.length + ":" + str);
- }
- // bencode an integer
- function bint(num) {
- return "i" + num + "e";
- }
- // bencode a list
- function blist(list) {
- var str, enclist;
- enclist = [];
- for(var key in list) {
- enclist.push(bencode(list[key]));
- }
- // enclist.sort(); WTF that's the dumbest thing I've ever seen.
- str = "l";
- for(var enckey in enclist) {
- str += enclist[enckey];
- }
- return str + "e";
- }
- // bencode a dictionary
- function bdict(dict) {
- var str, enclist;
- enclist = [];
- for(var key in dict) {
- enclist.push(bstring(key) + bencode(dict[key]));
- }
- enclist.sort();
- str = "d";
- for(var enckey in enclist) {
- str += enclist[enckey];
- }
- return str + "e";
- }
- module.exports.encode = bencode;
- module.exports.decode = bdecode;
|