source: trunk/lib/Node.cc @ 757

Last change on this file since 757 was 757, checked in by Peter Johansson, 12 years ago

closes #185 and avoid copying in recursive log function

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 9.0 KB
Line 
1// $Id: Node.cc 757 2009-01-27 23:00:58Z peter $
2
3/*
4  Copyright (C) 2005, 2006, 2007 Jari Häkkinen, Peter Johansson
5  Copyright (C) 2008 Jari Häkkinen
6  Copyright (C) 2009 Peter Johansson
7
8  This file is part of svndigest, http://dev.thep.lu.se/svndigest
9
10  svndigest is free software; you can redistribute it and/or modify it
11  under the terms of the GNU General Public License as published by
12  the Free Software Foundation; either version 3 of the License, or
13  (at your option) any later version.
14
15  svndigest is distributed in the hope that it will be useful, but
16  WITHOUT ANY WARRANTY; without even the implied warranty of
17  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
18  General Public License for more details.
19
20  You should have received a copy of the GNU General Public License
21  along with svndigest. If not, see <http://www.gnu.org/licenses/>.
22*/
23
24#include "Node.h"
25
26#include "Date.h"
27#include "HtmlStream.h"
28#include "html_utility.h"
29#include "SVNlog.h"
30#include "SVNproperty.h"
31#include "utility.h"
32
33#include <cassert>
34#include <ctime>
35#include <fstream>
36#include <iostream>
37#include <sstream>
38
39#include <dirent.h>
40#include <sys/stat.h>
41
42namespace theplu{
43namespace svndigest{
44
45  std::string Node::project_=std::string();
46
47  Node::Node(const unsigned int level, const std::string& path, 
48             const std::string& local_path)
49    : level_(level), path_(path), stats_(path), log_(NULL), 
50      svninfo_(path)
51  { 
52    SVNproperty property(path);
53    binary_=property.binary();
54    svndigest_ignore_=property.svndigest_ignore();
55    if (Node::project_==std::string()) // no root directory in local path
56      Node::project_ = file_name(path);
57    else if (local_path.empty())
58      local_path_ = file_name(path);
59    else
60      local_path_ = local_path + "/" + file_name(path);
61
62    struct stat nodestat;                // C api from sys/stat.h
63    lstat(path.c_str(),&nodestat);   // C api from sys/stat.h
64    link_ = S_ISLNK(nodestat.st_mode);
65  }
66
67
68  Node::~Node(void)
69  {
70    if (log_)
71      delete log_;
72  }
73
74
75  std::string Node::author(void) const
76  { 
77    if (ignore())
78      return svninfo_.last_changed_author(); 
79    assert(log().commits().size());
80    return log().commits().back().author();
81  }
82
83
84  bool Node::dir(void) const
85  {
86    return false;
87  }
88
89
90  void Node::html_tablerow(std::ostream& os, 
91                           const std::string& stats_type,
92                           const std::string& css_class,
93                           const std::string& user) const
94  {
95    const Stats& stats = stats_[stats_type];
96    os << "<tr class=\"" << css_class << "\">\n"
97       << "<td class=\"" << node_type() << "\">";
98    if (svndigest_ignore())
99      os << name() << " (<i>svndigest:ignore</i>)";
100    else if (binary())
101      os << name() << " (<i>binary</i>)";
102    else if (link_)
103      os << name() << " (<i>link</i>)";
104    // there is no output for nodes when user has zero contribution
105    else if (user!="all" && !stats.lines(user))
106      os << name();
107    else
108      os << anchor(href(), name()); 
109    os << "</td>\n"; 
110    if (user=="all") {
111      os << "<td>" << stats.lines() << "</td>\n"
112         << "<td>" << stats.code() << "</td>\n"
113         << "<td>" << stats.comments() << "</td>\n"
114         << "<td>" << stats.empty() << "</td>\n";
115    }
116    else {
117      os << "<td>" << stats.lines(user); 
118      if (stats.lines(user)) 
119        os << " (" << percent(stats.lines(user),stats.lines()) << "%)"; 
120      os << "</td>\n";
121      os << "<td>" << stats.code(user); 
122      if (stats.code(user)) 
123        os << " (" << percent(stats.code(user),stats.code()) << "%)"; 
124      os << "</td>\n";
125      os << "<td>" << stats.comments(user); 
126      if (stats.comments(user)) 
127        os << " (" << percent(stats.comments(user),stats.comments()) << "%)"; 
128      os << "</td>\n";
129      os << "<td>" << stats.empty(user); 
130      if (stats.empty(user)) 
131        os << " (" << percent(stats.empty(user),stats.empty()) << "%)"; 
132      os << "</td>\n";
133
134    }
135
136    os << "<td>" << trac_revision(last_changed_rev()) << "</td>\n"
137       << "<td>" << author() << "</td>\n"
138       << "</tr>\n";
139  }
140
141
142  svn_revnum_t Node::last_changed_rev(void) const
143  {
144    if (ignore())
145      return svninfo_.last_changed_rev();
146    assert(log().commits().size());
147    return log().commits().back().revision();
148  }
149
150
151  const SVNlog& Node::log(void) const
152  {
153    if (!log_) {
154      if (ignore())
155        log_ = new SVNlog;
156      else {
157        log_ = new SVNlog(path());
158        log_core(*log_);
159      }
160    }
161    return *log_;
162  }
163
164
165  std::string Node::name(void) const 
166  { 
167    std::string res = file_name(path_); 
168    return res;
169  }
170
171
172  std::string Node::output_dir(void) const
173  {
174    return output_dir_;
175  }
176
177
178  void Node::path_anchor(std::ostream& os) const
179  {
180    os << "<h2 class=\"path\">\n";
181    std::vector<std::string> words;
182    words.reserve(level_+1);
183    std::string word;
184    words.push_back(Node::project_);
185    std::stringstream ss(local_path());
186    while(getline(ss,word,'/'))
187      if (!word.empty()) // ignore double slash in path
188        words.push_back(word);
189    if (words.size()==1)
190      os << anchor("index.html", Node::project_,0, "View " + Node::project_);
191    else {
192      for (size_t i=0; i<words.size()-1; ++i){
193        os << anchor("index.html", words[i], level_-i, "View " + words[i]);
194        os << "<span class=\"sep\">/</span>";
195      }
196      os << anchor(href(), words.back(), level_+2-words.size(), 
197             "View " + words.back()); 
198    }
199    os << "\n</h2>\n";
200  }
201
202
203  void Node::print(const bool verbose) const
204  {
205    if (ignore())
206      return;
207    if (verbose)
208      std::cout << "Printing output for " << path_ << std::endl;
209    const SVNlog& log = this->log();
210    typedef std::map<std::string, Stats*>::const_iterator iter;
211
212    const iter end(stats_.stats().end());
213    for (iter i=stats_.stats().begin();i!=end; ++i){
214      print_core(i->first, "all", "total", log);
215      print_core(i->first, "all", "code", log);
216      print_core(i->first, "all", "comments", log);
217      print_core(i->first, "all", "empty", log);
218      for (std::set<std::string>::const_iterator j=i->second->authors().begin();
219         j!=i->second->authors().end(); ++j) {
220        print_core(i->first, *j, "total", log);
221        print_core(i->first, *j, "code", log);
222        print_core(i->first, *j, "comments", log);
223        print_core(i->first, *j, "empty", log);
224      }
225    }
226      print_core(verbose);
227  }
228
229
230  void Node::print_author_summary(std::ostream& os, 
231                                  const Stats& stats,
232                                  const std::string& line_type,
233                                  const SVNlog& log) const
234  { 
235    HtmlStream hs(os);
236    os << "<h3>Author Summary</h3>";
237    os << "<table class=\"listings\">\n";
238    os << "<thead>\n";
239    os << "<tr>\n";
240    os << "<th>Author</th>\n";
241    os << "<th>Lines</th>\n";
242    os << "<th>Code</th>\n";
243    os << "<th>Comments</th>\n";
244    os << "<th>Other</th>\n";
245    os << "<th>Revision</th>\n";
246    os << "<th>Date</th>\n";
247    os << "</tr>\n</thead>\n";
248    os << "<tbody>\n";
249
250    std::string color("light");
251    if (!dir()) {
252      os << "<tr class=\"" << color << "\">\n";
253      os << "<td class=\"directory\" colspan=\"7\">";
254      os << anchor("index.html", "../");
255      os << "</td>\n</tr>\n";
256    }
257
258    // print authors
259    const std::string timefmt("%Y-%m-%d  %H:%M");
260    for (std::set<std::string>::const_iterator i=stats.authors().begin();
261         i!=stats.authors().end(); ++i){
262      if (color=="dark")
263        color="light";
264      else
265        color="dark";
266      os << "<tr class=\"" << color << "\"><td>"; 
267      os << anchor(*i+"/"+line_type+"/"+output_path()
268                   ,*i, level_+2, "View statistics for "+*i); 
269      os << "</td><td>" << stats.lines(*i)
270         << "</td><td>" << stats.code(*i)
271         << "</td><td>" << stats.comments(*i)
272         << "</td><td>" << stats.empty(*i);
273      if (log.exist(*i)) {
274        const Commitment& lc(log.latest_commit(*i));
275        os << "</td>" << "<td>" << trac_revision(lc.revision()) 
276           << "</td>" << "<td>";
277        hs << Date(lc.date())(timefmt);
278      }
279      else {
280        os << "</td>" << "<td>N/A"
281           << "</td>" << "<td>N/A";
282      }
283      os << "</td></tr>\n";
284    }
285
286    os << "<tr class=\"" << color << "\">\n";
287    os << "<td>"; 
288    if (dir())
289      if (local_path().empty())
290        os << anchor("all/"+line_type+"/index.html"
291                     ,"Total", level_+2, "View statistics for all"); 
292      else
293        os << anchor("all/"+line_type+"/"+local_path()+"/index.html"
294                     ,"Total", level_+2, "View statistics for all"); 
295    else
296      os << anchor("all/"+line_type+"/"+local_path()+".html"
297                   ,"Total", level_+2, "View statistics for all"); 
298    os << "</td>\n";
299    os << "<td>" << stats.lines() << "</td>\n";
300    os << "<td>" << stats.code() << "</td>\n";
301    os << "<td>" << stats.comments() << "</td>\n";
302    os << "<td>" << stats.empty() << "</td>\n";
303    const Commitment& lc(log.latest_commit());
304    os << "<td>" << trac_revision(lc.revision()) << "</td>\n";
305    os << "<td>";
306    hs << Date(lc.date())(timefmt);
307    os << "</td>\n";
308    os << "</tr>\n";
309    os << "</tbody>\n";
310    os << "</table>\n";
311  }
312
313 
314  void Node::print_copyright(std::map<std::string,Alias>& alias, 
315                             bool verbose) const
316  {
317    // map with last rev for every year
318    std::map<int, svn_revnum_t> year2rev;
319    SVNlog log(path());
320    typedef std::vector<Commitment>::const_iterator LogIterator;
321    for (LogIterator i=log.commits().begin(); i<log.commits().end(); ++i){
322      time_t sec = str2time(i->date());
323      tm* timeinfo = gmtime(&sec);
324      year2rev[timeinfo->tm_year] = i->revision();
325    }
326    print_copyright(alias, verbose, year2rev);
327  }
328
329
330  std::string Node::url(void) const
331  {
332    return svninfo_.url();
333  }
334
335}} // end of namespace svndigest and namespace theplu
Note: See TracBrowser for help on using the repository browser.