我想使用Rest API和PHP将blob从一个存储帐户复制到同一订阅中的另一个存储帐户。我可以使用下面的代码
在同一存储帐户中从一个容器到另一个容器执行复制blob<?php
$date = gmdate('D, d M Y H:i:s \G\M\T');
$account_name = "accname";
$destcontainername = "destcontainer";
$blobname = "blob.png";
$sourcecontainer = "sourcecontainer";
$account_key = "asdf";
$canonicalizedHeaders = "x-ms-copy-source:https://".$account_name.".blob.core.windows.net/".$sourcecontainer."/".$blobname."\nx-ms-date:$date\nx-ms-version:2015-04-05";
$canonicalizedResource = "/$account_name/$destcontainername/$blobname";
$arraysign = array();
$arraysign[] = 'PUT'; /*HTTP Verb*/
$arraysign[] = ''; /*Content-Encoding*/
$arraysign[] = ''; /*Content-Language*/
$arraysign[] = ''; /*Content-Length (include value when zero)*/
$arraysign[] = ''; /*Content-MD5*/
$arraysign[] = ''; /*Content-Type*/
$arraysign[] = ''; /*Date*/
$arraysign[] = ''; /*If-Modified-Since */
$arraysign[] = ''; /*If-Match*/
$arraysign[] = ''; /*If-None-Match*/
$arraysign[] = ''; /*If-Unmodified-Since*/
$arraysign[] = ''; /*Range*/
$arraysign[] = $canonicalizedHeaders; /*CanonicalizedHeaders*/
$arraysign[] = $canonicalizedResource; /*CanonicalizedResource*/
$stringtosign = implode("\n", $arraysign);
$signature = 'SharedKey'.' '.$account_name.':'.base64_encode(hash_hmac('sha256', $stringtosign, base64_decode($account_key), true));
$endpoint = 'https://'.$account_name.'.blob.core.windows.net';
$url = $endpoint.'/'.$destcontainername.'/'.$blobname;
$headers = [
'x-ms-copy-source:https://'.$account_name.'.blob.core.windows.net/'.$sourcecontainer.'/'.$blobname,
"x-ms-date:{$date}",
'x-ms-version:2015-04-05',
'Accept:application/json;odata=nometadata',
'Accept-Charset:UTF-8',
'Content-Length:0',
"Authorization:{$signature}"
];
$ch = curl_init();
curl_setopt($ch, CURLOPT_URL, $url);
curl_setopt($ch, CURLOPT_HTTPHEADER, $headers);
curl_setopt($ch, CURLOPT_CUSTOMREQUEST, 'PUT');
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
$response = curl_exec($ch);
echo curl_error($ch);
curl_close($ch);
echo '<pre>';print_r($response);
我想知道是否必须使用相同的副本blob rest API来完成它。如果是,则为$canonicalizedHeaders
和$canonicalizedResource
以及resquest rest API。我可以使用下面的powershell命令将blob从一个存储帐户复制到另一个存储帐户
Start-AzureStorageBlobCopy -DestContainer $destinationContainerName -DestContext $destinationContext -SrcBlob $vhdName -Context $sourceContext -SrcContainer $sourceSAContainerName
答案 0 :(得分:1)
您也可以使用相同的代码跨存储帐户复制blob。虽然$canonicalizedResource
保持不变,但您需要在x-ms-copy-source
的{{1}}标题中包含源blob的网址。
要记住的一件重要事情是,您在$canonicalizedHeaders
标头中指定的源blob网址必须可公开访问。这意味着如果您使用该URL并将其粘贴到浏览器的地址栏中,您应该能够访问该blob。如果源blob容器的ACL是x-ms-copy-source
或Blob
,那么您可以简单地指定blob URL(Container
),但是如果源blob容器的ACL是https://sourceaccountname.blob.core.windows.net/sourceblobcontainer/sourceblobname
那么你会需要在源blob上创建Private
且至少具有Shared Access Signature (SAS)
权限,并使用SAS URL来获取此标头的值。
另一件需要理解的是,跨存储帐户进行复制是一种异步操作。因此,当上面的代码成功执行时,blob复制操作将排队。在删除源blob或对其进行任何更改之前,必须检查复制操作是否已完成,否则复制操作将失败。