{error={name=HTTPError, message=Signature not valid.}} Bitmex Api 连接错误

{error={name=HTTPError, message=Signature not valid.}} Bitmex Api connection error

我一直在尽力解释 bitmex's indications for api connection,但这就是我得到的全部:

function balanceBitmex() {

  var apiKey = '***';
  var apiSecret = '***';

  var verb = 'get';
  var path = '/api/v1/user/wallet';

  var expires = Number(new Date().getTime() +5).toFixed(0);
  var expires = expires.toString();


  var data = '';
  var data = JSON.stringify(data);
  var currency = '';
  var count= '';
  var concat = verb+path+expires+data;


  var apiSignature = Utilities.computeHmacSha256Signature(apiSecret, concat);
  apiSignature = apiSignature.map(function(e) {
      var v = (e < 0 ? e + 256 : e).toString(16);
      return v.length == 1 ? "0" + v : v;
  }).join("");


  var params = {
    'method': 'get',
    'headers': {'contentType': 'application/json',
                'api-expires': expires,
                'api-key': apiKey,
                'api-signature': apiSignature},
    'muteHttpExceptions': true
  };

  //var params = params + data;

  var url = "https://www.bitmex.com/api/v1/user/wallet?currency=XBt";

  var data = UrlFetchApp.fetch(url, params);
  var data = JSON.parse(data.getContentText());

我的脚本有什么问题? bitmex api explorer - user wallet

函数Utilities.computeHmacSha256Signature(String, String)接受两个参数,顺序如下:

  1. value:为其生成散列的输入值。
  2. key:用于生成散列的密钥。

在您的情况下,您实际上是使用 concat 作为密钥来加密 apiSecret。然而,它应该是相反的——apiSecret 应该是用来加密 concat 的密钥。您只需更改:

var apiSignature = Utilities.computeHmacSha256Signature(apiSecret, concat);

对于以下内容:

var apiSignature = Utilities.computeHmacSha256Signature(concat, apiSecret);

此外,您可以使用以下代码计算BitMex documentation中提供的示例并验证其是否正常工作。

function testEncryption() {
  var apiSecret = 'chNOOS4KvNXR_Xq4k4c9qsfoKWvnDecLATCRlcBwyKDYnWgO';
  var verb = 'GET';
  var path = '/api/v1/instrument';
  var expires = 1518064236; 
  var data = '';

  var apiSignature = Utilities.computeHmacSha256Signature(verb + path + expires + data, apiSecret);
  apiSignature = apiSignature.map(function(e) {
      var v = (e < 0 ? e + 256 : e).toString(16);
      return v.length == 1 ? "0" + v : v;
  }).join("");

  Logger.log(apiSignature);
}

上述代码 运行 的结果是 c7682d435d0cfe87c16098df34ef2eb5a549d4c5a3c2b1f0f77b8af73423bf00,与 BitMex 文档中显示的结果相同。

关于您提供的代码,为了使 API 正常工作,还有一些错误需要解决:

  1. 动词参数:您指定的参数是get,但API只接受此参数为大写(GET)。使用前者,签名会不同,从而导致请求失败。
  2. path querystring:在计算签名时必须存在查询字符串(在您的请求 currency=XBt 的情况下),但在您的代码中它不是(注意行 var path = '/api/v1/user/wallet';)

你的最终代码,在修复了上面的所有错误并清理了一些未使用的变量之后,可能如下所示:

function balanceBitmex() {
  var apiKey = '***'
  var apiSecret = '***';

  var verb = 'GET';

  var path = '/api/v1/user/wallet?currency=XBt';

  var expires = Math.floor((Date.now() / 1000) + 600).toFixed(0);

  var concat = verb + path + expires;

  var apiSignature = Utilities.computeHmacSha256Signature(concat, apiSecret);
  apiSignature = apiSignature.map(function(e) {
      var v = (e < 0 ? e + 256 : e).toString(16);
      return v.length == 1 ? "0" + v : v;
  }).join("");

  var params = {
    'headers': {
      'api-expires': expires,
      'api-key': apiKey,
      'api-signature': apiSignature
    },
    'muteHttpExceptions': true
  };

  var url = "https://www.bitmex.com/api/v1/user/wallet?currency=XBt";

  var response = UrlFetchApp.fetch(url, params);
  var data = JSON.parse(response.getContentText());

  Logger.log(data);
}