OpenSearchClient.php 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322
  1. <?php
  2. /*
  3. * Licensed to the Apache Software Foundation (ASF) under one
  4. * or more contributor license agreements. See the NOTICE file
  5. * distributed with this work for additional information
  6. * regarding copyright ownership. The ASF licenses this file
  7. * to you under the Apache License, Version 2.0 (the
  8. * "License"); you may not use this file except in compliance
  9. * with the License. You may obtain a copy of the License at
  10. *
  11. * http://www.apache.org/licenses/LICENSE-2.0
  12. *
  13. * Unless required by applicable law or agreed to in writing,
  14. * software distributed under the License is distributed on an
  15. * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
  16. * KIND, either express or implied. See the License for the
  17. * specific language governing permissions and limitations
  18. * under the License.
  19. */
  20. namespace OpenSearch\Client;
  21. use OpenSearch\Generated\OpenSearch\OpenSearch;
  22. use OpenSearch\Generated\OpenSearch\Constant;
  23. use OpenSearch\Generated\Common\OpenSearchResult;
  24. use OpenSearch\Generated\Common\TraceInfo;
  25. class OpenSearchClient extends OpenSearch {
  26. const METHOD_GET = 'GET';
  27. const METHOD_POST = 'POST';
  28. const METHOD_PUT = 'PUT';
  29. const METHOD_DELETE = 'DELETE';
  30. const METHOD_PATCH = 'PATCH';
  31. const API_VERSION = '3';
  32. const API_TYPE = 'openapi';
  33. const SDK_VERSION = '3.3.0';
  34. const SDK_TYPE = 'opensearch_sdk';
  35. private $debug = false;
  36. public $timeout = 10;
  37. public $connectTimeout = 1;
  38. /**
  39. * 构造方法。
  40. *
  41. * @param string $accessKey 指定您的accessKeyId,在 https://ak-console.aliyun.com/#/accesskey 中可以创建。
  42. * @param string $secret 指定您的secret。
  43. * @param string $host 指定您要访问的区域的endPoint,在控制台应用详情页中有指定。
  44. * @param array @options 指定一些可选参数,debug:true/false,是否开启debug模式(默认不开启),gzip:true/false 是否开启gzip压缩(默认不开启),timeout:超时时间,seconds(默认10秒),connectTimeout: 连接超时时间,seconds(默认1秒)
  45. * @return void
  46. */
  47. public function __construct($accessKey, $secret, $host, $options = array()) {
  48. $args = array(
  49. 'accessKey' => trim($accessKey),
  50. 'secret' => trim($secret),
  51. 'host' => trim($host),
  52. 'options' => $options
  53. );
  54. if (isset($options['gzip'])) {
  55. $args['gzip'] = $options['gzip'];
  56. }
  57. if (isset($options['timeout'])) {
  58. $args['timeout'] = $options['timeout'];
  59. }
  60. if (isset($options['connectTimeout'])) {
  61. $args['connectTimeout'] = $options['connectTimeout'];
  62. }
  63. if (isset($options['securityToken'])) {
  64. $args['securityToken'] = $options['securityToken'];
  65. }
  66. if (isset($options['debug'])) {
  67. $this->debug = (boolean) $options['debug'];
  68. }
  69. parent::__construct($args);
  70. }
  71. /**
  72. * 发送一个GET请求。
  73. *
  74. * @param string $uri 发起GET请求的uri。
  75. * @param array $params 发起GET请求的参数,以param_key => param_value的方式体现。
  76. * @return \OpenSearch\Generated\Common\OpenSearchResult
  77. */
  78. public function get($uri, $params = array()) {
  79. return $this->call($uri, $params, '', self::METHOD_GET);
  80. }
  81. /**
  82. * 发送一个PUT请求。
  83. *
  84. * @param string $uri 发起PUT请求的uri。
  85. * @param string $body 发起PUT请求的body体,为一个原始的json格式的string。
  86. * @return \OpenSearch\Generated\Common\OpenSearchResult
  87. */
  88. public function put($uri, $body = '') {
  89. return $this->call($uri, array(), $body, self::METHOD_PUT);
  90. }
  91. /**
  92. * 发送一个POST请求。
  93. *
  94. * @param string $uri 发起POST请求的uri。
  95. * @param string $body 发起POST请求的body体,为一个原始的json格式的string。
  96. * @return \OpenSearch\Generated\Common\OpenSearchResult
  97. */
  98. public function post($uri, $body = '') {
  99. return $this->call($uri, array(), $body, self::METHOD_POST);
  100. }
  101. /**
  102. * 发送一个DELETE请求。
  103. *
  104. * @param string $uri 发起DELETE请求的uri。
  105. * @param string $body 发起DELETE请求的body体,为一个原始的json格式的string。
  106. * @return \OpenSearch\Generated\Common\OpenSearchResult
  107. */
  108. public function delete($uri, $body = '') {
  109. return $this->call($uri, array(), $body, self::METHOD_DELETE);
  110. }
  111. /**
  112. * 发送一个PATCH请求。
  113. *
  114. * @param string $uri 发起PATCH请求的uri。
  115. * @param string $body 发起PATCH请求的body体,为一个原始的json格式的string。
  116. * @return \OpenSearch\Generated\Common\OpenSearchResult
  117. */
  118. public function patch($uri, $body = '') {
  119. return $this->call($uri, array(), $body, self::METHOD_PATCH);
  120. }
  121. /**
  122. * 发送一个请求。
  123. *
  124. * @param string $uri 发起请求的uri。
  125. * @param array $params 指定的url中的query string 列表。
  126. * @param string $body 发起请求的body体,为一个原始的json格式的string。
  127. * @param string $method 发起请求的方法,有GET/POST/DELETE/PUT/PATCH等
  128. * @return \OpenSearch\Generated\Common\OpenSearchResult
  129. */
  130. public function call($uri, array $params, $body, $method) {
  131. $path = "/v" . self::API_VERSION . "/" . self::API_TYPE . "{$uri}";
  132. $url = $this->host . $path;
  133. $items = array();
  134. $items['method'] = $method;
  135. $items['request_path'] = $path;
  136. $items['content_type'] = "application/json";
  137. $items['accept_language'] = "zh-cn";
  138. $items['date'] = gmdate('Y-m-d\TH:i:s\Z');
  139. $items['opensearch_headers'] = array();
  140. $items['content_md5'] = "";
  141. $items['opensearch_headers']['X-Opensearch-Nonce'] = $this->_nonce();
  142. if ($this->securityToken) {
  143. $items['opensearch_headers']['X-Opensearch-Security-Token']
  144. = $this->securityToken;
  145. }
  146. if ($method != self::METHOD_GET) {
  147. if (!empty($body)) {
  148. $items['content_md5'] = md5($body);
  149. $items['body_json'] = $body;
  150. }
  151. }
  152. $items['query_params'] = $params;
  153. $signature = $this->_signature($this->secret, $items);
  154. $items['authorization'] = "OPENSEARCH {$this->accessKey}:{$signature}";
  155. return $this->_curl($url, $items);
  156. }
  157. private function _nonce() {
  158. return intval(microtime(true) * 1000) . mt_rand(10000, 99999);
  159. }
  160. private function _signature($secret, $items) {
  161. $params = isset($items['query_params']) ? $items['query_params'] : "";
  162. $signature = '';
  163. $string = '';
  164. $string .= strtoupper($items['method']) . "\n";
  165. $string .= $items['content_md5'] . "\n";
  166. $string .= $items['content_type'] . "\n";
  167. $string .= $items['date'] . "\n";
  168. $headers = self::_filter($items['opensearch_headers']);
  169. foreach($headers as $key => $value){
  170. $string .= strtolower($key) . ":" . $value."\n";
  171. }
  172. $resource = str_replace('%2F', '/', rawurlencode($items['request_path']));
  173. $sortParams = self::_filter($params);
  174. $queryString = $this->_buildQuery($sortParams);
  175. $canonicalizedResource = $resource;
  176. if(!empty($queryString)){
  177. $canonicalizedResource .= '?'.$queryString;
  178. }
  179. $string .= $canonicalizedResource;
  180. $signature = base64_encode(hash_hmac('sha1', $string, $secret, true));
  181. return $signature;
  182. }
  183. private function _buildQuery($params) {
  184. $query = '';
  185. if (version_compare(PHP_VERSION, '5.4.0', '>=')) {
  186. $query = !empty($params) ? http_build_query($params, null, '&', PHP_QUERY_RFC3986) : '';
  187. } else {
  188. $arg = '';
  189. foreach ($params as $key => $val) {
  190. $arg .= rawurlencode($key) . "=" . rawurlencode($val) . "&";
  191. }
  192. $query = substr($arg, 0, count($arg) - 2);
  193. }
  194. return $query;
  195. }
  196. private function _filter($parameters = array()){
  197. $params = array();
  198. if(!empty($parameters)){
  199. foreach ($parameters as $key => $val) {
  200. if ($key == "Signature" ||$val === "" || $val === NULL){
  201. continue;
  202. } else {
  203. $params[$key] = $parameters[$key];
  204. }
  205. }
  206. uksort($params,'strnatcasecmp');
  207. reset($params);
  208. }
  209. return $params;
  210. }
  211. private function _getHeaders($items) {
  212. $headers = array();
  213. $headers[] = 'Content-Type: '.$items['content_type'];
  214. $headers[] = 'Date: '.$items['date'];
  215. $headers[] = 'Accept-Language: '.$items['accept_language'];
  216. $headers[] = 'Content-Md5: '.$items['content_md5'];
  217. $headers[] = 'Authorization: '.$items['authorization'];
  218. if (is_array($items['opensearch_headers'])) {
  219. foreach($items['opensearch_headers'] as $key => $value){
  220. $headers[] = $key . ": " . $value;
  221. }
  222. }
  223. return $headers;
  224. }
  225. private function _curl($url, $items) {
  226. $method = strtoupper($items['method']);
  227. $options = array(
  228. CURLOPT_HTTP_VERSION => 'CURL_HTTP_VERSION_1_1',
  229. CURLOPT_CONNECTTIMEOUT => $this->connectTimeout,
  230. CURLOPT_TIMEOUT => $this->timeout,
  231. CURLOPT_CUSTOMREQUEST => $method,
  232. CURLOPT_HEADER => false,
  233. CURLOPT_RETURNTRANSFER => true,
  234. CURLOPT_USERAGENT => "opensearch/php sdk " . self::SDK_VERSION . "/" . PHP_VERSION,
  235. CURLOPT_HTTPHEADER => $this->_getHeaders($items),
  236. );
  237. if ($method == self::METHOD_GET) {
  238. $query = $this->_buildQuery($items['query_params']);
  239. $url .= preg_match('/\?/i', $url) ? '&' . $query : '?' . $query;
  240. } else{
  241. if(!empty($items['body_json'])){
  242. $options[CURLOPT_POSTFIELDS] = $items['body_json'];
  243. }
  244. }
  245. if ($this->gzip) {
  246. $options[CURLOPT_ENCODING] = 'gzip';
  247. }
  248. if ($this->debug) {
  249. $out = fopen('php://temp','rw');
  250. $options[CURLOPT_VERBOSE] = true;
  251. $options[CURLOPT_STDERR] = $out;
  252. }
  253. $session = curl_init($url);
  254. curl_setopt_array($session, $options);
  255. $response = curl_exec($session);
  256. curl_close($session);
  257. $openSearchResult = new OpenSearchResult();
  258. $openSearchResult->result = $response;
  259. if ($this->debug) {
  260. $openSearchResult->traceInfo = $this->getDebugInfo($out, $items);
  261. }
  262. return $openSearchResult;
  263. }
  264. private function getDebugInfo($handler, $items) {
  265. rewind($handler);
  266. $trace = new TraceInfo();
  267. $header = stream_get_contents($handler);
  268. fclose($handler);
  269. $trace->tracer = "\n" . $header;
  270. return $trace;
  271. }
  272. }