Accéder directement au contenu
Créez un compte ou connecter-vous
Logo de la documentation Stripe
/
Demander à l'assistant IA
Créez un compteConnectez-vous
Démarrer
Paiements
Revenus
Plateformes et places de marché
Gestion de fonds
Ressources pour les développeurs
API et SDKAide
Aperçu
Billing
PrésentationÀ propos des API Billing
Abonnements
Invoicing
Facturation à la consommation
Facturation à l'utilisation avancée
Devis
Gestion des clients
Facturation avec d'autres produits
Recouvrement de revenus
Automatisations
Tester votre intégration
Tax
Présentation
Utiliser Stripe Tax
Gérer la conformité
Rapports
Présentation
Sélectionner un rapport
Configurer des rapports
Rapports sur plusieurs comptes
API de rapport
Comptabilisation des revenus
Données
Présentation
Requête sur les données de l'entreprise
    Présentation
    Données relatives aux transactions
    Données relatives aux litiges et à la fraude
    Données sur l'ensemble des frais
    Données de facturation
    Données fiscales
    Données relatives aux comptes connectés
    Données relatives à l'émission de cartes
    Actualisation des données
    Schéma de données
Sigma
Data Pipeline
Importer des données externes
États-Unis
Français (France)
AccueilRevenusQuery business data

Interroger les données de facturation

Utilisez Sigma ou Data Pipeline pour récupérer les informations de facturation.

Stripe Billing comprend différents composants qui, associés, fournissent des factures ponctuelles et périodiques. Les différents aspects des données de facturation sont disponibles dans plusieurs tables. Toutes les tables spécifiques à la facturation sont accessibles dans la section Facturation du schéma, les tables principales étant subscriptions et invoices.

Pour aller plus loin, vous pouvez utiliser les tables supplémentaires qui représentent les composants des abonnements et des factures, comme prices, products ou coupons. En outre, la table customers fait partie intégrante de la facturation et contient des données pour lesquelles vous devrez peut-être créer un rapport.

Abonnements

Chaque ligne du tableau subscriptions représente les données d’un objet Subscription individuel. Ces mêmes informations sont récupérées par l’API ou accessibles dans le Dashboard Stripe. Vous pouvez créer un rapport pour chaque abonnement créé sur votre compte.

Nous vous recommandons de commencer par cette table pour la création de rapports sur vos abonnés actuels. Vous pouvez la lier à d’autres tables et ainsi explorer vos données plus en détails.

L’exemple suivant récupère une liste d’abonnements marqués comme impayés, ainsi que les coordonnées du client, le cas échéant.

select subscriptions.id, subscriptions.customer_id, customers.email from subscriptions inner join customers on customers.id = subscriptions.customer_id where subscriptions.status = 'unpaid' limit 5
idcustomer_idAdresse e-mail
sub_APkFrb15HwmehLpcus_neRybB3A6pBwwDuflorence.martin@exemple.com
sub_Cm0nZOeXAbg5OBBcus_fK0A8km06HSuXBNpierre.durand@exemple.fr
sub_JmyaeSQNGlwN1vmcus_Tjn9u65O55wVMG2philippe.legrain@exemple.fr
sub_Wj6KK29AeO9pILfcus_wcYDTb8s5asUf1ksylvie.dumont@exemple.fr
sub_giWcYE2cvngXOoXcus_k3vkOkwLw3ZECWNelias.boudet@exemple.fr

Clients

Les données relatives aux objets Customer figurent dans la table customers (elles ne font pas partie du groupe Tables Billing). Elles sont couramment utilisées dans le cadre des rapports basés sur la facturation et peuvent être liées à un certain nombre de tables. Elles sont également utiles si vous créez des paiements avec des informations de paiement enregistrées.

L’exemple suivant génère une liste de clients disposant d’un abonnement et qui bénéficient actuellement d’une période d’essai, avec leur ID et leur adresse e-mail.

select customers.id, customers.email, subscriptions.price_id from subscriptions inner join customers on customers.id = subscriptions.customer_id where subscriptions.status = 'trialing' limit 5
idAdresse e-mailprice_id
cus_tTb7ctTXFXHQ3l7florence.martin@exemple.comruby-pro-522
cus_O8FLwt0OOiZzjGLpierre.durand@exemple.frruby-pro-522
cus_QexCPEDOegIXJJZrichard.dupont@exemple.frgold-basic-221
cus_M1CnkoSklewKrqHsylvie.dumont@exemple.frgold-basic-221
cus_PVchNMfzju7JmtPelias.boudet@exemple.frsilver-pro-498

Produits et tarifs

Les produits décrivent les articles que vos clients peuvent acheter avec un abonnement. Les tarifs sont liés aux produits et définissent le coût, l’intervalle de facturation et la devise. Lorsque vous consultez des données dans le tableau subscriptions, vous pouvez les lier à subscription_items. En outre, vous pouvez les lier à products.id en utilisant le price_product_id de l’article.

L’exemple suivant renvoie une liste des abonnements actifs avec le nom du produit et son libellé de relevé bancaire :

active_subscription_products.sql
with active_subscriptions as ( select s.id as subscription_id, p.name as product_name, p.statement_descriptor from subscriptions s join subscription_items si on s.id = si.subscription_id join products p on si.price_product_id = p.id where s.status = 'active' ) select subscription_id, subscription_item_id, price_id, product_name, statement_descriptor from active_subscriptions order by 1,2
idnomstatement_descriptor
sub_Y0akzcOp1cYqsj8ruby-pro-522Ruby Pro
sub_np0bf6jR2ZFGMhxgold-basic-221Gold Basic
sub_KQw3FtXXF2pWVHJsilver-pro-498Silver Pro
sub_pUCCWV4Q5BN7je9diamond-mid-244Diamond Mid
sub_I3ysKEhaE5CtRYyruby-standard-196Ruby Standard

Niveaux de tarifs

Lorsque vous utilisez des niveaux de tarifs pour vos abonnements, le tableau price_tiers peut fournir des données spécifiques sur chaque niveau. Par exemple, si vous voulez comprendre des éléments du premier niveau de vos abonnements, tels que la quantité maximale et le montant unitaire utilisé, utilisez la requête suivante :

tiered_prices.sql
with subscription_item_prices as ( select si.subscription_id, si.price_id, p.currency from subscription_items si join prices p on si.price_id = p.id ), price_tier_details as ( select sp.subscription_id, pt.price_id, pt.upto, stringify_amount(sp.currency, pt.amount, '.') as tier_price, sp.currency from subscription_item_prices sp join price_tiers pt on sp.price_id = pt.price_id ) select ptd.subscription_id, ptd.price_id, ptd.upto, ptd.tier_price, ptd.currency from price_tier_details ptd order by ptd.subscription_id, ptd.price_id, ptd.upto asc
subscription_idprice_iduptotier_pricedevise
sub_ldxL4G8H0jg4Ik3price_mXUgBHjGZBu2eAq302USD
sub_Y6uRf1TMgdSvrcwprice_gpcqM1q5q633w2K601USD
sub_5hBgoo0ICUfOrccprice_z9BJgLEgiZYTAkK900,50USD

Factures

Utilisation des factures

Pour en savoir plus sur les factures et les postes de facture, consultez notre documentation consacrée aux factures.

La table invoices contient des données sur les objets Invoice. Chaque abonnement génère une facture récurrente qui représente le montant dû par le client, à savoir le montant de l’abonnement ainsi que tous les postes de facture supplémentaires qui ont été créés (et répertoriés comme tels).

Les factures sont constituées de postes de facture individuels. Ces postes correspondent aux abonnements pour lesquels le client est facturé, ainsi qu’aux postes de facture qui ont été créés et appliqués à la facture. Pour décomposer une facture et analyser chacun de ses postes, utilisez la table invoice_line_items.

La colonne source_id de cette table contient l’ID de l’abonnement (par exemple, sub_vVamfmp01qjLU4Q) ou du poste de facture (par exemple, ii_v5MUNzjVjkhcWBw) auquel correspond le poste. La colonne source_type permet de déterminer si les postes représentent un abonnement ou un poste de facture.

Contrairement à d’autres clés étrangères, la colonne subscription de la table invoice_line_items n’est pas toujours renseignée. Si le poste de facture correspondant est un abonnement, cette colonne est vide : son ID apparaît déjà dans la colonne source_id.

Postes de facture

Les données relatives aux postes de facture sont disponibles dans la table invoice_items. Les postes de facture sont couramment utilisés pour spécifier un montant supplémentaire (ou déduire un montant) qui est appliqué à la facture suivante au début du cycle de facturation suivant. Par exemple, vous pouvez créer un poste de facture si vous devez facturer votre client car il a dépassé son plafond mensuel, ou si vous devez offrir un crédit sur la facture suivante pour un service non utilisé.

L’exemple suivant récupère toutes les factures et tous les ID de paiement associés pour un abonnement spécifique.

select id, charge_id, amount_due from invoices where subscription_id = 'sub_ALJXL9gBYtv6GJ'
idnom
in_Q1PfFx90m5h9ybHch_pVlwe3d3Oyf9xCi1999
in_7vU1tN6yOXG6rtFch_2w7bqwb9zpVXhc81999
in_ZYXD7WhgbpE76f61999ch_yemt4IwvrolGb6C
in_bUAzqbBcyeUHXn41999ch_ezUlAZR3GC01lQ3
in_a8HGZX6H6KNt5yi1999ch_p11LzO7Tm5Yjv3d

Totaux de factures et ristournes

Le sous-total de la facture correspond à la somme de tous les abonnements, postes de facture et calculs au prorata sur la facture avant l’application d’une ristourne. Le total de la facture correspond au montant après l’application des ristournes et des taxes :

invoice.total = invoice.subtotal - discount + invoice.tax

Aucune colonne ne correspond au montant de la réduction sur une facture. Vous pouvez cependant la calculer en faisant la somme de tous les montants de réduction des postes. La requête suivante renvoie une liste avec des factures, le début et la fin de leur période ainsi que le montant total de la remise pour chaque facture.

invoice_discounts.sql
with invoices_with_discounts as ( select invoice_id, sum(amount) as total_discount_amount from invoice_line_item_discount_amounts group by invoice_id ) select i.id as invoice_id, i.period_start, i.period_end, stringify_amount(i.currency, ilda.total_discount_amount, '.') as total_discount_amount i.currency from invoices i join invoices_with_discounts ilda on i.id = ilda.invoice_id order by i.id
invoice_idperiod_startperiod_endtotal_discount_amountdevise
in_LUCF2CerXFeELXm01/05/202401/06/202424,66USD
in_OJtgYVPpl8cK2Tc01/06/202401/07/202424,34USD
in_LzmazquFqrnSXid01/04/202401/05/202445,96USD

Utilisation des dates et des périodes de facturation

Les factures d’abonnement sont pré-facturées, ce qui signifie que le client effectue le paiement au début d’un cycle de facturation. La date est précisée dans la valeur period d’un poste. Par exemple, un client ayant souscrit un abonnement mensuel est facturé au début de chaque mois. S’il annule son abonnement à l’aide de cancel_at_period_end, son abonnement reste actif jusqu’à la fin du mois, date à laquelle il prendra fin.

Les valeurs period_start et period_end d’une facture indiquent la date de création des postes de facture. Elles ne correspondent pas forcément à la période de facturation du client. Par exemple, si un client est facturé le premier de chaque mois et qu’il dépasse son plafond mensuel le 15, vous pouvez créer un poste de facture pour les frais supplémentaires facturés au client. Ce poste est ensuite inclus dans la facture suivante, qui est créée le premier du mois suivant. Lorsque la facture suivante est générée, la date period_start correspond au 15 du mois précédent, date à laquelle le poste supplémentaire a été créé pour la première fois.

Facturation à l’usage

La facturation à l’usage vous permet de facturer vos clients en fonction de la quantité de produit ou service qu’ils utilisent.

Dispositif de mesure de la facturation

Un objet Meter indique comment agréger les événements de mesure sur une période de facturation. Les événements liés aux objets Meter représentent toutes les actions que les clients effectuent dans votre système (notamment les requêtes à l’API). Les objets Meter sont associés à des tarifs et constituent la base de la facture. Ces objets sont disponibles via la table billing_meters.

La requête suivante renvoie tous les dispositifs de mesure de la facturation actifs.

meters.sql
select id, status, display_name, default_aggregation_formula from billing_meters where status = 'ACTIVE' and livemode
IDétatdisplay_namedefault_aggregation_formula
mtr_se8XTHYEXEQO8JGACTIFalpaca_ai_tokenSOMME
mtr_5Nzj83M8saTMJT4ACTIFalpaca_ai_image_tokenNOMBRE

Récapitulatifs des événements de mesure de la facturation

Un objet Billing Meter Event Summary représente une vue agrégée des événements de mesure de la facturation d’un client sur une période donnée. Il représente la quantité cumulée consommée par un client au cours de cette période. Ces objets sont disponibles via la table billing_meter_event_summaries. Des récapitulatifs horaires sont disponibles, comme l’indique la colonne value_grouping_window.

La requête suivante renvoie tous les événements de mesure de la facturation non valides pour un client spécifique.

billing_meter_event_summaries.sql
select billing_meters.display_name, sum(billing_meter_event_summaries.aggregated_value) AS total_usage from billing_meter_event_summaries join billing_meters on billing_meters.id = billing_meter_event_summaries.meter_id where billing_meter_event_summaries.customer_id = 'cus_EDQkYj7P2Jf3sJ1' and billing_meter_event_summaries.start_time >= timestamp '2025-02-01 08:00' and billing_meter_event_summaries.end_time <= timestamp '2025-02-01 20:00' and value_grouping_window = 'hourly' group by display_name
display_nametotal_usage
alpaca_ai_token256717
alpaca_ai_image_token97

Billing analyse utilisation de compteurs

Les objets Analyses de l’utilisation du compteur de facturation représentent un résumé analytique de l’utilisation du compteur de facturation d’un client sur une période spécifiée. Ils peuvent être groupés ou filtrés par compteurs, dimensions et locataires pour alimenter des dashboards d’analytique client.

Le guide d’intégration illustre les formes de requête et de réponse.

Cette API est disponible en aperçu public. Click here pour demander l’accès à cette API.

Événement de mesure de la facturation non valides

Un objet Billing Meter Invalid Event représente un événement de mesure de la facturation qui n’a pas été validé. Ces objets sont disponibles via la table billing_meter_invalid_events. La table billing_meter_invalid_events_payload associée contient la charge utile de l’événement initial.

La requête suivante renvoie tous les événements de mesure de la facturation non valides pour un client spécifique.

billing_meter_invalid_events.sql
SELECT billing_meter_invalid_events.id as event_id, billing_meter_invalid_events.error_code, billing_meter_invalid_events.error_message FROM billing_meter_invalid_events JOIN billing_meter_invalid_events_payload ON billing_meter_invalid_events_payload.event_id = billing_meter_invalid_events.id WHERE billing_meter_invalid_events_payload.key = 'stripe_customer_id' AND billing_meter_invalid_events_payload.value = 'cus_EDQkYj7P2Jf3sJ1'
event_iderror_codeerror_message
X5Snj3j4vBufHtfNtY2ue0NToS4htfS4METER_NOT_FOUNDAucun dispositif de mesure ne correspond à event_name mtr_Bdi2PUqJtA8mU6s.
2fZCzSYXeZMIz1GRoGpR7WkjlvuofXs0METER_NOT_FOUNDAucun dispositif de mesure ne correspond à event_name mtr_1OaQKk4CTYeaCER.

Coupons

Un objet Coupon correspond à un montant ou à un pourcentage de réduction pouvant s’appliquer à des abonnements ou à des clients.

coupons.sql
select coupons.id, coupons.amount_off, coupons.percent_off from coupons where valid = false limit 5
idamount_offpercent_off
10FF10
SUMMER2525
10FREE10
15OFF15
FALL3030

Discounts

Une réduction correspond à l’application d’un bon de réduction représenté par un objet Discount. La requête suivante renvoie une liste des abonnements avec les réductions et les bons de réduction qui leur sont associés :

discounts.sql
select subscriptions.id as subscription_id, t.discount_id, coupons.id as coupon_id from subscriptions cross join unnest(split(subscriptions.discounts, ',')) as t(discount_id) join discounts on discounts.id = t.discount_id join coupons on coupons.id = discounts.coupon_id limit 3
subscription_iddiscount_idcoupon_id
sub_I4nSJAindGAGhmZdi_1NAMhOOlwLBhtuX10OFF
sub_j32X5WN0bPa6E2Tdi_TGgF6pY7Osp2zCI25OFF
sub_NtKo6V5dY7CJBWNdi_VUTUMQJZq7pXswc10FREE

Codes promotionnels

Un code promotionnel est un code utilisable par le client pour obtenir un bon de réduction. La requête suivante fournit une liste des codes promotionnels relatifs à un bon de réduction spécifique et affiche le nombre de fois où chaque code a été utilisé :

promotion_codes.sql
select promotion_codes.id as promotion_code_id, promotion_codes.code as promotion_code, promotion_codes.times_redeemed from promotion_codes limit 3
promotion_code_idcodetimes_redeemed
promo_ILYtn7OJMvlcD2Y10OFF1
promo_HXcErEacFRBuX3u25OFF2
promo_F0eAchkEVbEuBEH10FREE3

Événements liés à la modification des éléments d’abonnement

Le tableau subscription_item_change_events suit les modifications apportées aux postes d’abonnement qui ont une incidence sur le revenu récurrent mensuel (MRR) et les quantités d’abonnement. Utilisez ce tableau pour calculer le MRR des clients individuels, des produits ou des offres pour créer des définitions d’indicateurs personnalisés pour vos modèles économiques et pour suivre les modifications des quantités d’abonnement.

Mise en garde

Ce tableau fournit des données plus récentes que la source des mesures MRR dans l’aperçu de Billing du Dashboard Stripe. Ainsi, les données pour le MRR du dernier jour et du jour en cours peuvent être plus précises et différer de ce que vous voyez dans le Dashboard.

Événements de modification des éléments d’abonnement v2 Public preview

Le tableau subscription_item_change_events_v2_beta remplace le tableau subscription_item_change_events par une actualisation des données améliorée. Les données de ce tableau conservent une actualisation de 3 heures dans Sigma et de 7 heures dans Stripe Data Pipeline. Elle partage le même schéma avec l’ensemble de données existant. Vous pouvez effectuer cette requête en ajoutant le suffixe v2_beta au tableau dans les mêmes exemples de requêtes de modèle.

Les données peuvent être modifiées

Ce tableau fournit des données plus récentes et plus cohérentes avec le Dashboard de Stripe que l’ensemble de données existant. Cela signifie que les futures livraisons de cet ensemble de données pourront contenir des données mises à jour pour le MRR du jour précédent et du jour actuel. Attendez que toutes les données soient finalisées (48 heures maximum) avant d’utiliser ces données de manière incrémentielle.

local_event_timestamp et event_timestamp

Ce tableau comprend deux colonnes d’horodatage :

  • event_timestamp : horodatage UTC.
  • local_event_timestamp : horodatage dans votre fuseau horaire local. Il s’agit généralement du fuseau horaire de la personne qui a créé votre compte Stripe.

devise

Vous y trouverez la devise de règlement du poste d’abonnement sous la forme d’un code ISO à trois lettres minuscules. La devise doit être prise en charge par Stripe.

mrr_change

La colonne mrr_change indique l’impact positif ou négatif d’un événement sur votre MRR dans la sous-unité de la devise de règlement de l’élément d’abonnement (par exemple, les cents pour le dollar américain).

quantity_change

La colonne quantity_change indique l’augmentation ou la diminution de la quantité associée à un poste d’abonnement auquel un client est abonné.

event_type

Type d’événementDéfinitions
ACTIVE_STARTL’élément d’abonnement a commencé à contribuer au MRR.
ACTIVE_ENDL’élément d’abonnement a cessé de contribuer au MRR.
ACTIVE_UPGRADELa contribution du poste d’abonnement au MRR a augmenté. Cela peut se produire lorsque le tarif d’un poste d’abonnement augmente ou lorsque la quantité associée à ce poste d’abonnement augmente.
ACTIVE_DOWNGRADELa contribution du poste d’abonnement au MRR a diminué. Cela peut se produire lorsque le tarif d’un poste d’abonnement diminue ou lorsque la quantité associée à ce poste d’abonnement diminue.
ACTIVE_QUANTITY_INCREASELa quantité associée au poste d’abonnement a augmenté, mais le MRR n’a pas été affecté. Cette situation peut se produire si vous utilisez une tarification progressive et que la quantité doit être supérieure à un certain seuil avant que le tarif ne soit modifié.
ACTIVE_QUANTITY_DECREASELa quantité associée au poste d’abonnement a diminué, mais le MRR n’a pas été affecté. Cette situation peut se produire si vous utilisez une tarification progressive et que la quantité doit être inférieure à un certain seuil avant que le tarif ne soit modifié.

Remarque

Certaines actions de l’utilisateur peuvent créer plusieurs événements. Ainsi, pour le même subscription_id, vous pourriez voir un événement avec un event_type ACTIVE_END sur un élément, puis directement après un événement avec un event_type ACTIVE_START sur un autre élément.

Autres colonnes

Les autres colonnes (product_id, price_id, customer_id, subscription_id et subscription_item_id) contiennent les identifiants liés à l’événement de modification de l’élément d’abonnement.

Exemples de requêtes

Pour des exemples supplémentaires et actualisés, veuillez vous référer à la section Abonnements de la bibliothèque de modèles de requêtes dans la barre latérale de Sigma.

Pour calculer le revenu récurrent mensuel (MRR) et le nombre d’abonnés actifs à partir de ce tableau, vous devrez utiliser des fonctions de fenêtre. De plus, si vos clients utilisent des devises différentes, vous devrez effectuer des calculs de change. Le calcul vise à suivre le MRR mensuel et l’évolution du nombre d’abonnés actifs, en faisant la distinction entre les nouveaux abonnements, les réactivations, les expansions, les contractions et les désabonnements. Les résultats finaux sont présentés en unités monétaires mineures, par exemple en cents pour le dollar américain.

WITH ts_grouped_sub_item_events AS ( SELECT local_event_timestamp, customer_id, currency, sum(mrr_change) AS mrr_change FROM subscription_item_change_events_v2_beta GROUP BY 1, 2, 3 ), ts_grouped_sub_item_events_with_mrr AS ( SELECT *, date_trunc( 'day', date(local_event_timestamp) ) AS local_event_date, -- Stripe defines an "active subscriber" as a customer with non-zero MRR. -- Therefore instead of summing up event_type to get subscription count (and its diff), -- We count the amount of revenue on each customer instead and later check its movement from / to zero sum(mrr_change) over ( PARTITION by customer_id ORDER BY local_event_timestamp ASC ) AS mrr, -- We count the # of times MRR has actually changed, and use nullif to ignore events that do not impact MRR -- Otherwise we may confuse between new vs. reactivation count(nullif(mrr_change, 0)) over ( PARTITION by customer_id ORDER BY local_event_timestamp ASC ) AS mrr_change_count FROM ts_grouped_sub_item_events ), ts_grouped_sub_item_events_with_previous_mrr AS ( SELECT *, coalesce( last_value(mrr) IGNORE nulls OVER ( PARTITION by customer_id ORDER BY local_event_timestamp ASC ROWS BETWEEN UNBOUNDED PRECEDING AND 1 PRECEDING ), 0 ) AS previous_mrr FROM ts_grouped_sub_item_events_with_mrr ), customer_events AS ( SELECT *, CASE WHEN mrr = 0 AND previous_mrr > 0 THEN 'ACTIVE_END' WHEN mrr > 0 AND previous_mrr = 0 AND mrr_change_count = 1 THEN 'ACTIVE_START' WHEN mrr > 0 AND previous_mrr = 0 AND mrr_change_count > 1 THEN 'REACTIVATE' WHEN mrr > previous_mrr THEN 'ACTIVE_UPGRADE' WHEN mrr < previous_mrr THEN 'ACTIVE_DOWNGRADE' ELSE NULL END AS cus_event_type FROM ts_grouped_sub_item_events_with_previous_mrr ), date_grouped_customer_events AS ( SELECT local_event_date, currency, sum(mrr_change) AS mrr_change, sum( CASE cus_event_type WHEN 'ACTIVE_START' THEN mrr_change ELSE 0 END ) AS new_mrr, sum( CASE cus_event_type WHEN 'REACTIVATE' THEN mrr_change ELSE 0 END ) AS reactivation_mrr, sum( CASE cus_event_type WHEN 'ACTIVE_UPGRADE' THEN mrr_change ELSE 0 END ) AS expansion_mrr, sum( CASE cus_event_type WHEN 'ACTIVE_DOWNGRADE' THEN mrr_change ELSE 0 END ) AS contraction_mrr, sum( CASE cus_event_type WHEN 'ACTIVE_END' THEN mrr_change ELSE 0 END ) AS churn_mrr, sum( CASE WHEN mrr = 0 AND previous_mrr > 0 THEN -1 WHEN mrr > 0 AND previous_mrr = 0 THEN 1 ELSE 0 END ) AS active_subscribers_change, sum( CASE cus_event_type WHEN 'ACTIVE_END' THEN 1 ELSE 0 END ) AS churned_subscribers, sum( CASE cus_event_type WHEN 'ACTIVE_START' THEN 1 ELSE 0 END ) AS new_subscribers, sum( CASE cus_event_type WHEN 'REACTIVATE' THEN 1 ELSE 0 END ) AS reactivated_subscribers FROM customer_events GROUP BY 1, 2 ), -- Prepare the multi dimensional table with all days + currency combinations and conversion rate metadata -- note that exchange_rates_from_usd contains one row for every date from 2010-01-07 until today -- which is why we don't need to generate a separate date series for the full table dates_with_rate_per_usd AS ( SELECT -- We use previous day's closing rates in precomputed metrics date - INTERVAL '1' DAY AS fx_date, cast( json_parse(buy_currency_exchange_rates) AS map(varchar, double) ) AS rate_per_usd FROM exchange_rates_from_usd ), currencies AS ( SELECT DISTINCT(currency) FROM subscription_item_change_events_v2_beta ), first_default_currency AS ( SELECT default_currency FROM accounts WHERE default_currency IS NOT NULL LIMIT 1 ), dates_x_currencies_with_conversion_rate AS ( SELECT fx_date as local_date, currency, default_currency, 1 / rate_per_usd [currency] * rate_per_usd [coalesce(default_currency, 'usd')] AS conversion_rate FROM dates_with_rate_per_usd CROSS JOIN currencies CROSS JOIN first_default_currency ORDER BY 1, 2 ), daily_metrics_by_currency AS ( SELECT dpc.local_date, dpc.currency, dpc.conversion_rate, coalesce( sum(mrr_change) over ( PARTITION by dpc.currency ORDER BY dpc.local_date ASC ), 0 ) AS mrr, coalesce( round( sum(mrr_change) over ( PARTITION by dpc.currency ORDER BY dpc.local_date ASC ) * dpc.conversion_rate ), 0 ) AS converted_mrr, coalesce(round(new_mrr * conversion_rate), 0) AS converted_new_mrr, coalesce(round(reactivation_mrr * conversion_rate), 0) AS converted_reactivation_mrr, coalesce(round(expansion_mrr * conversion_rate), 0) AS converted_expansion_mrr, coalesce(round(contraction_mrr * conversion_rate), 0) AS converted_contraction_mrr, coalesce(round(churn_mrr * conversion_rate), 0) AS converted_churn_mrr, coalesce(dgce.mrr_change, 0) AS mrr_change, coalesce(dgce.new_mrr, 0) AS new_mrr, coalesce(dgce.reactivation_mrr, 0) AS reactivation_mrr, coalesce(dgce.expansion_mrr, 0) AS expansion_mrr, coalesce(dgce.contraction_mrr, 0) AS contraction_mrr, coalesce(dgce.churn_mrr, 0) AS churn_mrr, coalesce( sum(active_subscribers_change) over ( PARTITION by dpc.currency ORDER BY dpc.local_date ASC ), 0 ) AS active_subscribers, coalesce(dgce.active_subscribers_change, 0) AS active_subscribers_change, coalesce(dgce.churned_subscribers, 0) AS churned_subscribers, coalesce(dgce.new_subscribers, 0) AS new_subscribers, coalesce(dgce.reactivated_subscribers, 0) AS reactivated_subscribers FROM dates_x_currencies_with_conversion_rate dpc LEFT JOIN date_grouped_customer_events dgce ON dpc.local_date = dgce.local_event_date AND dpc.currency = dgce.currency ), daily_metrics AS ( SELECT local_date, sum(converted_mrr) AS mrr, sum(converted_new_mrr) AS new_mrr, sum(converted_reactivation_mrr) AS reactivation_mrr, sum(converted_expansion_mrr) AS expansion_mrr, sum(converted_contraction_mrr) AS contraction_mrr, sum(converted_churn_mrr) AS churn_mrr, -- Customer can only have active subscription in a single currency at a time, as a result this doesn't result in over-counting subscriber changes -- This also matches the precomputed metrics logic in billing dashboard / CSV download sum(active_subscribers) AS active_subscribers, sum(churned_subscribers) AS churned_subscribers, sum(new_subscribers) AS new_subscribers, sum(reactivated_subscribers) AS reactivated_subscribers FROM daily_metrics_by_currency GROUP BY 1 ), daily_metrics_with_derived AS ( SELECT *, mrr - lag(mrr) over ( ORDER BY local_date ) - new_mrr - reactivation_mrr - expansion_mrr - contraction_mrr - churn_mrr AS fx_adjustment_mrr, lag(mrr) over ( ORDER BY local_date ) AS previous_mrr FROM daily_metrics ), -- Turn daily into monthly metrics monthly_metrics_with_derived AS ( SELECT date_trunc('month', local_date) AS local_month_start, max_by(mrr, local_date) AS ending_mrr, sum(new_mrr) AS new_mrr, sum(reactivation_mrr) AS reactivation_mrr, sum(expansion_mrr) AS expansion_mrr, sum(contraction_mrr) AS contraction_mrr, sum(churn_mrr) AS churn_mrr, sum(fx_adjustment_mrr) AS fx_adjustment_mrr, max_by(active_subscribers, local_date) AS ending_subscribers, sum(churned_subscribers) AS churned_subscribers, sum(new_subscribers) AS new_subscribers, sum(reactivated_subscribers) AS reactivated_subscribers FROM daily_metrics_with_derived GROUP BY 1 ) SELECT local_month_start, ending_mrr - fx_adjustment_mrr - churn_mrr - contraction_mrr - expansion_mrr - reactivation_mrr - new_mrr AS beginning_mrr, new_mrr, reactivation_mrr, expansion_mrr, contraction_mrr, churn_mrr, fx_adjustment_mrr, ending_mrr, -- Churned subscribers is a positive number in CSV reports instead of negative for churn / contraction mrr ending_subscribers - (-1 * churned_subscribers) - reactivated_subscribers - new_subscribers AS beginning_subscribers, new_subscribers, reactivated_subscribers, churned_subscribers, ending_subscribers FROM monthly_metrics_with_derived ORDER BY 1 DESC
local_month_startbeginning_mrrnew_mrrreactivation_mrrexpansion_mrrcontraction_mrrchurn_mrrfx_adjustment_mrrending_mrrbeginning_subscribersnew_subscribersreactivated_subscriberschurned_subscribersending_subscribers
01/05/2024100072149104000040000000100216149930012
01/04/20241000651497180000-1800100072149730012
2024-03-01100066099124000-1074010006514972027
2024-02-011000660991000000-1000010006609971017
01/01/20241000381022921601998-175-3042010006609954027
2023-12-0110003810200000010003810250005
2023-11-0110003710210000000010003810241005
2023-10-0110003710200000010003710240004
2023-09-0110003710200000010003710240004
2023-08-01100033902005 0000-1800010003710250014
2023-07-011000370650000-3159-410003390260015
01/06/202310003640235336900-2742110003706561346
2023-05-011000348982748030437-83-31598010003640273046
2023-04-01100034065933000-100010003489862017
2023-03-01100002715313500000010003406542006
2023-02-011000060486086060880-15507010000271552034
2023-01-011000060483043000-3043010000604851015
2022-12-011001521342591001363600-30000-1505574-22100006048960105
2022-11-01100178232486883333621878-10600-68939701001521347161159
2022-10-0110003619313633312000020600-10000-124894010017823274267
Cette page vous a-t-elle été utile ?
OuiNon
  • Besoin d'aide ? Contactez le service Support.
  • Consultez notre log des modifications.
  • Des questions ? Contactez l'équipe commerciale.
  • LLM ? Lire llms.txt.
  • Propulsé par Markdoc