ObjectUploader.php 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150
  1. <?php
  2. namespace Aws\S3;
  3. use GuzzleHttp\Promise\PromiseInterface;
  4. use GuzzleHttp\Promise\PromisorInterface;
  5. use GuzzleHttp\Psr7;
  6. use Psr\Http\Message\StreamInterface;
  7. /**
  8. * Uploads an object to S3, using a PutObject command or a multipart upload as
  9. * appropriate.
  10. */
  11. class ObjectUploader implements PromisorInterface
  12. {
  13. const DEFAULT_MULTIPART_THRESHOLD = 16777216;
  14. private $client;
  15. private $bucket;
  16. private $key;
  17. private $body;
  18. private $acl;
  19. private $options;
  20. private static $defaults = [
  21. 'before_upload' => null,
  22. 'concurrency' => 3,
  23. 'mup_threshold' => self::DEFAULT_MULTIPART_THRESHOLD,
  24. 'params' => [],
  25. 'part_size' => null,
  26. ];
  27. private $addContentMD5;
  28. /**
  29. * @param S3ClientInterface $client The S3 Client used to execute
  30. * the upload command(s).
  31. * @param string $bucket Bucket to upload the object, or
  32. * an S3 access point ARN.
  33. * @param string $key Key of the object.
  34. * @param mixed $body Object data to upload. Can be a
  35. * StreamInterface, PHP stream
  36. * resource, or a string of data to
  37. * upload.
  38. * @param string $acl ACL to apply to the copy
  39. * (default: private).
  40. * @param array $options Options used to configure the
  41. * copy process. Options passed in
  42. * through 'params' are added to
  43. * the sub command(s).
  44. */
  45. public function __construct(
  46. S3ClientInterface $client,
  47. $bucket,
  48. $key,
  49. $body,
  50. $acl = 'private',
  51. array $options = []
  52. ) {
  53. $this->client = $client;
  54. $this->bucket = $bucket;
  55. $this->key = $key;
  56. $this->body = Psr7\Utils::streamFor($body);
  57. $this->acl = $acl;
  58. $this->options = $options + self::$defaults;
  59. // Handle "add_content_md5" option.
  60. $this->addContentMD5 = isset($options['add_content_md5'])
  61. && $options['add_content_md5'] === true;
  62. }
  63. /**
  64. * @return PromiseInterface
  65. */
  66. public function promise(): PromiseInterface
  67. {
  68. /** @var int $mup_threshold */
  69. $mup_threshold = $this->options['mup_threshold'];
  70. if ($this->requiresMultipart($this->body, $mup_threshold)) {
  71. // Perform a multipart upload.
  72. return (new MultipartUploader($this->client, $this->body, [
  73. 'bucket' => $this->bucket,
  74. 'key' => $this->key,
  75. 'acl' => $this->acl
  76. ] + $this->options))->promise();
  77. }
  78. // Perform a regular PutObject operation.
  79. $command = $this->client->getCommand('PutObject', [
  80. 'Bucket' => $this->bucket,
  81. 'Key' => $this->key,
  82. 'Body' => $this->body,
  83. 'ACL' => $this->acl,
  84. 'AddContentMD5' => $this->addContentMD5
  85. ] + $this->options['params']);
  86. if (is_callable($this->options['before_upload'])) {
  87. $this->options['before_upload']($command);
  88. }
  89. return $this->client->executeAsync($command);
  90. }
  91. public function upload()
  92. {
  93. return $this->promise()->wait();
  94. }
  95. /**
  96. * Determines if the body should be uploaded using PutObject or the
  97. * Multipart Upload System. It also modifies the passed-in $body as needed
  98. * to support the upload.
  99. *
  100. * @param StreamInterface $body Stream representing the body.
  101. * @param integer $threshold Minimum bytes before using Multipart.
  102. *
  103. * @return bool
  104. */
  105. private function requiresMultipart(StreamInterface &$body, $threshold)
  106. {
  107. // If body size known, compare to threshold to determine if Multipart.
  108. if ($body->getSize() !== null) {
  109. return $body->getSize() >= $threshold;
  110. }
  111. /**
  112. * Handle the situation where the body size is unknown.
  113. * Read up to 5MB into a buffer to determine how to upload the body.
  114. * @var StreamInterface $buffer
  115. */
  116. $buffer = Psr7\Utils::streamFor();
  117. Psr7\Utils::copyToStream($body, $buffer, MultipartUploader::PART_MIN_SIZE);
  118. // If body < 5MB, use PutObject with the buffer.
  119. if ($buffer->getSize() < MultipartUploader::PART_MIN_SIZE) {
  120. $buffer->seek(0);
  121. $body = $buffer;
  122. return false;
  123. }
  124. // If body >= 5 MB, then use multipart. [YES]
  125. if ($body->isSeekable() && $body->getMetadata('uri') !== 'php://input') {
  126. // If the body is seekable, just rewind the body.
  127. $body->seek(0);
  128. } else {
  129. // If the body is non-seekable, stitch the rewind the buffer and
  130. // the partially read body together into one stream. This avoids
  131. // unnecessary disc usage and does not require seeking on the
  132. // original stream.
  133. $buffer->seek(0);
  134. $body = new Psr7\AppendStream([$buffer, $body]);
  135. }
  136. return true;
  137. }
  138. }