惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

IntelliJ IDEA : IntelliJ IDEA – the Leading IDE for Professional Development in Java and Kotlin | The JetBrains Blog
IntelliJ IDEA : IntelliJ IDEA – the Leading IDE for Professional Development in Java and Kotlin | The JetBrains Blog
C
CXSECURITY Database RSS Feed - CXSecurity.com
博客园_首页
H
Hackread – Cybersecurity News, Data Breaches, AI and More
T
ThreatConnect
钛媒体:引领未来商业与生活新知
钛媒体:引领未来商业与生活新知
博客园 - 聂微东
H
Help Net Security
T
Threat Research - Cisco Blogs
Blog — PlanetScale
Blog — PlanetScale
A
Arctic Wolf
G
Google Developers Blog
量子位
U
Unit 42
I
InfoQ
V
V2EX
F
Fox-IT International blog
P
Privacy & Cybersecurity Law Blog
V
Visual Studio Blog
J
Java Code Geeks
大猫的无限游戏
大猫的无限游戏
C
CERT Recently Published Vulnerability Notes
博客园 - 三生石上(FineUI控件)
T
The Exploit Database - CXSecurity.com
T
Tailwind CSS Blog
SecWiki News
SecWiki News
Know Your Adversary
Know Your Adversary
MyScale Blog
MyScale Blog
宝玉的分享
宝玉的分享
The Hacker News
The Hacker News
Project Zero
Project Zero
Application and Cybersecurity Blog
Application and Cybersecurity Blog
月光博客
月光博客
Recent Commits to openclaw:main
Recent Commits to openclaw:main
奇客Solidot–传递最新科技情报
奇客Solidot–传递最新科技情报
G
GRAHAM CLULEY
C
Cisco Blogs
I
Intezer
Simon Willison's Weblog
Simon Willison's Weblog
O
OpenAI News
Recorded Future
Recorded Future
T
Tenable Blog
W
WeLiveSecurity
腾讯CDC
Stack Overflow Blog
Stack Overflow Blog
T
The Blog of Author Tim Ferriss
www.infosecurity-magazine.com
www.infosecurity-magazine.com
D
Docker
C
Cybersecurity and Infrastructure Security Agency CISA
PCI Perspectives
PCI Perspectives

Все публикации подряд на Хабре

Способна ли нейробиология подтвердить наличие сознания у искусственного интеллекта? И почему нет? Цифровой рубль как инструмент сбережения: миф или реальность? Зарядка для джависта Cursor пишет вам unit‑тесты за минуту. 5 паттернов, на которых эти тесты пропустят любой баг Щука на весло: почему случайная выручка опасна для молодого продукта Как я делал VPN-сервис в 2026 году Из разработчика в системные аналитики: практический путь в профессию Почему ваш сайт не продаёт, хотя SEO-шник клянётся, что всё хорошо Сейчас никому не нужны технологии, все обсуждают только ИИ От папки с созвонами до 5K+ юзеров: как pet-проект «для себя» встретился с реальными пользователями Отстаём своим путём Лучшие приложения для изучения иностранных языков: что выбрать в 2026 году Что не нужно знать топ менеджеру, что бы провалить внедрение AI | ИИ Одна строчка .Result роняет ваш ASP.NET Core при CPU 8 %: разбор hill-climbing в .NET 9 Миграции в Go-проекте: PostgreSQL в Docker и goose на практике Что такое OpSec, если углубится Российская компания на 50 человек платит 350 000 ₽ в год за софт, который дублирует сам себя Локализовать нельзя ошибиться. Как работает локализация в автономном транспорте и почему это — самая сложная задача. 1/2 Inside AI Meetup — как это было? Делимся записями докладов, фото и атмосферой Делаем сайт из картинки в нейронке Один простой механизм управляет практически всем в игре Cities: Skylines Встраиваемая векторная БД для RAG на .NET 8: когда внешние сервисы избыточны Gemini-3.5-flash догнал GPT-5.5 на 97/S и в 2.5× дешевле. Но главное — китайцы выигрывают по цене и качеству JavaScript. Работа с большими файлами в браузере. Часть 2/2: Создание 5Gb файлов в браузере Как визуализировать задачи и зависимости в проекте: обзор трекеров, Gantt, графов и whiteboard-инструментов Как команда становится AI Native: методология из 4 этапов Как дебажить distroless-контейнер в Kubernetes без shell: ephemeral containers на практике ИИ не автоматизировал разработчиков. Он сделал кое-что хуже Как оплатить обучение за границей из России в 2026 году: способы, цены, рейтинги Сложный поиск альтернативных частиц Хиггса Тест батареек Camelion Plus Почему компании строят свои конструкторы баннеров: разбор паттерна, который никто не называет Структурированное логирование и трейсинг в Node.js: @cleverbrush/log и @cleverbrush/otel Странные образования на поверхности Венеры ставят в тупик планетологов Шифрование на уровне протокола Пять самых крупных ошибок, которые допускают компании при внедрении SRE Приложения для Битрикс24, которые реально экономят время Анатомия Claude Code. Первичный анализ и наполнение контекста Как изменились требования к разработчикам в эпоху AI: опыт техлида Распродажа «Большой Пятерки» в PlayStation – Days Of Play GIT: Как ломать и чинить историю правильно Разбираемся в ML без воды: от базы до Attention. Часть 6: Логистическая регрессия Решето как гипотетический контейнер для жидких субстанций Лучшие нейросети для генерации изображений — как создать картинку с помощью ИИ в 2026 году Волны гасят ветер: во что упирается развитие ИИ в теории длинных волн Кондратьева Почему на самом деле нельзя делить на ноль? Физический и аксиоматический подходы Zero Trust для подрядного доступа: четыре слоя Identity, Device, Access и Monitoring Zero Trust для подрядного доступа: четыре слоя Identity, Device, Access и Monitoring Базовый командный runtime для терминальных AI-агентов Спектр. Контекст создания, трудности, боли и победы Задолбал нейрослоп: честный разбор, почему мы не можем без него C3D Converter: Plug and Play Почему технические директора не проходят в CIO: портфель проектов против навыка «докрутить» SaaS умирает? Я сравнил 8 публикаций Q1 2026 с тем, что вижу внутри Kaiten Взрослый BIM для детского сада на 230 мест: крупнейший застройщик Черноземья ОДСК – сделал комплексный проект в nanoCAD Privacy-by-design: что наш edge не пишет на диск и почему это сложнее, чем кажется Civilization VII: что изменилось в механике смены эпох после патча 1.4.0 Готовые решения для интернет-магазина на 1С-Битрикс: разбираю рынок изнутри На РОИ появились инициативы с требованием ограничения полномочий РКН и блокировок Фаундер написал 15 страниц про рынок и поднял на этом $10M Формула интегрирования по частям с точки зрения дифференциальной геометрии Plan-tango: как я перестал гонять план между Claude Code и Codex руками Динозавры в проде: сколько лет языкам программирования и кто до сих пор зарабатывает на «мёртвых» Как стать postgres в чужом облаке: краш-тест безопасности управляемых БД Погружение в новый проект: как не потерять месяц жизни Простой гайд по Kling Motion Control от А до Я Семантический слой: архитектура, подходы и роль в эпоху AI-аналитики Гоняться за оптовиками и чуть не закрыться, придумать «стартовый набор новичка» и удвоить выручку НЕкурс про разработку безопасного программного обеспечения (РБПО) Теология возможных миров. Есть ли боги в мультивселенной, или мультивселенная и есть Бог? Что делать, если не прошли переаккредитацию ИТ-компании в 2026 году: пошаговый план действий Нейросеть для работы с текстом — как генерировать чистый и уникальный текст для студентов Прокачать SQLite и сократить векторы в видеоформате — открытые инструменты для работы с эмбеддингами Киберзадачи в сеттинге Minecraft. Школьники в финале ВсоШ по инфобезу Windows 11 будет работать быстрее на всех компьютерах. Теперь официально Кэширование в Symfony: как мы сломали авторизацию и починили ее через Lock Стажеры uAcademy*. Опыт кураторства дипломов: почему стажировок недостаточно Команда выросла, методы — остались «Ошибка выжившего» на примере спортсменов Испытание временем — как тестировать цифровой двойник, если физического объекта ещё не существует Как обычный кухонный таймер на ESP32 превратился в домашний центр уведомлений Как мы научили СХД TATLIN.OBJECT мигрировать данные из S3-хранилища MinIO Онлайн-приключение для IT-команд, как альтернатива корпоративу в Zoom Экскурсия по «зоопарку» сетевого трафика: топ-10 аномалий внутри вашего периметра Книга: «System Design. Проектирование мобильных систем. Подготовка к сложному интервью» Критическое мышление руководителя: как один красивый слайд может привести к дорогой ошибке Ecommerce на Laravel, или как мы собрали headless-слой для фронтов (6 часть) Обновление macOS для инженеров поддержки Делаем ностальгический фильмоскоп на Raspberry Pi Zero 2 W От баз данных до инструментов для ИИ-экосистем: проекты, которые получили гранты Yandex Open Source Больше, чем просто безопасность, или Зачем контролировать зависимости Тот неловкий момент, когда письмо от Джованни из Швейцарии не оказалось обманом Почему AAA-игры проваливаются? Разбираем примеры Как запустить 3D-приложение на сервере без GPU: от SwiftShader до WARP Благоустраиваем Firefox: встроенный VPN Современный Angular: Заменяем жизненные циклы на сигналы HR-бот на базе RAG: архитектура корпоративной базы знаний для ресторанного холдинга Почему ИИ не заменит аналитика при подготовке технического задания InSales без пушей: как бесплатно перенести уведомления о заказах в Telegram на Yandex Cloud Serverless Александрийская библиотека: краткая история античной системы хранения
Написать биллинг за неделю: туториал на два года
lidia_zakhar · 2026-05-29 · via Все публикации подряд на Хабре

Написать биллинг за неделю: туториал на два года

Уровень сложностиПростой

Время на прочтение5 мин

Охват и читатели2

Туториал

Меня зовут Лидия, я основатель LBX биллинга для SaaS-продуктов. Мы провели десятки внедрений: от команд, которые только запускают монетизацию, до сервисов с несколькими тысячами активных подписчиков. Видели, как команды наступают на одни и те же грабли. Некоторые с хрустом. Вот четыре ситуации, которые встречаются чаще всего.

Грабли №1. «У нас простая логика, напишем сами»

ЮKassa и CloudPayments поддерживают рекуррентные платежи. Настраивается периодичность, дата списания в месяце, сумма, количество повторений. На первый взгляд есть всё необходимое.

Но рекуррент — это только механизм списания. Он не знает, что клиент перешёл на другой тариф в середине месяца и сумма должна измениться. Не знает, что один пользователь подключился первого числа, другой семнадцатого, и периоды у них разные. Не знает, что делать, если платёж не прошёл: ждать три дня, заблокировать доступ или дать льготный период. Не знает, что клиент попросил паузу на месяц.

Всю эту логику нужно строить поверх платёжной системы самостоятельно.

Льготный период уже отдельная история. Заблокировать юзера немедленно при первой просрочке, значит рисковать его потерять. Карта могла устареть, платёж завис на стороне банка, бухгалтер был в отпуске. 

Грабли №2. «Пока клиентов немного — справляемся вручную»

На старте большинство команд ведут учет подписок и оплат вручную. Счёт сформировали в банке или 1С, отправили клиенту, дождались оплаты, отметили в таблице. Двадцать клиентов занимает пару часов в неделю. Раздражает, но еще терпимо.

При пятидесяти клиентах час превращается в несколько дней. Счёт нужно не только сформировать, но и отследить — оплатили или нет. Платежи разносить вручную: открыть банк, найти входящий перевод, сопоставить с клиентом по ИНН, обновить статус. Если платёж пришёл с некорректным назначением или частично, то разбирать отдельно.

Параллельно копятся исключения. Один клиент договорился о скидке, бухгалтер помнит, что ему выставлять меньше. Другой платит не первого, а двадцатого, так договорились при подключении. Третьему по какой-то причине счёт уходит на квартал вперёд. Каждое такое исключение живёт в голове конкретного человека или в заметке в Notion. Когда этот человек уходит в отпуск или увольняется, исключения теряются. Клиент получает счёт на неправильную сумму. Или не получает вовсе.

Грабли №3. «Автоматизировали — и думали, что готово»

Команда дорастает до автоматизации: счета генерируются и отправляются сами, входящие платежи разносятся по ИНН автоматически. Ощущение, что проект биллинга наконец закрыт.

Но к этому моменту исключения переехали из голов людей в код. Для одного клиента в базе захардкожена спеццена. Для другого — отдельное условие в логике расчёта периода. Для третьего написан отдельный обработчик, про который знает только тот разработчик, который его писал.

Формально, это уже не ручная работа. Но это хрупкая конструкция: изменить условия для пользователя, значит идти в код. Добавить новый тип исключения, ещё один if где-то в обработчике. Разобраться, почему конкретному клиенту выставляется именно такая сумма уже становится задачей разработчика, а не бухгалтера.

Грабли №4. «Добавим позже»

Хотим запустить новый тариф — нужно писать код.
Хотим добавить опцию — нужно писать код.
Хотим скорректировать цену — нужно писать код, тестировать, выкатывать.

Разработчики заняты продуктом. Новый тариф откладывается на следующий спринт. Потом ещё на один. «Добавим позже» в большинстве случаев означает «никогда».

То же самое с архитектурными решениями по биллингу. Переделать модель данных на работающей системе с живыми пользователями это один из самых болезненных технических проектов. Не потому что сложно технически, а потому что цена ошибки высока. Каждый раз, когда команда говорит «сделаем нормально позже», она повышает стоимость этого «позже».

Собственно туториал

Ни одни из этих граблей не уникальны - это стандартная эволюция биллинга в большинстве SaaS команд. Рекуррент в платёжной системе, таблица с оплатами, исключения в коде проходят почти все. Отличается только момент, когда это начинает мешать.

Хорошая новость: грабли стандартные, и о них можно знать заранее.

Разделить тарифный план и условия конкретного клиента. Публичный тариф и то, на что фактически подписан пользователь, разные сущности в модели данных. Скидка, индивидуальная цена, нестандартный расчётный период хранятся в клиентском договоре, а не как исключение в логике тарифа. Это делает все отклонения от стандарта явными и видимыми без чтения кода.

Хранить историю событий, а не только текущее состояние. Не «клиент на тарифе PRO», а «клиент перешёл на тариф PRO 15 марта в 14:32». Без событийной истории перерасчёт за прошлый период невозможен, разбор спорной ситуации невозможен, аудит невозможен. Добавить это задним числом — отдельное приключение.

Не хранить расчётный период как фиксированный объект. Он должен вычисляться на основе истории: дата подключения, смены тарифа, паузы, апгрейды в середине цикла. Если есть полная история, то период всегда можно пересчитать корректно. Если нет, то при любом нестандартном кейсе система начинает выдавать неверные суммы, которые сложно объяснить клиенту и ещё сложнее исправить задним числом.

Предусмотреть состояния подписки. Минимальный набор: активна, ожидает оплаты, заблокирована, отменена. Каждое состояние имеет свою логику доступа, начислений и автоматических действий системы. Состояние «активен / неактивен» это не покрывает.

Обеспечить идемпотентность денежных операций. Одно и то же списание не должно выполняться дважды при сетевом сбое или перезапуске. Каждая транзакция должна иметь уникальный идентификатор, повторный вызов возвращать тот же результат без повторного списания. Это базовое требование к надёжности, которое на старте кажется излишним, но его лучше заложить сразу.

Управление тарифами без изменений кода. Добавление тарифа, изменение цены, запуск новой опции, всё это должно настраиваться через интерфейс, а не через задачу разработчику. Это снимает зависимость коммерческих решений от бэклога и позволяет тестировать ценообразование в спокойном режиме.

Разделить выставление счёта и факт оплаты. Счёт формируется в начале расчётного периода. Оплата фиксируется, когда деньги фактически поступили. Между ними есть состояние ожидания с собственной логикой и сроками. Без этого разделения корректный учёт задолженностей и работа с просрочками существенно усложняются.

Итог

Биллинг редко попадает в список приоритетных задач. Есть продукт, есть метрики, есть задачи, которые видны пользователям. Биллинг не в этом списке, пока не начинает мешать. Именно поэтому архитектурные решения по нему так часто откладываются, и именно поэтому потом возвращаются в виде технического долга с высоким приоритетом.

Я ни разу не встречала команду, которая пожалела, что нормально заложила биллинг с самого начала. Зато команд, которые пожалели об обратном, хватает.

А как у вас? Писали ли биллинг сами? интересно услышать, что оказалось неожиданно сложным и на каком этапе стало понятно, что задача переросла первоначальную оценку в пару спринтов. Или, наоборот, написали и всё до сих пор работает.