python - 使用 Python 使用 Binance API 进行交易的问题
问题描述
我正在尝试在不使用外部库的情况下对美国版 Binance API 进行交易。
我可以使用 GET 请求和urllib
. 第一个示例代码有效,我可以毫无问题地传递我的值(这些值是私有的,它们不会显示在这里并且位于API_KEY
)。SECRET_KEY
settings.py
进行交易需要 POST,我不确定我哪里出错了,我的 POST 请求不起作用,但 GET 请求可以正常工作。根据我对发出 POST 请求的文档的理解,我应该使用对参数进行编码 并将urllib.parse.urlencode()
其传递给.data
urllib.request.Request()
这样做不会引发错误,但是当我尝试打开请求时urllib.request.urlopen()
出现错误:
Traceback (most recent call last):
File "C:\Users\user\PycharmProjects\test\test.py", line 80, in <module> place_trade(symbol='BTCUSD', side='BUY', order_type='MARKET', quantity=1)
File "C:\Users\user\PycharmProjects\test\test.py", line 73, in place_trade response = urllib.request.urlopen(req)
File "C:\Users\user\AppData\Local\Programs\Python\Python39\lib\urllib\request.py", line 214, in urlopen return opener.open(url, data, timeout)
File "C:\Users\user\AppData\Local\Programs\Python\Python39\lib\urllib\request.py", line 523, in open response = meth(req, response)
File "C:\Users\user\AppData\Local\Programs\Python\Python39\lib\urllib\request.py", line 632, in http_response response = self.parent.error(
File "C:\Users\user\AppData\Local\Programs\Python\Python39\lib\urllib\request.py", line 561, in error return self._call_chain(*args)
File "C:\Users\user\AppData\Local\Programs\Python\Python39\lib\urllib\request.py", line 494, in _call_chain result = func(*args)
File "C:\Users\user\AppData\Local\Programs\Python\Python39\lib\urllib\request.py", line 641, in http_error_default raise HTTPError(req.full_url, code, msg, hdrs, fp)
urllib.error.HTTPError: HTTP Error 500: Internal Server Error
HTTP 返回代码状态:
HTTP 5XX 返回码用于内部错误;问题在币安这边。重要的是不要将此视为失败操作;执行状态为 UNKNOWN,可能已经成功。
但是我不认为是这样,因为我可以get_account_balance()
毫无问题地运行其他代码的功能。我不确定我做错了什么,除了发出 GET 和 POST 请求外,两个代码几乎相同。
获取帐户余额的代码 - 工作正常:
import json
import time
import hmac
import settings
import hashlib
import urllib.parse
import urllib.request
def get_account_balance():
# Setup header with API_KEY
headers = {'X-MBX-APIKEY': settings.API_KEY}
# Params requires timestamp in MS
params = {'timestamp': int(time.time() * 1000)}
# Encode params into url
url = 'https://api.binance.us/api/v3/account?' + urllib.parse.urlencode(params)
# Create a HMAC SHA256 signature
secret = bytes(settings.SECRET_KEY.encode('utf-8'))
signature = hmac.new(secret, urllib.parse.urlencode(params).encode('utf-8'), hashlib.sha256).hexdigest()
# Add signature to url
url += f'&signature={signature}'
# Make a request
req = urllib.request.Request(url, headers=headers)
# Read and decode response
response = urllib.request.urlopen(req).read().decode('utf-8')
# Convert to json
response_json = json.loads(response)
# Print balances for all coins not at 0
for entry in response_json['balances']:
if entry['free'] == '0.00000000':
continue
print(entry)
get_account_balance()
进行交易的代码 - 不起作用:
import json
import time
import hmac
import settings
import hashlib
import urllib.parse
import urllib.request
def place_trade(symbol, side, order_type, quantity):
# Setup header with API_KEY
headers = {'X-MBX-APIKEY': settings.API_KEY}
# Params require symbol, side, type, quantity and timestamp (for market orders)
params = {
'symbol': symbol,
'side': side,
'type': order_type,
'quantity': quantity,
'timestamp': int(time.time() * 1000)
}
# Encode params into url
url = 'https://api.binance.us/api/v3/order/test?' + urllib.parse.urlencode(params)
# Create a HMAC SHA256 signature
secret = bytes(settings.SECRET_KEY.encode('utf-8'))
signature = hmac.new(secret, urllib.parse.urlencode(params).encode('utf-8'), hashlib.sha256).hexdigest()
# Add signature to url
url += f'&signature={signature}'
# Encode params
data = urllib.parse.urlencode(params).encode('ascii')
# Make a POST request
req = urllib.request.Request(url, data, headers)
# Open request and convert to string and then to json
response = urllib.request.urlopen(req) # <- line with error
response_str = response.read().decode('utf-8')
response_json = json.loads(response_str)
print(response_json)
place_trade(symbol='BTCUSD', side='BUY', order_type='MARKET', quantity=1)
参考
示例中使用了此端点,但两个端点的功能相同并且具有相同的错误
我还查看了库python-binance
中的示例
编辑
我可以使用requests
图书馆成功订购。我浏览了库的源代码,但无法弄清楚如何使用正确格式化 POST 请求urllib
使用requests
库进行交易的代码 - 作品:
import hmac
import time
import hashlib
import requests
import settings
import urllib.parse
session = requests.session()
session.headers.update({'X-MBX-APIKEY': settings.API_KEY})
url = 'https://api.binance.us/api/v3/order/test'
params = {
'symbol': 'BTCUSD',
'side': 'BUY',
'type': 'MARKET',
'quantity': 1,
'timestamp': int(time.time() * 1000)
}
secret = bytes(settings.SECRET_KEY.encode('utf-8'))
signature = hmac.new(secret, urllib.parse.urlencode(params).encode('utf-8'), hashlib.sha256).hexdigest()
params['signature'] = signature
result = session.post(url, params)
print(result)
print(result.text)
解决方案
问题出在以下几行:
# Encode params into url
url = 'https://api.binance.us/api/v3/order/test?' + urllib.parse.urlencode(params)
# Add signature to url
url += f'&signature={signature}'
显然,当使用urllib
GET 请求时,必须将有效负载编码到 url 本身中,但是 POST 请求需要您将它们传递到data
参数中:
data = urllib.parse.urlencode(params).encode('ascii')
req = urllib.request.Request(url, data=data, headers=headers)
在我的问题中,我通过 url 和data
参数传递了我的有效负载。删除 url 有效负载可解决此问题。对于遇到此问题的任何人的旁注,?
在 url 后放置一个问号对于 POST 请求是可选的,但在使用时不是 GET 请求urllib
在没有外部库的情况下发布交易的工作代码:
import json
import time
import hmac
import hashlib
import settings
import urllib.parse
import urllib.request
params = {
'symbol': 'BTCUSD',
'side': 'BUY',
'type': 'MARKET',
'quantity': 1,
'timestamp': int(time.time() * 1000)
}
secret = bytes(settings.SECRET_KEY.encode('utf-8'))
signature = hmac.new(secret, urllib.parse.urlencode(params).encode('utf-8'), hashlib.sha256).hexdigest()
params['signature'] = signature
url = 'https://api.binance.us/api/v3/order/test'
headers = {'X-MBX-APIKEY': settings.API_KEY}
data = urllib.parse.urlencode(params).encode('ascii')
req = urllib.request.Request(url, data=data, headers=headers)
response = urllib.request.urlopen(req)
response_str = response.read().decode('utf-8')
response_json = json.loads(response_str)
print(response_json)
推荐阅读
- python - NumPy 在 ArcMap 中将绕组而不是数字写入 CSV
- jquery - JQuery在选择更改事件上添加类
- python - Keras 时间序列我可以一次预测下一个 6 个月吗
- email - 在 Outlook VSTO 的活动资源管理器窗格中显示邮件列表
- python - 无法正确获取 twitter 抓取中的 min_position
- python - 带有管道和 GridSearchCV 的 StandardScaler
- blazor - 如何从其父组件调用组件公共方法?
- java - SQL Server 与 jdbc eclipse 的连接
- javascript - 在移动设备的滑动上创建简单的纯 Javascript 滑块
- spring-security-saml2 - 当 application.yml 只有一个 IdP 时,如何避免“选择身份提供者”页面?