Skip to content

Commit

Permalink
Simplifies proactive 3-D Secure APIs
Browse files Browse the repository at this point in the history
  • Loading branch information
chrissrogers committed Sep 25, 2024
1 parent a6069aa commit 7973024
Show file tree
Hide file tree
Showing 6 changed files with 42 additions and 72 deletions.
1 change: 1 addition & 0 deletions lib/recurly.js
Original file line number Diff line number Diff line change
Expand Up @@ -74,6 +74,7 @@ const DEFAULTS = {
preflightDeviceDataCollector: true,
proactive: {
enabled: false,
gatewayCode: ''
}
}
},
Expand Down
13 changes: 6 additions & 7 deletions lib/recurly/risk/risk.js
Original file line number Diff line number Diff line change
Expand Up @@ -56,15 +56,14 @@ export class Risk {
* @return {Promise}
*/
static preflight ({ recurly, number, month, year, cvv }) {
function resolveRoute (recurly) {
let route = '/risk/preflights';
if (recurly.config.risk.threeDSecure.proactive.enabled) {
route += `?proactive=true&gatewayCode=${recurly.config.risk.threeDSecure.proactive.gateway_code}`;
}
return route;
const data = {};

if (recurly.config.risk.threeDSecure.proactive.enabled) {
data.proactive = true;
data.gateway_code = recurly.config.risk.threeDSecure.proactive.gatewayCode;
}

return recurly.request.get({ route: resolveRoute(recurly) })
return recurly.request.get({ route: '/risk/preflights', data })
.then(({ preflights }) => {
debug('received preflight instructions', preflights);
return ThreeDSecure.preflight({ recurly, number, month, year, cvv, preflights });
Expand Down
28 changes: 14 additions & 14 deletions lib/recurly/risk/three-d-secure/strategy/braintree.js
Original file line number Diff line number Diff line change
Expand Up @@ -11,31 +11,31 @@ export default class BraintreeStrategy extends ThreeDSecureStrategy {
}

static preflight ({ recurly, number, month, year, cvv }) {
const { proactive } = recurly.config.risk.threeDSecure;
const { enabled, gatewayCode } = recurly.config.risk.threeDSecure.proactive;

if(!proactive.enabled) {
debug('performing preflight for', { gatewayCode });

if (!enabled) {
return Promise.resolve();
}

const data = {
gatewayType: BraintreeStrategy.strategyName,
gatewayCode: proactive.gateway_code,
gateway_type: BraintreeStrategy.strategyName,
gateway_code: gatewayCode,
number,
month,
year,
cvv,
cvv
};

// we don't really need to do anything once we get a response except resolve with relevant data instead of session_id
// we don't really need to do anything once we get a response except
// resolve with relevant data instead of session_id
return recurly.request.post({ route: '/risk/authentications', data })
.then(({ paymentMethodNonce, clientToken, bin }) => (
{
payment_method_nonce: paymentMethodNonce,
client_token: clientToken,
bin,
proactive: true
}
));
.then(({ paymentMethodNonce, clientToken, bin }) => ({
payment_method_nonce: paymentMethodNonce,
client_token: clientToken,
bin,
}));
}


Expand Down
52 changes: 19 additions & 33 deletions lib/recurly/risk/three-d-secure/three-d-secure.js
Original file line number Diff line number Diff line change
Expand Up @@ -71,6 +71,11 @@ export class ThreeDSecure extends RiskConcern {
'05': { height: '100%', width: '100%' }
}

static VALID_ACTION_TOKEN_TYPES = [
'three_d_secure_action',
'three_d_secure_proactive_action'
];

/**
* Returns a strateggy for a given gateway type
*
Expand Down Expand Up @@ -108,7 +113,7 @@ export class ThreeDSecure extends RiskConcern {
}, Promise.resolve([]));
}

constructor ({ risk, actionTokenId, challengeWindowSize, proactiveTokenId }) {
constructor ({ risk, actionTokenId, challengeWindowSize }) {
const existingConcern = risk.concerns.find((concern) => concern instanceof ThreeDSecure);
if (existingConcern) {
throw errors('3ds-multiple-instances', { name: 'ThreeDSecure', expect: 'to be the only concern' });
Expand All @@ -117,28 +122,25 @@ export class ThreeDSecure extends RiskConcern {
super({ risk });

this.actionTokenId = actionTokenId;
this.proactiveTokenId = proactiveTokenId;

this.validateChallengeWindowSize(challengeWindowSize);
this.challengeWindowSize = challengeWindowSize || this.constructor.CHALLENGE_WINDOW_SIZE_DEFAULT;

if (!actionTokenId && !proactiveTokenId) {
if (!actionTokenId) {
throw errors('invalid-option', { name: 'actionTokenId', expect: 'a three_d_secure_action_token_id' });
}

this.recurly.request.get({ route: `/tokens/${this.resolveToken()}` })
this.recurly.request.get({ route: `/tokens/${actionTokenId}` })
.catch(err => this.error(err))
.then(token => {
if (this.resolveToken() == this.actionTokenId) {
this.resolveActionToken(token);
} else {
this.resolveProactiveToken(token);
}
assertIsActionToken(token);
this.strategy = this.getStrategyForActionToken(token);
this.strategy.on('done', (...args) => this.onStrategyDone(...args));
this.markReady();
})
.catch(err => this.error(err));

this.report('create', { actionTokenId, proactiveTokenId });
this.report('create', { actionTokenId });
this.whenReady(() => this.report('ready', { strategy: this.strategy.strategyName }));
}

Expand Down Expand Up @@ -172,11 +174,6 @@ export class ThreeDSecure extends RiskConcern {
return new strategy({ threeDSecure: this, actionToken });
}

getStrategyForProactiveToken (token) {
const strategy = ThreeDSecure.getStrategyForGatewayType(token.three_d_secure.gateway.type);
return new strategy({ threeDSecure: this, proactiveToken: token });
}

/**
* Creates a ThreeDSecureActionResultToken from action results
*
Expand All @@ -189,9 +186,9 @@ export class ThreeDSecure extends RiskConcern {
const data = {
type: 'three_d_secure_action_result',
three_d_secure_action_token_id: this.actionTokenId,
proactive_three_d_secure_token_id: this.proactiveTokenId,
results
};

debug('submitting results for tokenization', data);
return this.recurly.request.post({ route: '/tokens', data });
}
Expand Down Expand Up @@ -225,24 +222,13 @@ export class ThreeDSecure extends RiskConcern {
throw new Error(`Invalid challengeWindowSize. Expected any of ${validWindowSizes}, got ${challengeWindowSize}`);
}
}

resolveToken () {
return this.actionTokenId || this.proactiveTokenId;
}

resolveActionToken (token) {
assertIsActionToken(token);
this.strategy = this.getStrategyForActionToken(token);
this.strategy.on('done', (...args) => this.onStrategyDone(...args));
}

resolveProactiveToken (token) {
this.strategy = this.getStrategyForProactiveToken(token);
this.strategy.on('done', (...args) => this.onStrategyDone(...args));
}
}

function assertIsActionToken (token) {
if (token && token.type === 'three_d_secure_action') return;
throw errors('invalid-option', { name: 'actionTokenId', expect: 'a three_d_secure_action_token_id' });
if (VALID_ACTION_TOKEN_TYPES.includes(token?.type)) return;

throw errors('invalid-option', {
name: 'actionTokenId',
expect: `a token of type: ${VALID_ACTION_TOKEN_TYPES.join(',')}`
});
}
19 changes: 1 addition & 18 deletions lib/recurly/token.js
Original file line number Diff line number Diff line change
Expand Up @@ -174,9 +174,7 @@ function token (customerData, bus, done) {

const { number, month, year, cvv } = inputs;
Risk.preflight({ recurly: this, number, month, year, cvv })
.then(results => {
enrichInputs(this, inputs, results);
})
.then(results => inputs.risk = results)
.then(() => this.request.post({ route: '/token', data: inputs, done: complete }))
.done();
}
Expand All @@ -188,19 +186,4 @@ function token (customerData, bus, done) {
}
done(null, res);
}

function enrichInputs (recurly, inputs, results) {
if (results.length === 0) return;

inputs.risk = [];

results.forEach(result => {
if (result.processor === 'braintree_blue') {
inputs.proactive = recurly.config.risk.threeDSecure.proactive;
inputs.proactive.params = result;
} else {
inputs.risk.push(result);
}
});
}
}
1 change: 1 addition & 0 deletions types/lib/configure.d.ts
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,7 @@ export type RecurlyOptions = {
preflightDeviceDataCollector?: boolean;
proactive?: {
enabled: true;
gatewayCode: string;
}
}
};
Expand Down

0 comments on commit 7973024

Please sign in to comment.