hydrator.php 5.5 KB

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