source: trunk/lib/first_page.cc @ 317

Last change on this file since 317 was 317, checked in by Peter Johansson, 14 years ago

fixing truncation issue in log message. fixes #188

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 6.1 KB
Line 
1// $Id: first_page.cc 317 2007-05-17 18:49:09Z peter $
2
3/*
4  Copyright (C) 2006, 2007 Peter Johansson
5
6  This file is part of svndigest, http://lev.thep.lu.se/trac/svndigest
7
8  svndigest is free software; you can redistribute it and/or modify it
9  under the terms of the GNU General Public License as published by
10  the Free Software Foundation; either version 2 of the License, or
11  (at your option) any later version.
12
13  svndigest is distributed in the hope that it will be useful, but
14  WITHOUT ANY WARRANTY; without even the implied warranty of
15  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16  General Public License for more details.
17
18  You should have received a copy of the GNU General Public License
19  along with this program; if not, write to the Free Software
20  Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
21  02111-1307, USA.
22*/
23
24#include "first_page.h"
25
26#include "html_utility.h"
27
28#include "Commitment.h"
29#include "Configuration.h"
30#include "Date.h"
31#include "HtmlStream.h"
32#include "Stats.h"
33#include "SVNlog.h"
34#include "Trac.h"
35#include "utility.h"
36#include <config.h> // this header file is created by configure
37
38#include <algorithm>
39#include <cassert>
40#include <fstream>
41#include <iostream>
42#include <sstream>
43#include <stdexcept>
44#include <string>
45#include <sys/param.h>
46#include <unistd.h>
47#include <vector>
48
49namespace theplu{
50namespace svndigest{
51
52  void print_main_page(const std::string& dir, const SVNlog& log,
53                       const Stats& stats, std::string url)
54  {
55    std::string filename="index.html";
56    std::ofstream os(filename.c_str());
57    print_header(os, dir, 0, "all", "main", "index.html");
58
59    using namespace std;
60    set<string> authors;
61    authors.insert(log.author().begin(), log.author().end());
62    // erase invalid authors
63    authors.erase("");
64    authors.erase("no author");
65
66    vector<Commitment> latest_commit;
67    latest_commit.reserve(authors.size());
68    for (set<string>::const_iterator i(authors.begin()); i!=authors.end(); ++i)
69      latest_commit.push_back(log.latest_commit(*i));
70
71    print_general_information(os, log, authors.size(), url);
72    sort(latest_commit.begin(), latest_commit.end(), GreaterRevision());
73    print_authors(os, latest_commit, stats);
74    print_recent_logs(os, log);
75    os << "<hr width=100% />";
76    print_footer(os);
77    os.close();
78
79  }
80
81  void print_general_information(std::ostream& os, const SVNlog& log, 
82                                 size_t nof_authors, std::string url)
83  { 
84    Date begin(log.date()[0]);
85    if (log.date().size()>1)
86      begin = std::min(begin, Date(log.date()[1]));
87    Date end(log.date().back());
88    Date now;
89    std::string timefmt("%a, %e %b %Y");
90
91    os << "<div class=\"main\">" 
92       << "<table class=\"main\"><thead><tr><th colspan=\"2\">" 
93       << "General Information"
94       << "</th></tr></thead>\n"
95       << "<tr><td>URL:</td><td>" 
96       << url << "</td></tr>\n"
97       << "<tr><td>First Revision:</td><td>" 
98       << begin(timefmt) << "</td></tr>\n"
99       << "<tr><td>Latest Revision:</td><td>" 
100       << end(timefmt) << "</td></tr>\n"
101       << "<tr><td>Report Generated:</td><td>" 
102       << now(timefmt) << "</td></tr>\n"
103       << "<tr><td>Repository Age:</td><td>"; 
104    os << now.difftime(begin);
105    os << "</td></tr>\n"
106       << "<tr><td>Number of Authors:</td><td>" << nof_authors
107       << "</td></tr>\n"
108       << "<tr><td>Revisions:</td><td>" << log.revision().size() 
109       << "</td></tr>\n"
110       << "</table></div>\n";
111  }
112
113
114  void print_authors(std::ostream& os, 
115                     const std::vector<Commitment>& lc,
116                     const Stats& stats)
117  {
118    os << "<div class=\"main\">"
119       << "<table class=\"main\"><thead><tr><th colspan=\"2\">" 
120       << "Authors"
121       << "</th></tr></thead>\n";
122
123    os << "<tr><td>Author</td>" 
124       << "<td>Number of Lines</td>"
125       << "<td>Code Lines</td>"
126       << "<td>Comment Lines</td>"
127       << "<td>Latest Commitment</td>"
128       <<"</tr>";
129
130    std::string timefmt("%b %d %H:%M:%S %Y");
131    using namespace std;
132    for (vector<Commitment>::const_iterator i=lc.begin(); i!=lc.end(); ++i) {
133      os << "<tr><td>"; 
134      if (!stats.lines(i->author()))
135        os << i->author();
136      else
137        os << anchor(string(i->author()+"/total/index.html"),i->author());
138      os << "</td><td>" << stats.lines(i->author()) << " (" 
139         << 100*stats.lines(i->author())/stats.lines() << "%)</td>"
140         << "<td>" << stats.code(i->author()) << " (" 
141         << 100*stats.code(i->author())/stats.code() << "%)</td>"
142         << "<td>" << stats.comments(i->author()) << " (" 
143         << 100*stats.comments(i->author())/stats.comments() << "%)</td>"
144         << "<td>" << i->date()(timefmt) << "</td>"
145         <<"</tr>";
146    }
147
148    os << "</table></div>\n";
149   
150  }
151
152
153  void print_recent_logs(std::ostream& os, const SVNlog& log)
154  {
155    os << "<div class=\"main\">\n"
156       << "<table class=\"main\"><thead><tr><th colspan=\"2\">" 
157       << "Recent Log"
158       << "</th></tr></thead>\n";
159
160    std::vector<std::string>::const_reverse_iterator a=log.author().rbegin();
161    std::vector<std::string>::const_reverse_iterator d=log.date().rbegin();
162    std::vector<std::string>::const_reverse_iterator m=log.message().rbegin();
163    std::vector<size_t>::const_reverse_iterator r=log.revision().rbegin();
164    assert(log.author().size()==log.date().size());
165    assert(log.author().size()==log.message().size());
166    assert(log.author().size()==log.revision().size());
167    os << "<tr><td>Author</td><td>Date</td><td>Rev</td><td>Message</td></tr>\n";
168    HtmlStream hs(os);
169    std::string timefmt("%b %d %H:%M:%S %Y");
170    const size_t maxlength = 80;
171    const Configuration& conf = Configuration::instance();
172    for (size_t i=0; i<10 && a!=log.author().rend(); ++i) {
173      os << "<tr><td>" << anchor(*a+"/total/index.html",*a) << "</td>";
174      Date date(*d);
175      os << "<td>" << date(timefmt) << "</td>";
176      os << "<td>"; 
177      os << trac_revision(*r);
178      os << "</td>";
179      os << "<td>";
180      std::string mess = *m;
181      // replace newlines with space
182      std::replace(mess.begin(), mess.end(), '\n', ' ');
183      mess = htrim(mess);
184
185      if (conf.trac_root().empty()) {
186        // truncate message if too long
187        if (mess.size()>maxlength)
188          mess = mess.substr(0,maxlength-3) + "...";
189        hs << mess;
190      }
191      else {// make anchors to trac
192        Trac trac(hs);
193        trac.print(mess, maxlength);
194      }
195
196      os << "</td></tr>";
197      ++a;
198      ++d;
199      ++m;
200      ++r;
201    }
202    os << "</table></div>\n";
203  }
204
205
206}} // end of namespace svndigest and namespace theplu
Note: See TracBrowser for help on using the repository browser.