source: trunk/lib/ClassicStats.cc @ 519

Last change on this file since 519 was 519, checked in by Jari Häkkinen, 14 years ago

trac moved to new location.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 8.8 KB
RevLine 
[84]1// $Id: ClassicStats.cc 519 2007-12-23 20:14:50Z jari $
[14]2
[84]3/*
4  Copyright (C) 2005 Peter Johansson
[381]5  Copyright (C) 2006, 2007 Jari Häkkinen, Peter Johansson
[84]6
[519]7  This file is part of svndigest, http://trac.thep.lu.se/svndigest
[84]8
[149]9  svndigest is free software; you can redistribute it and/or modify it
[84]10  under the terms of the GNU General Public License as published by
11  the Free Software Foundation; either version 2 of the License, or
12  (at your option) any later version.
13
[149]14  svndigest is distributed in the hope that it will be useful, but
[84]15  WITHOUT ANY WARRANTY; without even the implied warranty of
[149]16  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
[84]17  General Public License for more details.
18
19  You should have received a copy of the GNU General Public License
20  along with this program; if not, write to the Free Software
21  Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
22  02111-1307, USA.
23*/
24
[487]25#include "ClassicStats.h"
[129]26
[465]27#include "Functor.h"
[73]28#include "GnuplotFE.h"
[138]29#include "SVNblame.h"
[129]30#include "SVNinfo.h"
[14]31#include "utility.h"
32
[41]33#include <algorithm>
[65]34#include <cassert>
[63]35#include <cstdlib>
[454]36#include <fstream>
[60]37#include <iostream>
[454]38#include <iterator>
[14]39#include <map>
40#include <numeric>
41#include <string>
[138]42#include <sstream>
[44]43#include <unistd.h>
[23]44#include <utility>
[14]45#include <vector>
46
[63]47
[14]48namespace theplu{
[149]49namespace svndigest{
[14]50
51
[487]52  ClassicStats::ClassicStats(const std::string& path)
53    : Stats(path)
[80]54  {
55  }
56
57
[487]58  std::vector<u_int> ClassicStats::accumulated(const Map_& map) const
[14]59  {
[36]60    // sum of all users
[487]61    std::vector<u_int> sum(revision()+1);
[118]62    sum=std::accumulate(map.begin(), map.end(), sum,
[36]63                        PairValuePlus<std::string,u_int>());
[14]64
65    // calculate accumulated sum
[74]66    std::vector<u_int> accum(sum.size());
[60]67    std::partial_sum(sum.begin(),sum.end(),accum.begin());
[63]68    assert(sum.size()==accum.size());
[14]69    return accum;
70  }
71
[487]72  std::vector<u_int> ClassicStats::accumulated(const Map_& map, 
[118]73                                        const std::string& user) const
[14]74  {
[118]75    if (!map.count(user))
[487]76      return std::vector<u_int>(last_changed_rev(),0);
[118]77    std::vector<u_int> vec=(map.find(user))->second;
[131]78
79    // static_cast to remove annoying compiler warning
[487]80    if (vec.size() < static_cast<size_t>(revision()+1))
81      vec.insert(vec.end(), revision()+1-vec.size(), 0);
[35]82
[74]83    std::vector<u_int> accum(vec.size());
[14]84    std::partial_sum(vec.begin(),vec.end(),accum.begin());
85    return accum;
86  }
87
[487]88  void ClassicStats::add(const std::string& user, const u_int& rev, 
89                         const Parser::line_type& lt)
[14]90  {
[482]91    assert(user.size());
[487]92    add_author(user);
[118]93
94    std::vector<u_int>* total = &(total_[user]);
[265]95    assert(total);
[118]96    if (total->size() < rev+1){
[487]97      total->reserve(revision() + 1);
[118]98      total->insert(total->end(), rev - total->size(), 0);
99      total->push_back(1);
[14]100    }
101    else
[173]102      ++(*total)[rev];
[118]103
104    std::vector<u_int>* code = &(code_[user]);
[265]105    assert(code);
[118]106    if (code->size() < rev+1){
[487]107      code->reserve(revision() + 1);
[118]108      code->insert(code->end(), rev - code->size(), 0);
109      if (lt == Parser::code)
110        code->push_back(1);
111      else 
112        code->push_back(0);
113    }
114    else if (lt == Parser::code)
[173]115      ++(*code)[rev];
[118]116
117    std::vector<u_int>* comments = &(comments_[user]);
[265]118    assert(comments);
[118]119    if (comments->size() < rev+1){
[487]120      comments->reserve(revision() + 1);
[118]121      comments->insert(comments->end(), rev - comments->size(), 0);
122      if (lt == Parser::comment)
123        comments->push_back(1);
124      else 
125        comments->push_back(0);
126    }
127    else if (lt == Parser::comment)
[173]128      ++(*comments)[rev];
[202]129
130    std::vector<u_int>* empty = &(empty_[user]);
[265]131    assert(empty);
[202]132    if (empty->size() < rev+1){
[487]133      empty->reserve(revision() + 1);
[202]134      empty->insert(empty->end(), rev - empty->size(), 0);
[208]135      if (lt == Parser::empty)
[202]136        empty->push_back(1);
137      else 
138        empty->push_back(0);
139    }
[208]140    else if (lt == Parser::empty)
[202]141      ++(*empty)[rev];
[14]142  }
143
[60]144
[487]145  bool ClassicStats::do_load_cache(std::istream& is)
[468]146  {
[482]147    svn_revnum_t rev;
148    is >> rev;
[487]149    if (rev<last_changed_rev()){
[482]150      return false; // cache is not up to date
151    }
[487]152    reset();
[468]153    size_t a_size=0;
154    is >> a_size;
155    std::string str;
[487]156    while (authors().size()<a_size){
[468]157      getline(is, str);
[482]158      assert(str.size());
[487]159      add_author(str);
[468]160    }
[482]161    getline(is, str);
162    if (str!=code_cache()){
163      return false;
164    }
[468]165    load(is, code_);
[482]166    getline(is, str);
167    getline(is, str);
168    if (str!=comments_cache()){
169      return false;
170    }
[468]171    load(is, comments_);
[482]172    getline(is, str);
173    getline(is, str);
174    if (str!=empty_cache()){
175      return false;
176    }
[468]177    load(is, empty_);
[482]178    getline(is, str);
179    getline(is, str);
180    if (str!=total_cache()){
181      return false;
182    }
[468]183    load(is, total_);
[482]184    getline(is,str);
185    getline(is,str);
186    return str==end_of_cache();
[468]187  }
188
189
[487]190  void ClassicStats::load(std::istream& is, Map_& m)
[468]191  {
[487]192    while (m.size() < authors().size() && is.good()) {
[468]193      std::string name;
194      std::getline(is, name);
[482]195      assert(name.size());
[468]196      std::vector<u_int>& vec=m[name];
197      size_t revs=0;
198      is >> revs;
199      vec.reserve(revs);
200      while (vec.size() < revs) {
201        u_int tmp;
202        is >> tmp;
203        vec.push_back(tmp);
204      }
205    }
206  }
207
208
[487]209  void ClassicStats::do_parse(const std::string& path)
[60]210  {
[118]211    Parser parser(path);
212    std::vector<Parser::line_type>::const_iterator count=parser.type().begin();
213
[185]214    SVNblame svn_blame(path);
[381]215    while (svn_blame.valid()) {
216      add(svn_blame.author(), svn_blame.revision(), *count);
217      svn_blame.next_line();
[361]218      ++count;
[138]219    }
[60]220  }
221
222
[487]223  std::string ClassicStats::do_plot(const std::string& filename,
224                                    const std::string& linetype) const
[345]225  {
226    plot_init(filename);
227    GnuplotFE* gp=GnuplotFE::instance();
[209]228    const Map_* stat=NULL;
229    if (linetype=="total")
230      stat = &total_;
231    else if (linetype=="code")
232      stat = &code_;
233    else if (linetype=="comments")
234      stat = &comments_;
235    else if (linetype=="empty")
236      stat = &empty_;
237    assert(stat);
238    std::vector<u_int> total=accumulated(*stat);   
[76]239    double yrange_max=1.03*total.back()+1;
[74]240    gp->yrange(yrange_max);
[371]241
242    typedef std::vector<std::pair<std::string, std::vector<u_int> > > vec_type;
243    vec_type author_cont;
244    author_cont.reserve(stat->size());
245    for (MapConstIter_ i= stat->begin(); i != stat->end(); ++i) {
246      author_cont.push_back(std::make_pair(i->first,
247                                           accumulated(*stat,i->first)));
248    }
249
250    LessReversed<std::vector<u_int> > lr;
251    PairSecondCompare<std::string, std::vector<u_int>, 
252      LessReversed<std::vector<u_int> > > compare(lr);
253    std::sort(author_cont.begin(), author_cont.end(), compare);
254
255    size_t plotno=author_cont.size();
[77]256    std::stringstream ss;
[371]257    vec_type::iterator end(author_cont.end());
258    for (vec_type::iterator i(author_cont.begin()); i!=end; ++i) {
[77]259      ss.str("");
260      ss << "set key height " << 2*plotno;
261      gp->command(ss.str());
262      ss.str("");
[371]263      ss << i->second.back() << " " << i->first;
[74]264      gp->yrange(yrange_max);
[77]265      gp->linetitle(ss.str());
266      ss.str("");
[371]267      ss << "steps " << --plotno+2;
[77]268      gp->linestyle(ss.str());
[382]269      gp->plot(i->second);
[39]270    }
[77]271    ss.str("");
272    ss << total.back() << " total";
[76]273    gp->command("set key height 0");
[77]274    gp->linetitle(ss.str());
[76]275    gp->linestyle("steps 1");
276    gp->plot(total);
277
[73]278    gp->command("unset multiplot");
[74]279    gp->yrange();
[36]280
[101]281    return filename;
[34]282  }
283
[118]284
[487]285  void ClassicStats::do_print(std::ostream& os) const
[382]286  {
[487]287    os << last_changed_rev() << " ";
288    os << authors().size() << " ";
[382]289
[487]290    std::copy(authors().begin(), authors().end(), 
[454]291              std::ostream_iterator<std::string>(os, "\n"));
[482]292    os << code_cache() << "\n";
[487]293    do_print(os, code_);
[482]294    os << "\n" << comments_cache() << "\n";
[487]295    do_print(os, comments_);
[482]296    os << "\n" << empty_cache() << "\n";
[487]297    do_print(os, empty_);
[482]298    os << "\n" << total_cache() << "\n";
[487]299    do_print(os, total_);
[482]300    os << "\n" << end_of_cache() << "\n";
[454]301  }
302
303
[487]304  void ClassicStats::do_print(std::ostream& os, const Map_& m) const
[454]305  {
306    for (MapConstIter_ i(m.begin()); i!=m.end(); ++i){
307      os << i->first << "\n";
[482]308      os << i->second.size() << " ";
[454]309      std::copy(i->second.begin(), i->second.end(),
[482]310                std::ostream_iterator<u_int>(os, " "));
[454]311    }
312  }
313
[487]314 
315  std::vector<u_int> ClassicStats::vector(std::string type, 
316                                          std::string user) const
317  {
318    const Map_* map = NULL;
319    if (type=="code")
320      map = &code_;
321    else if (type=="comments")
322      map = &comments_;
323    else if (type=="other")
324      map = &empty_;
325    assert(map && "type is of invalid type");
326    if (!map->count(user))
327      return std::vector<u_int>(last_changed_rev(),0);
328    const std::vector<u_int>& vec=(map->find(user))->second;
329   
330    std::vector<u_int> accum(vec.size());
331    std::partial_sum(vec.begin(),vec.end(),accum.begin());
332    // static_cast to remove annoying compiler warning
333    if (accum.size() < static_cast<size_t>(revision()+1))
334      accum.insert(accum.end(), revision()+1-vec.size(), accum.back());
335    return accum;
[454]336
[487]337  }
338
339
340  ClassicStats& ClassicStats::operator+=(const ClassicStats& rhs)
[14]341  {
[487]342    base_add(rhs.code_.begin(), rhs.code_.end(), code_);
343    base_add(rhs.comments_.begin(), rhs.comments_.end(), comments_);
344    base_add(rhs.empty_.begin(), rhs.empty_.end(), empty_);
345    base_add(rhs.total_.begin(), rhs.total_.end(), total_);
[118]346   
[487]347    base_add(rhs);
[14]348    return *this;
349  }
350
[149]351}} // end of namespace svndigest and namespace theplu
Note: See TracBrowser for help on using the repository browser.