| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162 | 
							- package plumbing
 
- import (
 
- 	"unicode/utf8"
 
- 	"github.com/sergi/go-diff/diffmatchpatch"
 
- 	"gopkg.in/src-d/go-git.v4"
 
- 	"gopkg.in/src-d/go-git.v4/plumbing"
 
- 	"gopkg.in/src-d/go-git.v4/plumbing/object"
 
- 	"gopkg.in/src-d/go-git.v4/utils/merkletrie"
 
- 	"gopkg.in/src-d/hercules.v9/internal/core"
 
- )
 
- // LinesStatsCalculator measures line statistics for each text file in the commit.
 
- type LinesStatsCalculator struct {
 
- 	core.NoopMerger
 
- }
 
- // LineStats holds the numbers of inserted, deleted and changed lines.
 
- type LineStats struct {
 
- 	// Added is the number of added lines by a particular developer in a particular day.
 
- 	Added int
 
- 	// Removed is the number of removed lines by a particular developer in a particular day.
 
- 	Removed int
 
- 	// Changed is the number of changed lines by a particular developer in a particular day.
 
- 	Changed int
 
- }
 
- const (
 
- 	// DependencyLineStats is the identifier of the data provided by LinesStatsCalculator - line
 
- 	// statistics for each file in the commit.
 
- 	DependencyLineStats = "line_stats"
 
- )
 
- // Name of this PipelineItem. Uniquely identifies the type, used for mapping keys, etc.
 
- func (lsc *LinesStatsCalculator) Name() string {
 
- 	return "LinesStats"
 
- }
 
- // Provides returns the list of names of entities which are produced by this PipelineItem.
 
- // Each produced entity will be inserted into `deps` of dependent Consume()-s according
 
- // to this list. Also used by core.Registry to build the global map of providers.
 
- func (lsc *LinesStatsCalculator) Provides() []string {
 
- 	arr := [...]string{DependencyLineStats}
 
- 	return arr[:]
 
- }
 
- // Requires returns the list of names of entities which are needed by this PipelineItem.
 
- // Each requested entity will be inserted into `deps` of Consume(). In turn, those
 
- // entities are Provides() upstream.
 
- func (lsc *LinesStatsCalculator) Requires() []string {
 
- 	arr := [...]string{DependencyTreeChanges, DependencyBlobCache, DependencyFileDiff}
 
- 	return arr[:]
 
- }
 
- // ListConfigurationOptions returns the list of changeable public properties of this PipelineItem.
 
- func (lsc *LinesStatsCalculator) ListConfigurationOptions() []core.ConfigurationOption {
 
- 	return nil
 
- }
 
- // Configure sets the properties previously published by ListConfigurationOptions().
 
- func (lsc *LinesStatsCalculator) Configure(facts map[string]interface{}) error {
 
- 	return nil
 
- }
 
- // Initialize resets the temporary caches and prepares this PipelineItem for a series of Consume()
 
- // calls. The repository which is going to be analysed is supplied as an argument.
 
- func (lsc *LinesStatsCalculator) Initialize(repository *git.Repository) error {
 
- 	return nil
 
- }
 
- // Consume runs this PipelineItem on the next commit data.
 
- // `deps` contain all the results from upstream PipelineItem-s as requested by Requires().
 
- // Additionally, DependencyCommit is always present there and represents the analysed *object.Commit.
 
- // This function returns the mapping with analysis results. The keys must be the same as
 
- // in Provides(). If there was an error, nil is returned.
 
- func (lsc *LinesStatsCalculator) Consume(deps map[string]interface{}) (map[string]interface{}, error) {
 
- 	result := map[object.ChangeEntry]LineStats{}
 
- 	if deps[core.DependencyIsMerge].(bool) {
 
- 		// we ignore merge commit diffs
 
- 		// TODO(vmarkovtsev): handle them better
 
- 		return map[string]interface{}{DependencyLineStats: result}, nil
 
- 	}
 
- 	treeDiff := deps[DependencyTreeChanges].(object.Changes)
 
- 	cache := deps[DependencyBlobCache].(map[plumbing.Hash]*CachedBlob)
 
- 	fileDiffs := deps[DependencyFileDiff].(map[string]FileDiffData)
 
- 	for _, change := range treeDiff {
 
- 		action, err := change.Action()
 
- 		if err != nil {
 
- 			return nil, err
 
- 		}
 
- 		switch action {
 
- 		case merkletrie.Insert:
 
- 			blob := cache[change.To.TreeEntry.Hash]
 
- 			lines, err := blob.CountLines()
 
- 			if err != nil {
 
- 				// binary
 
- 				continue
 
- 			}
 
- 			result[change.To] = LineStats{
 
- 				Added:   lines,
 
- 				Removed: 0,
 
- 				Changed: 0,
 
- 			}
 
- 		case merkletrie.Delete:
 
- 			blob := cache[change.From.TreeEntry.Hash]
 
- 			lines, err := blob.CountLines()
 
- 			if err != nil {
 
- 				// binary
 
- 				continue
 
- 			}
 
- 			result[change.From] = LineStats{
 
- 				Added:   0,
 
- 				Removed: lines,
 
- 				Changed: 0,
 
- 			}
 
- 		case merkletrie.Modify:
 
- 			thisDiffs := fileDiffs[change.To.Name]
 
- 			var added, removed, changed, removedPending int
 
- 			for _, edit := range thisDiffs.Diffs {
 
- 				switch edit.Type {
 
- 				case diffmatchpatch.DiffEqual:
 
- 					if removedPending > 0 {
 
- 						removed += removedPending
 
- 					}
 
- 					removedPending = 0
 
- 				case diffmatchpatch.DiffInsert:
 
- 					delta := utf8.RuneCountInString(edit.Text)
 
- 					if removedPending > delta {
 
- 						changed += delta
 
- 						removed += removedPending - delta
 
- 					} else {
 
- 						changed += removedPending
 
- 						added += delta - removedPending
 
- 					}
 
- 					removedPending = 0
 
- 				case diffmatchpatch.DiffDelete:
 
- 					removedPending = utf8.RuneCountInString(edit.Text)
 
- 				}
 
- 			}
 
- 			if removedPending > 0 {
 
- 				removed += removedPending
 
- 			}
 
- 			result[change.To] = LineStats{
 
- 				Added:   added,
 
- 				Removed: removed,
 
- 				Changed: changed,
 
- 			}
 
- 		}
 
- 	}
 
- 	return map[string]interface{}{DependencyLineStats: result}, nil
 
- }
 
- // Fork clones this PipelineItem.
 
- func (lsc *LinesStatsCalculator) Fork(n int) []core.PipelineItem {
 
- 	return core.ForkSamePipelineItem(lsc, n)
 
- }
 
- func init() {
 
- 	core.Registry.Register(&LinesStatsCalculator{})
 
- }
 
 
  |