strategy.js 1.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566
  1. var passport = require('passport-strategy')
  2. , siwe = require('siwe')
  3. , util = require('util')
  4. , SessionStore = require('./store/session');
  5. function Strategy(options, verify) {
  6. if (typeof options == 'function') {
  7. verify = options;
  8. options = {};
  9. }
  10. if (!verify) { throw new TypeError('EthereumStrategy requires a verify function'); }
  11. this.name = 'ethereum';
  12. this._verify = verify;
  13. this._passReqToCallback = options.passReqToCallback;
  14. this._store = options.store || new SessionStore();
  15. }
  16. /**
  17. * Inherit from `passport.Strategy`.
  18. */
  19. util.inherits(Strategy, passport.Strategy);
  20. Strategy.prototype.authenticate = function(req, options) {
  21. var message = req.body.message
  22. , signature = req.body.signature;
  23. if (!message) { return this.fail({ message: 'Missing message' }, 400); }
  24. if (!signature) { return this.fail({ message: 'Missing signature' }, 400); }
  25. var self = this;
  26. var siweMessage = new siwe.SiweMessage(message);
  27. this._store.verify(req, siweMessage.nonce, function(err, ok, info) {
  28. if (!ok) {
  29. return self.fail(info, 403);
  30. }
  31. siweMessage.validate(signature)
  32. .then(function(message) {
  33. function verified(err, user, info) {
  34. if (err) { return self.error(err); }
  35. if (!user) { return self.fail(info); }
  36. self.success(user, info);
  37. }
  38. try {
  39. if (self._passReqToCallback) {
  40. self._verify(req, message.address, verified);
  41. } else {
  42. self._verify(message.address, verified);
  43. }
  44. } catch (ex) {
  45. return self.error(ex);
  46. }
  47. })
  48. .catch(function(err) {
  49. return self.fail({ message: 'Invalid signature' });
  50. });
  51. });
  52. };
  53. /**
  54. * Expose `Strategy`.
  55. */
  56. module.exports = Strategy;