+@unittest.skipUnless("unit" in limits, "Unit skipped")
+class ComputationTest(WebMockTestCase):
+ def test_compute_value(self):
+ compute = mock.Mock()
+ portfolio.Computation.compute_value("foo", "buy", compute_value=compute)
+ compute.assert_called_with("foo", "ask")
+
+ compute.reset_mock()
+ portfolio.Computation.compute_value("foo", "sell", compute_value=compute)
+ compute.assert_called_with("foo", "bid")
+
+ compute.reset_mock()
+ portfolio.Computation.compute_value("foo", "ask", compute_value=compute)
+ compute.assert_called_with("foo", "ask")
+
+ compute.reset_mock()
+ portfolio.Computation.compute_value("foo", "bid", compute_value=compute)
+ compute.assert_called_with("foo", "bid")
+
+ compute.reset_mock()
+ portfolio.Computation.computations["test"] = compute
+ portfolio.Computation.compute_value("foo", "bid", compute_value="test")
+ compute.assert_called_with("foo", "bid")
+
+
+@unittest.skipUnless("unit" in limits, "Unit skipped")
+class TradeTest(WebMockTestCase):
+
+ def test_values_assertion(self):
+ value_from = portfolio.Amount("BTC", "1.0")
+ value_from.linked_to = portfolio.Amount("ETH", "10.0")
+ value_to = portfolio.Amount("BTC", "1.0")
+ trade = portfolio.Trade(value_from, value_to, "ETH")
+ self.assertEqual("BTC", trade.base_currency)
+ self.assertEqual("ETH", trade.currency)
+
+ with self.assertRaises(AssertionError):
+ portfolio.Trade(value_from, value_to, "ETC")
+ with self.assertRaises(AssertionError):
+ value_from.linked_to = None
+ portfolio.Trade(value_from, value_to, "ETH")
+ with self.assertRaises(AssertionError):
+ value_from.currency = "ETH"
+ portfolio.Trade(value_from, value_to, "ETH")
+
+ value_from = portfolio.Amount("BTC", 0)
+ trade = portfolio.Trade(value_from, value_to, "ETH")
+ self.assertEqual(0, trade.value_from.linked_to)
+
+ def test_action(self):
+ value_from = portfolio.Amount("BTC", "1.0")
+ value_from.linked_to = portfolio.Amount("ETH", "10.0")
+ value_to = portfolio.Amount("BTC", "1.0")
+ trade = portfolio.Trade(value_from, value_to, "ETH")
+
+ self.assertIsNone(trade.action)
+
+ value_from = portfolio.Amount("BTC", "1.0")
+ value_from.linked_to = portfolio.Amount("BTC", "1.0")
+ value_to = portfolio.Amount("BTC", "2.0")
+ trade = portfolio.Trade(value_from, value_to, "BTC")
+
+ self.assertIsNone(trade.action)
+
+ value_from = portfolio.Amount("BTC", "0.5")
+ value_from.linked_to = portfolio.Amount("ETH", "10.0")
+ value_to = portfolio.Amount("BTC", "1.0")
+ trade = portfolio.Trade(value_from, value_to, "ETH")
+
+ self.assertEqual("acquire", trade.action)
+
+ value_from = portfolio.Amount("BTC", "0")
+ value_from.linked_to = portfolio.Amount("ETH", "0")
+ value_to = portfolio.Amount("BTC", "-1.0")
+ trade = portfolio.Trade(value_from, value_to, "ETH")
+
+ self.assertEqual("acquire", trade.action)
+
+ def test_order_action(self):
+ value_from = portfolio.Amount("BTC", "0.5")
+ value_from.linked_to = portfolio.Amount("ETH", "10.0")
+ value_to = portfolio.Amount("BTC", "1.0")
+ trade = portfolio.Trade(value_from, value_to, "ETH")
+
+ self.assertEqual("buy", trade.order_action(False))
+ self.assertEqual("sell", trade.order_action(True))
+
+ value_from = portfolio.Amount("BTC", "0")
+ value_from.linked_to = portfolio.Amount("ETH", "0")
+ value_to = portfolio.Amount("BTC", "-1.0")
+ trade = portfolio.Trade(value_from, value_to, "ETH")
+
+ self.assertEqual("sell", trade.order_action(False))
+ self.assertEqual("buy", trade.order_action(True))
+
+ def test_trade_type(self):
+ value_from = portfolio.Amount("BTC", "0.5")
+ value_from.linked_to = portfolio.Amount("ETH", "10.0")
+ value_to = portfolio.Amount("BTC", "1.0")
+ trade = portfolio.Trade(value_from, value_to, "ETH")
+
+ self.assertEqual("long", trade.trade_type)
+
+ value_from = portfolio.Amount("BTC", "0")
+ value_from.linked_to = portfolio.Amount("ETH", "0")
+ value_to = portfolio.Amount("BTC", "-1.0")
+ trade = portfolio.Trade(value_from, value_to, "ETH")
+
+ self.assertEqual("short", trade.trade_type)
+
+ def test_filled_amount(self):
+ value_from = portfolio.Amount("BTC", "0.5")
+ value_from.linked_to = portfolio.Amount("ETH", "10.0")
+ value_to = portfolio.Amount("BTC", "1.0")
+ trade = portfolio.Trade(value_from, value_to, "ETH")
+
+ order1 = mock.Mock()
+ order1.filled_amount.return_value = portfolio.Amount("ETH", "0.3")
+
+ order2 = mock.Mock()
+ order2.filled_amount.return_value = portfolio.Amount("ETH", "0.01")
+ trade.orders.append(order1)
+ trade.orders.append(order2)
+
+ self.assertEqual(portfolio.Amount("ETH", "0.31"), trade.filled_amount())
+ order1.filled_amount.assert_called_with(in_base_currency=False)
+ order2.filled_amount.assert_called_with(in_base_currency=False)
+
+ self.assertEqual(portfolio.Amount("ETH", "0.31"), trade.filled_amount(in_base_currency=False))
+ order1.filled_amount.assert_called_with(in_base_currency=False)
+ order2.filled_amount.assert_called_with(in_base_currency=False)
+
+ self.assertEqual(portfolio.Amount("ETH", "0.31"), trade.filled_amount(in_base_currency=True))
+ order1.filled_amount.assert_called_with(in_base_currency=True)
+ order2.filled_amount.assert_called_with(in_base_currency=True)
+
+ @mock.patch.object(helper, "get_ticker")
+ @mock.patch.object(portfolio.Computation, "compute_value")
+ @mock.patch.object(portfolio.Trade, "filled_amount")
+ @mock.patch.object(portfolio, "Order")
+ def test_prepare_order(self, Order, filled_amount, compute_value, get_ticker):
+ Order.return_value = "Order"
+
+ with self.subTest(desc="Nothing to do"):
+ value_from = portfolio.Amount("BTC", "10")
+ value_from.rate = D("0.1")
+ value_from.linked_to = portfolio.Amount("FOO", "100")
+ value_to = portfolio.Amount("BTC", "10")
+ trade = portfolio.Trade(value_from, value_to, "FOO", market="market")
+
+ trade.prepare_order()
+
+ filled_amount.assert_not_called()
+ compute_value.assert_not_called()
+ self.assertEqual(0, len(trade.orders))
+ Order.assert_not_called()
+
+ get_ticker.return_value = { "inverted": False }
+ with self.subTest(desc="Already filled"),\
+ mock.patch("portfolio.ReportStore") as report_store:
+ filled_amount.return_value = portfolio.Amount("FOO", "100")
+ compute_value.return_value = D("0.125")
+
+ value_from = portfolio.Amount("BTC", "10")
+ value_from.rate = D("0.1")
+ value_from.linked_to = portfolio.Amount("FOO", "100")
+ value_to = portfolio.Amount("BTC", "0")
+ trade = portfolio.Trade(value_from, value_to, "FOO", market="market")
+
+ trade.prepare_order()
+
+ filled_amount.assert_called_with(in_base_currency=False)
+ compute_value.assert_called_with(get_ticker.return_value, "sell", compute_value="default")
+ self.assertEqual(0, len(trade.orders))
+ report_store.log_error.assert_called_with("prepare_order", message=mock.ANY)
+ Order.assert_not_called()
+
+ with self.subTest(action="dispose", inverted=False):
+ filled_amount.return_value = portfolio.Amount("FOO", "60")
+ compute_value.return_value = D("0.125")
+
+ value_from = portfolio.Amount("BTC", "10")
+ value_from.rate = D("0.1")
+ value_from.linked_to = portfolio.Amount("FOO", "100")
+ value_to = portfolio.Amount("BTC", "1")
+ trade = portfolio.Trade(value_from, value_to, "FOO", market="market")
+
+ trade.prepare_order()
+
+ filled_amount.assert_called_with(in_base_currency=False)
+ compute_value.assert_called_with(get_ticker.return_value, "sell", compute_value="default")
+ self.assertEqual(1, len(trade.orders))
+ Order.assert_called_with("sell", portfolio.Amount("FOO", 30),
+ D("0.125"), "BTC", "long", "market",
+ trade, close_if_possible=False)
+
+ with self.subTest(action="acquire", inverted=False):
+ filled_amount.return_value = portfolio.Amount("BTC", "3")
+ compute_value.return_value = D("0.125")
+
+ value_from = portfolio.Amount("BTC", "1")
+ value_from.rate = D("0.1")
+ value_from.linked_to = portfolio.Amount("FOO", "10")
+ value_to = portfolio.Amount("BTC", "10")
+ trade = portfolio.Trade(value_from, value_to, "FOO", market="market")
+
+ trade.prepare_order()
+
+ filled_amount.assert_called_with(in_base_currency=True)
+ compute_value.assert_called_with(get_ticker.return_value, "buy", compute_value="default")
+ self.assertEqual(1, len(trade.orders))
+
+ Order.assert_called_with("buy", portfolio.Amount("FOO", 48),
+ D("0.125"), "BTC", "long", "market",
+ trade, close_if_possible=False)
+
+ with self.subTest(close_if_possible=True):
+ filled_amount.return_value = portfolio.Amount("FOO", "0")
+ compute_value.return_value = D("0.125")
+
+ value_from = portfolio.Amount("BTC", "10")
+ value_from.rate = D("0.1")
+ value_from.linked_to = portfolio.Amount("FOO", "100")
+ value_to = portfolio.Amount("BTC", "0")
+ trade = portfolio.Trade(value_from, value_to, "FOO", market="market")
+
+ trade.prepare_order()
+
+ filled_amount.assert_called_with(in_base_currency=False)
+ compute_value.assert_called_with(get_ticker.return_value, "sell", compute_value="default")
+ self.assertEqual(1, len(trade.orders))
+ Order.assert_called_with("sell", portfolio.Amount("FOO", 100),
+ D("0.125"), "BTC", "long", "market",
+ trade, close_if_possible=True)
+
+ get_ticker.return_value = { "inverted": True, "original": {} }
+ with self.subTest(action="dispose", inverted=True):
+ filled_amount.return_value = portfolio.Amount("FOO", "300")
+ compute_value.return_value = D("125")
+
+ value_from = portfolio.Amount("BTC", "10")
+ value_from.rate = D("0.01")
+ value_from.linked_to = portfolio.Amount("FOO", "1000")
+ value_to = portfolio.Amount("BTC", "1")
+ trade = portfolio.Trade(value_from, value_to, "FOO", market="market")
+
+ trade.prepare_order(compute_value="foo")
+
+ filled_amount.assert_called_with(in_base_currency=True)
+ compute_value.assert_called_with(get_ticker.return_value["original"], "buy", compute_value="foo")
+ self.assertEqual(1, len(trade.orders))
+ Order.assert_called_with("buy", portfolio.Amount("BTC", D("4.8")),
+ D("125"), "FOO", "long", "market",
+ trade, close_if_possible=False)
+
+ with self.subTest(action="acquire", inverted=True):
+ filled_amount.return_value = portfolio.Amount("BTC", "4")
+ compute_value.return_value = D("125")
+
+ value_from = portfolio.Amount("BTC", "1")
+ value_from.rate = D("0.01")
+ value_from.linked_to = portfolio.Amount("FOO", "100")
+ value_to = portfolio.Amount("BTC", "10")
+ trade = portfolio.Trade(value_from, value_to, "FOO", market="market")
+
+ trade.prepare_order(compute_value="foo")
+
+ filled_amount.assert_called_with(in_base_currency=False)
+ compute_value.assert_called_with(get_ticker.return_value["original"], "sell", compute_value="foo")
+ self.assertEqual(1, len(trade.orders))
+ Order.assert_called_with("sell", portfolio.Amount("BTC", D("5")),
+ D("125"), "FOO", "long", "market",
+ trade, close_if_possible=False)
+
+
+ @mock.patch.object(portfolio.Trade, "prepare_order")
+ def test_update_order(self, prepare_order):
+ order_mock = mock.Mock()
+ new_order_mock = mock.Mock()
+
+ value_from = portfolio.Amount("BTC", "0.5")
+ value_from.linked_to = portfolio.Amount("ETH", "10.0")
+ value_to = portfolio.Amount("BTC", "1.0")
+ trade = portfolio.Trade(value_from, value_to, "ETH")
+ prepare_order.return_value = new_order_mock
+
+ for i in [0, 1, 3, 4, 6]:
+ with self.subTest(tick=i),\
+ mock.patch.object(portfolio.ReportStore, "log_order") as log_order:
+ trade.update_order(order_mock, i)
+ order_mock.cancel.assert_not_called()
+ new_order_mock.run.assert_not_called()
+ log_order.assert_called_once_with(order_mock, i,
+ update="waiting", compute_value=None, new_order=None)
+
+ order_mock.reset_mock()
+ new_order_mock.reset_mock()
+ trade.orders = []
+
+ with mock.patch.object(portfolio.ReportStore, "log_order") as log_order:
+ trade.update_order(order_mock, 2)
+ order_mock.cancel.assert_called()
+ new_order_mock.run.assert_called()
+ prepare_order.assert_called()
+ log_order.assert_called()
+ self.assertEqual(2, log_order.call_count)
+ calls = [
+ mock.call(order_mock, 2, update="adjusting",
+ compute_value='lambda x, y: (x[y] + x["average"]) / 2',
+ new_order=new_order_mock),
+ mock.call(order_mock, 2, new_order=new_order_mock),
+ ]
+ log_order.assert_has_calls(calls)
+
+ order_mock.reset_mock()
+ new_order_mock.reset_mock()
+ trade.orders = []
+
+ with mock.patch.object(portfolio.ReportStore, "log_order") as log_order:
+ trade.update_order(order_mock, 5)
+ order_mock.cancel.assert_called()
+ new_order_mock.run.assert_called()
+ prepare_order.assert_called()
+ self.assertEqual(2, log_order.call_count)
+ log_order.assert_called()
+ calls = [
+ mock.call(order_mock, 5, update="adjusting",
+ compute_value='lambda x, y: (x[y]*2 + x["average"]) / 3',
+ new_order=new_order_mock),
+ mock.call(order_mock, 5, new_order=new_order_mock),
+ ]
+ log_order.assert_has_calls(calls)
+
+ order_mock.reset_mock()
+ new_order_mock.reset_mock()
+ trade.orders = []
+
+ with mock.patch.object(portfolio.ReportStore, "log_order") as log_order:
+ trade.update_order(order_mock, 7)
+ order_mock.cancel.assert_called()
+ new_order_mock.run.assert_called()
+ prepare_order.assert_called_with(compute_value="default")
+ log_order.assert_called()
+ self.assertEqual(2, log_order.call_count)
+ calls = [
+ mock.call(order_mock, 7, update="market_fallback",
+ compute_value='default',
+ new_order=new_order_mock),
+ mock.call(order_mock, 7, new_order=new_order_mock),
+ ]
+ log_order.assert_has_calls(calls)
+
+ order_mock.reset_mock()
+ new_order_mock.reset_mock()
+ trade.orders = []
+
+ for i in [10, 13, 16]:
+ with self.subTest(tick=i), mock.patch.object(portfolio.ReportStore, "log_order") as log_order:
+ trade.update_order(order_mock, i)
+ order_mock.cancel.assert_called()
+ new_order_mock.run.assert_called()
+ prepare_order.assert_called_with(compute_value="default")
+ log_order.assert_called()
+ self.assertEqual(2, log_order.call_count)
+ calls = [
+ mock.call(order_mock, i, update="market_adjust",
+ compute_value='default',
+ new_order=new_order_mock),
+ mock.call(order_mock, i, new_order=new_order_mock),
+ ]
+ log_order.assert_has_calls(calls)
+
+ order_mock.reset_mock()
+ new_order_mock.reset_mock()
+ trade.orders = []
+
+ for i in [8, 9, 11, 12]:
+ with self.subTest(tick=i), mock.patch.object(portfolio.ReportStore, "log_order") as log_order:
+ trade.update_order(order_mock, i)
+ order_mock.cancel.assert_not_called()
+ new_order_mock.run.assert_not_called()
+ log_order.assert_called_once_with(order_mock, i, update="waiting",
+ compute_value=None, new_order=None)
+
+ order_mock.reset_mock()
+ new_order_mock.reset_mock()
+ trade.orders = []
+
+
+ @mock.patch.object(portfolio.ReportStore, "print_log")
+ def test_print_with_order(self, print_log):
+ value_from = portfolio.Amount("BTC", "0.5")
+ value_from.linked_to = portfolio.Amount("ETH", "10.0")
+ value_to = portfolio.Amount("BTC", "1.0")
+ trade = portfolio.Trade(value_from, value_to, "ETH")
+
+ order_mock1 = mock.Mock()
+ order_mock1.__repr__ = mock.Mock()
+ order_mock1.__repr__.return_value = "Mock 1"
+ order_mock2 = mock.Mock()
+ order_mock2.__repr__ = mock.Mock()
+ order_mock2.__repr__.return_value = "Mock 2"
+ order_mock1.mouvements = []
+ mouvement_mock1 = mock.Mock()
+ mouvement_mock1.__repr__ = mock.Mock()
+ mouvement_mock1.__repr__.return_value = "Mouvement 1"
+ mouvement_mock2 = mock.Mock()
+ mouvement_mock2.__repr__ = mock.Mock()
+ mouvement_mock2.__repr__.return_value = "Mouvement 2"
+ order_mock2.mouvements = [
+ mouvement_mock1, mouvement_mock2
+ ]
+ trade.orders.append(order_mock1)
+ trade.orders.append(order_mock2)
+
+ trade.print_with_order()
+
+ print_log.assert_called()
+ calls = print_log.mock_calls
+ self.assertEqual("Trade(0.50000000 BTC [10.00000000 ETH] -> 1.00000000 BTC in ETH, acquire)", str(calls[0][1][0]))
+ self.assertEqual("\tMock 1", str(calls[1][1][0]))
+ self.assertEqual("\tMock 2", str(calls[2][1][0]))
+ self.assertEqual("\t\tMouvement 1", str(calls[3][1][0]))
+ self.assertEqual("\t\tMouvement 2", str(calls[4][1][0]))
+
+ def test__repr(self):
+ value_from = portfolio.Amount("BTC", "0.5")
+ value_from.linked_to = portfolio.Amount("ETH", "10.0")
+ value_to = portfolio.Amount("BTC", "1.0")
+ trade = portfolio.Trade(value_from, value_to, "ETH")
+
+ self.assertEqual("Trade(0.50000000 BTC [10.00000000 ETH] -> 1.00000000 BTC in ETH, acquire)", str(trade))
+
+ def test_as_json(self):
+ value_from = portfolio.Amount("BTC", "0.5")
+ value_from.linked_to = portfolio.Amount("ETH", "10.0")
+ value_to = portfolio.Amount("BTC", "1.0")
+ trade = portfolio.Trade(value_from, value_to, "ETH")
+
+ as_json = trade.as_json()
+ self.assertEqual("acquire", as_json["action"])
+ self.assertEqual(D("0.5"), as_json["from"])
+ self.assertEqual(D("1.0"), as_json["to"])
+ self.assertEqual("ETH", as_json["currency"])
+ self.assertEqual("BTC", as_json["base_currency"])
+
+@unittest.skipUnless("unit" in limits, "Unit skipped")
+class OrderTest(WebMockTestCase):
+ def test_values(self):
+ order = portfolio.Order("buy", portfolio.Amount("ETH", 10),
+ D("0.1"), "BTC", "long", "market", "trade")
+ self.assertEqual("buy", order.action)
+ self.assertEqual(10, order.amount.value)
+ self.assertEqual("ETH", order.amount.currency)
+ self.assertEqual(D("0.1"), order.rate)
+ self.assertEqual("BTC", order.base_currency)
+ self.assertEqual("market", order.market)
+ self.assertEqual("long", order.trade_type)
+ self.assertEqual("pending", order.status)
+ self.assertEqual("trade", order.trade)
+ self.assertIsNone(order.id)
+ self.assertFalse(order.close_if_possible)
+
+ def test__repr(self):
+ order = portfolio.Order("buy", portfolio.Amount("ETH", 10),
+ D("0.1"), "BTC", "long", "market", "trade")
+ self.assertEqual("Order(buy long 10.00000000 ETH at 0.1 BTC [pending])", repr(order))
+
+ order = portfolio.Order("buy", portfolio.Amount("ETH", 10),
+ D("0.1"), "BTC", "long", "market", "trade",
+ close_if_possible=True)
+ self.assertEqual("Order(buy long 10.00000000 ETH at 0.1 BTC [pending] ✂)", repr(order))
+
+ def test_as_json(self):
+ order = portfolio.Order("buy", portfolio.Amount("ETH", 10),
+ D("0.1"), "BTC", "long", "market", "trade")
+ mouvement_mock1 = mock.Mock()
+ mouvement_mock1.as_json.return_value = 1
+ mouvement_mock2 = mock.Mock()
+ mouvement_mock2.as_json.return_value = 2
+
+ order.mouvements = [mouvement_mock1, mouvement_mock2]
+ as_json = order.as_json()
+ self.assertEqual("buy", as_json["action"])
+ self.assertEqual("long", as_json["trade_type"])
+ self.assertEqual(10, as_json["amount"])
+ self.assertEqual("ETH", as_json["currency"])
+ self.assertEqual("BTC", as_json["base_currency"])
+ self.assertEqual(D("0.1"), as_json["rate"])
+ self.assertEqual("pending", as_json["status"])
+ self.assertEqual(False, as_json["close_if_possible"])
+ self.assertIsNone(as_json["id"])
+ self.assertEqual([1, 2], as_json["mouvements"])
+
+ def test_account(self):
+ order = portfolio.Order("buy", portfolio.Amount("ETH", 10),
+ D("0.1"), "BTC", "long", "market", "trade")
+ self.assertEqual("exchange", order.account)
+
+ order = portfolio.Order("sell", portfolio.Amount("ETH", 10),
+ D("0.1"), "BTC", "short", "market", "trade")
+ self.assertEqual("margin", order.account)
+
+ def test_pending(self):
+ order = portfolio.Order("buy", portfolio.Amount("ETH", 10),
+ D("0.1"), "BTC", "long", "market", "trade")
+ self.assertTrue(order.pending)
+ order.status = "open"
+ self.assertFalse(order.pending)
+
+ def test_open(self):
+ order = portfolio.Order("buy", portfolio.Amount("ETH", 10),
+ D("0.1"), "BTC", "long", "market", "trade")
+ self.assertFalse(order.open)
+ order.status = "open"
+ self.assertTrue(order.open)
+
+ def test_finished(self):
+ order = portfolio.Order("buy", portfolio.Amount("ETH", 10),
+ D("0.1"), "BTC", "long", "market", "trade")
+ self.assertFalse(order.finished)
+ order.status = "closed"
+ self.assertTrue(order.finished)
+ order.status = "canceled"
+ self.assertTrue(order.finished)
+ order.status = "error"
+ self.assertTrue(order.finished)
+
+ @mock.patch.object(portfolio.Order, "fetch")
+ @mock.patch("portfolio.ReportStore")
+ def test_cancel(self, report_store, fetch):