在Jquery中使用Last.fm进行身份验证 - 提供的方法签名无效

时间:2015-05-25 00:34:34

标签: javascript jquery api md5 last.fm

我正在尝试授权Last.fm会话,并且正在努力正确签署会话密钥请求。

我一直在接收Invalid method signature supplied但是当我md5哈希我认为查询应该包含在JS之外时,我得到相同的签名。我必须在字符串中包含错误的数据,但我无法弄清楚是什么。

我知道还有其他一些问题,我已经看过所有问题,看看这里出了什么问题,但我发誓看起来对我来说是正确的。

这是签名算法和Ajax调用。我也试图留下足够的样本数据。

// Set elsewhere but hacked into this example:
var last_fm_data = {
    'last_token':'TOKEN876234876', 
    'user': 'bob',
    'secret': 'SECRET348264386'
};

// Kick it off.
last_fm_call('auth.getSession', {'token': last_fm_data['last_token']});


// Low level API call, purely builds a POSTable object and calls it. 
function last_fm_call(method, data){  
    // param data - dictionary.

    last_fm_data[method] = false; 
    // Somewhere to put the result after callback.

    // Append some static variables
    data['api_key'] = "APIKEY1323454";
    data['format'] = 'json';
    data['method'] = method;
    post_data = last_fm_sign(data);

    $.ajax({
      type: "post",
      url: last_url,
      data: post_data,
      success: function(res){
          last_fm_data[method] = res;
          console.log(res['key'])// Should return session key.
      },
      dataType: 'json'
     });
}

function last_fm_sign(params){
    ss = "";
    st = [];
    so = {};
    Object.keys(params).forEach(function(key){
        st.push(key); // Get list of object keys
    });
    st.sort(); // Alphabetise it 
    st.forEach(function(std){
        ss = ss + std + params[std]; // build string
        so[std] = params[std];  // return object in exact same order JIC
    });    
        // console.log(ss + last_fm_data['secret']);
        // api_keyAPIKEY1323454formatjsonmethodauth.getSessiontokenTOKEN876234876SECRET348264386
    hashed_sec = unescape(encodeURIComponent($.md5(ss + last_fm_data['secret'])));
    so['signature'] = hashed_sec; // Correct when calculated elsewhere.
    return so; // Returns signed POSTable object
}

任何人都可以看到我在这里失踪了吗?我完全不知道为什么这不能以请求的格式返回正确签名的POSTable对象here。谢谢你的时间。

编辑:如果我没有得到任何建议,我们不能感谢任何人的时间!没有人有过last.fm的经验吗?

2 个答案:

答案 0 :(得分:3)

在调查了与last.fm api电话相关的代码和其他帖子之后,我发现@george lee实际上是正确的。在生成format时,您不需要提供auth_sign

除此之外,您需要在应用$.md5()auth_sign函数后将encodeURIComponent()应用于unescape()字符串。像这样。

hashed_sec = $.md5(unescape(encodeURIComponent(ss + last_fm_data['secret'])));

在拨打ajax电话时,您需要将api_key, token & api_sig作为data传递。但是看到你的代码,就会发现你正在传递api_key, token, format, method & signature

因此,您需要从format, method & signature来电的data字段中删除ajax

相反,您需要将api_key, token & api_sig传递给data字段。

因此评论data['format'] = 'json';行后的最终代码将如下所示。

    // Set elsewhere but hacked into this example:
    var last_fm_data = {
        'last_token':'TOKEN876234876', 
        'user': 'bob',
        'secret': 'SECRET348264386'
    };

    // Kick it off.
    last_fm_call('auth.getSession', {'token': last_fm_data['last_token']});


    // Low level API call, purely builds a POSTable object and calls it. 
    function last_fm_call(method, data){  
        // param data - dictionary.
        last_fm_data[method] = false; 
        // Somewhere to put the result after callback.

        // Append some static variables
        data['api_key'] = "APIKEY1323454";
        //data['format'] = 'json';
        data['method'] = method;

        post_data = last_fm_sign(data);

        $.ajax({
          type: "POST",
          url: last_url,
          data: post_data,
          success: function(res){
              last_fm_data[method] = res;
              console.log(res['key'])// Should return session key.
          },
          dataType: 'json'
         });
    }

    function last_fm_sign(params){
        ss = "";
        st = [];
        so = {};
        so['api_key'] = params['api_key'];
        so['token'] = params['token'];
        Object.keys(params).forEach(function(key){
            st.push(key); // Get list of object keys
        });
        st.sort(); // Alphabetise it 
        st.forEach(function(std){
            ss = ss + std + params[std]; // build string
        });
        ss += last_fm_data['secret'];
            // console.log(ss + last_fm_data['secret']);
            // api_keyAPIKEY1323454formatjsonmethodauth.getSessiontokenTOKEN876234876SECRET348264386
        hashed_sec = $.md5(unescape(encodeURIComponent(ss)));
        so['api_sig'] = hashed_sec; // Correct when calculated elsewhere.
        return so; // Returns signed POSTable object
    }

Please refer to this link.

答案 1 :(得分:1)

所以在测试一些响应时,我找到了解决方案。有2个问题。 编辑见下文( 第一个是需要删除

data['format'] = 'json';

正如George Lee所指出的那样。谢谢乔治。 )

另一个问题是我错误地命名了一个变量,因此使用错误的名称进行了POST。这条线

so['signature'] = hashed_sec;

应该是

so['api_sig'] = hashed_sec;

我在Pankaj的回答中注意到了这一点,但不幸的是,他的其余答案(包括方法)是不正确的。进行这两项更改解决了呼叫并正确签名。

感谢所有的建议!

编辑: 再多玩一次后,我发现了

data['format'] = 'json';

是正确的,但它不会随签名进行哈希处理。 在散列工作之后将data['format'] = 'json';添加到POST对象,并且在此实例中将返回JSON而不是XML - 这是首选方法。在我找到的任何地方都没有记录散列后添加,所以你去了。 新的工作代码如下,这显示了用--------------------

表示的2行
// Set elsewhere but hacked into this example:
var last_fm_data = {
    'last_token':'TOKEN876234876', 
    'user': 'bob',
    'secret': 'SECRET348264386'
};

// Kick it off.
last_fm_call('auth.getSession', {'token': last_fm_data['last_token']});


// Low level API call, purely builds a POSTable object and calls it. 
function last_fm_call(method, data){  
    // param data - dictionary.

    last_fm_data[method] = false; 
    // Somewhere to put the result after callback.

    // Append some static variables
    data['api_key'] = "APIKEY1323454";
    data['method'] = method;
    post_data = last_fm_sign(data);
    // THEN ADD THE FORMAT ---------------------------------------
    post_data['format'] = 'json';
    $.ajax({
      type: "post",
      url: last_url,
      data: post_data,
      success: function(res){
          last_fm_data[method] = res;
          console.log(res['key'])// Should return session key.
      },
      dataType: 'json'
     });
}

function last_fm_sign(params){
    ss = "";
    st = [];
    so = {};
    Object.keys(params).forEach(function(key){
        st.push(key); // Get list of object keys
    });
    st.sort(); // Alphabetise it 
    st.forEach(function(std){
        ss = ss + std + params[std]; // build string
        so[std] = params[std];  // return object in exact same order JIC
    });    
        // console.log(ss + last_fm_data['secret']);
        // api_keyAPIKEY1323454formatjsonmethodauth.getSessiontokenTOKEN876234876SECRET348264386
    hashed_sec = unescape(encodeURIComponent($.md5(ss + last_fm_data['secret'])));
    so['api_sig'] = hashed_sec; // RENAMED THIS ----------------------------
    return so; // Returns signed POSTable object
}