Skip to content

Commit a315629

Browse files
authoredSep 13, 2020
Merge pull request #1 from dtcooper/master
Update
2 parents be89b43 + d46429e commit a315629

File tree

7 files changed

+93
-28
lines changed

7 files changed

+93
-28
lines changed
 

‎LICENSE

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
MIT License
22

3-
Copyright (c) 2011-2019, David Cooper <david@dtcooper.com>
4-
Copyright (c) 2017-2019, Carey Metcalfe <carey@cmetcalfe.ca>
3+
Copyright (c) 2011-2020, David Cooper <david@dtcooper.com>
4+
Copyright (c) 2017-2020, Carey Metcalfe <carey@cmetcalfe.ca>
55

66
Permission is hereby granted, free of charge, to any person obtaining a copy
77
of this software and associated documentation files (the "Software"), to deal

‎README.md

+52-1
Original file line numberDiff line numberDiff line change
@@ -15,7 +15,58 @@ FIT files
1515
by [ANT](http://www.thisisant.com/).
1616
- The SDK, code examples, and detailed documentation can be found in the
1717
[ANT FIT SDK](http://www.thisisant.com/resources/fit).
18-
18+
19+
20+
Usage
21+
-----
22+
A simple example of printing records from a fit file:
23+
24+
```python
25+
import fitparse
26+
27+
# Load the FIT file
28+
fitfile = fitparse.FitFile("my_activity.fit")
29+
30+
# Iterate over all messages of type "record"
31+
# (other types include "device_info", "file_creator", "event", etc)
32+
for record in fitfile.get_messages("record"):
33+
34+
# Records can contain multiple pieces of data (ex: timestamp, latitude, longitude, etc)
35+
for data in record:
36+
37+
# Print the name and value of the data (and the units if it has any)
38+
if data.units:
39+
print(" * {}: {} ({})".format(data.name, data.value, data.units))
40+
else:
41+
print(" * {}: {}".format(data.name, data.value))
42+
43+
print("---")
44+
```
45+
46+
The library also provides a `fitdump` script for command line usage:
47+
```
48+
$ fitdump --help
49+
usage: fitdump [-h] [-v] [-o OUTPUT] [-t {readable,json}] [-n NAME] [--ignore-crc] FITFILE
50+
51+
Dump .FIT files to various formats
52+
53+
positional arguments:
54+
FITFILE Input .FIT file (Use - for stdin)
55+
56+
optional arguments:
57+
-h, --help show this help message and exit
58+
-v, --verbose
59+
-o OUTPUT, --output OUTPUT
60+
File to output data into (defaults to stdout)
61+
-t {readable,json}, --type {readable,json}
62+
File type to output. (DEFAULT: readable)
63+
-n NAME, --name NAME Message name (or number) to filter
64+
--ignore-crc Some devices can write invalid crc's, ignore these.
65+
```
66+
67+
See the documentation for more: http://dtcooper.github.com/python-fitparse
68+
69+
1970
Major Changes From Original Version
2071
-----------------------------------
2172

‎fitparse/__init__.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@
33
from fitparse.processors import FitFileDataProcessor, StandardUnitsDataProcessor
44

55

6-
__version__ = '1.1.0'
6+
__version__ = '1.2.0'
77
__all__ = [
88
'FitFileDataProcessor', 'FitFile', 'FitParseError',
99
'StandardUnitsDataProcessor', 'DataMessage'

‎fitparse/processors.py

+8-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
import datetime
2-
from fitparse.utils import scrub_method_name
2+
from fitparse.utils import scrub_method_name, is_iterable
33

44
# Datetimes (uint32) represent seconds since this UTC_REFERENCE
55
UTC_REFERENCE = 631065600 # timestamp for UTC 00:00 Dec 31 1989
@@ -107,7 +107,13 @@ def process_field_distance(self, field_data):
107107

108108
def process_field_speed(self, field_data):
109109
if field_data.value is not None:
110-
field_data.value *= 60.0 * 60.0 / 1000.0
110+
factor = 60.0 * 60.0 / 1000.0
111+
112+
# record.enhanced_speed field can be a tuple
113+
if is_iterable(field_data.value):
114+
field_data.value = tuple(x * factor for x in field_data.value)
115+
else:
116+
field_data.value *= factor
111117
field_data.units = 'km/h'
112118

113119
def process_units_semicircles(self, field_data):

‎fitparse/records.py

+14-14
Original file line numberDiff line numberDiff line change
@@ -120,6 +120,12 @@ def get(self, field_name, as_dict=False):
120120
if field_data.is_named(field_name):
121121
return field_data.as_dict() if as_dict else field_data
122122

123+
def get_raw_value(self, field_name):
124+
field_data = self.get(field_name)
125+
if field_data:
126+
return field_data.raw_value
127+
return None
128+
123129
def get_value(self, field_name):
124130
# SIMPLIFY: get rid of this completely
125131
field_data = self.get(field_name)
@@ -433,11 +439,8 @@ def parse_string(string):
433439

434440
def add_dev_data_id(message):
435441
global DEV_TYPES
436-
dev_data_index = message.get('developer_data_index').raw_value
437-
if message.get('application_id'):
438-
application_id = message.get('application_id').raw_value
439-
else:
440-
application_id = None
442+
dev_data_index = message.get_raw_value('developer_data_index')
443+
application_id = message.get_raw_value('application_id')
441444

442445
# Note that nothing in the spec says overwriting an existing type is invalid
443446
DEV_TYPES[dev_data_index] = {'dev_data_index': dev_data_index, 'application_id': application_id, 'fields': {}}
@@ -446,15 +449,12 @@ def add_dev_data_id(message):
446449
def add_dev_field_description(message):
447450
global DEV_TYPES
448451

449-
dev_data_index = message.get('developer_data_index').raw_value
450-
field_def_num = message.get('field_definition_number').raw_value
451-
base_type_id = message.get('fit_base_type_id').raw_value
452-
field_name = message.get('field_name').raw_value
453-
units = message.get('units').raw_value
454-
455-
native_field_num = message.get('native_field_num')
456-
if native_field_num is not None:
457-
native_field_num = native_field_num.raw_value
452+
dev_data_index = message.get_raw_value('developer_data_index')
453+
field_def_num = message.get_raw_value('field_definition_number')
454+
base_type_id = message.get_raw_value('fit_base_type_id')
455+
field_name = message.get_raw_value('field_name') or "unnamed_dev_field_%s" % field_def_num
456+
units = message.get_raw_value("units")
457+
native_field_num = message.get_raw_value('native_field_num')
458458

459459
if dev_data_index not in DEV_TYPES:
460460
raise FitParseError("No such dev_data_index=%s found" % (dev_data_index))

‎fitparse/utils.py

+4-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,9 @@
11
import io
22
import re
3-
from collections import Iterable
3+
try:
4+
from collections.abc import Iterable
5+
except ImportError:
6+
from collections import Iterable
47

58

69
class FitParseError(ValueError):

‎scripts/fitdump

+12-7
Original file line numberDiff line numberDiff line change
@@ -80,7 +80,7 @@ class RecordJSONEncoder(json.JSONEncoder):
8080
def default(self, obj):
8181
if isinstance(obj, types.GeneratorType):
8282
return list(obj)
83-
if isinstance(obj, datetime.datetime):
83+
if isinstance(obj, (datetime.datetime, datetime.time)):
8484
return obj.isoformat()
8585
if isinstance(obj, fitparse.DataMessage):
8686
return {
@@ -107,12 +107,17 @@ def main(args=None):
107107
as_dict=options.as_dict
108108
)
109109

110-
if options.type == "json":
111-
json.dump(records, fp=options.output, cls=RecordJSONEncoder)
112-
elif options.type == "readable":
113-
options.output.writelines(format_message(n, record, options)
114-
for n, record in enumerate(records, 1))
115-
110+
try:
111+
if options.type == "json":
112+
json.dump(records, fp=options.output, cls=RecordJSONEncoder)
113+
elif options.type == "readable":
114+
options.output.writelines(format_message(n, record, options)
115+
for n, record in enumerate(records, 1))
116+
finally:
117+
try:
118+
options.output.close()
119+
except IOError:
120+
pass
116121

117122
if __name__ == '__main__':
118123
try:

0 commit comments

Comments
 (0)
Please sign in to comment.