====== Рівень абстракції NyanORM ======
===== Навіщо це все? =====
{{:nyanorm.gif? |}}
Робить НЯКУ з вашою базою. І надає дуже зручні засоби, щоб ви докладали мінімум зусиль і мозкових звивин при роботі з БД Ubilling. Ви зможете описувати ваші моделі даних, і роботу з ними, більше не малюючи в ручну SELECT `something` FROM `tablename`... та інші остогидлі вам конструкції. Також NyanORM візьме на себе всю вашу рутинну роботу з попередньої підготовки і трансформації ваших даних. Завданням NyanORM **не бути** всеосяжним монстром, який реалізує всі можливі кейси роботи з вашими даними. Завданням NyanORM є - бути простим та зручним. З самого початку, він задумувався для того, щоб скоротити обсяги рутинного, нудного й однотипного коду, з якого складається робота із сутностями в 90-95% прикладних модулів. Власне тільки ці кейси він і повинен перекривати, не накладаючи на вас жодних додаткових обмежень.
===== Особливості =====
Чому саме NyanORM, а не будь-яка інша стороння бібліотека з приставкою ORM у назві? А з таких причин:
* Не накладає жодних обмежень на іменування табличок БД.
* Не диктує жодних умов щодо полів і вмісту цих табличок.
* Дозволяє без проблем працювати з уже наявними структурами даних.
* Дозволяє міксувати код як з його використанням так і в старому стилі.
* Не вимагає довгого і клопіткого опису моделей, десь в окремому місці, перед тим як почати працювати.
* Ідеальний для надшвидкого прототипування бізнес-логіки прикладних модулів.
* Доступний у будь-який момент і завжди з будь-якого вашого модулю.
* Написаний у наскільки це можливо мінімалістичному стилі з максимальним використанням наявних механік Ubilling.
* Мінімізує оверхеди за пам'яттю і коллбеками. Але це не точно.
* Дозволяє наслідуванням модифікувати як завгодно будь-яку модель і методи роботи з нею, взагалі ніяк себе не обмежуючи.
* Не виставляє жодних вимог і засобів щодо фільтрації даних. Ви можете використовувати взагалі що захочете.
* Працює однаково відмінно на PHP5 і PHP7 та на PHP8. Легасі так.
* Максимально очевидний синтаксис і механіки вимагають для свого розуміння мінімального рівня IQ.
* Зашкалюючий рівень каваю.
На жаль, ми не знайшли жодного наявного рішення, що відповідає **всім** особливостям/вимогам вище. Це заощадило б нам дуже багато зусиль і часу, але ні. Тому NyanORM.
\\
\\
Якщо вам буде так спокійніше, то ми вдумливо подивилися на те, як працюють Eloquent, Hibernate, Doctrine, Propel, RedBean... і зробили навпаки.
===== Це обов'язково використовувати? =====
**Ні! ** Ніхто вас ні в чому не обмежує і не примушує використовувати будь-що таке специфічне у вашому новому коді, і тим паче навіть не натякає, що ви маєте переписати весь ваш старий код з використанням цього об'єкта. Можливо, вам просто сподобається, який вигляд може мати ваш код без ручного збирання запитів до БД, і з використанням високорівневої абстракції.
===== З чого почати? =====
Для прикладу, давайте розглянемо дуже типовий кейс. Нам потрібно отримати, наприклад, платежі за поточний рік, із сумою більше нуля і типом платежів "готівка". Також ми хочемо мати їх в асоціативному масиві, де ключем буде id платежу. Як це б виглядало в нашій старій реальності:
$query = "SELECT * from `payments` WHERE `summ`>'0' AND `date` LIKE '" . curyear() . "-%' AND `cashtypeid`='1'";
$all = simple_queryall($query);
$rawPayments = array();
if (!empty($all)) {
foreach ($all as $io => $each) {
$rawPayments[$each['id']] = $each;
}
}
debarr($rawPayments);
Правда знайома конструкція? Остогидло, правда ж? А тепер давайте теж саме, але красиво. Для початку нам потрібна модель для таблички payments. Для цього ми або успадковуємо базову модель, використовуючи ім'я таблички як ім'я класу:
class payments extends NyanORM {}
$payments = new payments();
або робимо те саме з використанням чорної магії
$payments = new nya_payments(); //воу, тут відбувається магія ;)
Так, усе після префіксу **nya_** буде розгорнуто в ім'я таблички as is і для неї буде автоматично згенеровано модель.
отримуємо всі записи
$allPayments=$payments->getAll(); //отримуємо всі записи з моделі
Так. Це типу моделька для таблички payments у ловеркейсі. Також ви можете максимально прямолінійно виставити ім'я таблички, з якою ми будемо працювати через конструктор класу. Наприклад так:
$bablo=new NyanORM('payments');
debarr($bablo->getAll());
Ми щось відволіклися. Коротше модель у нас є. Що там із нашим початковим планом щодо вибірки цілком конкретних платежів? А ось що:
class payments extends NyanORM { } // створюємо модель наслідуванням, оскільки це краще дружить з автокомплітом деяких IDE
$payments = new payments(); // створюємо об'єкт моделі
$payments->where('summ', '>', '0'); //тут і далі вішаємо наші умови
$payments->where('date', 'LIKE', curyear() . '%');
$payments->where('cashtypeid', '=', '1'); // так, це готівка
$rawPayments = $payments->getAll('id'); // автоматичне групування результату по id
debarr($rawPayments);
Давайте ще раз, на прикладі, але з чимось іншим. Нехай це будуть світчі. І нехай ми хочемо просто отримати всі світчі.
$switches = new nya_switches();
$allSwitches = $switches->getAll();
ну або якось так без чорної магії, в лоб:
$switches = new NyanORM('switches');
$allSwitches = $switches->getAll();
Куди вже простіше?
===== Про параметри моделей =====
Як можна було помітити, для встановлення параметрів наших запитів до моделей ми можемо використовувати метод where() з трьома досить очевидними параметрами. Також досить очевидно, що кілька where(), що йдуть послідовно, будуть збережені в кумулятивних структурах і надалі інтерпретовані як AND. А що якщо ми раптово захочемо крім платежів із готівкою отримувати їх також із типом 4 (нехай це будуть якісь штуки із самообслуговування)? Очевидно, що ми хочемо умову OR. А як її зробити? А дуже просто.
$payments->where('cashtypeid', '=', '1');
$payments->orWhere('cashtypeid','=','4'); //Ага, все настільки очевидно
Також якщо нам дуже захочеться пописати шматочки запитів руками, згадавши старе, або якщо хочеться зробити щось таке "таке", чого з коробки не зрозуміло, як зробити, за допомогою конструктора запитів NyanORM, тоді ви можете використовувати whereRaw('expression')/orWhereRaw('expression'), наприклад так:
$payments = new nya_payments();
$payments->whereRaw("`summ`>'0' AND `date` LIKE '" . curyear() . "-%' AND `cashtypeid`='1'");
$rawPayments = $payments->getAll('id');
debarr($rawPayments);
Що дасть нам цілком собі ідентичний результат. Але не красиво ж, правда?
===== Про очищення параметрів =====
Слід, до речі, зауважити, що після виконання методів типу getAll(), delete() і їм подібних. Усі раніше встановлені вами параметри моделей, як-от where, limit, order, data будуть скинуті. Це зроблено з метою безпеки. Наприклад ось вибираєте ви ці платежі, розглядаєте їх, а потім, уже забувши, що ви це робили кількома екранами коду вище, ви вирішили з якоїсь причини зробити delete(). І все. Вам результат не сподобався. Саме тому за замовчуванням відбувається автоматичне очищення цих параметрів. Метод delete(), наприклад, також своєю чергою нічого не робить без зазначених явно where і кидається на вас винятками.
\\
Якщо з якоїсь причини, ви не хочете, щоб відбувалося автоматичне очищення параметрів моделей, ви можете встановити параметр $flushParams цих очищувальних методів у значення false.
\\
Також у будь-який момент ви можете самостійно очистити стан будь-яких параметрів конкретної моделі, використавши відповідний сеттер з усіма порожніми параметрами.
$payments->where();
$payments->limit();
$payments->orderBy();
$payments->data();
$payments->selectable();
Помітили, як ненав'язливо ми вам показали, які ще параметри (кумулятивні структури) у моделей бувають? Так? ;)
===== Видалення даних =====
Ви не повірите. Усе настільки ж лінійно і просто. Давайте будемо видаляти запис із таблички abstractdevices з id 666?
$devices = new nya_abstractdevices();
$devices->where('id','=','666');
$devices->delete();
===== Кумулятивна структура data() =====
Кумулятивна структура **data** призначена для зберігання даних, які будуть надалі використані під час виклику методів **create()** або **save()**. Власне має вона лише два параметри, а саме **field** та **value**. Досить не важко здогадатися як її використовувати:
$object->data('somefield', 'new value');
$object->data('anotherfield', 'це теж типу якісь дані');
===== Створення та зміна записів =====
Пам'ятаєте кумулятивну структуру **data()**? Вона нам знадобиться для створення записів у моделі або зміни наявних. Давайте створимо новий запис приблизно для такої таблички:
CREATE TABLE IF NOT EXISTS `someobjects` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`name` varchar(255) NOT NULL,
`text` text,
PRIMARY KEY (`id`)
) ENGINE=MyISAM DEFAULT CHARSET=utf8 AUTO_INCREMENT=1;
Усе дуже прямолінійно.
$object = new nya_someobjects();
$object->data('name', 'а це типу ім`я');
$object->data('text', 'а це якби текст запису');
$object->create();
Зауважте, ми не вказували ручками NULL для автоінкрементного поля id, як так? А так, що у методу **create()** за замовчуванням встановлений параметр $autoAiId=true, який робить це неявно. Якщо у вашій табличці немає автоінкрементного поля `id` або іншого подібного primary key, ви маєте встановити цей параметр у false. Власне ім'я поля головного ключа таблички ви завжди можете перепризначити за допомогою наслідування. Він міститься в протектед проперті **defaultPk**.
\\
\\
Окей, запис створити ми створили, а як отримати його id? Для цього є зручний метод **getLastId()**, який отримує останній **defaultPk** з таблички.
Ось як це працює:
deb($object->getLastId()); // ой... повертає 15
Окей, припустимо, ми раптово захотіли тепер змінити всі або якесь конкретне із полів у цій табличці. як бути? Усе точно так само, як і з **create()** тільки за допомогою **save()**, але тепер нам ще знадобитися **where()**. Припустимо ми будемо редагувати останній запис у цій табличці:
$idToModify=$object->getLastId();
$object->data('text', 'ого, це ж нове значення для text!');
$object->where('id', '=', $idToModify);
$object->save();
===== Увімкнення режиму відлагодження =====
Ми знаємо. Ви звикли використовувати для налагодження ваших модулів усілякі print_r/debarr. Усе, відвикаємо. Тепер можна легко і невимушено увімкнути режим налагодження або глибокого налагодження і отримати нормальний лог і виведення всього, що відбувається з моделлю.
$model->setDebug(true);
Усе, тепер усі запити до БД виводитимуться прямо у ваш стандартний в'ю, а також записуватимуться разом із часом у дебаг-лог, який ви зможете дивитися реалтайм методом
$ tail -F exports/nyanorm.log
Також вам може захотітися врубити режим глибокого відлагодження. Тоді в цей же лог, буде дампитися також стан всієї моделі цілком на кожен пчих. Робиться це так:
$model->setDebug(true,true);
===== Про виключення =====
Якщо ви зовсім знахабнієте від вседозволеності NyanORM вам в обличчя можуть бути викинуті такі винятки:
* **MEOW_WHERE_STRUCT_EMPTY** - кумулятивна структура where порожня. А вона потрібна. Дуже.
* **MEOW_DATA_STRUCT_EMPTY** - кумулятивна структура data порожня. І вона теж комусь дуже потрібна.
* **MEOW_JOIN_WRONG_TYPE** - неправильний тип JOIN. Допустимі тільки INNER, LEFT, RIGHT.
* **MEOW_NO_FIELD_NAME** - не встановлено обов'язкове ім'я поля.
===== Принципова схема =====
Це десь ось настільки високорівнева штука.
{{:nyanorm_scheme0.png?|}}
Тож так, у модулях, де швидкість роботи з даними може бути вузьким місцем, можливо, доведеться використовувати більш традиційний підхід із використанням api.mysql.
===== Що ще? =====
Коротше ось поки що вам практичні приклади використання цього у вигляді хеллоуворлда. Але оскільки я хеллоуворлди писати не вмію, ось вам тудушка. Як кажуть розумні люди - не вмієш писати хеллоуворлди - пиши тудушки.
Працювати наш TODO-list буде на наступній табличці в БД:
CREATE TABLE IF NOT EXISTS `todo` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`text` text,
PRIMARY KEY (`id`)
) ENGINE=MyISAM DEFAULT CHARSET=utf8 AUTO_INCREMENT=1;
А ось і весь код нашого модуля:
$todo = new nya_todo(); //Створюємо модель даних за допомогою магії.
//Власне todo після префікса nya_ це і є наша табличка.
$moduleBaseUrl = '?module=testing'; //базовий URL нашого модулю
$messages = new UbillingMessageHelper(); //хочемо штатні красиві повідомлення
$result = '';
//Рендеримо формочку створення, використовуючи інлайнову збірку за допомогою [[apiastral|Astral]].
$inputs = wf_TextInput('newtasktext', __('Task'), '', false, 40);
$inputs .= wf_Submit(__('Create'));
$creationForm = wf_Form('', 'POST', $inputs, 'glamour');
show_window(__('Create new task'), $creationForm);
//Ловимо непорожній newtasktext як сигнал для створення нового запису
if (ubRouting::checkPost(array('newtasktext'))) {
//заповнюємо новими даними структуру data, попередньо відфільтрувавши їх засобами ubRouting
$todo->data('text', ubRouting::post('newtasktext', 'mres'));
$todo->create(); //кажемо моделі що "створи запис" на підставі структури вище.
ubRouting::nav($moduleBaseUrl); //повертаємось у морду модуля
}
//Ловимо запит на видалення тудушки, цього разу GET-ом.
if (ubRouting::checkGet('deletetodo')) {
//виставляємо параметр where у id, котра видаляється, попередньо переконавшись, що це буде циферка
$todo->where('id', '=', ubRouting::get('deletetodo', 'int'));
$todo->delete(); //кажемо моделі "вдалися"
ubRouting::nav($moduleBaseUrl);
}
//Ловимо запит на зміну наявного запису. Для сигналізації про початок чекаємо не порожній текст і айдишку.
if (ubRouting::checkPost(array('edittodoid', 'edittodotext'))) {
//Далі ж усе очевидно, правда? Виставляємо де, виставляємо, що поміняти, фільтруємо, говоримо "збережися".
$todo->where('id', '=', ubRouting::post('edittodoid', 'int'));
$todo->data('text', ubRouting::post('edittodotext', 'mres'));
$todo->save();
ubRouting::nav($moduleBaseUrl);
}
//Показуємо наявні завдання, які нам потрібно зробити.
$todo->orderBy('id', 'DESC'); //хочемо сортування від свіжих до древніх
$allTodos = $todo->getAll(); //дістаємо взагалі все, що бачимо з моделі.
if (!empty($allTodos)) {
$cells = wf_TableCell(__('Text'));
$cells .= wf_TableCell(__('Actions'));
$rows = wf_TableRow($cells, 'row1');
foreach ($allTodos as $io => $each) {
$cells = wf_TableCell($each['text']);
$actControls = wf_JSAlert($moduleBaseUrl . '&deletetodo=' . $each['id'], web_delete_icon(), $messages->getDeleteAlert());
//Прямо тут, збираємо формочку редагування кожного завдання і пихаємо її в контроли.
$editInputs = wf_HiddenInput('edittodoid', $each['id']);
$editInputs .= wf_TextInput('edittodotext', __('Text'), $each['text'], false, 40);
$editInputs .= wf_Submit(__('Save'));
$editForm = wf_Form('', 'POST', $editInputs, 'glamour');
$actControls .= wf_modalAuto(web_edit_icon(), __('Edit'), $editForm);
//Фу так робити.
$cells .= wf_TableCell($actControls);
$rows .= wf_TableRow($cells, 'row5');
}
$result .= wf_TableBody($rows, '100%', 0, 'sortable');
} else {
$result .= $messages->getStyledMessage(__('Nothing to show'), 'warning');
}
show_window(__('Sample TODO list'), $result);