308 lines
		
	
	
		
			7.1 KiB
		
	
	
	
		
			PHP
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			308 lines
		
	
	
		
			7.1 KiB
		
	
	
	
		
			PHP
		
	
	
		
			Executable File
		
	
	
	
	
| <?php namespace Illuminate\Database;
 | |
| 
 | |
| use Illuminate\Support\Str;
 | |
| use InvalidArgumentException;
 | |
| use Illuminate\Database\Connectors\ConnectionFactory;
 | |
| 
 | |
| class DatabaseManager implements ConnectionResolverInterface {
 | |
| 
 | |
| 	/**
 | |
| 	 * The application instance.
 | |
| 	 *
 | |
| 	 * @var \Illuminate\Foundation\Application
 | |
| 	 */
 | |
| 	protected $app;
 | |
| 
 | |
| 	/**
 | |
| 	 * The database connection factory instance.
 | |
| 	 *
 | |
| 	 * @var \Illuminate\Database\Connectors\ConnectionFactory
 | |
| 	 */
 | |
| 	protected $factory;
 | |
| 
 | |
| 	/**
 | |
| 	 * The active connection instances.
 | |
| 	 *
 | |
| 	 * @var array
 | |
| 	 */
 | |
| 	protected $connections = array();
 | |
| 
 | |
| 	/**
 | |
| 	 * The custom connection resolvers.
 | |
| 	 *
 | |
| 	 * @var array
 | |
| 	 */
 | |
| 	protected $extensions = array();
 | |
| 
 | |
| 	/**
 | |
| 	 * Create a new database manager instance.
 | |
| 	 *
 | |
| 	 * @param  \Illuminate\Foundation\Application  $app
 | |
| 	 * @param  \Illuminate\Database\Connectors\ConnectionFactory  $factory
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	public function __construct($app, ConnectionFactory $factory)
 | |
| 	{
 | |
| 		$this->app = $app;
 | |
| 		$this->factory = $factory;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get a database connection instance.
 | |
| 	 *
 | |
| 	 * @param  string  $name
 | |
| 	 * @return \Illuminate\Database\Connection
 | |
| 	 */
 | |
| 	public function connection($name = null)
 | |
| 	{
 | |
| 		list($name, $type) = $this->parseConnectionName($name);
 | |
| 
 | |
| 		// If we haven't created this connection, we'll create it based on the config
 | |
| 		// provided in the application. Once we've created the connections we will
 | |
| 		// set the "fetch mode" for PDO which determines the query return types.
 | |
| 		if ( ! isset($this->connections[$name]))
 | |
| 		{
 | |
| 			$connection = $this->makeConnection($name);
 | |
| 
 | |
| 			$this->setPdoForType($connection, $type);
 | |
| 
 | |
| 			$this->connections[$name] = $this->prepare($connection);
 | |
| 		}
 | |
| 
 | |
| 		return $this->connections[$name];
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Parse the connection into an array of the name and read / write type.
 | |
| 	 *
 | |
| 	 * @param  string  $name
 | |
| 	 * @return array
 | |
| 	 */
 | |
| 	protected function parseConnectionName($name)
 | |
| 	{
 | |
| 		$name = $name ?: $this->getDefaultConnection();
 | |
| 
 | |
| 		return Str::endsWith($name, ['::read', '::write'])
 | |
|                             ? explode('::', $name, 2) : [$name, null];
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Disconnect from the given database and remove from local cache.
 | |
| 	 *
 | |
| 	 * @param  string  $name
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	public function purge($name = null)
 | |
| 	{
 | |
| 		$this->disconnect($name);
 | |
| 
 | |
| 		unset($this->connections[$name]);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Disconnect from the given database.
 | |
| 	 *
 | |
| 	 * @param  string  $name
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	public function disconnect($name = null)
 | |
| 	{
 | |
| 		if (isset($this->connections[$name = $name ?: $this->getDefaultConnection()]))
 | |
| 		{
 | |
| 			$this->connections[$name]->disconnect();
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Reconnect to the given database.
 | |
| 	 *
 | |
| 	 * @param  string  $name
 | |
| 	 * @return \Illuminate\Database\Connection
 | |
| 	 */
 | |
| 	public function reconnect($name = null)
 | |
| 	{
 | |
| 		$this->disconnect($name = $name ?: $this->getDefaultConnection());
 | |
| 
 | |
| 		if ( ! isset($this->connections[$name]))
 | |
| 		{
 | |
| 			return $this->connection($name);
 | |
| 		}
 | |
| 
 | |
| 		return $this->refreshPdoConnections($name);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Refresh the PDO connections on a given connection.
 | |
| 	 *
 | |
| 	 * @param  string  $name
 | |
| 	 * @return \Illuminate\Database\Connection
 | |
| 	 */
 | |
| 	protected function refreshPdoConnections($name)
 | |
| 	{
 | |
| 		$fresh = $this->makeConnection($name);
 | |
| 
 | |
| 		return $this->connections[$name]
 | |
|                                 ->setPdo($fresh->getPdo())
 | |
|                                 ->setReadPdo($fresh->getReadPdo());
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Make the database connection instance.
 | |
| 	 *
 | |
| 	 * @param  string  $name
 | |
| 	 * @return \Illuminate\Database\Connection
 | |
| 	 */
 | |
| 	protected function makeConnection($name)
 | |
| 	{
 | |
| 		$config = $this->getConfig($name);
 | |
| 
 | |
| 		// First we will check by the connection name to see if an extension has been
 | |
| 		// registered specifically for that connection. If it has we will call the
 | |
| 		// Closure and pass it the config allowing it to resolve the connection.
 | |
| 		if (isset($this->extensions[$name]))
 | |
| 		{
 | |
| 			return call_user_func($this->extensions[$name], $config, $name);
 | |
| 		}
 | |
| 
 | |
| 		$driver = $config['driver'];
 | |
| 
 | |
| 		// Next we will check to see if an extension has been registered for a driver
 | |
| 		// and will call the Closure if so, which allows us to have a more generic
 | |
| 		// resolver for the drivers themselves which applies to all connections.
 | |
| 		if (isset($this->extensions[$driver]))
 | |
| 		{
 | |
| 			return call_user_func($this->extensions[$driver], $config, $name);
 | |
| 		}
 | |
| 
 | |
| 		return $this->factory->make($config, $name);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Prepare the database connection instance.
 | |
| 	 *
 | |
| 	 * @param  \Illuminate\Database\Connection  $connection
 | |
| 	 * @return \Illuminate\Database\Connection
 | |
| 	 */
 | |
| 	protected function prepare(Connection $connection)
 | |
| 	{
 | |
| 		$connection->setFetchMode($this->app['config']['database.fetch']);
 | |
| 
 | |
| 		if ($this->app->bound('events'))
 | |
| 		{
 | |
| 			$connection->setEventDispatcher($this->app['events']);
 | |
| 		}
 | |
| 
 | |
| 		// Here we'll set a reconnector callback. This reconnector can be any callable
 | |
| 		// so we will set a Closure to reconnect from this manager with the name of
 | |
| 		// the connection, which will allow us to reconnect from the connections.
 | |
| 		$connection->setReconnector(function($connection)
 | |
| 		{
 | |
| 			$this->reconnect($connection->getName());
 | |
| 		});
 | |
| 
 | |
| 		return $connection;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Prepare the read write mode for database connection instance.
 | |
| 	 *
 | |
| 	 * @param  \Illuminate\Database\Connection  $connection
 | |
| 	 * @param  string  $type
 | |
| 	 * @return \Illuminate\Database\Connection
 | |
| 	 */
 | |
| 	protected function setPdoForType(Connection $connection, $type = null)
 | |
| 	{
 | |
| 		if ($type == 'read')
 | |
| 		{
 | |
| 			$connection->setPdo($connection->getReadPdo());
 | |
| 		}
 | |
| 		elseif ($type == 'write')
 | |
| 		{
 | |
| 			$connection->setReadPdo($connection->getPdo());
 | |
| 		}
 | |
| 
 | |
| 		return $connection;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get the configuration for a connection.
 | |
| 	 *
 | |
| 	 * @param  string  $name
 | |
| 	 * @return array
 | |
| 	 *
 | |
| 	 * @throws \InvalidArgumentException
 | |
| 	 */
 | |
| 	protected function getConfig($name)
 | |
| 	{
 | |
| 		$name = $name ?: $this->getDefaultConnection();
 | |
| 
 | |
| 		// To get the database connection configuration, we will just pull each of the
 | |
| 		// connection configurations and get the configurations for the given name.
 | |
| 		// If the configuration doesn't exist, we'll throw an exception and bail.
 | |
| 		$connections = $this->app['config']['database.connections'];
 | |
| 
 | |
| 		if (is_null($config = array_get($connections, $name)))
 | |
| 		{
 | |
| 			throw new InvalidArgumentException("Database [$name] not configured.");
 | |
| 		}
 | |
| 
 | |
| 		return $config;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get the default connection name.
 | |
| 	 *
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function getDefaultConnection()
 | |
| 	{
 | |
| 		return $this->app['config']['database.default'];
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Set the default connection name.
 | |
| 	 *
 | |
| 	 * @param  string  $name
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	public function setDefaultConnection($name)
 | |
| 	{
 | |
| 		$this->app['config']['database.default'] = $name;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Register an extension connection resolver.
 | |
| 	 *
 | |
| 	 * @param  string    $name
 | |
| 	 * @param  callable  $resolver
 | |
| 	 * @return void
 | |
| 	 */
 | |
| 	public function extend($name, callable $resolver)
 | |
| 	{
 | |
| 		$this->extensions[$name] = $resolver;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Return all of the created connections.
 | |
| 	 *
 | |
| 	 * @return array
 | |
| 	 */
 | |
| 	public function getConnections()
 | |
| 	{
 | |
| 		return $this->connections;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Dynamically pass methods to the default connection.
 | |
| 	 *
 | |
| 	 * @param  string  $method
 | |
| 	 * @param  array   $parameters
 | |
| 	 * @return mixed
 | |
| 	 */
 | |
| 	public function __call($method, $parameters)
 | |
| 	{
 | |
| 		return call_user_func_array(array($this->connection(), $method), $parameters);
 | |
| 	}
 | |
| 
 | |
| }
 |