Робить НЯКУ з вашою базою. І надає дуже зручні засоби, щоб ви докладали мінімум зусиль і мозкових звивин при роботі з БД Ubilling. Ви зможете описувати ваші моделі даних, і роботу з ними, більше не малюючи в ручну SELECT `something` FROM `tablename`… та інші остогидлі вам конструкції. Також NyanORM візьме на себе всю вашу рутинну роботу з попередньої підготовки і трансформації ваших даних. Завданням NyanORM не бути всеосяжним монстром, який реалізує всі можливі кейси роботи з вашими даними. Завданням NyanORM є - бути простим та зручним. З самого початку, він задумувався для того, щоб скоротити обсяги рутинного, нудного й однотипного коду, з якого складається робота із сутностями в 90-95% прикладних модулів. Власне тільки ці кейси він і повинен перекривати, не накладаючи на вас жодних додаткових обмежень.
Чому саме NyanORM, а не будь-яка інша стороння бібліотека з приставкою ORM у назві? А з таких причин:
На жаль, ми не знайшли жодного наявного рішення, що відповідає всім особливостям/вимогам вище. Це заощадило б нам дуже багато зусиль і часу, але ні. Тому 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 призначена для зберігання даних, які будуть надалі використані під час виклику методів 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 вам в обличчя можуть бути викинуті такі винятки:
Це десь ось настільки високорівнева штука.
Тож так, у модулях, де швидкість роботи з даними може бути вузьким місцем, можливо, доведеться використовувати більш традиційний підхід із використанням 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);