Skip to content

Commit

Permalink
Update Schwab Equity Award parser to support Schwab's updated JSON fo…
Browse files Browse the repository at this point in the history
…rmat. (#467)

Co-authored-by: Ruslan Sayfutdinov <[email protected]>
  • Loading branch information
thibwk and KapJI authored Jan 29, 2024
1 parent d1864b5 commit 325a7ea
Show file tree
Hide file tree
Showing 5 changed files with 311 additions and 43 deletions.
2 changes: 1 addition & 1 deletion README.md
Original file line number Diff line number Diff line change
Expand Up @@ -49,7 +49,7 @@ You will need several input files:
- Exported transaction history from Schwab in CSV format since the beginning.
Or at least since you first acquired the shares, which you were holding during the tax year. Schwab only allows to download transaction for the last 4 years so keep it safe. After that you may need to restore transactions from PDF statements.
[See example](https://github.com/KapJI/capital-gains-calculator/blob/main/tests/test_data/schwab_transactions.csv).
- Exported transaction history from Schwab Equity Awards (e.g. for Alphabet/Google employees) since the beginning. These are to be downloaded in JSON format. Instructions are available at the top of the [parser file](../main/cgt_calc/parsers/schwab_equity_award_json.py).
- Exported transaction history from Schwab Equity Awards (e.g. for Alphabet/Google employees) since the beginning (Note: Schwab now allows for the whole history of Equity Awards account transactions to be downloaded). These are to be downloaded in JSON format. Instructions are available at the top of the [parser file](../main/cgt_calc/parsers/schwab_equity_award_json.py).
- Exported transaction history from Trading 212.
You can use several files here since Trading 212 limit the statements to 1 year periods.
[See example](https://github.com/KapJI/capital-gains-calculator/tree/main/tests/test_data/trading212).
Expand Down
159 changes: 120 additions & 39 deletions cgt_calc/parsers/schwab_equity_award_json.py
Original file line number Diff line number Diff line change
@@ -1,16 +1,20 @@
"""Charles Schwab Equity Award JSON export parser.
To get the data from Schwab:
1. Open https://client.schwab.com/Apps/accounts/transactionhistory/#/
1. Open https://client.schwab.com/app/accounts/history/#/
2. Make sure Equity Award Center is selected
3. Select date range ALL and click SEARCH
4. In chrome devtools, look for an API call to
https://ausgateway.schwab.com/api/is.TransactionHistoryWeb/TransactionHistoryInterface/TransactionHistory/equity-award-center/transactions
5. Copy response JSON inside schwab_input.json and run schwab.py
3. Select date range 'Previous 4 Years' and click SEARCH
4. At the top right, click on Export and select type JSON
5. If you have had Equity Awards for more than 4 years, good news:
Schwab now allows to export all the data history (which you do need
to calculate the CGT). In that case:
* repeat the process to export older data
* manually combine the data into a single file
"""

from __future__ import annotations

from dataclasses import InitVar, dataclass
import datetime
from decimal import Decimal
import json
Expand All @@ -28,6 +32,58 @@
# Delay between a (sale) trade, and when it is settled.
SETTLEMENT_DELAY = 2 * CustomBusinessDay(calendar=USFederalHolidayCalendar())

OPTIONAL_DETAILS_NAME = "Details"

field2schema = {"transactions": 1, "Transactions": 2}


@dataclass
class FieldNames:
"""Names of the fields in the Schwab JSON data, depending on the schema version."""

# Note that the schema version is not an official Schwab one, just something
# we use internally in this code:
schema_version: InitVar[int] = 2

transactions: str = "Transactions"
description: str = "Description"
action: str = "Action"
symbol: str = "Symbol"
quantity: str = "Quantity"
amount: str = "Amount"
fees: str = "FeesAndCommissions"
transac_details: str = "TransactionDetails"
shares: str = "Shares"
vest_date: str = "VestDate"
vest_fair_market_value: str = "VestFairMarketValue"
award_date: str = "AwardDate"
award_id: str = "AwardId"
date: str = "Date"
sale_price: str = "SalePrice"

def __post_init__(self, schema_version: int) -> None:
"""Set correct field names if the schema is not the default one.
Automatically run on object initialization.
"""
if schema_version == 1:
self.transactions = "transactions"
self.description = "description"
self.action = "action"
self.symbol = "symbol"
self.quantity = "quantity"
self.amount = "amount"
self.fees = "totalCommissionsAndFees"
self.transac_details = "transactionDetails"
self.shares = "shares"
self.vest_date = "vestDate"
self.vest_fair_market_value = "vestFairMarketValue"
self.award_date = "awardDate"
self.award_id = "awardName"
self.date = "eventDate"
self.sale_price = "salePrice"


# We want enough decimals to cover what Schwab gives us (up to 4 decimals)
# divided by the share-split factor (20), so we keep 6 decimals.
# We don't want more decimals than necessary or we risk converting
Expand Down Expand Up @@ -132,46 +188,45 @@ def _is_integer(number: Decimal) -> bool:
class SchwabTransaction(BrokerTransaction):
"""Represent single Schwab transaction."""

def __init__(
self,
row: JsonRowType,
file: str,
) -> None:
def __init__(self, row: JsonRowType, file: str, field_names: FieldNames) -> None:
"""Create a new SchwabTransaction from a JSON row."""
description = row["description"]
self.raw_action = row["action"]
names = field_names
description = row[names.description]
self.raw_action = row[names.action]
action = action_from_str(self.raw_action)
symbol = row.get("symbol")
symbol = row.get(names.symbol)
symbol = TICKER_RENAMES.get(symbol, symbol)
quantity = _decimal_from_number_or_str(row, "quantity")
amount = _decimal_from_number_or_str(row, "amount")
fees = _decimal_from_number_or_str(row, "totalCommissionsAndFees")
if row["action"] == "Deposit":
if len(row["transactionDetails"]) != 1:
quantity = _decimal_from_number_or_str(row, names.quantity)
amount = _decimal_from_number_or_str(row, names.amount)
fees = _decimal_from_number_or_str(row, names.fees)
if row[names.action] == "Deposit":
if len(row[names.transac_details]) != 1:
raise ParsingError(
file,
"Expected a single transactionDetails for a Deposit, but "
f"found {len(row['transactionDetails'])}",
"Expected a single Transaction Details for a Deposit, but "
f"found {len(row[names.transac_details])}",
)
if OPTIONAL_DETAILS_NAME in row[names.transac_details][0]:
details = row[names.transac_details][0]["Details"]
else:
details = row[names.transac_details][0]
date = datetime.datetime.strptime(
row["transactionDetails"][0]["vestDate"], "%m/%d/%Y"
details[names.vest_date], "%m/%d/%Y"
).date()
# Schwab only provide this one as string:
price = _decimal_from_str(
row["transactionDetails"][0]["vestFairMarketValue"]
)
price = _decimal_from_str(details[names.vest_fair_market_value])
if amount == Decimal(0):
amount = price * quantity
description = (
f"Vest from Award Date "
f'{row["transactionDetails"][0]["awardDate"]} '
f'(ID {row["transactionDetails"][0]["awardName"]})'
f"{details[names.award_date]} "
f"(ID {details[names.award_id]})"
)
elif row["action"] == "Sale":
elif row[names.action] == "Sale":
# Schwab's data export shows the settlement date,
# whereas HMRC wants the trade date:
date = (
datetime.datetime.strptime(row["eventDate"], "%m/%d/%Y").date()
datetime.datetime.strptime(row[names.date], "%m/%d/%Y").date()
- SETTLEMENT_DELAY
).date() # type: ignore[attr-defined]

Expand All @@ -184,10 +239,17 @@ def __init__(
subtransac_have_quantities = True
subtransac_shares_sum = Decimal() # Decimal 0
found_share_decimals = False
for subtransac in row["transactionDetails"]:

details = row[names.transac_details][0].get(
OPTIONAL_DETAILS_NAME, row[names.transac_details][0]
)

for subtransac in row[names.transac_details]:
subtransac = subtransac.get(OPTIONAL_DETAILS_NAME, subtransac)

if "shares" in subtransac:
# Schwab only provides this one as a string:
shares = _decimal_from_str(subtransac["shares"])
shares = _decimal_from_str(subtransac[names.shares])
subtransac_shares_sum += shares
if not _is_integer(shares):
found_share_decimals = True
Expand All @@ -203,11 +265,18 @@ def __init__(
# amount, and sale price of the sub-transactions.
# We can only work-out the correct quantity if all
# sub-transactions have the same price:
price_str = row["transactionDetails"][0]["salePrice"]

first_subtransac = row[names.transac_details][0]
first_subtransac = first_subtransac.get(
OPTIONAL_DETAILS_NAME, first_subtransac
)
price_str = first_subtransac[names.sale_price]
price = _decimal_from_str(price_str)

for subtransac in row["transactionDetails"][1:]:
if subtransac["salePrice"] != price_str:
for subtransac in row[names.transac_details][1:]:
subtransac = subtransac.get(OPTIONAL_DETAILS_NAME, subtransac)

if subtransac[names.sale_price] != price_str:
raise ParsingError(
file,
"Impossible to work out quantity of sale of "
Expand All @@ -217,9 +286,10 @@ def __init__(
)

quantity = (amount + fees) / price

else:
raise ParsingError(
file, f'Parsing for action {row["action"]} is not implemented!'
file, f"Parsing for action {row[names.action]} is not implemented!"
)

currency = "USD"
Expand Down Expand Up @@ -276,18 +346,29 @@ def read_schwab_equity_award_json_transactions(
"Cloud not parse content as JSON",
) from exception

if "transactions" not in data or not isinstance(data["transactions"], list):
for field_name, schema_version in field2schema.items():
if field_name in data:
fields = FieldNames(schema_version)
break
if not fields:
raise ParsingError(
transactions_file,
f"Expected top level field ({', '.join(field2schema.keys())}) "
"not found: the JSON data is not in the expected format",
)

if not isinstance(data[fields.transactions], list):
raise ParsingError(
transactions_file,
"no 'transactions' list found: the JSON data is not "
f"'{fields.transactions}' is not a list: the JSON data is not "
"in the expected format",
)

transactions = [
SchwabTransaction(transac, transactions_file)
for transac in data["transactions"]
SchwabTransaction(transac, transactions_file, fields)
for transac in data[fields.transactions]
# Skip as not relevant for CGT
if transac["action"] not in {"Journal", "Wire Transfer"}
if transac[fields.action] not in {"Journal", "Wire Transfer"}
]
transactions.reverse()
return list(transactions)
Expand Down
File renamed without changes.
144 changes: 144 additions & 0 deletions tests/test_data/schwab_equity_award_v2.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,144 @@
{
"FromDate": "01/01/2020",
"ToDate": "12/23/2023",
"Transactions": [
{
"Date": "09/27/2023",
"Action": "Deposit",
"Symbol": "GOOG",
"Quantity": "13.6",
"Description": "RS",
"FeesAndCommissions": null,
"DisbursementElection": null,
"Amount": null,
"TransactionDetails": [
{
"Details": {
"AwardDate": "01/01/2019",
"AwardId": "C987654",
"VestDate": "09/25/2023",
"VestFairMarketValue": "$131.25"
}
}
]
},
{
"Date": "09/27/2023",
"Action": "Deposit",
"Symbol": "GOOG",
"Quantity": "4.911",
"Description": "RS",
"FeesAndCommissions": null,
"DisbursementElection": null,
"Amount": null,
"TransactionDetails": [
{
"Details": {
"AwardDate": "03/01/2020",
"AwardId": "C1234567",
"VestDate": "09/25/2023",
"VestFairMarketValue": "$131.25"
}
}
]
},
{
"Date": "09/01/2023",
"Action": "Journal",
"Symbol": "GOOG",
"Quantity": null,
"Description": "Journal To Account ...123",
"FeesAndCommissions": "",
"DisbursementElection": null,
"Amount": "-$1,382.75",
"TransactionDetails": []
},

{
"Date": "08/31/2023",
"Action": "Sale",
"Symbol": "GOOG",
"Quantity": "14.40",
"Description": "Share Sale",
"FeesAndCommissions": "$0.02",
"DisbursementElection": "Journal",
"Amount": "$1,985.74",
"TransactionDetails": [
{
"Details": {
"Type": "RS",
"Shares": "14",
"SalePrice": "$137.90",
"SubscriptionDate": "",
"SubscriptionFairMarketValue": "",
"PurchaseDate": "",
"PurchasePrice": "",
"PurchaseFairMarketValue": "",
"DispositionType": null,
"GrantId": "C1234567",
"VestDate": "07/25/2023",
"VestFairMarketValue": "$121.88",
"GrossProceeds": "$1,930.60"
}
},
{
"Details": {
"Type": "RS",
"Shares": "0.40",
"SalePrice": "$137.90",
"SubscriptionDate": "",
"SubscriptionFairMarketValue": "",
"PurchaseDate": "",
"PurchasePrice": "",
"PurchaseFairMarketValue": "",
"DispositionType": null,
"GrantId": "C1234567",
"VestDate": "07/25/2023",
"VestFairMarketValue": "$121.88",
"GrossProceeds": "$55.16"
}
}
]
},
{
"Date": "04/27/2023",
"Action": "Deposit",
"Symbol": "GOOG",
"Quantity": "13.6",
"Description": "RS",
"FeesAndCommissions": null,
"DisbursementElection": null,
"Amount": null,
"TransactionDetails": [
{
"Details": {
"AwardDate": "01/01/2019",
"AwardId": "C987654",
"VestDate": "04/25/2023",
"VestFairMarketValue": "$106.78"
}
}
]
},
{
"Date": "04/27/2023",
"Action": "Deposit",
"Symbol": "GOOG",
"Quantity": "4.911",
"Description": "RS",
"FeesAndCommissions": null,
"DisbursementElection": null,
"Amount": null,
"TransactionDetails": [
{
"Details": {
"AwardDate": "03/01/2020",
"AwardId": "C1234567",
"VestDate": "04/25/2023",
"VestFairMarketValue": "$106.78"
}
}
]
}
]
}
Loading

0 comments on commit 325a7ea

Please sign in to comment.