You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
221 lines
5.5 KiB
221 lines
5.5 KiB
10 months ago
|
<?php
|
||
|
|
||
|
/**
|
||
|
* Copyright 2019 Huawei Technologies Co.,Ltd.
|
||
|
* Licensed under the Apache License, Version 2.0 (the "License"); you may not use
|
||
|
* this file except in compliance with the License. You may obtain a copy of the
|
||
|
* License at
|
||
|
*
|
||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||
|
*
|
||
|
* Unless required by applicable law or agreed to in writing, software distributed
|
||
|
* under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
|
||
|
* CONDITIONS OF ANY KIND, either express or implied. See the License for the
|
||
|
* specific language governing permissions and limitations under the License.
|
||
|
*
|
||
|
*/
|
||
|
|
||
|
/**
|
||
|
* This sample demonstrates how to multipart upload an object concurrently by copy mode
|
||
|
* to OBS using the OBS SDK for PHP.
|
||
|
*/
|
||
|
|
||
|
if (file_exists ( 'vendor/autoload.php' )) {
|
||
|
require 'vendor/autoload.php';
|
||
|
} else {
|
||
|
require '../vendor/autoload.php'; // sample env
|
||
|
}
|
||
|
|
||
|
if (file_exists ( 'obs-autoloader.php' )) {
|
||
|
require 'obs-autoloader.php';
|
||
|
} else {
|
||
|
require '../obs-autoloader.php'; // sample env
|
||
|
}
|
||
|
|
||
|
use Obs\ObsClient;
|
||
|
use Obs\ObsException;
|
||
|
|
||
|
$ak = '*** Provide your Access Key ***';
|
||
|
|
||
|
$sk = '*** Provide your Secret Key ***';
|
||
|
|
||
|
$endpoint = 'https://your-endpoint:443';
|
||
|
|
||
|
$bucketName = 'my-obs-bucket-demo';
|
||
|
|
||
|
$sourceBucketName = $bucketName;
|
||
|
|
||
|
$sourceObjectKey = 'my-obs-object-key-demo';
|
||
|
|
||
|
|
||
|
$objectKey = $sourceObjectKey . '-back';
|
||
|
|
||
|
|
||
|
/*
|
||
|
* Constructs a obs client instance with your account for accessing OBS
|
||
|
*/
|
||
|
$obsClient = ObsClient::factory ( [
|
||
|
'key' => $ak,
|
||
|
'secret' => $sk,
|
||
|
'endpoint' => $endpoint,
|
||
|
'socket_timeout' => 30,
|
||
|
'connect_timeout' => 10
|
||
|
] );
|
||
|
|
||
|
try
|
||
|
{
|
||
|
/*
|
||
|
* Create bucket
|
||
|
*/
|
||
|
printf("Create a new bucket for demo\n\n");
|
||
|
$obsClient -> createBucket(['Bucket' => $bucketName]);
|
||
|
|
||
|
|
||
|
$sampleFilePath = '/temp/test.txt'; //sample large file path
|
||
|
// you can prepare a large file in you filesystem first
|
||
|
createSampleFile($sampleFilePath);
|
||
|
|
||
|
/*
|
||
|
* Upload an object to your source bucket
|
||
|
*/
|
||
|
$obsClient -> putObject(['Bucket' => $sourceBucketName, 'Key' => $sourceObjectKey, 'SourceFile' => $sampleFilePath]);
|
||
|
|
||
|
/*
|
||
|
* Claim a upload id firstly
|
||
|
*/
|
||
|
$resp = $obsClient -> initiateMultipartUpload(['Bucket' => $bucketName, 'Key' => $objectKey]);
|
||
|
$uploadId = $resp['UploadId'];
|
||
|
printf("Claiming a new upload id %s\n\n", $uploadId);
|
||
|
|
||
|
|
||
|
$partSize = 5 * 1024 * 1024;
|
||
|
$resp = $obsClient -> getObjectMetadata(['Bucket' => $sourceBucketName, 'Key' => $sourceObjectKey]);
|
||
|
$objectSize = $resp['ContentLength'];
|
||
|
|
||
|
$partCount = $objectSize % $partSize === 0 ? intval($objectSize / $partSize) : intval($objectSize / $partSize) + 1;
|
||
|
|
||
|
if($partCount > 10000){
|
||
|
throw new \RuntimeException('Total parts count should not exceed 10000');
|
||
|
}
|
||
|
printf("Total parts count %d\n\n", $partCount);
|
||
|
|
||
|
/*
|
||
|
* Upload multiparts by copy mode
|
||
|
*/
|
||
|
$promise = null;
|
||
|
$parts = [];
|
||
|
printf("Begin to upload multiparts to OBS by copy mode\n\n");
|
||
|
for($i = 0; $i < $partCount; $i++){
|
||
|
$rangeStart = $i * $partSize;
|
||
|
$rangeEnd = ($i + 1 === $partCount) ? $objectSize - 1 : $rangeStart + $partSize - 1;
|
||
|
$partNumber = $i + 1;
|
||
|
$p = $obsClient -> copyPartAsync([
|
||
|
'Bucket' => $bucketName,
|
||
|
'Key' => $objectKey,
|
||
|
'UploadId' => $uploadId,
|
||
|
'PartNumber' => $partNumber,
|
||
|
'CopySource'=>sprintf('%s/%s', $sourceBucketName, $sourceObjectKey),
|
||
|
'CopySourceRange' => sprintf('bytes=%d-%d', $rangeStart, $rangeEnd)
|
||
|
], function($exception, $resp) use (&$parts, $partNumber){
|
||
|
$parts[] = ['PartNumber' => $partNumber, 'ETag' => $resp['ETag']];
|
||
|
printf ( "Part#" . strval ( $partNumber ) . " done\n\n" );
|
||
|
});
|
||
|
|
||
|
if($promise === null){
|
||
|
$promise = $p;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
/*
|
||
|
* Waiting for all parts finished
|
||
|
*/
|
||
|
$promise->wait();
|
||
|
|
||
|
usort($parts, function($a, $b){
|
||
|
if($a['PartNumber'] === $b['PartNumber']){
|
||
|
return 0;
|
||
|
}
|
||
|
return $a['PartNumber'] > $b['PartNumber'] ? 1 : -1;
|
||
|
});
|
||
|
|
||
|
/*
|
||
|
* Verify whether all parts are finished
|
||
|
*/
|
||
|
if(count($parts) !== $partCount){
|
||
|
throw new \RuntimeException('Upload multiparts fail due to some parts are not finished yet');
|
||
|
}
|
||
|
|
||
|
|
||
|
printf("Succeed to complete multiparts into an object named %s\n\n", $objectKey);
|
||
|
|
||
|
/*
|
||
|
* View all parts uploaded recently
|
||
|
*/
|
||
|
printf("Listing all parts......\n");
|
||
|
$resp = $obsClient -> listParts(['Bucket' => $bucketName, 'Key' => $objectKey, 'UploadId' => $uploadId]);
|
||
|
foreach ($resp['Parts'] as $part)
|
||
|
{
|
||
|
printf("\tPart#%d, ETag=%s\n", $part['PartNumber'], $part['ETag']);
|
||
|
}
|
||
|
printf("\n");
|
||
|
|
||
|
/*
|
||
|
* Complete to upload multiparts
|
||
|
*/
|
||
|
$resp = $obsClient->completeMultipartUpload([
|
||
|
'Bucket' => $bucketName,
|
||
|
'Key' => $objectKey,
|
||
|
'UploadId' => $uploadId,
|
||
|
'Parts'=> $parts
|
||
|
]);
|
||
|
|
||
|
if(file_exists($sampleFilePath)){
|
||
|
unlink($sampleFilePath);
|
||
|
}
|
||
|
|
||
|
} catch ( ObsException $e ) {
|
||
|
echo 'Response Code:' . $e->getStatusCode () . PHP_EOL;
|
||
|
echo 'Error Message:' . $e->getExceptionMessage () . PHP_EOL;
|
||
|
echo 'Error Code:' . $e->getExceptionCode () . PHP_EOL;
|
||
|
echo 'Request ID:' . $e->getRequestId () . PHP_EOL;
|
||
|
echo 'Exception Type:' . $e->getExceptionType () . PHP_EOL;
|
||
|
} finally{
|
||
|
$obsClient->close ();
|
||
|
}
|
||
|
|
||
|
|
||
|
function createSampleFile($filePath)
|
||
|
{
|
||
|
if(file_exists($filePath)){
|
||
|
return;
|
||
|
}
|
||
|
$filePath = iconv('UTF-8', 'GBK', $filePath);
|
||
|
if(is_string($filePath) && $filePath !== '')
|
||
|
{
|
||
|
$fp = null;
|
||
|
$dir = dirname($filePath);
|
||
|
try{
|
||
|
if(!is_dir($dir))
|
||
|
{
|
||
|
mkdir($dir,0755,true);
|
||
|
}
|
||
|
|
||
|
if(($fp = fopen($filePath, 'w')))
|
||
|
{
|
||
|
|
||
|
for($i=0;$i< 1000000;$i++){
|
||
|
fwrite($fp, uniqid() . "\n");
|
||
|
fwrite($fp, uniqid() . "\n");
|
||
|
if($i % 100 === 0){
|
||
|
fflush($fp);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}finally{
|
||
|
if($fp){
|
||
|
fclose($fp);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|