|
1
|
|
|
from unittest import TestCase |
|
2
|
|
|
from mock import patch |
|
3
|
|
|
import time |
|
4
|
|
|
import hashlib |
|
5
|
|
|
import hmac |
|
6
|
|
|
import requests |
|
7
|
|
|
import requests_mock |
|
8
|
|
|
import btcde |
|
9
|
|
|
|
|
10
|
|
|
@requests_mock.Mocker() |
|
11
|
|
|
class TestBtcdeAPIDocu(TestCase): |
|
12
|
|
|
'''Tests are as in bitcoin.de API documentation. |
|
13
|
|
|
https://www.bitcoin.de/de/api/tapi/v2/docu''' |
|
14
|
|
|
|
|
15
|
|
|
def sortParams(self, url, params={}): |
|
16
|
|
|
'''To sort params for url string.''' |
|
17
|
|
|
self.encoded_string = '' |
|
18
|
|
|
if len(params) > 0: |
|
19
|
|
|
for key, value in sorted(params.items()): |
|
20
|
|
|
self.encoded_string += str(key) + '=' + str(value) + '&' |
|
21
|
|
|
self.encoded_string = self.encoded_string[:-1] |
|
22
|
|
|
self.url = url + '?' + self.encoded_string |
|
23
|
|
|
else: |
|
24
|
|
|
self.url = url |
|
25
|
|
|
return self.url, self.encoded_string |
|
26
|
|
|
|
|
27
|
|
|
def verifySignature(self, url, method, params): |
|
28
|
|
|
'''To verify API Signature.''' |
|
29
|
|
|
self.XAPINONCE += 1 |
|
30
|
|
|
self.url, self.encoded_string = self.sortParams(url, params) |
|
31
|
|
|
if method == 'POST': |
|
32
|
|
|
md5_encoded_query_string = hashlib.md5(self.encoded_string.encode()).hexdigest() |
|
33
|
|
|
else: |
|
34
|
|
|
md5_encoded_query_string = hashlib.md5(b'').hexdigest() |
|
35
|
|
|
hmac_data = method + '#' + \ |
|
36
|
|
|
self.url + '#' + self.XAPIKEY + \ |
|
37
|
|
|
'#' + str(self.XAPINONCE) + '#' + md5_encoded_query_string |
|
38
|
|
|
hmac_signed = hmac.new(bytearray(self.XAPISECRET.encode()), msg=hmac_data.encode(), digestmod=hashlib.sha256).hexdigest() |
|
39
|
|
|
return hmac_signed |
|
40
|
|
|
|
|
41
|
|
|
def setUp(self): |
|
42
|
|
|
self.XAPIKEY = 'f00b4r' |
|
43
|
|
|
self.XAPISECRET = 'b4rf00' |
|
44
|
|
|
self.XAPINONCE = int(time.time()) |
|
45
|
|
|
btcde.nonce = self.XAPINONCE |
|
46
|
|
|
self.conn = btcde.Connection(self.XAPIKEY, self.XAPISECRET) |
|
47
|
|
|
|
|
48
|
|
|
def tearDown(self): |
|
49
|
|
|
del self.XAPIKEY |
|
50
|
|
|
del self.XAPISECRET |
|
51
|
|
|
del self.XAPINONCE |
|
52
|
|
|
del self.conn |
|
53
|
|
|
|
|
54
|
|
|
def test_signature_post(self, m): |
|
55
|
|
|
'''Test the signature on a post request.''' |
|
56
|
|
|
params = {'type': 'buy', |
|
57
|
|
|
'trading_pair': 'btceur', |
|
58
|
|
|
'max_amount': 10, |
|
59
|
|
|
'price': 1337} |
|
60
|
|
|
response = {"order_id": "A1234BC", |
|
61
|
|
|
"errors": [], |
|
62
|
|
|
"credits": 8,} |
|
63
|
|
|
m.post(requests_mock.ANY, json=response, status_code=201) |
|
64
|
|
|
btcde.createOrder(self.conn, params.get('type'), params.get('trading_pair'), params.get('max_amount'), params.get('price')) |
|
65
|
|
|
history = m.request_history |
|
66
|
|
|
request_signature = history[0].headers.get('X-API-SIGNATURE') |
|
67
|
|
|
verified_signature = self.verifySignature(btcde.orderuri, 'POST', params) |
|
68
|
|
|
self.assertEqual(request_signature, verified_signature) |
|
69
|
|
|
|
|
70
|
|
|
def test_signature_get(self, m): |
|
71
|
|
|
'''Test the signature on a get request.''' |
|
72
|
|
|
params = {'type': 'buy', |
|
73
|
|
|
'trading_pair': 'btceur',} |
|
74
|
|
|
response = {"orders":{"order_id": "A1B2D3", |
|
75
|
|
|
"trading_pair": "btceur", "type": "buy", |
|
76
|
|
|
"max_amount":0.5, "min_amount":0.1, |
|
77
|
|
|
"price":230.55, "max_volume":115.28, "min_volume":23.06, |
|
78
|
|
|
"order_requirements_fullfilled":True, |
|
79
|
|
|
"trading_partner_information":{ |
|
80
|
|
|
"username":"bla", "is_kyc_full":True, |
|
81
|
|
|
"trust_level":"gold", "bank_name":"Sparkasse", |
|
82
|
|
|
"bic":"HASPDEHHXXX", "seat_of_bank":"DE", "rating": 99, |
|
83
|
|
|
"amount_trades": 52}, |
|
84
|
|
|
"order_requirements":{"min_trust_level":"gold", |
|
85
|
|
|
"only_kyc_full":True, "seat_of_bank":["DE", "NL"],"payment_option":1, } |
|
86
|
|
|
}, "errors":[], |
|
87
|
|
|
"credits":12 } |
|
88
|
|
|
m.get(requests_mock.ANY, json=response, status_code=200) |
|
89
|
|
|
btcde.showOrderbook(self.conn, params.get('type'), params.get('trading_pair')) |
|
90
|
|
|
history = m.request_history |
|
91
|
|
|
request_signature = history[0].headers.get('X-API-SIGNATURE') |
|
92
|
|
|
verified_signature = self.verifySignature(btcde.orderuri, 'GET', params) |
|
93
|
|
|
self.assertEqual(request_signature, verified_signature) |
|
94
|
|
|
|
|
95
|
|
|
def test_signature_delete(self, m): |
|
96
|
|
|
'''Test the signature on a delete request.''' |
|
97
|
|
|
order_id = 'A1234BC' |
|
98
|
|
|
trading_pair = 'btceur' |
|
99
|
|
|
params = {'order_id': order_id, |
|
100
|
|
|
'trading_pair': trading_pair} |
|
101
|
|
|
m.delete(requests_mock.ANY, json={}, status_code=200) |
|
102
|
|
|
btcde.deleteOrder(self.conn, order_id, trading_pair) |
|
103
|
|
|
history = m.request_history |
|
104
|
|
|
request_signature = history[0].headers.get('X-API-SIGNATURE') |
|
105
|
|
|
url = btcde.orderuri + "/" + order_id + "/" + trading_pair |
|
106
|
|
|
verified_signature = self.verifySignature(url, 'DELETE', params) |
|
107
|
|
|
self.assertEqual(request_signature, verified_signature) |
|
108
|
|
|
|