-
Notifications
You must be signed in to change notification settings - Fork 44
/
Copy pathdataset-stats.R
444 lines (384 loc) · 28.9 KB
/
dataset-stats.R
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
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
#
# (c) 2012 -- 2014 Georgios Gousios <[email protected]>
#
# BSD licensed, see LICENSE in top level dir
#
rm(list = ls(all = TRUE))
source(file = "R/packages.R")
source(file = "R/cmdline.R")
source(file = "R/utils.R")
library(RMySQL)
library(ggplot2)
library(reshape)
library(sqldf)
library(digest)
library(xtable)
print("Running database queries...")
con <- dbConnect(dbDriver("MySQL"), user = mysql.user, password = mysql.passwd,
dbname = mysql.db, host = mysql.host)
# Overall project statistics
# if (file.exists(overall.dataset.stats)) {
# print(sprintf("%s file found", overall.dataset.stats))
# projectstats <- read.csv(file = overall.dataset.stats)
# } else {
# print(sprintf("File not found %s", overall.dataset.stats))
# print("Running project stats query, this will take long...")
# # This will take a LONG time. Think >0.5 hours.
# res <- dbSendQuery(con, "select concat(u.login, '/', p.name) as name, (select count(*) from commits c, project_commits pc where pc.project_id = p.id and pc.commit_id = c.id) as commits, (select count(*) from watchers where repo_id = p.id) as watchers, (select count(*) from pull_requests where base_repo_id = p.id) as pull_requests, (select count(*) from issues where repo_id = p.id) as issues, (select count(*) from project_members where repo_id = p.id) as project_members, (select count(distinct c.author_id) from commits c, project_commits pc where pc.project_id = p.id and pc.commit_id = c.id) as contributors, (select count(*) from projects p1 where p1.forked_from = p.id) as forks, (select count(*) from issue_comments ic, issues i where ic.issue_id=i.id and i.repo_id = p.id) as issue_comments, (select count(*) from pull_requests pr, pull_request_comments prc where pr.base_repo_id=p.id and prc.pull_request_id = pr.id) as pull_req_comments, p.language from projects p, users u where p.forked_from is null and p.deleted is false and u.id = p.owner_id group by p.id;")
# projectstats <- fetch(res, n = -1)
# write.csv(projectstats, file = overall.dataset.stats)
# projectstats
# }
# Total repos
res <- dbSendQuery(con, "select count(*) as cnt from projects")
repos <- fetch(res, n = -1)
print(sprintf("Total repos: %d",repos$cnt))
# Total users
res <- dbSendQuery(con, "select count(*) as cnt from users")
users <- fetch(res, n = -1)
print(sprintf("Total users: %d",users$cnt))
# Original repos
q <- "select count(*) as cnt
from projects
where forked_from is null
and deleted is false
and name not regexp '^.*\\.github\\.com$'
and name <> 'try_git'
and name <> 'dotfiles'
and name <> 'vimfiles'"
res <- dbSendQuery(con, unwrap(q))
orig_repos <- fetch(res, n = -1)
print(sprintf("Original repos: %f",orig_repos$cnt))
# % of original repos
print(sprintf("Original repos: %f",(orig_repos$cnt/repos$cnt) * 100))
# Original repositories that received a single commit in 2013
q <- "select count(*) as cnt
from projects p
where forked_from is null
and deleted is false
and name not regexp '^.*\\.github\\.com$'
and name <> 'try_git'
and name <> 'dotfiles'
and name <> 'vimfiles'
and exists (
select *
from project_commits pc, commits c
where pc.project_id = p.id
and c.id = pc.commit_id
and unix_timestamp(c.created_at) between 1359676800 and 1377993600)"
res <- dbSendQuery(con, unwrap(q))
repos_with_commits <- fetch(res, n = -1)
print(sprintf("Original repos with commits Feb-Aug 2013: %f", repos_with_commits$cnt))
# Original repositories that received a single commit in 2012
q <- "select count(*) as cnt
from projects p
where forked_from is null
and deleted is false
and name not regexp '^.*\\.github\\.com$'
and name <> 'try_git'
and name <> 'dotfiles'
and name <> 'vimfiles'
and exists (
select *
from project_commits pc, commits c
where pc.project_id = p.id
and c.id = pc.commit_id
and unix_timestamp(c.created_at) between 1328054400 and 1346457600)"
res <- dbSendQuery(con, unwrap(q))
repos_with_commits.2012 <- fetch(res, n = -1)
print(sprintf("Original repos with commits Feb-Aug 2012: %f", repos_with_commits.2012$cnt))
# Total pull requests
res <- dbSendQuery(con, "select count(*) as cnt from pull_requests")
pullreqs <- fetch(res, n = -1)
print(sprintf("Total pull requests: %d",pullreqs$cnt))
# % of pull req comments by non-repo members
q <- "select count(pr.id) as cnt
from pull_requests pr, pull_request_comments prc
where pr.id = prc.pull_request_id
and not exists (
select *
from project_commits pc, commits c
where pc.commit_id = c.id
and pc.project_id = pr.base_repo_id
and c.author_id = prc.user_id);"
res <- dbSendQuery(con, unwrap(q))
prc_non_members <- fetch(res, n = -1)
print(sprintf("Pull request comments by non project members: %f", prc_non_members$cnt))
#
# % of pull req comments by non-repo members
print(sprintf("% comments from non-repo members: %f",(prc_non_members$cnt/pullreqs$cnt) * 100))
# Pull req comments
q <- "select i.pr_id, ic_cnt + prc_cnt as cnt, i.issue_id
from
(select pr.id as pr_id, i.issue_id as issue_id, count(ic.comment_id) as ic_cnt
from pull_requests pr
left outer join issues i on pr.id = i.pull_request_id
left outer join issue_comments ic on i.id = ic.issue_id
where pr.base_repo_id = i.repo_id group by pr.id) as i,
(select pr.id as pr_id, count(prc.comment_id) as prc_cnt
from projects p
join pull_requests pr on p.id = pr.base_repo_id
left outer join pull_request_comments prc on prc.pull_request_id = pr.id
where p.forked_from is null group by pr.id) as pr
where pr.pr_id = i.pr_id"
res <- dbSendQuery(con, unwrap(q))
prs <- fetch(res, n = -1)
print(sprintf("Num discussion comments per pulreq (mean): %f", mean(prs$cnt)))
print(sprintf("Num discussion comments per pulreq (median): %f", median(prs$cnt))
print(sprintf("Num discussion comments per pulreq (95 perc): %d", quantile(prs$cnt, 0.95)))
print(sprintf("Num discussion comments per pulreq (5 perc): %d", quantile(prs$cnt, 0.05)))
# Original repos with > 1 committers and 0 pull reqs in 2013
q <- "select count(*) as cnt
from projects p
where forked_from is null
and deleted = false
and name not regexp '^.*\\.github\\.com$'
and name <> 'try_git'
and name <> 'dotfiles'
and name <> 'vimfiles' and (
select count(c.author_id)
from project_commits pc, commits c
where pc.project_id = p.id
and c.id = pc.commit_id
and year(c.created_at) = 2013) > 1
and not exists (
select pr.id
from pull_requests pr, pull_request_history prh
where pr.base_repo_id = p.id and prh.pull_request_id = pr.id
and year(prh.created_at)=2013)"
res <- dbSendQuery(con, "")
repos_with_co_op_devs <- fetch(res, n = -1)$cnt
print(sprintf("Repos with > 1 committers and 0 pull reqs in 2013: %d", repos_with_co_op_devs))
## Pull req usage comparison 2012 -- 2013
# Original repositories that have > 1 committers in Feb - Aug 2012
q <- "select count(*) as cnt
from projects p
where forked_from is null
and deleted is false
and name not regexp '^.*\\.github\\.com$'
and name <> 'try_git'
and name <> 'dotfiles'
and name <> 'vimfiles'
and (
select count(distinct c.author_id)
from project_commits pc, commits c
where pc.project_id = p.id
and c.id = pc.commit_id
and unix_timestamp(c.created_at) between 1328054400 and 1346457600) > 1
and not exists (select * from pull_requests pr where p.id = pr.base_repo_id)"
res <- dbSendQuery(con, unwrap(q))
repos_with_commits_2012 <- fetch(res, n = -1)
print(sprintf("Original repos with commits in Feb - Aug 2012: %f", repos_with_commits_2012$cnt))
# Original repos that received a pullreq in Feb - Aug 2012
res <- dbSendQuery(con, "select count(*) as cnt from projects p where p.forked_from is null and deleted is false and p.name not regexp '^.*\\.github\\.com$' and p.name <> 'try_git' and p.name <> 'dotfiles' and exists (select pr.id from pull_requests pr, pull_request_history prh where pr.base_repo_id = p.id and prh.pull_request_id = pr.id and prh.action = 'opened' and unix_timestamp(prh.created_at) between 1328054400 and 1346457600)")
orig_repos_pullreqs_2012 <- fetch(res, n = -1)
print(sprintf("Repos that received a pull request in Feb - Aug 2012: %s", orig_repos_pullreqs_2012$cnt))
print(sprintf("Perc repos that received a pull request in Feb - Aug 2012: %f", (orig_repos_pullreqs_2012$cnt/repos_with_commits_2012$cnt) * 100))
# Original repositories that received a single commit in Feb - Aug 2013
res <- dbSendQuery(con, "select count(*) as cnt from projects p where forked_from is null and deleted is false and name not regexp '^.*\\.github\\.com$' and name <> 'try_git' and name <> 'dotfiles' and name <> 'vimfiles' and (select count(distinct c.author_id) from project_commits pc, commits c where pc.project_id = p.id and c.id = pc.commit_id and unix_timestamp(c.created_at) between 1359676800 and 1377993600) > 1 and not exists (select * from pull_requests pr where p.id = pr.base_repo_id)")
repos_with_commits_2013 <- fetch(res, n = -1)
print(sprintf("Original repos with commit Feb - Aug 2013s: %f", repos_with_commits_2013$cnt))
# Original repos that received a pullreq in Feb - Aug 2013
res <- dbSendQuery(con, "select count(*) as cnt from projects p where p.forked_from is null and deleted is false and p.name not regexp '^.*\\.github\\.com$' and p.name <> 'try_git' and p.name <> 'dotfiles' and exists (select pr.id from pull_requests pr, pull_request_history prh where pr.base_repo_id = p.id and prh.pull_request_id = pr.id and prh.action = 'opened' and unix_timestamp(prh.created_at) between 1359676800 and 1377993600)")
orig_repos_pullreqs <- fetch(res, n = -1)
print(sprintf("Repos that received a pull request in Feb - Aug 2013: %s", orig_repos_pullreqs$cnt))
print(sprintf("Perc repos that received a pull request in Feb - Aug 2013: %f", (orig_repos_pullreqs$cnt/repos_with_commits_2013$cnt) * 100))
# Distribution of pull requests per project
res <- dbSendQuery(con, "select pr.base_repo_id as repoid, count(*) as cnt from projects p, pull_requests pr where p.forked_from is null and p.name not regexp '^.*\\.github\\.com$' and p.name <> 'try_git' and p.name <> 'dotfiles' and p.name <> 'vimfiles' and pr.base_repo_id = p.id and exists (select prh.created_at from pull_request_history prh where prh.pull_request_id = pr.id and prh.action='opened' and year(prh.created_at) > 2010) group by pr.base_repo_id order by count(*) desc")
pullreqs <- fetch(res, n = -1)
print(sprintf("Pullreqs per project (mean): %f", mean(pullreqs$cnt)))
print(sprintf("Pullreqs per project (median): %f", median(pullreqs$cnt)))
print(sprintf("Pullreqs per project (95 perc): %d", quantile(pullreqs$cnt, 0.95)))
print(sprintf("Pullreqs per project (5 perc): %d", quantile(pullreqs$cnt, 0.05)))
store.pdf(qplot(cnt, data = subset(pullreqs, cnt > 10), geom = "histogram", log = "x", ylab = "Number of projects", xlab = "Number of pull requests (log)"), plot.location, "pull-req-freq.pdf")
# Overall pull req stats - opened
res <- dbSendQuery(con, "select count(*) as cnt from projects p, pull_requests pr where p.forked_from is null and p.name not regexp '^.*\\.github\\.com$' and p.name <> 'try_git' and p.name <> 'dotfiles' and p.name <> 'vimfiles'and pr.base_repo_id = p.id and exists (select prh.created_at from pull_request_history prh where prh.pull_request_id = pr.id and prh.action='opened' and year(prh.created_at) > 2010)")
opened_pullreqs <- fetch(res, n = -1)$cnt
# Overall pull req stats - closed
res <- dbSendQuery(con, "select count(*) as cnt from projects p, pull_requests pr where p.forked_from is null and p.name not regexp '^.*\\.github\\.com$' and p.name <> 'try_git' and p.name <> 'dotfiles' and p.name <> 'vimfiles' and pr.base_repo_id = p.id and exists (select prh.created_at from pull_request_history prh where prh.pull_request_id = pr.id and prh.action='closed' and year(prh.created_at) > 2010)")
closed_pullreqs <- fetch(res, n = -1)$cnt
# Overall pull req stats - merged
res <- dbSendQuery(con, "select count(*) as cnt from projects p, pull_requests pr where p.forked_from is null and p.name not regexp '^.*\\.github\\.com$' and p.name <> 'try_git' and p.name <> 'dotfiles' and p.name <> 'vimfiles' and pr.base_repo_id = p.id and pr.merged = true and exists (select prh.created_at from pull_request_history prh where prh.pull_request_id = pr.id and year(prh.created_at) > 2010)")
merged_pullreqs <- fetch(res, n = -1)$cnt
print(sprintf("Perc merged pull requests: %f", (merged_pullreqs/opened_pullreqs) * 100))
# Pull reqs per month plot
res <- dbSendQuery(con, "select last_day(prh.created_at) as cdate, count(*) as pull_reqs from pull_requests pr, pull_request_history prh where prh.pull_request_id = pr.id and prh.action = 'opened' and unix_timestamp(prh.created_at) between 1328054400 and 1396224000 group by month(prh.created_at), year(prh.created_at) order by prh.created_at")
pullreqs_per_month <- fetch(res, n = -1)
res <- dbSendQuery(con, "select cdate, count(repo) as repos_with_pull_reqs from (select last_day(prh.created_at) as cdate, pr.base_repo_id as repo from pull_requests pr, pull_request_history prh where pr.id = prh.pull_request_id and prh.action = 'opened' and unix_timestamp(prh.created_at) between 1328054400 and 1396224000 group by pr.base_repo_id, month(prh.created_at), year(prh.created_at) order by cdate) as a group by cdate")
repos_with_pullreqs_per_month <- fetch(res, n = -1)
pullreqs_per_month <- merge(pullreqs_per_month, repos_with_pullreqs_per_month, by = "cdate")
pullreqs_per_month$cdate <- as.POSIXct(pullreqs_per_month$cdate, origin = "1970-01-01")
#pullreqs_per_month$ratio <- pullreqs_per_month$pull_reqs / pullreqs_per_month$repos
store.pdf(ggplot(pullreqs_per_month, aes(x = cdate)) +
#scale_x_datetime(breaks = date_breaks(width = "6 months")) +
scale_x_datetime(breaks = c(as.POSIXct(1327968000, origin = "1970-01-01"),
as.POSIXct(1343775600, origin = "1970-01-01"),
as.POSIXct(1359676800, origin = "1970-01-01"),
as.POSIXct(1375311600, origin = "1970-01-01"))) +
scale_y_continuous(limits=c(1,5)) +
geom_line(aes(y = ratio, colour = "1")) +
stat_smooth(aes(y = ratio, color = "2"), method = "loess", formula = y ~ x^2, size = 1, alpha = 0) +
xlab("Date") +
ylab("Number of pull requests per repo per month")+
scale_colour_manual(values=c("red", "blue"), labels = c("actual", "trend")) +
theme(legend.title=element_blank()),
plot.location,"num-pullreqs-month.pdf")
pullreqs_per_month <- melt(pullreqs_per_month, id.vars = "cdate")
store.pdf(ggplot(pullreqs_per_month, aes(x = cdate, y = value, fill = variable)) +
scale_x_datetime() +
geom_bar(position = "dodge", stat = "identity") +
stat_smooth(method = "loess", formula = y ~ x^2, size = 1, alpha = 0) +
facet_grid(. ~ variable) +
scale_y_continuous(limits=c(0,160000)) +
xlab("Date") + ylab("Number") +
theme(legend.position="none", axis.text.x = element_text(angle = 90)), plot.location, "pullreqs-per-month.pdf")
# Correlation between pull requests and watchers
res <- dbSendQuery(con, "select count(pr.id) as pullreqs, (select count(*) from watchers w where w.repo_id = pr.base_repo_id) as watchers from pull_requests pr group by pr.base_repo_id")
pr_watchers <- fetch(res, n = -1)
cor_pr_watch <- cor.test(pr_watchers$pullreqs, pr_watchers$watchers, method="spearman")
print(sprintf("Spearman correlation pullreqs-watchers: %f (n = %d, p = %f)",
cor_pr_watch$estimate, nrow(pr_watchers), cor_pr_watch$p.value))
# Forked repos
res <- dbSendQuery(con, "select count(*) as cnt from projects where forked_from is not null")
forked_repos <- fetch(res, n = -1)
print(sprintf("Perc forked repos: %f", (forked_repos/repos) * 100))
# Issue tracker % usage for projects
res <- dbSendQuery(con, "select count(*) as cnt from projects p where p.forked_from is null and p.name not regexp '^.*\\.github\\.com$' and p.name <> 'try_git' and p.name <> 'dotfiles' and p.name <> 'vimfiles' and exists (select pr.id from pull_requests pr, pull_request_history prh where pr.base_repo_id = p.id and prh.pull_request_id = pr.id and year(prh.created_at)=2012) and exists (select i.id from issues i where i.repo_id = p.id and year(i.created_at)=2012)")
issue_pr_repos <- fetch(res, n = -1)
print(sprintf("Perc projects with pull reqs and issues (overall): %f", (issue_pr_repos/orig_repos) * 100))
print(sprintf("Perc projects with pull reqs and issues: %f", (issue_pr_repos/orig_repos_pullreqs) * 100))
# Commiter raise due to pull requests
# res <- dbSendQuery(con, "select p.id from projects p, users u, commits c, project_commits pc where p.forked_from is null and u.id = p.owner_id and pc.commit_id = c.id and pc.project_id = p.id and exists (select c.id from project_commits pc, commits c where year(c.created_at) = 2012 and month (c.created_at)=12 and pc.commit_id = c.id and pc.project_id = p.id ) and exists (select c.id from project_commits pc, commits c where year(c.created_at) < 2007 and pc.commit_id = c.id and pc.project_id = p.id ) and exists (select pr.id from pull_requests pr, pull_request_history prh where prh.pull_request_id = pr.id and pr.base_repo_id = p.id and prh.action='opened' and year(prh.created_at) = 2010) group by p.id having count(c.id) > 1000 order by p.created_at asc limit 200")
# Run the above query and handpick projects
# 1334 rails rails
# 3189 junit KentBeck
# 6313 jquery jquery
# 4367 phpbb3 phpbb
# 18716 rubygems rubygems
# 9316 cakephp cakephp
# 2602 monodevelop mono
# 847 puppet puppetlabs
# 5591 facter puppetlabs
res <- dbSendQuery(con, "select p.name as project, last_day(c.created_at) as timestamp, count(distinct c.author_id) as num_authors from commits c, project_commits pc, projects p where pc.commit_id = c.id and pc.project_id = p.id and year(c.created_at) < 2013 and p.id in (3189, 6313, 4367, 18716, 9316, 2602, 847, 5591) group by p.id, month(c.created_at), year(c.created_at) order by p.name, year(c.created_at), month(c.created_at)")
devs_per_month <- fetch(res, n = -1)
devs_per_month$month <- as.POSIXct(devs_per_month$timestamp, origin = "1970-01-01")
store.pdf(ggplot(devs_per_month, aes(x = month, y = num_authors, colour = project)) +
scale_x_datetime() +
stat_smooth(method = "loess", formula = y ~ x^2, size = 1, alpha = 0) +
annotate("pointrange", x = as.POSIXct(1283174614, origin = "1970-01-01"), ymin = 0, ymax = 20, y = 20) +
annotate("text", x = as.POSIXct(1283174614, origin = "1970-01-01"), y = 20, label ="Github introduces pull requests", size = 4) +
ylim(0, 22) +
xlab("Date") +
ylab("Number of active committers per month"), plot.location,"num-commiters-after-pr.pdf")
res <- dbSendQuery(con, "select a.project as project, avg(a.num_authors) as avg_after, avg(b.num_authors) as avg_before from (select p.id as project, last_day(c.created_at) as timestamp_after_pr, count(distinct c.author_id) as num_authors from commits c, project_commits pc, projects p where pc.commit_id = c.id and pc.project_id = p.id and exists(select pc1.* from project_commits pc1, commits c1 where pc1.commit_id = c1.id and year(c1.created_at) = 2010 and month(c1.created_at) < 9 and pc1.project_id = p.id) and c.created_at between makedate(2010, 213) and makedate(2011, 244) and p.forked_from is null group by p.id, month(c.created_at), year(c.created_at)) as a, (select p.id as project, last_day(c.created_at) as timestamp_before_pr, count(distinct c.author_id) as num_authors from commits c, project_commits pc, projects p where pc.commit_id = c.id and pc.project_id = p.id and exists(select pc1.* from project_commits pc1, commits c1 where pc1.commit_id = c1.id and year(c1.created_at) = 2010 and month(c1.created_at) < 9 and pc1.project_id = p.id) and c.created_at between makedate(2009, 244) and makedate(2010, 212) and p.forked_from is null group by p.id, month(c.created_at), year(c.created_at)) as b where a.project = b.project and exists(select pr.id from pull_requests pr, pull_request_history prh where pr.base_repo_id = a.project and prh.pull_request_id = pr.id and prh.created_at between makedate(2010, 213) and makedate(2011, 244)) group by a.project")
mean_commiters_per_month <- fetch(res, n = -1)
a <- subset(mean_commiters_per_month, avg_after > 2)
w <- wilcox.test(a$avg_after, a$avg_before, paired = TRUE)
print(sprintf("Wilcox: mean devs before/after pullreqs: n = %d, V = %f, p < %f", nrow(a), w$statistic, w$p.value))
print(sprintf("Cliff's delta on number of avg number of committers before/after pullreqs: %f", cliffs.d(a$avg_after, a$avg_before)))
ranksum(a$avg_after, a$avg_before)
# Drive-by commits pull reqs
res <- dbSendQuery(con, "select count(*) as cnt from pull_requests pr, pull_request_history prh where prh.action = 'opened' and prh.pull_request_id = pr.id and year(prh.created_at) = 2012 and not exists (select c.author_id from commits c, project_commits pc where pc.project_id = pr.base_repo_id and c.created_at < prh.created_at and c.author_id = pr.user_id) and 1 = (select count(*) from pull_request_commits prc where prc.pull_request_id = pr.id)")
drive_by_pr <- fetch(res, n = -1)$cnt
print(sprintf("Perc drive by pull requests: %f", (drive_by_pr/opened_pullreqs) * 100))
print(sprintf("Perc one pull req repos: %f", (drive_by_pr/forked_repos) * 100))
# Load CSV files
all <- load.data(project.list)
# Discussion comments from internal vs externals
project_ids <- lapply(unique(all$project_name), function(x) {
printf("Aquiring id for project %s", x)
details <- strsplit(as.character(x),'/')[[1]]
res <- dbSendQuery(con, sprintf("select p.id from projects p, users u where u.id = p.owner_id and u.login = '%s' and p.name = '%s'", details[1], details[2]))
fetch(res, n = -1)$id
})
comments <- data.frame()
for (pid in project_ids) {
printf("Running for project %d ", pid)
res <- dbSendQuery(con, sprintf("select a.p_id, concat(u.login, '/', p.name) as project_name, (select count(pm.user_id) from project_members pm where pm.user_id = a.user_id and pm.repo_id = a.p_id) as is_member, count(distinct user_id) as num_users, sum(a.cnt) as num_comments from ( (select pr.base_repo_id as p_id, ic.user_id as user_id, count(ic.comment_id) as cnt from projects p join pull_requests pr on p.id = pr.base_repo_id left outer join issues i on pr.pullreq_id = i.issue_id left outer join issue_comments ic on i.id = ic.issue_id where p.forked_from is null and p.id = %d and pr.base_repo_id = i.repo_id group by pr.base_repo_id, ic.user_id) union (select pr.base_repo_id as p_id, prc.user_id as user_id, count(prc.comment_id) as cnt from projects p join pull_requests pr on p.id = pr.base_repo_id left outer join pull_request_comments prc on prc.pull_request_id = pr.id where p.forked_from is null and p.id = %d group by pr.base_repo_id, prc.user_id) ) as a, users u, projects p where p.owner_id = u.id and p.id = a.p_id group by a.p_id, is_member", pid, pid))
d <- fetch(res, n = -1)
comments <- rbind(comments, d)
comments
}
comments <- subset(comments, num_users > 0 & num_comments >0)
processed_comments <- sqldf("select project_name, (select c1.num_users from comments c1 where c1.p_id = c.p_id and c1.is_member = 0)/sum(c.num_users) as commenters, (select c2.num_comments from comments c2 where c2.p_id = c.p_id and c2.is_member = 0)/sum(c.num_comments) as comments from comments c group by c.p_id", drv="SQLite")
processed_comments <- melt(processed_comments, 'project_name', na.rm = TRUE)
processed_comments$project_name <- as.factor(processed_comments$project_name)
processed_comments$value <- processed_comments$value * 100
#processed_comments$foo <- match(processed_comments$value[processed_comments$variable == "comments"], sort(processed_comments$value[processed_comments$variable == "comments"]))
processed_comments$foo <- match(processed_comments$value[processed_comments$variable == "commenters"], sort(processed_comments$value[processed_comments$variable == "commenters"]))
processed_comments$foo <- as.factor(processed_comments$foo)
p <- ggplot(processed_comments, aes(x = foo, y = value, fill = variable)) +
scale_x_discrete() + xlab("Project") + ylab("%") +
geom_bar(position="dodge") +
#theme(axis.text.x=element_text(angle = 90, size = 6), legend.position="none") +
theme(axis.text.x=element_blank(), legend.position="none") +
#ggtitle("% of external commenters/comments per project") +
facet_grid(. ~ variable)
store.pdf(p, plot.location, "perc-external-commenters-comments.pdf")
# Code review, pimp the data frame with info
has.code.review <- function(pr_id) {
q = sprintf("select count(*) as cnt from pull_request_comments prc where prc.pull_request_id = %d", pr_id)
res <- dbSendQuery(con,q)
num_code_review <- fetch(res, n = -1)
num_code_review > 0
}
all$code_review <- lapply(all$pull_req_id, has.code.review)
# Various statistics on comments from community
mean_external_contribs <- aggregate(perc_external_contribs ~ project_name, all, mean)
joined <- merge(mean_external_contribs, processed_comments, by = "project_name")
a <- subset(joined, variable == "commenters", perc_external_contribs)
b <- subset(joined, variable == "comments", value)
cor.test(a$perc_external_contribs, b$value, method="spearman")
b <- subset(joined, variable == "commenters", value)
cor.test(a$perc_external_contribs, b$value, method="spearman")
nrow(subset(processed_comments, variable == "commenters" & value > 50))
###
# Overall statistics table
cache <- new.env()
overall.stats <- data.frame(
description = c('Number of participants', 'Number of comments',
'Number of commits', 'Time to merge (min)',
'Time to close (min)'),
query = c(
"select (select count(distinct(prh.actor_id)) from pull_request_history prh where prh.pull_request_id = pr.id) + (select count(distinct(ie.actor_id)) from issue_events ie, issues i where i.id = ie.issue_id and i.pull_request_id = pr.id) as participants from pull_requests pr group by id;",
"select (select count(*) from issue_comments ic, issues i where ic.issue_id = i.id and i.pull_request_id = pr.id) + (select count(*) from pull_request_comments prc where prc.pull_request_id = pr.id ) as num_comments from pull_requests pr group by pr.id;",
"select count(*) as number_of_commits from pull_request_commits group by pull_request_id;",
"select timestampdiff(minute, a.created_at, b.created_at) as mergetime_minutes from pull_request_history a, pull_request_history b where a.action = 'opened' and b.action ='merged' and a.pull_request_id = b.pull_request_id group by a.pull_request_id;",
"select timestampdiff(minute, a.created_at, b.created_at) as lifetime_minutes from pull_request_history a, pull_request_history b where a.action = 'opened' and b.action ='closed' and not exists (select * from pull_request_history c where c.pull_request_id = a.pull_request_id and c.action = 'merged') and a.pull_request_id = b.pull_request_id group by a.pull_request_id;"
)
)
do.query <- function(con, cache, x) {
print(x)
md5 <- digest(x)
if (is.null(cache[[md5]])) {
res <- dbSendQuery(con, x)
r <- fetch(res, n = -1)[[1]]
cache[[md5]] <- r
}
cache[[md5]]
}
fix <- function(x) { if(x < 0) {0} else {x}}
overall.stats$min <- lapply(overall.stats$query, function(x){fix(min(do.query(con, cache, as.character(x))))})
overall.stats$quant_5 <- lapply(overall.stats$query, function(x){fix(quantile(do.query(con, cache, as.character(x)),0.05))})
overall.stats$median <- lapply(overall.stats$query, function(x){fix(median(do.query(con, cache, as.character(x))))})
overall.stats$mean <- lapply(overall.stats$query, function(x){fix(mean(do.query(con, cache, as.character(x))))})
overall.stats$quant_95 <- lapply(overall.stats$query, function(x){fix(quantile(do.query(con, cache, as.character(x)),0.95))})
overall.stats$max <- lapply(overall.stats$query, function(x){fix(max(do.query(con, cache, as.character(x))))})
overall.stats$histogram <- lapply(overall.stats$query, function(x){
data <- do.query(con, cache, as.character(x))
fname <- paste(plot.location, sprintf("hist-%s.pdf",digest(as.character(x))), sep="/")
par(mar=c(0,0,0,0))
plot.window(c(0,1),c(0,1), xaxs='i', yaxs='i')
pdf(file = fname , width = 6, height = 3)
hist(log(data), probability = TRUE, col = "red", border = "white",
breaks = 10, xlab = "", ylab = "", axes = F, main = NULL)
dev.off()
sprintf("\\includegraphics[scale = 0.1, clip = true, trim= 50px 60px 50px 60px]{hist-%s.pdf}", digest(as.character(x)))
})
table <- xtable(overall.stats[,-c(2)], label="tab.overall.stats",
caption="Descriptive statistics for pull requests across all projects in the GHTorrent dataset (1.8M). Historgrams are in log scale.",
align = c("r","r", rep("c", 7)))
print.xtable(table, file = paste(latex.location, "overall-stats.tex", sep = "/"),
floating.environment = "table*",
include.rownames = F, size = c(-1),
sanitize.text.function = function(str)gsub("_","\\_",str,fixed=TRUE))