SqlOrmModel.php 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218
  1. <?php
  2. namespace KarmaFW\Database\Sql;
  3. use \KarmaFW\App;
  4. class SqlOrmModel
  5. {
  6. protected $db;
  7. protected $table_row = [];
  8. protected $status = 'ready';
  9. protected $primary_keys = null; // example: ['id_product'] ... or ... ['id_product', 'id_category']
  10. protected $primary_key_values = null; // example: ['id_product' => 14] ... or ... ['id_product' => 123, 'id_category' => 14]
  11. public $table_name;
  12. public $autosave = false;
  13. public function __construct($table_name=null, $primary_key_values=[], $db=null)
  14. {
  15. if (is_null($db)) {
  16. $db = App::getDb();
  17. }
  18. $this->db = $db;
  19. $this->table_name = $table_name;
  20. $this->primary_key_values = $primary_key_values;
  21. if (! empty($primary_key_values)) {
  22. $this->load($primary_key_values);
  23. }
  24. }
  25. public function __set($name, $value)
  26. {
  27. $this->table_row[$name] = $value;
  28. $this->status = 'unsaved';
  29. if ($this->autosave) {
  30. $this->save();
  31. }
  32. }
  33. public function __get($name)
  34. {
  35. if (array_key_exists($name, $this->table_row)) {
  36. return $this->table_row[$name];
  37. }
  38. $trace = debug_backtrace();
  39. trigger_error(
  40. 'Propriété non-définie via __get() : ' . $name .
  41. ' dans ' . $trace[0]['file'] .
  42. ' à la ligne ' . $trace[0]['line'],
  43. E_USER_NOTICE);
  44. return null;
  45. }
  46. /** Depuis PHP 5.1.0 */
  47. public function __isset($name)
  48. {
  49. return isset($this->table_row[$name]);
  50. }
  51. /** Depuis PHP 5.1.0 */
  52. public function __unset($name)
  53. {
  54. unset($this->table_row[$name]);
  55. }
  56. public function clear()
  57. {
  58. return $this->loadFromArray([]);
  59. }
  60. public function asArray()
  61. {
  62. // Alias of toArray
  63. return $this->toArray();
  64. }
  65. public function getArray()
  66. {
  67. // Alias of toArray
  68. return $this->toArray();
  69. }
  70. public function toArray()
  71. {
  72. return $this->table_row;
  73. }
  74. public function toJSON()
  75. {
  76. return json_encode($this->table_row);
  77. }
  78. public function fetchPrimaryKeys()
  79. {
  80. if (empty($this->table_name)) {
  81. return false;
  82. }
  83. $this->primary_keys = [];
  84. $columns = $this->db->listTableColumns($this->table_name);
  85. foreach ($columns as $column) {
  86. if ($column['Key'] == 'PRI') {
  87. $this->primary_keys[] = $column['Field'];
  88. }
  89. }
  90. return $this->primary_keys;
  91. }
  92. public function setPrimaryKeysValues(array $primary_key_values=null)
  93. {
  94. $this->primary_key_values = $primary_key_values;
  95. }
  96. public function loadFromArray($data, $primary_key_values=null)
  97. {
  98. $this->primary_key_values = $primary_key_values;
  99. $this->table_row = array_slice($data, 0);
  100. //pre($this->table_row);
  101. if (! empty($this->primary_keys)) {
  102. foreach ($this->primary_keys as $column_name) {
  103. if (isset($this->table_row[$column_name])) {
  104. $this->primary_key_values[$column_name] = $this->table_row[$column_name];
  105. }
  106. }
  107. }
  108. return $this;
  109. }
  110. public function load(array $primary_key_values)
  111. {
  112. $this->primary_key_values = $primary_key_values;
  113. $data = $this->db->createQuery()->tableSelectOne($this->table_name, $this->primary_key_values, ['limit' => 1]);
  114. if (is_null($data)) {
  115. $this->primary_key_values = [];
  116. $this->table_row = [];
  117. return null;
  118. }
  119. $result = $this->loadFromArray($data, $primary_key_values);
  120. return $result;
  121. }
  122. public function save($force = false)
  123. {
  124. if (empty($this->primary_keys)) {
  125. // on recupere les primary_keys depuis le schema de ma table sql
  126. $this->fetchPrimaryKeys();
  127. if (empty($this->primary_keys)) {
  128. // cannot update because no primary key found
  129. return false;
  130. }
  131. }
  132. if ($this->status == 'saved' && ! $force) {
  133. return true;
  134. }
  135. if (empty($this->primary_key_values)) {
  136. // INSERT
  137. $id = $this->db->createQuery()->tableInsert($this->table_name, $this->table_row);
  138. if (! empty($id)) {
  139. if (count($this->primary_keys) > 1) {
  140. // TODO: gerer correctement les index multiples (normalement un seul des champ est en autoincrement. TODO...)
  141. $this->primary_key_values = [];
  142. foreach ($this->primary_keys as $column_name) {
  143. if (! isset($this->table_row[$column_name])) {
  144. $this->table_row[$column_name] = null; // TODO: trouver mieux que null
  145. }
  146. $this->primary_key_values = [ $column_name => $this->table_row[$column_name] ];
  147. }
  148. } else {
  149. $column_name = $this->primary_keys[0];
  150. $this->primary_key_values = [ $column_name => $id ];
  151. // on affecte le insert_id a sa variable equivalente en php
  152. $this->table_row[$column_name] = $id;
  153. }
  154. }
  155. } else {
  156. // UPDATE
  157. $this->db->createQuery()->tableUpdate($this->table_name, $this->table_row, $this->primary_key_values, ['limit' => 1]);
  158. }
  159. $this->status = 'saved';
  160. return true;
  161. }
  162. }