Standard topic modeling approaches assume the order of documents does not matter, making them unsuitable for time-stamped corpora. In contrast, dynamic topic modeling approaches track how language changes and topics evolve over time. We have developed a two-level approach for dynamic topic modeling via Non-negative Matrix Factorization (NMF), which links together topics identified in snapshots of text sources appearing over time.
Details of this approach are described in the following paper (Link):
Exploring the Political Agenda of the European Parliament Using a Dynamic Topic Modeling Approach
Derek Greene, James P. Cross. Political Analysis, 2016.
This repository contains a Python reference implementation of the above approach.
Tested with Python 2.7 and Python 3.5, and requiring the following packages, which are available via PIP:
- Required: numpy >= 1.8.0
- Required: scikit-learn >= 0.14
- Required for utility tools: prettytable >= 0.7.2
- Required for automatic model selection: gensim >= 0.10.3
To perform dynamic topic modeling, the input corpus of documents should consist of plain text files (one document per file), organised into two or more sub-directories. Each of these sub-directories should correspond to a unique time window, representing a different time interval. The names of these sub-directories is arbitrary, once their alphabetic ordering corresponds to their order in time (e.g 2000, 2001, 2002; month1, month2, month3; 2010-q1, 2010-q2, 2010-q3).
The dynamic topic modeling process consists of three steps, discussed below. The archive 'data/sample.zip' contains a sample corpus of 1,324 news articles divided into three time windows (month1, month2, month3), which is used to illustrate these steps.
Before applying dynamic topic modeling, the first step is to pre-process the documents from each time window (i.e. sub-directory), to produce a document-term matrix for those windows. This involves tokenizing the documents, removing common stop-words, and building a document-term matrix for the time window. In the example below, we parse all .txt files in the sub-directories of 'data/sample'. The output files will be stored in the directory 'data'. Note that the final options below indicate that we want to apply TF-IDF term weighting and document length normalization to the documents before writing each matrix.
python prep-text.py data/sample/month1 data/sample/month2 data/sample/month3 -o data --tfidf --norm
The result of this process will be a collection of Joblib binary files (*.pkl and *.npy) written to the directory 'data', where the prefix of each corresponds to the name of each time window (e.g. month1, month2 etc).
Once the data has been pre-processed, the next step is to generate the window topics, where a topic model is created by applying NMF to each the pre-process data for each time window. For the example data, we apply it to the three months. If we want to use the same number of topics for each window (e.g. 5 topics), we can run the following, where results are written to the directory 'out':
python find-window-topics.py data/*.pkl -k 5 -o out
When the process has completed, we can view the descriptiors (i.e. the top ranked terms) for the resulting window topics as follows:
python display-topics.py out/month1_windowtopics_k05.pkl out/month2_windowtopics_k05.pkl out/month3_windowtopics_k05.pkl
The top terms and document IDs can be exported from a NMF results file to two individual comma-separated files using 'export-csv.py'. For instance, to export the top 50 terms and document IDs for a single results file:
python export-csv.py out/month1_windowtopics_k05.pkl -t 50
Once the window topics have been created, we combine the results for the time windows to generate dynamic topics that span across multiple time windows. If we want to specify a fixed number of dynamic topics (e.g. k=5), we can run the following, where results are written to the directory 'out':
python find-dynamic-topics.py out/month1_windowtopics_k05.pkl out/month2_windowtopics_k05.pkl out/month3_windowtopics_k05.pkl -k 5 -o out
In this case the results will be written to 'out/dynamictopics_k05.pkl'. When the process has completed, we can view the dynamic topic descriptiors using 'display-topics.py':
python display-topics.py out/dynamictopics_k05.pkl
For the sample corpus, the output for the top 10 terms for 5 dynamic topics should look like:
+------+------------+-----------+------------+--------+----------+
| Rank | D01 | D02 | D03 | D04 | D05 |
+------+------------+-----------+------------+--------+----------+
| 1 | blair | chelsea | people | best | growth |
| 2 | labour | game | mobile | band | economy |
| 3 | election | club | users | music | oil |
| 4 | government | united | software | film | sales |
| 5 | minister | arsenal | microsoft | album | prices |
| 6 | brown | league | technology | awards | market |
| 7 | party | players | net | show | bank |
| 8 | prime | cup | phone | number | economic |
| 9 | howard | liverpool | computer | award | profits |
| 10 | told | football | security | top | company |
+------+------------+-----------+------------+--------+----------+
The examples above involve using a manually-specified number of topics, for both window topics and dynamic topics. In cases where this number is not known in advance, a variety of strategies exist for automatically or semi-automatically choosing a number of topics. This package contains an implementation of the TC-W2V topic coherence measure, which can be used to compare different topic models and subsequently choose a model with a suitable number of topics. More details on the TC-W2V are included in the paper:
An Analysis of the Coherence of Descriptors in Topic Modeling
D. O'Callaghan, D. Greene, J. Carthy, P. Cunningham.
Expert Systems with Applications (ESWA), 2015.
The approach involves a number of steps, listed below. Again these steps are illustrated using the sample corpus.
As well as preparing the input text corpus, we also need to build a Word2Vec model from all of the documents in our corpus. The script 'prep-word2vec.py' uses Gensim to build a Skipgram (SG) Word2Vec model. All of the text files in the specified sub-directories are used to build the model, which is written to the file 'out/w2v-model.bin'.
python prep-word2vec.py data/sample -o out -m sg
Next, we use topic coherence based on the pre-built Word2Vec model to evaluate a range of different values for the number of topics k for each time window. We use the same 'find-window-topics.py' script, but specify a comma-separated range of values to try (kmin,kmax) (e.g. 4,10 will test all numbers of topics from k=4 to k=10), and also specify the path to Word2Vec model file:
python find-window-topics.py data/*.pkl -k 4,10 -o out -m out/w2v-model.bin -w selected.csv
The script will apply NMF to each time window for each value of k, writing a result file each time to the directory 'out'. The output of the above for the sample data will also include the following top 3 recommendations for the number of topics for each of the three time windows:
Top recommendations for number of topics for 'month1': 5,6,7
...
Top recommendations for number of topics for 'month2': 8,10,7
...
Top recommendations for number of topics for 'month3': 8,6,10
The top recommended number of topics for each window will be stored in selected.csv
We can also run automatic selection for the number of dynamic topics, by running the script 'find-dynamic-topics.py' and specifying a comma-separated range kmin,kmax and the path to the Word2Vec model built on the entire corpus:
python find-dynamic-topics.py out/month1_windowtopics_k05.pkl out/month2_windowtopics_k08.pkl out/month3_windowtopics_k08.pkl -k 4,10 -o out -m out/w2v-model.bin
Applying this to the sample corpus for the range [4,10] results in the recommendation of 5 topics:
Top recommendations for number of dynamic topics: 5,10,6
The corresponding results will be written to 'out/dynamictopics_k05.pkl'. When the process has completed, we can view the dynamic topic descriptiors using:
python display-topics.py out/dynamictopics_k05.pkl
For the sample corpus, the output for the top 10 terms for 5 dynamic topics should look similar to:
+------+------------+-----------+------------+--------+----------+
| Rank | D01 | D02 | D03 | D04 | D05 |
+------+------------+-----------+------------+--------+----------+
| 1 | people | chelsea | blair | best | growth |
| 2 | mobile | club | labour | band | economy |
| 3 | users | game | election | music | oil |
| 4 | software | arsenal | government | film | sales |
| 5 | phone | united | party | album | market |
| 6 | microsoft | league | brown | show | prices |
| 7 | technology | players | minister | awards | bank |
| 8 | net | liverpool | howard | number | economic |
| 9 | computer | cup | prime | tv | profits |
| 10 | security | football | tory | award | company |
+------+------------+-----------+------------+--------+----------+
To track the individual topics from each window that contribute to the overall dynamic topics, run the script 'track-dynamic-topics.py', specifying the file path for the output of dynamic topic modeling, following by the paths for all of the selected individual window topic models (ordered by time window). Make sure that the window topic files were the ones used to generate the dynamic topics.
Note that multiple topics in a single time window can be related to a single dynamic topic. Following on from the example the above, to see the tracking for all dynamic topics, run:
python track-dynamic-topics.py out/dynamictopics_k05.pkl out/month1_windowtopics_k05.pkl out/month2_windowtopics_k08.pkl out/month3_windowtopics_k08.pkl
To view tracking for only a subset of dynamic topics, specify one or more topic numbers comma separated:
python track-dynamic-topics.py out/dynamictopics_k05.pkl out/month1_windowtopics_k05.pkl out/month2_windowtopics_k08.pkl out/month3_windowtopics_k08.pkl -d 1,4
For the sample corpus, the output for tracking the dynamic topics D01 and D04 will contain the top-ranked terms for both the overall dynamic topics and the associated time window topics:
- Dynamic Topic: D01
+------+------------+-------------+------------+------------+-------------+-------------+
| Rank | Overall | Window 1 | Window 2 | Window 3 | Window 3(2) | Window 3(3) |
+------+------------+-------------+------------+------------+-------------+-------------+
| 1 | people | microsoft | people | broadband | virus | mobile |
| 2 | mobile | mobile | technology | tv | spam | phone |
| 3 | users | users | phone | digital | software | phones |
| 4 | software | software | users | high | mail | camera |
| 5 | phone | people | computer | dvd | security | mobiles |
| 6 | microsoft | security | software | service | attacks | people |
| 7 | technology | net | sites | bt | net | cameras |
| 8 | net | information | site | people | microsoft | handsets |
| 9 | computer | programs | net | definition | users | technology |
| 10 | security | computer | microsoft | technology | search | video |
+------+------------+-------------+------------+------------+-------------+-------------+
- Dynamic Topic: D04
+------+---------+----------+----------+-------------+----------+
| Rank | Overall | Window 1 | Window 2 | Window 2(2) | Window 3 |
+------+---------+----------+----------+-------------+----------+
| 1 | best | film | album | show | music |
| 2 | band | best | band | tv | band |
| 3 | music | awards | number | series | best |
| 4 | film | award | chart | bbc | album |
| 5 | album | actor | music | super | rock |
| 6 | show | director | awards | channel | show |
| 7 | awards | oscar | best | music | singer |
| 8 | number | films | song | jackson | number |
| 9 | tv | actress | single | viewers | song |
| 10 | award | star | top | television | awards |
+------+---------+----------+----------+-------------+----------+
To create a partition file (i.e. a single membership topic model) for all documents in the complete corpus relative to the dynamic topics, run:
python create-dynamic-partition.py -o out/dynamic-combined.pkl out/dynamictopics_k05.pkl out/month1_windowtopics_k05.pkl out/month2_windowtopics_k08.pkl out/month3_windowtopics_k08.pkl