tags:

views:

694

answers:

3

All,

I have a class function to interface with the RESTful API for Last.FM - its purpose is to grab the most recent tracks for my user. Here it is:

private static $base_url = 'http://ws.audioscrobbler.com/2.0/';

public static function getTopTracks($options = array())
{
  $options = array_merge(array(
    'user' => 'bachya',
    'period' => NULL,
    'api_key' => 'xxxxx...', // obfuscated, obviously
  ), $options);

  $options['method'] = 'user.getTopTracks';

  // Initialize cURL request and set parameters
  $ch = curl_init();
  curl_setopt_array($ch, array(
    CURLOPT_URL            => self::$base_url,
    CURLOPT_POST           => TRUE,
    CURLOPT_POSTFIELDS     => $options,
    CURLOPT_RETURNTRANSFER => TRUE,
    CURLOPT_TIMEOUT        => 30,
    CURLOPT_USERAGENT      => 'Mozilla/4.0 (compatible; MSIE 5.01; Windows NT 5.0)'
  ));

  $results = curl_exec($ch);
  return $results;
}

This returns "Empty reply from server". I know that some have suggested that this error comes from some fault in network infrastructure; I do not believe this to be true in my case. If I run a cURL request through the command line, I get my data; the Last.FM service is up and accessible.

Before I go to those folks and see if anything has changed, I wanted to check with you fine folks and see if there's some issue in my code that would be causing this.

Thanks!

ANSWER: @Jan Kuboschek helped me stumble onto what is (maybe) going on here. By giving CURLOPT_POSTFIELDS an associative array, a particular content-type is specified that may not work with certain RESTful services. A smarter solution is to manually create a URL-encoded version of that data and pass that as the CURLOPT_POSTFIELDS.

For more info, check out: http://www.brandonchecketts.com/archives/array-versus-string-in-curlopt_postfields

+4  A: 

A common issue are spaces in the URL - beginning, in the middle, or trailing. Did you check that out?

Edit - per comments below, spacing is not the issue.

I ran your code and had the same problem - no output whatsoever. I tried the URL and with a GET request, the server talks to me. I would do the following:

  1. Use the following as $base_url: $base_url = 'http://ws.audioscrobbler.com/2.0/?user=bachya&period=&api_key=xxx&method=user.getTopTracks';

  2. Remove the post fields from your request.

Edit I moved your code out of the class since I didn't have the rest and modified it. The following code runs perfect for me. If these changes don't work for you, I suggest that your error is in a different function.

<?php


function getTopTracks()
{
  $base_url = 'http://ws.audioscrobbler.com/2.0/?user=bachya&amp;period=&amp;api_key=8066d2ebfbf1e1a8d1c32c84cf65c91c&amp;method=user.getTopTracks';
  $options = array_merge(array(
    'user' => 'bachya',
    'period' => NULL,
    'api_key' => 'xxxxx...', // obfuscated, obviously
  ));

  $options['method'] = 'user.getTopTracks';

  // Initialize cURL request and set parameters
  $ch = curl_init($base_url);
  curl_setopt_array($ch, array(
    CURLOPT_URL            => $base_url,
    CURLOPT_RETURNTRANSFER => TRUE,
    CURLOPT_TIMEOUT        => 30,
    CURLOPT_USERAGENT      => 'Mozilla/4.0 (compatible; MSIE 5.01; Windows NT 5.0)'
  ));

  $results = curl_exec($ch);
  return $results;
}

echo getTopTracks();

?>
Jan Kuboschek
Unfortunately, that's not the problem - I updated my original post to show the URL ($base_url) that I use. Thanks, though. :-\
ABach
Updated my answer. Let me know if this still doesn't work. I couldn't test further since the server told me that I needed a valid API key.
Jan Kuboschek
Okay, tried that - it seems to hang ("work") a little longer. I var dump the curl_exec results and get `false` in return... I'd rather not put out my API key, but they're easy enough to get.
ABach
Ok, this is bugging me now. How do I get an API key?
Jan Kuboschek
Check here: http://www.last.fm/api/account - thanks for being as bugged as I am! :)
ABach
Take a look at the update. It's working for me now.
Jan Kuboschek
Okay - you hit on something that I've updated my original post with. Thank you for your help! :)
ABach
You're welcome!
Jan Kuboschek
Good work! -----
Pekka
A: 

The server received your request, but sent an empty response. Check the result of curl_getinfo($ch, CURLINFO_HTTP_CODE) to find out if the server responded with an HTTP error code.

Update: Ok so the server responds with the 100 Continue HTTP status code. In that case, this should solve your problem:

curl_setopt($ch, CURLOPT_HTTPHEADER, array('Expect:'));

I found this here: PHP and cURL: Disabling 100-continue header. Hope it works!

cygri
Var-dumping this returns `100` - helpful, I hope?
ABach
Yes—I updated my answer.
cygri
One more update: Jan's solution above uses GET instead of POST, and as Michał pointed out that's actually better for read API calls. The `100 Continue` stuff doesn't apply to GET, so it neatly avoids the real problem. If you want to access write services as well, then you should use POST for those calls, and try my answer.
cygri
A: 

According to Last.FM API documentation you should use GET method instead of POST to pass parameters. When I've changed POST to GET I've received the answer about incorrect key.

Michał Pipa
I don't see anything in that document that specifies the required use of a GET request in passing parameters. Could you show me where?
ABach
Michał Pipa