usage_credits

Demo app for the 💳✨ usage_credits gem

The basics: get and spend credits

 

Subscribe to a plan: Buy credit packs: Get arbitrary bonuses: Spend credits:

Ledger status

Date Amount Category Expiration
Feb 19, 6:22 AM +5100 credits credit_pack_purchase
Feb 19, 6:41 AM +1000 credits credit_pack_purchase
Feb 19, 7:04 AM +20 credits referral_bonus referral
Feb 19, 7:23 AM +10 credits bonus expiring_bonus_credits 🗑️ expired
Feb 19, 9:22 AM +5100 credits credit_pack_purchase
Feb 19, 10:41 AM +5100 credits credit_pack_purchase
Feb 19, 11:40 AM +10 credits credit_pack_purchase
Feb 19, 2:20 PM +1000 credits credit_pack_purchase

Payment records (pay gem)

The user has paid $695.99 total

Pay::Charge (6)
# {
                        :id => 2131,
               :customer_id => 3201,
           :subscription_id => nil,
              :processor_id => "8RRh7D8IO7PqXaDBANiIA",
                    :amount => 19900,
                  :currency => nil,
    :application_fee_amount => nil,
           :amount_refunded => nil,
                  :metadata => {
         "purchase_type" => "credit_pack",
             "pack_name" => "pro",
               "credits" => 5000,
         "bonus_credits" => 100,
           "price_cents" => 19900,
        "price_currency" => "USD"
    },
                      :data => {
        "payment_method_type" => "card",
                      "brand" => "Fake",
                      "last4" => 1234,
                  "exp_month" => 2,
                   "exp_year" => 2026
    },
            :stripe_account => nil,
                :created_at => 2026-02-19 06:22:17.614930000 UTC +00:00,
                :updated_at => 2026-02-19 06:22:17.614930000 UTC +00:00,
                      :type => "Pay::FakeProcessor::Charge"
}
# {
                        :id => 2133,
               :customer_id => 3201,
           :subscription_id => nil,
              :processor_id => "OSyTeBj_Ov8vIZNGU1SJ_",
                    :amount => 19900,
                  :currency => nil,
    :application_fee_amount => nil,
           :amount_refunded => nil,
                  :metadata => {
         "purchase_type" => "credit_pack",
             "pack_name" => "pro",
               "credits" => 5000,
         "bonus_credits" => 100,
           "price_cents" => 19900,
        "price_currency" => "USD"
    },
                      :data => {
        "payment_method_type" => "card",
                      "brand" => "Fake",
                      "last4" => 1234,
                  "exp_month" => 2,
                   "exp_year" => 2026
    },
            :stripe_account => nil,
                :created_at => 2026-02-19 09:22:30.846935000 UTC +00:00,
                :updated_at => 2026-02-19 09:22:30.846935000 UTC +00:00,
                      :type => "Pay::FakeProcessor::Charge"
}
# {
                        :id => 2139,
               :customer_id => 3201,
           :subscription_id => nil,
              :processor_id => "UOI22yVwKv9wQT1zvMkD8",
                    :amount => 4900,
                  :currency => nil,
    :application_fee_amount => nil,
           :amount_refunded => nil,
                  :metadata => {
         "purchase_type" => "credit_pack",
             "pack_name" => "starter",
               "credits" => 1000,
         "bonus_credits" => 0,
           "price_cents" => 4900,
        "price_currency" => "USD"
    },
                      :data => {
        "payment_method_type" => "card",
                      "brand" => "Fake",
                      "last4" => 1234,
                  "exp_month" => 2,
                   "exp_year" => 2026
    },
            :stripe_account => nil,
                :created_at => 2026-02-19 14:20:21.718226000 UTC +00:00,
                :updated_at => 2026-02-19 14:20:21.718226000 UTC +00:00,
                      :type => "Pay::FakeProcessor::Charge"
}
# {
                        :id => 2134,
               :customer_id => 3201,
           :subscription_id => nil,
              :processor_id => "YFOhrocvM3AP2hQShVppm",
                    :amount => 19900,
                  :currency => nil,
    :application_fee_amount => nil,
           :amount_refunded => nil,
                  :metadata => {
         "purchase_type" => "credit_pack",
             "pack_name" => "pro",
               "credits" => 5000,
         "bonus_credits" => 100,
           "price_cents" => 19900,
        "price_currency" => "USD"
    },
                      :data => {
        "payment_method_type" => "card",
                      "brand" => "Fake",
                      "last4" => 1234,
                  "exp_month" => 2,
                   "exp_year" => 2026
    },
            :stripe_account => nil,
                :created_at => 2026-02-19 10:41:40.885996000 UTC +00:00,
                :updated_at => 2026-02-19 10:41:40.885996000 UTC +00:00,
                      :type => "Pay::FakeProcessor::Charge"
}
# {
                        :id => 2138,
               :customer_id => 3201,
           :subscription_id => nil,
              :processor_id => "dKyc2_OIkOPDfhruy9rKR",
                    :amount => 99,
                  :currency => nil,
    :application_fee_amount => nil,
           :amount_refunded => nil,
                  :metadata => {
         "purchase_type" => "credit_pack",
             "pack_name" => "tiny",
               "credits" => 10,
         "bonus_credits" => 0,
           "price_cents" => 99,
        "price_currency" => "USD"
    },
                      :data => {
        "payment_method_type" => "card",
                      "brand" => "Fake",
                      "last4" => 1234,
                  "exp_month" => 2,
                   "exp_year" => 2026
    },
            :stripe_account => nil,
                :created_at => 2026-02-19 11:40:13.581796000 UTC +00:00,
                :updated_at => 2026-02-19 11:40:13.581796000 UTC +00:00,
                      :type => "Pay::FakeProcessor::Charge"
}
# {
                        :id => 2132,
               :customer_id => 3201,
           :subscription_id => nil,
              :processor_id => "vdqzzT87Wozx3HNnYvpYg",
                    :amount => 4900,
                  :currency => nil,
    :application_fee_amount => nil,
           :amount_refunded => nil,
                  :metadata => {
         "purchase_type" => "credit_pack",
             "pack_name" => "starter",
               "credits" => 1000,
         "bonus_credits" => 0,
           "price_cents" => 4900,
        "price_currency" => "USD"
    },
                      :data => {
        "payment_method_type" => "card",
                      "brand" => "Fake",
                      "last4" => 1234,
                  "exp_month" => 2,
                   "exp_year" => 2026
    },
            :stripe_account => nil,
                :created_at => 2026-02-19 06:41:58.238096000 UTC +00:00,
                :updated_at => 2026-02-19 06:41:58.238096000 UTC +00:00,
                      :type => "Pay::FakeProcessor::Charge"
}
Pay::Subscription (0) none

Model records

UsageCredits::Fulfillment (6)
# {
                          :id => 3287,
                   :wallet_id => 3206,
                 :source_type => "Pay::Charge",
                   :source_id => 2131,
    :credits_last_fulfillment => 5100,
            :fulfillment_type => "credit_pack",
           :last_fulfilled_at => 2026-02-19 06:22:17.622327000 UTC +00:00,
         :next_fulfillment_at => nil,
          :fulfillment_period => nil,
                    :stops_at => nil,
                    :metadata => {
        "purchase_charge_id" => 2131,
              "purchased_at" => "2026-02-19T06:22:17.614Z",
             "purchase_type" => "credit_pack",
                 "pack_name" => "pro",
                   "credits" => 5000,
             "bonus_credits" => 100,
               "price_cents" => 19900,
            "price_currency" => "USD"
    },
                  :created_at => 2026-02-19 06:22:17.624494000 UTC +00:00,
                  :updated_at => 2026-02-19 06:22:17.624494000 UTC +00:00
}
# {
                          :id => 3288,
                   :wallet_id => 3206,
                 :source_type => "Pay::Charge",
                   :source_id => 2132,
    :credits_last_fulfillment => 1000,
            :fulfillment_type => "credit_pack",
           :last_fulfilled_at => 2026-02-19 06:41:58.249784000 UTC +00:00,
         :next_fulfillment_at => nil,
          :fulfillment_period => nil,
                    :stops_at => nil,
                    :metadata => {
        "purchase_charge_id" => 2132,
              "purchased_at" => "2026-02-19T06:41:58.238Z",
             "purchase_type" => "credit_pack",
                 "pack_name" => "starter",
                   "credits" => 1000,
             "bonus_credits" => 0,
               "price_cents" => 4900,
            "price_currency" => "USD"
    },
                  :created_at => 2026-02-19 06:41:58.250317000 UTC +00:00,
                  :updated_at => 2026-02-19 06:41:58.250317000 UTC +00:00
}
# {
                          :id => 3289,
                   :wallet_id => 3206,
                 :source_type => "Pay::Charge",
                   :source_id => 2133,
    :credits_last_fulfillment => 5100,
            :fulfillment_type => "credit_pack",
           :last_fulfilled_at => 2026-02-19 09:22:30.856455000 UTC +00:00,
         :next_fulfillment_at => nil,
          :fulfillment_period => nil,
                    :stops_at => nil,
                    :metadata => {
        "purchase_charge_id" => 2133,
              "purchased_at" => "2026-02-19T09:22:30.846Z",
             "purchase_type" => "credit_pack",
                 "pack_name" => "pro",
                   "credits" => 5000,
             "bonus_credits" => 100,
               "price_cents" => 19900,
            "price_currency" => "USD"
    },
                  :created_at => 2026-02-19 09:22:30.857099000 UTC +00:00,
                  :updated_at => 2026-02-19 09:22:30.857099000 UTC +00:00
}
# {
                          :id => 3290,
                   :wallet_id => 3206,
                 :source_type => "Pay::Charge",
                   :source_id => 2134,
    :credits_last_fulfillment => 5100,
            :fulfillment_type => "credit_pack",
           :last_fulfilled_at => 2026-02-19 10:41:40.894296000 UTC +00:00,
         :next_fulfillment_at => nil,
          :fulfillment_period => nil,
                    :stops_at => nil,
                    :metadata => {
        "purchase_charge_id" => 2134,
              "purchased_at" => "2026-02-19T10:41:40.885Z",
             "purchase_type" => "credit_pack",
                 "pack_name" => "pro",
                   "credits" => 5000,
             "bonus_credits" => 100,
               "price_cents" => 19900,
            "price_currency" => "USD"
    },
                  :created_at => 2026-02-19 10:41:40.894976000 UTC +00:00,
                  :updated_at => 2026-02-19 10:41:40.894976000 UTC +00:00
}
# {
                          :id => 3295,
                   :wallet_id => 3206,
                 :source_type => "Pay::Charge",
                   :source_id => 2138,
    :credits_last_fulfillment => 10,
            :fulfillment_type => "credit_pack",
           :last_fulfilled_at => 2026-02-19 11:40:13.592480000 UTC +00:00,
         :next_fulfillment_at => nil,
          :fulfillment_period => nil,
                    :stops_at => nil,
                    :metadata => {
        "purchase_charge_id" => 2138,
              "purchased_at" => "2026-02-19T11:40:13.581Z",
             "purchase_type" => "credit_pack",
                 "pack_name" => "tiny",
                   "credits" => 10,
             "bonus_credits" => 0,
               "price_cents" => 99,
            "price_currency" => "USD"
    },
                  :created_at => 2026-02-19 11:40:13.593392000 UTC +00:00,
                  :updated_at => 2026-02-19 11:40:13.593392000 UTC +00:00
}
# {
                          :id => 3296,
                   :wallet_id => 3206,
                 :source_type => "Pay::Charge",
                   :source_id => 2139,
    :credits_last_fulfillment => 1000,
            :fulfillment_type => "credit_pack",
           :last_fulfilled_at => 2026-02-19 14:20:21.726696000 UTC +00:00,
         :next_fulfillment_at => nil,
          :fulfillment_period => nil,
                    :stops_at => nil,
                    :metadata => {
        "purchase_charge_id" => 2139,
              "purchased_at" => "2026-02-19T14:20:21.718Z",
             "purchase_type" => "credit_pack",
                 "pack_name" => "starter",
                   "credits" => 1000,
             "bonus_credits" => 0,
               "price_cents" => 4900,
            "price_currency" => "USD"
    },
                  :created_at => 2026-02-19 14:20:21.727232000 UTC +00:00,
                  :updated_at => 2026-02-19 14:20:21.727232000 UTC +00:00
}
UsageCredits::Transaction (8)
# {
                :id => 5404,
         :wallet_id => 3206,
            :amount => 5100,
          :category => "credit_pack_purchase",
        :expires_at => nil,
    :fulfillment_id => nil,
          :metadata => {
        "purchase_charge_id" => 2131,
              "purchased_at" => "2026-02-19T06:22:17.614Z",
         "credits_fulfilled" => true,
              "fulfilled_at" => "2026-02-19T06:22:17.618Z",
             "purchase_type" => "credit_pack",
                 "pack_name" => "pro",
                   "credits" => 5000,
             "bonus_credits" => 100,
               "price_cents" => 19900,
            "price_currency" => "USD"
    },
        :created_at => 2026-02-19 06:22:17.620229000 UTC +00:00,
        :updated_at => 2026-02-19 06:22:17.620229000 UTC +00:00
}
# {
                :id => 5405,
         :wallet_id => 3206,
            :amount => 1000,
          :category => "credit_pack_purchase",
        :expires_at => nil,
    :fulfillment_id => nil,
          :metadata => {
        "purchase_charge_id" => 2132,
              "purchased_at" => "2026-02-19T06:41:58.238Z",
         "credits_fulfilled" => true,
              "fulfilled_at" => "2026-02-19T06:41:58.246Z",
             "purchase_type" => "credit_pack",
                 "pack_name" => "starter",
                   "credits" => 1000,
             "bonus_credits" => 0,
               "price_cents" => 4900,
            "price_currency" => "USD"
    },
        :created_at => 2026-02-19 06:41:58.247680000 UTC +00:00,
        :updated_at => 2026-02-19 06:41:58.247680000 UTC +00:00
}
# {
                :id => 5406,
         :wallet_id => 3206,
            :amount => 20,
          :category => "referral_bonus",
        :expires_at => nil,
    :fulfillment_id => nil,
          :metadata => {
        "reason" => "referral"
    },
        :created_at => 2026-02-19 07:04:08.218964000 UTC +00:00,
        :updated_at => 2026-02-19 07:04:08.218964000 UTC +00:00
}
# {
                :id => 5407,
         :wallet_id => 3206,
            :amount => 10,
          :category => "bonus",
        :expires_at => 2026-02-19 07:24:45.319065000 UTC +00:00,
    :fulfillment_id => nil,
          :metadata => {
        "reason" => "expiring_bonus_credits"
    },
        :created_at => 2026-02-19 07:23:45.324188000 UTC +00:00,
        :updated_at => 2026-02-19 07:23:45.324188000 UTC +00:00
}
# {
                :id => 5408,
         :wallet_id => 3206,
            :amount => 5100,
          :category => "credit_pack_purchase",
        :expires_at => nil,
    :fulfillment_id => nil,
          :metadata => {
        "purchase_charge_id" => 2133,
              "purchased_at" => "2026-02-19T09:22:30.846Z",
         "credits_fulfilled" => true,
              "fulfilled_at" => "2026-02-19T09:22:30.851Z",
             "purchase_type" => "credit_pack",
                 "pack_name" => "pro",
                   "credits" => 5000,
             "bonus_credits" => 100,
               "price_cents" => 19900,
            "price_currency" => "USD"
    },
        :created_at => 2026-02-19 09:22:30.852595000 UTC +00:00,
        :updated_at => 2026-02-19 09:22:30.852595000 UTC +00:00
}
# {
                :id => 5409,
         :wallet_id => 3206,
            :amount => 5100,
          :category => "credit_pack_purchase",
        :expires_at => nil,
    :fulfillment_id => nil,
          :metadata => {
        "purchase_charge_id" => 2134,
              "purchased_at" => "2026-02-19T10:41:40.885Z",
         "credits_fulfilled" => true,
              "fulfilled_at" => "2026-02-19T10:41:40.891Z",
             "purchase_type" => "credit_pack",
                 "pack_name" => "pro",
                   "credits" => 5000,
             "bonus_credits" => 100,
               "price_cents" => 19900,
            "price_currency" => "USD"
    },
        :created_at => 2026-02-19 10:41:40.892566000 UTC +00:00,
        :updated_at => 2026-02-19 10:41:40.892566000 UTC +00:00
}
# {
                :id => 5417,
         :wallet_id => 3206,
            :amount => 10,
          :category => "credit_pack_purchase",
        :expires_at => nil,
    :fulfillment_id => nil,
          :metadata => {
        "purchase_charge_id" => 2138,
              "purchased_at" => "2026-02-19T11:40:13.581Z",
         "credits_fulfilled" => true,
              "fulfilled_at" => "2026-02-19T11:40:13.588Z",
             "purchase_type" => "credit_pack",
                 "pack_name" => "tiny",
                   "credits" => 10,
             "bonus_credits" => 0,
               "price_cents" => 99,
            "price_currency" => "USD"
    },
        :created_at => 2026-02-19 11:40:13.589883000 UTC +00:00,
        :updated_at => 2026-02-19 11:40:13.589883000 UTC +00:00
}
# {
                :id => 5418,
         :wallet_id => 3206,
            :amount => 1000,
          :category => "credit_pack_purchase",
        :expires_at => nil,
    :fulfillment_id => nil,
          :metadata => {
        "purchase_charge_id" => 2139,
              "purchased_at" => "2026-02-19T14:20:21.718Z",
         "credits_fulfilled" => true,
              "fulfilled_at" => "2026-02-19T14:20:21.723Z",
             "purchase_type" => "credit_pack",
                 "pack_name" => "starter",
                   "credits" => 1000,
             "bonus_credits" => 0,
               "price_cents" => 4900,
            "price_currency" => "USD"
    },
        :created_at => 2026-02-19 14:20:21.724910000 UTC +00:00,
        :updated_at => 2026-02-19 14:20:21.724910000 UTC +00:00
}
UsageCredits::Allocation (0) none

Audit trail

Audit trail
[
    [0] [
        [0] "credit_pack_purchase",
        [1] 5100,
        [2] {
            "purchase_charge_id" => 2131,
                  "purchased_at" => "2026-02-19T06:22:17.614Z",
             "credits_fulfilled" => true,
                  "fulfilled_at" => "2026-02-19T06:22:17.618Z",
                 "purchase_type" => "credit_pack",
                     "pack_name" => "pro",
                       "credits" => 5000,
                 "bonus_credits" => 100,
                   "price_cents" => 19900,
                "price_currency" => "USD"
        }
    ],
    [1] [
        [0] "credit_pack_purchase",
        [1] 1000,
        [2] {
            "purchase_charge_id" => 2132,
                  "purchased_at" => "2026-02-19T06:41:58.238Z",
             "credits_fulfilled" => true,
                  "fulfilled_at" => "2026-02-19T06:41:58.246Z",
                 "purchase_type" => "credit_pack",
                     "pack_name" => "starter",
                       "credits" => 1000,
                 "bonus_credits" => 0,
                   "price_cents" => 4900,
                "price_currency" => "USD"
        }
    ],
    [2] [
        [0] "referral_bonus",
        [1] 20,
        [2] {
            "reason" => "referral"
        }
    ],
    [3] [
        [0] "bonus",
        [1] 10,
        [2] {
            "reason" => "expiring_bonus_credits"
        }
    ],
    [4] [
        [0] "credit_pack_purchase",
        [1] 5100,
        [2] {
            "purchase_charge_id" => 2133,
                  "purchased_at" => "2026-02-19T09:22:30.846Z",
             "credits_fulfilled" => true,
                  "fulfilled_at" => "2026-02-19T09:22:30.851Z",
                 "purchase_type" => "credit_pack",
                     "pack_name" => "pro",
                       "credits" => 5000,
                 "bonus_credits" => 100,
                   "price_cents" => 19900,
                "price_currency" => "USD"
        }
    ],
    [5] [
        [0] "credit_pack_purchase",
        [1] 5100,
        [2] {
            "purchase_charge_id" => 2134,
                  "purchased_at" => "2026-02-19T10:41:40.885Z",
             "credits_fulfilled" => true,
                  "fulfilled_at" => "2026-02-19T10:41:40.891Z",
                 "purchase_type" => "credit_pack",
                     "pack_name" => "pro",
                       "credits" => 5000,
                 "bonus_credits" => 100,
                   "price_cents" => 19900,
                "price_currency" => "USD"
        }
    ],
    [6] [
        [0] "credit_pack_purchase",
        [1] 10,
        [2] {
            "purchase_charge_id" => 2138,
                  "purchased_at" => "2026-02-19T11:40:13.581Z",
             "credits_fulfilled" => true,
                  "fulfilled_at" => "2026-02-19T11:40:13.588Z",
                 "purchase_type" => "credit_pack",
                     "pack_name" => "tiny",
                       "credits" => 10,
                 "bonus_credits" => 0,
                   "price_cents" => 99,
                "price_currency" => "USD"
        }
    ],
    [7] [
        [0] "credit_pack_purchase",
        [1] 1000,
        [2] {
            "purchase_charge_id" => 2139,
                  "purchased_at" => "2026-02-19T14:20:21.718Z",
             "credits_fulfilled" => true,
                  "fulfilled_at" => "2026-02-19T14:20:21.723Z",
                 "purchase_type" => "credit_pack",
                     "pack_name" => "starter",
                       "credits" => 1000,
                 "bonus_credits" => 0,
                   "price_cents" => 4900,
                "price_currency" => "USD"
        }
    ]
]

Loaded config

Literal, raw config file
      # frozen_string_literal: true

UsageCredits.configure do |config|

  # Default test operations
  operation :small_operation do
    costs 1.credit
  end

  # operation :test_operation do
  #   costs 10.credits + 1.credits_per(:mb)
  #   validate ->(params) { params[:size].to_i <= 100.megabytes }, "File too large (max: 100 MB)"
  # end

  operation :prohibitive_operation do
    costs 1_000_000.credits
  end

  operation :this_operation_will_always_fail do
    costs 42.credits
  end


  # Define test credit packs
  credit_pack :tiny do
    gives 10.credits
    costs 99.cents
  end

  credit_pack :starter do
    gives 1000.credits
    costs 49.dollars
  end

  credit_pack :pro do
    gives 5000.credits
    bonus 100.credits  # Optional: give bonus credits
    costs 199.dollars
    currency :usd
  end


  # Define subscriptions
  subscription_plan :test_plan do
    processor_plan(:fake_processor, "abcdef123456")
    gives 10.credits.every(:month)
    unused_credits :expire
  end

  # Alert when balance drops below 100 credits
  # Set to nil to disable low balance alerts
  # config.low_balance_threshold = 20.credits

  # # Handle low credit balance alerts
  # config.on_low_balance do |user|
  #   # Send notification to user when their balance drops below the threshold
  #   ApplicationMailer.generic_email(to: user.email, body: "Heads up! You're low on credits.", subject: "Low credits alert").deliver_now
  # end

end

    
Credit-consuming operations
Credit packs
Credit subscription plans
Rest of the config
      {
  "default_currency": "usd",
  "rounding_strategy": "ceil",
  "credit_formatter": {},
  "fulfillment_grace_period": 300,
  "allow_negative_balance": false,
  "low_balance_threshold": null,
  "low_balance_callback": null
}