hydrate.php 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272
  1. <?php namespace System\DB\Eloquent;
  2. class Hydrate {
  3. /**
  4. * Load the array of hydrated models.
  5. *
  6. * @param object $eloquent
  7. * @return array
  8. */
  9. public static function from($eloquent)
  10. {
  11. // -----------------------------------------------------
  12. // Load the base models.
  13. // -----------------------------------------------------
  14. $results = static::base(get_class($eloquent), $eloquent->query->get());
  15. // -----------------------------------------------------
  16. // Load all of the eager relationships.
  17. // -----------------------------------------------------
  18. if (count($results) > 0)
  19. {
  20. foreach ($eloquent->includes as $include)
  21. {
  22. // -----------------------------------------------------
  23. // Verify the relationship is defined.
  24. // -----------------------------------------------------
  25. if ( ! method_exists($eloquent, $include))
  26. {
  27. throw new \Exception("Attempting to eager load [$include], but the relationship is not defined.");
  28. }
  29. // -----------------------------------------------------
  30. // Eagerly load the relationship.
  31. // -----------------------------------------------------
  32. static::eagerly($eloquent, $include, $results);
  33. }
  34. }
  35. return $results;
  36. }
  37. /**
  38. * Hydrate the base models for a query.
  39. *
  40. * @param string $class
  41. * @param array $models
  42. * @return array
  43. */
  44. private static function base($class, $models)
  45. {
  46. // -----------------------------------------------------
  47. // Initialize the hydrated model array.
  48. // -----------------------------------------------------
  49. $results = array();
  50. // -----------------------------------------------------
  51. // Hydrate the models from the results.
  52. // -----------------------------------------------------
  53. foreach ($models as $model)
  54. {
  55. // -----------------------------------------------------
  56. // Instantiate a new model instance.
  57. // -----------------------------------------------------
  58. $result = new $class;
  59. // -----------------------------------------------------
  60. // Set the model's attributes.
  61. // -----------------------------------------------------
  62. $result->attributes = (array) $model;
  63. // -----------------------------------------------------
  64. // Indicate that the model already exists.
  65. // -----------------------------------------------------
  66. $result->exists = true;
  67. // -----------------------------------------------------
  68. // Add the hydrated model to the array of models.
  69. // The array is keyed by the primary keys of the models.
  70. // -----------------------------------------------------
  71. $results[$result->id] = $result;
  72. }
  73. return $results;
  74. }
  75. /**
  76. * Eagerly load a relationship.
  77. *
  78. * @param object $eloquent
  79. * @param string $include
  80. * @param array $results
  81. * @return void
  82. */
  83. private static function eagerly($eloquent, $include, &$results)
  84. {
  85. // -----------------------------------------------------
  86. // Get the relationship Eloquent model.
  87. //
  88. // We spoof the "belongs_to" key to allow the query
  89. // to be fetched without any problems.
  90. // -----------------------------------------------------
  91. $eloquent->attributes[$spoof = $include.'_id'] = 0;
  92. $model = $eloquent->$include();
  93. unset($eloquent->attributes[$spoof]);
  94. // -----------------------------------------------------
  95. // Reset the WHERE clause on the query.
  96. // -----------------------------------------------------
  97. $model->query->where = 'WHERE 1 = 1';
  98. // -----------------------------------------------------
  99. // Reset the bindings on the query.
  100. // -----------------------------------------------------
  101. $model->query->bindings = array();
  102. // -----------------------------------------------------
  103. // Initialize the relationship on the parent models.
  104. // -----------------------------------------------------
  105. foreach ($results as &$result)
  106. {
  107. $result->ignore[$include] = (strpos($eloquent->relating, 'has_many') === 0) ? array() : null;
  108. }
  109. // -----------------------------------------------------
  110. // Eagerly load a 1:1 or 1:* relationship.
  111. // -----------------------------------------------------
  112. if ($eloquent->relating == 'has_one' or $eloquent->relating == 'has_many')
  113. {
  114. static::eagerly_load_one_or_many($eloquent->relating_key, $eloquent->relating, $include, $model, $results);
  115. }
  116. // -----------------------------------------------------
  117. // Eagerly load a 1:1 (belonging) relationship.
  118. // -----------------------------------------------------
  119. elseif ($eloquent->relating == 'belongs_to')
  120. {
  121. static::eagerly_load_belonging($eloquent->relating_key, $include, $model, $results);
  122. }
  123. // -----------------------------------------------------
  124. // Eagerly load a *:* relationship.
  125. // -----------------------------------------------------
  126. else
  127. {
  128. static::eagerly_load_many_to_many($eloquent->relating_key, $eloquent->relating_table, strtolower(get_class($eloquent)).'_id', $include, $model, $results);
  129. }
  130. }
  131. /**
  132. * Eagerly load a 1:1 or 1:* relationship.
  133. *
  134. * @param string $relating_key
  135. * @param string $relating
  136. * @param string $include
  137. * @param object $model
  138. * @param array $results
  139. * @return void
  140. */
  141. private static function eagerly_load_one_or_many($relating_key, $relating, $include, $model, &$results)
  142. {
  143. // -----------------------------------------------------
  144. // Get the related models.
  145. // -----------------------------------------------------
  146. $inclusions = $model->where_in($relating_key, array_keys($results))->get();
  147. // -----------------------------------------------------
  148. // Match the child models with their parent.
  149. // -----------------------------------------------------
  150. foreach ($inclusions as $key => $inclusion)
  151. {
  152. if ($relating == 'has_one')
  153. {
  154. $results[$inclusion->$relating_key]->ignore[$include] = $inclusion;
  155. }
  156. else
  157. {
  158. $results[$inclusion->$relating_key]->ignore[$include][$inclusion->id] = $inclusion;
  159. }
  160. }
  161. }
  162. /**
  163. * Eagerly load a 1:1 belonging relationship.
  164. *
  165. * @param string $relating_key
  166. * @param string $include
  167. * @param object $model
  168. * @param array $results
  169. * @return void
  170. */
  171. private static function eagerly_load_belonging($relating_key, $include, $model, &$results)
  172. {
  173. // -----------------------------------------------------
  174. // Gather the keys from the parent models.
  175. // -----------------------------------------------------
  176. $keys = array();
  177. foreach ($results as &$result)
  178. {
  179. $keys[] = $result->$relating_key;
  180. }
  181. // -----------------------------------------------------
  182. // Get the related models.
  183. // -----------------------------------------------------
  184. $inclusions = $model->where_in('id', array_unique($keys))->get();
  185. // -----------------------------------------------------
  186. // Match the child models with their parent.
  187. // -----------------------------------------------------
  188. foreach ($results as &$result)
  189. {
  190. $result->ignore[$include] = $inclusions[$result->$relating_key];
  191. }
  192. }
  193. /**
  194. * Eagerly load a many-to-many relationship.
  195. *
  196. * @param string $relating_key
  197. * @param string $relating_table
  198. * @param string $foreign_key
  199. * @param string $include
  200. * @param object $model
  201. * @param array $results
  202. * @return void
  203. */
  204. private static function eagerly_load_many_to_many($relating_key, $relating_table, $foreign_key, $include, $model, &$results)
  205. {
  206. // -----------------------------------------------------
  207. // Reset the SELECT clause.
  208. // -----------------------------------------------------
  209. $model->query->select = null;
  210. // -----------------------------------------------------
  211. // Retrieve the raw results as stdClasses.
  212. //
  213. // We also add the foreign key to the select which will allow us
  214. // to match the models back to their parents.
  215. // -----------------------------------------------------
  216. $inclusions = $model->query->where_in($relating_key, array_keys($results))->get(Meta::table(get_class($model)).'.*', $relating_table.'.'.$foreign_key);
  217. // -----------------------------------------------------
  218. // Get the class name of the related model.
  219. // -----------------------------------------------------
  220. $class = get_class($model);
  221. // -----------------------------------------------------
  222. // Create the related models.
  223. // -----------------------------------------------------
  224. foreach ($inclusions as $inclusion)
  225. {
  226. $related = new $class;
  227. $related->exists = true;
  228. $related->attributes = (array) $inclusion;
  229. // -----------------------------------------------------
  230. // Remove the foreign key from the attributes since it
  231. // was only added to the query to help us match the models.
  232. // -----------------------------------------------------
  233. unset($related->attributes[$foreign_key]);
  234. // -----------------------------------------------------
  235. // Add the related model to the parent model's array.
  236. // -----------------------------------------------------
  237. $results[$inclusion->$foreign_key]->ignore[$include][$inclusion->id] = $related;
  238. }
  239. }
  240. }