| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242 | 
							- <?php
 
- namespace Aws\S3;
 
- use Aws\Arn\ArnParser;
 
- use Aws\Multipart\AbstractUploadManager;
 
- use Aws\ResultInterface;
 
- use GuzzleHttp\Psr7;
 
- class MultipartCopy extends AbstractUploadManager
 
- {
 
-     use MultipartUploadingTrait;
 
-     /** @var string|array */
 
-     private $source;
 
-     /** @var string */
 
-     private $sourceVersionId;
 
-     /** @var ResultInterface */
 
-     private $sourceMetadata;
 
-     /**
 
-      * Creates a multipart upload for copying an S3 object.
 
-      *
 
-      * The valid configuration options are as follows:
 
-      *
 
-      * - acl: (string) ACL to set on the object being upload. Objects are
 
-      *   private by default.
 
-      * - before_complete: (callable) Callback to invoke before the
 
-      *   `CompleteMultipartUpload` operation. The callback should have a
 
-      *   function signature like `function (Aws\Command $command) {...}`.
 
-      * - before_initiate: (callable) Callback to invoke before the
 
-      *   `CreateMultipartUpload` operation. The callback should have a function
 
-      *   signature like `function (Aws\Command $command) {...}`.
 
-      * - before_upload: (callable) Callback to invoke before `UploadPartCopy`
 
-      *   operations. The callback should have a function signature like
 
-      *   `function (Aws\Command $command) {...}`.
 
-      * - bucket: (string, required) Name of the bucket to which the object is
 
-      *   being uploaded.
 
-      * - concurrency: (int, default=int(5)) Maximum number of concurrent
 
-      *   `UploadPart` operations allowed during the multipart upload.
 
-      * - key: (string, required) Key to use for the object being uploaded.
 
-      * - params: (array) An array of key/value parameters that will be applied
 
-      *   to each of the sub-commands run by the uploader as a base.
 
-      *   Auto-calculated options will override these parameters. If you need
 
-      *   more granularity over parameters to each sub-command, use the before_*
 
-      *   options detailed above to update the commands directly.
 
-      * - part_size: (int, default=int(5242880)) Part size, in bytes, to use when
 
-      *   doing a multipart upload. This must between 5 MB and 5 GB, inclusive.
 
-      * - state: (Aws\Multipart\UploadState) An object that represents the state
 
-      *   of the multipart upload and that is used to resume a previous upload.
 
-      *   When this option is provided, the `bucket`, `key`, and `part_size`
 
-      *   options are ignored.
 
-      * - source_metadata: (Aws\ResultInterface) An object that represents the
 
-      *   result of executing a HeadObject command on the copy source.
 
-      *
 
-      * @param S3ClientInterface $client Client used for the upload.
 
-      * @param string|array $source Location of the data to be copied (in the
 
-      *                       form /<bucket>/<key>).  If the key contains a '?'
 
-      *                       character, instead pass an array of source_key,
 
-      *                       source_bucket, and source_version_id.
 
-      * @param array $config Configuration used to perform the upload.
 
-      */
 
-     public function __construct(
 
-         S3ClientInterface $client,
 
-         $source,
 
-         array $config = []
 
-     ) {
 
-         if (is_array($source)) {
 
-             $this->source = $source;
 
-         } else {
 
-             $this->source = $this->getInputSource($source);
 
-         }
 
-         parent::__construct(
 
-             $client,
 
-             array_change_key_case($config) + ['source_metadata' => null]
 
-         );
 
-     }
 
-     /**
 
-      * An alias of the self::upload method.
 
-      *
 
-      * @see self::upload
 
-      */
 
-     public function copy()
 
-     {
 
-         return $this->upload();
 
-     }
 
-     protected function loadUploadWorkflowInfo()
 
-     {
 
-         return [
 
-             'command' => [
 
-                 'initiate' => 'CreateMultipartUpload',
 
-                 'upload' => 'UploadPartCopy',
 
-                 'complete' => 'CompleteMultipartUpload',
 
-             ],
 
-             'id' => [
 
-                 'bucket' => 'Bucket',
 
-                 'key' => 'Key',
 
-                 'upload_id' => 'UploadId',
 
-             ],
 
-             'part_num' => 'PartNumber',
 
-         ];
 
-     }
 
-     protected function getUploadCommands(callable $resultHandler)
 
-     {
 
-         $parts = ceil($this->getSourceSize() / $this->determinePartSize());
 
-         for ($partNumber = 1; $partNumber <= $parts; $partNumber++) {
 
-             // If we haven't already uploaded this part, yield a new part.
 
-             if (!$this->state->hasPartBeenUploaded($partNumber)) {
 
-                 $command = $this->client->getCommand(
 
-                     $this->info['command']['upload'],
 
-                     $this->createPart($partNumber, $parts) + $this->getState()->getId()
 
-                 );
 
-                 $command->getHandlerList()->appendSign($resultHandler, 'mup');
 
-                 yield $command;
 
-             }
 
-         }
 
-     }
 
-     private function createPart($partNumber, $partsCount)
 
-     {
 
-         $data = [];
 
-         // Apply custom params to UploadPartCopy data
 
-         $config = $this->getConfig();
 
-         $params = isset($config['params']) ? $config['params'] : [];
 
-         foreach ($params as $k => $v) {
 
-             $data[$k] = $v;
 
-         }
 
-         // The source parameter here is usually a string, but can be overloaded as an array
 
-         // if the key contains a '?' character to specify where the query parameters start
 
-         if (is_array($this->source)) {
 
-             $key = str_replace('%2F', '/', rawurlencode($this->source['source_key']));
 
-             $bucket = $this->source['source_bucket'];
 
-         } else {
 
-             list($bucket, $key) = explode('/', ltrim($this->source, '/'), 2);
 
-             $key = implode(
 
-                 '/',
 
-                 array_map(
 
-                     'urlencode',
 
-                     explode('/', rawurldecode($key))
 
-                 )
 
-             );
 
-         }
 
-         $uri = ArnParser::isArn($bucket) ? '' : '/';
 
-         $uri .= $bucket . '/' . $key;
 
-         $data['CopySource'] = $uri;
 
-         $data['PartNumber'] = $partNumber;
 
-         if (!empty($this->sourceVersionId)) {
 
-             $data['CopySource'] .= "?versionId=" . $this->sourceVersionId;
 
-         }
 
-         $defaultPartSize = $this->determinePartSize();
 
-         $startByte = $defaultPartSize * ($partNumber - 1);
 
-         $data['ContentLength'] = $partNumber < $partsCount
 
-             ? $defaultPartSize
 
-             : $this->getSourceSize() - ($defaultPartSize * ($partsCount - 1));
 
-         $endByte = $startByte + $data['ContentLength'] - 1;
 
-         $data['CopySourceRange'] = "bytes=$startByte-$endByte";
 
-         return $data;
 
-     }
 
-     protected function extractETag(ResultInterface $result)
 
-     {
 
-         return $result->search('CopyPartResult.ETag');
 
-     }
 
-     protected function getSourceMimeType()
 
-     {
 
-         return $this->getSourceMetadata()['ContentType'];
 
-     }
 
-     protected function getSourceSize()
 
-     {
 
-         return $this->getSourceMetadata()['ContentLength'];
 
-     }
 
-     private function getSourceMetadata()
 
-     {
 
-         if (empty($this->sourceMetadata)) {
 
-             $this->sourceMetadata = $this->fetchSourceMetadata();
 
-         }
 
-         return $this->sourceMetadata;
 
-     }
 
-     private function fetchSourceMetadata()
 
-     {
 
-         if ($this->config['source_metadata'] instanceof ResultInterface) {
 
-             return $this->config['source_metadata'];
 
-         }
 
-         //if the source variable was overloaded with an array, use the inputs for key and bucket
 
-         if (is_array($this->source)) {
 
-             $headParams = [
 
-                 'Key' => $this->source['source_key'],
 
-                 'Bucket' => $this->source['source_bucket']
 
-             ];
 
-             if (isset($this->source['source_version_id'])) {
 
-                 $this->sourceVersionId = $this->source['source_version_id'];
 
-                 $headParams['VersionId'] = $this->sourceVersionId;
 
-             }
 
-         //otherwise, use the default source parsing behavior
 
-         } else {
 
-             list($bucket, $key) = explode('/', ltrim($this->source, '/'), 2);
 
-             $headParams = [
 
-                 'Bucket' => $bucket,
 
-                 'Key' => $key,
 
-             ];
 
-             if (strpos($key, '?')) {
 
-                 list($key, $query) = explode('?', $key, 2);
 
-                 $headParams['Key'] = $key;
 
-                 $query = Psr7\Query::parse($query, false);
 
-                 if (isset($query['versionId'])) {
 
-                     $this->sourceVersionId = $query['versionId'];
 
-                     $headParams['VersionId'] = $this->sourceVersionId;
 
-                 }
 
-             }
 
-         }
 
-         return $this->client->headObject($headParams);
 
-     }
 
-     /**
 
-      * Get the url decoded input source, starting with a slash if it is not an
 
-      * ARN to standardize the source location syntax.
 
-      *
 
-      * @param string $inputSource The source that was passed to the constructor
 
-      * @return string The source, starting with a slash if it's not an arn
 
-      */
 
-     private function getInputSource($inputSource)
 
-     {
 
-         $sourceBuilder = ArnParser::isArn($inputSource) ? '' : '/';
 
-         $sourceBuilder .= ltrim(rawurldecode($inputSource), '/');
 
-         return $sourceBuilder;
 
-     }
 
- }
 
 
  |