PHP curl 更改 Content-Type 为 application/x-www-form-urlencoded,这样做是否合适?

9
我可以帮您进行翻译。下面是翻译的结果:

我想从我的服务器直接上传视频到Youtube,我正在使用PHP curl。

我需要的请求格式如下:

POST /feeds/api/users/default/uploads HTTP/1.1
Host: uploads.gdata.youtube.com
Authorization: Bearer ACCESS_TOKEN
GData-Version: 2
X-GData-Key: key=adf15ee97731bca89da876c...a8dc
Slug: video-test.mp4
Content-Type: multipart/related; boundary="f93dcbA3"
Content-Length: 1941255
Connection: close

--f93dcbA3
Content-Type: application/atom+xml; charset=UTF-8

<?xml version="1.0"?>
<entry xmlns="http://www.w3.org/2005/Atom"
  xmlns:media="http://search.yahoo.com/mrss/"
  xmlns:yt="http://gdata.youtube.com/schemas/2007">
  <media:group>
    <media:title type="plain">Bad Wedding Toast</media:title>
    <media:description type="plain">
      I gave a bad toast at my friend's wedding.
    </media:description>
    <media:category
      scheme="http://gdata.youtube.com/schemas/2007/categories.cat">People
    </media:category>
    <media:keywords>toast, wedding</media:keywords>
  </media:group>
</entry>
--f93dcbA3
Content-Type: video/mp4
Content-Transfer-Encoding: binary

<Binary File Data>
--f93dcbA3--

这是我拥有的内容:
$content = $this->buildRequestContent();

$ch = curl_init();

$curlConfig = array(
    CURLOPT_URL => 'http://uploads.gdata.youtube.com/feeds/api/users/default/uploads',
    CURLOPT_POST => true,
    CURLOPT_RETURNTRANSFER => true,
    CURLOPT_BINARYTRANSFER => true,
    CURLOPT_POSTFIELDS => $content,
    CURLOPT_HTTPHEADER, array(
        "Authorization" => sprintf("GoogleLogin auth=%s", $this->accessToken),
        "GData-Version" => 2,
        "X-GData-Key" => sprintf("key=%s", $this->developerKey),
        "Slug" => sprintf("%s", $this->video->getFilename()),
        "Content-Type" => sprintf("multipart/related; boundary=\"%s\"", $this->boundaryString),
        "Content-Length" => strlen($content),
        "Connection" => "close"
    ),
);

curl_setopt_array($ch, $curlConfig);

$result = curl_exec($ch);

转储结果显示,curl将Content-Type更改为application/x-www-form-urlencoded,而Youtube当然不支持它。我将二进制内容(视频)放入CURLOPT_POSTFIELDS中,也许这是错误的,但我不知道如何设置请求体。那么,我该如何保留我设置的Content-Type呢?

1
根据文档,CURLOPT_POST 会自动将 Content-Type 设置为 application/x-www-form-urlencoded。尝试设置此选项和 CURLOPT_HEADER,可以使用单个语句,或者至少在设置方法后使用 curl_setopt 将后者设置为单个语句。 - Havelock
1
工作了。你能把这个发表为答案,这样我就可以接受它了吗? - Johannes Klauß
2个回答

14

接受的答案对我没有帮助,我在您的代码中发现了另一个问题。根据当前的PHP 文档,CURLOPT_HTTPHEADER必须像这样设置:

curl_setopt($ch, CURLOPT_HTTPHEADER, array(
    'Content-type: text/plain',
    'Content-length: 100'
));

必须是字符串列表,而不是哈希表,例如array('Content-Type' => 'text/plain')。 希望这能节省某些人的调试时间。

2
换句话说,您不能将关联数组传递给CURLOPT_HTTPHEADER,只能传递简单的数组。这个问题让我疯了好几个小时... - andreszs

8
根据有关选项参数的文档部分,将HTTP方法设置为POST会默认将内容类型设置为application/x-www-form-urlencoded。因此,我怀疑一次性设置所有选项会导致您的Content-Type被覆盖。
我的建议是,在设置方法后,使用单个语句设置内容类型。
$curlConfig = array(
    CURLOPT_URL => 'http://uploads.gdata.youtube.com/feeds/api/users/default/uploads',
    CURLOPT_POST => true,
    CURLOPT_RETURNTRANSFER => true,
    CURLOPT_BINARYTRANSFER => true,
    CURLOPT_POSTFIELDS => $content,
);

curl_setopt_array($ch, $curlConfig);

curl_setopt($ch, CURLOPT_HTTPHEADER, array(
    'Authorization: ' . sprintf('GoogleLogin auth=%s', $this->accessToken),
    'GData-Version: 2',
    'X-GData-Key: ' . sprintf('key=%s', $this->developerKey),
    'Slug: ' . sprintf('%s', $this->video->getFilename()),
    'Content-Type: ' . sprintf('multipart/related; boundary="%s"',
                                  $this->boundaryString),
    'Content-Length: ' . strlen($content),
    'Connection: close'
));

请更新您的示例,它不是一个二维数组。应该是:array("Authorization: " . sprintf...,"Content-Type: " . sprintf...)。 - John Congdon
@JohnCongdon:没错,谢谢!有趣的是,在两年多的时间里,只有三个赞和“被接受的答案”,没有人注意到这一点。(我相信你的意思是“不是关联数组”,而不是“不是二维数组”;)) - Havelock
PHP cURL 真是令人遗憾。不仅 CURLOPT_POST 覆盖了 Content-Type,而且还阻止了其他 HTTP 标头同时设置。 感谢您提供的解决方案! - NeverEndingQueue
我无法让这个解决方案正常工作。我已经将标题的设置与其他选项分开,但它仍然添加了额外的内容类型。 - dmarra
@dmarra 我建议您使用专门的客户端库,例如 GuzzleHttp。 - Havelock

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