233 lines
6.2 KiB
C++
233 lines
6.2 KiB
C++
|
#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;
|
||
|
}
|