xauth.js 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421
  1. /*
  2. Copyright 2011 - Tommi Laukkanen (www.substanceofcode.com)
  3. This file is part of TwimGo.
  4. NewsFlow is free software: you can redistribute it and/or modify
  5. it under the terms of the GNU Lesser General Public License as published by
  6. the Free Software Foundation, either version 3 of the License, or
  7. (at your option) any later version.
  8. Foobar is distributed in the hope that it will be useful,
  9. but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  11. GNU Lesser General Public License for more details.
  12. You should have received a copy of the GNU Lesser General Public License
  13. along with NewsFlow. If not, see <http://www.gnu.org/licenses/>.
  14. */
  15. Qt.include("lib/sha1.js")
  16. String.prototype.startsWith = function(str) {return (this.match("^"+str)==str)}
  17. function XAuth() {
  18. var OAUTH_CONSUMER_TOKEN = "rUeGDPrcxbYN6r63JA39Kg";
  19. var OAUTH_CONSUMER_SECRET = "EoNvzuQ5gz0LhbbssK4B9e9REdlN52ngl6YoN6eUqrs";
  20. var HMACSHA1SignatureType = "HMAC-SHA1";
  21. var OAuthVersion = "1.0";
  22. var OAuthParameterPrefix = "oauth_";
  23. var OAuthConsumerKeyKey = "oauth_consumer_key";
  24. var OAuthCallbackKey = "oauth_callback";
  25. var OAuthVersionKey = "oauth_version";
  26. var OAuthSignatureMethodKey = "oauth_signature_method";
  27. var OAuthSignatureKey = "oauth_signature";
  28. var OAuthTimestampKey = "oauth_timestamp";
  29. var OAuthNonceKey = "oauth_nonce";
  30. var OAuthTokenKey = "oauth_token";
  31. var OAuthTokenSecretKey = "oauth_token_secret";
  32. var OAuthVerifier = "oauth_verifier";
  33. var token = "";
  34. var tokenSecret = "";
  35. var timeCorrection = 0;
  36. var showError;
  37. var OAUTH_ACCESS_TOKEN_URL = "https://api.twitter.com/oauth/access_token";
  38. this.setErrorCallback = function(callback) {
  39. showError = callback;
  40. }
  41. this.setTokenAndSecret = function(newToken, newTokenSecret) {
  42. token = newToken;
  43. tokenSecret = newTokenSecret;
  44. }
  45. function doWebRequest(method, url, callback, errorCallback) {
  46. var doc = new XMLHttpRequest();
  47. //console.log("HTTPRequest to " + url);
  48. doc.onreadystatechange = function() {
  49. var shortURL = url;
  50. if (doc.readyState == XMLHttpRequest.HEADERS_RECEIVED) {
  51. var status = doc.status;
  52. //console.log("HTTP status on headers " + status);
  53. if(timeCorrection==0) {
  54. var twitterDateString = doc.getResponseHeader("Date");
  55. //console.log("Twitter date: " + twitterDateString);
  56. if(twitterDateString.length>0) {
  57. var twitterDate = parseDate(twitterDateString);
  58. timeCorrection = twitterDate.getTime() - (new Date()).getTime();
  59. //console.log("Twitter time offset: " + timeCorrection);
  60. } else {
  61. //console.log("Couldn't parse Twitter time offset");
  62. }
  63. }
  64. if(status!=200) {
  65. if(shortURL.length>32) {
  66. shortURL = shortURL.substring(0,40) + "..";
  67. }
  68. showError("Twitter API returned " + status + " " + doc.statusText + " on URL " + shortURL);
  69. return;
  70. }
  71. } else if (doc.readyState == XMLHttpRequest.DONE) {
  72. var status = doc.status;
  73. if(timeCorrection==0) {
  74. var twitterDateString = doc.getResponseHeader("Date");
  75. //console.log("Twitter date: " + twitterDateString);
  76. if(twitterDateString.length>0) {
  77. var twitterDate = parseDate(twitterDateString);
  78. timeCorrection = twitterDate.getTime() - (new Date()).getTime();
  79. //console.log("Twitter time offset: " + timeCorrection);
  80. } else {
  81. //console.log("Couldn't parse Twitter time offset");
  82. }
  83. }
  84. if(status!=200) {
  85. console.log("HTTP status " + status);
  86. if(shortURL.length>40) {
  87. shortURL = shortURL.substring(0,40) + "..";
  88. }
  89. showError("Twitter API returned " + status + " " + doc.statusText + " on URL " + shortURL);
  90. return;
  91. }
  92. var data = doc.responseText;
  93. if(typeof(data)=="undefined") {
  94. data = "";
  95. }
  96. //console.log("data=" + data);
  97. //console.log("Callback=" + callback);
  98. callback(data);
  99. }
  100. }
  101. doc.open(method, url);
  102. doc.send();
  103. }
  104. this.getQueryParameters = function(url) {
  105. var questionMarkIndex = url.indexOf("?");
  106. if(questionMarkIndex<0) {
  107. return new Array();
  108. }
  109. var parameters = url.substring(questionMarkIndex+1);
  110. var params = new Array();
  111. var para = parameters.split("&");
  112. for(var i=0; i<para.length; i++) {
  113. if(para[i].startsWith("oauth")==false) {
  114. var nameValue = para[i].split("=");
  115. var q = [nameValue[0], nameValue[1]];
  116. params.push(q);
  117. }
  118. }
  119. return params;
  120. }
  121. this.generateSignatureBase = function(url,consumerKey,token,tokenSecret,httpMethod,timeStamp,nonce,signatureType) {
  122. if (typeof(token)=="undefined")
  123. {
  124. token = "";
  125. }
  126. if (typeof(tokenSecret)=="undefined")
  127. {
  128. tokenSecret = "";
  129. }
  130. var parameters = this.getQueryParameters(url);
  131. parameters.push( [OAuthVersionKey, OAuthVersion] );
  132. parameters.push( [OAuthNonceKey, nonce] );
  133. parameters.push( [OAuthTimestampKey, timeStamp] );
  134. parameters.push( [OAuthSignatureMethodKey, signatureType] );
  135. parameters.push( [OAuthConsumerKeyKey, consumerKey] );
  136. if (typeof(token)!="undefined" && token!="")
  137. {
  138. parameters.push( [OAuthTokenKey, token] );
  139. }
  140. this.sortParameters( parameters );
  141. var normalizedUrl = this.getSchemeAndHost(url);
  142. normalizedUrl += this.getAbsolutePath(url);
  143. var normalizedRequestParameters = this.normalizeRequestParameters(parameters);
  144. var signatureBase = "";
  145. signatureBase += httpMethod + "&";
  146. signatureBase += this.encode(normalizedUrl) + "&";
  147. signatureBase += this.encode(normalizedRequestParameters);
  148. var returnObject = new Object();
  149. returnObject.normalizedUrl = normalizedUrl;
  150. returnObject.normalizedRequestParameters = normalizedRequestParameters;
  151. returnObject.signatureBase = signatureBase;
  152. return returnObject;
  153. }
  154. this.getSchemeAndHost = function(url) {
  155. var startIndex = url.indexOf("//")+2;
  156. var endIndex = url.indexOf("/", startIndex);
  157. return url.substring(0,endIndex);
  158. }
  159. this.getAbsolutePath = function(url) {
  160. var startIndex = url.indexOf("//")+2;
  161. var endIndex = url.indexOf("/", startIndex);
  162. var questionMark = url.indexOf("?");
  163. if(questionMark>0) {
  164. return url.substring(endIndex, questionMark);
  165. } else {
  166. return url.substring(endIndex);
  167. }
  168. }
  169. this.sortParameters = function(items) {
  170. items.sort();
  171. }
  172. this.generateSignature = function(url, consumerKey, consumerSecret,token,tokenSecret,httpMethod,timeStamp,nonce) {
  173. var obj = this.generateSignatureBase(
  174. url,
  175. consumerKey,
  176. token,
  177. tokenSecret,
  178. httpMethod,
  179. timeStamp,
  180. nonce,
  181. HMACSHA1SignatureType);
  182. var signatureBase = obj.signatureBase;
  183. var tokenSec = "";
  184. if(typeof(tokenSecret)!="undefined") {
  185. tokenSec = tokenSecret;
  186. }
  187. var key = this.encode(consumerSecret) + "&" + this.encode(tokenSec);
  188. var signature = this.getSignature(signatureBase, key);
  189. obj.signature = signature;
  190. return obj;
  191. }
  192. this.getSignature = function(message, key) {
  193. var b64pad = '=';
  194. var signature = b64_hmac_sha1(key, message);
  195. return signature;
  196. }
  197. this.normalizeRequestParameters = function( parameters ) {
  198. var sb = "";
  199. for(var i in parameters) {
  200. var par = parameters[i];
  201. sb += par[0] + "=" + par[1] + "&";
  202. }
  203. return sb.substring(0, sb.length-1);
  204. }
  205. this.generateTimeStamp = function() {
  206. //var d = new Date();
  207. //var utc = Date.UTC(d.getFullYear(), d.getMonth(), d.getDate(), d.getHours(), d.getMinutes(), d.getSeconds());
  208. //var t = utc + 0;
  209. var p = (new Date()).getTime() + timeCorrection;
  210. //console.log("t: " + t);
  211. //console.log("p: " + p);
  212. return Math.floor(p / 1000);
  213. }
  214. // pass in the 'created_at' string returned from twitter //
  215. // stamp arrives formatted as Tue Apr 07 22:52:51 +0000 2009 //
  216. // Wed, 10 Aug 2011 19:30:24 GMT
  217. function parseDate(stamp)
  218. {
  219. try {
  220. var parts = stamp.split(" ");
  221. var dayName;
  222. var monthName;
  223. var day;
  224. var time;
  225. var hours;
  226. var minutes;
  227. var seconds;
  228. var offset;
  229. var year;
  230. // Wed, 10 Aug 2011 19:30:24 GMT
  231. dayName = parts[0];
  232. day = parseInt(parts[1], 10);
  233. monthName = parts[2];
  234. year = parseInt(parts[3], 10);
  235. time = parts[4].split(":");
  236. hours = parseInt(time[0], 10);
  237. minutes = parseInt(time[1], 10);
  238. seconds = parseInt(time[2], 10);
  239. var month = 0;
  240. if(monthName=="Jan") {
  241. month = 0;
  242. } else if(monthName=="Feb") {
  243. month = 1;
  244. } else if(monthName=="Mar") {
  245. month = 2;
  246. } else if(monthName=="Apr") {
  247. month = 3;
  248. } else if(monthName=="May") {
  249. month = 4;
  250. } else if(monthName=="Jun") {
  251. month = 5;
  252. } else if(monthName=="Jul") {
  253. month = 6;
  254. } else if(monthName=="Aug") {
  255. month = 7;
  256. } else if(monthName=="Sep") {
  257. month = 8;
  258. } else if(monthName=="Oct") {
  259. month = 9;
  260. } else if(monthName=="Nov") {
  261. month = 10;
  262. } else if(monthName=="Dec") {
  263. month = 11;
  264. }
  265. var dt = new Date();
  266. // dt.setDate(day);
  267. // dt.setYear(year);
  268. // dt.setMonth(month);
  269. // dt.setHours(hours);
  270. // dt.setMinutes(minutes);
  271. // dt.setSeconds(seconds);
  272. dt.setUTCDate(day);
  273. dt.setYear(year);
  274. dt.setUTCMonth(month);
  275. dt.setUTCHours(hours);
  276. dt.setUTCMinutes(minutes);
  277. dt.setUTCSeconds(seconds);
  278. return dt;
  279. } catch(err) {
  280. console.log("Error while parsing date: " + err);
  281. return new Date();
  282. }
  283. }
  284. var nonceChars = "0123456789ABCDEFGHIJKLMNOPQRSTUVWXTZabcdefghiklmnopqrstuvwxyz";
  285. this.generateNonce = function(length) {
  286. var chars = nonceChars;
  287. var result = "";
  288. for (var i = 0; i < length; ++i) {
  289. var rnum = Math.floor(Math.random() * chars.length);
  290. result += chars.substring(rnum, rnum+1);
  291. }
  292. return result;
  293. }
  294. // private String unreservedCharactersPattern = "[a-zA-Z0-9\\-\\._~]";
  295. // private String unreservedCharacters = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789-._~";
  296. this.encode = function(s) {
  297. if (typeof(s)=="undefined" || s=="") {
  298. return "";
  299. }
  300. s = encodeURIComponent(s);
  301. // Now replace the values which encodeURIComponent doesn't do
  302. // encodeURIComponent ignores: - _ . ! ~ * ' ( )
  303. // OAuth dictates the only ones you can ignore are: - _ . ~
  304. // Source: http://developer.mozilla.org/en/docs/Core_JavaScript_1.5_Reference:Global_Functions:encodeURIComponent
  305. s = s.replace(/\!/g, "%21");
  306. s = s.replace(/\*/g, "%2A");
  307. s = s.replace(/\'/g, "%27");
  308. s = s.replace(/\(/g, "%28");
  309. s = s.replace(/\)/g, "%29");
  310. return s;
  311. }
  312. this.webRequest = function(isPost, url, parameters, callback) {
  313. var method = "GET"
  314. //Setup postData for signing.
  315. //Add the postData to the querystring.
  316. var postData = "";
  317. if (isPost)
  318. {
  319. method = "POST";
  320. }
  321. if (typeof(parameters)!=undefined && parameters!=null && parameters.length>0)
  322. {
  323. //Decode the parameters and re-encode using the oAuth UrlEncode method.
  324. for(var i in parameters) {
  325. var q = parameters[i];
  326. if(typeof(q[0]!="undefined" && q[0]!="" && typeof(q[1])!="undefined" && q[1]!="")) {
  327. if(postData.length>0) {
  328. postData += "&";
  329. }
  330. postData += q[0] + "=" + this.encode(q[1]);
  331. }
  332. }
  333. if (url.indexOf("?") > 0)
  334. {
  335. url += "&";
  336. }
  337. else
  338. {
  339. url += "?";
  340. }
  341. url += postData;
  342. }
  343. //console.log("URL before: " + url);
  344. while(url.indexOf("?&")>0) {
  345. url = url.replace("?&", "?");
  346. }
  347. //console.log("URL after: " + url);
  348. var nonce = this.generateNonce(16);
  349. var timeStamp = this.generateTimeStamp();
  350. //Generate Signature
  351. var obj = this.generateSignature(
  352. url,
  353. OAUTH_CONSUMER_TOKEN,
  354. OAUTH_CONSUMER_SECRET,
  355. token,
  356. tokenSecret,
  357. method,
  358. timeStamp,
  359. nonce);
  360. var sig = obj.signature;
  361. var outUrl = obj.normalizedUrl;
  362. var querystring = obj.normalizedRequestParameters;
  363. //if(token.length>0) {
  364. if(querystring.length>0) {
  365. querystring += "&";
  366. }
  367. querystring += "oauth_signature=" + this.encode(sig);
  368. //}
  369. if (querystring.length > 0)
  370. {
  371. outUrl += "?";
  372. }
  373. doWebRequest(method, outUrl + querystring, callback);
  374. }
  375. }