Skip to content

Commit 1e46c5a

Browse files
committed
change jwt.sign to return errors on callback instead of throwing errors
1 parent 65aadb4 commit 1e46c5a

File tree

2 files changed

+33
-10
lines changed

2 files changed

+33
-10
lines changed

sign.js

+16-9
Original file line numberDiff line numberDiff line change
@@ -48,13 +48,20 @@ module.exports = function(payload, secretOrPrivateKey, options, callback) {
4848
typ: typeof payload === 'object' ? 'JWT' : undefined
4949
}, options.header);
5050

51+
function failure (err) {
52+
if (callback) {
53+
return callback(err);
54+
}
55+
throw err;
56+
}
57+
5158
if (typeof payload === 'undefined') {
52-
throw new Error('payload is required');
59+
return failure(new Error('payload is required'));
5360
} else if (typeof payload === 'object') {
5461
var payload_validation_result = registered_claims_schema.validate(payload);
5562

5663
if (payload_validation_result.error) {
57-
throw payload_validation_result.error;
64+
return failure(payload_validation_result.error);
5865
}
5966

6067
payload = xtend(payload);
@@ -64,22 +71,22 @@ module.exports = function(payload, secretOrPrivateKey, options, callback) {
6471
});
6572

6673
if (invalid_options.length > 0) {
67-
throw new Error('invalid ' + invalid_options.join(',') + ' option for ' + (typeof payload ) + ' payload' );
74+
return failure(new Error('invalid ' + invalid_options.join(',') + ' option for ' + (typeof payload ) + ' payload' ));
6875
}
6976
}
7077

7178
if (typeof payload.exp !== 'undefined' && typeof options.expiresIn !== 'undefined') {
72-
throw new Error('Bad "options.expiresIn" option the payload already has an "exp" property.');
79+
return failure(new Error('Bad "options.expiresIn" option the payload already has an "exp" property.'));
7380
}
7481

7582
if (typeof payload.nbf !== 'undefined' && typeof options.notBefore !== 'undefined') {
76-
throw new Error('Bad "options.notBefore" option the payload already has an "nbf" property.');
83+
return failure(new Error('Bad "options.notBefore" option the payload already has an "nbf" property.'));
7784
}
7885

7986
var validation_result = sign_options_schema.validate(options);
8087

8188
if (validation_result.error) {
82-
throw validation_result.error;
89+
return failure(validation_result.error);
8390
}
8491

8592
var timestamp = payload.iat || Math.floor(Date.now() / 1000);
@@ -93,22 +100,22 @@ module.exports = function(payload, secretOrPrivateKey, options, callback) {
93100
if (typeof options.notBefore !== 'undefined') {
94101
payload.nbf = timespan(options.notBefore);
95102
if (typeof payload.nbf === 'undefined') {
96-
throw new Error('"notBefore" should be a number of seconds or string representing a timespan eg: "1d", "20h", 60');
103+
return failure(new Error('"notBefore" should be a number of seconds or string representing a timespan eg: "1d", "20h", 60'));
97104
}
98105
}
99106

100107
if (typeof options.expiresIn !== 'undefined' && typeof payload === 'object') {
101108
payload.exp = timespan(options.expiresIn);
102109
if (typeof payload.exp === 'undefined') {
103-
throw new Error('"expiresIn" should be a number of seconds or string representing a timespan eg: "1d", "20h", 60');
110+
return failure(new Error('"expiresIn" should be a number of seconds or string representing a timespan eg: "1d", "20h", 60'));
104111
}
105112
}
106113

107114
Object.keys(options_to_payload).forEach(function (key) {
108115
var claim = options_to_payload[key];
109116
if (typeof options[key] !== 'undefined') {
110117
if (typeof payload[claim] !== 'undefined') {
111-
throw new Error('Bad "options.' + key + '" option. The payload already has an "' + claim + '" property.');
118+
return failure(new Error('Bad "options.' + key + '" option. The payload already has an "' + claim + '" property.'));
112119
}
113120
payload[claim] = options[key];
114121
}

test/async_sign.tests.js

+17-1
Original file line numberDiff line numberDiff line change
@@ -17,12 +17,28 @@ describe('signing a token asynchronously', function() {
1717
});
1818
});
1919

20-
it('should throw error', function(done) {
20+
it('should return error when secret is not a cert for RS256', function(done) {
2121
//this throw an error because the secret is not a cert and RS256 requires a cert.
2222
jwt.sign({ foo: 'bar' }, secret, { algorithm: 'RS256' }, function (err) {
2323
expect(err).to.be.ok();
2424
done();
2525
});
2626
});
27+
28+
it('should return error on wrong arguments', function(done) {
29+
//this throw an error because the secret is not a cert and RS256 requires a cert.
30+
jwt.sign({ foo: 'bar' }, secret, { notBefore: {} }, function (err) {
31+
expect(err).to.be.ok();
32+
done();
33+
});
34+
});
35+
36+
it('should return error on wrong arguments (2)', function(done) {
37+
jwt.sign('string', 'secret', {noTimestamp: true}, function (err) {
38+
expect(err).to.be.ok();
39+
expect(err).to.be.instanceof(Error);
40+
done();
41+
});
42+
});
2743
});
2844
});

0 commit comments

Comments
 (0)