hydrator.php 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203
  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 belongs_to key to allow the query to be fetched without
  58. // any problems, since the belongs_to method actually gets the attribute.
  59. $eloquent->attributes[$spoof = $include.'_id'] = 0;
  60. $relationship = $eloquent->$include();
  61. unset($eloquent->attributes[$spoof]);
  62. // Reset the WHERE clause and bindings on the query. We'll add our own WHERE clause soon.
  63. // This will allow us to load a range of related models instead of only one.
  64. $relationship->query->reset_where();
  65. // Initialize the relationship attribute on the parents. As expected, "many" relationships
  66. // are initialized to an array and "one" relationships are initialized to null.
  67. foreach ($parents as &$parent)
  68. {
  69. $parent->ignore[$include] = (in_array($eloquent->relating, array('has_many', 'has_and_belongs_to_many'))) ? array() : null;
  70. }
  71. if (in_array($relating = $eloquent->relating, array('has_one', 'has_many', 'belongs_to')))
  72. {
  73. return static::$relating($relationship, $parents, $eloquent->relating_key, $include);
  74. }
  75. else
  76. {
  77. static::has_and_belongs_to_many($relationship, $parents, $eloquent->relating_key, $eloquent->relating_table, $include);
  78. }
  79. }
  80. /**
  81. * Eagerly load a 1:1 relationship.
  82. *
  83. * @param object $relationship
  84. * @param array $parents
  85. * @param string $relating_key
  86. * @param string $relating
  87. * @param string $include
  88. * @return void
  89. */
  90. private static function has_one($relationship, &$parents, $relating_key, $include)
  91. {
  92. foreach ($relationship->where_in($relating_key, array_keys($parents))->get() as $key => $child)
  93. {
  94. $parents[$child->$relating_key]->ignore[$include] = $child;
  95. }
  96. }
  97. /**
  98. * Eagerly load a 1:* relationship.
  99. *
  100. * @param object $relationship
  101. * @param array $parents
  102. * @param string $relating_key
  103. * @param string $relating
  104. * @param string $include
  105. * @return void
  106. */
  107. private static function has_many($relationship, &$parents, $relating_key, $include)
  108. {
  109. foreach ($relationship->where_in($relating_key, array_keys($parents))->get() as $key => $child)
  110. {
  111. $parents[$child->$relating_key]->ignore[$include][$child->id] = $child;
  112. }
  113. }
  114. /**
  115. * Eagerly load a 1:1 belonging relationship.
  116. *
  117. * @param object $relationship
  118. * @param array $parents
  119. * @param string $relating_key
  120. * @param string $include
  121. * @return void
  122. */
  123. private static function belongs_to($relationship, &$parents, $relating_key, $include)
  124. {
  125. $keys = array();
  126. foreach ($parents as &$parent)
  127. {
  128. $keys[] = $parent->$relating_key;
  129. }
  130. $children = $relationship->where_in('id', array_unique($keys))->get();
  131. foreach ($parents as &$parent)
  132. {
  133. if (array_key_exists($parent->$relating_key, $children))
  134. {
  135. $parent->ignore[$include] = $children[$parent->$relating_key];
  136. }
  137. }
  138. }
  139. /**
  140. * Eagerly load a many-to-many relationship.
  141. *
  142. * @param object $relationship
  143. * @param array $parents
  144. * @param string $relating_key
  145. * @param string $relating_table
  146. * @param string $include
  147. *
  148. * @return void
  149. */
  150. private static function has_and_belongs_to_many($relationship, &$parents, $relating_key, $relating_table, $include)
  151. {
  152. // The model "has and belongs to many" method sets the SELECT clause; however, we need
  153. // to clear it here since we will be adding the foreign key to the select.
  154. $relationship->query->select = null;
  155. $relationship->query->where_in($relating_table.'.'.$relating_key, array_keys($parents));
  156. // The foreign key is added to the select to allow us to easily match the models back to their parents.
  157. // Otherwise, there would be no apparent connection between the models to allow us to match them.
  158. $children = $relationship->query->get(array(Model::table(get_class($relationship)).'.*', $relating_table.'.'.$relating_key));
  159. $class = get_class($relationship);
  160. foreach ($children as $child)
  161. {
  162. $related = new $class;
  163. $related->attributes = (array) $child;
  164. $related->exists = true;
  165. // Remove the foreign key since it was only added to the query to help match the models.
  166. unset($related->attributes[$relating_key]);
  167. $parents[$child->$relating_key]->ignore[$include][$child->id] = $related;
  168. }
  169. }
  170. }