3 from decimal
import Decimal
as D
4 from unittest
import mock
6 class AmountTest(unittest
.TestCase
):
8 amount
= portfolio
.Amount("BTC", "0.65")
9 self
.assertEqual(D("0.65"), amount
.value
)
10 self
.assertEqual("BTC", amount
.currency
)
12 def test_in_currency(self
):
13 amount
= portfolio
.Amount("ETC", 10)
15 self
.assertEqual(amount
, amount
.in_currency("ETC", None))
17 ticker_mock
= unittest
.mock
.Mock()
18 with mock
.patch
.object(portfolio
.Trade
, 'get_ticker', new
=ticker_mock
):
19 ticker_mock
.return_value
= None
21 self
.assertRaises(Exception, amount
.in_currency
, "ETH", None)
23 with mock
.patch
.object(portfolio
.Trade
, 'get_ticker', new
=ticker_mock
):
24 ticker_mock
.return_value
= {
30 converted_amount
= amount
.in_currency("ETH", None)
32 self
.assertEqual(D("3.0"), converted_amount
.value
)
33 self
.assertEqual("ETH", converted_amount
.currency
)
34 self
.assertEqual(amount
, converted_amount
.linked_to
)
35 self
.assertEqual("bar", converted_amount
.ticker
["foo"])
37 converted_amount
= amount
.in_currency("ETH", None, action
="bid", compute_value
="default")
38 self
.assertEqual(D("2"), converted_amount
.value
)
40 converted_amount
= amount
.in_currency("ETH", None, compute_value
="ask")
41 self
.assertEqual(D("4"), converted_amount
.value
)
43 converted_amount
= amount
.in_currency("ETH", None, rate
=D("0.02"))
44 self
.assertEqual(D("0.2"), converted_amount
.value
)
47 amount
= portfolio
.Amount("SC", -120)
48 self
.assertEqual(120, abs(amount
).value
)
49 self
.assertEqual("SC", abs(amount
).currency
)
51 amount
= portfolio
.Amount("SC", 10)
52 self
.assertEqual(10, abs(amount
).value
)
53 self
.assertEqual("SC", abs(amount
).currency
)
56 amount1
= portfolio
.Amount("XVG", "12.9")
57 amount2
= portfolio
.Amount("XVG", "13.1")
59 self
.assertEqual(26, (amount1
+ amount2
).value
)
60 self
.assertEqual("XVG", (amount1
+ amount2
).currency
)
62 amount3
= portfolio
.Amount("ETH", "1.6")
63 with self
.assertRaises(Exception):
66 amount4
= portfolio
.Amount("ETH", 0.0)
67 self
.assertEqual(amount1
, amount1
+ amount4
)
70 amount
= portfolio
.Amount("XVG", "12.9")
72 self
.assertEqual(amount
, 0 + amount
)
73 with self
.assertRaises(Exception):
77 amount1
= portfolio
.Amount("XVG", "13.3")
78 amount2
= portfolio
.Amount("XVG", "13.1")
80 self
.assertEqual(D("0.2"), (amount1
- amount2
).value
)
81 self
.assertEqual("XVG", (amount1
- amount2
).currency
)
83 amount3
= portfolio
.Amount("ETH", "1.6")
84 with self
.assertRaises(Exception):
87 amount4
= portfolio
.Amount("ETH", 0.0)
88 self
.assertEqual(amount1
, amount1
- amount4
)
91 amount
= portfolio
.Amount("XEM", 11)
93 self
.assertEqual(D("38.5"), (amount
* D("3.5")).value
)
94 self
.assertEqual(D("33"), (amount
* 3).value
)
96 with self
.assertRaises(Exception):
100 amount
= portfolio
.Amount("XEM", 11)
102 self
.assertEqual(D("38.5"), (D("3.5") * amount
).value
)
103 self
.assertEqual(D("33"), (3 * amount
).value
)
105 def test__floordiv(self
):
106 amount
= portfolio
.Amount("XEM", 11)
108 self
.assertEqual(D("5.5"), (amount
/ 2).value
)
109 self
.assertEqual(D("4.4"), (amount
/ D("2.5")).value
)
112 amount
= portfolio
.Amount("XEM", 11)
114 self
.assertEqual(D("5.5"), (amount
/ 2).value
)
115 self
.assertEqual(D("4.4"), (amount
/ D("2.5")).value
)
118 amount1
= portfolio
.Amount("BTD", 11.3)
119 amount2
= portfolio
.Amount("BTD", 13.1)
121 self
.assertTrue(amount1
< amount2
)
122 self
.assertFalse(amount2
< amount1
)
123 self
.assertFalse(amount1
< amount1
)
125 amount3
= portfolio
.Amount("BTC", 1.6)
126 with self
.assertRaises(Exception):
130 amount1
= portfolio
.Amount("BTD", 11.3)
131 amount2
= portfolio
.Amount("BTD", 13.1)
132 amount3
= portfolio
.Amount("BTD", 11.3)
134 self
.assertFalse(amount1
== amount2
)
135 self
.assertFalse(amount2
== amount1
)
136 self
.assertTrue(amount1
== amount3
)
137 self
.assertFalse(amount2
== 0)
139 amount4
= portfolio
.Amount("BTC", 1.6)
140 with self
.assertRaises(Exception):
143 amount5
= portfolio
.Amount("BTD", 0)
144 self
.assertTrue(amount5
== 0)
147 amount1
= portfolio
.Amount("BTX", 32)
148 self
.assertEqual("32.00000000 BTX", str(amount1
))
150 amount2
= portfolio
.Amount("USDT", 12000)
151 amount1
.linked_to
= amount2
152 self
.assertEqual("32.00000000 BTX [12000.00000000 USDT]", str(amount1
))
154 def test__repr(self
):
155 amount1
= portfolio
.Amount("BTX", 32)
156 self
.assertEqual("Amount(32.00000000 BTX)", repr(amount1
))
158 amount2
= portfolio
.Amount("USDT", 12000)
159 amount1
.linked_to
= amount2
160 self
.assertEqual("Amount(32.00000000 BTX -> Amount(12000.00000000 USDT))", repr(amount1
))
162 amount3
= portfolio
.Amount("BTC", 0.1)
163 amount2
.linked_to
= amount3
164 self
.assertEqual("Amount(32.00000000 BTX -> Amount(12000.00000000 USDT -> Amount(0.10000000 BTC)))", repr(amount1
))
166 class PortfolioTest(unittest
.TestCase
):
169 if self
.json_response
is not None:
170 portfolio
.Portfolio
.data
= self
.json_response
173 super(PortfolioTest
, self
).setUp()
175 with open("test_portfolio.json") as example
:
177 self
.json_response
= json
.load(example
)
179 self
.patcher
= mock
.patch
.multiple(portfolio
.Portfolio
, data
=None, liquidities
={})
182 @mock.patch.object(urllib3
, "disable_warnings")
183 @mock.patch.object(urllib3
.poolmanager
.PoolManager
, "request")
184 @mock.patch.object(portfolio
.Portfolio
, "URL", new
="foo://bar")
185 def test_get_cryptoportfolio(self
, request
, disable_warnings
):
186 request
.side_effect
= [
187 type('', (), { "data": '{ "foo": "bar" }
' }),
188 type('', (), { "data": 'System Error' }),
189 Exception("Connection error"),
192 portfolio.Portfolio.get_cryptoportfolio()
193 self.assertIn("foo", portfolio.Portfolio.data)
194 self.assertEqual("bar", portfolio.Portfolio.data["foo"])
195 request.assert_called_with("GET", "foo://bar")
198 portfolio.Portfolio.get_cryptoportfolio()
199 self.assertIsNone(portfolio.Portfolio.data)
200 request.assert_called_with("GET", "foo://bar")
203 portfolio.Portfolio.data = "foo"
204 portfolio.Portfolio.get_cryptoportfolio()
205 request.assert_called_with("GET", "foo://bar")
206 self.assertEqual("foo", portfolio.Portfolio.data)
207 disable_warnings.assert_called_with()
209 @mock.patch.object(portfolio.Portfolio, "get_cryptoportfolio")
210 def test_parse_cryptoportfolio(self, mock_get):
211 mock_get.side_effect = self.fill_data
213 portfolio.Portfolio.parse_cryptoportfolio()
215 self.assertListEqual(
217 list(portfolio.Portfolio.liquidities.keys()))
219 liquidities = portfolio.Portfolio.liquidities
220 self.assertEqual(10, len(liquidities["medium"].keys()))
221 self.assertEqual(10, len(liquidities["high"].keys()))
223 expected = {'BTC': 2857, 'DGB': 1015, 'DOGE': 1805, 'SC': 623, 'ZEC': 3701}
224 self.assertDictEqual(expected, liquidities["high"]['2018-01-08'])
226 expected = {'ETC': 1000, 'FCT': 1000, 'GAS': 1000, 'NAV': 1000, 'OMG': 1000, 'OMNI': 1000, 'PPC': 1000, 'RIC': 1000, 'VIA': 1000, 'XCP': 1000}
227 self.assertDictEqual(expected, liquidities["medium"]['2018-01-08'])
229 # It doesn't refetch the data when available
230 portfolio
.Portfolio
.parse_cryptoportfolio()
231 mock_get
.assert_called_once_with()
233 portfolio
.Portfolio
.data
["portfolio_1"]["holding"]["direction"][3] = "short"
234 self
.assertRaises(AssertionError, portfolio
.Portfolio
.parse_cryptoportfolio
)
236 @mock.patch.object(portfolio
.Portfolio
, "get_cryptoportfolio")
237 def test_repartition_pertenthousand(self
, mock_get
):
238 mock_get
.side_effect
= self
.fill_data
240 expected_medium
= {'USDT': 1000, 'ETC': 1000, 'FCT': 1000, 'OMG': 1000, 'STEEM': 1000, 'STRAT': 1000, 'XEM': 1000, 'XMR': 1000, 'XVC': 1000, 'ZRX': 1000}
241 expected_high
= {'USDT': 1226, 'BTC': 1429, 'ETC': 1127, 'ETH': 1569, 'FCT': 3341, 'GAS': 1308}
243 self
.assertEqual(expected_medium
, portfolio
.Portfolio
.repartition_pertenthousand())
244 self
.assertEqual(expected_medium
, portfolio
.Portfolio
.repartition_pertenthousand(liquidity
="medium"))
245 self
.assertEqual(expected_high
, portfolio
.Portfolio
.repartition_pertenthousand(liquidity
="high"))
250 class BalanceTest(unittest
.TestCase
):
252 super(BalanceTest
, self
).setUp()
254 self
.fetch_balance
= {
280 self
.patcher
= mock
.patch
.multiple(portfolio
.Balance
, known_balances
={})
283 def test_values(self
):
284 balance
= portfolio
.Balance("BTC", 0.65, 0.35, 0.30)
285 self
.assertEqual(0.65, balance
.total
.value
)
286 self
.assertEqual(0.35, balance
.free
.value
)
287 self
.assertEqual(0.30, balance
.used
.value
)
288 self
.assertEqual("BTC", balance
.currency
)
290 balance
= portfolio
.Balance
.from_hash("BTC", { "total": 0.65, "free": 0.35, "used": 0.30}
)
291 self
.assertEqual(0.65, balance
.total
.value
)
292 self
.assertEqual(0.35, balance
.free
.value
)
293 self
.assertEqual(0.30, balance
.used
.value
)
294 self
.assertEqual("BTC", balance
.currency
)
296 @mock.patch.object(portfolio
.Trade
, "get_ticker")
297 def test_in_currency(self
, get_ticker
):
298 portfolio
.Balance
.known_balances
= {
299 "BTC": portfolio
.Balance("BTC", "0.65", "0.35", "0.30"),
300 "ETH": portfolio
.Balance("ETH", 3, 3, 0),
303 get_ticker
.return_value
= {
309 amounts
= portfolio
.Balance
.in_currency("BTC", market
)
310 self
.assertEqual("BTC", amounts
["ETH"].currency
)
311 self
.assertEqual(D("0.65"), amounts
["BTC"].value
)
312 self
.assertEqual(D("0.30"), amounts
["ETH"].value
)
314 amounts
= portfolio
.Balance
.in_currency("BTC", market
, compute_value
="bid")
315 self
.assertEqual(D("0.65"), amounts
["BTC"].value
)
316 self
.assertEqual(D("0.27"), amounts
["ETH"].value
)
318 amounts
= portfolio
.Balance
.in_currency("BTC", market
, compute_value
="bid", type="used")
319 self
.assertEqual(D("0.30"), amounts
["BTC"].value
)
320 self
.assertEqual(0, amounts
["ETH"].value
)
322 def test_currencies(self
):
323 portfolio
.Balance
.known_balances
= {
324 "BTC": portfolio
.Balance("BTC", "0.65", "0.35", "0.30"),
325 "ETH": portfolio
.Balance("ETH", 3, 3, 0),
327 self
.assertListEqual(["BTC", "ETH"], list(portfolio
.Balance
.currencies()))
329 @mock.patch.object(portfolio
.market
, "fetch_balance")
330 def test_fetch_balances(self
, fetch_balance
):
331 fetch_balance
.return_value
= self
.fetch_balance
333 portfolio
.Balance
.fetch_balances(portfolio
.market
)
334 self
.assertNotIn("XMR", portfolio
.Balance
.currencies())
335 self
.assertListEqual(["USDT", "XVG"], list(portfolio
.Balance
.currencies()))
337 portfolio
.Balance
.known_balances
["ETC"] = portfolio
.Balance("ETC", "1", "0", "1")
338 portfolio
.Balance
.fetch_balances(portfolio
.market
)
339 self
.assertEqual(0, portfolio
.Balance
.known_balances
["ETC"].total
)
340 self
.assertListEqual(["USDT", "XVG", "ETC"], list(portfolio
.Balance
.currencies()))
342 @mock.patch.object(portfolio
.Portfolio
, "repartition_pertenthousand")
343 @mock.patch.object(portfolio
.market
, "fetch_balance")
344 def test_dispatch_assets(self
, fetch_balance
, repartition
):
345 fetch_balance
.return_value
= self
.fetch_balance
346 portfolio
.Balance
.fetch_balances(portfolio
.market
)
348 self
.assertNotIn("XEM", portfolio
.Balance
.currencies())
350 repartition
.return_value
= {
355 amounts
= portfolio
.Balance
.dispatch_assets(portfolio
.Amount("BTC", "10.1"))
356 self
.assertIn("XEM", portfolio
.Balance
.currencies())
357 self
.assertEqual(D("2.6"), amounts
["BTC"].value
)
358 self
.assertEqual(D("7.5"), amounts
["XEM"].value
)
360 @mock.patch.object(portfolio
.Portfolio
, "repartition_pertenthousand")
361 @mock.patch.object(portfolio
.Trade
, "get_ticker")
362 @mock.patch.object(portfolio
.Trade
, "compute_trades")
363 def test_prepare_trades(self
, compute_trades
, get_ticker
, repartition
):
364 repartition
.return_value
= {
368 def _get_ticker(c1
, c2
, market
):
369 if c1
== "USDT" and c2
== "BTC":
370 return { "average": D("0.0001") }
371 if c1
== "XVG" and c2
== "BTC":
372 return { "average": D("0.000001") }
373 if c1
== "XEM" and c2
== "BTC":
374 return { "average": D("0.001") }
375 self
.fail("Should be called with {}, {}".format(c1
, c2
))
376 get_ticker
.side_effect
= _get_ticker
379 market
.fetch_balance
.return_value
= {
381 "free": D("10000.0"),
383 "total": D("10000.0")
386 "free": D("10000.0"),
388 "total": D("10000.0")
391 portfolio
.Balance
.prepare_trades(market
)
392 compute_trades
.assert_called()
394 call
= compute_trades
.call_args
395 self
.assertEqual(market
, call
[1]["market"])
396 self
.assertEqual(1, call
[0][0]["USDT"].value
)
397 self
.assertEqual(D("0.01"), call
[0][0]["XVG"].value
)
398 self
.assertEqual(D("0.2525"), call
[0][1]["BTC"].value
)
399 self
.assertEqual(D("0.7575"), call
[0][1]["XEM"].value
)
401 @unittest.skip("TODO")
402 def test_update_trades(self
):
405 def test__repr(self
):
406 balance
= portfolio
.Balance("BTX", 3, 1, 2)
407 self
.assertEqual("Balance(BTX [1.00000000 BTX/2.00000000 BTX/3.00000000 BTX])", repr(balance
))
412 class TradeTest(unittest
.TestCase
):
416 super(TradeTest
, self
).setUp()
418 self
.patcher
= mock
.patch
.multiple(portfolio
.Trade
,
420 ticker_cache_timestamp
=self
.time
.time(),
425 def test_get_ticker(self
):
427 market
.fetch_ticker
.side_effect
= [
428 { "bid": 1, "ask": 3 }
,
429 portfolio
.ccxt
.ExchangeError("foo"),
430 { "bid": 10, "ask": 40 }
,
431 portfolio
.ccxt
.ExchangeError("foo"),
432 portfolio
.ccxt
.ExchangeError("foo"),
435 ticker
= portfolio
.Trade
.get_ticker("ETH", "ETC", market
)
436 market
.fetch_ticker
.assert_called_with("ETH/ETC")
437 self
.assertEqual(1, ticker
["bid"])
438 self
.assertEqual(3, ticker
["ask"])
439 self
.assertEqual(2, ticker
["average"])
440 self
.assertFalse(ticker
["inverted"])
442 ticker
= portfolio
.Trade
.get_ticker("ETH", "XVG", market
)
443 self
.assertEqual(0.0625, ticker
["average"])
444 self
.assertTrue(ticker
["inverted"])
445 self
.assertIn("original", ticker
)
446 self
.assertEqual(10, ticker
["original"]["bid"])
448 ticker
= portfolio
.Trade
.get_ticker("XVG", "XMR", market
)
449 self
.assertIsNone(ticker
)
451 market
.fetch_ticker
.assert_has_calls([
452 mock
.call("ETH/ETC"),
453 mock
.call("ETH/XVG"),
454 mock
.call("XVG/ETH"),
455 mock
.call("XVG/XMR"),
456 mock
.call("XMR/XVG"),
459 market2
= mock
.Mock()
460 market2
.fetch_ticker
.side_effect
= [
461 { "bid": 1, "ask": 3 }
,
462 { "bid": 1.2, "ask": 3.5 }
,
464 ticker1
= portfolio
.Trade
.get_ticker("ETH", "ETC", market2
)
465 ticker2
= portfolio
.Trade
.get_ticker("ETH", "ETC", market2
)
466 ticker3
= portfolio
.Trade
.get_ticker("ETC", "ETH", market2
)
467 market2
.fetch_ticker
.assert_called_once_with("ETH/ETC")
468 self
.assertEqual(1, ticker1
["bid"])
469 self
.assertDictEqual(ticker1
, ticker2
)
470 self
.assertDictEqual(ticker1
, ticker3
["original"])
472 ticker4
= portfolio
.Trade
.get_ticker("ETH", "ETC", market2
, refresh
=True)
473 ticker5
= portfolio
.Trade
.get_ticker("ETH", "ETC", market2
)
474 self
.assertEqual(1.2, ticker4
["bid"])
475 self
.assertDictEqual(ticker4
, ticker5
)
477 market3
= mock
.Mock()
478 market3
.fetch_ticker
.side_effect
= [
479 { "bid": 1, "ask": 3 }
,
480 { "bid": 1.2, "ask": 3.5 }
,
482 ticker6
= portfolio
.Trade
.get_ticker("ETH", "ETC", market3
)
483 portfolio
.Trade
.ticker_cache_timestamp
-= 4
484 ticker7
= portfolio
.Trade
.get_ticker("ETH", "ETC", market3
)
485 portfolio
.Trade
.ticker_cache_timestamp
-= 2
486 ticker8
= portfolio
.Trade
.get_ticker("ETH", "ETC", market3
)
487 self
.assertDictEqual(ticker6
, ticker7
)
488 self
.assertEqual(1.2, ticker8
["bid"])
490 @unittest.skip("TODO")
491 def test_values_assertion(self
):
494 @unittest.skip("TODO")
495 def test_fetch_fees(self
):
498 @unittest.skip("TODO")
499 def test_compute_trades(self
):
502 @unittest.skip("TODO")
503 def test_action(self
):
506 @unittest.skip("TODO")
507 def test_action(self
):
510 @unittest.skip("TODO")
511 def test_order_action(self
):
514 @unittest.skip("TODO")
515 def test_prepare_order(self
):
518 @unittest.skip("TODO")
519 def test_all_orders(self
):
522 @unittest.skip("TODO")
523 def test_follow_orders(self
):
526 @unittest.skip("TODO")
527 def test_compute_value(self
):
530 @unittest.skip("TODO")
531 def test__repr(self
):
537 class AcceptanceTest(unittest
.TestCase
):
541 super(AcceptanceTest
, self
).setUp()
544 mock
.patch
.multiple(portfolio
.Balance
, known_balances
={}),
545 mock
.patch
.multiple(portfolio
.Portfolio
, data
=None, liquidities
={}),
546 mock
.patch
.multiple(portfolio
.Trade
,
548 ticker_cache_timestamp
=self
.time
.time(),
551 mock
.patch
.multiple(portfolio
.Computation
,
552 computations
=portfolio
.Computation
.computations
)
554 for patcher
in self
.patchers
:
557 def test_success_sell_only_necessary(self
):
572 "total": D("1000.0"),
583 def fetch_ticker(symbol
):
584 if symbol
== "ETH/BTC":
590 if symbol
== "ETC/BTC":
596 if symbol
== "XVG/BTC":
602 if symbol
== "BTD/BTC":
608 if symbol
== "USDT/BTC":
609 raise portfolio
.ccxt
.ExchangeError
610 if symbol
== "BTC/USDT":
612 "symbol": "BTC/USDT",
616 self
.fail("Shouldn't have been called with {}".format(symbol
))
619 market
.fetch_balance
.return_value
= fetch_balance
620 market
.fetch_ticker
.side_effect
= fetch_ticker
621 with mock
.patch
.object(portfolio
.Portfolio
, "repartition_pertenthousand", return_value
=repartition
):
623 portfolio
.Balance
.prepare_trades(market
)
625 balances
= portfolio
.Balance
.known_balances
626 self
.assertEqual(portfolio
.Amount("ETH", 1), balances
["ETH"].total
)
627 self
.assertEqual(portfolio
.Amount("ETC", 4), balances
["ETC"].total
)
628 self
.assertEqual(portfolio
.Amount("XVG", 1000), balances
["XVG"].total
)
631 trades
= portfolio
.Trade
.trades
632 self
.assertEqual(portfolio
.Amount("BTC", D("0.15")), trades
["ETH"].value_from
)
633 self
.assertEqual(portfolio
.Amount("BTC", D("0.05")), trades
["ETH"].value_to
)
634 self
.assertEqual("sell", trades
["ETH"].action
)
636 self
.assertEqual(portfolio
.Amount("BTC", D("0.01")), trades
["ETC"].value_from
)
637 self
.assertEqual(portfolio
.Amount("BTC", D("0.05")), trades
["ETC"].value_to
)
638 self
.assertEqual("buy", trades
["ETC"].action
)
640 self
.assertNotIn("BTC", trades
)
642 self
.assertEqual(portfolio
.Amount("BTC", D("0.00")), trades
["BTD"].value_from
)
643 self
.assertEqual(portfolio
.Amount("BTC", D("0.01")), trades
["BTD"].value_to
)
644 self
.assertEqual("buy", trades
["BTD"].action
)
646 self
.assertEqual(portfolio
.Amount("BTC", D("0.00")), trades
["USDT"].value_from
)
647 self
.assertEqual(portfolio
.Amount("BTC", D("0.01")), trades
["USDT"].value_to
)
648 self
.assertEqual("buy", trades
["USDT"].action
)
650 self
.assertEqual(portfolio
.Amount("BTC", D("0.04")), trades
["XVG"].value_from
)
651 self
.assertEqual(portfolio
.Amount("BTC", D("0.00")), trades
["XVG"].value_to
)
652 self
.assertEqual("sell", trades
["XVG"].action
)
655 portfolio
.Trade
.prepare_orders(only
="sell", compute_value
=lambda x
, y
: x
["bid"] * D("1.001"))
657 all_orders
= portfolio
.Trade
.all_orders()
658 self
.assertEqual(2, len(all_orders
))
659 self
.assertEqual(2, 3*all_orders
[0].amount
.value
)
660 self
.assertEqual(D("0.14014"), all_orders
[0].rate
)
661 self
.assertEqual(1000, all_orders
[1].amount
.value
)
662 self
.assertEqual(D("0.00003003"), all_orders
[1].rate
)
665 def create_order(symbol
, type, action
, amount
, price
=None):
666 self
.assertEqual("limit", type)
667 if symbol
== "ETH/BTC":
668 self
.assertEqual("bid", action
)
669 self
.assertEqual(2, 3*amount
)
670 self
.assertEqual(D("0.14014"), price
)
671 elif symbol
== "XVG/BTC":
672 self
.assertEqual("bid", action
)
673 self
.assertEqual(1000, amount
)
674 self
.assertEqual(D("0.00003003"), price
)
676 self
.fail("I shouldn't have been called")
681 market
.create_order
.side_effect
= create_order
684 portfolio
.Trade
.run_orders()
686 self
.assertEqual("open", all_orders
[0].status
)
687 self
.assertEqual("open", all_orders
[1].status
)
689 market
.fetch_order
.return_value
= { "status": "closed" }
690 with mock
.patch
.object(portfolio
.time
, "sleep") as sleep
:
692 portfolio
.Trade
.follow_orders(verbose
=False)
694 sleep
.assert_called_with(30)
696 for order
in all_orders
:
697 self
.assertEqual("closed", order
.status
)
701 "free": D("1.0") / 3,
703 "total": D("1.0") / 3,
721 market
.fetch_balance
.return_value
= fetch_balance
723 with mock
.patch
.object(portfolio
.Portfolio
, "repartition_pertenthousand", return_value
=repartition
):
725 portfolio
.Balance
.update_trades(market
, only
="buy", compute_value
="average")
727 balances
= portfolio
.Balance
.known_balances
728 self
.assertEqual(portfolio
.Amount("ETH", 1 / D("3")), balances
["ETH"].total
)
729 self
.assertEqual(portfolio
.Amount("ETC", 4), balances
["ETC"].total
)
730 self
.assertEqual(portfolio
.Amount("BTC", D("0.134")), balances
["BTC"].total
)
731 self
.assertEqual(portfolio
.Amount("XVG", 0), balances
["XVG"].total
)
734 trades
= portfolio
.Trade
.trades
735 self
.assertEqual(portfolio
.Amount("BTC", D("0.15")), trades
["ETH"].value_from
)
736 self
.assertEqual(portfolio
.Amount("BTC", D("0.05")), trades
["ETH"].value_to
)
737 self
.assertEqual("sell", trades
["ETH"].action
)
739 self
.assertEqual(portfolio
.Amount("BTC", D("0.01")), trades
["ETC"].value_from
)
740 self
.assertEqual(portfolio
.Amount("BTC", D("0.0485")), trades
["ETC"].value_to
)
741 self
.assertEqual("buy", trades
["ETC"].action
)
743 self
.assertNotIn("BTC", trades
)
745 self
.assertEqual(portfolio
.Amount("BTC", D("0.00")), trades
["BTD"].value_from
)
746 self
.assertEqual(portfolio
.Amount("BTC", D("0.0097")), trades
["BTD"].value_to
)
747 self
.assertEqual("buy", trades
["BTD"].action
)
749 self
.assertEqual(portfolio
.Amount("BTC", D("0.00")), trades
["USDT"].value_from
)
750 self
.assertEqual(portfolio
.Amount("BTC", D("0.0097")), trades
["USDT"].value_to
)
751 self
.assertEqual("buy", trades
["USDT"].action
)
753 self
.assertEqual(portfolio
.Amount("BTC", D("0.04")), trades
["XVG"].value_from
)
754 self
.assertEqual(portfolio
.Amount("BTC", D("0.00")), trades
["XVG"].value_to
)
755 self
.assertEqual("sell", trades
["XVG"].action
)
758 portfolio
.Trade
.prepare_orders(only
="buy", compute_value
=lambda x
, y
: x
["ask"] * D("0.999"))
760 all_orders
= portfolio
.Trade
.all_orders(state
="pending")
761 self
.assertEqual(3, len(all_orders
))
762 self
.assertEqual(portfolio
.Amount("ETC", D("15.4")), all_orders
[0].amount
)
763 self
.assertEqual(D("0.002997"), all_orders
[0].rate
)
764 self
.assertEqual("ask", all_orders
[0].action
)
765 self
.assertEqual(portfolio
.Amount("BTD", D("9.7")), all_orders
[1].amount
)
766 self
.assertEqual(D("0.0011988"), all_orders
[1].rate
)
767 self
.assertEqual("ask", all_orders
[1].action
)
768 self
.assertEqual(portfolio
.Amount("BTC", D("0.0097")), all_orders
[2].amount
)
769 self
.assertEqual(D("15984"), all_orders
[2].rate
)
770 self
.assertEqual("bid", all_orders
[2].action
)
772 with mock
.patch
.object(portfolio
.time
, "sleep") as sleep
:
774 portfolio
.Trade
.follow_orders(verbose
=False)
776 sleep
.assert_called_with(30)
779 for patcher
in self
.patchers
:
782 if __name__
== '__main__':