From 893196f08991fb4cf1979b6e18c4202ae25871be Mon Sep 17 00:00:00 2001 From: Anton Kropp Date: Mon, 12 Nov 2018 15:07:02 -0800 Subject: [PATCH] Adding client telemetry headers --- init.php | 1 + lib/ApiRequestor.php | 53 +++++++++++++ lib/RequestTelemetry.php | 21 +++++ lib/Stripe.php | 23 ++++++ lib/Util/Util.php | 10 +++ tests/Stripe/StripeTelemetryTest.php | 111 +++++++++++++++++++++++++++ tests/TestCase.php | 1 + 7 files changed, 220 insertions(+) create mode 100644 lib/RequestTelemetry.php create mode 100644 tests/Stripe/StripeTelemetryTest.php diff --git a/init.php b/init.php index 85fc5c548..10ea220d6 100644 --- a/init.php +++ b/init.php @@ -49,6 +49,7 @@ // Plumbing require(dirname(__FILE__) . '/lib/ApiResponse.php'); +require(dirname(__FILE__) . '/lib/RequestTelemetry.php'); require(dirname(__FILE__) . '/lib/StripeObject.php'); require(dirname(__FILE__) . '/lib/ApiRequestor.php'); require(dirname(__FILE__) . '/lib/ApiResource.php'); diff --git a/lib/ApiRequestor.php b/lib/ApiRequestor.php index 4a147d01a..7cf851877 100644 --- a/lib/ApiRequestor.php +++ b/lib/ApiRequestor.php @@ -24,6 +24,11 @@ class ApiRequestor */ private static $_httpClient; + /** + * @var RequestTelemetry + */ + private static $requestTelemetry; + /** * ApiRequestor constructor. * @@ -39,6 +44,30 @@ public function __construct($apiKey = null, $apiBase = null) $this->_apiBase = $apiBase; } + /** + * Creates a telemetry json blob for use in 'X-Stripe-Client-Telemetry' headers + * @static + * + * @param RequestTelemetry $requestTelemetry + * @return string + */ + private static function _telemetryJson($requestTelemetry) + { + $payload = array( + 'last_request_metrics' => array( + 'request_id' => $requestTelemetry->requestId, + 'request_duration_ms' => $requestTelemetry->requestDuration, + )); + + $result = json_encode($payload); + if ($result != false) { + return $result; + } else { + Stripe::getLogger()->error("Serializing telemetry payload failed!"); + return "{}"; + } + } + /** * @static * @@ -332,6 +361,10 @@ private function _requestRaw($method, $url, $params, $headers) $defaultHeaders['Stripe-Account'] = Stripe::$accountId; } + if (Stripe::$enableTelemetry && self::$requestTelemetry != null) { + $defaultHeaders["X-Stripe-Client-Telemetry"] = self::_telemetryJson(self::$requestTelemetry); + } + $hasFile = false; $hasCurlFile = class_exists('\CURLFile', false); foreach ($params as $k => $v) { @@ -356,6 +389,8 @@ private function _requestRaw($method, $url, $params, $headers) $rawHeaders[] = $header . ': ' . $value; } + $requestStartMs = Util\Util::currentTimeMillis(); + list($rbody, $rcode, $rheaders) = $this->httpClient()->request( $method, $absUrl, @@ -363,6 +398,14 @@ private function _requestRaw($method, $url, $params, $headers) $params, $hasFile ); + + if (array_key_exists('request-id', $rheaders)) { + self::$requestTelemetry = new RequestTelemetry( + $rheaders['request-id'], + Util\Util::currentTimeMillis() - $requestStartMs + ); + } + return [$rbody, $rcode, $rheaders, $myApiKey]; } @@ -442,6 +485,16 @@ public static function setHttpClient($client) self::$_httpClient = $client; } + /** + * @static + * + * Resets any stateful telemetry data + */ + public static function resetTelemetry() + { + self::$requestTelemetry = null; + } + /** * @return HttpClient\ClientInterface */ diff --git a/lib/RequestTelemetry.php b/lib/RequestTelemetry.php new file mode 100644 index 000000000..5ddbb0561 --- /dev/null +++ b/lib/RequestTelemetry.php @@ -0,0 +1,21 @@ +requestId = $requestId; + $this->requestDuration = $requestDuration; + } +} diff --git a/lib/Stripe.php b/lib/Stripe.php index 2c2c38246..e1672a258 100644 --- a/lib/Stripe.php +++ b/lib/Stripe.php @@ -46,6 +46,9 @@ class Stripe // @var int Maximum number of request retries public static $maxNetworkRetries = 0; + // @var boolean Whether client telemetry is enabled. Defaults to false. + public static $enableTelemetry = false; + // @var float Maximum delay between retries, in seconds private static $maxNetworkRetryDelay = 2.0; @@ -239,4 +242,24 @@ public static function getInitialNetworkRetryDelay() { return self::$initialNetworkRetryDelay; } + + /** + * @return bool Whether client telemetry is enabled + */ + public static function getEnableTelemetry() + { + return self::$enableTelemetry; + } + + /** + * @param bool $enableTelemetry Enables client telemetry. + * + * Client telemetry enables timing and request metrics to be sent back to Stripe as an HTTP Header + * with the current request. This enables Stripe to do latency and metrics analysis without adding extra + * overhead (such as extra network calls) on the client. + */ + public static function setEnableTelemetry($enableTelemetry) + { + self::$enableTelemetry = $enableTelemetry; + } } diff --git a/lib/Util/Util.php b/lib/Util/Util.php index 7f257b051..b9804c20d 100644 --- a/lib/Util/Util.php +++ b/lib/Util/Util.php @@ -333,4 +333,14 @@ public static function normalizeId($id) } return [$id, $params]; } + + /** + * Returns UNIX timestamp in milliseconds + * + * @return float current time in millis + */ + public static function currentTimeMillis() + { + return round(microtime(true) * 1000); + } } diff --git a/tests/Stripe/StripeTelemetryTest.php b/tests/Stripe/StripeTelemetryTest.php new file mode 100644 index 000000000..966677138 --- /dev/null +++ b/tests/Stripe/StripeTelemetryTest.php @@ -0,0 +1,111 @@ +getMockBuilder("HttpClient\ClientInterface") + ->setMethods(array('request')) + ->getMock(); + + $stub->expects($this->any()) + ->method("request") + ->with( + $this->anything(), + $this->anything(), + $this->callback(function ($headers) use (&$requestheaders) { + foreach ($headers as $index => $header) { + // capture the requested headers and format back to into an assoc array + $components = explode(": ", $header, 2); + $requestheaders[$components[0]] = $components[1]; + } + + return true; + }), + $this->anything(), + $this->anything() + )->willReturn(array(self::FAKE_VALID_RESPONSE, 200, ["request-id" => "123"])); + + ApiRequestor::setHttpClient($stub); + + // make one request to capture its result + Charge::all(); + $this->assertArrayNotHasKey('X-Stripe-Client-Telemetry', $requestheaders); + + // make another request and verify telemetry isn't sent + Charge::all(); + $this->assertArrayNotHasKey('X-Stripe-Client-Telemetry', $requestheaders); + + ApiRequestor::setHttpClient(null); + } + + public function testTelemetrySetIfEnabled() + { + Stripe::setEnableTelemetry(true); + + $requestheaders = null; + + $stub = $this + ->getMockBuilder("HttpClient\ClientInterface") + ->setMethods(array('request')) + ->getMock(); + + $stub->expects($this->any()) + ->method("request") + ->with( + $this->anything(), + $this->anything(), + $this->callback(function ($headers) use (&$requestheaders) { + // capture the requested headers and format back to into an assoc array + foreach ($headers as $index => $header) { + $components = explode(": ", $header, 2); + $requestheaders[$components[0]] = $components[1]; + } + + return true; + }), + $this->anything(), + $this->anything() + )->willReturn(array(self::FAKE_VALID_RESPONSE, 200, ["request-id" => "123"])); + + ApiRequestor::setHttpClient($stub); + + // make one request to capture its result + Charge::all(); + $this->assertArrayNotHasKey('X-Stripe-Client-Telemetry', $requestheaders); + + // make another request to send the previous + Charge::all(); + $this->assertArrayHasKey('X-Stripe-Client-Telemetry', $requestheaders); + + $data = json_decode($requestheaders['X-Stripe-Client-Telemetry'], true); + $this->assertEquals('123', $data['last_request_metrics']['request_id']); + $this->assertNotNull($data['last_request_metrics']['request_duration_ms']); + + ApiRequestor::setHttpClient(null); + } +} diff --git a/tests/TestCase.php b/tests/TestCase.php index 93be3d128..e52e9c1f7 100644 --- a/tests/TestCase.php +++ b/tests/TestCase.php @@ -52,6 +52,7 @@ protected function tearDown() { // Restore original values Stripe::$apiBase = $this->origApiBase; + Stripe::setEnableTelemetry(false); Stripe::setApiKey($this->origApiKey); Stripe::setClientId($this->origClientId); Stripe::setApiVersion($this->origApiVersion);