-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathprocess_empower_transactions.py
123 lines (103 loc) · 4.03 KB
/
process_empower_transactions.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
"""
Utilities to inspect and manipulate transactions exported by Empower to
make them work with the tools to analyze mint transaction data
"""
import pandas as pd
import numpy as np
import sys
import expenses_config as ec
# Replace the category with the label for a given row
def use_tag_as_category(row, skip_categories=[]):
if row["Category"] in skip_categories:
print(
f"Skipping {row['Date']}: {row['Description']} of {row['Amount']}"
f"for {row['Category']}..."
)
elif not row.isna()["Labels"]:
if ", " == row["Labels"]:
print("Found improperly split transaction")
print(
f"{row['Date']}: {row['Description']} of {row['Amount']} for "
f"{row['Category']}..."
)
print("Please fix it manually and rerun")
sys.exit(0)
if "," in row["Labels"]:
print(
f"Multiple tags found for {row['Date']}: {row['Description']} "
f"of {row['Amount']}"
)
print("Please clean data")
sys.exit(0)
row["Empower Category"] = row["Category"]
row["Category"] = row["Labels"]
return row
def empower_to_mint_format(empower_transactions):
"""
Reads an export of transaction data from Empower, ie:
https://home.personalcapital.com/page/login/app#/all-transactions
and returns a dataframe modified to match the format and column
names used when transaction data is exported from mint
"""
parse_dates = ["Date"]
try:
# Read CSV files into dataframes
empower_df = pd.read_csv(empower_transactions, parse_dates=parse_dates)
# Remove empty rows
empower_df = empower_df.dropna(how="all")
except BaseException as e:
raise ValueError(
f"Failed to read Transaction data from {empower_transactions}: {e}"
)
# Validate required columns exist
required_cols = {"Date", "Account", "Description", "Amount", "Category", "Tags"}
if not required_cols.issubset(set(empower_df.columns)):
raise ValueError("Missing required columns empower transaction data")
# Convert Amount column to float
print(f"Converting {len(empower_df)-1} Empower transactions to mint format")
empower_df["Amount"] = empower_df["Amount"].astype(float)
# Convert Empower data to Mint format
empower_df = empower_df.rename(
columns={"Account": "Account Name", "Tags": "Labels"}
)
empower_df["Transaction Type"] = np.where(
empower_df["Amount"] >= 0, "credit", "debit"
)
empower_df["Amount"] = empower_df["Amount"].abs()
# Empower supports few categories than Mint so, if enabled override the
# Category with the label
if hasattr(ec, "USE_EMPOWER_LABELS"):
if hasattr(ec, "SKIP_CATEGORIES"):
empower_df = empower_df.apply(
use_tag_as_category, args=(ec.SKIP_CATEGORIES), axis=1
)
else:
empower_df = empower_df.apply(use_tag_as_category, axis=1)
# Index on the date
empower_df.set_index(["Date"], inplace=True)
return empower_df
def compare_csv_files(file1, file2, file3):
"""
Find all the values in the "Category" column of file1
that don't exists in file2 or file3
"""
df1 = pd.read_csv(file1)
df2 = pd.read_csv(file2)
df3 = pd.read_csv(file3)
# Read in all the values in the Category column of file1
categories_file1 = set(df1["Category"])
# Read in all the column names and values from file2 and file3
categories_file2 = set(df2.columns).union(set(df2.values.flatten()))
categories_file3 = set(df3.columns).union(set(df3.values.flatten()))
# return a list of categories not in file2 or file3
categories_not_in_file2_or_file3 = (
categories_file1 - categories_file2 - categories_file3
)
return list(categories_not_in_file2_or_file3)
def main():
"""
Read in empower data
"""
empower_to_mint_format(ec.PATH_TO_NEW_TRANSACTIONS)
if __name__ == "__main__":
main()