本文我们接着上文继续来探讨一下如何使用使用 PHP 来操作 Ceph S3 接口。
安装 AWS PHP SDK
很简单,直接使用 composer
安装就好了。
composer install aws/aws-sdk-php
你也可以直接新建 composer.json
文件,然后加上如下配置:
{
"require": {
"aws/aws-sdk-php": "^3.154"
}
}
然后再执行:
composer update
建立连接
use AwsS3S3Client;
define('AWS_KEY', 'MTAU54XU6ILI6097SNK8');
define('AWS_SECRET_KEY', 'TfYnLGXnOsQ6eXd8j98T3zYlwoewDFwTRwhqygFU');
define('DOMAIN', 'http://s3.rockyang.org');
require __DIR__.'/vendor/autoload.php';
// Instantiate the S3 class and point it at the desired host
$client = new S3Client([
'region' => '',
'version' => '2012-10-17',
'endpoint' => DOMAIN,
'credentials' => [
'key' => AWS_KEY,
'secret' => AWS_SECRET_KEY
],
// Set the S3 class to use objects.dreamhost.com/bucket
// instead of bucket.objects.dreamhost.com
'use_path_style_endpoint' => true
]);
获取 Bucket 列表
该 API 将返回一个 ` AWSResult` 对象。
// list all the buckets
$listResponse = $client->listBuckets();
$buckets = $listResponse['Buckets'];
foreach ($buckets as $bucket) {
echo $bucket['Name'] . " " . $bucket['CreationDate'] . "
";
}
输出:
bucket1 2020-10-19T22:05:39.000Z
bucket1 2020-10-19T22:05:48.000Z
bucket1 2020-10-19T22:07:18.000Z
创建一个 Bucket
$client->createBucket(['Bucket' => 'my-new-bucket']);
删除一个 Bucket
$client->deleteBucket(['Bucket' => 'my-old-bucket']);
获取文件列表
$objectsListResponse = $client->listObjects(['Bucket' => $bucketname]);
$objects = $objectsListResponse['Contents'] ?? [];
foreach ($objects as $object) {
echo $object['Key'] . " " . $object['Size'] . " " . $object['LastModified'] . "
";
}
上传文件
上传一个文本文件:
$client->putObject([
'Bucket' => 'my-bucket-name',
'Key' => 'hello.txt',
'Body' => "Hello World!" // 文件内容
]);
上传一个二进制文件:
$sourceFile = "hello.jpg";
$r = $client->putObject([
'Bucket' => 'my-bucket-name',
'Key' => 'hello.jpg',
'SourceFile' => $sourceFile
]);
$r = $r->toArray();
$res = $r['@metadata'];
if (isset($res["statusCode"]) && $res["statusCode"] == 200) {
printf("Upload successfully. url: %s
", $r["ObjectURL"]);
} else {
echo "File upload failed.
";
}
删除文件
$client->deleteObject(['Bucket' => 'my-bucket-name', 'Key' => 'hello.txt']);
下载文件
$object = $client->getObject(['Bucket' => 'my-bucket-name', 'Key' => 'poetry.pdf']);
file_put_contents('/home/larry/documents/poetry.pdf', $object['Body']->getContents());
生成文件下载地址
$hello_url = $client->getObjectUrl('my-bucket-name', 'hello.txt');
echo $hello_url."
";
$secret_plans_cmd = $client->getCommand('GetObject', ['Bucket' => 'my-bucket-name', 'Key' => 'secret_plans.txt']);
$request = $client->createPresignedRequest($secret_plans_cmd, '+1 hour');
echo $request->getUri()."
";
输出结果:
http://objects.dreamhost.com/my-bucket-name/hello.txt
http://objects.dreamhost.com/my-bucket-name/secret_plans.txt?X-Amz-Content-Sha256=UNSIGNED-PAYLOAD&X-Amz-Algorithm=AWS4-HMAC-SHA256&X-Amz-Credential=sandboxAccessKey%2F20190116%2F%2Fs3%2Faws4_request&X-Amz-Date=20190116T125520Z&X-Amz-SignedHeaders=host&X-Amz-Expires=3600&X-Amz-Signature=61921f07c73d7695e47a2192cf55ae030f34c44c512b2160bb5a936b2b48d923
参考文档
http://docs.ceph.org.cn/radosgw/s3/php/
PHP S3 样例
新建一个连接
下面的代码会新建一个连接,这样你就可以和服务器交互.
<?php
define('AWS_KEY', 'place access key here');
define('AWS_SECRET_KEY', 'place secret key here');
define('AWS_CANONICAL_ID', 'your DHO Username');
define('AWS_CANONICAL_NAME', 'Also your DHO Username!');
$HOST = 'objects.dreamhost.com';
// require the amazon sdk for php library
require_once 'AWSSDKforPHP/sdk.class.php';
// Instantiate the S3 class and point it at the desired host
$Connection = new AmazonS3(array(
'key' => AWS_KEY,
'secret' => AWS_SECRET_KEY,
'canonical_id' => AWS_CANONICAL_ID,
'canonical_name' => AWS_CANONICAL_NAME,
));
$Connection->set_hostname($HOST);
$Connection->allow_hostname_override(false);
// Set the S3 class to use objects.dreamhost.com/bucket
// instead of bucket.objects.dreamhost.com
$Connection->enable_path_style();
列出用户的所有 BUCKET
下面的代码会生成 CFSimpleXML 类型的对象列表,它代表你拥有的bucket。这也会打印出每个bucket 的 bucket 名和创建时间。
<?php
$ListResponse = $Connection->list_buckets();
$Buckets = $ListResponse->body->Buckets->Bucket;
foreach ($Buckets as $Bucket) {
echo $Bucket->Name . " " . $Bucket->CreationDate . "
";
}
输出形式类似下面这样:
mahbuckat1 2011-04-21T18:05:39.000Z
mahbuckat2 2011-04-21T18:05:48.000Z
mahbuckat3 2011-04-21T18:07:18.000Z
新建一个 BUCKET
下面的代码会新建一个名为 my-new-bucket 的bucket,并返回一个 CFResponse 对象
Note
这个命令需要指定 region 作为第二个参数, 所以我们使用 AmazonS3::REGION_US_E1, 因为它的内容是 ''
<?php
$Connection->create_bucket('my-new-bucket', AmazonS3::REGION_US_E1);
列出 BUCKET 的内容
下面的代码会输出 CFSimpleXML 对象的一个数组,它代表了bucket内的对象。这也会打印出每一个对象的名字、文件尺寸和最近修改时间。
<?php
$ObjectsListResponse = $Connection->list_objects($bucketname);
$Objects = $ObjectsListResponse->body->Contents;
foreach ($Objects as $Object) {
echo $Object->Key . " " . $Object->Size . " " . $Object->LastModified . "
";
}
Note
如果在这个 bucket 中有超过1000个对象,你需要检查 $ObjectListResponse->body->isTruncated 的值,然后使用列出的最后一个 key 的名字再次运行。一直这样 做直到 isTruncated 的值不再是 true。
如果该 bucket 内有文件,输出形式类似下面这样:
myphoto1.jpg 251262 2011-08-08T21:35:48.000Z
myphoto2.jpg 262518 2011-08-08T21:38:01.000Z
删除 BUCKET
下面的代码会删除名为 my-old-bucket 的 bucket,并返回一个 CFResponse 对象。
Note
Bucket必须为空!否则它不会工作!
<?php
$Connection->delete_bucket('my-old-bucket');
强制删除非空 BUCKETS
下面的代码会删除一个 bucket 即使它不是空的。
<?php
$Connection->delete_bucket('my-old-bucket', 1);
新建一个对象
下面的代码会新建一个内容是字符串``”Hello World!”`` 的文件 hello.txt。
<?php
$Connection->create_object('my-bucket-name', 'hello.txt', array(
'body' => "Hello World!",
));
修改一个对象的 ACL
下面的代码会将对象 hello.txt 的权限变为公开可读,而将 secret_plans.txt 的权限设为私有。
<?php
$Connection->set_object_acl('my-bucket-name', 'hello.txt', AmazonS3::ACL_PUBLIC);
$Connection->set_object_acl('my-bucket-name', 'secret_plans.txt', AmazonS3::ACL_PRIVATE);
删除一个对象
下面的代码会删除对象 goodbye.txt
<?php
$Connection->delete_object('my-bucket-name', 'goodbye.txt');
下载一个对象 (到文件)
下面的代码会下载对象 perl_poetry.pdf 并将它存到位置 C:UserslarryDocuments
<?php
$FileHandle = fopen('/home/larry/documents/poetry.pdf', 'w+');
$Connection->get_object('my-bucket-name', 'poetry.pdf', array(
'fileDownload' => $FileHandle,
));
生成对象的下载 URLS (带签名和不带签名)
下面的代码会为 hello.txt 生成一个无签名为下 载URL。这个操作会生效是因为前面我们已经设置 hello.txt 的ACL 为公开可读。下面的代码同时会 为 secret_plans.txt 生成一个有效时间是一个小 时的带签名的下载 URL。带签名的下载URL 在这个时间内 是可用的,即使对象的权限是私有(当时间到期后URL 将不 可用)。
<?php
my $plans_url = $Connection->get_object_url('my-bucket-name', 'hello.txt');
echo $plans_url . "
";
my $secret_url = $Connection->get_object_url('my-bucket-name', 'secret_plans.txt', '1 hour');
echo $secret_url . "
";
输出形式类似下面这样:
http://objects.dreamhost.com/my-bucket-name/hello.txt
http://objects.dreamhost.com/my-bucket-name/secret_plans.txt?Signature=XXXXXXXXXXXXXXXXXXXXXXXXXXX&Expires=13160270