home
/
zktecojo
/
public_html
/
vendor
/
laravel
/
framework
/
src
/
Illuminate
/
Database
/
Eloquent
➕ New
📤 Upload
✎ Editing:
Builder.php
← Back
<?php namespace Illuminate\Database\Eloquent; use Closure; use Illuminate\Database\Query\Expression; use Illuminate\Database\Eloquent\Relations\Relation; use Illuminate\Database\Query\Builder as QueryBuilder; class Builder { /** * The base query builder instance. * * @var \Illuminate\Database\Query\Builder */ protected $query; /** * The model being queried. * * @var \Illuminate\Database\Eloquent\Model */ protected $model; /** * The relationships that should be eager loaded. * * @var array */ protected $eagerLoad = array(); /** * All of the registered builder macros. * * @var array */ protected $macros = array(); /** * A replacement for the typical delete function. * * @var \Closure */ protected $onDelete; /** * The methods that should be returned from query builder. * * @var array */ protected $passthru = array( 'toSql', 'lists', 'insert', 'insertGetId', 'pluck', 'count', 'min', 'max', 'avg', 'sum', 'exists', 'getBindings', ); /** * Create a new Eloquent query builder instance. * * @param \Illuminate\Database\Query\Builder $query * @return void */ public function __construct(QueryBuilder $query) { $this->query = $query; } /** * Find a model by its primary key. * * @param mixed $id * @param array $columns * @return \Illuminate\Database\Eloquent\Model|static|null */ public function find($id, $columns = array('*')) { if (is_array($id)) { return $this->findMany($id, $columns); } $this->query->where($this->model->getQualifiedKeyName(), '=', $id); return $this->first($columns); } /** * Find a model by its primary key. * * @param array $id * @param array $columns * @return \Illuminate\Database\Eloquent\Model|Collection|static */ public function findMany($id, $columns = array('*')) { if (empty($id)) return $this->model->newCollection(); $this->query->whereIn($this->model->getQualifiedKeyName(), $id); return $this->get($columns); } /** * Find a model by its primary key or throw an exception. * * @param mixed $id * @param array $columns * @return \Illuminate\Database\Eloquent\Model|static * * @throws \Illuminate\Database\Eloquent\ModelNotFoundException */ public function findOrFail($id, $columns = array('*')) { if ( ! is_null($model = $this->find($id, $columns))) return $model; throw (new ModelNotFoundException)->setModel(get_class($this->model)); } /** * Execute the query and get the first result. * * @param array $columns * @return \Illuminate\Database\Eloquent\Model|static|null */ public function first($columns = array('*')) { return $this->take(1)->get($columns)->first(); } /** * Execute the query and get the first result or throw an exception. * * @param array $columns * @return \Illuminate\Database\Eloquent\Model|static * * @throws \Illuminate\Database\Eloquent\ModelNotFoundException */ public function firstOrFail($columns = array('*')) { if ( ! is_null($model = $this->first($columns))) return $model; throw (new ModelNotFoundException)->setModel(get_class($this->model)); } /** * Execute the query as a "select" statement. * * @param array $columns * @return \Illuminate\Database\Eloquent\Collection|static[] */ public function get($columns = array('*')) { $models = $this->getModels($columns); // If we actually found models we will also eager load any relationships that // have been specified as needing to be eager loaded, which will solve the // n+1 query issue for the developers to avoid running a lot of queries. if (count($models) > 0) { $models = $this->eagerLoadRelations($models); } return $this->model->newCollection($models); } /** * Pluck a single column from the database. * * @param string $column * @return mixed */ public function pluck($column) { $result = $this->first(array($column)); if ($result) return $result->{$column}; } /** * Chunk the results of the query. * * @param int $count * @param callable $callback * @return void */ public function chunk($count, callable $callback) { $results = $this->forPage($page = 1, $count)->get(); while (count($results) > 0) { // On each chunk result set, we will pass them to the callback and then let the // developer take care of everything within the callback, which allows us to // keep the memory low for spinning through large result sets for working. call_user_func($callback, $results); $page++; $results = $this->forPage($page, $count)->get(); } } /** * Get an array with the values of a given column. * * @param string $column * @param string $key * @return array */ public function lists($column, $key = null) { $results = $this->query->lists($column, $key); // If the model has a mutator for the requested column, we will spin through // the results and mutate the values so that the mutated version of these // columns are returned as you would expect from these Eloquent models. if ($this->model->hasGetMutator($column)) { foreach ($results as $key => &$value) { $fill = array($column => $value); $value = $this->model->newFromBuilder($fill)->$column; } } return $results; } /** * Get a paginator for the "select" statement. * * @param int $perPage * @param array $columns * @return \Illuminate\Pagination\Paginator */ public function paginate($perPage = null, $columns = array('*')) { $perPage = $perPage ?: $this->model->getPerPage(); $paginator = $this->query->getConnection()->getPaginator(); if (isset($this->query->groups)) { return $this->groupedPaginate($paginator, $perPage, $columns); } return $this->ungroupedPaginate($paginator, $perPage, $columns); } /** * Get a paginator for a grouped statement. * * @param \Illuminate\Pagination\Factory $paginator * @param int $perPage * @param array $columns * @return \Illuminate\Pagination\Paginator */ protected function groupedPaginate($paginator, $perPage, $columns) { $results = $this->get($columns)->all(); return $this->query->buildRawPaginator($paginator, $results, $perPage); } /** * Get a paginator for an ungrouped statement. * * @param \Illuminate\Pagination\Factory $paginator * @param int $perPage * @param array $columns * @return \Illuminate\Pagination\Paginator */ protected function ungroupedPaginate($paginator, $perPage, $columns) { $total = $this->query->getPaginationCount(); // Once we have the paginator we need to set the limit and offset values for // the query so we can get the properly paginated items. Once we have an // array of items we can create the paginator instances for the items. $page = $paginator->getCurrentPage($total); $this->query->forPage($page, $perPage); return $paginator->make($this->get($columns)->all(), $total, $perPage); } /** * Get a paginator only supporting simple next and previous links. * * This is more efficient on larger data-sets, etc. * * @param int $perPage * @param array $columns * @return \Illuminate\Pagination\Paginator */ public function simplePaginate($perPage = null, $columns = array('*')) { $paginator = $this->query->getConnection()->getPaginator(); $page = $paginator->getCurrentPage(); $perPage = $perPage ?: $this->model->getPerPage(); $this->query->skip(($page - 1) * $perPage)->take($perPage + 1); return $paginator->make($this->get($columns)->all(), $perPage); } /** * Update a record in the database. * * @param array $values * @return int */ public function update(array $values) { return $this->query->update($this->addUpdatedAtColumn($values)); } /** * Increment a column's value by a given amount. * * @param string $column * @param int $amount * @param array $extra * @return int */ public function increment($column, $amount = 1, array $extra = array()) { $extra = $this->addUpdatedAtColumn($extra); return $this->query->increment($column, $amount, $extra); } /** * Decrement a column's value by a given amount. * * @param string $column * @param int $amount * @param array $extra * @return int */ public function decrement($column, $amount = 1, array $extra = array()) { $extra = $this->addUpdatedAtColumn($extra); return $this->query->decrement($column, $amount, $extra); } /** * Add the "updated at" column to an array of values. * * @param array $values * @return array */ protected function addUpdatedAtColumn(array $values) { if ( ! $this->model->usesTimestamps()) return $values; $column = $this->model->getUpdatedAtColumn(); return array_add($values, $column, $this->model->freshTimestampString()); } /** * Delete a record from the database. * * @return mixed */ public function delete() { if (isset($this->onDelete)) { return call_user_func($this->onDelete, $this); } return $this->query->delete(); } /** * Run the default delete function on the builder. * * @return mixed */ public function forceDelete() { return $this->query->delete(); } /** * Register a replacement for the default delete function. * * @param \Closure $callback * @return void */ public function onDelete(Closure $callback) { $this->onDelete = $callback; } /** * Get the hydrated models without eager loading. * * @param array $columns * @return \Illuminate\Database\Eloquent\Model[] */ public function getModels($columns = array('*')) { // First, we will simply get the raw results from the query builders which we // can use to populate an array with Eloquent models. We will pass columns // that should be selected as well, which are typically just everything. $results = $this->query->get($columns); $connection = $this->model->getConnectionName(); $models = array(); // Once we have the results, we can spin through them and instantiate a fresh // model instance for each records we retrieved from the database. We will // also set the proper connection name for the model after we create it. foreach ($results as $result) { $models[] = $model = $this->model->newFromBuilder($result); $model->setConnection($connection); } return $models; } /** * Eager load the relationships for the models. * * @param array $models * @return array */ public function eagerLoadRelations(array $models) { foreach ($this->eagerLoad as $name => $constraints) { // For nested eager loads we'll skip loading them here and they will be set as an // eager load on the query to retrieve the relation so that they will be eager // loaded on that query, because that is where they get hydrated as models. if (strpos($name, '.') === false) { $models = $this->loadRelation($models, $name, $constraints); } } return $models; } /** * Eagerly load the relationship on a set of models. * * @param array $models * @param string $name * @param \Closure $constraints * @return array */ protected function loadRelation(array $models, $name, Closure $constraints) { // First we will "back up" the existing where conditions on the query so we can // add our eager constraints. Then we will merge the wheres that were on the // query back to it in order that any where conditions might be specified. $relation = $this->getRelation($name); $relation->addEagerConstraints($models); call_user_func($constraints, $relation); $models = $relation->initRelation($models, $name); // Once we have the results, we just match those back up to their parent models // using the relationship instance. Then we just return the finished arrays // of models which have been eagerly hydrated and are readied for return. $results = $relation->getEager(); return $relation->match($models, $results, $name); } /** * Get the relation instance for the given relation name. * * @param string $relation * @return \Illuminate\Database\Eloquent\Relations\Relation */ public function getRelation($relation) { // We want to run a relationship query without any constrains so that we will // not have to remove these where clauses manually which gets really hacky // and is error prone while we remove the developer's own where clauses. $query = Relation::noConstraints(function() use ($relation) { return $this->getModel()->$relation(); }); $nested = $this->nestedRelations($relation); // If there are nested relationships set on the query, we will put those onto // the query instances so that they can be handled after this relationship // is loaded. In this way they will all trickle down as they are loaded. if (count($nested) > 0) { $query->getQuery()->with($nested); } return $query; } /** * Get the deeply nested relations for a given top-level relation. * * @param string $relation * @return array */ protected function nestedRelations($relation) { $nested = array(); // We are basically looking for any relationships that are nested deeper than // the given top-level relationship. We will just check for any relations // that start with the given top relations and adds them to our arrays. foreach ($this->eagerLoad as $name => $constraints) { if ($this->isNested($name, $relation)) { $nested[substr($name, strlen($relation.'.'))] = $constraints; } } return $nested; } /** * Determine if the relationship is nested. * * @param string $name * @param string $relation * @return bool */ protected function isNested($name, $relation) { $dots = str_contains($name, '.'); return $dots && starts_with($name, $relation.'.'); } /** * Add a basic where clause to the query. * * @param string $column * @param string $operator * @param mixed $value * @param string $boolean * @return $this */ public function where($column, $operator = null, $value = null, $boolean = 'and') { if ($column instanceof Closure) { $query = $this->model->newQueryWithoutScopes(); call_user_func($column, $query); $this->query->addNestedWhereQuery($query->getQuery(), $boolean); } else { call_user_func_array(array($this->query, 'where'), func_get_args()); } return $this; } /** * Add an "or where" clause to the query. * * @param string $column * @param string $operator * @param mixed $value * @return \Illuminate\Database\Eloquent\Builder|static */ public function orWhere($column, $operator = null, $value = null) { return $this->where($column, $operator, $value, 'or'); } /** * Add a relationship count condition to the query. * * @param string $relation * @param string $operator * @param int $count * @param string $boolean * @param \Closure|null $callback * @return \Illuminate\Database\Eloquent\Builder|static */ public function has($relation, $operator = '>=', $count = 1, $boolean = 'and', Closure $callback = null) { $relation = $this->getHasRelationQuery($relation); $query = $relation->getRelationCountQuery($relation->getRelated()->newQuery(), $this); if ($callback) call_user_func($callback, $query); return $this->addHasWhere($query, $relation, $operator, $count, $boolean); } /** * Add a relationship count condition to the query. * * @param string $relation * @param string $boolean * @param \Closure|null $callback * @return \Illuminate\Database\Eloquent\Builder|static */ public function doesntHave($relation, $boolean = 'and', Closure $callback = null) { return $this->has($relation, '<', 1, $boolean, $callback); } /** * Add a relationship count condition to the query with where clauses. * * @param string $relation * @param \Closure $callback * @param string $operator * @param int $count * @return \Illuminate\Database\Eloquent\Builder|static */ public function whereHas($relation, Closure $callback, $operator = '>=', $count = 1) { return $this->has($relation, $operator, $count, 'and', $callback); } /** * Add a relationship count condition to the query with where clauses. * * @param string $relation * @param \Closure|null $callback * @return \Illuminate\Database\Eloquent\Builder|static */ public function whereDoesntHave($relation, Closure $callback = null) { return $this->doesntHave($relation, 'and', $callback); } /** * Add a relationship count condition to the query with an "or". * * @param string $relation * @param string $operator * @param int $count * @return \Illuminate\Database\Eloquent\Builder|static */ public function orHas($relation, $operator = '>=', $count = 1) { return $this->has($relation, $operator, $count, 'or'); } /** * Add a relationship count condition to the query with where clauses and an "or". * * @param string $relation * @param \Closure $callback * @param string $operator * @param int $count * @return \Illuminate\Database\Eloquent\Builder|static */ public function orWhereHas($relation, Closure $callback, $operator = '>=', $count = 1) { return $this->has($relation, $operator, $count, 'or', $callback); } /** * Add the "has" condition where clause to the query. * * @param \Illuminate\Database\Eloquent\Builder $hasQuery * @param \Illuminate\Database\Eloquent\Relations\Relation $relation * @param string $operator * @param int $count * @param string $boolean * @return \Illuminate\Database\Eloquent\Builder */ protected function addHasWhere(Builder $hasQuery, Relation $relation, $operator, $count, $boolean) { $this->mergeWheresToHas($hasQuery, $relation); if (is_numeric($count)) { $count = new Expression($count); } return $this->where(new Expression('('.$hasQuery->toSql().')'), $operator, $count, $boolean); } /** * Merge the "wheres" from a relation query to a has query. * * @param \Illuminate\Database\Eloquent\Builder $hasQuery * @param \Illuminate\Database\Eloquent\Relations\Relation $relation * @return void */ protected function mergeWheresToHas(Builder $hasQuery, Relation $relation) { // Here we have the "has" query and the original relation. We need to copy over any // where clauses the developer may have put in the relationship function over to // the has query, and then copy the bindings from the "has" query to the main. $relationQuery = $relation->getBaseQuery(); $hasQuery = $hasQuery->getModel()->removeGlobalScopes($hasQuery); $hasQuery->mergeWheres( $relationQuery->wheres, $relationQuery->getBindings() ); $this->query->mergeBindings($hasQuery->getQuery()); } /** * Get the "has relation" base query instance. * * @param string $relation * @return \Illuminate\Database\Eloquent\Builder */ protected function getHasRelationQuery($relation) { return Relation::noConstraints(function() use ($relation) { return $this->getModel()->$relation(); }); } /** * Set the relationships that should be eager loaded. * * @param mixed $relations * @return $this */ public function with($relations) { if (is_string($relations)) $relations = func_get_args(); $eagers = $this->parseRelations($relations); $this->eagerLoad = array_merge($this->eagerLoad, $eagers); return $this; } /** * Parse a list of relations into individuals. * * @param array $relations * @return array */ protected function parseRelations(array $relations) { $results = array(); foreach ($relations as $name => $constraints) { // If the "relation" value is actually a numeric key, we can assume that no // constraints have been specified for the eager load and we'll just put // an empty Closure with the loader so that we can treat all the same. if (is_numeric($name)) { $f = function() {}; list($name, $constraints) = array($constraints, $f); } // We need to separate out any nested includes. Which allows the developers // to load deep relationships using "dots" without stating each level of // the relationship with its own key in the array of eager load names. $results = $this->parseNested($name, $results); $results[$name] = $constraints; } return $results; } /** * Parse the nested relationships in a relation. * * @param string $name * @param array $results * @return array */ protected function parseNested($name, $results) { $progress = array(); // If the relation has already been set on the result array, we will not set it // again, since that would override any constraints that were already placed // on the relationships. We will only set the ones that are not specified. foreach (explode('.', $name) as $segment) { $progress[] = $segment; if ( ! isset($results[$last = implode('.', $progress)])) { $results[$last] = function() {}; } } return $results; } /** * Call the given model scope on the underlying model. * * @param string $scope * @param array $parameters * @return \Illuminate\Database\Query\Builder */ protected function callScope($scope, $parameters) { array_unshift($parameters, $this); return call_user_func_array(array($this->model, $scope), $parameters) ?: $this; } /** * Get the underlying query builder instance. * * @return \Illuminate\Database\Query\Builder|static */ public function getQuery() { return $this->query; } /** * Set the underlying query builder instance. * * @param \Illuminate\Database\Query\Builder $query * @return void */ public function setQuery($query) { $this->query = $query; } /** * Get the relationships being eagerly loaded. * * @return array */ public function getEagerLoads() { return $this->eagerLoad; } /** * Set the relationships being eagerly loaded. * * @param array $eagerLoad * @return void */ public function setEagerLoads(array $eagerLoad) { $this->eagerLoad = $eagerLoad; } /** * Get the model instance being queried. * * @return \Illuminate\Database\Eloquent\Model */ public function getModel() { return $this->model; } /** * Set a model instance for the model being queried. * * @param \Illuminate\Database\Eloquent\Model $model * @return $this */ public function setModel(Model $model) { $this->model = $model; $this->query->from($model->getTable()); return $this; } /** * Extend the builder with a given callback. * * @param string $name * @param \Closure $callback * @return void */ public function macro($name, Closure $callback) { $this->macros[$name] = $callback; } /** * Get the given macro by name. * * @param string $name * @return \Closure */ public function getMacro($name) { return array_get($this->macros, $name); } /** * Dynamically handle calls into the query instance. * * @param string $method * @param array $parameters * @return mixed */ public function __call($method, $parameters) { if (isset($this->macros[$method])) { array_unshift($parameters, $this); return call_user_func_array($this->macros[$method], $parameters); } elseif (method_exists($this->model, $scope = 'scope'.ucfirst($method))) { return $this->callScope($scope, $parameters); } $result = call_user_func_array(array($this->query, $method), $parameters); return in_array($method, $this->passthru) ? $result : $this; } /** * Force a clone of the underlying query builder when cloning. * * @return void */ public function __clone() { $this->query = clone $this->query; } }
💾 Save Changes
Cancel
📤 Upload File
×
Select File
Upload
Cancel
➕ Create New
×
Type
📄 File
📁 Folder
Name
Create
Cancel
✎ Rename Item
×
Current Name
New Name
Rename
Cancel
🔐 Change Permissions
×
Target File
Permission (e.g., 0755, 0644)
0755
0644
0777
Apply
Cancel