如何在PHP中发送“multipart/related”内容类型

7
我需要用PHP发送multipart/related内容。其中包括3个文件(2个XML,1个PDF)。XML文件需要以7位编码方式编码,而PDF文件需要以base64编码方式编码。
我可以像这样构建一个文件,但是我不知道如何在PHP中使用curl发送它。
内容应该像这样(我删除了大部分编码的pdf)。此示例来自另一个闭源应用程序:
MIME-Version: 1.0
Content-Type: multipart/related; 
    boundary="----=_Part_0_869724450.1481019442425"

------=_Part_0_869724450.1481019442425
Content-Type: application/vnd.cip4-jmf+xml; name=SubmitQueueEntry.jmf
Content-Transfer-Encoding: 7bit
Content-Disposition: attachment; filename=SubmitQueueEntry.jmf
Content-ID: <5cba3621:158d3a34526:-7ffe@hp.com>
Content-Length: 465

<?xml version="1.0" encoding="UTF-8"?>
<JMF xmlns="http://www.CIP4.org/JDFSchema_1_1" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" SenderID="Alces 0.9.9.1" TimeStamp="2012-08-21T14:55:08-06:00" Version="1.3">
  <Command ID="ALCES_YECIYJ_4_20120821145508" Type="SubmitQueueEntry" xsi:type="CommandSubmitQueueEntry">
    <QueueSubmissionParams ReturnJMF="http://YOURHOSTNAME:9090/jmf" URL="cid:5cba3621:158d3a34526:-7fff@hp.com" />
  </Command>
</JMF>

------=_Part_0_869724450.1481019442425
Content-Type: application/vnd.cip4-jdf+xml; name=test.pdf.jdf
Content-Transfer-Encoding: 7bit
Content-Disposition: attachment; filename=test.pdf.jdf
Content-ID: <5cba3621:158d3a34526:-7fff@hp.com>
Content-Length: 1536

<?xml version="1.0" encoding="UTF-8"?>
<JDF xmlns="http://www.CIP4.org/JDFSchema_1_1" Type="Combined" ID="rootNodeId" Status="Waiting" JobPartID="My Job Part ID" Version="1.3" Types="DigitalPrinting" DescriptiveName="My Job" JobID="My Job ID">
   <Comment Name="JobSpec">Photobook</Comment><ResourcePool>
      <Media Class="Consumable" ID="M001" Status="Available" />
      <DigitalPrintingParams Class="Parameter" ID="DPP001" Status="Available" />
      <RunList ID="RunList_1" Status="Available" Class="Parameter">
            <LayoutElement>
               <FileSpec MimeType="application/pdf" URL="cid:5cba3621:158d3a34526:-8000@hp.com" />
            </LayoutElement>
      </RunList>
      <Component ID="Component" ComponentType="FinalProduct" Status="Unavailable" Class="Quantity" />
   <NodeInfo ID="NI001" Class="Parameter" Status="Available" LastEnd="2015-01-21T13:14:40" JobPriority="50"><Comment Name="Instructions">Emboss with gold stitch</Comment><GeneralID IDUsage="EmbossText" IDValue="Sara and Michael's Wedding,EmbossFontSize 20pt" /></NodeInfo><CustomerInfo Class="Parameter" ID="CI001" Status="Available" /></ResourcePool>
   <ResourceLinkPool>
      <MediaLink rRef="M001" Usage="Input" />
      <DigitalPrintingParamsLink rRef="DPP001" Usage="Input" />
      <RunListLink rRef="RunList_1" Usage="Input" />
      <ComponentLink Usage="Output" rRef="Component" Amount="1" />
   <NodeInfoLink rRef="NI001" Usage="Input" /><CustomerInfoLink rRef="CI001" Usage="Input" /></ResourceLinkPool>
</JDF>

------=_Part_0_869724450.1481019442425
Content-Type: application/octet-stream; name=_113HN_test.pdf
Content-Transfer-Encoding: base64
Content-Disposition: attachment; filename=_113HN_test.pdf
Content-ID: <5cba3621:158d3a34526:-8000@hp.com>
Content-Length: 576230

JVBERi0xLjUNJeLjz9MNCjEgMCBvYmoNPDwvTWV0YWRhdGEgMiAwIFIvT0NQcm9wZXJ0aWVzPDwv
RDw8L09OWzUgMCBSXS9PcmRlciA2IDAgUi9SQkdyb3Vwc1tdPj4vT0NHc1s1IDAgUl0+Pi9QYWdl
cyAzIDAgUi9UeXBlL0NhdGFsb2c+Pg1lbmRvYmoNMiAwIG9iag08PC9MZW5ndGggMjcwNDIvU3Vi
dHlwZS9YTUwvVHlwZS9NZXRhZGF0YT4+c3RyZWFtDQo8P3hwYWNrZXQgYmVnaW49Iu+7vyIgaWQ9
...
bg0KMDAwMDM2NjI5NSAwMDAwMCBuDQowMDAwMzk1NDY2IDAwMDAwIG4NCjAwMDA0MTk5MjggMDAw
MDAgbg0KdHJhaWxlcg08PC9TaXplIDQxL1Jvb3QgMSAwIFIvSW5mbyA0MCAwIFIvSURbPEM3MjlE
QzVEMUYwODQzNDA4NEY0QTlBNEJBQTE4RjhCPjxDMjU2RDIxQjA5Q0Y0MjQ4QTA5REIzRDgxNjQw
NkMzMT5dPj4Nc3RhcnR4cmVmDTQyMDEyMQ0lJUVPRg0=
------=_Part_0_869724450.1481019442425--

我尝试了以下方法,但结果为空:
  $url="1.2.3.4";
  $data = array('name' => basename($filePath), 'file' => '@' . realpath($filePath));
  $data = array('file' => '@' . realpath($filePath));

  curl_setopt($ch, CURLOPT_POST, 1);
  curl_setopt($ch, CURLOPT_POSTFIELDS, $data);
  curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
  curl_setopt($ch, CURLOPT_TIMEOUT, 300);
  curl_setopt($ch, CURLOPT_HTTPHEADER, array(
    'Content-type: multipart/related'
    )); 
  $result = curl_exec($ch);

  $info = curl_getinfo($ch);
//  curl_close($ch);

  if ($result === false || $info['http_code'] != 200) {
    $output = "No cURL data returned for $url [" . $info['http_code'] . "]";
    if (curl_error($ch))
      $output .= "\n" . curl_error($ch);
  }
  else {
    // 'OK' status; format $output data if necessary here:
    echo 'succes';
  }

也许使用curl选项来构建完整内容会是更好的方法?而不是在之前“手动”构建内容... 预先感谢您!

“empty result” 这个短语具体指什么?您在 if/else 语句中期望得到非空输出。您是否收到了 errorsuccess 消息?如果收到了 error 消息,返回的 http_code 是多少? - user2560539
请查看 https://dev59.com/i4Pba4cB1Zd3GeqPu6KW ,简而言之,curl 没有内置的功能来使用 multipart/related 编码请求,因此您需要自己手动进行编码。 - hanshenrik
另外,由于您需要处理curl和发送文件,请同时查看CURLFile类http://php.net/manual/en/class.curlfile.php#class.curlfile以及此处的第一个示例http://php.net/manual/en/curlfile.construct.php。 - user2560539
最后但并非最不重要的是,也请查看这个例子http://php.net/manual/en/curlfile.construct.php#115160,我认为它涵盖了这个主题。 - user2560539
1个回答

3
根据https://dev59.com/i4Pba4cB1Zd3GeqPu6KW#25998544,libcurl没有内置的机制将请求编码为multipart/related(此人是curl的作者和主要开发者,所以他应该知道),这意味着您必须自己对请求进行编码。
我没有服务器可供测试,但我认为它可能会像这样:
<?php
declare(strict_types = 1);
$files = array (
        new MultipartRelatedFile ( 'foo.php' ),
        new MultipartRelatedFile ( 'bar.php' ),
        new MultipartRelatedFile ( 'baz.php' ) 
);
$body = generateMultipartRelatedBody ( $files, $boundary );
$ch = curl_init ();
curl_setopt_array ( $ch, array (
        CURLOPT_HTTPHEADER => array (
                'Content-Type: multipart/related; boundary="' . $boundary.'"' 
        ),
        CURLOPT_POST => true,
        CURLOPT_POSTFIELDS => $body,
        CURLOPT_URL => 'http://127.0.0.1:9999' 
) );
curl_exec ( $ch );
class MultipartRelatedFile {
    public $path;
    public $postname;
    public $content_type;
    function __construct(string $path) {
        if (! is_readable ( $path )) {
            throw new \Exception ( 'file is not readable!' );
        }
        $this->path = $path;
        $this->content_type = mime_content_type ( $path );
        $this->postname = basename ( $path );
    }
}
/**
 * generate http body for a multipart/related POST request
 *
 * @param MultipartRelatedFile[] $files
 * @param string|null $boundary
 */
function generateMultipartRelatedBody(array $files, string &$boundary = NULL): string {
    if (null === $boundary) {
        $boundary = bin2hex ( random_bytes ( 10 ) );
    }
    $n = "\r\n";
    $body = '';
    foreach ( $files as $file ) {
        $body .= $boundary . $n;
        $body .= 'Content-Disposition: attachment; filename=' . $file->postname . $n;
        $body .= "Content-Type: " . $file->content_type . $n;
        $body .= $n . file_get_contents ( $file->path ) . $n;
    }
    $body .= $boundary . '--';
    return $body;
}
警告:除了在netcat服务器中确保它看起来正确,这是未经测试的代码。 我还想指出规范没有提到任何Content-Length头,并暗示如果以其他方式提供文件名(如在Content-Disposition中)且未使用start头,则Content-Id是可选的。 无论如何,可以在此处找到规范:https://www.rfc-editor.org/rfc/rfc2112

编辑:忘记在初始Content-Type头的边界周围添加引号("),已修复。


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接