hydrator.php 5.9 KB

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