hydrator.php 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213
  1. <?php namespace System\DB\Eloquent;
  2. use System\DB\Eloquent;
  3. class Hydrator {
  4. /**
  5. * Load the array of hydrated models and their eager relationships.
  6. *
  7. * @param object $eloquent
  8. * @return array
  9. */
  10. public static function hydrate($eloquent)
  11. {
  12. $results = static::base(get_class($eloquent), $eloquent->query->get());
  13. if (count($results) > 0)
  14. {
  15. foreach ($eloquent->includes as $include)
  16. {
  17. if ( ! method_exists($eloquent, $include))
  18. {
  19. throw new \Exception("Attempting to eager load [$include], but the relationship is not defined.");
  20. }
  21. static::eagerly($eloquent, $results, $include);
  22. }
  23. }
  24. return $results;
  25. }
  26. /**
  27. * Hydrate the base models for a query.
  28. *
  29. * The resulting model array is keyed by the primary keys of the models.
  30. * This allows the models to easily be matched to their children.
  31. *
  32. * @param string $class
  33. * @param array $results
  34. * @return array
  35. */
  36. private static function base($class, $results)
  37. {
  38. $models = array();
  39. foreach ($results as $result)
  40. {
  41. $model = new $class;
  42. $model->attributes = (array) $result;
  43. $model->exists = true;
  44. $models[$model->id] = $model;
  45. }
  46. return $models;
  47. }
  48. /**
  49. * Eagerly load a relationship.
  50. *
  51. * @param object $eloquent
  52. * @param array $parents
  53. * @param string $include
  54. * @return void
  55. */
  56. private static function eagerly($eloquent, &$parents, $include)
  57. {
  58. // Get the relationship Eloquent model.
  59. //
  60. // We temporarily spoof the belongs_to key to allow the query to be fetched without
  61. // any problems, since the belongs_to method actually gets the attribute.
  62. $eloquent->attributes[$spoof = $include.'_id'] = 0;
  63. $relationship = $eloquent->$include();
  64. unset($eloquent->attributes[$spoof]);
  65. // Reset the WHERE clause and bindings on the query. We'll add our own WHERE clause soon.
  66. $relationship->query->where = 'WHERE 1 = 1';
  67. $relationship->query->bindings = array();
  68. // Initialize the relationship attribute on the parents. As expected, "many" relationships
  69. // are initialized to an array and "one" relationships are initialized to null.
  70. foreach ($parents as &$parent)
  71. {
  72. $parent->ignore[$include] = (strpos($eloquent->relating, 'has_many') === 0) ? array() : null;
  73. }
  74. if ($eloquent->relating == 'has_one')
  75. {
  76. static::eagerly_load_one($relationship, $parents, $eloquent->relating_key, $include);
  77. }
  78. elseif ($eloquent->relating == 'has_many')
  79. {
  80. static::eagerly_load_many($relationship, $parents, $eloquent->relating_key, $include);
  81. }
  82. elseif ($eloquent->relating == 'belongs_to')
  83. {
  84. static::eagerly_load_belonging($relationship, $parents, $eloquent->relating_key, $include);
  85. }
  86. else
  87. {
  88. static::eagerly_load_many_to_many($relationship, $parents, $eloquent->relating_key, $eloquent->relating_table, $include);
  89. }
  90. }
  91. /**
  92. * Eagerly load a 1:1 relationship.
  93. *
  94. * @param object $relationship
  95. * @param array $parents
  96. * @param string $relating_key
  97. * @param string $relating
  98. * @param string $include
  99. * @return void
  100. */
  101. private static function eagerly_load_one($relationship, &$parents, $relating_key, $include)
  102. {
  103. foreach ($relationship->where_in($relating_key, array_keys($parents))->get() as $key => $child)
  104. {
  105. $parents[$child->$relating_key]->ignore[$include] = $child;
  106. }
  107. }
  108. /**
  109. * Eagerly load a 1:* relationship.
  110. *
  111. * @param object $relationship
  112. * @param array $parents
  113. * @param string $relating_key
  114. * @param string $relating
  115. * @param string $include
  116. * @return void
  117. */
  118. private static function eagerly_load_many($relationship, &$parents, $relating_key, $include)
  119. {
  120. foreach ($relationship->where_in($relating_key, array_keys($parents))->get() as $key => $child)
  121. {
  122. $parents[$child->$relating_key]->ignore[$include][$child->id] = $child;
  123. }
  124. }
  125. /**
  126. * Eagerly load a 1:1 belonging relationship.
  127. *
  128. * @param object $relationship
  129. * @param array $parents
  130. * @param string $relating_key
  131. * @param string $include
  132. * @return void
  133. */
  134. private static function eagerly_load_belonging($relationship, &$parents, $relating_key, $include)
  135. {
  136. $keys = array();
  137. foreach ($parents as &$parent)
  138. {
  139. $keys[] = $parent->$relating_key;
  140. }
  141. $children = $relationship->where_in('id', array_unique($keys))->get();
  142. foreach ($parents as &$parent)
  143. {
  144. if (array_key_exists($parent->$relating_key, $children))
  145. {
  146. $parent->ignore[$include] = $children[$parent->$relating_key];
  147. }
  148. }
  149. }
  150. /**
  151. * Eagerly load a many-to-many relationship.
  152. *
  153. * @param object $relationship
  154. * @param array $parents
  155. * @param string $relating_key
  156. * @param string $relating_table
  157. * @param string $include
  158. *
  159. * @return void
  160. */
  161. private static function eagerly_load_many_to_many($relationship, &$parents, $relating_key, $relating_table, $include)
  162. {
  163. $relationship->query->select = null;
  164. $relationship->query->where_in($relating_table.'.'.$relating_key, array_keys($parents));
  165. // The foreign key is added to the select to allow us to easily match the models back to their parents.
  166. $children = $relationship->query->get(array(Eloquent::table(get_class($relationship)).'.*', $relating_table.'.'.$relating_key));
  167. $class = get_class($relationship);
  168. foreach ($children as $child)
  169. {
  170. $related = new $class;
  171. $related->attributes = (array) $child;
  172. $related->exists = true;
  173. // Remove the foreign key since it was added to the query to help match to the children.
  174. unset($related->attributes[$relating_key]);
  175. $parents[$child->$relating_key]->ignore[$include][$child->id] = $related;
  176. }
  177. }
  178. }