У Ruby часто згадується термін PORO (Plain Old Ruby Object), але що це таке і чому він важливий?
PORO: простими словами
PORO — це просто звичайний об'єкт Ruby. Він не успадковується від специфічних бібліотек або класів фреймворків, таких як ActiveRecord, ActionController чи інші. PORO — це об'єкт, який:
- Легкий і мінімалістичний.
- Має лише ті методи, які потрібні для вирішення конкретної задачі.
- Не залежить від зовнішніх фреймворків чи бібліотек.
Переваги PORO
-
Простота: PORO дозволяє зберігати код чистим і зрозумілим.
-
Гнучкість: Ви можете використовувати PORO у будь-якому Ruby-проєкті без прив'язки до фреймворків.
-
Покриття тестами: PORO легко тестувати, оскільки вони не мають залежностей від складної екосистеми фреймворків.
Коли використовувати PORO?
- Коли вам потрібно реалізувати специфічну бізнес-логіку, яка не підходить до моделей, контролерів або хелперів.
- Коли ви хочете створити маленькі, самостійні класи.
- Коли вам потрібно зменшити залежність від фреймворків і зробити код більш модульним.
PORO важливий для гнучкого дизайну, оскільки дозволяє створювати незалежні компоненти, які не залежать від фреймворків, таких як Rails. Наприклад, уявіть бізнес-логіку для розрахунку зарплати. Якщо вона прив’язана до ActiveRecord, її використання залежить від доступу до бази даних. Це ускладнює тестування та обмежує можливість повторного використання в інших контекстах, таких як API або мікросервіси.
Завдяки PORO ви можете винести логіку в окремий об’єкт, який працює лише з вхідними даними та не має прив’язок до зовнішніх залежностей. Такий підхід спрощує тестування, підвищує модульність коду та дозволяє легко адаптувати його для нових потреб. Це робить PORO ефективним інструментом для створення чистої архітектури додатків.
Використання PORO в Rails
Уявімо, що у вашому Rails-проекті є потреба обчислити загальну вартість замовлення:
class OrderTotalCalculator
def initialize(order)
@order = order
end
def total
@order.items.sum { |item| item.price * item.quantity }
end
end
# Використання:
order = Order.find(1) # Приклад ActiveRecord моделі
calculator = OrderTotalCalculator.new(order)
puts calculator.total
У цьому прикладі OrderTotalCalculator є PORO. Він відокремлений від моделей і контролерів Rails, що робить його незалежним і легким для тестування.
Як тестувати PORO?
Завдяки своїй простоті, PORO легко тестувати за допомогою RSpec чи MiniTest:
RSpec.describe DiscountCalculator do
it 'розраховує правильну ціну зі знижкою' do
calculator = DiscountCalculator.new(100, 10)
expect(calculator.final_price).to eq(90.0)
end
end
PORO — це основа чистого і простого дизайну в Ruby. Вони (прості об'єкти без залежностей) допомагають створювати гнучкі, незалежні компоненти, які легко розширювати, підтримувати та тестувати.