forked from akka/akka
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathauthors.scala
executable file
·69 lines (55 loc) · 2.13 KB
/
authors.scala
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
#!/bin/sh
exec scala "$0" "$@"
!#
/*
* Usage:
* authors.scala tag1 tag2
*
* or if on non unixy os:
* scala authors.scala tag1 tag2
*
* requires scala 2.11.x+ and command line git on path
*/
import scala.sys.process._
require(args.length == 2, "usage: authors prevTag currTag")
val gitCmd = "git log --no-merges --shortstat -z --minimal -w -C " + args(0) + ".." + args(1)
case class Stats(name: String, email: String, commits: Int = 0, inserts: Int = 0, deletes: Int = 0, filesChanged: Int = 0)
val AuthorExp = """Author: (.*) <([^>]+)>""".r
val FilesExp = """(\d+)\sfile[s]? changed""".r
val InsertionsExp = """(\d+)\sinsertion[s]?\(\+\)""".r
val DeletionsExp = """(\d+)\sdeletion[s]?\(-\)""".r
val entries = gitCmd.lines_!.foldLeft("")(_ + "\n" + _).split('\0')
val map = entries.foldLeft(Map.empty[String, Stats]) { (map, entry) =>
val lines = entry.trim.split('\n')
val authorLine = lines(1)
val summary = lines.last
val statsEntry = authorLine match {
case AuthorExp(name, email) =>
map.get(name.toLowerCase).orElse {
// look for same email, but different name
map.values.find(_.email.equalsIgnoreCase(email)).orElse {
Some(Stats(name, email))
}
}
case _ =>
println(s"Unparseable author line: \n$authorLine\n in entry $entry")
None
}
val updatedEntry =
statsEntry.map(entry => summary.trim.split(',').map(_.trim).foldLeft(entry.copy(commits = entry.commits + 1)) {
case (entry, FilesExp(f)) => entry.copy(filesChanged = entry.filesChanged + f.toInt)
case (entry, InsertionsExp(a)) => entry.copy(inserts = entry.inserts + a.toInt)
case (entry, DeletionsExp(d)) => entry.copy(deletes = entry.deletes + d.toInt)
case (entry, uff) =>
println(s"Couldn't parse summary section for $entry '$uff'")
entry
})
updatedEntry.fold(
map
)(entry => map + (entry.name.toLowerCase -> entry))
}
val sorted = map.values.toSeq.sortBy(s => (s.commits, s.inserts + s.deletes)).reverse
println("commits added removed")
sorted.foreach { entry =>
println("%7d%7d%9d %s".format(entry.commits, entry.inserts, entry.deletes, entry.name))
}