source: trunk/lib/Node.cc @ 828

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

fixes #396

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