delphi - 额外的 HTTP header 会影响 MWS Feed API 提交吗?

标签 delphi amazon-web-services delphi-xe indy amazon-mws

调试 Amazon MWS Feed 提交是很麻烦的,因为无论您做错什么,Ama 都会返回相同的毫无意义的错误消息:

<Error>
  <Type>Sender</Type>
  <Code>InvalidParameterValue</Code>
  <Message>Either Action or Operation query parameter must be present.</Message>
</Error>

我 100% 确信我正确构建了 StringToSign 并计算了 HMAC-SHA256 等。我花了几周时间收集并采用了一堆有用的函数来进行哈希、签名、base64ing 等 MWS 请求。它们都是用纯 Pascal 编写的,并且都在订单和产品 API 上进行了测试。

现在,当谈到 Feeds API 时,我陷入了上述错误。所有参数均与MWS Scratchpad 生成的参数相同。我测试了 MWS Scratchpad 生成的提交 StringToSign,但没有成功。

到目前为止我注意到的是:MWS Scratchpad 和我的应用程序生成的 header 的数量/值之间存在差异。

便笺本生成以下标题(至少会显示它们):

  Host: mws.amazonservices.ca
  x-amazon-user-agent: AmazonJavascriptScratchpad/1.0 (Language=Javascript)
  Content-Type: text/xml

我的应用程序使用 Indy(在 XE4 中)TIdHTTP 发出请求。当亚马逊返回上述错误时,Request.RawHeaders.Text包含以下内容:

  Content-Length: 251
  x-amazon-user-agent: MyApp/1.1(Language=Delphi;Platform=Windows7)
  Content-Type: text/xml
  Host: mws.amazonservices.ca
  Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8
  Accept-Encoding: identity
  User-Agent: MyApp/1.1(Language=Delphi;Platform=Windows7)

看起来额外的 header 已添加到 Request默认对象。我的第一个问题是:这些额外的 header 理论上会影响行为吗? IE。他们能成为搅局者吗?

我的第二个问题:是否有任何选项可以修改 Indy HTTP.Request 中的默认标题列表?为了继续调试,我宁愿排除额外的 header 以查看请求是否有效。

更新:(签名字符串)

  AWSAccessKeyId=<AWSAccessKeyId>
  &Action=GetFeedSubmissionList
  &Merchant=<MerchantId>
  &SignatureMethod=HmacSHA256
  &SignatureVersion=2
  &Timestamp=2015-07-26T09%3A04%3A59Z
  &Version=2009-01-01
  &Signature=1OI0PVgL3uh5sFXxjCzaaWEwGmW6h5e0dgLUFkPgoXg%3D

更新:(完整的 HTTP 请求/响应由 TIdLogFile 提供)

  Stat Connected.
  Sent 28.07.2015 12:28:11:
    POST / HTTP/1.1<EOL>
    Content-Type: text/xml; charset=us-ascii<EOL>
    Content-Length: 279<EOL>
    x-amazon-user-agent: MyAppNameAndVer<EOL>
    Host: mws.amazonservices.ca<EOL>
    Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8<EOL>
    Accept-Encoding: identity<EOL>
    User-Agent: Mozilla/3.0 (compatible; Indy Library)<EOL><EOL>

  Sent 28.07.2015 12:28:11: 
    AWSAccessKeyId=<AWSAccessKeyId>
    &Action=GetFeedSubmissionList
    &MWSAuthToken=<AWSAuthToken>
    &Merchant=<MerchantId>
    &SignatureMethod=HmacSHA256
    &SignatureVersion=2
    &Timestamp=2015-07-28T10%3A28%3A09Z
    &Version=2009-01-01
    &Signature=I6euLIiVDzjZ8bbdtF840K0TJCkGh4NrUbQPtQtu78A%3D

  Recv 28.07.2015 12:28:11:
    HTTP/1.1 400 Bad Request<EOL>
    Date: Tue, 28 Jul 2015 10:28:10 GMT<EOL>
    Server: AmazonMWS<EOL>
    x-mws-request-id: 7e63280d-1db5-4d10-af40-587747d032a8<EOL>
    x-mws-timestamp: 2015-07-28T10:28:11.048Z<EOL>
    x-mws-response-context: OmAlguEmW20QT07uIdb9d25xkX+JSS7uFr1rDvXIoqXMvbUFzUMt1b5Xl2WzDaJszbwr25N/J4c=<EOL>
    Content-Type: text/xml<EOL>
    Content-Length: 324<EOL>
    Vary: User-Agent<EOL>
    nnCoection: close<EOL><EOL>
    <?xml version="1.0"?><LF><ErrorResponse xmlns="https://mws.amazonservices.com/"><LF>  <Error><LF>    <Type>Sender</Type><LF>    <Code>InvalidParameterValue</Code><LF>    <Message>Either Action or Operation query parameter must be present.</Message><LF>  </Error><LF>  <RequestID>7e63280d-1db5-4d10-af40-587747d032a8</RequestID><LF></ErrorResponse><LF>
  Stat Disconnected.

请注意:为了您的方便,我对响应进行了格式化,在 <EOL> 之后添加了 CR

令人困惑nnCoection header 解释如下:Cneonction and nnCoection HTTP headers


我的问题现在已经解决了。以下是我感谢那些没有违反社区标准的人的方式;-)

Days := 1;
repeat
  IsSuccessful := DebugAmazonFeedSubmissionProc;
  Inc(Days);
until IsSuccessful or (Days = 14);

if not IsSuccessful then
begin
  AskTheGods(@StackOverflow);
  Sleep(1000); // ...wake up and drink a cup'o'tea :)
  Expert := TRemyLebeau.Create('delphi');
  try
    Expert.ReviewTheCode;
    Expert.GetTheJobDone;
  finally
    // You may dispose the instance here,
    // but the Class is one of the most valuable assets of the Community.
    // Thank you, Remy!
  end;
end;

最佳答案

您的请求 Content-Type header 错误。您将其设置为 text/xml; charset=us-ascii,但您实际上并没有开始发送 XML 数据。您需要将 Content-Type(通过 TIdHTTP.Request.ContentType 属性)设置为 application/x-www-form-urlencoded ,根据 MWS GetFeedSubmissionList文档。

TIdHTTP.Post()TStrings 版本会为您设置 ContentType,但您正在使用 TStream 版本,因此您必须手动设置 ContentType。在这种情况下,我确实建议您切换到 TStrings 版本,并让它为您处理表单字段的编码:

var
  PostData: TStringList;
begin
  PostData := TStringList.Create;
  try
    PostData.Add('AWSAccessKeyId=<AWSAccessKeyId>');
    PostData.Add('Action=GetFeedSubmissionList');
    PostData.Add('MWSAuthToken=<AWSAuthToken>');
    PostData.Add('Merchant=<MerchantId>');
    PostData.Add('SignatureMethod=HmacSHA256');
    PostData.Add('SignatureVersion=2');
    PostData.Add('Timestamp=2015-07-28T10:28:09Z'); // <-- NOT percent encoded yet!
    PostData.Add('Version=2009-01-01');
    PostData.Add('Signature=I6euLIiVDzjZ8bbdtF840K0TJCkGh4NrUbQPtQtu78A=');  // <-- NOT percent encoded yet!

    IdHTTP1.Request.CustomHeaders.Values['x-amazon-user-agent'] := 'MyAppNameAndVer';
    IdHTTP1.Post('http://mws.amazonservices.ca', PostData);
  finally
    PostData.Free;
  end;
end;

关于delphi - 额外的 HTTP header 会影响 MWS Feed API 提交吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31630075/

相关文章:

delphi - Delphi XE 中不再有 xercesxmldom 单元?

database - 在 Delphi 中反转主从连接

php - 如何使用 AWS PHP sdk 以 JSON 形式检索 Amazon S3 存储桶的内容?

amazon-web-services - Cloudformation - 将多个堆栈连接在一起

amazon-web-services - 如何卡住 SQS 队列以防止任何读取?

delphi - 如何在应用程序的主窗体可见后立即出现对话框?

delphi - 在 Delphi XE2 中同时调试多个应用程序

delphi - TServerSocket : Confusion with Socket Objects

c++ - 如何在 Windows (win32) 上获取每个线程的 cpu 使用率

delphi - 为什么有时 IDE 将类方法或字段标记为错误(红色下划线)