forked from derekgreene/dynamic-nmf
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtrack-dynamic-topics.py
executable file
·111 lines (99 loc) · 4.55 KB
/
track-dynamic-topics.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
#!/usr/bin/env python
"""
Script to track the individual topics from each window that contribute to an overall set of dynamic topics.
Usage:
python track-dynamic-topics.py -l out/dynamictopics_k05.pkl out/month1_windowtopics_k05.pkl out/month2_windowtopics_k05.pkl out/month3_windowtopics_k05.pkl
"""
import os, sys
import logging as log
from optparse import OptionParser
from prettytable import PrettyTable
import unsupervised.nmf, unsupervised.rankings
# --------------------------------------------------------------
def main():
parser = OptionParser(usage="usage: %prog [options] dynamic_topics window_topics1 window_topics2...")
parser.add_option("-t", "--top", action="store", type="int", dest="top", help="number of top terms to display", default=10)
parser.add_option("-l","--long", action="store_true", dest="long_display", help="long format display")
parser.add_option("-d", "--dynamic", action="store", type="string", dest="dynamic_required", help="to view a subset of dynamic topics, specify one or more topic numbers comma separated", default=None)
(options, args) = parser.parse_args()
if( len(args) < 3 ):
parser.error( "Must specify at least a dynamic topic file, followed by two or more window topic files (in order of time window)" )
log.basicConfig(level=20, format='%(message)s')
# Load dynamic results: (doc_ids, terms, term_rankings, partition, W, H, labels)
dynamic_in_path = args[0]
dynamic_res = unsupervised.nmf.load_nmf_results( dynamic_in_path )
dynamic_k = len(dynamic_res[2])
dynamic_term_rankings = unsupervised.rankings.truncate_term_rankings( dynamic_res[2], options.top )
log.info( "Loaded model with %d dynamic topics from %s" % (dynamic_k, dynamic_in_path) )
# Create a map of window topic label -> dynamic topic
assigned_window_map = {}
dynamic_partition = dynamic_res[3]
for idx, window_topic_label in enumerate(dynamic_res[0]):
assigned_window_map[window_topic_label] = dynamic_partition[idx]
all_tracked_topics = []
for i in range(dynamic_k):
all_tracked_topics.append( [] )
if not options.dynamic_required is None:
dynamic_required = [ int(x) for x in options.dynamic_required.split(",") ]
# Load all of the individual window topics models
# Note: We should have one result for each window
window_num = 0
for in_path in args[1:]:
window_num += 1
log.debug( "Loading window topics for window %d from %s ..." % ( window_num, in_path ) )
# Load window results: (doc_ids, terms, term_rankings, partition, W, H, labels)
window_res = unsupervised.nmf.load_nmf_results( in_path )
window_k = len(window_res[2])
window_term_rankings = unsupervised.rankings.truncate_term_rankings( window_res[2], options.top )
log.info( "Loaded model with %d window topics from %s" % (window_k, in_path) )
for idx, window_topic_label in enumerate(window_res[6]):
dynamic_topic_idx = assigned_window_map[window_topic_label]
ranking = window_term_rankings[idx]
all_tracked_topics[dynamic_topic_idx].append( (window_num,ranking) )
# Now display each dynamic topic by building a PrettyTable
for i in range(dynamic_k):
# do we want to display this dynamic topic?
if (not options.dynamic_required is None) and ( not (i+1) in dynamic_required ) :
continue
dynamic_topic_label = dynamic_res[6][i]
log.info("- Dynamic Topic: %s" % dynamic_topic_label )
# long format?
if options.long_display:
# standard format?
current_labels = [ "Overall" ]
current_rankings = [ dynamic_term_rankings[i] ]
for t in all_tracked_topics[i]:
current_labels.append( "Window %d" % t[0] )
current_rankings.append( t[1] )
log.info( unsupervised.rankings.format_term_rankings_long( current_rankings, current_labels, options.top ).strip() )
else:
header = ["Rank", "Overall"]
for t in all_tracked_topics[i]:
field = "Window %d" % t[0]
# deal with multiple window topics from same window
suffix = 1
while field in header:
suffix += 1
field = "Window %d(%d)" % (t[0],suffix)
header.append( field )
tab = PrettyTable(header)
tab.align["Rank"] = "r"
for label in header[1:]:
tab.align[label] = "l"
# add the term rows
for pos in range(options.top):
row = [ str(pos+1) ]
# the term from the overall (dynamic) topic
row.append( dynamic_term_rankings[i][pos] )
# the term for each time window topic
for t in all_tracked_topics[i]:
# have we run out of terms?
if len(t[1]) <= pos:
row.append( "" )
else:
row.append( t[1][pos] )
tab.add_row( row )
log.info( tab )
# --------------------------------------------------------------
if __name__ == "__main__":
main()