connect.js 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197
  1. 'use strict';
  2. var connect = require('../lib/connect').connect;
  3. var credentialsFromUrl = require('../lib/connect').credentialsFromUrl;
  4. var defs = require('../lib/defs');
  5. var assert = require('assert');
  6. var util = require('./util');
  7. var net = require('net');
  8. var fail = util.fail, succeed = util.succeed, latch = util.latch,
  9. kCallback = util.kCallback,
  10. succeedIfAttributeEquals = util.succeedIfAttributeEquals;
  11. var format = require('util').format;
  12. var URL = process.env.URL || 'amqp://localhost';
  13. var urlparse = require('url-parse');
  14. suite("Credentials", function() {
  15. function checkCreds(creds, user, pass, done) {
  16. if (creds.mechanism != 'PLAIN') {
  17. return done('expected mechanism PLAIN');
  18. }
  19. if (creds.username != user || creds.password != pass) {
  20. return done(format("expected '%s', '%s'; got '%s', '%s'",
  21. user, pass, creds.username, creds.password));
  22. }
  23. done();
  24. }
  25. test("no creds", function(done) {
  26. var parts = urlparse('amqp://localhost');
  27. var creds = credentialsFromUrl(parts);
  28. checkCreds(creds, 'guest', 'guest', done);
  29. });
  30. test("usual user:pass", function(done) {
  31. var parts = urlparse('amqp://user:pass@localhost')
  32. var creds = credentialsFromUrl(parts);
  33. checkCreds(creds, 'user', 'pass', done);
  34. });
  35. test("missing user", function(done) {
  36. var parts = urlparse('amqps://:password@localhost');
  37. var creds = credentialsFromUrl(parts);
  38. checkCreds(creds, '', 'password', done);
  39. });
  40. test("missing password", function(done) {
  41. var parts = urlparse('amqps://username:@localhost');
  42. var creds = credentialsFromUrl(parts);
  43. checkCreds(creds, 'username', '', done);
  44. });
  45. test("escaped colons", function(done) {
  46. var parts = urlparse('amqp://user%3Aname:pass%3Aword@localhost')
  47. var creds = credentialsFromUrl(parts);
  48. checkCreds(creds, 'user:name', 'pass:word', done);
  49. });
  50. });
  51. suite("Connect API", function() {
  52. test("Connection refused", function(done) {
  53. connect('amqp://localhost:23450', {},
  54. kCallback(fail(done), succeed(done)));
  55. });
  56. // %% this ought to fail the promise, rather than throwing an error
  57. test("bad URL", function() {
  58. assert.throws(function() {
  59. connect('blurble');
  60. });
  61. });
  62. test("wrongly typed open option", function(done) {
  63. var url = require('url');
  64. var parts = url.parse(URL, true);
  65. var q = parts.query || {};
  66. q.frameMax = 'NOT A NUMBER';
  67. parts.query = q;
  68. var u = url.format(parts);
  69. connect(u, {}, kCallback(fail(done), succeed(done)));
  70. });
  71. test("serverProperties", function(done) {
  72. var url = require('url');
  73. var parts = url.parse(URL, true);
  74. var config = parts.query || {};
  75. connect(config, {}, function(err, connection) {
  76. if (err) { return done(err); }
  77. assert.equal(connection.serverProperties.product, 'RabbitMQ');
  78. done();
  79. });
  80. });
  81. test("using custom heartbeat option", function(done) {
  82. var url = require('url');
  83. var parts = url.parse(URL, true);
  84. var config = parts.query || {};
  85. config.heartbeat = 20;
  86. connect(config, {}, kCallback(succeedIfAttributeEquals('heartbeat', 20, done), fail(done)));
  87. });
  88. test("wrongly typed heartbeat option", function(done) {
  89. var url = require('url');
  90. var parts = url.parse(URL, true);
  91. var config = parts.query || {};
  92. config.heartbeat = 'NOT A NUMBER';
  93. connect(config, {}, kCallback(fail(done), succeed(done)));
  94. });
  95. test("using plain credentials", function(done) {
  96. var url = require('url');
  97. var parts = url.parse(URL, true);
  98. var u = 'guest', p = 'guest';
  99. if (parts.auth) {
  100. var auth = parts.auth.split(":");
  101. u = auth[0], p = auth[1];
  102. }
  103. connect(URL, {credentials: require('../lib/credentials').plain(u, p)},
  104. kCallback(succeed(done), fail(done)));
  105. });
  106. test("using amqplain credentials", function(done) {
  107. var url = require('url');
  108. var parts = url.parse(URL, true);
  109. var u = 'guest', p = 'guest';
  110. if (parts.auth) {
  111. var auth = parts.auth.split(":");
  112. u = auth[0], p = auth[1];
  113. }
  114. connect(URL, {credentials: require('../lib/credentials').amqplain(u, p)},
  115. kCallback(succeed(done), fail(done)));
  116. });
  117. test("using unsupported mechanism", function(done) {
  118. var creds = {
  119. mechanism: 'UNSUPPORTED',
  120. response: function() { return Buffer.from(''); }
  121. };
  122. connect(URL, {credentials: creds},
  123. kCallback(fail(done), succeed(done)));
  124. });
  125. test("with a given connection timeout", function(done) {
  126. var timeoutServer = net.createServer(function() {}).listen(31991);
  127. connect('amqp://localhost:31991', {timeout: 50}, function(err, val) {
  128. timeoutServer.close();
  129. if (val) done(new Error('Expected connection timeout, did not'));
  130. else done();
  131. });
  132. });
  133. });
  134. suite('Errors on connect', function() {
  135. var server
  136. teardown(function() {
  137. if (server) {
  138. server.close();
  139. }
  140. })
  141. test("closes underlying connection on authentication error", function(done) {
  142. var bothDone = latch(2, done);
  143. server = net.createServer(function(socket) {
  144. socket.once('data', function(protocolHeader) {
  145. assert.deepStrictEqual(
  146. protocolHeader,
  147. Buffer.from("AMQP" + String.fromCharCode(0,0,9,1))
  148. );
  149. util.runServer(socket, function(send, wait) {
  150. send(defs.ConnectionStart,
  151. {versionMajor: 0,
  152. versionMinor: 9,
  153. serverProperties: {},
  154. mechanisms: Buffer.from('PLAIN'),
  155. locales: Buffer.from('en_US')});
  156. wait(defs.ConnectionStartOk)().then(function() {
  157. send(defs.ConnectionClose,
  158. {replyCode: 403,
  159. replyText: 'ACCESS_REFUSED - Login was refused using authentication mechanism PLAIN',
  160. classId: 0,
  161. methodId: 0});
  162. });
  163. });
  164. });
  165. // Wait for the connection to be closed after the authentication error
  166. socket.once('end', function() {
  167. bothDone();
  168. });
  169. }).listen(0);
  170. connect('amqp://localhost:' + server.address().port, {}, function(err) {
  171. if (!err) bothDone(new Error('Expected authentication error'));
  172. bothDone();
  173. });
  174. });
  175. });