-
Notifications
You must be signed in to change notification settings - Fork 73
/
Copy pathdynamodb_to_csv.py
executable file
·174 lines (143 loc) · 5.47 KB
/
dynamodb_to_csv.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
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
#!/usr/bin/env python3
"""
dynamodb_to_csv.py
==================
Python3 / boto3 script to dump all data in a DynamoDB table to CSV (or JSON).
Requirements
------------
* Python 3.4+
* boto3
Canonical Source
----------------
https://github.com/jantman/misc-scripts/blob/master/dynamodb_to_csv.py
License
-------
Copyright 2019 Jason Antman <jason@jasonantman.com> <http://www.jasonantman.com>
Free for any use provided that patches are submitted back to me.
CHANGELOG
---------
2019-08-01 Jason Antman <jason@jasonantman.com>:
- serialize Decimals as floats
2019-03-18 Jason Antman <jason@jasonantman.com>:
- fix bug in limiting fields
2019-03-03 Jason Antman <jason@jasonantman.com>:
- ability to output to JSON instead
- ability to load to DynamoDB from JSON
- handle dynamodb-local via ``DYNAMO_ENDPOINT`` environment variable
2019-02-20 Jason Antman <jason@jasonantman.com>:
- initial version of script
"""
import sys
import os
import argparse
import logging
import csv
import io
import json
from decimal import Decimal
try:
import boto3
except ImportError:
sys.stderr.write("ERROR: You must 'pip install boto3'")
raise SystemExit(1)
FORMAT = '%(asctime)s %(levelname)s:%(name)s:%(message)s'
logging.basicConfig(level=logging.INFO, format=FORMAT)
logger = logging.getLogger()
for lname in ['urllib3', 'boto3', 'botocore']:
l = logging.getLogger(lname)
l.setLevel(logging.WARNING)
l.propagate = True
class CustomEncoder(json.JSONEncoder):
def default(self, o):
if isinstance(o, Decimal):
return float(str(o))
# Let the base class default method raise the TypeError
return json.JSONEncoder.default(self, o)
class DynamoDumper(object):
def __init__(self):
logger.debug('Connecting to DynamoDB')
kwargs = {}
if 'AWS_DEFAULT_REGION' in os.environ:
kwargs['region_name'] = os.environ['AWS_DEFAULT_REGION']
if 'DYNAMO_ENDPOINT' in os.environ:
kwargs['endpoint_url'] = os.environ['DYNAMO_ENDPOINT']
self._dynamo = boto3.resource('dynamodb', **kwargs)
def run(self, table_name, fields=None, sort_field=None, as_json=False):
records, all_fields = self._get_data(table_name)
if fields is None:
fields = sorted(all_fields)
if as_json:
print(json.dumps(records, cls=CustomEncoder))
else:
print(self._to_csv(records, fields, sort_field))
def load_from_json(self, table_name, fname):
table = self._dynamo.Table(table_name)
with open(fname, 'r') as fh:
records = json.loads(fh.read())
count = 0
for r in records:
table.put_item(Item=r)
count += 1
print('Loaded %d items into DynamoDB table %s' % (count, table_name))
def _to_csv(self, records, fields, sort_field):
output = io.StringIO()
writer = csv.DictWriter(output, fieldnames=fields, restval='')
writer.writeheader()
rownum = 0
if sort_field is not None:
records = sorted(records, key=lambda x: x[sort_field])
for r in records:
rownum += 1
for k, v in r.items():
# format lists nicely
if isinstance(v, type([])):
r[k] = ', '.join(v)
writer.writerow({
x: r[x] for x in fields
})
return output.getvalue()
def _get_data(self, table_name):
table = self._dynamo.Table(table_name)
all_fields = set()
records = []
logger.info('Scanning DynamoDB table: %s', table_name)
for item in table.scan()['Items']:
records.append(item)
for k in item.keys():
all_fields.add(k)
logger.debug('Retrieved %d records from DynamoDB', len(records))
return records, list(all_fields)
def parse_args(argv):
p = argparse.ArgumentParser(description='Dump DynamoDB table to CSV')
p.add_argument('-f', '--field-order', dest='field_order', action='store',
type=str, default=None,
help='CSV list of field names, to output columns in this '
'order. Fields not listed will not be output. If not '
'specified, will output all fields in alphabetical '
'order.')
p.add_argument('-s', '--sort-field', dest='sort_field', type=str,
action='store', default=None,
help='Optional, name of field to sort on')
p.add_argument('-j', '--json', dest='json', action='store_true',
default=False,
help='dump to JSON instead of CSV '
'(ignores -f/--field-order')
p.add_argument('-r', '--reverse', dest='reverse', action='store', type=str,
default=False,
help='reverse - load FROM json file (filename specified in '
'this option) TO dynamodb table')
p.add_argument('TABLE_NAME', action='store', type=str,
help='DynamoDB table name to dump')
args = p.parse_args(argv)
return args
if __name__ == "__main__":
args = parse_args(sys.argv[1:])
if args.field_order is not None:
args.field_order = args.field_order.split(',')
if args.reverse:
DynamoDumper().load_from_json(args.TABLE_NAME, args.reverse)
else:
DynamoDumper().run(
args.TABLE_NAME, fields=args.field_order,
sort_field=args.sort_field, as_json=args.json
)