-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathbulk-git-ops.sh
executable file
·163 lines (145 loc) · 4.39 KB
/
bulk-git-ops.sh
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
#!/usr/bin/env bash
source ./logging.sh
logdir="$HOME/.bulk_git_ops"
# SUMMARY
#
# If remote has new changes, git fetch origin would download objects
# and should cause mtime of .git to change. We can use this fact
# to filter active repos for update.
#
# Examples assume that repos you contribute to are spread across
# remote hosts, and (hopefully) namespaced sanely. I organise my
# sources as follows.
#
# ~/src/{github,gitlab,bitbucket}/{usernames..}/{reponames..}
#
# USAGE
#
# Source the file, and use the functions as command line utilities as follows.
#
# source ~/src/path/to/this/script/bulk-git-ops.sh
#
# QUERY: Count repos that are stale:
#
# ls_git_projects ~/src/ | take_stale | count_repos_by_remote
#
# QUERY: Count repos that are active (within 12 hours by default):
#
# ls_git_projects ~/src/ | take_active | count_repos_by_remote
#
#
# EXECUTE! Use 'xgit' to apply simple git commands to the given repos, with logs to STDERR/stty
#
# ls_git_projects ~/src/bitbucket | xgit fetch # bitbucket-hosted repos
#
# EXECUTE! Use 'proc_repos' to apply custom functions to, with logs to STDERR/stty
#
# ls_git_projects ~/src/bitbucket | proc_repos git_fetch # all repos
# ls_git_projects ~/src/bitbucket | take_stale | proc_repos git_fetch # only stale repos
# ls_git_projects ~/src/bitbucket | take_active | proc_repos git_fetch # only active repos
#
# EXECUTE! What's the current branch? Logs to STDERR/stty
#
# ls_git_projects ~/src/bitbucket | proc_repos git_branch_current # all repos
#
# EXECUTE! With logs redirected to hidden dir for logging (you must create it by hand first)
#
# mkdir -p "${logdir}"
# ls_git_projects ~/src/bitbucket | proc_repos git_branch_current 2>> "${logdir}/bulkops.log"
# tail "${logdir}/bulkops.log"
#
ls_git_projects() {
# Spit out list of root dirs of .git-controlled projects
local basedir="${1:-$PWD}"
if [[ -d ${basedir} ]]
then find ${basedir} -type d -name '.git' | xargs dirname | xargs realpath
else >&2 printf "%s\n" "ERROR ${basedir} is not a valid base directory."
return 1
fi
}
identity() {
printf "%s\n" "$@"
}
proc_repos() {
# Apply any given operation on the given repos. Use in a pipeline.
local func=${1:-identity}
local repo_dir
while read repo_dir
do $func "${repo_dir}"
log_info "Applied ${func} to ${repo_dir}"
done
}
xgit() {
# Apply a git operation on the given repos. Use in a pipeline.
local git_cmd="${@}"
local repo_dir
while read repo_dir
do git --git-dir="${repo_dir}/.git" $git_cmd
log_info "Applied git ${git_cmd} to ${repo_dir}."
done
}
# git_fetch_statuses() {
# local basedir=${1:-"$PWD"}
# find ${basedir} \
# -type d -name '.git' \
# -prune -print \
# -execdir git fetch -q \;
# }
__with_git_dir() {
local repo_dir="${1}"
shift
git --git-dir="${repo_dir}/.git" "$@"
}
git_fetch() {
local repo_dir="${1}"
__with_git_dir "${repo_dir}" fetch -q
}
git_status() {
__with_git_dir "${1}" status
}
git_branch_current() {
local repo_dir="${1}"
__with_git_dir "${repo_dir}" rev-parse --abbrev-ref=strict HEAD
}
__is_repo_active() {
# A directory's mtime increments only if files/sub-dirs are
# created and/or deleted.
#
# We can use this plus the fact that the .git directory's
# contents would change IFF new objects are created in it,
# or old ones garbage collected.
#
# Fetch operations are idempotent, and no-ops if local has not
# diverged from remote.
#
# Events we care about, to classify as "active":
# - Fetch from remote updates .git
# - Active development updates .git (stage/commit)
#
# Events that don't matter usually (infrequent), but are worth
# interpreting as "active":
#
# - Objects collected / compacted automatically by git locally
#
local repo_dir="${1}"
local stale_hrs="${2:-12}"
local hrs_ago=$(( ($(date +%s)
- $(stat -c %Y "${repo_dir}/.git"))
/ 3600 ))
[[ $hrs_ago -le $stale_hrs ]]
}
take_active() {
local repo_dir
while read repo_dir
do __is_repo_active "${repo_dir}" && printf "%s\n" "${repo_dir}"
done
}
take_stale() {
local repo_dir
while read repo_dir
do __is_repo_active "${repo_dir}" || printf "%s\n" "${repo_dir}"
done
}
count_repos_by_remote() {
grep -E -o "github|gitlab|bitbucket" | sort | uniq -c
}