/wp-content/plugins/w3-total-cache/lib/Azure/MicrosoftAzureStorage/Common/Internal/RetryMiddlewareFactory.php

https://bitbucket.org/vaderdeveloper/autovidas · PHP · 234 lines · 110 code · 15 blank · 109 comment · 17 complexity · 54f9975b17569dc866dbddf56f84ac82 MD5 · raw file

  1. <?php
  2. /**
  3. * LICENSE: The MIT License (the "License")
  4. * you may not use this file except in compliance with the License.
  5. * You may obtain a copy of the License at
  6. * https://github.com/azure/azure-storage-php/LICENSE
  7. *
  8. * Unless required by applicable law or agreed to in writing, software
  9. * distributed under the License is distributed on an "AS IS" BASIS,
  10. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  11. * See the License for the specific language governing permissions and
  12. * limitations under the License.
  13. *
  14. * PHP version 5
  15. *
  16. * @category Microsoft
  17. * @package MicrosoftAzure\Storage\Common\Internal
  18. * @author Azure Storage PHP SDK <dmsh@microsoft.com>
  19. * @copyright 2016 Microsoft Corporation
  20. * @license https://github.com/azure/azure-storage-php/LICENSE
  21. * @link https://github.com/azure/azure-storage-php
  22. */
  23. namespace MicrosoftAzure\Storage\Common\Internal;
  24. use MicrosoftAzure\Storage\Common\Internal\Resources;
  25. use MicrosoftAzure\Storage\Common\Internal\Validate;
  26. use GuzzleHttp\Middleware;
  27. /**
  28. * This class provides static functions that creates retry handlers for Guzzle
  29. * HTTP clients to handle retry policy.
  30. *
  31. * @category Microsoft
  32. * @package MicrosoftAzure\Storage\Common\Internal
  33. * @author Azure Storage PHP SDK <dmsh@microsoft.com>
  34. * @copyright 2016 Microsoft Corporation
  35. * @license https://github.com/azure/azure-storage-php/LICENSE
  36. * @version Release: 0.11.0
  37. * @link https://github.com/azure/azure-storage-php
  38. */
  39. class RetryMiddlewareFactory
  40. {
  41. const LINEAR_INTERVAL_ACCUMULATION = 'Linear';
  42. const EXPONENTIAL_INTERVAL_ACCUMULATION = 'Exponential';
  43. const GENERAL_RETRY_TYPE = 'General';
  44. const APPEND_BLOB_RETRY_TYPE = 'Append Blob Retry';
  45. /**
  46. * Create the retry handler for the Guzzle client, according to the given
  47. * attributes.
  48. *
  49. * @param string $type The type that controls the logic of
  50. * the decider of the retry handler.
  51. * @param int $numberOfRetries The maximum number of retries.
  52. * @param int $interval The minimum interval between each retry
  53. * @param string $accumulationMethod If the interval increases linearly or
  54. * exponentially.
  55. *
  56. * @return RetryMiddleware A RetryMiddleware object that contains
  57. * the logic of how the request should be
  58. * handled after a response.
  59. */
  60. public static function create(
  61. $type = self::GENERAL_RETRY_TYPE,
  62. $numberOfRetries = Resources::DEFAULT_NUMBER_OF_RETRIES,
  63. $interval = Resources::DEAFULT_RETRY_INTERVAL,
  64. $accumulationMethod = self::LINEAR_INTERVAL_ACCUMULATION
  65. ) {
  66. //Validate the input parameters
  67. //type
  68. Validate::isTrue(
  69. $type == self::GENERAL_RETRY_TYPE ||
  70. $type == self::APPEND_BLOB_RETRY_TYPE,
  71. sprintf(
  72. Resources::INVALID_PARAM_GENERAL,
  73. 'type'
  74. )
  75. );
  76. //numberOfRetries
  77. Validate::isTrue(
  78. $numberOfRetries > 0,
  79. sprintf(
  80. Resources::INVALID_NEGATIVE_PARAM,
  81. 'numberOfRetries'
  82. )
  83. );
  84. //interval
  85. Validate::isTrue(
  86. $interval > 0,
  87. sprintf(
  88. Resources::INVALID_NEGATIVE_PARAM,
  89. 'interval'
  90. )
  91. );
  92. //accumulationMethod
  93. Validate::isTrue(
  94. $accumulationMethod == self::LINEAR_INTERVAL_ACCUMULATION ||
  95. $accumulationMethod == self::EXPONENTIAL_INTERVAL_ACCUMULATION,
  96. sprintf(
  97. Resources::INVALID_PARAM_GENERAL,
  98. 'accumulationMethod'
  99. )
  100. );
  101. //Get the interval calculator according to the type of the
  102. //accumulation method.
  103. $intervalCalculator =
  104. $accumulationMethod == self::LINEAR_INTERVAL_ACCUMULATION ?
  105. self::createLinearDelayCalculator($interval) :
  106. self::createExponentialDelayCalculator($interval);
  107. //Get the retry decider according to the type of the retry and
  108. //the number of retries.
  109. $retryDecider = self::createRetryDecider($type, $numberOfRetries);
  110. //construct the retry middle ware.
  111. return Middleware::retry($retryDecider, $intervalCalculator);
  112. }
  113. /**
  114. * Create the retry decider for the retry handler. It will return a callable
  115. * that accepts the number of retries, the request, the response and the
  116. * exception, and return the decision for a retry.
  117. *
  118. * @param string $type The type of the retry handler.
  119. *
  120. * @return callable The callable that will return if the request should
  121. * be retried.
  122. */
  123. protected static function createRetryDecider($type, $maxRetries)
  124. {
  125. return function (
  126. $retries,
  127. $request,
  128. $response = null,
  129. $exception = null
  130. ) use (
  131. $type,
  132. $maxRetries
  133. ) {
  134. //Exceeds the retry limit. No retry.
  135. if ($retries >= $maxRetries) {
  136. return false;
  137. }
  138. //Not retriable error, won't retry.
  139. if (!$response) {
  140. return false;
  141. } else {
  142. if ($type == self::GENERAL_RETRY_TYPE) {
  143. return self::generalRetryDecider($response->getStatusCode());
  144. } else {
  145. return self::appendBlobRetryDecider($response->getStatusCode());
  146. }
  147. }
  148. return true;
  149. };
  150. }
  151. /**
  152. * Decide if the given status code indicate the request should be retried.
  153. *
  154. * @param int $statusCode status code of the previous request.
  155. *
  156. * @return bool true if the request should be retried.
  157. */
  158. protected static function generalRetryDecider($statusCode)
  159. {
  160. $retry = false;
  161. if ($statusCode == 408) {
  162. $retry = true;
  163. } elseif ($statusCode >= 500) {
  164. if ($statusCode != 501 && $statusCode != 505) {
  165. $retry = true;
  166. }
  167. }
  168. return $retry;
  169. }
  170. /**
  171. * Decide if the given status code indicate the request should be retried.
  172. * This is for append blob.
  173. *
  174. * @param int $statusCode status code of the previous request.
  175. *
  176. * @return bool true if the request should be retried.
  177. */
  178. protected static function appendBlobRetryDecider($statusCode)
  179. {
  180. //The retry logic is different for append blob.
  181. //First it will need to record the former status code if it is
  182. //server error. Then if the following request is 412 then it
  183. //needs to be retried. Currently this is not implemented so will
  184. //only adapt to the general retry decider.
  185. //TODO: add logic for append blob's retry when implemented.
  186. $retry = self::generalRetryDecider($statusCode);
  187. return $retry;
  188. }
  189. /**
  190. * Create the delay calculator that increases the interval linearly
  191. * according to the number of retries.
  192. *
  193. * @param int $interval the minimum interval of the retry.
  194. *
  195. * @return callable a calculator that will return the interval
  196. * according to the number of retries.
  197. */
  198. protected static function createLinearDelayCalculator($interval)
  199. {
  200. return function ($retries) use ($interval) {
  201. return $retries * $interval;
  202. };
  203. }
  204. /**
  205. * Create the delay calculator that increases the interval exponentially
  206. * according to the number of retries.
  207. *
  208. * @param int $interval the minimum interval of the retry.
  209. *
  210. * @return callable a calculator that will return the interval
  211. * according to the number of retries.
  212. */
  213. protected static function createExponentialDelayCalculator($interval)
  214. {
  215. return function ($retries) use ($interval) {
  216. return $interval * ((int)\pow(2, $retries));
  217. };
  218. }
  219. }