mirror of
https://github.com/fleetdm/fleet
synced 2026-05-23 08:58:41 +00:00
150 lines
5.2 KiB
JavaScript
Vendored
150 lines
5.2 KiB
JavaScript
Vendored
module.exports = {
|
|
|
|
|
|
friendlyName: 'Save billing info and subscribe',
|
|
|
|
|
|
description: '',
|
|
|
|
|
|
inputs: {
|
|
|
|
quoteId: {
|
|
type: 'number',
|
|
required: true,
|
|
description: 'The quote to use (determines the price and number of hosts.)'
|
|
},
|
|
|
|
paymentSource: {
|
|
required: true,
|
|
description: 'New payment source info to use (instead of the saved default payment source).',
|
|
extendedDescription: 'If provided, this will also be saved as the new default payment source for the customer, replacing the existing default payment source (if there is one.)',
|
|
type: {
|
|
stripeToken: 'string',
|
|
billingCardLast4: 'string',
|
|
billingCardBrand: 'string',
|
|
billingCardExpMonth: 'string',
|
|
billingCardExpYear: 'string',
|
|
},
|
|
example: {
|
|
stripeToken: 'tok_199k3qEXw14QdSnRwmsK99MH',
|
|
billingCardLast4: '4242',
|
|
billingCardBrand: 'visa',
|
|
billingCardExpMonth: '08',
|
|
billingCardExpYear: '2023',
|
|
}
|
|
},
|
|
|
|
},
|
|
|
|
|
|
exits: {
|
|
|
|
couldNotSaveBillingInfo: {
|
|
description: 'The billing information provided could not be saved.',
|
|
responseType: 'badRequest'
|
|
},
|
|
|
|
},
|
|
|
|
|
|
fn: async function (inputs) {
|
|
const stripe = require('stripe')(sails.config.custom.stripeSecret);
|
|
|
|
let quoteRecord = await Quote.findOne({id: inputs.quoteId});
|
|
if(!quoteRecord) {
|
|
throw new Error(`Consistency violation: The specified quote (${inputs.quoteId}) no longer seems to exist.`);
|
|
}
|
|
|
|
// If this user has a subscription, we'll throw an error.
|
|
let doesUserHaveAnExistingSubscription = await Subscription.findOne({user: this.req.me.id});
|
|
if(doesUserHaveAnExistingSubscription) {
|
|
throw new Error(`Consistency violation: The requesting user (${this.req.me.emailAddress}) already has an existing subscription!`);
|
|
}
|
|
|
|
// What if the stripe customer id doesn't already exist on the user?
|
|
if (!this.req.me.stripeCustomerId) {
|
|
throw new Error(`Consistency violation: The logged-in user's (${this.req.me.emailAddress}) Stripe customer id has somehow gone missing!`);
|
|
}
|
|
|
|
// Write new payment card info ("token") to Stripe's API.
|
|
await sails.helpers.stripe.saveBillingInfo.with({
|
|
stripeCustomerId: this.req.me.stripeCustomerId,
|
|
token: inputs.paymentSource.stripeToken
|
|
})
|
|
.intercept({ type: 'StripeCardError' }, 'couldNotSaveBillingInfo');
|
|
|
|
// Save payment card info to our database.
|
|
await User.updateOne({ id: this.req.me.id })
|
|
.set({
|
|
hasBillingCard: true,
|
|
billingCardBrand: inputs.paymentSource.billingCardBrand,
|
|
billingCardLast4: inputs.paymentSource.billingCardLast4,
|
|
billingCardExpMonth: inputs.paymentSource.billingCardExpMonth,
|
|
billingCardExpYear: inputs.paymentSource.billingCardExpYear,
|
|
});
|
|
|
|
// Create the subscription for this order in Stripe
|
|
const subscription = await stripe.subscriptions.create({
|
|
customer: this.req.me.stripeCustomerId,
|
|
items: [
|
|
{
|
|
price: sails.config.custom.stripeSubscriptionPriceId,
|
|
quantity: quoteRecord.numberOfHosts,
|
|
},
|
|
],
|
|
});
|
|
|
|
// Generate the license key for this subscription
|
|
let licenseKey = await sails.helpers.createLicenseKey.with({
|
|
numberOfHosts: quoteRecord.numberOfHosts,
|
|
organization: this.req.me.organization ? this.req.me.organization : this.req.me.emailAddress,
|
|
validTo: subscription.current_period_end
|
|
});
|
|
|
|
// Create the subscription record for this order.
|
|
await Subscription.create({
|
|
organization: this.req.me.organization,
|
|
numberOfHosts: quoteRecord.numberOfHosts,
|
|
subscriptionPrice: quoteRecord.quotedPrice,
|
|
user: this.req.me.id,
|
|
stripeSubscriptionId: subscription.id,
|
|
nextBillingAt: subscription.current_period_end * 1000,
|
|
fleetLicenseKey: licenseKey,
|
|
});
|
|
|
|
// Send a POST request to Zapier
|
|
await sails.helpers.http.post(
|
|
'https://hooks.zapier.com/hooks/catch/3627242/blhrvf1/',
|
|
{
|
|
'emailAddress': this.req.me.emailAddress,
|
|
'numberOfHosts': quoteRecord.numberOfHosts,
|
|
'subscriptionPrice': quoteRecord.quotedPrice,
|
|
'nextBillingTimestamp': new Date(subscription.current_period_end * 1000).toISOString(),
|
|
'webhookSecret': sails.config.custom.zapierSandboxWebhookSecret
|
|
}
|
|
)
|
|
.timeout(5000)
|
|
.tolerate(['non200Response', 'requestFailed'], (err)=>{
|
|
// Note that Zapier responds with a 2xx status code even if something goes wrong, so just because this message is not logged doesn't mean everything is hunky dory. More info: https://github.com/fleetdm/fleet/pull/6380#issuecomment-1204395762
|
|
sails.log.warn(`When a user purchased a Fleet Premium license, a lead/contact could not be updated in the CRM for this email address: ${this.req.me.emailAddress}. Raw error: ${err}`);
|
|
return;
|
|
});
|
|
|
|
// Send the order confirmation template email
|
|
await sails.helpers.sendTemplateEmail.with({
|
|
to: this.req.me.emailAddress,
|
|
from: sails.config.custom.fromEmail,
|
|
fromName: sails.config.custom.fromName,
|
|
subject: 'Your Fleet Premium order',
|
|
template: 'email-order-confirmation',
|
|
templateData: {
|
|
firstName: this.req.me.firstName,
|
|
lastName: this.req.me.lastName,
|
|
}
|
|
});
|
|
|
|
}
|
|
|
|
|
|
};
|