eloquent.php 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516
  1. <?php namespace System\DB;
  2. abstract class Eloquent {
  3. /**
  4. * Indicates if the model exists in the database.
  5. *
  6. * @var bool
  7. */
  8. public $exists = false;
  9. /**
  10. * The model attributes.
  11. *
  12. * @var array
  13. */
  14. public $attributes = array();
  15. /**
  16. * The model's dirty attributes.
  17. *
  18. * @var array
  19. */
  20. public $dirty = array();
  21. /**
  22. * The model's ignored attributes.
  23. *
  24. * Ignored attributes will not be saved to the database, and
  25. * are primarily used to hold relationships.
  26. *
  27. * @var array
  28. */
  29. public $ignore = array();
  30. /**
  31. * The relationships that should be eagerly loaded.
  32. *
  33. * @var array
  34. */
  35. public $includes = array();
  36. /**
  37. * The relationship type the model is currently resolving.
  38. *
  39. * @var string
  40. */
  41. public $relating;
  42. /**
  43. * The foreign key of the "relating" relationship.
  44. *
  45. * @var string
  46. */
  47. public $relating_key;
  48. /**
  49. * The table name of the model being resolved. Used during many-to-many eager loading.
  50. *
  51. * @var string
  52. */
  53. public $relating_table;
  54. /**
  55. * The model query instance.
  56. *
  57. * @var Query
  58. */
  59. public $query;
  60. /**
  61. * Get the table name for a model.
  62. *
  63. * @param string $class
  64. * @return string
  65. */
  66. public static function table($class)
  67. {
  68. // -----------------------------------------------------
  69. // Check for a table name override.
  70. // -----------------------------------------------------
  71. if (property_exists($class, 'table'))
  72. {
  73. return $class::$table;
  74. }
  75. return \System\Str::lower(\System\Inflector::plural($class));
  76. }
  77. /**
  78. * Factory for creating new Eloquent model instances.
  79. *
  80. * @param string $class
  81. * @return object
  82. */
  83. public static function make($class)
  84. {
  85. // -----------------------------------------------------
  86. // Instantiate the Eloquent model.
  87. // -----------------------------------------------------
  88. $model = new $class;
  89. // -----------------------------------------------------
  90. // Set the fluent query builder on the model.
  91. // -----------------------------------------------------
  92. $model->query = Query::table(static::table($class));
  93. return $model;
  94. }
  95. /**
  96. * Create a new model instance and set the relationships
  97. * that should be eagerly loaded.
  98. *
  99. * @return mixed
  100. */
  101. public static function with()
  102. {
  103. // -----------------------------------------------------
  104. // Create a new model instance.
  105. // -----------------------------------------------------
  106. $model = static::make(get_called_class());
  107. // -----------------------------------------------------
  108. // Set the relationships that should be eager loaded.
  109. // -----------------------------------------------------
  110. $model->includes = func_get_args();
  111. return $model;
  112. }
  113. /**
  114. * Get a model by the primary key.
  115. *
  116. * @param int $id
  117. * @return mixed
  118. */
  119. public static function find($id)
  120. {
  121. return static::make(get_called_class())->where('id', '=', $id)->first();
  122. }
  123. /**
  124. * Get an array of models from the database.
  125. *
  126. * @return array
  127. */
  128. private function _get()
  129. {
  130. return Eloquent\Hydrator::hydrate($this);
  131. }
  132. /**
  133. * Get the first model result
  134. *
  135. * @return mixed
  136. */
  137. private function _first()
  138. {
  139. return (count($results = Eloquent\Hydrator::hydrate($this->take(1))) > 0) ? reset($results) : null;
  140. }
  141. /**
  142. * Retrieve the query for a 1:1 relationship.
  143. *
  144. * @param string $model
  145. * @param string $foreign_key
  146. * @return mixed
  147. */
  148. public function has_one($model, $foreign_key = null)
  149. {
  150. $this->relating = __FUNCTION__;
  151. return $this->has_one_or_many($model, $foreign_key);
  152. }
  153. /**
  154. * Retrieve the query for a 1:* relationship.
  155. *
  156. * @param string $model
  157. * @param string $foreign_key
  158. * @return mixed
  159. */
  160. public function has_many($model, $foreign_key = null)
  161. {
  162. $this->relating = __FUNCTION__;
  163. return $this->has_one_or_many($model, $foreign_key);
  164. }
  165. /**
  166. * Retrieve the query for a 1:1 or 1:* relationship.
  167. *
  168. * @param string $model
  169. * @param string $foreign_key
  170. * @return mixed
  171. */
  172. private function has_one_or_many($model, $foreign_key)
  173. {
  174. // -----------------------------------------------------
  175. // Determine the foreign key for the relationship.
  176. //
  177. // The foreign key is typically the name of the related
  178. // model with an appeneded _id.
  179. // -----------------------------------------------------
  180. $this->relating_key = (is_null($foreign_key)) ? \System\Str::lower(get_class($this)).'_id' : $foreign_key;
  181. return static::make($model)->where($this->relating_key, '=', $this->id);
  182. }
  183. /**
  184. * Retrieve the query for a 1:1 belonging relationship.
  185. *
  186. * @param string $model
  187. * @param string $foreign_key
  188. * @return mixed
  189. */
  190. public function belongs_to($model, $foreign_key = null)
  191. {
  192. $this->relating = __FUNCTION__;
  193. // -----------------------------------------------------
  194. // Determine the foreign key of the relationship.
  195. // -----------------------------------------------------
  196. if ( ! is_null($foreign_key))
  197. {
  198. $this->relating_key = $foreign_key;
  199. }
  200. else
  201. {
  202. // -----------------------------------------------------
  203. // Get the calling function name.
  204. // -----------------------------------------------------
  205. list(, $caller) = debug_backtrace(false);
  206. // -----------------------------------------------------
  207. // Determine the foreign key for the relationship.
  208. //
  209. // The foreign key for belonging relationships is the
  210. // name of the relationship method with an appended _id.
  211. // -----------------------------------------------------
  212. $this->relating_key = $caller['function'].'_id';
  213. }
  214. return static::make($model)->where('id', '=', $this->attributes[$this->relating_key]);
  215. }
  216. /**
  217. * Retrieve the query for a *:* relationship.
  218. *
  219. * @param string $model
  220. * @param string $table
  221. * @return mixed
  222. */
  223. public function has_and_belongs_to_many($model, $table = null)
  224. {
  225. $this->relating = __FUNCTION__;
  226. // -----------------------------------------------------
  227. // Determine the intermediate table name.
  228. // -----------------------------------------------------
  229. if ( ! is_null($table))
  230. {
  231. $this->relating_table = $table;
  232. }
  233. else
  234. {
  235. // -----------------------------------------------------
  236. // By default, the intermediate table name is the plural
  237. // names of the models arranged alphabetically and
  238. // concatenated with an underscore.
  239. // -----------------------------------------------------
  240. $models = array(\System\Inflector::plural($model), \System\Inflector::plural(get_class($this)));
  241. sort($models);
  242. $this->relating_table = \System\Str::lower($models[0].'_'.$models[1]);
  243. }
  244. // -----------------------------------------------------
  245. // Determine the foreign key for the relationship.
  246. // -----------------------------------------------------
  247. $this->relating_key = $this->relating_table.'.'.\System\Str::lower(get_class($this)).'_id';
  248. return static::make($model)
  249. ->select(static::table($model).'.*')
  250. ->join($this->relating_table, static::table($model).'.id', '=', $this->relating_table.'.'.\System\Str::lower($model).'_id')
  251. ->where($this->relating_key, '=', $this->id);
  252. }
  253. /**
  254. * Save the model to the database.
  255. *
  256. * @return bool
  257. */
  258. public function save()
  259. {
  260. // -----------------------------------------------------
  261. // If the model doesn't have any dirty attributes, there
  262. // is no need to save it to the database.
  263. // -----------------------------------------------------
  264. if ($this->exists and count($this->dirty) == 0)
  265. {
  266. return true;
  267. }
  268. // -----------------------------------------------------
  269. // Get the class name of the Eloquent model.
  270. // -----------------------------------------------------
  271. $model = get_class($this);
  272. // -----------------------------------------------------
  273. // Get a fresh query instance for the model.
  274. // -----------------------------------------------------
  275. $this->query = Query::table(static::table($model));
  276. // -----------------------------------------------------
  277. // Set the creation and update timestamps.
  278. // -----------------------------------------------------
  279. if (property_exists($model, 'timestamps') and $model::$timestamps)
  280. {
  281. $this->updated_at = date('Y-m-d H:i:s');
  282. if ( ! $this->exists)
  283. {
  284. $this->created_at = $this->updated_at;
  285. }
  286. }
  287. // -----------------------------------------------------
  288. // If the model already exists in the database, we only
  289. // need to update it. Otherwise, we'll insert it.
  290. // -----------------------------------------------------
  291. if ($this->exists)
  292. {
  293. $result = $this->query->where('id', '=', $this->attributes['id'])->update($this->dirty) == 1;
  294. }
  295. else
  296. {
  297. $this->attributes['id'] = $this->query->insert_get_id($this->attributes);
  298. $result = $this->exists = is_numeric($this->id);
  299. }
  300. // -----------------------------------------------------
  301. // The dirty attributes can be cleared after each save.
  302. // -----------------------------------------------------
  303. $this->dirty = array();
  304. return $result;
  305. }
  306. /**
  307. * Delete a model from the database.
  308. */
  309. public function delete($id = null)
  310. {
  311. // -----------------------------------------------------
  312. // If the method is being called from an existing model,
  313. // only delete that model from the database.
  314. // -----------------------------------------------------
  315. if ($this->exists)
  316. {
  317. return Query::table(static::table(get_class($this)))->delete($this->id) == 1;
  318. }
  319. return $this->query->delete($id);
  320. }
  321. /**
  322. * Magic method for retrieving model attributes.
  323. */
  324. public function __get($key)
  325. {
  326. // -----------------------------------------------------
  327. // Check the ignored attributes first.
  328. // -----------------------------------------------------
  329. if (array_key_exists($key, $this->ignore))
  330. {
  331. return $this->ignore[$key];
  332. }
  333. // -----------------------------------------------------
  334. // Is the attribute actually a relationship?
  335. // -----------------------------------------------------
  336. if (method_exists($this, $key))
  337. {
  338. // -----------------------------------------------------
  339. // Get the query / model for the relationship.
  340. // -----------------------------------------------------
  341. $model = $this->$key();
  342. return ($this->relating == 'has_one' or $this->relating == 'belongs_to')
  343. ? $this->ignore[$key] = $model->first()
  344. : $this->ignore[$key] = $model->get();
  345. }
  346. return (array_key_exists($key, $this->attributes)) ? $this->attributes[$key] : null;
  347. }
  348. /**
  349. * Magic Method for setting model attributes.
  350. */
  351. public function __set($key, $value)
  352. {
  353. // -----------------------------------------------------
  354. // If the key is a relationship, add it to the ignored.
  355. // -----------------------------------------------------
  356. if (method_exists($this, $key))
  357. {
  358. $this->ignore[$key] = $value;
  359. }
  360. // -----------------------------------------------------
  361. // Set the attribute and add it to the dirty array.
  362. // -----------------------------------------------------
  363. else
  364. {
  365. $this->attributes[$key] = $value;
  366. $this->dirty[$key] = $value;
  367. }
  368. }
  369. /**
  370. * Magic Method for determining if a model attribute is set.
  371. */
  372. public function __isset($key)
  373. {
  374. return (array_key_exists($key, $this->attributes) or array_key_exists($key, $this->ignore));
  375. }
  376. /**
  377. * Magic Method for unsetting model attributes.
  378. */
  379. public function __unset($key)
  380. {
  381. unset($this->attributes[$key]);
  382. unset($this->ignore[$key]);
  383. unset($this->dirty[$key]);
  384. }
  385. /**
  386. * Magic Method for handling dynamic method calls.
  387. */
  388. public function __call($method, $parameters)
  389. {
  390. // -----------------------------------------------------
  391. // Retrieve an array of models.
  392. // -----------------------------------------------------
  393. if ($method == 'get')
  394. {
  395. return $this->_get();
  396. }
  397. // -----------------------------------------------------
  398. // Retrieve the first model result.
  399. // -----------------------------------------------------
  400. if ($method == 'first')
  401. {
  402. return $this->_first();
  403. }
  404. // -----------------------------------------------------
  405. // If the method is an aggregate function, just return
  406. // the aggregate value from the query.
  407. // -----------------------------------------------------
  408. if (in_array($method, array('count', 'sum', 'min', 'max', 'avg')))
  409. {
  410. return call_user_func_array(array($this->query, $method), $parameters);
  411. }
  412. // -----------------------------------------------------
  413. // Pass the method call to the query instance.
  414. // -----------------------------------------------------
  415. call_user_func_array(array($this->query, $method), $parameters);
  416. return $this;
  417. }
  418. /**
  419. * Magic Method for handling dynamic static method calls.
  420. */
  421. public static function __callStatic($method, $parameters)
  422. {
  423. $model = static::make(get_called_class());
  424. // -----------------------------------------------------
  425. // Retrieve the entire table of models.
  426. // -----------------------------------------------------
  427. if ($method == 'get')
  428. {
  429. return $model->_get();
  430. }
  431. // -----------------------------------------------------
  432. // Retrieve the first model result.
  433. // -----------------------------------------------------
  434. if ($method == 'first')
  435. {
  436. return $model->_first();
  437. }
  438. // -----------------------------------------------------
  439. // If the method is an aggregate function, just return
  440. // the aggregate value from the query.
  441. // -----------------------------------------------------
  442. if (in_array($method, array('count', 'sum', 'min', 'max', 'avg')))
  443. {
  444. return call_user_func_array(array($model->query, $method), $parameters);
  445. }
  446. // -----------------------------------------------------
  447. // Pass the method call to the query instance.
  448. // -----------------------------------------------------
  449. call_user_func_array(array($model->query, $method), $parameters);
  450. return $model;
  451. }
  452. }