-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcommon.sh
560 lines (483 loc) · 16.3 KB
/
common.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
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
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
#!/usr/bin/env bash
### Consts for colors to use inside 'sed'
RED_ES="\x1b[31m"
GREEN_ES="\x1b[32m"
YELLOW_ES="\x1b[33m"
BLUE_ES="\x1b[34m"
PURPLE_ES="\x1b[35m"
CYAN_ES="\x1b[36m"
GRAY_ES="\x1b[37m"
ENDCOLOR_ES="\x1b[0m"
### Function sets git config value
# $1: name
# $2: value
# $3: global flag
# Returns: value
function set_config_value {
if [ -z $3 ]; then
git config --local $1 $2
else
git config --global $1 $2
fi
echo "$2"
}
### Function should be used in default case in script mode selection
# $1: script name
# $2: entered mode
function wrong_mode {
if [ -n "$2" ]; then
echo -e "Unknown mode ${YELLOW}$2${ENDCOLOR} for ${YELLOW}gitb $1${ENDCOLOR}"
echo -e "Use ${GREEN}gitb $1 help${ENDCOLOR} to get usage info"
exit
fi
}
### Function echoes (true return) url to current user's repo (remote)
# Return: url to repo
function get_repo {
repo=$(git config --get remote.${origin_name}.url)
repo="${repo/"com:"/"com/"}"
repo="${repo/"io:"/"io/"}"
repo="${repo/"org:"/"org/"}"
repo="${repo/"net:"/"net/"}"
repo="${repo/"dev:"/"dev/"}"
repo="${repo/"ru:"/"ru/"}"
repo="${repo/"git@"/"https://"}"
repo="${repo/".git"/""}"
echo "$repo"
}
### Function echoes (true return) name of current repo
function get_repo_name {
repo=$(get_repo)
echo "${repo##*/}"
}
### Function prints current config
function print_configuration {
echo -e "${YELLOW}Current configuration:${ENDCOLOR}"
echo -e "\tuser.name:\t${YELLOW}$(get_config_value user.name)${ENDCOLOR}"
echo -e "\tuser.email:\t${YELLOW}$(get_config_value user.email)${ENDCOLOR}"
echo -e "\tdefault:\t${YELLOW}$main_branch${ENDCOLOR}"
echo -e "\tseparator:\t${YELLOW}$sep${ENDCOLOR}"
echo -e "\teditor:\t\t${YELLOW}$editor${ENDCOLOR}"
if [ "$ticket_name" != "" ]; then
echo -e "\tticket:\t\t${YELLOW}$ticket_name${ENDCOLOR}"
fi
if [ "$scopes" != "" ]; then
echo -e "\tscopes:\t\t${YELLOW}$scopes${ENDCOLOR}"
fi
}
### Function to escape substring in string
# $1: string
# $2: substring to escape
# Returns: provided string with escaped substring
function escape {
string="$1"
sub="$2"
escaped="\\$sub"
echo "${string//${sub}/${escaped}}"
}
### Function checks code against 0 and show error
# $1: return code
# $2: command output (error message)
# $3: command name
# Using of global:
# * git_add
function check_code {
if [ $1 != 0 ]; then
echo
echo
echo -e "${RED}Error during $3!${ENDCOLOR}"
echo -e "$2"
if [ -n "$git_add" ]; then
git restore --staged $git_add
fi
exit $1
fi
}
### Function asks user to enter yes or no, it will exit if user answers 'no'
# $1: what to write in console on success
# $2: flag no echo
function yes_no_choice {
while [ true ]; do
read -n 1 -s choice
if [ "$choice" == "y" ]; then
if [ -n "$1" ]; then
echo -e "${YELLOW}$1${ENDCOLOR}"
if [ -z $2 ]; then
echo
fi
fi
return
fi
if [ "$choice" == "n" ]; then
exit
fi
done
}
### Function waits a number from user and returns result of choice from a provided list
# $1: list of values
# Returns:
# * choice_result
# * pressed_alt
# Using of global:
# * git_add
function choose {
values=("$@")
number_of_values=${#values[@]}
while [ true ]; do
if [ $number_of_values -gt 9 ]; then
read -p "$read_prefix" -e -n 2 choice
else
read -p "$read_prefix" -n 1 -s choice
fi
if [ "$choice" == "0" ] || [ "$choice" == "00" ]; then
if [ -n "$git_add" ]; then
git restore --staged $git_add
fi
if [ $number_of_values -le 9 ]; then
printf $choice
fi
exit
fi
re='^[0-9=]+$'
if ! [[ $choice =~ $re ]]; then
if [ -n "$git_add" ]; then
git restore --staged $git_add
fi
exit
fi
if [ "$choice" == "=" ] || [ "$choice" == "==" ]; then
pressed_alt="true"
break
fi
index=$(($choice-1))
choice_result=${values[index]}
if [ -n "$choice_result" ]; then
if [ $number_of_values -le 9 ]; then
printf $choice
fi
break
else
if [ $number_of_values -gt 9 ]; then
if [ -n "$git_add" ]; then
git restore --staged $git_add
fi
exit
fi
fi
done
}
### Function prints fiels from git status in a pretty way
function git_status {
status_output=$(git status --short)
status_output=$(echo "$status_output" | sed "s/^ D/${RED_ES}\tDeleted: ${ENDCOLOR_ES}/")
status_output=$(echo "$status_output" | sed "s/^D /${GREEN_ES}Staged\t${RED_ES}Deleted: ${ENDCOLOR_ES}/")
status_output=$(echo "$status_output" | sed "s/^ M/${YELLOW_ES}\tModified:${ENDCOLOR_ES}/")
status_output=$(echo "$status_output" | sed "s/^MM/${GRAY_ES}Old\t${YELLOW_ES}Modified:${ENDCOLOR_ES}/")
status_output=$(echo "$status_output" | sed "s/^AM/${GRAY_ES}Old\t${YELLOW_ES}Modified:${ENDCOLOR_ES}/")
status_output=$(echo "$status_output" | sed "s/^M /${GREEN_ES}Staged\t${YELLOW_ES}Modified:${ENDCOLOR_ES}/")
status_output=$(echo "$status_output" | sed "s/^A/${GREEN_ES}Staged\tAdded: ${ENDCOLOR_ES}/")
status_output=$(echo "$status_output" | sed "s/^??/${GREEN_ES}\tAdded: ${ENDCOLOR_ES}/")
echo -e "$status_output"
}
### Function prints the list of commits
# $1: number of last commits to show
# $2: what to add before commit line
# * <empty> - nothing
# * tab
# * number
# $3: from which place (commit, branch) show commits (empty for default)
# Returns:
# commits_info
# commits_hash
function commit_list {
ref=$3
if [[ "$(git --no-pager log -n 1 2>&1)" == *"does not have any commits yet"* ]]; then
if [[ "$3" == *"HEAD"* ]]; then
ref="$(echo $3 | sed 's/HEAD..//')"
else
return
fi
fi
IFS=$'\n'
read -rd '' -a commits_info <<<"$(git --no-pager log -n $1 --pretty="${YELLOW_ES}%h${ENDCOLOR_ES} | %s | ${BLUE_ES}%an${ENDCOLOR_ES} | ${GREEN_ES}%cr${ENDCOLOR_ES}" $ref | column -ts'|')"
read -rd '' -a commits_hash <<<"$(git --no-pager log -n $1 --pretty="%h"$ref)"
for index in "${!commits_info[@]}"
do
line=${commits_info[index]}
if [ $2 == "number" ]; then
line="$(($index+1)). ${line}"
elif [ $2 == "tab" ]; then
line="\t${line}"
fi
echo -e "$line"
done
}
### Function prints the list of refs from reflog
# $1: number of last refs to show
# Returns:
# refs_info
# refs_hash
function ref_list {
IFS=$'\n'
read -rd '' -a refs_info <<<"$(git --no-pager reflog -n $1 --pretty="${YELLOW_ES}%h${ENDCOLOR_ES} | ${BLUE_ES}%gd${ENDCOLOR_ES} | %gs | ${GREEN_ES}%cr${ENDCOLOR_ES}" | column -ts'|')"
read -rd '' -a refs_hash <<<"$(git --no-pager reflog -n $1 --pretty="%gd")"
# Remove HEAD@{0}
refs_info=("${refs_info[@]:1}")
refs_hash=("${refs_hash[@]:1}")
for index in "${!refs_info[@]}"
do
line="$(($index+1)). ${refs_info[index]}"
echo -e "$line"
done
}
### Function prints the list of commits and user should choose one
# $1: number of last commits to show
# Returns:
# commit_hash - hash of selected commit
# Using of global:
# * git_add
function choose_commit {
commit_list $1 "number"
if [ $1 -gt 9 ]; then
echo "00. Exit"
else
echo "0. Exit"
fi
echo "Enter = to show more"
echo
read_prefix="Enter commit number: "
choose "${commits_hash[@]}"
commit_hash=$choice_result
if [ -n "$pressed_alt" ]; then
commit_list 50 "number"
echo "00. Exit"
echo
choose "${commits_hash[@]}"
commit_hash=$choice_result
fi
echo
}
### Function prints provided stat in a nice format with colors
# $1: stats after pull or commit like 'README.md | 1 +\n1 file changed, 1 insertion(+)'
function print_changes_stat {
IFS=$'\n' read -rd '' -a stats <<< "$1"
result_stat=""
bottom_line=""
number_of_stats=${#stats[@]}
for index in "${!stats[@]}"
do
s=$(echo ${stats[index]} | sed -e 's/^[[:space:]]*//')
s=$(sed "s/+/${GREEN_ES}+${ENDCOLOR_ES}/g" <<< ${s})
s=$(sed "s/-/${RED_ES}-${ENDCOLOR_ES}/g" <<< ${s})
if [ $(($index+1)) == $number_of_stats ]; then
#s=$(sed '1 s/,/|/' <<< ${s})
bottom_line="${s}"
break
fi
result_stat="${result_stat}\n${s}"
done
echo -e "$(echo -e "${result_stat}" | column -ts'|')"
echo -e "$bottom_line"
}
### Function sets to variables push_list and history_from actual push log information
# $1: current branch
# $2: main branch
# $3: origin name
# Returns:
# push_list - unpushed commits
# history_from - branch or commit from which history was calculated
function get_push_list {
push_list_check=$(git --no-pager log $3/$1..HEAD 2>&1)
if [[ $push_list_check != *"unknown revision or path not in the working tree"* ]]; then
push_list=$(commit_list 999 "tab" $3/$1..HEAD)
history_from="$3/$1"
return
fi
# Case with new repo without any branch
if [[ $push_list_check == *"unknown revision or path not in the working tree"* ]]; then
if [[ $1 == $2 ]]; then
push_list=$(commit_list 999 "tab")
history_from="$3/$1"
return
fi
fi
base_commit=$(diff -u <(git rev-list --first-parent $1) <(git rev-list --first-parent $2) | sed -ne 's/^ //p' | head -1)
if [ -n "$base_commit" ]; then
push_list=$(commit_list 999 "tab" $base_commit..HEAD)
history_from="${base_commit::7}"
else
push_list=$(commit_list 999 "tab" $3/$2..HEAD)
history_from="$3/$2"
fi
}
### Function prints list of branches
# $1: possible values:
# * no value prints all local branches
# * 'remote' - all remote
# * 'delete' - all local without main and current
# * 'merge' - all local without current
# Using of global:
# * current_branch
# * main_branch
# Returns:
# * number_of_branches
# * branches_first_main
# * to_exit
function list_branches {
args="--sort=-committerdate"
if [[ "$1" == "remote" ]]; then
args="--sort=-committerdate -r"
fi
branches_str=$(git --no-pager branch $args --format="%(refname:short)")
branches_info_str=$(git --no-pager branch $args --format="${BLUE_ES}%(refname:short)${ENDCOLOR_ES} | %(contents:subject) | ${YELLOW_ES}%(objectname:short)${ENDCOLOR_ES} | ${GREEN_ES}%(committerdate:relative)${ENDCOLOR_ES}" | column -ts'|' )
IFS=$'\n'
read -rd '' -a branches <<< "$branches_str"
read -rd '' -a branches_info <<< "$branches_info_str"
number_of_branches=${#branches[@]}
if [[ "$1" == "remote" ]]; then
# There is origin/HEAD
((number_of_branches=number_of_branches-1))
fi
if [[ "$number_of_branches" == 0 ]]; then
echo
echo -e "${YELLOW}There is no branches${ENDCOLOR}"
to_exit="true"
return
fi
branch_to_check="${branches[0]}"
if [[ "$1" == "remote" ]]; then
# Remove 'origin/'
branch_to_check="${branches[1]}"
branch_to_check="$(sed "s/${origin_name}\///g" <<< ${branch_to_check})"
fi
if [[ "$number_of_branches" == 1 ]] && [[ "${branch_to_check}" == "${current_branch}" ]]; then
echo
echo -e "There is only one branch: ${YELLOW}${current_branch}${ENDCOLOR}"
to_exit="true"
return
fi
if [[ "$1" == "delete" ]] && [[ "$number_of_branches" == 2 ]] && [[ "${current_branch}" != "${main_branch}" ]]; then
echo
echo -e "${YELLOW}There are no branches to delete${ENDCOLOR}"
to_exit="true"
return
fi
### Main should be the first
branches_first_main=(${main_branch})
branches_info_first_main=("dummy")
if [[ "$1" == "delete" ]]; then
branches_first_main=()
branches_info_first_main=()
fi
if [[ "$1" == "merge" ]] && [[ "$current_branch" == "$main_branch" ]]; then
branches_first_main=()
branches_info_first_main=()
fi
for index in "${!branches[@]}"
do
branch_to_check="${branches[index]}"
if [[ "$1" == "delete" ]]; then
if [[ "$branch_to_check" == "${current_branch}"* ]] || [[ "$branch_to_check" == "${main_branch}"* ]]; then
continue
fi
fi
if [[ "$1" == "merge" ]]; then
if [[ "$branch_to_check" == "${current_branch}"* ]]; then
continue
fi
fi
if [[ "$1" == "remote" ]]; then
branch_to_check="$(sed "s/${origin_name}\///g" <<< ${branch_to_check})"
fi
if [[ "$branch_to_check" == "${main_branch}"* ]]; then
branches_info_first_main[0]="${branches_info[index]}"
elif [[ "$branch_to_check" != "HEAD->"* ]] && [[ "$branch_to_check" != "$origin_name" ]]; then
branches_first_main+=(${branches[index]})
branches_info_first_main+=("${branches_info[index]}")
fi
done
for index in "${!branches_info_first_main[@]}"
do
branch=$(escape "${branches_first_main[index]}" "/")
if [[ "$1" == "remote" ]] && [[ "$branch" != "origin"* ]]; then
branch="$origin_name\/$branch"
fi
branch_line="${branches_info_first_main[index]}"
if [ "${branches_first_main[index]}" == "$current_branch" ]; then
echo -e "$(($index+1)). * $branch_line"
else
echo -e "$(($index+1)). $branch_line"
fi
done
}
### This function prints the list of branches and user should choose one
# $1: possible values:
# * no value prints all local branches
# * 'remote' - choose from all remote
# * 'delete' - choose from all local without main and current
# * 'merge' - all local without current
# Using of global:
# * origin_name
# * current_branch
# * main_branch
# Returns:
# * branch_name
function choose_branch {
list_branches $1
if [ -n "$to_exit" ]; then
exit
fi
echo
printf "Enter branch number: "
choose "${branches_first_main[@]}"
branch_name=$choice_result
if [[ "$1" == "remote" ]]; then
branch_name=$(sed "s/${origin_name}\///g" <<< ${branch_name})
fi
echo
}
### Function handles switch result
# $1: name of the branch to switch
# $2: pass it if you want to disable push log and moved changes
function switch {
switch_output=$(git switch $1 2>&1)
switch_code=$?
## Switch is OK
if [ "$switch_code" == 0 ]; then
if [ "$current_branch" == "$1" ]; then
echo -e "${GREEN}Already on '$1'${ENDCOLOR}"
else
echo -e "${GREEN}Switched to branch '$1'${ENDCOLOR}"
changes=$(git_status)
if [ -n "$changes" ] && [ -z $2 ]; then
echo
echo -e "${YELLOW}Moved changes:${ENDCOLOR}"
echo -e "$changes"
fi
fi
if [ -z $2 ]; then
get_push_list $1 ${main_branch} ${origin_name}
if [ -n "$push_list" ]; then
echo
count=$(echo -e "$push_list" | wc -l | sed 's/^ *//;s/ *$//')
echo -e "Your branch ${YELLOW}$1${ENDCOLOR} is ahead ${YELLOW}${history_from}${ENDCOLOR} by ${BOLD}$count${ENDCOLOR} commits"
echo -e "$push_list"
fi
fi
return
fi
## There are uncommited files with conflicts
if [[ $switch_output == *"Your local changes to the following files would be overwritten"* ]]; then
conflicts="$(echo "$switch_output" | tail -r | tail -n +3 | tail -r | tail -n +2)"
echo -e "${RED}Changes would be overwritten by switch to '$1':${ENDCOLOR}"
echo -e "${conflicts//[[:blank:]]/}"
echo
echo -e "${YELLOW}Commit these files and try to switch for one more time${ENDCOLOR}"
exit
fi
if [ $switch_code -ne 0 ]; then
echo -e "${RED}Cannot switch to '$main_branch'! Error message:${ENDCOLOR}"
echo -e "$switch_output"
exit $switch_code
fi
}