test_hawkrequest.js 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236
  1. /* Any copyright is dedicated to the Public Domain.
  2. http://creativecommons.org/publicdomain/zero/1.0/ */
  3. "use strict";
  4. Cu.import("resource://gre/modules/Log.jsm");
  5. Cu.import("resource://services-common/utils.js");
  6. Cu.import("resource://services-common/hawkrequest.js");
  7. // https://github.com/mozilla/fxa-auth-server/wiki/onepw-protocol#wiki-use-session-certificatesign-etc
  8. var SESSION_KEYS = {
  9. sessionToken: h("a0a1a2a3a4a5a6a7 a8a9aaabacadaeaf"+
  10. "b0b1b2b3b4b5b6b7 b8b9babbbcbdbebf"),
  11. tokenID: h("c0a29dcf46174973 da1378696e4c82ae"+
  12. "10f723cf4f4d9f75 e39f4ae3851595ab"),
  13. reqHMACkey: h("9d8f22998ee7f579 8b887042466b72d5"+
  14. "3e56ab0c094388bf 65831f702d2febc0"),
  15. };
  16. function do_register_cleanup() {
  17. Services.prefs.resetUserPrefs();
  18. // remove the pref change listener
  19. let hawk = new HAWKAuthenticatedRESTRequest("https://example.com");
  20. hawk._intl.uninit();
  21. }
  22. function run_test() {
  23. Log.repository.getLogger("Services.Common.RESTRequest").level =
  24. Log.Level.Trace;
  25. initTestLogging("Trace");
  26. run_next_test();
  27. }
  28. add_test(function test_intl_accept_language() {
  29. let testCount = 0;
  30. let languages = [
  31. "zu-NP;vo", // Nepalese dialect of Zulu, defaulting to Volapük
  32. "fa-CG;ik", // Congolese dialect of Farsei, defaulting to Inupiaq
  33. ];
  34. function setLanguagePref(lang) {
  35. let acceptLanguage = Cc["@mozilla.org/supports-string;1"]
  36. .createInstance(Ci.nsISupportsString);
  37. acceptLanguage.data = lang;
  38. Services.prefs.setComplexValue(
  39. "intl.accept_languages", Ci.nsISupportsString, acceptLanguage);
  40. }
  41. let hawk = new HAWKAuthenticatedRESTRequest("https://example.com");
  42. Services.prefs.addObserver("intl.accept_languages", checkLanguagePref, false);
  43. setLanguagePref(languages[testCount]);
  44. function checkLanguagePref() {
  45. var _done = false;
  46. CommonUtils.nextTick(function() {
  47. // Ensure we're only called for the number of entries in languages[].
  48. do_check_true(testCount < languages.length);
  49. do_check_eq(hawk._intl.accept_languages, languages[testCount]);
  50. testCount++;
  51. if (testCount < languages.length) {
  52. // Set next language in prefs; Pref service will call checkNextLanguage.
  53. setLanguagePref(languages[testCount]);
  54. return;
  55. }
  56. // We've checked all the entries in languages[]. Cleanup and move on.
  57. do_print("Checked " + testCount + " languages. Removing checkLanguagePref as pref observer.");
  58. Services.prefs.removeObserver("intl.accept_languages", checkLanguagePref);
  59. run_next_test();
  60. return;
  61. });
  62. }
  63. });
  64. add_test(function test_hawk_authenticated_request() {
  65. let onProgressCalled = false;
  66. let postData = {your: "data"};
  67. // An arbitrary date - Feb 2, 1971. It ends in a bunch of zeroes to make our
  68. // computation with the hawk timestamp easier, since hawk throws away the
  69. // millisecond values.
  70. let then = 34329600000;
  71. let clockSkew = 120000;
  72. let timeOffset = -1 * clockSkew;
  73. let localTime = then + clockSkew;
  74. // Set the accept-languages pref to the Nepalese dialect of Zulu.
  75. let acceptLanguage = Cc['@mozilla.org/supports-string;1'].createInstance(Ci.nsISupportsString);
  76. acceptLanguage.data = 'zu-NP'; // omit trailing ';', which our HTTP libs snip
  77. Services.prefs.setComplexValue('intl.accept_languages', Ci.nsISupportsString, acceptLanguage);
  78. let credentials = {
  79. id: "eyJleHBpcmVzIjogMTM2NTAxMDg5OC4x",
  80. key: "qTZf4ZFpAMpMoeSsX3zVRjiqmNs=",
  81. algorithm: "sha256"
  82. };
  83. let server = httpd_setup({
  84. "/elysium": function(request, response) {
  85. do_check_true(request.hasHeader("Authorization"));
  86. // check that the header timestamp is our arbitrary system date, not
  87. // today's date. Note that hawk header timestamps are in seconds, not
  88. // milliseconds.
  89. let authorization = request.getHeader("Authorization");
  90. let tsMS = parseInt(/ts="(\d+)"/.exec(authorization)[1], 10) * 1000;
  91. do_check_eq(tsMS, then);
  92. // This testing can be a little wonky. In an environment where
  93. // pref("intl.accept_languages") === 'en-US, en'
  94. // the header is sent as:
  95. // 'en-US,en;q=0.5'
  96. // hence our fake value for acceptLanguage.
  97. let lang = request.getHeader("Accept-Language");
  98. do_check_eq(lang, acceptLanguage);
  99. let message = "yay";
  100. response.setStatusLine(request.httpVersion, 200, "OK");
  101. response.bodyOutputStream.write(message, message.length);
  102. }
  103. });
  104. function onProgress() {
  105. onProgressCalled = true;
  106. }
  107. function onComplete(error) {
  108. do_check_eq(200, this.response.status);
  109. do_check_eq(this.response.body, "yay");
  110. do_check_true(onProgressCalled);
  111. Services.prefs.resetUserPrefs();
  112. let pref = Services.prefs.getComplexValue(
  113. "intl.accept_languages", Ci.nsIPrefLocalizedString);
  114. do_check_neq(acceptLanguage.data, pref.data);
  115. server.stop(run_next_test);
  116. }
  117. let url = server.baseURI + "/elysium";
  118. let extra = {
  119. now: localTime,
  120. localtimeOffsetMsec: timeOffset
  121. };
  122. let request = new HAWKAuthenticatedRESTRequest(url, credentials, extra);
  123. // Allow hawk._intl to respond to the language pref change
  124. CommonUtils.nextTick(function() {
  125. request.post(postData, onComplete, onProgress);
  126. });
  127. });
  128. add_test(function test_hawk_language_pref_changed() {
  129. let languages = [
  130. "zu-NP", // Nepalese dialect of Zulu
  131. "fa-CG", // Congolese dialect of Farsi
  132. ];
  133. let credentials = {
  134. id: "eyJleHBpcmVzIjogMTM2NTAxMDg5OC4x",
  135. key: "qTZf4ZFpAMpMoeSsX3zVRjiqmNs=",
  136. algorithm: "sha256",
  137. };
  138. function setLanguage(lang) {
  139. let acceptLanguage = Cc["@mozilla.org/supports-string;1"].createInstance(Ci.nsISupportsString);
  140. acceptLanguage.data = lang;
  141. Services.prefs.setComplexValue("intl.accept_languages", Ci.nsISupportsString, acceptLanguage);
  142. }
  143. let server = httpd_setup({
  144. "/foo": function(request, response) {
  145. do_check_eq(languages[1], request.getHeader("Accept-Language"));
  146. response.setStatusLine(request.httpVersion, 200, "OK");
  147. },
  148. });
  149. let url = server.baseURI + "/foo";
  150. let postData = {};
  151. let request;
  152. setLanguage(languages[0]);
  153. // A new request should create the stateful object for tracking the current
  154. // language.
  155. request = new HAWKAuthenticatedRESTRequest(url, credentials);
  156. CommonUtils.nextTick(testFirstLanguage);
  157. function testFirstLanguage() {
  158. do_check_eq(languages[0], request._intl.accept_languages);
  159. // Change the language pref ...
  160. setLanguage(languages[1]);
  161. CommonUtils.nextTick(testRequest);
  162. }
  163. function testRequest() {
  164. // Change of language pref should be picked up, which we can see on the
  165. // server by inspecting the request headers.
  166. request = new HAWKAuthenticatedRESTRequest(url, credentials);
  167. request.post({}, function(error) {
  168. do_check_null(error);
  169. do_check_eq(200, this.response.status);
  170. Services.prefs.resetUserPrefs();
  171. server.stop(run_next_test);
  172. });
  173. }
  174. });
  175. add_task(function test_deriveHawkCredentials() {
  176. let credentials = deriveHawkCredentials(
  177. SESSION_KEYS.sessionToken, "sessionToken");
  178. do_check_eq(credentials.algorithm, "sha256");
  179. do_check_eq(credentials.id, SESSION_KEYS.tokenID);
  180. do_check_eq(CommonUtils.bytesAsHex(credentials.key), SESSION_KEYS.reqHMACkey);
  181. });
  182. // turn formatted test vectors into normal hex strings
  183. function h(hexStr) {
  184. return hexStr.replace(/\s+/g, "");
  185. }