From 449282dda0751906ae3130b7546428c6c545a5f8 Mon Sep 17 00:00:00 2001 From: Benjamin Piouffle Date: Thu, 29 Feb 2024 12:13:00 +0100 Subject: [PATCH] feat(Contributions): ability to pause non-transferable --- cron/daily/60-clean-orders.js | 8 ++ ...ncel-subscriptions-for-cancelled-orders.ts | 40 ++++-- server/constants/activities.ts | 2 + server/constants/order-status.ts | 1 + server/graphql/loaders/index.js | 123 +++++++++++------- server/graphql/schemaV1.graphql | 1 + server/graphql/schemaV2.graphql | 33 +++++ server/graphql/v1/CollectiveInterface.js | 5 +- .../v2/mutation/HostApplicationMutations.ts | 34 +++-- server/graphql/v2/object/AccountStats.js | 65 +++++++-- server/graphql/v2/object/AmountStats.js | 3 +- server/lib/emailTemplates.ts | 1 + server/lib/notifications/email.ts | 1 + server/lib/recurring-contributions.js | 11 +- server/lib/timeline.ts | 1 + server/lib/webhooks.js | 2 +- server/models/Collective.ts | 8 +- server/models/Order.ts | 92 ++++++++++--- templates/emails/collective.unhosted.hbs | 5 +- templates/emails/subscription.paused.hbs | 24 ++++ .../mutation/HostApplicationMutations.test.ts | 97 +++++++++++++- .../lib/recurring-contributions.test.js | 4 +- test/server/models/Order.test.js | 95 +++++++++++++- 23 files changed, 539 insertions(+), 117 deletions(-) create mode 100644 templates/emails/subscription.paused.hbs diff --git a/cron/daily/60-clean-orders.js b/cron/daily/60-clean-orders.js index ed4f2472d932..f341f94d4c5e 100644 --- a/cron/daily/60-clean-orders.js +++ b/cron/daily/60-clean-orders.js @@ -20,6 +20,14 @@ Promise.all([ AND "Orders"."createdAt" < (NOW() - interval '2 month') `, ), + // Mark all paused Orders as EXPIRED after 1 year + sequelize.query( + `UPDATE "Orders" + SET "status" = 'EXPIRED', "updatedAt" = NOW() + WHERE "Orders"."status" = 'PAUSED' + AND "Orders"."updatedAt" < (NOW() - interval '1 year') + `, + ), ]).then(() => { console.log('>>> Clean Orders: done'); process.exit(0); diff --git a/cron/hourly/70-cancel-subscriptions-for-cancelled-orders.ts b/cron/hourly/70-cancel-subscriptions-for-cancelled-orders.ts index c6e6ad4e4d24..3b90e3cc3c92 100644 --- a/cron/hourly/70-cancel-subscriptions-for-cancelled-orders.ts +++ b/cron/hourly/70-cancel-subscriptions-for-cancelled-orders.ts @@ -10,7 +10,7 @@ import OrderStatuses from '../../server/constants/order-status'; import logger from '../../server/lib/logger'; import { reportErrorToSentry } from '../../server/lib/sentry'; import { sleep } from '../../server/lib/utils'; -import models, { Op } from '../../server/models'; +import models, { Collective, Op } from '../../server/models'; import { OrderModelInterface } from '../../server/models/Order'; /** @@ -32,17 +32,29 @@ const getHostFromOrder = async order => { return models.Collective.findByPk(hostIds[0]); }; -const getOrderCancelationReason = (collective, order, orderHost) => { - if (order.TierId && !order.Tier) { - return ['DELETED_TIER', `Order tier deleted`]; +const getOrderCancelationReason = ( + collective: Collective, + order: OrderModelInterface, + orderHost: Collective, +): { + code: 'TRANSFER' | 'DELETED_TIER' | 'ARCHIVED_ACCOUNT' | 'UNHOSTED_COLLECTIVE' | 'CHANGED_HOST' | 'CANCELLED_ORDER'; + message: string; +} => { + if (order.status === 'PAUSED' && order.data?.pausedForTransfer) { + return { + code: 'TRANSFER', + message: `Your contribution to the Collective was paused. We'll inform you when it will be ready for re-activation.`, + }; + } else if (order.TierId && !order.Tier) { + return { code: 'DELETED_TIER', message: `Order tier deleted` }; } else if (collective.deactivatedAt) { - return ['ARCHIVED_ACCOUNT', `@${collective.slug} archived their account`]; + return { code: 'ARCHIVED_ACCOUNT', message: `@${collective.slug} archived their account` }; } else if (!collective.HostCollectiveId) { - return ['UNHOSTED_COLLECTIVE', `@${collective.slug} was un-hosted`]; + return { code: 'UNHOSTED_COLLECTIVE', message: `@${collective.slug} was un-hosted` }; } else if (collective.HostCollectiveId !== orderHost.id) { - return ['CHANGED_HOST', `@${collective.slug} changed host`]; + return { code: 'CHANGED_HOST', message: `@${collective.slug} changed host` }; } else { - return ['CANCELLED_ORDER', `Order cancelled`]; + return { code: 'CANCELLED_ORDER', message: `Order cancelled` }; } }; @@ -53,7 +65,7 @@ const getOrderCancelationReason = (collective, order, orderHost) => { */ export async function run() { const orphanOrders = await models.Order.findAll({ - where: { status: OrderStatuses.CANCELLED }, + where: { status: [OrderStatuses.CANCELLED, OrderStatuses.PAUSED] }, include: [ { model: models.Tier, @@ -85,14 +97,14 @@ export async function run() { for (const order of accountOrders) { try { const host = await getHostFromOrder(order); - const [reasonCode, reason] = getOrderCancelationReason(collective, order, host); + const reason = getOrderCancelationReason(collective, order, host); logger.debug( `Cancelling subscription ${order.Subscription.id} from order ${order.id} of @${collectiveHandle} (host: ${host.slug})`, ); if (!process.env.DRY) { - await order.Subscription.deactivate(reason, host); + await order.Subscription.deactivate(reason.message, host); await models.Activity.create({ - type: activities.SUBSCRIPTION_CANCELED, + type: reason.code === 'TRANSFER' ? activities.SUBSCRIPTION_PAUSED : activities.SUBSCRIPTION_CANCELED, CollectiveId: order.CollectiveId, FromCollectiveId: order.FromCollectiveId, HostCollectiveId: order.collective.HostCollectiveId, @@ -102,8 +114,8 @@ export async function run() { subscription: order.Subscription, collective: order.collective.minimal, fromCollective: order.fromCollective.minimal, - reasonCode: reasonCode, - reason: reason, + reasonCode: reason.code, + reason: reason.message, order: order.info, tier: order.Tier?.info, }, diff --git a/server/constants/activities.ts b/server/constants/activities.ts index e4a25ff314fc..50efddbc51f7 100644 --- a/server/constants/activities.ts +++ b/server/constants/activities.ts @@ -83,6 +83,7 @@ enum ActivityTypes { CONTRIBUTION_REJECTED = 'contribution.rejected', SUBSCRIPTION_ACTIVATED = 'subscription.activated', SUBSCRIPTION_CANCELED = 'subscription.canceled', + SUBSCRIPTION_PAUSED = 'subscription.paused', TICKET_CONFIRMED = 'ticket.confirmed', ORDER_CANCELED_ARCHIVED_COLLECTIVE = 'order.canceled.archived.collective', ORDER_PENDING = 'order.pending', @@ -239,6 +240,7 @@ export const ActivitiesPerClass: Record = { ActivityTypes.PAYMENT_FAILED, ActivityTypes.SUBSCRIPTION_ACTIVATED, ActivityTypes.SUBSCRIPTION_CANCELED, + ActivityTypes.SUBSCRIPTION_PAUSED, ActivityTypes.SUBSCRIPTION_CONFIRMED, ], [ActivityClasses.ACTIVITIES_UPDATES]: [ diff --git a/server/constants/order-status.ts b/server/constants/order-status.ts index 59b81e074f99..79b458dd4f7b 100644 --- a/server/constants/order-status.ts +++ b/server/constants/order-status.ts @@ -24,6 +24,7 @@ enum OrderStatuses { // Disputed charges from Stripe DISPUTED = 'DISPUTED', REFUNDED = 'REFUNDED', + PAUSED = 'PAUSED', // In review charges from Stripe, IN_REVIEW = 'IN_REVIEW', } diff --git a/server/graphql/loaders/index.js b/server/graphql/loaders/index.js index 6fda0be9c9da..18d70e2e53ce 100644 --- a/server/graphql/loaders/index.js +++ b/server/graphql/loaders/index.js @@ -1,6 +1,6 @@ import DataLoader from 'dataloader'; import { createContext } from 'dataloader-sequelize'; -import { get, groupBy } from 'lodash'; +import { get, groupBy, isNil } from 'lodash'; import moment from 'moment'; import { CollectiveType } from '../../constants/collectives'; @@ -372,54 +372,79 @@ export const loaders = req => { }) .then(results => sortResults(ids, results, 'CollectiveId')), ), - activeRecurringContributions: new DataLoader(ids => - models.Order.findAll({ - attributes: [ - 'Order.CollectiveId', - 'Order.currency', - 'Subscription.interval', - [ - sequelize.fn( - 'SUM', - sequelize.literal(`COALESCE("Order"."totalAmount", 0) - COALESCE("Order"."platformTipAmount", 0)`), - ), - 'total', - ], - ], - where: { - CollectiveId: { [Op.in]: ids }, - status: 'ACTIVE', - }, - group: ['Subscription.interval', 'CollectiveId', 'Order.currency'], - include: [ - { - model: models.Subscription, - attributes: [], - where: { isActive: true }, - }, - ], - raw: true, - }).then(rows => { - const results = groupBy(rows, 'CollectiveId'); - return Promise.all( - ids.map(async collectiveId => { - const stats = { CollectiveId: Number(collectiveId), monthly: 0, yearly: 0, currency: null }; - if (results[collectiveId]) { - for (const result of results[collectiveId]) { - const interval = result.interval === 'month' ? 'monthly' : 'yearly'; - // If it's the first total collected, set the currency - if (!stats.currency) { - stats.currency = result.currency; - } - const fxRate = await getFxRate(result.currency, stats.currency); - stats[interval] += result.total * fxRate; - } - } - return stats; - }), - ); - }), - ), + activeRecurringContributions: { + buildLoader({ currency, hasPortability = undefined, includeChildren = undefined } = {}) { + const key = `${currency}-${hasPortability}-${includeChildren}`; + if (!context.loaders.Collective.stats.activeRecurringContributions[key]) { + const collectiveIdCol = !includeChildren + ? 'CollectiveId' + : sequelize.fn('COALESCE', sequelize.col('collective.ParentCollectiveId'), sequelize.col('collective.id')); + + context.loaders.Collective.stats.activeRecurringContributions[key] = new DataLoader(ids => + models.Order.findAll({ + attributes: [ + [collectiveIdCol, 'CollectiveId'], + 'Order.currency', + 'Subscription.interval', + [sequelize.fn('COUNT', sequelize.literal(`DISTINCT "Order"."id"`)), 'count'], + [ + sequelize.fn( + 'SUM', + sequelize.literal(`COALESCE("Order"."totalAmount", 0) - COALESCE("Order"."platformTipAmount", 0)`), + ), + 'total', + ], + ], + where: { + status: 'ACTIVE', + }, + group: ['Subscription.interval', collectiveIdCol, 'Order.currency'], + include: [ + { + association: 'collective', + attributes: [], + required: true, + where: !includeChildren ? { id: ids } : { [Op.or]: [{ id: ids }, { ParentCollectiveId: ids }] }, + }, + { + model: models.Subscription, + attributes: [], + required: true, + where: { + isActive: true, + ...(!isNil(hasPortability) && { isManagedExternally: !hasPortability }), + }, + }, + ], + raw: true, + }).then(rows => { + const results = groupBy(rows, 'CollectiveId'); + return Promise.all( + ids.map(async collectiveId => { + const stats = { + CollectiveId: Number(collectiveId), + monthly: 0, + monthlyCount: 0, + yearly: 0, + yearlyCount: 0, + }; + if (results[collectiveId]) { + for (const result of results[collectiveId]) { + const interval = result.interval === 'month' ? 'monthly' : 'yearly'; + const fxRate = await getFxRate(result.currency, currency); + stats[interval] += result.total * fxRate; + stats[`${interval}Count`] += result.count; + } + } + return stats; + }), + ); + }), + ); + } + return context.loaders.Collective.stats.activeRecurringContributions[key]; + }, + }, orders: new DataLoader(async collectiveIds => { const stats = await sequelize.query( `SELECT * FROM "CollectiveOrderStats" WHERE "CollectiveId" IN (:collectiveIds)`, diff --git a/server/graphql/schemaV1.graphql b/server/graphql/schemaV1.graphql index 1f5445e7dd3a..87620f88d068 100644 --- a/server/graphql/schemaV1.graphql +++ b/server/graphql/schemaV1.graphql @@ -733,6 +733,7 @@ enum OrderStatus { EXPIRED DISPUTED REFUNDED + PAUSED IN_REVIEW } diff --git a/server/graphql/schemaV2.graphql b/server/graphql/schemaV2.graphql index e55f30b83610..aa23fc7a7492 100644 --- a/server/graphql/schemaV2.graphql +++ b/server/graphql/schemaV2.graphql @@ -1194,6 +1194,7 @@ enum OrderStatus { EXPIRED DISPUTED REFUNDED + PAUSED IN_REVIEW } @@ -4616,6 +4617,27 @@ type AccountStats { """ frequency: ContributionFrequency! = MONTHLY ): Amount + @deprecated(reason: "2024-03-04: Use activeRecurringContributionsBreakdown while we migrate to better semantics.") + + """ + Returns some statistics about active recurring contributions, broken down by frequency + """ + activeRecurringContributionsBreakdown( + """ + Return only the stats for this frequency + """ + frequency: ContributionFrequency + + """ + Filter contributions on whether they can be ported to another fiscal host directly + """ + hasPortability: Boolean + + """ + Include contributions to children accounts (Projects and Events) + """ + includeChildren: Boolean = false + ): [AmountStats!]! """ Returns expense tags for collective sorted by popularity @@ -5310,6 +5332,7 @@ enum ActivityType { CONTRIBUTION_REJECTED SUBSCRIPTION_ACTIVATED SUBSCRIPTION_CANCELED + SUBSCRIPTION_PAUSED TICKET_CONFIRMED ORDER_CANCELED_ARCHIVED_COLLECTIVE ORDER_PENDING @@ -13850,6 +13873,7 @@ enum ActivityAndClassesType { CONTRIBUTION_REJECTED SUBSCRIPTION_ACTIVATED SUBSCRIPTION_CANCELED + SUBSCRIPTION_PAUSED TICKET_CONFIRMED ORDER_CANCELED_ARCHIVED_COLLECTIVE ORDER_PENDING @@ -16215,7 +16239,16 @@ type Mutation { The account to unhost """ account: AccountReferenceInput! + + """ + An optional message to explain the reason for unhosting + """ message: String + + """ + If true, contributions will be paused rather than canceled + """ + pauseContributions: Boolean! = true ): Account! """ diff --git a/server/graphql/v1/CollectiveInterface.js b/server/graphql/v1/CollectiveInterface.js index f14b0ce414b4..82ee8a724002 100644 --- a/server/graphql/v1/CollectiveInterface.js +++ b/server/graphql/v1/CollectiveInterface.js @@ -522,7 +522,10 @@ export const CollectiveStatsType = new GraphQLObjectType({ activeRecurringContributions: { type: GraphQLJSON, resolve(collective, args, req) { - return req.loaders.Collective.stats.activeRecurringContributions.load(collective.id); + const loader = req.loaders.Collective.stats.activeRecurringContributions.buildLoader({ + currency: collective.currency, + }); + return loader.load(collective.id); }, }, }; diff --git a/server/graphql/v2/mutation/HostApplicationMutations.ts b/server/graphql/v2/mutation/HostApplicationMutations.ts index 38dabee2baa9..10b605933ac8 100644 --- a/server/graphql/v2/mutation/HostApplicationMutations.ts +++ b/server/graphql/v2/mutation/HostApplicationMutations.ts @@ -1,6 +1,6 @@ import config from 'config'; import express from 'express'; -import { GraphQLList, GraphQLNonNull, GraphQLObjectType, GraphQLString } from 'graphql'; +import { GraphQLBoolean, GraphQLList, GraphQLNonNull, GraphQLObjectType, GraphQLString } from 'graphql'; import { GraphQLJSON } from 'graphql-scalars'; import { activities } from '../../../constants'; @@ -12,7 +12,7 @@ import { purgeAllCachesForAccount, purgeCacheForCollective } from '../../../lib/ import emailLib from '../../../lib/email'; import * as github from '../../../lib/github'; import { OSCValidator, ValidatedRepositoryInfo } from '../../../lib/osc-validator'; -import { getPolicy, hasPolicy } from '../../../lib/policies'; +import { getPolicy } from '../../../lib/policies'; import { stripHTML } from '../../../lib/sanitize-html'; import twoFactorAuthLib from '../../../lib/two-factor-authentication'; import models, { Collective, Op, sequelize } from '../../../models'; @@ -20,7 +20,7 @@ import ConversationModel from '../../../models/Conversation'; import { HostApplicationStatus } from '../../../models/HostApplication'; import { processInviteMembersInput } from '../../common/members'; import { checkRemoteUserCanUseAccount, checkRemoteUserCanUseHost, checkScope } from '../../common/scope-check'; -import { Forbidden, NotFound, Unauthorized, ValidationFailed } from '../../errors'; +import { Forbidden, NotFound, ValidationFailed } from '../../errors'; import { GraphQLProcessHostApplicationAction } from '../enum/ProcessHostApplicationAction'; import { fetchAccountWithReference, GraphQLAccountReferenceInput } from '../input/AccountReferenceInput'; import { GraphQLInviteMemberInput } from '../input/InviteMemberInput'; @@ -224,6 +224,12 @@ const HostApplicationMutations = { }, message: { type: GraphQLString, + description: 'An optional message to explain the reason for unhosting', + }, + pauseContributions: { + type: new GraphQLNonNull(GraphQLBoolean), + description: 'If true, contributions will be paused rather than canceled', + defaultValue: true, }, }, resolve: async (_, args, req: express.Request): Promise => { @@ -238,19 +244,19 @@ const HostApplicationMutations = { if (!host) { return account; } - if (!req.remoteUser.isAdminOfCollective(host) && !(req.remoteUser.isRoot() && checkScope(req, 'root'))) { - throw new Unauthorized(); - } - if (await hasPolicy(host, POLICIES.REQUIRE_2FA_FOR_ADMINS)) { - await twoFactorAuthLib.validateRequest(req, { - alwaysAskForToken: true, - requireTwoFactorAuthEnabled: true, - FromCollectiveId: host.id, - }); + const isHostAdmin = req.remoteUser.isAdminOfCollective(host); + const isAccountAdmin = req.remoteUser.isAdminOfCollective(account); + if (!isHostAdmin && !isAccountAdmin && !(req.remoteUser.isRoot() && checkScope(req, 'root'))) { + throw new Forbidden('Only the host admin or the account admin can trigger this action'); } - await account.changeHost(null); + await twoFactorAuthLib.enforceForAccountsUserIsAdminOf(req, [account, host], { alwaysAskForToken: true }); + + await account.changeHost(null, req.remoteUser, { + pauseContributions: args.pauseContributions, + messageForContributors: args.message, + }); await models.Activity.create({ type: activities.COLLECTIVE_UNHOSTED, @@ -262,6 +268,8 @@ const HostApplicationMutations = { collective: account.info, host: host.info, message: args.message, + isHostAdmin, + isAccountAdmin, }, }); diff --git a/server/graphql/v2/object/AccountStats.js b/server/graphql/v2/object/AccountStats.js index 541854e01b12..9021b1112a57 100644 --- a/server/graphql/v2/object/AccountStats.js +++ b/server/graphql/v2/object/AccountStats.js @@ -4,7 +4,6 @@ import { get, has, pick } from 'lodash'; import moment from 'moment'; import { getCollectiveIds } from '../../../lib/budget'; -import { getFxRate } from '../../../lib/currency'; import queries from '../../../lib/queries'; import sequelize, { QueryTypes } from '../../../lib/sequelize'; import { computeDatesAsISOStrings } from '../../../lib/utils'; @@ -359,11 +358,16 @@ export const GraphQLAccountStats = new GraphQLObjectType({ type: GraphQLJSON, deprecationReason: '2022-10-21: Use activeRecurringContributionsV2 while we migrate to better semantics.', resolve(collective, args, req) { - return req.loaders.Collective.stats.activeRecurringContributions.load(collective.id); + const loader = req.loaders.Collective.stats.activeRecurringContributions.buildLoader({ + currency: collective.currency, + }); + return loader.load(collective.id); }, }, activeRecurringContributionsV2: { type: GraphQLAmount, + deprecationReason: + '2024-03-04: Use activeRecurringContributionsBreakdown while we migrate to better semantics.', args: { frequency: { type: new GraphQLNonNull(GraphQLContributionFrequency), @@ -376,17 +380,60 @@ export const GraphQLAccountStats = new GraphQLObjectType({ if (!['monthly', 'yearly'].includes(key)) { throw new Error('Unsupported frequency.'); } - const stats = await req.loaders.Collective.stats.activeRecurringContributions.load(collective.id); - const currency = collective.currency; - // There is no guarantee that stats are returned in collective.currency, we convert to be sure - const fxRate = await getFxRate(stats.currency, currency); - const value = Math.round(stats[key] * fxRate); + const loader = req.loaders.Collective.stats.activeRecurringContributions.buildLoader({ + currency: collective.currency, + }); + const stats = await loader.load(collective.id); return { - value: value, - currency: currency, + value: stats[key], + currency: collective.currency, }; }, }, + activeRecurringContributionsBreakdown: { + description: 'Returns some statistics about active recurring contributions, broken down by frequency', + type: new GraphQLNonNull(new GraphQLList(new GraphQLNonNull(GraphQLAmountStats))), + args: { + frequency: { + type: GraphQLContributionFrequency, + description: 'Return only the stats for this frequency', + }, + hasPortability: { + type: GraphQLBoolean, + description: 'Filter contributions on whether they can be ported to another fiscal host directly', + }, + includeChildren: { + type: GraphQLBoolean, + description: 'Include contributions to children accounts (Projects and Events)', + defaultValue: false, + }, + }, + async resolve(collective, args, req) { + const interval = args.frequency?.toLowerCase(); + if (interval && !['monthly', 'yearly'].includes(interval)) { + throw new Error('Unsupported frequency.'); + } + const currency = collective.currency; + const loader = req.loaders.Collective.stats.activeRecurringContributions.buildLoader({ + hasPortability: args.hasPortability, + includeChildren: args.includeChildren, + currency, + }); + const stats = await loader.load(collective.id); + const getStatsForInterval = interval => ({ + label: interval, + count: stats[`${interval}Count`], + amount: stats[interval], + currency, + }); + + if (interval) { + return [getStatsForInterval(interval)]; + } else { + return ['monthly', 'yearly'].map(getStatsForInterval); + } + }, + }, expensesTags: { type: new GraphQLList(GraphQLAmountStats), description: 'Returns expense tags for collective sorted by popularity', diff --git a/server/graphql/v2/object/AmountStats.js b/server/graphql/v2/object/AmountStats.js index 1cd75ed82342..163a1860dc39 100644 --- a/server/graphql/v2/object/AmountStats.js +++ b/server/graphql/v2/object/AmountStats.js @@ -1,4 +1,5 @@ import { GraphQLInt, GraphQLNonNull, GraphQLObjectType, GraphQLString } from 'graphql'; +import { isNil } from 'lodash'; import { GraphQLAmount } from '../object/Amount'; @@ -14,7 +15,7 @@ export const GraphQLAmountStats = new GraphQLObjectType({ type: new GraphQLNonNull(GraphQLAmount), description: 'Total amount for this label', resolve(entry) { - if (entry.amount) { + if (!isNil(entry.amount)) { return { value: entry.amount, currency: entry.currency }; } }, diff --git a/server/lib/emailTemplates.ts b/server/lib/emailTemplates.ts index 7f43d2fdd002..527438a163e1 100644 --- a/server/lib/emailTemplates.ts +++ b/server/lib/emailTemplates.ts @@ -92,6 +92,7 @@ export const templateNames = [ 'report.platform', 'report.platform.weekly', 'subscription.canceled', + 'subscription.paused', 'taxform.request', 'ticket.confirmed', 'ticket.confirmed.fearlesscitiesbrussels', diff --git a/server/lib/notifications/email.ts b/server/lib/notifications/email.ts index 3ddb7b13bbe6..f9fb436585b8 100644 --- a/server/lib/notifications/email.ts +++ b/server/lib/notifications/email.ts @@ -239,6 +239,7 @@ export const notifyByEmail = async (activity: Activity) => { break; case ActivityTypes.SUBSCRIPTION_CANCELED: + case ActivityTypes.SUBSCRIPTION_PAUSED: await notify.user(activity); break; diff --git a/server/lib/recurring-contributions.js b/server/lib/recurring-contributions.js index a22673f0aafd..b93fc8091515 100644 --- a/server/lib/recurring-contributions.js +++ b/server/lib/recurring-contributions.js @@ -31,13 +31,22 @@ export const MAX_RETRIES = 6; export async function ordersWithPendingCharges({ limit, startDate } = {}) { return models.Order.findAndCountAll({ where: { + status: { [Op.not]: status.PAUSED }, SubscriptionId: { [Op.ne]: null }, deletedAt: null, }, limit: limit, include: [ { model: models.User, as: 'createdByUser' }, - { model: models.Collective, as: 'collective', required: true }, + { + model: models.Collective, + as: 'collective', + required: true, + where: { + HostCollectiveId: { [Op.not]: null }, + isActive: true, + }, + }, { model: models.Collective, as: 'fromCollective', required: true }, { model: models.PaymentMethod, as: 'paymentMethod' }, { model: models.Tier, as: 'Tier' }, diff --git a/server/lib/timeline.ts b/server/lib/timeline.ts index 63a70f77287f..79af3ad2a34a 100644 --- a/server/lib/timeline.ts +++ b/server/lib/timeline.ts @@ -154,6 +154,7 @@ const makeTimelineQuery = async ( ActivityTypes.PAYMENT_CREDITCARD_EXPIRING, ActivityTypes.PAYMENT_FAILED, ActivityTypes.SUBSCRIPTION_CANCELED, + ActivityTypes.SUBSCRIPTION_PAUSED, ], ); } diff --git a/server/lib/webhooks.js b/server/lib/webhooks.js index 884b9b23a3b2..e5d3f145b36b 100644 --- a/server/lib/webhooks.js +++ b/server/lib/webhooks.js @@ -142,7 +142,7 @@ export const sanitizeActivity = activity => { cleanActivity.data = pick(activity.data, ['recipient.name']); cleanActivity.data.tier = getTierInfo(activity.data.tier); cleanActivity.data.order = getOrderInfo(activity.data.order); - } else if (type === activities.SUBSCRIPTION_CANCELED) { + } else if (type === activities.SUBSCRIPTION_CANCELED || type === activities.SUBSCRIPTION_PAUSED) { cleanActivity.data = pick(activity.data, ['subscription.id']); cleanActivity.data.order = getOrderInfo(activity.data.order); cleanActivity.data.tier = getTierInfo(activity.data.tier); diff --git a/server/models/Collective.ts b/server/models/Collective.ts index d2829b8180c2..ee37323de99a 100644 --- a/server/models/Collective.ts +++ b/server/models/Collective.ts @@ -52,6 +52,7 @@ import { SupportedCurrency } from '../constants/currencies'; import expenseStatus from '../constants/expense-status'; import expenseTypes from '../constants/expense-type'; import FEATURE from '../constants/feature'; +import OrderStatuses from '../constants/order-status'; import { PAYMENT_METHOD_SERVICE, PAYMENT_METHOD_TYPE } from '../constants/paymentMethods'; import plans from '../constants/plans'; import POLICIES, { Policies } from '../constants/policies'; @@ -2412,7 +2413,12 @@ class Collective extends Model< }); } - await Order.cancelNonTransferableActiveOrdersByCollectiveId(this.id); + // Pause or cancel all orders that cannot be transferred + await Order.updateNonTransferableActiveOrdersStatusesByCollectiveId( + this.id, + options?.pauseContributions ? OrderStatuses.PAUSED : OrderStatuses.CANCELLED, + 'Changing host', + ); const virtualCards = await VirtualCard.findAll({ where: { CollectiveId: this.id } }); await Promise.all(virtualCards.map(virtualCard => virtualCard.delete())); diff --git a/server/models/Order.ts b/server/models/Order.ts index e9a04ce64703..7f9cf7b156e2 100644 --- a/server/models/Order.ts +++ b/server/models/Order.ts @@ -1,4 +1,5 @@ import { TaxType } from '@opencollective/taxes'; +import config from 'config'; import debugLib from 'debug'; import { get } from 'lodash'; import { @@ -41,7 +42,11 @@ interface OrderModelStaticInterface { generateDescription(collective, amount, interval, tier): string; cancelActiveOrdersByCollective(collectiveId: number): Promise<[affectedCount: number]>; cancelActiveOrdersByTierId(tierId: number): Promise<[affectedCount: number]>; - cancelNonTransferableActiveOrdersByCollectiveId(collectiveId: number): Promise<[affectedCount: number]>; + updateNonTransferableActiveOrdersStatusesByCollectiveId( + collectiveId: number, + newStatus: OrderStatus, + context: string, + ): Promise; clearExpiredLocks(): Promise<[null, number]>; } @@ -667,30 +672,79 @@ Order.cancelActiveOrdersByTierId = function (tierId: number) { }; /** - * Cancels all orders with subscriptions that cannot be transferred when changing hosts (i.e. PayPal) + * Update the status of all non-transferable active orders for the given collective. + * + * The only type of contributions that can be transferred are non-Stripe Connect credit card subscriptions. + * + * Note: for externally managed subscriptions (PayPal), marking the orders as CANCELLED will trigger the + * cancellation of the associated subscriptions on the payment provider. + * See `cron/hourly/70-cancel-subscriptions-for-cancelled-orders.ts`. */ -Order.cancelNonTransferableActiveOrdersByCollectiveId = function (collectiveId: number) { - return sequelize.query( +Order.updateNonTransferableActiveOrdersStatusesByCollectiveId = async function ( + collectiveId: number, + newStatus: OrderStatus, + updateContext = 'Bulk update for non-transferable active orders', +): Promise { + const [orders] = await sequelize.query( ` - UPDATE public."Orders" - SET - status = 'CANCELLED', - "updatedAt" = NOW() - WHERE id IN ( - SELECT "Orders".id FROM public."Orders" - INNER JOIN public."Subscriptions" ON "Subscriptions".id = "Orders"."SubscriptionId" - WHERE - "Orders".status NOT IN ('PAID', 'CANCELLED', 'REJECTED', 'EXPIRED') AND - "Subscriptions"."isManagedExternally" AND - "Subscriptions"."isActive" AND - "Orders"."CollectiveId" = ? - ) - `, + UPDATE "Orders" + SET + status = :newStatus, + "updatedAt" = NOW(), + "data" = COALESCE("data", '{}'::JSONB) || JSONB_BUILD_OBJECT('updateContext', :updateContext) + WHERE id IN ( + SELECT "Orders".id FROM "Orders" + INNER JOIN "Subscriptions" ON "Subscriptions".id = "Orders"."SubscriptionId" + LEFT JOIN "PaymentMethods" pm ON "Orders"."PaymentMethodId" = pm.id + LEFT JOIN "PaymentMethods" spm ON pm."SourcePaymentMethodId" = spm.id + WHERE "Orders"."CollectiveId" = :collectiveId + AND "Subscriptions"."isActive" + AND status NOT IN ('PAID', 'PAUSED', 'CANCELLED', 'REJECTED', 'EXPIRED') + AND CASE + WHEN spm.id IS NULL THEN ( + pm.service != 'stripe' + OR pm.type != 'creditcard' + OR COALESCE(pm.data#>>'{stripeAccount}', :platformStripeAccountId) != :platformStripeAccountId + ) ELSE ( + spm.service != 'stripe' + OR spm.type != 'creditcard' + OR COALESCE(spm.data#>>'{stripeAccount}', :platformStripeAccountId) != :platformStripeAccountId + ) + END + ) + RETURNING id, "SubscriptionId" + `, { type: QueryTypes.UPDATE, - replacements: [collectiveId], + raw: true, + replacements: { + collectiveId, + newStatus, + updateContext, + platformStripeAccountId: config.stripe.accountId, + }, }, ); + + // Update related subscriptions that are managed internally. For the ones managed externally, we need to get + // the payment provider's confirmation before marking them as cancelled (see `cron/hourly/70-cancel-subscriptions-for-cancelled-orders.ts`). + if ([OrderStatus.CANCELLED, OrderStatus.PAUSED].includes(newStatus) && orders.length) { + await sequelize.query( + ` + UPDATE "Subscriptions" + SET "isActive" = FALSE, "updatedAt" = NOW() + WHERE "id" IN (:subscriptionIds) + AND "isManagedExternally" = FALSE + `, + { + replacements: { + subscriptionIds: orders.map(order => order.SubscriptionId), + }, + }, + ); + } + + return orders.length; }; Temporal(Order, sequelize); diff --git a/templates/emails/collective.unhosted.hbs b/templates/emails/collective.unhosted.hbs index d9430955a57e..b4299e52ab2d 100644 --- a/templates/emails/collective.unhosted.hbs +++ b/templates/emails/collective.unhosted.hbs @@ -6,11 +6,14 @@ Subject: Important: {{{collective.name}}} has been un-hosted by {{{host.name}}} Hi {{collective.name}},

You have been un-hosted by {{> linkCollective collective=host}}. This Fiscal Host will no longer manage money on behalf of your Collective. Any active recurring PayPal contributions or Virtual Cards will be cancelled. + {{#if pauseContributions}} + Once you're ready to start receiving contributions again (either with a new Fiscal Host or as an Independent Collective), we will help you reach out to your contributors to let them know how to resume their contributions. + {{/if}}

{{#if message}}

- Message from {{host.name}}: + Message from {{#if isHostAdmin}}{{host.name}}{{else}}{{collective.name}}{{/if}}:

{{message}}
{{/if}} diff --git a/templates/emails/subscription.paused.hbs b/templates/emails/subscription.paused.hbs new file mode 100644 index 000000000000..d67b22eb99f4 --- /dev/null +++ b/templates/emails/subscription.paused.hbs @@ -0,0 +1,24 @@ +Subject: Your contribution to {{{collective.name}}} has been paused + +{{> header}} + +

{{> greeting}}

+ +

Your recurring contribution to {{collective.name}} for {{currency subscription.amount currency=subscription.currency}}/{{subscription.interval}} has been paused.

+ +{{#if reason}} +

The reason given by the collective is:

+
{{reason}}
+{{/if}} + +

+ We'll inform you when your contribution is ready for reactivation. Without any further action from you, your contribution will remained paused. +

+ +

Warmly,

+ +

+ – {{collective.name}} +

+ +{{> footer}} diff --git a/test/server/graphql/v2/mutation/HostApplicationMutations.test.ts b/test/server/graphql/v2/mutation/HostApplicationMutations.test.ts index d1a13ee7a205..3f0e72eee92a 100644 --- a/test/server/graphql/v2/mutation/HostApplicationMutations.test.ts +++ b/test/server/graphql/v2/mutation/HostApplicationMutations.test.ts @@ -14,6 +14,7 @@ import { VirtualCardStatus } from '../../../../../server/models/VirtualCard'; import * as stripeVirtualCardService from '../../../../../server/paymentProviders/stripe/virtual-cards'; import { randEmail } from '../../../../stores'; import { + fakeActiveHost, fakeCollective, fakeEvent, fakeHost, @@ -92,8 +93,8 @@ const PROCESS_HOST_APPLICATION_MUTATION = gql` `; const REMOVE_HOST_MUTATION = gql` - mutation UnhostAccount($account: AccountReferenceInput!, $message: String) { - removeHost(account: $account, message: $message) { + mutation UnhostAccount($account: AccountReferenceInput!, $message: String, $pauseContributions: Boolean) { + removeHost(account: $account, message: $message, pauseContributions: $pauseContributions) { id slug name @@ -410,7 +411,7 @@ describe('server/graphql/v2/mutation/HostApplicationMutations', () => { ); }); - it('requires token with host scope', async () => { + it("can't remove if unauthenticated", async () => { const result = await graphqlQueryV2(REMOVE_HOST_MUTATION, { account: { id: 'some id', @@ -420,6 +421,14 @@ describe('server/graphql/v2/mutation/HostApplicationMutations', () => { expect(result.errors[0].message).to.equal('You need to be logged in to manage hosted accounts.'); }); + it("can't remove if not an admin", async () => { + const randomUser = await fakeUser(); + const collective = await fakeCollective(); + const result = await graphqlQueryV2(REMOVE_HOST_MUTATION, { account: { legacyId: collective.id } }, randomUser); + expect(result.errors).to.exist; + expect(result.errors[0].message).to.equal('Only the host admin or the account admin can trigger this action'); + }); + it('results in error if account does not exist', async () => { const result = await graphqlQueryV2( REMOVE_HOST_MUTATION, @@ -475,6 +484,62 @@ describe('server/graphql/v2/mutation/HostApplicationMutations', () => { expect(unhostingActivity.data.host.id).to.eq(host.id); }); + it('pauses the contributions if requested', async () => { + const host = await fakeActiveHost(); + const collective = await fakeCollective({ HostCollectiveId: host.id }); + const transferableOrder = await fakeOrder( + { + status: OrderStatuses.ACTIVE, + CollectiveId: collective.id, + interval: 'month', + subscription: { isManagedExternally: false, isActive: true }, + }, + { + withSubscription: true, + }, + ); + + const nonTransferableOrder = await fakeOrder( + { + status: OrderStatuses.ACTIVE, + CollectiveId: collective.id, + interval: 'month', + subscription: { isManagedExternally: true, isActive: true }, + }, + { + withSubscription: true, + }, + ); + + const result = await graphqlQueryV2( + REMOVE_HOST_MUTATION, + { + account: { legacyId: collective.id }, + message: 'We are transitioning to a new host', + pauseContributions: true, + }, + rootUser, + ); + + expect(result.errors).to.not.exist; + expect(result.data.removeHost.host).to.be.null; + + const unhostingActivity = await models.Activity.findOne({ + where: { type: activities.COLLECTIVE_UNHOSTED, CollectiveId: collective.id }, + }); + expect(unhostingActivity).to.exist; + expect(unhostingActivity.data.message).to.eq('We are transitioning to a new host'); + expect(unhostingActivity.data.collective.id).to.eq(collective.id); + expect(unhostingActivity.data.host.id).to.eq(host.id); + + await nonTransferableOrder.reload(); + expect(nonTransferableOrder.status).to.eq(OrderStatuses.PAUSED); + expect(nonTransferableOrder.data.pausedForTransfer).to.be.true; + + await transferableOrder.reload(); + expect(transferableOrder.status).to.eq(OrderStatuses.ACTIVE); + }); + it('validates if collective does not have a parent account', async () => { const host = await fakeHost(); const collective = await fakeCollective({ HostCollectiveId: host.id }); @@ -529,6 +594,30 @@ describe('server/graphql/v2/mutation/HostApplicationMutations', () => { expect(result.errors[0].message).to.equal('Unable to change host: you still have a balance of $24.00'); }); + it('removes the host as a collective admin', async () => { + const host = await fakeActiveHost(); + const collective = await fakeCollective({ HostCollectiveId: host.id }); + const result = await graphqlQueryV2( + REMOVE_HOST_MUTATION, + { + account: { legacyId: collective.id }, + message: 'removing as collective admin', + }, + rootUser, + ); + + expect(result.errors).to.not.exist; + expect(result.data.removeHost.host).to.be.null; + + const unhostingActivity = await models.Activity.findOne({ + where: { type: activities.COLLECTIVE_UNHOSTED, CollectiveId: collective.id }, + }); + expect(unhostingActivity).to.exist; + expect(unhostingActivity.data.message).to.eq('removing as collective admin'); + expect(unhostingActivity.data.collective.id).to.eq(collective.id); + expect(unhostingActivity.data.host.id).to.eq(host.id); + }); + it('validates if user is admin of target host collective', async () => { const user = await fakeUser(); await fakeCollective({ admin: user }); @@ -545,7 +634,7 @@ describe('server/graphql/v2/mutation/HostApplicationMutations', () => { user, ); expect(result.errors).to.exist; - expect(result.errors[0].message).to.equal('You need to be authenticated to perform this action'); + expect(result.errors[0].message).to.equal('Only the host admin or the account admin can trigger this action'); }); it('removes the host from a hosted collective using host admin', async () => { diff --git a/test/server/lib/recurring-contributions.test.js b/test/server/lib/recurring-contributions.test.js index faa21ed14597..41777fe43966 100644 --- a/test/server/lib/recurring-contributions.test.js +++ b/test/server/lib/recurring-contributions.test.js @@ -18,7 +18,7 @@ import { } from '../../../server/lib/recurring-contributions'; import models from '../../../server/models'; import { randEmail } from '../../stores'; -import { fakeOrder } from '../../test-helpers/fake-data'; +import { fakeCollective, fakeOrder } from '../../test-helpers/fake-data'; import * as utils from '../../utils'; async function createOrderWithSubscription(interval, date, quantity = 1) { @@ -472,7 +472,7 @@ describe('server/lib/recurring-contributions', () => { beforeEach(async () => { await utils.resetTestDB(); user = await models.User.createUserWithCollective({ email: randEmail(), name: 'Test McTesterson' }); - collective = await models.Collective.create({ name: 'Parcel' }); + collective = await fakeCollective({ name: 'Parcel' }); tier = await models.Tier.create({ name: 'backer', amount: 0, CollectiveId: collective.id }); }); diff --git a/test/server/models/Order.test.js b/test/server/models/Order.test.js index e3a920a1d079..7a26ee3c2a19 100644 --- a/test/server/models/Order.test.js +++ b/test/server/models/Order.test.js @@ -4,7 +4,7 @@ import sinon from 'sinon'; import models from '../../../server/models'; import { randEmail } from '../../stores'; -import { fakeOrder } from '../../test-helpers/fake-data'; +import { fakeCollective, fakeOrder, fakePaymentMethod, randStr } from '../../test-helpers/fake-data'; import * as utils from '../../utils'; describe('server/models/Order', () => { @@ -122,4 +122,97 @@ describe('server/models/Order', () => { ]); }); }); + + describe('updateNonTransferableActiveOrdersStatusesByCollectiveId', () => { + it('cancels non transferable active orders', async () => { + const collective = await fakeCollective(); + + // Payment methods + const paypalPm = await fakePaymentMethod({ + service: 'paypal', + type: 'subscription', + }); + const stripePm = await fakePaymentMethod({ + service: 'stripe', + type: 'creditcard', + }); + const stripeGiftCardPm = await fakePaymentMethod({ + service: 'opencollective', + type: 'giftcard', + SourcePaymentMethodId: stripePm.id, + }); + const stripeConnectPm = await fakePaymentMethod({ + service: 'stripe', + type: 'creditcard', + data: { stripeAccount: randStr() }, + }); + const stripeConnectGiftCardPm = await fakePaymentMethod({ + service: 'opencollective', + type: 'giftcard', + SourcePaymentMethodId: stripeConnectPm.id, + }); + + // Orders + const fakeActiveOrder = async PaymentMethodId => { + return fakeOrder( + { + CollectiveId: collective.id, + status: 'ACTIVE', + PaymentMethodId, + }, + { + withSubscription: true, + }, + ); + }; + + const activePaypalOrder = await fakeActiveOrder(paypalPm.id); + const activeStripeOrder = await fakeActiveOrder(stripePm.id); + const activeStripeGiftCardOrder = await fakeActiveOrder(stripeGiftCardPm.id); + const activeStripeConnectOrder = await fakeActiveOrder(stripeConnectPm.id); + const activeStripeConnectGiftCardOrder = await fakeActiveOrder(stripeConnectGiftCardPm.id); + const inactiveOrder = await fakeOrder({ + CollectiveId: collective.id, + status: 'CANCELLED', + PaymentMethodId: paypalPm.id, + }); + + // Trigger the update + const nbUpdated = await models.Order.updateNonTransferableActiveOrdersStatusesByCollectiveId( + collective.id, + 'PAUSED', + 'Testing', + ); + + // Check the results + expect(nbUpdated).to.equal(3); + await activePaypalOrder.reload({ include: [models.Subscription] }); + await activeStripeOrder.reload({ include: [models.Subscription] }); + await activeStripeGiftCardOrder.reload({ include: [models.Subscription] }); + await activeStripeConnectOrder.reload({ include: [models.Subscription] }); + await activeStripeConnectGiftCardOrder.reload({ include: [models.Subscription] }); + await inactiveOrder.reload({ include: [models.Subscription] }); + + expect(activePaypalOrder.status).to.equal('PAUSED'); + expect(activeStripeOrder.status).to.equal('ACTIVE'); // Do not touch stripe orders since they can be transfered + expect(activeStripeGiftCardOrder.status).to.equal('ACTIVE'); + expect(activeStripeConnectOrder.status).to.equal('PAUSED'); + expect(activeStripeConnectGiftCardOrder.status).to.equal('PAUSED'); + expect(inactiveOrder.status).to.equal('CANCELLED'); + + expect(activePaypalOrder.data.updateContext).to.equal('Testing'); + expect(activeStripeOrder.data?.updateContext).to.not.exist; + expect(activeStripeGiftCardOrder.data?.updateContext).to.not.exist; + expect(activeStripeConnectOrder.data.updateContext).to.equal('Testing'); + expect(activeStripeConnectGiftCardOrder.data.updateContext).to.equal('Testing'); + expect(inactiveOrder.data?.updateContext).to.not.exist; + + // Check the subscriptions + expect(activePaypalOrder.Subscription.isActive).to.be.false; + expect(activeStripeOrder.Subscription.isActive).to.be.true; + expect(activeStripeGiftCardOrder.Subscription.isActive).to.be.true; + expect(activeStripeConnectOrder.Subscription.isActive).to.be.false; + expect(activeStripeConnectGiftCardOrder.Subscription.isActive).to.be.false; + }); + }); });