| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150 | 
							- <?php
 
- namespace Aws\S3;
 
- use GuzzleHttp\Promise\PromiseInterface;
 
- use GuzzleHttp\Promise\PromisorInterface;
 
- use GuzzleHttp\Psr7;
 
- use Psr\Http\Message\StreamInterface;
 
- /**
 
-  * Uploads an object to S3, using a PutObject command or a multipart upload as
 
-  * appropriate.
 
-  */
 
- class ObjectUploader implements PromisorInterface
 
- {
 
-     const DEFAULT_MULTIPART_THRESHOLD = 16777216;
 
-     private $client;
 
-     private $bucket;
 
-     private $key;
 
-     private $body;
 
-     private $acl;
 
-     private $options;
 
-     private static $defaults = [
 
-         'before_upload' => null,
 
-         'concurrency'   => 3,
 
-         'mup_threshold' => self::DEFAULT_MULTIPART_THRESHOLD,
 
-         'params'        => [],
 
-         'part_size'     => null,
 
-     ];
 
-     private $addContentMD5;
 
-     /**
 
-      * @param S3ClientInterface $client         The S3 Client used to execute
 
-      *                                          the upload command(s).
 
-      * @param string            $bucket         Bucket to upload the object, or
 
-      *                                          an S3 access point ARN.
 
-      * @param string            $key            Key of the object.
 
-      * @param mixed             $body           Object data to upload. Can be a
 
-      *                                          StreamInterface, PHP stream
 
-      *                                          resource, or a string of data to
 
-      *                                          upload.
 
-      * @param string            $acl            ACL to apply to the copy
 
-      *                                          (default: private).
 
-      * @param array             $options        Options used to configure the
 
-      *                                          copy process. Options passed in
 
-      *                                          through 'params' are added to
 
-      *                                          the sub command(s).
 
-      */
 
-     public function __construct(
 
-         S3ClientInterface $client,
 
-         $bucket,
 
-         $key,
 
-         $body,
 
-         $acl = 'private',
 
-         array $options = []
 
-     ) {
 
-         $this->client = $client;
 
-         $this->bucket = $bucket;
 
-         $this->key = $key;
 
-         $this->body = Psr7\Utils::streamFor($body);
 
-         $this->acl = $acl;
 
-         $this->options = $options + self::$defaults;
 
-         // Handle "add_content_md5" option.
 
-         $this->addContentMD5 = isset($options['add_content_md5'])
 
-             && $options['add_content_md5'] === true;
 
-     }
 
-     /**
 
-      * @return PromiseInterface
 
-      */
 
-     public function promise(): PromiseInterface
 
-     {
 
-         /** @var int $mup_threshold */
 
-         $mup_threshold = $this->options['mup_threshold'];
 
-         if ($this->requiresMultipart($this->body, $mup_threshold)) {
 
-             // Perform a multipart upload.
 
-             return (new MultipartUploader($this->client, $this->body, [
 
-                     'bucket' => $this->bucket,
 
-                     'key'    => $this->key,
 
-                     'acl'    => $this->acl
 
-                 ] + $this->options))->promise();
 
-         }
 
-         // Perform a regular PutObject operation.
 
-         $command = $this->client->getCommand('PutObject', [
 
-                 'Bucket' => $this->bucket,
 
-                 'Key'    => $this->key,
 
-                 'Body'   => $this->body,
 
-                 'ACL'    => $this->acl,
 
-                 'AddContentMD5' => $this->addContentMD5
 
-             ] + $this->options['params']);
 
-         if (is_callable($this->options['before_upload'])) {
 
-             $this->options['before_upload']($command);
 
-         }
 
-         return $this->client->executeAsync($command);
 
-     }
 
-     public function upload()
 
-     {
 
-         return $this->promise()->wait();
 
-     }
 
-     /**
 
-      * Determines if the body should be uploaded using PutObject or the
 
-      * Multipart Upload System. It also modifies the passed-in $body as needed
 
-      * to support the upload.
 
-      *
 
-      * @param StreamInterface $body      Stream representing the body.
 
-      * @param integer             $threshold Minimum bytes before using Multipart.
 
-      *
 
-      * @return bool
 
-      */
 
-     private function requiresMultipart(StreamInterface &$body, $threshold)
 
-     {
 
-         // If body size known, compare to threshold to determine if Multipart.
 
-         if ($body->getSize() !== null) {
 
-             return $body->getSize() >= $threshold;
 
-         }
 
-         /**
 
-          * Handle the situation where the body size is unknown.
 
-          * Read up to 5MB into a buffer to determine how to upload the body.
 
-          * @var StreamInterface $buffer
 
-          */
 
-         $buffer = Psr7\Utils::streamFor();
 
-         Psr7\Utils::copyToStream($body, $buffer, MultipartUploader::PART_MIN_SIZE);
 
-         // If body < 5MB, use PutObject with the buffer.
 
-         if ($buffer->getSize() < MultipartUploader::PART_MIN_SIZE) {
 
-             $buffer->seek(0);
 
-             $body = $buffer;
 
-             return false;
 
-         }
 
-         // If body >= 5 MB, then use multipart. [YES]
 
-         if ($body->isSeekable() && $body->getMetadata('uri') !== 'php://input') {
 
-             // If the body is seekable, just rewind the body.
 
-             $body->seek(0);
 
-         } else {
 
-             // If the body is non-seekable, stitch the rewind the buffer and
 
-             // the partially read body together into one stream. This avoids
 
-             // unnecessary disc usage and does not require seeking on the
 
-             // original stream.
 
-             $buffer->seek(0);
 
-             $body = new Psr7\AppendStream([$buffer, $body]);
 
-         }
 
-         return true;
 
-     }
 
- }
 
 
  |