UrlSigner.php 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119
  1. <?php
  2. namespace Aws\CloudFront;
  3. use GuzzleHttp\Psr7;
  4. use GuzzleHttp\Psr7\Uri;
  5. use Psr\Http\Message\UriInterface;
  6. /**
  7. * Creates signed URLs for Amazon CloudFront resources.
  8. */
  9. class UrlSigner
  10. {
  11. private $signer;
  12. /**
  13. * @param $keyPairId string ID of the key pair
  14. * @param $privateKey string Path to the private key used for signing
  15. *
  16. * @throws \RuntimeException if the openssl extension is missing
  17. * @throws \InvalidArgumentException if the private key cannot be found.
  18. */
  19. public function __construct($keyPairId, $privateKey)
  20. {
  21. $this->signer = new Signer($keyPairId, $privateKey);
  22. }
  23. /**
  24. * Create a signed Amazon CloudFront URL.
  25. *
  26. * Keep in mind that URLs meant for use in media/flash players may have
  27. * different requirements for URL formats (e.g. some require that the
  28. * extension be removed, some require the file name to be prefixed
  29. * - mp4:<path>, some require you to add "/cfx/st" into your URL).
  30. *
  31. * @param string $url URL to sign (can include query
  32. * string string and wildcards)
  33. * @param string|integer|null $expires UTC Unix timestamp used when signing
  34. * with a canned policy. Not required
  35. * when passing a custom $policy.
  36. * @param string $policy JSON policy. Use this option when
  37. * creating a signed URL for a custom
  38. * policy.
  39. *
  40. * @return string The file URL with authentication parameters
  41. * @throws \InvalidArgumentException if the URL provided is invalid
  42. * @link http://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/WorkingWithStreamingDistributions.html
  43. */
  44. public function getSignedUrl($url, $expires = null, $policy = null)
  45. {
  46. // Determine the scheme of the url
  47. $urlSections = explode('://', $url);
  48. if (count($urlSections) < 2) {
  49. throw new \InvalidArgumentException("Invalid URL: {$url}");
  50. }
  51. // Get the real scheme by removing wildcards from the scheme
  52. $scheme = str_replace('*', '', $urlSections[0]);
  53. $uri = new Uri($scheme . '://' . $urlSections[1]);
  54. $query = Psr7\Query::parse($uri->getQuery(), PHP_QUERY_RFC3986);
  55. $signature = $this->signer->getSignature(
  56. $this->createResource($scheme, (string) $uri),
  57. $expires,
  58. $policy
  59. );
  60. $uri = $uri->withQuery(
  61. http_build_query($query + $signature, '', '&', PHP_QUERY_RFC3986)
  62. );
  63. return $scheme === 'rtmp'
  64. ? $this->createRtmpUrl($uri)
  65. : (string) $uri;
  66. }
  67. private function createRtmpUrl(UriInterface $uri)
  68. {
  69. // Use a relative URL when creating Flash player URLs
  70. $result = ltrim($uri->getPath(), '/');
  71. if ($query = $uri->getQuery()) {
  72. $result .= '?' . $query;
  73. }
  74. return $result;
  75. }
  76. /**
  77. * @param $scheme
  78. * @param $url
  79. *
  80. * @return string
  81. */
  82. private function createResource($scheme, $url)
  83. {
  84. switch ($scheme) {
  85. case 'http':
  86. case 'http*':
  87. case 'https':
  88. return $url;
  89. case 'rtmp':
  90. $parts = parse_url($url);
  91. $pathParts = pathinfo($parts['path']);
  92. $resource = ltrim(
  93. $pathParts['dirname'] . '/' . $pathParts['basename'],
  94. '/'
  95. );
  96. // Add a query string if present.
  97. if (isset($parts['query'])) {
  98. $resource .= "?{$parts['query']}";
  99. }
  100. return $resource;
  101. }
  102. throw new \InvalidArgumentException("Invalid URI scheme: {$scheme}. "
  103. . "Scheme must be one of: http, https, or rtmp");
  104. }
  105. }