Вход Регистрация
Файл: system/vendor/illuminate/database/Eloquent/FactoryBuilder.php
Строк: 375
<?php

namespace IlluminateDatabaseEloquent;

use 
FakerGenerator as Faker;
use 
IlluminateSupportTraitsMacroable;
use 
InvalidArgumentException;

class 
FactoryBuilder
{
    use 
Macroable;

    
/**
     * The model definitions in the container.
     *
     * @var array
     */
    
protected $definitions;

    
/**
     * The model being built.
     *
     * @var string
     */
    
protected $class;

    
/**
     * The database connection on which the model instance should be persisted.
     *
     * @var string
     */
    
protected $connection;

    
/**
     * The model states.
     *
     * @var array
     */
    
protected $states;

    
/**
     * The model after making callbacks.
     *
     * @var array
     */
    
protected $afterMaking = [];

    
/**
     * The model after creating callbacks.
     *
     * @var array
     */
    
protected $afterCreating = [];

    
/**
     * The states to apply.
     *
     * @var array
     */
    
protected $activeStates = [];

    
/**
     * The Faker instance for the builder.
     *
     * @var FakerGenerator
     */
    
protected $faker;

    
/**
     * The number of models to build.
     *
     * @var int|null
     */
    
protected $amount null;

    
/**
     * Create an new builder instance.
     *
     * @param  string  $class
     * @param  array  $definitions
     * @param  array  $states
     * @param  array  $afterMaking
     * @param  array  $afterCreating
     * @param  FakerGenerator  $faker
     * @return void
     */
    
public function __construct($class, array $definitions, array $states,
                                array 
$afterMaking, array $afterCreatingFaker $faker)
    {
        
$this->class $class;
        
$this->faker $faker;
        
$this->states $states;
        
$this->definitions $definitions;
        
$this->afterMaking $afterMaking;
        
$this->afterCreating $afterCreating;
    }

    
/**
     * Set the amount of models you wish to create / make.
     *
     * @param  int  $amount
     * @return $this
     */
    
public function times($amount)
    {
        
$this->amount $amount;

        return 
$this;
    }

    
/**
     * Set the state to be applied to the model.
     *
     * @param  string  $state
     * @return $this
     */
    
public function state($state)
    {
        return 
$this->states([$state]);
    }

    
/**
     * Set the states to be applied to the model.
     *
     * @param  array|mixed  $states
     * @return $this
     */
    
public function states($states)
    {
        
$this->activeStates is_array($states) ? $states func_get_args();

        return 
$this;
    }

    
/**
     * Set the database connection on which the model instance should be persisted.
     *
     * @param  string  $name
     * @return $this
     */
    
public function connection($name)
    {
        
$this->connection $name;

        return 
$this;
    }

    
/**
     * Create a model and persist it in the database if requested.
     *
     * @param  array  $attributes
     * @return Closure
     */
    
public function lazy(array $attributes = [])
    {
        return function () use (
$attributes) {
            return 
$this->create($attributes);
        };
    }

    
/**
     * Create a collection of models and persist them to the database.
     *
     * @param  array  $attributes
     * @return IlluminateDatabaseEloquentCollection|IlluminateDatabaseEloquentModel|mixed
     */
    
public function create(array $attributes = [])
    {
        
$results $this->make($attributes);

        if (
$results instanceof Model) {
            
$this->store(collect([$results]));

            
$this->callAfterCreating(collect([$results]));
        } else {
            
$this->store($results);

            
$this->callAfterCreating($results);
        }

        return 
$results;
    }

    
/**
     * Create a collection of models and persist them to the database.
     *
     * @param  iterable  $records
     * @return IlluminateDatabaseEloquentCollection|mixed
     */
    
public function createMany(iterable $records)
    {
        return (new 
$this->class)->newCollection(array_map(function ($attribute) {
            return 
$this->create($attribute);
        }, 
$records));
    }

    
/**
     * Set the connection name on the results and store them.
     *
     * @param  IlluminateSupportCollection  $results
     * @return void
     */
    
protected function store($results)
    {
        
$results->each(function ($model) {
            if (! isset(
$this->connection)) {
                
$model->setConnection($model->newQueryWithoutScopes()->getConnection()->getName());
            }

            
$model->save();
        });
    }

    
/**
     * Create a collection of models.
     *
     * @param  array  $attributes
     * @return IlluminateDatabaseEloquentCollection|IlluminateDatabaseEloquentModel|mixed
     */
    
public function make(array $attributes = [])
    {
        if (
$this->amount === null) {
            return 
tap($this->makeInstance($attributes), function ($instance) {
                
$this->callAfterMaking(collect([$instance]));
            });
        }

        if (
$this->amount 1) {
            return (new 
$this->class)->newCollection();
        }

        
$instances = (new $this->class)->newCollection(array_map(function () use ($attributes) {
            return 
$this->makeInstance($attributes);
        }, 
range(1$this->amount)));

        
$this->callAfterMaking($instances);

        return 
$instances;
    }

    
/**
     * Create an array of raw attribute arrays.
     *
     * @param  array  $attributes
     * @return mixed
     */
    
public function raw(array $attributes = [])
    {
        if (
$this->amount === null) {
            return 
$this->getRawAttributes($attributes);
        }

        if (
$this->amount 1) {
            return [];
        }

        return 
array_map(function () use ($attributes) {
            return 
$this->getRawAttributes($attributes);
        }, 
range(1$this->amount));
    }

    
/**
     * Get a raw attributes array for the model.
     *
     * @param  array  $attributes
     * @return mixed
     *
     * @throws InvalidArgumentException
     */
    
protected function getRawAttributes(array $attributes = [])
    {
        if (! isset(
$this->definitions[$this->class])) {
            throw new 
InvalidArgumentException("Unable to locate factory for [{$this->class}].");
        }

        
$definition call_user_func(
            
$this->definitions[$this->class],
            
$this->faker$attributes
        
);

        return 
$this->expandAttributes(
            
array_merge($this->applyStates($definition$attributes), $attributes)
        );
    }

    
/**
     * Make an instance of the model with the given attributes.
     *
     * @param  array  $attributes
     * @return IlluminateDatabaseEloquentModel
     */
    
protected function makeInstance(array $attributes = [])
    {
        return 
Model::unguarded(function () use ($attributes) {
            
$instance = new $this->class(
                
$this->getRawAttributes($attributes)
            );

            if (isset(
$this->connection)) {
                
$instance->setConnection($this->connection);
            }

            return 
$instance;
        });
    }

    
/**
     * Apply the active states to the model definition array.
     *
     * @param  array  $definition
     * @param  array  $attributes
     * @return array
     *
     * @throws InvalidArgumentException
     */
    
protected function applyStates(array $definition, array $attributes = [])
    {
        foreach (
$this->activeStates as $state) {
            if (! isset(
$this->states[$this->class][$state])) {
                if (
$this->stateHasAfterCallback($state)) {
                    continue;
                }

                throw new 
InvalidArgumentException("Unable to locate [{$state}] state for [{$this->class}].");
            }

            
$definition array_merge(
                
$definition,
                
$this->stateAttributes($state$attributes)
            );
        }

        return 
$definition;
    }

    
/**
     * Get the state attributes.
     *
     * @param  string  $state
     * @param  array  $attributes
     * @return array
     */
    
protected function stateAttributes($state, array $attributes)
    {
        
$stateAttributes $this->states[$this->class][$state];

        if (! 
is_callable($stateAttributes)) {
            return 
$stateAttributes;
        }

        return 
$stateAttributes($this->faker$attributes);
    }

    
/**
     * Expand all attributes to their underlying values.
     *
     * @param  array  $attributes
     * @return array
     */
    
protected function expandAttributes(array $attributes)
    {
        foreach (
$attributes as &$attribute) {
            if (
is_callable($attribute) && ! is_string($attribute) && ! is_array($attribute)) {
                
$attribute $attribute($attributes);
            }

            if (
$attribute instanceof static) {
                
$attribute $attribute->create()->getKey();
            }

            if (
$attribute instanceof Model) {
                
$attribute $attribute->getKey();
            }
        }

        return 
$attributes;
    }

    
/**
     * Run after making callbacks on a collection of models.
     *
     * @param  IlluminateSupportCollection  $models
     * @return void
     */
    
public function callAfterMaking($models)
    {
        
$this->callAfter($this->afterMaking$models);
    }

    
/**
     * Run after creating callbacks on a collection of models.
     *
     * @param  IlluminateSupportCollection  $models
     * @return void
     */
    
public function callAfterCreating($models)
    {
        
$this->callAfter($this->afterCreating$models);
    }

    
/**
     * Call after callbacks for each model and state.
     *
     * @param  array  $afterCallbacks
     * @param  IlluminateSupportCollection  $models
     * @return void
     */
    
protected function callAfter(array $afterCallbacks$models)
    {
        
$states array_merge(['default'], $this->activeStates);

        
$models->each(function ($model) use ($states$afterCallbacks) {
            foreach (
$states as $state) {
                
$this->callAfterCallbacks($afterCallbacks$model$state);
            }
        });
    }

    
/**
     * Call after callbacks for each model and state.
     *
     * @param  array  $afterCallbacks
     * @param  IlluminateDatabaseEloquentModel  $model
     * @param  string  $state
     * @return void
     */
    
protected function callAfterCallbacks(array $afterCallbacks$model$state)
    {
        if (! isset(
$afterCallbacks[$this->class][$state])) {
            return;
        }

        foreach (
$afterCallbacks[$this->class][$state] as $callback) {
            
$callback($model$this->faker);
        }
    }

    
/**
     * Determine if the given state has an "after" callback.
     *
     * @param  string  $state
     * @return bool
     */
    
protected function stateHasAfterCallback($state)
    {
        return isset(
$this->afterMaking[$this->class][$state]) ||
               isset(
$this->afterCreating[$this->class][$state]);
    }
}
Онлайн: 0
Реклама