client = $client; $this->options = $options; $this->partSize = isset( $options['PartSize'] ) ? $options['PartSize'] : self::DEFAULT_PART_SIZE; $this->concurrency = isset( $options['Concurrency'] ) ? $options['Concurrency'] : 10; $this->progress = isset( $options['Progress'] ) ? $options['Progress'] : function( $totalSize, $downloadedSize ) { } ; $this->parts = []; $this->partNumberList = []; $this->downloadedSize = 0; $this->totalSize = $contentLength; $this->saveAs = $saveAs; $this->resumableJson = []; $this->resumableJson = isset( $options['ResumableJson'] ) ? $options['ResumableJson'] : []; unset( $options['ResumableJson'] ); $this->resumableTaskFile = isset( $options['ResumableTaskFile'] ) ? $options['ResumableTaskFile'] : $saveAs . '.cosresumabletask'; $this->resumableDownload = isset( $options['ResumableDownload'] ) ? $options['ResumableDownload'] : false; } public function performdownloading() { if ( $this->resumableDownload ) { try { if ( file_exists( $this->resumableTaskFile ) ) { $origin_content = file_get_contents( $this->resumableTaskFile ); $this->resumableJsonLocal = json_decode( $origin_content, true ); if ( $this->resumableJsonLocal == null ) { $this->resumableJsonLocal = []; } else if ( $this->resumableJsonLocal['LastModified'] != $this->resumableJson['LastModified'] || $this->resumableJsonLocal['ContentLength'] != $this->resumableJson['ContentLength'] || $this->resumableJsonLocal['ETag'] != $this->resumableJson['ETag'] || $this->resumableJsonLocal['Crc64ecma'] != $this->resumableJson['Crc64ecma'] ) { $this->resumableDownload = false; } } } catch ( \Exception $e ) { $this->resumableDownload = false; } } try { if ($this->resumableDownload) { $this->fp = fopen( $this->saveAs, 'r+' ); } else { $this->fp = fopen( $this->saveAs, 'wb' ); } $rt = $this->donwloadParts(); $this->resumableJson['DownloadedBlocks'] = []; if (file_exists( $this->resumableTaskFile )) { unlink($this->resumableTaskFile); } } catch ( \Exception $e ) { $this->fp_resume = fopen( $this->resumableTaskFile, 'wb' ); fwrite( $this->fp_resume, json_encode( $this->resumableJson ) ); fclose( $this->fp_resume ); throw ( $e ); } finally { fclose( $this->fp ); } return $rt; } public function donwloadParts() { $uploadRequests = function () { $index = 1; $partSize = 0; for ( $offset = 0; $offset < $this->totalSize; ) { $partSize = $this->partSize; if ( $offset + $this->partSize >= $this->totalSize ) { $partSize = $this->totalSize - $offset; } $this->parts[$index]['PartSize'] = $partSize; $this->parts[$index]['Offset'] = $offset; $begin = $offset; $end = $offset + $partSize - 1; if ( !( $this->resumableDownload && isset( $this->resumableJsonLocal['DownloadedBlocks'] ) && in_array( ['from' => $begin, 'to' => $end], $this->resumableJsonLocal['DownloadedBlocks'] ) ) ) { $params = array( 'Bucket' => $this->options['Bucket'], 'Key' => $this->options['Key'], 'Range' => sprintf( 'bytes=%d-%d', $begin, $end ) ); $command = $this->client->getCommand( 'getObject', $params ); $request = $this->client->commandToRequestTransformer( $command ); $index += 1; yield $request; } else { $this->resumableJson['DownloadedBlocks'][] = ['from' => $begin, 'to' => $end]; $this->downloadedSize += $partSize; call_user_func_array( $this->progress, [$this->totalSize, $this->downloadedSize] ); } $offset += $partSize; } } ; $pool = new Pool( $this->client->httpClient, $uploadRequests(), [ 'concurrency' => $this->concurrency, 'fulfilled' => function ( $response, $index ) { $index = $index + 1; $stream = $response->getBody(); $offset = $this->parts[$index]['Offset']; $partsize = 8192; $begin = $offset; fseek( $this->fp, $offset ); while ( !$stream->eof() ) { $output = $stream->read( $partsize ); $writeLen = fwrite( $this->fp, $output ); $offset += $writeLen; } $end = $offset - 1; $this->resumableJson['DownloadedBlocks'][] = ['from' => $begin, 'to' => $end]; $partSize = $this->parts[$index]['PartSize']; $this->downloadedSize += $partSize; call_user_func_array( $this->progress, [$this->totalSize, $this->downloadedSize] ); } , 'rejected' => function ( $reason, $index ) { throw( $reason ); } ] ); $promise = $pool->promise(); $promise->wait(); } }