Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Kraken Margin Trading #120

Open
wants to merge 19 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 7 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -54,7 +54,7 @@ install:
pip install -r requirements.txt

install-dev: install
pip -r requirements-dev.txt
pip install -r requirements-dev.txt

# Setup virtual environment
venv:
Expand Down
73 changes: 73 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -147,3 +147,76 @@ Weitere Besonderheiten die sich so nicht im BMF-Schreiben wiederfinden, sind im
Bei Binance gibt es die Möglichkeit, andere Personen über einen Link zu werben.
Bei jeder Transaktion der Person erhält man einen kleinen Anteil derer Gebühren als Reward gutgeschrieben.
Die Einkünfte durch diese Rewards werden durch CoinTaxman als Einkünfte aus sonstigen Leistungen ausgewiesen und damit wie eine übliche Kunden-werben-Kunden-Prämie erfasst.

### Future- / Margin-Trading

#### Unterscheidung Veräußerungsgeschäft / Termingeschäft

> Gewinne aus Future-Trades stellen in der Regel Einkünfte aus Kapitalvermögen dar und unterliegen damit der Kapitalertragsteuer. Maßgebend für die steuerliche Beurteilung ist allerdings weniger die von der Börse gewählte Begrifflichkeit, sondern vielmehr die konkrete Ausgestaltung des angebotenen Finanzprodukts. Im Einzelfall kann deshalb unter Umständen auch bei Futures ein privates Veräußerungsgeschäft gemäß § 23 EStG vorliegen, das zu einer Besteuerung nach dem persönlichen Einkommensteuersatz führt. Im Kern kommt es für die Abgrenzung darauf an, ob das Geschäft wie beim Spot Trading auf die Lieferung einer Kryptowährung abzielt (dann ist § 23 EStG einschlägig) oder ob die Lieferung lediglich einen Differenzausgleich darstellt (dann liegen Kapitaleinkünfte gemäß § 23 Abs. 2 Satz. 1 Nr. 3 EStG vor). [...]
>
> Parallel dazu lassen sich die Überlegungen auf das Margin Trading übertragen, weshalb Gewinne aus Margin Trades immer nur dann unter Kapitaleinkünfte (§ 20 EStG) fallen, wenn keine Lieferung einer Kryptowährung, sondern ein Differenzausgleich durchgeführt wird. Kommt es hingegen zu einer Lieferung einer Kryptowährung, liegt ein privates Veräußerungsgeschäft gemäß § 23 Abs. 1 Satz 1 Nr. 2 EStG vor.
>
> [...]
>
> Stammen die erhaltenen Bitcoins aus einer Auszahlung resultierend aus einem Differenzausgleich, ist anschließend eine steuerfreie Veräußerung möglich. Da die Gewinne in Bitcoin bereits nach Maßgabe der Kapitalertragsteuer gemäß § 20 EStG versteuert wurden, greift auch keine Jahresfrist.
>
> [...]
>
> Solange die Position offen ist, muss diese auch nicht versteuert werden. Erst ab dem Zeitpunkt, in dem Investoren tatsächlich Einnahmen zugeflossen sind, findet die Besteuerung statt (sog. Zufluss-/Abflussprinzip).
>
> [...]
>
> Entstandene Gebühren fallen unter die sog. Werbungskosten. Da in den meisten Fällen beim Future Trading Kapitaleinkünfte vorliegen, sind die Werbungskosten bereits durch den Pauschbetrag in Höhe von 801 Euro (bzw. 1.602 Euro für verheiratete Paare) abgegolten. Die Gebühren können deshalb nicht gesondert steuerlich geltend gemacht werden, um die Steuerlast zu mindern.

[Quelle](https://winheller.com/blog/besteuerung-future-margin-trading/)
[Wörtlich zitiert vom 18.02.2022]

#### Fallbeispiel

> Person A schließt mit Person B einen Vertrag, der A das Recht einräumt, in Zukunft einen BTC von B zu erhalten, der momentan 35.000 Euro wert ist. Wird der Kontrakt fällig und A erhält von B den BTC, liegt ein Fall von § 23 Abs. 1 Satz 1 Nr. 2 EStG vor. Das heißt für B, dass er die Veräußerung des BTC mit seinem persönlichen Einkommensteuersatz versteuern muss. Für A bedeutet das hingegen, dass eine Anschaffung vorliegt und damit die Jahresfrist gilt.
>
> Treffen A und B hingegen im oben geschilderten Fall die Vereinbarung, dass A am Ende des Vertrages die Wahl hat, ob er einen BTC bekommt oder alternativ den Gegenwert der Differenz zum aktuellen Kurs, dann liegt ein Termingeschäft gem. § 20 Abs. 2 Satz 1 Nr. 3 EStG vor. Das gilt auch dann, wenn die Differenz in BTC gezahlt wird. Steigt der Kurs von BTC am Ende des Kontrakts auf 37.000 Euro an, erhält A den Gegenwert der Differenz (37.000 Euro – 35.000 Euro = 2.000 Euro) in BTC. Der Gewinn muss von A pauschal mit 25 Prozent Kapitalertragsteuer versteuert werden, die Jahresfrist aus § 23 EStG greift hingegen nicht. Gegebenenfalls liegt bei B ein privates Veräußerungsgeschäft i.S.v. § 23 EStG vor.

[Quelle](https://hub.accointing.com/crypto-tax-regulations/germany/tax-break-germany-derivate-und-futures-winheller)
[Wörtlich zitiert vom 18.02.2022]

#### Werbungskosten für Termingeschäfte

> Ab dem VZ 2009 ist als Werbungskosten ein Betrag von 801 € bzw. 1 602 € bei Zusammenveranlagung abzuziehen (Sparer-Pauschbetrag, § 20 Abs. 9 EStG); der Abzug der tatsächlichen Werbungskosten ist ausgeschlossen. Die früheren Regelungen zum Werbungskosten Pauschbetrag und Sparer-Freibetrag wurden mit Einführung der Abgeltungsteuer aufgehoben.
>
> [...]
>
> In folgenden Fällen sind die Kosten auch ab 2009 weiterhin abzugsfähig:
> - Veräußerungskosten und Kosten in Zusammenhang mit Termingeschäften werden bei der Veräußerungsgewinnermittlung nach § 20 Abs. 4 EStG berücksichtigt.
>
> [...]
>

[Quelle](https://datenbank.nwb.de/Dokument/97088/)
[Wörtlich zitiert vom 18.02.2022]

#### Verrechnung von Verlusten aus Termingeschäften

> Während es vor dem 01.01.2021 möglich war, Verluste aus Termingeschäften uneingeschränkt mit den Einkünften aus Kapitalvermögen zu verrechnen, ist dies aufgrund des neu eingeführten § 20 Abs. 6 Satz 5 EStG seit 2021 nicht mehr ohne Weiteres möglich:
> 1. Verluste dürfen nur noch mit Gewinnen aus Termingeschäften und mit Erträgen aus Stillhaltergeschäften verrechnet werden.
> 2. Außerdem ist die Verlustverrechnung auf 20.000 Euro jährlich begrenzt.
>
> Zwar können die nicht verrechneten Verluste in die Folgejahre vorgetragen werden. Aber auch dann ist eine Verlustverrechnung der Höhe nach auf 20.000 Euro pro Jahr begrenzt. Das führt faktisch zu einer Mindestbesteuerung von Gewinnen.

[Quelle](https://www.winheller.com/bankrecht-finanzrecht/bitcointrading/bitcoinundsteuer/verlustverrechnung.html)
[Wörtlich zitiert vom 18.02.2022]

#### Zusammenfassung

Zusammenfassung der Besteuerung des Margin-Tradings in meinen Worten:
- Gewinne/Verluste werden besteuert, sobald die Margin-Positionen ausgeglichen bzw. geschlossen werden
- Wird eine Margin-Position ausgeglichen ("settled"), d.h. die Kryptowährung wird zu Vertragsende zum Startpreis ge-/verkauft, liegt ein privates Veräußerungsgeschäft vor
- Für private Veräußerungsgeschäfte gelten die oben angeführten Regeln, inklusive der einjährigen Haltefrist
- Wird eine Margin-Position geschlossen ("closed", Differenzausgleich), liegt ein Termingeschäft vor
- Die Gewinne bzw. Verluste fallen unter Kapitaleinkünfte (§ 20 EStG)
- Erhaltene Kryptowährung aus Differenzausgleichen kann steuerfrei veräußert werden
- Es gibt keine einjährige Haltefrist
- Gebühren können nur abgezogen werden, wenn der Freibetrag von 801 / 1602 Euro bereits ausgeschöpft wird
- Die Verlustrechnung ist auf 20.000 Euro jährlich begrenzt und darf nicht mit Gewinnen aus privaten Veräußerungsgeschäften verrechnet werden
- Steht es dem Investor bis zum Ende offen, ob eine Margin-Position ausgeglichen ("settled") oder geschlossen ("closed") werden kann, liegt automatisch ein Termingeschäft vor und es gelten die gleichen Regelungen wie für geschlossene Positionen. Dies trifft für folgende Börsen zu:
- Kraken
4 changes: 3 additions & 1 deletion src/balance_queue.py
Original file line number Diff line number Diff line change
Expand Up @@ -41,7 +41,9 @@ def not_sold(self) -> decimal.Decimal:
"""
not_sold = self.op.change - self.sold
# If the left over amount is <= 0, this coin shouldn't be in the queue.
assert not_sold > 0, f"{not_sold=} should be > 0"
assert (
not_sold > 0
), f"{not_sold=} {self.op.coin} should be > 0 ({self.op.type_name})"
return not_sold


Expand Down
51 changes: 40 additions & 11 deletions src/book.py
Original file line number Diff line number Diff line change
Expand Up @@ -537,6 +537,7 @@ def _read_kraken_ledgers(self, file_path: Path) -> None:
"staking": "StakingInterest",
"deposit": "Deposit",
"withdrawal": "Withdrawal",
"rollover": "MarginFee",
}

with open(file_path, encoding="utf8") as f:
Expand Down Expand Up @@ -597,12 +598,24 @@ def _read_kraken_ledgers(self, file_path: Path) -> None:
if operation is None:
if _type == "trade":
operation = "Sell" if change < 0 else "Buy"
elif _type in ["margin trade", "rollover", "settled", "margin"]:
log.error(
f"{file_path} row {row}: Margin trading is currently not "
"supported. Please create an Issue or PR."
)
raise RuntimeError
elif _type == "margin":
# Margin positions for Kraken always fall under income from
# capital, as the user can decide until the end if it is closed
# or settled. "rollover" entries contain margin fees between
# start and end. The start of a margin position is denoted with
# a "margin" entry with zero change. For closed positions, the
# end is marked with a "margin" entry containing the net
# gain/loss of the position.
# if the change is zero, consider only the fees
if change == 0:
operation = "MarginFee"
elif change > 0:
operation = "MarginGain"
else:
operation = "MarginLoss"
elif _type == "settled":
# "settled" entries mark the end of settled positions
operation = "MarginGain" if change > 0 else "MarginLoss"
elif _type == "transfer":
if num_columns == 9:
# for backwards compatibility assume Airdrop for staking
Expand Down Expand Up @@ -636,7 +649,20 @@ def _read_kraken_ledgers(self, file_path: Path) -> None:
# Validate data.
assert operation
assert coin
assert change

# Margin trading: Add operations and fees to list
if operation in ["MarginFee", "MarginGain", "MarginLoss"]:
if operation == "MarginFee":
assert change == 0, "Margin fee should be only contain fee"
if change:
# add margin gain/losses to operation list
self.append_operation(
operation, utc_time, platform, change, coin, row, file_path
)
if fee != 0:
self.append_operation(
"MarginFee", utc_time, platform, fee, coin, row, file_path
)

# Skip duplicate entries for deposits / withdrawals and additional
# deposit / withdrawal lines for staking / unstaking / staking reward
Expand All @@ -652,7 +678,8 @@ def _read_kraken_ledgers(self, file_path: Path) -> None:
# == None: Initial value (first occurrence)
# == False: No operation has been appended (second occurrence)
# == True: Operation has already been appended, this should not happen
if operation in ["Deposit", "Withdrawal"]:
elif operation in ["Deposit", "Withdrawal"]:
assert change
# First, create the operations
op = self.create_operation(
operation, utc_time, platform, change, coin, row, file_path
Expand Down Expand Up @@ -748,6 +775,7 @@ def _read_kraken_ledgers(self, file_path: Path) -> None:

# for all other operation types
else:
assert change
self.append_operation(
operation, utc_time, platform, change, coin, row, file_path
)
Expand Down Expand Up @@ -1576,9 +1604,10 @@ def match_fees(self) -> None:
log.warning(
"Fee matching is not implemented for this case. "
"Your fees will be discarded and are not evaluated in "
"the tax evaluation.\n"
"Please create an Issue or PR.\n\n"
f"{matching_operations=}\n{fees=}"
"the tax evaluation. "
"Please create an Issue or PR. "
f"Found {len(matching_operations)} matching operations:\n"
f"{matching_operations=}\n{fees=}\n"
)

def resolve_trades(self) -> None:
Expand Down
52 changes: 51 additions & 1 deletion src/taxman.py
Original file line number Diff line number Diff line change
Expand Up @@ -170,7 +170,9 @@ def get_buy_cost(self, sc: tr.SoldCoin) -> decimal.Decimal:
"previously sold coins of the trade. "
"The calculated buy cost might be wrong. "
"This may lead to a false tax evaluation.\n"
f"{sc.op}"
f"{sc.op.type_name} {sc.op.change} {sc.op.coin} @ "
f"{sc.op.platform} {sc.op.utc_time}, "
f"row(s) {sc.op.line} of {sc.op.file_path.name}"
)
buy_value = self.price_data.get_cost(sc)
else:
Expand Down Expand Up @@ -475,6 +477,54 @@ def _evaluate_taxation_GERMANY(self, op: tr.Operation) -> None:
)
self.tax_report_entries.append(report_entry)

elif isinstance(op, tr.MarginFee):
# Fees for margin trading
self.remove_from_balance(op)
if in_tax_year(op):
taxation_type = "Kapitaleinkünfte aus Margin Trading"
Griffsano marked this conversation as resolved.
Show resolved Hide resolved
report_entry = tr.MarginReportEntry(
platform=op.platform,
amount=op.change,
Griffsano marked this conversation as resolved.
Show resolved Hide resolved
coin=op.coin,
utc_time=op.utc_time,
interest_in_fiat=-self.price_data.get_cost(op),
taxation_type=taxation_type,
remark=op.remark,
Griffsano marked this conversation as resolved.
Show resolved Hide resolved
)
self.tax_report_entries.append(report_entry)

elif isinstance(op, tr.MarginGain):
# Gains from margin trading
self.add_to_balance(op)
if in_tax_year(op):
taxation_type = "Kapitaleinkünfte aus Margin Trading"
report_entry = tr.MarginReportEntry(
platform=op.platform,
amount=op.change,
coin=op.coin,
utc_time=op.utc_time,
interest_in_fiat=self.price_data.get_cost(op),
taxation_type=taxation_type,
remark=op.remark,
)
self.tax_report_entries.append(report_entry)

elif isinstance(op, tr.MarginLoss):
# Losses from margin trading
self.remove_from_balance(op)
if in_tax_year(op):
taxation_type = "Kapitaleinkünfte aus Margin Trading"
report_entry = tr.MarginReportEntry(
platform=op.platform,
amount=op.change,
Griffsano marked this conversation as resolved.
Show resolved Hide resolved
coin=op.coin,
utc_time=op.utc_time,
interest_in_fiat=-self.price_data.get_cost(op),
taxation_type=taxation_type,
remark=op.remark,
)
self.tax_report_entries.append(report_entry)

elif isinstance(op, tr.Airdrop):
# Depending on how you received the coins, the taxation varies.
# If you didn't "do anything" to get the coins, the airdrop counts
Expand Down
24 changes: 24 additions & 0 deletions src/transaction.py
Original file line number Diff line number Diff line change
Expand Up @@ -179,6 +179,26 @@ class Sell(Transaction):
selling_value: Optional[decimal.Decimal] = None


class MarginFee(Transaction):
"""Fees for margin trading"""

pass


class MarginGain(Transaction):
"""Gains from margin trading.
This is already a taxable value, no buy/sell calculation required."""

pass


class MarginLoss(Transaction):
"""Losses from margin trading.
This is already a taxable value, no buy/sell calculation required."""

pass
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

When possible, it might be nicer to have classes MarginStart, MarginFee, MarginClose instead.
MarginClose should have a fees: list[MarginFee] variable. The fees must be linked to the MarginClose for tax evaluation.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The thing is that Kraken logs do not provide starts and closes of margins. Instead, the already computed gains or losses are logged. Therefore, we don't have to / can't compute the gains/losses ourselves.
An example can be found here: #97



class CoinLendInterest(Transaction):
pass

Expand Down Expand Up @@ -744,6 +764,10 @@ class StakingInterestReportEntry(InterestReportEntry):
event_type = "Staking Einkünfte"


class MarginReportEntry(InterestReportEntry):
event_type = "Margin-Trading"
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Does the labels from InterestReportEntry make sense here? All your MarginGain/MarginLoss.change are positive. So that the report should state that you always receive ("Erhalten am") coins. I believe that you have to update the labels. Also "Wert in EUR" doesn't make sense as we do not evaluate the value of the "bought" coins.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Changed to TaxReportEntry.
"Wert in EUR" makes sense for gains/losses/fees received/paid in crypto coins. However, since it's the same as "Gewinn/Verlust in EUR", I removed the column.
Not sure about the date (as it could be gain or loss), what do you think about "Erhalten oder ausgegeben am"?



class AirdropReportEntry(TaxReportEntry):
event_type = "Airdrops"

Expand Down