blob: bf8e4df02ee4f2d97606d3c8c3ab9b884b091d68 [file] [log] [blame]
#include <string>
#include <sstream>
#include <iostream>
#include <fstream>
#include <iomanip>
#include <map>
#include <list>
using namespace std;
// this function takes a line that may contain a name and/or email address,
// and returns just the name, while fixing the "bad cases".
std::string contributor_name(const std::string& line) {
string result;
// let's first take care of the case of isolated email addresses, like
// "user@localhost.localdomain" entries
if (line.find("markb@localhost.localdomain") != string::npos) {
return "Mark Borgerding";
}
if (line.find("kayhman@contact.intra.cea.fr") != string::npos) {
return "Guillaume Saupin";
}
// from there on we assume that we have a entry of the form
// either:
// Bla bli Blurp
// or:
// Bla bli Blurp <bblurp@email.com>
size_t position_of_email_address = line.find_first_of('<');
if (position_of_email_address != string::npos) {
// there is an e-mail address in <...>.
// Hauke once committed as "John Smith", fix that.
if (line.find("hauke.heibel") != string::npos)
result = "Hauke Heibel";
else {
// just remove the e-mail address
result = line.substr(0, position_of_email_address);
}
} else {
// there is no e-mail address in <...>.
if (line.find("convert-repo") != string::npos)
result = "";
else
result = line;
}
// remove trailing spaces
size_t length = result.length();
while (length >= 1 && result[length - 1] == ' ') result.erase(--length);
return result;
}
// parses hg churn output to generate a contributors map.
map<string, int> contributors_map_from_churn_output(const char* filename) {
map<string, int> contributors_map;
string line;
ifstream churn_out;
churn_out.open(filename, ios::in);
while (!getline(churn_out, line).eof()) {
// remove the histograms "******" that hg churn may draw at the end of some lines
size_t first_star = line.find_first_of('*');
if (first_star != string::npos) line.erase(first_star);
// remove trailing spaces
size_t length = line.length();
while (length >= 1 && line[length - 1] == ' ') line.erase(--length);
// now the last space indicates where the number starts
size_t last_space = line.find_last_of(' ');
// get the number (of changesets or of modified lines for each contributor)
int number;
istringstream(line.substr(last_space + 1)) >> number;
// get the name of the contributor
line.erase(last_space);
string name = contributor_name(line);
map<string, int>::iterator it = contributors_map.find(name);
// if new contributor, insert
if (it == contributors_map.end()) contributors_map.insert(pair<string, int>(name, number));
// if duplicate, just add the number
else
it->second += number;
}
churn_out.close();
return contributors_map;
}
// find the last name, i.e. the last word.
// for "van den Schbling" types of last names, that's not a problem, that's actually what we want.
string lastname(const string& name) {
size_t last_space = name.find_last_of(' ');
if (last_space >= name.length() - 1)
return name;
else
return name.substr(last_space + 1);
}
struct contributor {
string name;
int changedlines;
int changesets;
string url;
string misc;
contributor() : changedlines(0), changesets(0) {}
bool operator<(const contributor& other) { return lastname(name).compare(lastname(other.name)) < 0; }
};
void add_online_info_into_contributors_list(list<contributor>& contributors_list, const char* filename) {
string line;
ifstream online_info;
online_info.open(filename, ios::in);
while (!getline(online_info, line).eof()) {
string hgname, realname, url, misc;
size_t last_bar = line.find_last_of('|');
if (last_bar == string::npos) continue;
if (last_bar < line.length()) misc = line.substr(last_bar + 1);
line.erase(last_bar);
last_bar = line.find_last_of('|');
if (last_bar == string::npos) continue;
if (last_bar < line.length()) url = line.substr(last_bar + 1);
line.erase(last_bar);
last_bar = line.find_last_of('|');
if (last_bar == string::npos) continue;
if (last_bar < line.length()) realname = line.substr(last_bar + 1);
line.erase(last_bar);
hgname = line;
// remove the example line
if (hgname.find("MercurialName") != string::npos) continue;
list<contributor>::iterator it;
for (it = contributors_list.begin(); it != contributors_list.end() && it->name != hgname; ++it) {
}
if (it == contributors_list.end()) {
contributor c;
c.name = realname;
c.url = url;
c.misc = misc;
contributors_list.push_back(c);
} else {
it->name = realname;
it->url = url;
it->misc = misc;
}
}
}
int main() {
// parse the hg churn output files
map<string, int> contributors_map_for_changedlines = contributors_map_from_churn_output("churn-changedlines.out");
// map<string,int> contributors_map_for_changesets = contributors_map_from_churn_output("churn-changesets.out");
// merge into the contributors list
list<contributor> contributors_list;
map<string, int>::iterator it;
for (it = contributors_map_for_changedlines.begin(); it != contributors_map_for_changedlines.end(); ++it) {
contributor c;
c.name = it->first;
c.changedlines = it->second;
c.changesets = 0; // contributors_map_for_changesets.find(it->first)->second;
contributors_list.push_back(c);
}
add_online_info_into_contributors_list(contributors_list, "online-info.out");
contributors_list.sort();
cout << "{| cellpadding=\"5\"\n";
cout << "!\n";
cout << "! Lines changed\n";
cout << "!\n";
list<contributor>::iterator itc;
int i = 0;
for (itc = contributors_list.begin(); itc != contributors_list.end(); ++itc) {
if (itc->name.length() == 0) continue;
if (i % 2)
cout << "|-\n";
else
cout << "|- style=\"background:#FFFFD0\"\n";
if (itc->url.length())
cout << "| [" << itc->url << " " << itc->name << "]\n";
else
cout << "| " << itc->name << "\n";
if (itc->changedlines)
cout << "| " << itc->changedlines << "\n";
else
cout << "| (no information)\n";
cout << "| " << itc->misc << "\n";
i++;
}
cout << "|}" << endl;
}