source: branches/0.6-stable/lib/Node.cc @ 451

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

changing lev.thep.lu.se to trac.thep.lu.se

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