const {flowRight} = require('lodash'); const {mapKeyValues, mapQuery} = require('@nexes/mongo-utils'); const DomainEvents = require('@tryghost/domain-events'); const OfferCodeChangeEvent = require('../domain/events/OfferCodeChange'); const Offer = require('../domain/models/Offer'); const OfferStatus = require('../domain/models/OfferStatus'); const statusTransformer = mapKeyValues({ key: { from: 'status', to: 'active' }, values: [{ from: 'active', to: true }, { from: 'archived', to: false }] }); const rejectNonStatusTransformer = input => mapQuery(input, function (value, key) { if (key !== 'status') { return; } return { [key]: value }; }); const mongoTransformer = flowRight(statusTransformer, rejectNonStatusTransformer); /** * @typedef {object} BaseOptions * @prop {import('knex').Transaction} transacting */ /** * @typedef {object} ListOptions * @prop {import('knex').Transaction} transacting * @prop {string} filter */ class OfferRepository { /** * @param {{forge: (data: object) => import('bookshelf').Model}} OfferModel * @param {{forge: (data: object) => import('bookshelf').Model}} OfferRedemptionModel * @param {import('@tryghost/members-stripe-service')} stripeAPIService */ constructor(OfferModel, OfferRedemptionModel, stripeAPIService) { /** @private */ this.OfferModel = OfferModel; /** @private */ this.OfferRedemptionModel = OfferRedemptionModel; /** @private */ this.stripeAPIService = stripeAPIService; } /** * @template T * @param {(t: import('knex').Transaction) => Promise} cb * @returns {Promise} */ async createTransaction(cb) { return this.OfferModel.transaction(cb); } /** * @param {string} name * @param {BaseOptions} [options] * @returns {Promise} */ async existsByName(name, options) { const model = await this.OfferModel.findOne({name}, options); if (!model) { return false; } return true; } /** * @param {string} code * @param {BaseOptions} [options] * @returns {Promise} */ async existsByCode(code, options) { const model = await this.OfferModel.findOne({code}, options); if (!model) { return false; } return true; } /** * @private * @param {import('bookshelf').Model} model * @param {BaseOptions} options * @returns {Promise} */ async mapToOffer(model, options) { const json = model.toJSON(); const count = await this.OfferRedemptionModel.forge({offer_id: json.id}).count('id', { transacting: options.transacting }); return Offer.create({ id: json.id, name: json.name, code: json.code, display_title: json.portal_title, display_description: json.portal_description, type: json.discount_type === 'amount' ? 'fixed' : 'percent', amount: json.discount_amount, cadence: json.interval, currency: json.currency, duration: json.duration, duration_in_months: json.duration_in_months, redemptionCount: count, stripe_coupon_id: json.stripe_coupon_id, status: json.active ? 'active' : 'archived', tier: { id: json.product.id, name: json.product.name } }, null); } /** * @param {string} id * @param {BaseOptions} [options] * @returns {Promise} */ async getById(id, options) { const model = await this.OfferModel.findOne({id}, { ...options, withRelated: ['product'] }); return this.mapToOffer(model, options); } /** * @param {ListOptions} options * @returns {Promise} */ async getAll(options) { const models = await this.OfferModel.findAll({ ...options, mongoTransformer, withRelated: ['product'] }); const mapOptions = { transacting: options && options.transacting }; const offers = models.map(model => this.mapToOffer(model, mapOptions)); return Promise.all(offers); } /** * @param {Offer} offer * @param {BaseOptions} [options] * @returns {Promise} */ async save(offer, options) { /** @type any */ const data = { id: offer.id, name: offer.name.value, code: offer.code.value, portal_title: offer.displayTitle.value, portal_description: offer.displayDescription.value, discount_type: offer.type.value === 'fixed' ? 'amount' : 'percent', discount_amount: offer.amount.value, interval: offer.cadence.value, product_id: offer.tier.id, duration: offer.duration.value.type, duration_in_months: offer.duration.value.type === 'repeating' ? offer.duration.value.months : null, currency: offer.currency ? offer.currency.value : null, active: offer.status.equals(OfferStatus.create('active')) }; if (offer.codeChanged || offer.isNew) { const event = OfferCodeChangeEvent.create({ offerId: offer.id, previousCode: offer.oldCode, currentCode: offer.code }); DomainEvents.dispatch(event); } if (offer.isNew) { /** @type {import('stripe').Stripe.CouponCreateParams} */ const coupon = { name: offer.name.value, duration: offer.duration.value.type }; if (offer.duration.value.type === 'repeating') { coupon.duration_in_months = offer.duration.value.months; } if (offer.type.value === 'percent') { coupon.percent_off = offer.amount.value; } else { coupon.amount_off = offer.amount.value; coupon.currency = offer.currency.value; } const couponData = await this.stripeAPIService.createCoupon(coupon); data.stripe_coupon_id = couponData.id; await this.OfferModel.add(data, options); } else { await this.OfferModel.edit(data, {...options, id: data.id}); } } } module.exports = OfferRepository;