ConnectionParameters.php 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200
  1. <?php
  2. namespace Predis;
  3. use Predis\IConnectionParameters;
  4. use Predis\Options\IOption;
  5. class ConnectionParameters implements IConnectionParameters
  6. {
  7. private static $_defaultParameters;
  8. private static $_validators;
  9. private $_parameters;
  10. private $_userDefined;
  11. public function __construct($parameters = array())
  12. {
  13. self::ensureDefaults();
  14. if (!is_array($parameters)) {
  15. $parameters = $this->parseURI($parameters);
  16. }
  17. $this->_userDefined = array_keys($parameters);
  18. $this->_parameters = $this->filter($parameters) + self::$_defaultParameters;
  19. }
  20. private static function ensureDefaults()
  21. {
  22. if (!isset(self::$_defaultParameters)) {
  23. self::$_defaultParameters = array(
  24. 'scheme' => 'tcp',
  25. 'host' => '127.0.0.1',
  26. 'port' => 6379,
  27. 'database' => null,
  28. 'password' => null,
  29. 'connection_async' => false,
  30. 'connection_persistent' => false,
  31. 'connection_timeout' => 5.0,
  32. 'read_write_timeout' => null,
  33. 'alias' => null,
  34. 'weight' => null,
  35. 'path' => null,
  36. 'iterable_multibulk' => false,
  37. 'throw_errors' => true,
  38. );
  39. }
  40. if (!isset(self::$_validators)) {
  41. $bool = function($value) { return (bool) $value; };
  42. $float = function($value) { return (float) $value; };
  43. $int = function($value) { return (int) $value; };
  44. self::$_validators = array(
  45. 'port' => $int,
  46. 'connection_async' => $bool,
  47. 'connection_persistent' => $bool,
  48. 'connection_timeout' => $float,
  49. 'read_write_timeout' => $float,
  50. 'iterable_multibulk' => $bool,
  51. 'throw_errors' => $bool,
  52. );
  53. }
  54. }
  55. public static function define($parameter, $default, $callable = null)
  56. {
  57. self::ensureDefaults();
  58. self::$_defaultParameters[$parameter] = $default;
  59. if ($default instanceof IOption) {
  60. self::$_validators[$parameter] = $default;
  61. return;
  62. }
  63. if (!isset($callable)) {
  64. unset(self::$_validators[$parameter]);
  65. return;
  66. }
  67. if (!is_callable($callable)) {
  68. throw new \InvalidArgumentException(
  69. "The validator for $parameter must be a callable object"
  70. );
  71. }
  72. self::$_validators[$parameter] = $callable;
  73. }
  74. public static function undefine($parameter)
  75. {
  76. self::ensureDefaults();
  77. unset(self::$_defaultParameters[$parameter], self::$_validators[$parameter]);
  78. }
  79. private function parseURI($uri)
  80. {
  81. if (stripos($uri, 'unix') === 0) {
  82. // Hack to support URIs for UNIX sockets with minimal effort.
  83. $uri = str_ireplace('unix:///', 'unix://localhost/', $uri);
  84. }
  85. if (($parsed = @parse_url($uri)) === false || !isset($parsed['host'])) {
  86. throw new ClientException("Invalid URI: $uri");
  87. }
  88. if (isset($parsed['query'])) {
  89. foreach (explode('&', $parsed['query']) as $kv) {
  90. @list($k, $v) = explode('=', $kv);
  91. $parsed[$k] = $v;
  92. }
  93. unset($parsed['query']);
  94. }
  95. return $parsed;
  96. }
  97. private function filter(Array $parameters)
  98. {
  99. if (count($parameters) > 0) {
  100. $validators = array_intersect_key(self::$_validators, $parameters);
  101. foreach ($validators as $parameter => $validator) {
  102. $parameters[$parameter] = $validator($parameters[$parameter]);
  103. }
  104. }
  105. return $parameters;
  106. }
  107. public function __get($parameter)
  108. {
  109. $value = $this->_parameters[$parameter];
  110. if ($value instanceof IOption) {
  111. $this->_parameters[$parameter] = ($value = $value->getDefault());
  112. }
  113. return $value;
  114. }
  115. public function __isset($parameter)
  116. {
  117. return isset($this->_parameters[$parameter]);
  118. }
  119. public function isSetByUser($parameter)
  120. {
  121. return in_array($parameter, $this->_userDefined);
  122. }
  123. protected function getBaseURI()
  124. {
  125. if ($this->scheme === 'unix') {
  126. return "{$this->scheme}://{$this->path}";
  127. }
  128. return "{$this->scheme}://{$this->host}:{$this->port}";
  129. }
  130. protected function getDisallowedURIParts()
  131. {
  132. return array('scheme', 'host', 'port', 'password', 'path');
  133. }
  134. public function toArray()
  135. {
  136. return $this->_parameters;
  137. }
  138. public function __toString()
  139. {
  140. $query = array();
  141. $parameters = $this->toArray();
  142. $reject = $this->getDisallowedURIParts();
  143. foreach ($this->_userDefined as $param) {
  144. if (in_array($param, $reject) || !isset($parameters[$param])) {
  145. continue;
  146. }
  147. $value = $parameters[$param];
  148. $query[] = "$param=" . ($value === false ? '0' : $value);
  149. }
  150. if (count($query) === 0) {
  151. return $this->getBaseURI();
  152. }
  153. return $this->getBaseURI() . '/?' . implode('&', $query);
  154. }
  155. public function __sleep()
  156. {
  157. return array('_parameters', '_userDefined');
  158. }
  159. public function __wakeup()
  160. {
  161. self::ensureDefaults();
  162. }
  163. }