forked from oster/rtce-experiments-toolbox
-
Notifications
You must be signed in to change notification settings - Fork 1
/
chat-slicing-notes.py
executable file
·205 lines (159 loc) · 6.84 KB
/
chat-slicing-notes.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
#! /usr/bin/python -tt
import os
import re
import json
import gzip
from datetime import datetime, timedelta
import time
from collections import Counter
last_uid = -1
user_ids = {}
def reset_user_ids():
global last_uid
global user_ids
last_uid = -1
user_ids = {}
def generate_user_id(str_uid):
global last_uid
global user_ids
if str_uid in user_ids:
return user_ids[str_uid]
else:
last_uid = last_uid + 1
user_ids[str_uid] = last_uid
return last_uid
def load_pad_revisions_and_chat(db_file, padid):
with gzip.open(db_file, 'rb') as f:
revisions = []
pattern_rev_metadata = re.compile('{"key":"pad:'+padid+':revs:([0-9]+)"')
pattern_rev_data = re.compile('pad:'+padid)
chat_entries = []
pattern_chat_data = re.compile('{"key":"pad:'+padid+':chat:([0-9]+)"')
while True:
try:
line = f.next()
match = pattern_rev_metadata.match(line)
if match:
rev = int(match.group(1)) #-> to int ?
rev_partA = json.loads(line)
rev_partB = json.loads(f.next())
if not pattern_rev_data.match(rev_partB['key']):
raise Exception('missing pad content for rev:%i' % (rev))
if rev_partB['val']['head'] != rev:
raise Exception( 'head (%i) does not match with current rev (%i)' % (rev_partB['val']['head'], rev))
revision = {}
revision['rev'] = rev
revision['timestamp'] = rev_partA['val']['meta']['timestamp']
revision['datetime'] = datetime.fromtimestamp(int(revision['timestamp']) / 1000)
revision['author'] = rev_partA['val']['meta']['author']
revision['content'] = rev_partB['val']['atext']['text']
revisions.append(revision)
match = pattern_chat_data.match(line)
if match:
entry_num = int(match.group(1))
content = json.loads(line)
entry = {}
entry['num'] = entry_num
entry['timestamp'] = content['val']['time']
entry['datetime'] = datetime.fromtimestamp(int(entry['timestamp']) / 1000)
entry['message'] = content['val']['text']
entry['userid'] = generate_user_id(content['val']['userId'])
chat_entries.append(entry)
except StopIteration:
break
return (revisions, chat_entries)
def format_time(timestamp):
return timestamp.strftime('%H:%M:%S')
def get_revision(revisions, revision_num):
return [ rev for rev in revisions if rev['rev'] == revision_num ][0]
def get_revision_at_time(revisions, one_min_after_end_of_audio):
return [ rev for rev in revisions if rev['datetime'] <= one_min_after_end_of_audio ][-1]
def format_chat_entry(entry):
return "[%s] user%s:> %s" % (format_time(entry['datetime']), entry['userid'], entry['message'].encode('utf8'))
def print_chat_entries(entries):
for e in entries:
print format_chat_entry(e)
def split_list(f_cond, list):
selected_items = []
unselected_items = []
for item in list:
if f_cond(item):
selected_items.append(item)
else:
unselected_items.append(item)
return (selected_items, unselected_items)
def chat_word_counts(entries):
words_by_lines = [ entry['message'].split() for entry in entries ]
words = [word.lower() for words_in_line in words_by_lines for word in words_in_line]
word_counts = Counter(words)
return word_counts
def write_chat_word_counts_as_csv(filename, entries):
with open(filename, "w") as csv_file:
for (word, count) in chat_word_counts(entries).items():
csv_file.write('"%s", %i\n' % (word.encode('utf8'), count))
def slice_chat(num, experiment_name):
(revisions, chat_entries) = load_pad_revisions_and_chat(INPUT_DATA_PATH + group + '/dirty.db.gz', experiment_name + num)
chat_entries.sort(key=lambda entry: entry['datetime']) # just to be sure ;)
initial_doc_rev = get_revision(revisions, initial_doc_rev_num)
first_changes_rev = get_revision(revisions, first_changes_rev_num)
end_of_audio_rev = get_revision(revisions, end_of_audio_rev_num)
print "= Group: %s" % (num)
print "== first version - rev: %s (%s)" % (initial_doc_rev['rev'], format_time(initial_doc_rev['datetime']))
print "== first change version - rev: %s (%s)" % (first_changes_rev['rev'], format_time(first_changes_rev['datetime']))
print "== end of audio version - rev: %s (%s)" % (end_of_audio_rev['rev'], format_time(end_of_audio_rev['datetime']))
x_min_after_end_of_audio_revs = [ get_revision_at_time(revisions, end_of_audio_rev['datetime'] + timedelta(minutes=x)) for x in xrange(1,6)]
i = 0
for rev in x_min_after_end_of_audio_revs:
i = i + 1
print "== %i min. after end of audio version - rev: %s (%s)" % (i, rev['rev'], format_time(rev['datetime']))
entries = chat_entries
print '\t-- chat --'
(selected, entries) = split_list(lambda entry: entry['datetime'] <= initial_doc_rev['datetime'], entries)
print_chat_entries(selected)
write_chat_word_counts_as_csv(OUTPUT_DATA_PATH + experiment_name + num + '-before-init.csv', selected)
print '\t-- initial version --'
(selected, entries) = split_list(lambda entry: entry['datetime'] <= first_changes_rev['datetime'], entries)
print_chat_entries(selected)
write_chat_word_counts_as_csv(OUTPUT_DATA_PATH + experiment_name + num + '-before-first-change.csv', selected)
print '\t-- first change version --'
(selected, entries) = split_list(lambda entry: entry['datetime'] <= end_of_audio_rev['datetime'], entries)
print_chat_entries(selected)
write_chat_word_counts_as_csv(OUTPUT_DATA_PATH + experiment_name + num + '-before-end-audio.csv', selected)
print '\t-- end of audio --'
i = 0
for rev in x_min_after_end_of_audio_revs:
i = i + 1
(selected, entries) = split_list(lambda entry: entry['datetime'] <= rev['datetime'], entries)
print_chat_entries(selected)
write_chat_word_counts_as_csv(OUTPUT_DATA_PATH + experiment_name + num + '-before-%imin-after-end-audio.csv' % i, selected)
print '\t-- %i min after end of audio --' % (i)
print_chat_entries(entries)
write_chat_word_counts_as_csv(OUTPUT_DATA_PATH + experiment_name + num + '-after-5min-after-end-audio.csv', entries)
print '\t-- end of record --\n\n'
INPUT_DATA_PATH='./DATA-by-num/'
INPUT_DATA_JSON_FILE='./chat-slicing-data-notes.json'
OUTPUT_DATA_PATH='./DATA-chat-word-counts/'
# data_json = '''{
# "004": {
# "notes": {
# "init-rev": 3 ,
# "first-change-rev": 8 ,
# "end-of-audio-rev": 488
# }
# }
# }'''
# data = json.loads(data_json)
with open (INPUT_DATA_JSON_FILE, "r") as json_data_file:
data = json.loads(json_data_file.read())
for group in data.keys():
#for group in ['005']:
for experiment in data[group].keys():
if group == '014' and (experiment == "corrections" or experiment == "films"):
num = '015'
else:
num = group
reset_user_ids()
initial_doc_rev_num = data[group][experiment]["init-rev"]
first_changes_rev_num = data[group][experiment]["first-change-rev"]
end_of_audio_rev_num = data[group][experiment]["end-of-audio-rev"]
slice_chat(num, experiment)