1de02284cSSzymon Olewniczak<?php 2de02284cSSzymon Olewniczak 3de02284cSSzymon Olewniczak//if(!defined('DOKU_INC')) die(); 4de02284cSSzymon Olewniczak 5de02284cSSzymon Olewniczak 6de02284cSSzymon Olewniczak//~ abstract class BEZ_mdl_Dummy_Entity { 7de02284cSSzymon Olewniczak 8de02284cSSzymon Olewniczak //~ protected $model; 9de02284cSSzymon Olewniczak 10de02284cSSzymon Olewniczak //~ protected $id = NULL; 11de02284cSSzymon Olewniczak 12de02284cSSzymon Olewniczak //~ public function __get($property) { 13de02284cSSzymon Olewniczak //~ if ($property === 'id') { 14de02284cSSzymon Olewniczak //~ return $this->id; 15de02284cSSzymon Olewniczak //~ } 16de02284cSSzymon Olewniczak //~ } 17de02284cSSzymon Olewniczak 18de02284cSSzymon Olewniczak //~ public function get_table_singular() { 19de02284cSSzymon Olewniczak //~ $class = get_class($this); 20de02284cSSzymon Olewniczak //~ $exp = explode('_', $class); 21de02284cSSzymon Olewniczak //~ $singular = array_pop($exp); 22de02284cSSzymon Olewniczak //~ return lcfirst($singular); 23de02284cSSzymon Olewniczak //~ } 24de02284cSSzymon Olewniczak 25de02284cSSzymon Olewniczak //~ public function get_table_name() { 26de02284cSSzymon Olewniczak //~ $singlar = $this->get_table_singular(); 27de02284cSSzymon Olewniczak //~ return $singular.'s'; 28de02284cSSzymon Olewniczak //~ } 29de02284cSSzymon Olewniczak 30de02284cSSzymon Olewniczak //~ public function acl_of($field) { 31de02284cSSzymon Olewniczak //~ return $this->model->acl->check_field($this, $field); 32de02284cSSzymon Olewniczak //~ } 33de02284cSSzymon Olewniczak 34de02284cSSzymon Olewniczak //~ public function __construct($model) { 35de02284cSSzymon Olewniczak //~ $this->model = $model; 36de02284cSSzymon Olewniczak //~ } 37de02284cSSzymon Olewniczak//~ } 38de02284cSSzymon Olewniczak 39de02284cSSzymon Olewniczaknamespace dokuwiki\plugin\bez\mdl; 40de02284cSSzymon Olewniczak/* 41de02284cSSzymon Olewniczak * All fields are stored in object as strings. 42de02284cSSzymon Olewniczak * NULLs are converted to empty string. 43de02284cSSzymon Olewniczak * If any attribute in object === NULL -> it means that it was not initialized 44de02284cSSzymon Olewniczak * But we always inserts NULLs instead of empty strings. 45de02284cSSzymon Olewniczak * https://stackoverflow.com/questions/1267999/mysql-better-to-insert-null-or-empty-string 46de02284cSSzymon Olewniczak **/ 47de02284cSSzymon Olewniczak 48de02284cSSzymon Olewniczakuse dokuwiki\plugin\bez\meta\PermissionDeniedException; 49de02284cSSzymon Olewniczak 50de02284cSSzymon Olewniczakabstract class Entity {// extends BEZ_mdl_Dummy_Entity { 51de02284cSSzymon Olewniczak 52de02284cSSzymon Olewniczak /** @var Model */ 53de02284cSSzymon Olewniczak protected $model; 54de02284cSSzymon Olewniczak 55de02284cSSzymon Olewniczak /** @var Validator */ 56de02284cSSzymon Olewniczak protected $validator; 57de02284cSSzymon Olewniczak 58de02284cSSzymon Olewniczak //protected $columns_demendencies = array(); 59de02284cSSzymon Olewniczak 60de02284cSSzymon Olewniczak //protected $parse_int = array(); 61de02284cSSzymon Olewniczak 62de02284cSSzymon Olewniczak abstract public static function get_columns(); 63de02284cSSzymon Olewniczak 64*7fbf4c39SSzymon Olewniczak public static function get_select_columns() { 65*7fbf4c39SSzymon Olewniczak $class = get_called_class(); 66*7fbf4c39SSzymon Olewniczak return $class::get_columns(); 67*7fbf4c39SSzymon Olewniczak } 68*7fbf4c39SSzymon Olewniczak 69de02284cSSzymon Olewniczak// abstract public static function get_virtual_columns(); 70de02284cSSzymon Olewniczak 71de02284cSSzymon Olewniczak// private function is_dummy() { 72de02284cSSzymon Olewniczak// if (strstr(get_class(), 'Dummy') === false) { 73de02284cSSzymon Olewniczak// return false; 74de02284cSSzymon Olewniczak// } 75de02284cSSzymon Olewniczak// return true; 76de02284cSSzymon Olewniczak// } 77de02284cSSzymon Olewniczak// 78de02284cSSzymon Olewniczak// private function not_for_dummies() { 79de02284cSSzymon Olewniczak// if ($this->is_dummy()) { 80de02284cSSzymon Olewniczak// throw new Exception('dummy object doesn\'t contains data.'); 81de02284cSSzymon Olewniczak// } 82de02284cSSzymon Olewniczak// } 83de02284cSSzymon Olewniczak 84de02284cSSzymon Olewniczak public function get_assoc($filter=NULL) { 85de02284cSSzymon Olewniczak //$this->not_for_dummies(); 86de02284cSSzymon Olewniczak 87de02284cSSzymon Olewniczak $assoc = array(); 88de02284cSSzymon Olewniczak //$columns = array_merge($this->get_columns(), $this->get_virtual_columns()); 89de02284cSSzymon Olewniczak 90*7fbf4c39SSzymon Olewniczak $columns = $this->get_select_columns(); 91de02284cSSzymon Olewniczak if ($filter !== NULL) { 92de02284cSSzymon Olewniczak $columns = array_intersect($columns, $filter); 93de02284cSSzymon Olewniczak } 94de02284cSSzymon Olewniczak 95de02284cSSzymon Olewniczak foreach ($columns as $col) { 96de02284cSSzymon Olewniczak $assoc[$col] = $this->$col; 97de02284cSSzymon Olewniczak } 98de02284cSSzymon Olewniczak return $assoc; 99de02284cSSzymon Olewniczak } 100de02284cSSzymon Olewniczak 101de02284cSSzymon Olewniczak// public function get_table_singular() { 102de02284cSSzymon Olewniczak// $class = get_class($this); 103de02284cSSzymon Olewniczak// $exp = explode('_', $class); 104de02284cSSzymon Olewniczak// $singular = array_pop($exp); 105de02284cSSzymon Olewniczak// return lcfirst($singular); 106de02284cSSzymon Olewniczak// } 107de02284cSSzymon Olewniczak// 108de02284cSSzymon Olewniczak public function get_table_name() { 109de02284cSSzymon Olewniczak $class = (new \ReflectionClass($this))->getShortName(); 110de02284cSSzymon Olewniczak return lcfirst($class); 111de02284cSSzymon Olewniczak } 112de02284cSSzymon Olewniczak 113de02284cSSzymon Olewniczak// private function create_DateTime_from_column($column) { 114de02284cSSzymon Olewniczak// $cols = array_merge($this->get_columns(), $this->get_virtual_columns()); 115de02284cSSzymon Olewniczak// if (!in_array($column, $cols)) { 116de02284cSSzymon Olewniczak// throw new Exception("$column is not a column"); 117de02284cSSzymon Olewniczak// } 118de02284cSSzymon Olewniczak// $dt = $this->$column; 119de02284cSSzymon Olewniczak// if (is_numeric($dt)) { 120de02284cSSzymon Olewniczak// return new DateTime('@'.$dt); 121de02284cSSzymon Olewniczak// } 122de02284cSSzymon Olewniczak// return new DateTime($dt); 123de02284cSSzymon Olewniczak// } 124de02284cSSzymon Olewniczak// 125de02284cSSzymon Olewniczak// public function date_format($column) { 126de02284cSSzymon Olewniczak// $date = $this->create_DateTime_from_column($column); 127de02284cSSzymon Olewniczak// 128de02284cSSzymon Olewniczak// return $date->format('Y-m-d'); 129de02284cSSzymon Olewniczak// } 130de02284cSSzymon Olewniczak// 131de02284cSSzymon Olewniczak// public function datetime_format($column, $format='comment') { 132de02284cSSzymon Olewniczak// $date = $this->create_DateTime_from_column($column);; 133de02284cSSzymon Olewniczak// 134de02284cSSzymon Olewniczak// if ($format === 'comment') { 135de02284cSSzymon Olewniczak// return $dt->format('j') . ' ' . 136de02284cSSzymon Olewniczak// $this->model->action->getLang('mon'.$dt->format('n').'_a') . ' ' . 137de02284cSSzymon Olewniczak// ($dt->format('Y') === date('Y') ? '' : $dt->format('Y') . ' ') . 138de02284cSSzymon Olewniczak// $this->model->action->getLang('at_hour') . ' ' . 139de02284cSSzymon Olewniczak// $dt->format('G:i'); 140de02284cSSzymon Olewniczak// } 141de02284cSSzymon Olewniczak// } 142de02284cSSzymon Olewniczak// 143de02284cSSzymon Olewniczak// public function days_ago($column, $first_date = null) { 144de02284cSSzymon Olewniczak// if ($first_date === null) { 145de02284cSSzymon Olewniczak// $first_date = new DateTime(); 146de02284cSSzymon Olewniczak// } elseif (is_string($first_date)) { 147de02284cSSzymon Olewniczak// $first_date = $this->create_DateTime_from_column($first_date); 148de02284cSSzymon Olewniczak// } 149de02284cSSzymon Olewniczak// $interval = $first_date->diff($this->create_DateTime_from_column($column)); 150de02284cSSzymon Olewniczak// $sign = $interval->format('%R'); 151de02284cSSzymon Olewniczak// $days = $interval->format('%a'); 152de02284cSSzymon Olewniczak// if ($sign === '-') { 153de02284cSSzymon Olewniczak// return $days . ' ' . $this->model->action->getLang('days') . ' ' . 154de02284cSSzymon Olewniczak// $this->model->action->getLang('ago'); 155de02284cSSzymon Olewniczak// } 156de02284cSSzymon Olewniczak// return $days . ' ' . $this->model->action->getLang('days'); 157de02284cSSzymon Olewniczak// 158de02284cSSzymon Olewniczak// } 159de02284cSSzymon Olewniczak// 160de02284cSSzymon Olewniczak// public function float_localized($column) { 161de02284cSSzymon Olewniczak// $cols = array_merge($this->get_columns(), $this->get_virtual_columns()); 162de02284cSSzymon Olewniczak// if (!in_array($column, $cols)) { 163de02284cSSzymon Olewniczak// throw new Exception("$column is not a column"); 164de02284cSSzymon Olewniczak// } 165de02284cSSzymon Olewniczak// if ($this->$column === '') { 166de02284cSSzymon Olewniczak// return ''; 167de02284cSSzymon Olewniczak// } 168de02284cSSzymon Olewniczak// 169de02284cSSzymon Olewniczak// if (!is_numeric($this->$column)) { 170de02284cSSzymon Olewniczak// throw new Exception('Column ' . $column . ' isn\'t numeric'); 171de02284cSSzymon Olewniczak// } 172de02284cSSzymon Olewniczak// 173de02284cSSzymon Olewniczak// return sprintf('%.2f', $this->$column); 174de02284cSSzymon Olewniczak// } 175de02284cSSzymon Olewniczak 176de02284cSSzymon Olewniczak //set id when object is saved in database 177de02284cSSzymon Olewniczak// public function set_id($id) { 178de02284cSSzymon Olewniczak//// $this->not_for_dummies(); 179de02284cSSzymon Olewniczak// 180de02284cSSzymon Olewniczak// if ($this->id === NULL) { 181de02284cSSzymon Olewniczak// $this->id = $id; 182de02284cSSzymon Olewniczak// } else { 183de02284cSSzymon Olewniczak// throw new \Exception('id already set for issue #'.$this->id); 184de02284cSSzymon Olewniczak// } 185de02284cSSzymon Olewniczak// } 186de02284cSSzymon Olewniczak 187de02284cSSzymon Olewniczak// public function sqlite_date($time=NULL) { 188de02284cSSzymon Olewniczak// //SQLITE format: https://www.sqlite.org/lang_datefunc.html 189de02284cSSzymon Olewniczak// if ($time === NULL) { 190de02284cSSzymon Olewniczak// return date('Y-m-d H:i:s'); 191de02284cSSzymon Olewniczak// } else { 192de02284cSSzymon Olewniczak// return date('Y-m-d H:i:s', $time); 193de02284cSSzymon Olewniczak// } 194de02284cSSzymon Olewniczak// } 195de02284cSSzymon Olewniczak 196de02284cSSzymon Olewniczak public function __get($property) { 197de02284cSSzymon Olewniczak// $this->not_for_dummies(); 198de02284cSSzymon Olewniczak 199de02284cSSzymon Olewniczak// $columns = array_merge($this->get_columns(), $this->get_virtual_columns()); 200de02284cSSzymon Olewniczak if (!property_exists($this, $property) || !in_array($property, $this->get_columns())) { 201de02284cSSzymon Olewniczak throw new \Exception('there is no column: "'.$property. '"" in table: "' . $this->get_table_name() . '"'); 202de02284cSSzymon Olewniczak } 203de02284cSSzymon Olewniczak 204de02284cSSzymon Olewniczak //now only normal db columns has ACL, it should be fixed 205de02284cSSzymon Olewniczak if ($this->acl_of($property) < BEZ_PERMISSION_VIEW) { 206de02284cSSzymon Olewniczak throw new PermissionDeniedException(); 207de02284cSSzymon Olewniczak } 208de02284cSSzymon Olewniczak 209de02284cSSzymon Olewniczak return $this->$property; 210de02284cSSzymon Olewniczak 211de02284cSSzymon Olewniczak// if (property_exists($this, $property) && in_array($property, $this->get_columns())) { 212de02284cSSzymon Olewniczak//// if (in_array($property, $this->parse_int)) { 213de02284cSSzymon Olewniczak//// return (int)$this->$property; 214de02284cSSzymon Olewniczak//// } else { 215de02284cSSzymon Olewniczak// return $this->$property; 216de02284cSSzymon Olewniczak//// } 217de02284cSSzymon Olewniczak// } 218de02284cSSzymon Olewniczak } 219de02284cSSzymon Olewniczak 220de02284cSSzymon Olewniczak protected function set_property($property, $value) { 221de02284cSSzymon Olewniczak// $this->not_for_dummies(); 222de02284cSSzymon Olewniczak 223de02284cSSzymon Olewniczak if (!in_array($property, $this->get_columns())) { 224de02284cSSzymon Olewniczak throw new \Exception('trying to set not existing column'); 225de02284cSSzymon Olewniczak } 226de02284cSSzymon Olewniczak 227de02284cSSzymon Olewniczak //throws ValidationException 228de02284cSSzymon Olewniczak $this->validator->validate_field($property, $value); 229de02284cSSzymon Olewniczak 230de02284cSSzymon Olewniczak //throws PermissionDeniedException 231de02284cSSzymon Olewniczak $this->model->acl->can_change($this, $property); 232de02284cSSzymon Olewniczak 233de02284cSSzymon Olewniczak $this->$property = $value; 234de02284cSSzymon Olewniczak 235de02284cSSzymon Olewniczak// if (isset($this->columns_demendencies[$property])) { 236de02284cSSzymon Olewniczak// $method = $this->columns_demendencies[$property]; 237de02284cSSzymon Olewniczak// //convention -> run the method with the same name as $property 238de02284cSSzymon Olewniczak// if (!method_exists($this, $method)) { 239de02284cSSzymon Olewniczak// throw new Exception("denepndency method: $property not implemented"); 240de02284cSSzymon Olewniczak// } 241de02284cSSzymon Olewniczak// $this->$method(); 242de02284cSSzymon Olewniczak// } 243de02284cSSzymon Olewniczak } 244de02284cSSzymon Olewniczak 245de02284cSSzymon Olewniczak protected function set_property_array($array) { 246de02284cSSzymon Olewniczak foreach ($array as $k => $v) { 247de02284cSSzymon Olewniczak $this->set_property($k, $v); 248de02284cSSzymon Olewniczak } 249de02284cSSzymon Olewniczak } 250de02284cSSzymon Olewniczak 251de02284cSSzymon Olewniczak //by default do nothing 252de02284cSSzymon Olewniczak// private function update_virtual_columns() { 253de02284cSSzymon Olewniczak// } 254de02284cSSzymon Olewniczak// 255de02284cSSzymon Olewniczak 256de02284cSSzymon Olewniczak public function set_data($post) { 257de02284cSSzymon Olewniczak// $input = array_intersect($this->changable_fields($filter), array_keys($post), array_keys($this->validator->get_rules())); 258de02284cSSzymon Olewniczak 259de02284cSSzymon Olewniczak $val_data = $this->validator->validate($post); 260de02284cSSzymon Olewniczak if ($val_data === false) { 261de02284cSSzymon Olewniczak throw new ValidationException($this->get_table_name(), $this->validator->get_errors()); 262de02284cSSzymon Olewniczak } 263de02284cSSzymon Olewniczak 264de02284cSSzymon Olewniczak $this->set_property_array($val_data); 265de02284cSSzymon Olewniczak 266de02284cSSzymon Olewniczak// $this->update_virtual_columns(); 267de02284cSSzymon Olewniczak } 268de02284cSSzymon Olewniczak 269*7fbf4c39SSzymon Olewniczak public function changable_fields($filter=NULL) { 270*7fbf4c39SSzymon Olewniczak $fields = $this->model->acl->check($this); 271*7fbf4c39SSzymon Olewniczak 272*7fbf4c39SSzymon Olewniczak if ($filter !== NULL) { 273*7fbf4c39SSzymon Olewniczak $fields = array_filter($fields, function ($k) use ($filter) { 274*7fbf4c39SSzymon Olewniczak return in_array($k, $filter); 275*7fbf4c39SSzymon Olewniczak }, ARRAY_FILTER_USE_KEY); 276*7fbf4c39SSzymon Olewniczak } 277*7fbf4c39SSzymon Olewniczak 278*7fbf4c39SSzymon Olewniczak return array_keys(array_filter($fields, function ($var) { 279*7fbf4c39SSzymon Olewniczak return $var >= BEZ_PERMISSION_CHANGE; 280*7fbf4c39SSzymon Olewniczak })); 281*7fbf4c39SSzymon Olewniczak } 282de02284cSSzymon Olewniczak 283de02284cSSzymon Olewniczak public function acl_of($field) { 284de02284cSSzymon Olewniczak return $this->model->acl->check_field($this, $field); 285de02284cSSzymon Olewniczak } 286de02284cSSzymon Olewniczak 287de02284cSSzymon Olewniczak public function __construct($model) { 288de02284cSSzymon Olewniczak //by convention all defaults must be strings 289de02284cSSzymon Olewniczak// foreach ($defaults as $val) { 290de02284cSSzymon Olewniczak// if (!is_string($val)) { 291de02284cSSzymon Olewniczak// throw new Exception('all defaults must be strings'); 292de02284cSSzymon Olewniczak// } 293de02284cSSzymon Olewniczak// } 294de02284cSSzymon Olewniczak// $this->model = $model; 295de02284cSSzymon Olewniczak// 296de02284cSSzymon Olewniczak// if (!$this->is_dummy()) { 297de02284cSSzymon Olewniczak// $this->validator = new BEZ_mdl_Validator($this->model); 298de02284cSSzymon Olewniczak// } 299de02284cSSzymon Olewniczak //$this->helper = plugin_load('helper', 'bez'); 300de02284cSSzymon Olewniczak 301de02284cSSzymon Olewniczak $this->model = $model; 302de02284cSSzymon Olewniczak $this->validator = new Validator($this->model); 303de02284cSSzymon Olewniczak } 304de02284cSSzymon Olewniczak} 305