source: trunk/lib/Node.cc @ 453

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

fixes #253 - caching log in Node

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 7.9 KB
Line 
1// $Id: Node.cc 453 2007-08-17 23:18:42Z peter $
2
3/*
4  Copyright (C) 2005, 2006, 2007 Jari Häkkinen, Peter Johansson
5
6  This file is part of svndigest, http://trac.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 "Node.h"
25
26#include "Date.h"
27#include "html_utility.h"
28#include "SVNlog.h"
29#include "SVNproperty.h"
30#include "utility.h"
31
32#include <cassert>
33#include <ctime>
34#include <fstream>
35#include <iostream>
36#include <sstream>
37
38#include <dirent.h>
39#include <sys/stat.h>
40
41namespace theplu{
42namespace svndigest{
43
44  std::string Node::project_=std::string();
45
46  Node::Node(const u_int level, const std::string& path, 
47             const std::string& local_path)
48    : level_(level), path_(path), stats_(path), log_(NULL), 
49      svninfo_(path)
50  { 
51    SVNproperty property(path);
52    binary_=property.binary();
53    svndigest_ignore_=property.svndigest_ignore();
54    if (Node::project_==std::string()) // no root directory in local path
55      Node::project_ = file_name(path);
56    else if (local_path.empty())
57      local_path_ = file_name(path);
58    else
59      local_path_ = local_path + "/" + file_name(path);
60
61    struct stat nodestat;                // C api from sys/stat.h
62    lstat(path.c_str(),&nodestat);   // C api from sys/stat.h
63    link_ = S_ISLNK(nodestat.st_mode);
64  }
65
66
67  Node::~Node(void)
68  {
69    if (log_)
70      delete log_;
71  }
72
73
74  std::string Node::author(void) const
75  { 
76    if (ignore())
77      return svninfo_.last_changed_author(); 
78    assert(log().author().size());
79    return log().author().back();
80  }
81
82
83  bool Node::dir(void) const
84  {
85    return false;
86  }
87
88
89  void Node::html_tablerow(std::ostream& os, 
90                           const std::string& css_class,
91                           const std::string& user) const
92  {
93    os << "<tr class=\"" << css_class << "\">\n"
94       << "<td class=\"" << node_type() << "\">";
95    if (svndigest_ignore())
96      os << name() << " (<i>svndigest:ignore</i>)";
97    else if (binary())
98      os << name() << " (<i>binary</i>)";
99    else if (link_)
100      os << name() << " (<i>link</i>)";
101    // there is no output for nodes when user has zero contribution
102    else if (user!="all" && !stats_.lines(user))
103      os << name();
104    else
105      os << anchor(href(), name()); 
106    os << "</td>\n"; 
107    if (user=="all") {
108      os << "<td>" << stats_.lines() << "</td>\n"
109         << "<td>" << stats_.code() << "</td>\n"
110         << "<td>" << stats_.comments() << "</td>\n"
111         << "<td>" << stats_.empty() << "</td>\n";
112    }
113    else {
114      os << "<td>" << stats_.lines(user); 
115      if (stats_.lines(user)) 
116        os << " (" << percent(stats_.lines(user),stats_.lines()) << "%)"; 
117      os << "</td>\n";
118      os << "<td>" << stats_.code(user); 
119      if (stats_.code(user)) 
120        os << " (" << percent(stats_.code(user),stats_.code()) << "%)"; 
121      os << "</td>\n";
122      os << "<td>" << stats_.comments(user); 
123      if (stats_.comments(user)) 
124        os << " (" << percent(stats_.comments(user),stats_.comments()) << "%)"; 
125      os << "</td>\n";
126      os << "<td>" << stats_.empty(user); 
127      if (stats_.empty(user)) 
128        os << " (" << percent(stats_.empty(user),stats_.empty()) << "%)"; 
129      os << "</td>\n";
130
131    }
132
133    os << "<td>" << trac_revision(last_changed_rev()) << "</td>\n"
134       << "<td>" << author() << "</td>\n"
135       << "</tr>\n";
136  }
137
138
139  svn_revnum_t Node::last_changed_rev(void) const
140  {
141    if (ignore())
142      return svninfo_.last_changed_rev();
143    assert(log().revision().size());
144    return log().revision().back();
145  }
146
147
148  const SVNlog& Node::log(void) const
149  {
150    if (!log_)
151      if (ignore())
152        log_ = new SVNlog;
153      else
154        log_ = new SVNlog(log_core());
155    return *log_;
156  }
157
158
159  std::string Node::output_dir(void) const
160  {
161    return output_dir_;
162  }
163
164
165  void Node::path_anchor(std::ostream& os) const
166  {
167    os << "<h2 class=\"path\">\n";
168    std::vector<std::string> words;
169    words.reserve(level_+1);
170    std::string word;
171    words.push_back(Node::project_);
172    std::stringstream ss(local_path());
173    while(getline(ss,word,'/'))
174      if (!word.empty()) // ignore double slash in path
175        words.push_back(word);
176    if (words.size()==1)
177      os << anchor("index.html", Node::project_,0, "View " + Node::project_);
178    else {
179      for (size_t i=0; i<words.size()-1; ++i){
180        os << anchor("index.html", words[i], level_-i, "View " + words[i]);
181        os << "<span class=\"sep\">/</span>\n";
182      }
183      os << anchor(href(), words.back(), level_+2-words.size(), 
184             "View " + words.back()); 
185    }
186    os << "\n</h2>\n";
187  }
188
189
190  void Node::print(const bool verbose) const
191  {
192    if (ignore())
193      return;
194    if (verbose)
195      std::cout << "Printing output for " << path_ << std::endl;
196    SVNlog log(path_);
197    print_core("all", "total", log);
198    print_core("all", "code", log);
199    print_core("all", "comments", log);
200    print_core("all", "empty", log);
201
202    for (std::set<std::string>::const_iterator i = stats_.authors().begin();
203         i!=stats_.authors().end(); ++i) {
204      print_core(*i, "total", log);
205      print_core(*i, "code", log);
206      print_core(*i, "comments", log);
207      print_core(*i, "empty", log);
208    }
209   
210    print_core(verbose);
211  }
212
213
214  void Node::print_author_summary(std::ostream& os, std::string line_type,
215                                  const SVNlog& log) const
216  { 
217    os << "<h3>Author Summary</h3>";
218    os << "<table class=\"listings\">\n";
219    os << "<thead>\n";
220    os << "<tr>\n";
221    os << "<th>Author</th>\n";
222    os << "<th>Lines</th>\n";
223    os << "<th>Code</th>\n";
224    os << "<th>Comments</th>\n";
225    os << "<th>Other</th>\n";
226    os << "<th>Revision</th>\n";
227    os << "<th>Date</th>\n";
228    os << "</tr>\n</thead>\n";
229    os << "<tbody>\n";
230
231    std::string color("light");
232    if (!dir()) {
233      os << "<tr class=\"" << color << "\">\n";
234      os << "<td class=\"directory\" colspan=\"7\">";
235      os << anchor("index.html", "../");
236      os << "</td>\n</tr>\n";
237    }
238
239    // print authors
240    const std::string timefmt("%e/%m/%y %H:%M:%S");
241    for (std::set<std::string>::const_iterator i=stats_.authors().begin();
242         i!=stats_.authors().end(); ++i){
243      if (color=="dark")
244        color="light";
245      else
246        color="dark";
247      os << "<tr class=\"" << color << "\"><td>"; 
248      os << anchor(*i+"/"+line_type+"/"+output_path()
249                   ,*i, level_+2, "View statistics for "+*i); 
250      os << "</td><td>" << stats_.lines(*i)
251         << "</td><td>" << stats_.code(*i)
252         << "</td><td>" << stats_.comments(*i)
253         << "</td><td>" << stats_.empty(*i);
254      if (log.exist(*i)) {
255        Commitment lc(log.latest_commit(*i));
256        os << "</td>" << "<td>" << trac_revision(lc.revision()) 
257           << "</td>" << "<td>" << Date(lc.date())(timefmt);
258      }
259      else {
260        os << "</td>" << "<td>N/A"
261           << "</td>" << "<td>N/A";
262      }
263      os << "</td></tr>\n";
264    }
265
266    os << "<tr class=\"" << color << "\">\n";
267    os << "<td>"; 
268    if (dir())
269      if (local_path().empty())
270        os << anchor("all/"+line_type+"/index.html"
271                     ,"Total", level_+2, "View statistics for all"); 
272      else
273        os << anchor("all/"+line_type+"/"+local_path()+"/index.html"
274                     ,"Total", level_+2, "View statistics for all"); 
275    else
276      os << anchor("all/"+line_type+"/"+local_path()+".html"
277                   ,"Total", level_+2, "View statistics for all"); 
278    os << "</td>\n";
279    os << "<td>" << stats_.lines() << "</td>\n";
280    os << "<td>" << stats_.code() << "</td>\n";
281    os << "<td>" << stats_.comments() << "</td>\n";
282    os << "<td>" << stats_.empty() << "</td>\n";
283    Commitment lc(log.latest_commit());
284    os << "<td>" << trac_revision(lc.revision()) << "</td>\n";
285    os << "<td>" << Date(lc.date())(timefmt)<< "</td>\n";
286    os << "</tr>\n";
287    os << "</tbody>\n";
288    os << "</table>\n";
289  }
290
291 
292  std::string Node::url(void) const
293  {
294    return svninfo_.url();
295  }
296
297}} // end of namespace svndigest and namespace theplu
Note: See TracBrowser for help on using the repository browser.