forked from brunetton/redminetimesync
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathredminetimesync.py
executable file
·307 lines (277 loc) · 11.7 KB
/
redminetimesync.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
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
#!/usr/bin/python
from ConfigParser import RawConfigParser
from requests import ConnectionError
import datetime
import sqlite3
import math
import os
from pprint import pformat
import re
import sys
import yaml
from docopt import docopt # http://docopt.org/
import moment # https://pypi.python.org/pypi/moment
from redmine import Redmine # https://pypi.python.org/pypi/python-redmine
from redmine.exceptions import AuthError, ResourceNoFieldsProvidedError
ACTIVITIES_CONFIG_FILE = 'activities.config'
CONFIG_FILE = 'redminetimesync.config'
DB_TIMESTAMP_FORMAT = 'YYYY-MM-DD HH:mm:ss'
DOC = '''
Redmine / Hamster times entries synchronization
Usage:
{self_name} from <start> [(to <stop>)] [options]
{self_name} <date> [options]
{self_name} -h | --help
Options:
-a --auto Do not ask for manual validation for each day, sync all days in given interval
-n --no-date-confirm Do not ask for manual dates interval confirmation at begining
Note: start, stop and date could be :
- a date: ("12/10", "12/10/15", ...)
-> check for config file to change dates formats
- a number of days ago: ("3" means 3 days ago, "1" means yesterday, ...)
-> 0 means today
'''
def print_(string):
'''Print the string without end line break'''
print(string), # Here the end-line coma is intended
sys.stdout.flush()
def getTimeEntries(date, config):
'''Reads Sqlite Redmine DB file and return an array of explicit associative array for times entries,
filtering out entries that do not match issue_id_regexp defined in config file
Returns:
- activities_array:
array of dicts with 'description', 'label', 'issue_id', 'duration', 'comment', 'activity_id' keys
- total_duration: sum of all activities duration
'''
def fetchFromDatabase(db_filename, date):
'''Fetch data from an SQLITE3 database
Returns an iterable object with SELECT result'''
_date = "%{}%".format(date.format('YYYY-MM-DD'))
connection = sqlite3.connect(os.path.expanduser(db_filename))
dbCursor = connection.cursor()
dbCursor.execute("""SELECT
activities.name,facts.start_time,facts.end_time,facts.description,categories.name
FROM activities
JOIN facts ON activities.id = facts.activity_id
LEFT JOIN categories ON activities.category_id = categories.id
WHERE facts.start_time LIKE ?
ORDER BY start_time""", (_date,)
)
return dbCursor
db_filename = config.get('default', 'db')
time_entries = fetchFromDatabase(db_filename, date)
if os.path.exists(ACTIVITIES_CONFIG_FILE):
categories_association = yaml.load(open(ACTIVITIES_CONFIG_FILE, 'r'))
else:
categories_association = None
if config.has_option('default', 'redmine_default_activity_id'):
default_activity_id = config.get('default', 'redmine_default_activity_id')
else:
default_activity_id = None
activities = []
total_duration = 0
for time_entry in time_entries:
label = time_entry[0]
if not time_entry[2]:
print(u'\n** Warning: ignoring "{}": Not completed yet\n'.format(label))
continue
duration = (moment.date(time_entry[2], DB_TIMESTAMP_FORMAT) - moment.date(time_entry[1], DB_TIMESTAMP_FORMAT)).seconds / 3600.
assert duration > 0, "Duration for entry {} is not >0: {}".format(label, duration)
total_duration += duration
duration = round(duration, 1)
comment = time_entry[3]
# Try to find Redmine issue IDs from label using regexp defined in config file
match = re.match(config.get('default', 'issue_id_regexp'), label)
if match:
issue_id = match.group(1)
else:
print u'\n** Warning: ignoring entry "{}" : not able to find issue ID\n'.format(label)
continue
print u"* [{duration}h #{id}]: {label}".format(
duration=round(duration, 1), id=issue_id, label=label
)
if comment is not None:
print u" {}".format(comment)
# Try to find activity_id
category_name = time_entry[4]
if category_name is not None and categories_association is not None:
if category_name in categories_association:
activity_id = categories_association[category_name]
else:
print_(u' * Warning: unmatched category "{}"'.format(category_name))
if default_activity_id is not None:
activity_id = default_activity_id
print u'- assignated to default Redmine activity ID : {}'.format(default_activity_id)
else:
activity_id = None
else:
if default_activity_id is not None:
activity_id = default_activity_id
else:
activity_id = None
activities.append({
'description': label,
'label': label,
'issue_id': issue_id,
'duration': duration,
'comment': comment,
'activity_id': activity_id
})
if total_duration > 0:
print "\n\nTotal : {}h".format(round(total_duration, 1))
return activities, total_duration
def syncToRedmine(time_entries, date, redmine):
'''Push all given time_entries to Redmine'''
print_("-> Sending entries")
try:
for time_entry_infos in time_entries:
# Send this activity to Redmine
time_entry = redmine.time_entry.create(
spent_on=date.date, # converts Moment date to Datetime
issue_id=time_entry_infos['issue_id'],
hours=time_entry_infos['duration'],
activity_id=time_entry_infos['activity_id'],
comments=time_entry_infos['comment']
)
print_('.')
except ConnectionError as e:
print "Connection Error: {}".format(e.message)
print "\n"
def parse_date(datestr, date_formats):
'''Try all dates formats defined in date_formats array and returns a Moment object representing that date.
If format doesn't containt year, default assign current year to returned date (instead of 1900).
Returns: Moment object or None
'''
assert datestr
assert date_formats
for date_format in date_formats:
date_format = date_format.strip()
try:
date = moment.date(datestr, date_format)
if date_format.find('Y') == -1:
# date format doesn't containts year
current_year = datetime.date.today().year
return date.replace(year=current_year)
else:
return date
except ValueError:
pass
return None
def parse_command_line_args():
'''Parse command line args and returns args, from_date, to_date or for_date Moment dates.
nb: if from_date is not None; to_date could be None or not.
'''
def quit_with_parse_date_error(datestr, date_formats):
print "Error while parsing date '{}'.\nAccepted formats defined in config file are: {}."\
.format(datestr, date_formats)
sys.exit(-1)
def parse_date_or_days_ahead(datestr, config, quit_if_none=False):
'''Returns a moment date corresponding to given date, or days ahead number.
quit_if_none: quit programm if no date parsed
parse_date_or_days_ahead('4/10/2014') should return corresponding moment, if that format is defined in config file
parse_date_or_days_ahead('1') should return the date of yesterday
'''
# Try to find a formatted date
date_formats = config.get('default', 'date_formats').split(',')
date = parse_date(datestr, date_formats)
if date:
return date
# It's not a date; maybe is it a number corresponding to some days ago
if datestr.isdigit():
# It's a number, corresponding to some days ago from today. Retun that date
return moment.now().subtract(days=int(datestr))
if quit_if_none:
quit_with_parse_date_error(datestr, date_formats)
return None
args = docopt(DOC.format(self_name=os.path.basename(__file__)))
from_date = to_date = for_date = None
if args['from']:
from_date = parse_date_or_days_ahead(args['<start>'], config, quit_if_none=True)
if args['to']:
to_date = parse_date_or_days_ahead(args['<stop>'], config)
if args['<date>']:
for_date = parse_date_or_days_ahead(args['<date>'], config, quit_if_none=True)
return args, from_date, to_date, for_date
if __name__ == '__main__':
# Read config file
if not os.path.isfile(CONFIG_FILE):
print('Can\'t find config file: {}\nYou can copy template conf file and adapt.'.format(CONFIG_FILE))
sys.exit(-1)
config = RawConfigParser()
config.read(CONFIG_FILE)
# Parse command line parameters
args, from_date, to_date, for_date = parse_command_line_args()
# Get prefered date format from config file to display dates
date_format = config.get('default', 'date_formats')
if date_format.find(',') != -1:
# More than one format is defined, take first
date_format = (date_format.split(',')[0]).strip()
# print confirmation to user, to check dates
if from_date:
if to_date is None:
# implicitly takes today for to_date
to_date = moment.now()
question = "Sync tasks from {} to today (included) ?".format(from_date.format(date_format))
else:
question = "Sync tasks from {} to {} (included) ?".format(
from_date.format(date_format),
to_date.format(date_format)
)
elif for_date:
if args['<date>'] == '0':
question = "Sync tasks for today ?"
elif args['<date>'] == '1':
question = "Sync tasks for yesterday ({}) ?".format(for_date.format(date_format))
else:
question = "Sync tasks for {} ?".format(for_date.format(date_format))
assert question
if not args['--no-date-confirm']:
print question
print_("\nPress ENTER to validate ...")
try:
raw_input('')
print "\n"
except KeyboardInterrupt:
print "\n"
sys.exit()
# Connects to Redmine
print_('-> Connecting to Redmine...')
redmine = Redmine(config.get('redmine', 'url'), key=config.get('redmine', 'key'))
try:
redmine.auth()
except (AuthError, ConnectionError) as e:
print "\nConnection error: {}".format(e.message)
sys.exit(-1)
print_(' OK')
print "\n"
if for_date:
# only one date will be parsed
from_date = for_date
to_date = for_date
total_time = 0
total_sent_time = 0
date = from_date.clone()
while date <= to_date:
if not for_date:
print "{s} {formatted_date} {s}".format(s='*' * 20, formatted_date=date.format(date_format))
# Get time entries from local DB
time_entries, day_total = getTimeEntries(date, config)
if not time_entries:
print("\nNo time entries to send... have you been lazy ?\n\n\n")
date = date.add(days=1)
continue
# Wait for user validation
if not args['--auto']:
print "\nPress ENTER to synchronize those tasks ...",
try:
raw_input('')
except KeyboardInterrupt:
print "\n"
sys.exit()
total_time += day_total
syncToRedmine(time_entries, date, redmine)
sent_time = math.fsum(d['duration'] for d in time_entries)
total_sent_time += sent_time
date = date.add(days=1)
print
print "\n---> TOTAL: {}h found in Hamster - {}h sent to Redmine".format(round(total_time, 1), total_sent_time)