-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathmssb_1099b_to_txf.py
executable file
·81 lines (71 loc) · 2.45 KB
/
mssb_1099b_to_txf.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
#!/usr/bin/env python
"""mssb_1099b_to_txf converts simple Morgan Stanley (MSSB) 1099-B PDFs to TXF files."""
import datetime
import re
import subprocess
import sys
# Codes and structure are defined at
# https://www.taxdataexchange.org/txf/txf-spec.html
categories = {
'Short Term – Noncovered Securities': '711',
'Long Term – Noncovered Securities': '713',
}
# Match a section of sales for one sales category.
# The last line can say 'Total Short Term – Noncovered Securities' or
# 'Total Short Term Noncovered Securities' (without the hypen) so match
# only on "^Total".
categories_pattern = '|'.join(categories)
section_expr = re.compile(
r'^('+categories_pattern+r')'
r'(.*?)'
r'^Total', re.DOTALL|re.MULTILINE)
# Fields: RefNumber Description CUSIP Quantity DateAcquired DateSold
# GrossProceeds CostBasis
#
# Example:
# 1234 ALPHABET INC CL C
# 12345A678
# 1.000000 01/01/20 02/01/20 $2,000.00 $1,9999.00
#
# Example:
# 1234 ALPHABET INC CL C
# 12345A678
# 1.000000 VARIOUS 02/01/20 $2,000.00 $1,9999.00
row_expr = re.compile(
r'^(?P<descr>(\w| )+)\s+'
r'(?P<cusip>\w+)\s+'
r'(?P<quantity>\d*\.\d+)\s+'
r'(?P<acquired>(\d+/\d+/\d+|\w+))\s+'
r'(?P<sold>\d+/\d+/\d+)\s+'
r'(?P<proceeds>\$[0-9,.]+)\s+'
r'(?P<cost>\$[0-9,.]+)\s', re.DOTALL|re.MULTILINE)
def parseAndPrintRows(text, entry_code):
for match in row_expr.finditer(text):
print('TD')
print('N' + entry_code)
print('C1')
print('L1')
print('P' + match.group('descr'))
print('D' + match.group('acquired'))
print('D' + match.group('sold'))
# These have a leading dollar sign.
print(match.group('cost'))
print(match.group('proceeds'))
print("$") # Wash sale. Leaving blank. They aren't handled here.
print('^')
def parse_sections(text):
return section_expr.finditer(text)
def main():
if len(sys.argv) != 2:
sys.exit(f'Usage: {sys.argv[0]} path-to-1099b-pdf')
text = subprocess.check_output(['pdftotext', '-raw', sys.argv[1], '-']).decode()
print('V042')
print('A mssb_1099b_to_txf')
print('D ' + datetime.datetime.now().strftime('%m/%d/%Y'))
print('^')
for section_match in parse_sections(text):
entry_code = categories[section_match.group(1)]
contents = section_match.group(2)
parseAndPrintRows(contents, entry_code)
if __name__ == '__main__':
main()