-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy path.bashrc
545 lines (458 loc) · 17.1 KB
/
.bashrc
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
#!/usr/bin/env bash
PATH="$PATH:$HOME/.local/bin"
export TERM=xterm-256color
# Human editors
hash nano &> /dev/null && \
export EDITOR=nano
hash dte &> /dev/null && \
export EDITOR=dte
# Import user aliases/settings
[ -f ~/.bash-user ] && . ~/.bash-user
# Make and change directory at once
alias mkcd='_(){ mkdir -p "$(echo $@)"; cd "$(echo $@)"; }; _'
# Display the local time and the delta in human-readable format
alias dmesg='dmesg -e'
alias ls='ls --color=auto'
alias la='ls -a'
alias l='ls -alh'
# Common typos
alias sl='ls'
alias cd..='cd ..'
# System info shortcuts
alias df='df -h'
alias free='free -h'
alias du='du -h'
alias psg='ps aux | grep'
alias ncdu='ncdu -x'
# Network shortcuts
alias ports='netstat -tulanp'
alias myip='curl https://ifconfig.co; echo'
alias rcp='rsync -avzhHlP'
alias rcp-del='rsync -avzhHlP --delete'
# Git shortcuts
alias gs='git status'
alias ga='git add'
alias gc='git commit'
alias gp='git push'
alias gl='git pull'
alias gitlog="git log --pretty=format:'%Cred%h%Creset -%C(yellow)%d%Creset %s %Cgreen(%cr %an)%Creset' --abbrev-commit --date=relative"
alias gittree="git log --graph --date=short --branches --pretty=format:'%C(yellow)%h%C(reset) %ad | %C(75)%s%C(reset) %C(yellow)%d%C(reset) [%an]'"
# Default parameter to send to the "less" command
# -R: show ANSI colors correctly; -i: case insensitive search
export LESS="-R -i"
# Colored man
export LESS_TERMCAP_mb=$'\E[1;31m' # begin blink
export LESS_TERMCAP_md=$'\E[1;36m' # begin bold
export LESS_TERMCAP_me=$'\E[0m' # reset bold/blink
export LESS_TERMCAP_so=$'\E[01;45;33m' # begin reverse video
export LESS_TERMCAP_se=$'\E[0m' # reset reverse video
export LESS_TERMCAP_us=$'\E[1;32m' # begin underline
export LESS_TERMCAP_ue=$'\E[0m' # reset underline
# Enhanced tmux launcher: attaches to session or creates new one
# Can switch/create users when argument provided
# Example: t - attach/create tmux session for current user
# t testuser - attach/create tmux session for specified user
hash tmux &> /dev/null && \
t() {
# Enable mouse support for tmux
local TMUX_OPTS=${TMUX_OPTS:-'set-option -g mouse on \; set -g base-index 1 \; setw -g pane-base-index 1 \;'}
local TMUX_CMD="tmux $TMUX_OPTS attach 2>/dev/null || tmux $TMUX_OPTS new"
# If no arguments provided, attach to existing session or create new one
if [ -z "$1" ]; then
eval $TMUX_CMD
return
fi
# Try to switch to specified user
if su - "$1" -c "$TMUX_CMD" ; then
return
fi
# If user doesn't exist, propose to create it
echo ""
read -p "Do you want to create user '$1'? (y/n): " answer
if [[ $answer =~ ^[Yy]$ ]]; then
# Create user without password
if useradd -m "$1"; then
echo "User '$1' created successfully."
# Switch to new user and start tmux
su - "$1" -c "$TMUX_CMD"
else
echo "Failed to create user '$1'"
fi
else
echo "Operation cancelled."
fi
}
hash tmux &> /dev/null && \
_t_completion() {
local cur prev opts
COMPREPLY=()
cur="${COMP_WORDS[COMP_CWORD]}"
prev="${COMP_WORDS[COMP_CWORD-1]}"
# Если уже есть аргумент, не предлагаем автодополнение
if [[ ${#COMP_WORDS[@]} -gt 2 ]]; then
return 0
fi
opts=$(ls /home)
COMPREPLY=( $(compgen -W "${opts}" -- ${cur}) )
return 0
}
hash tmux &> /dev/null && complete -F _t_completion t
# Lightweight sandboxing
hash firejail &> /dev/null && \
ez() {
# Make cute path
local path="$PWD"
if [[ $path == $HOME* ]]; then
path="~${path#$HOME}"
fi
# EZ
env EZ_PATH=$path firejail --private="$(pwd)" "$@" bash
}
# Shortcut to create or open python virtual env
hash python3 &> /dev/null && \
pv() {
# Venv autodetect
local venv=`find . -maxdepth 2 -name "pyvenv.cfg" -type f -exec dirname {} \; | head -n1`
if [ -n "$venv" ]; then
source $venv/bin/activate
else
echo -e "\033[0;32mRunning: python3 -m venv venv --system-site-packages\033[m"
python3 -m venv venv --system-site-packages
source venv/bin/activate
fi
}
# FZF kill processes - list only the ones you can kill
hash fzf &> /dev/null && \
fkill() {
local pid
if [ "$UID" != "0" ]; then
pid=$(ps -f -u $UID | sed 1d | fzf -m | awk '{print $2}')
else
pid=$(ps -ef | sed 1d | fzf -m | awk '{print $2}')
fi
if [ "x$pid" != "x" ]
then
echo $pid | xargs kill -${1:-9}
fi
}
# FZF change dirrectory
hash fzf &> /dev/null && \
cd() {
if [[ "$#" != 0 ]]; then
builtin cd "$@";
return
fi
while true; do
local lsd=$(echo ".." && ls -p | grep '/$' | sed 's;/$;;')
local dir="$(printf '%s\n' "${lsd[@]}" |
fzf --reverse --preview '
__cd_nxt="$(echo {})";
__cd_path="$(echo $(pwd)/${__cd_nxt} | sed "s;//;/;")";
echo $__cd_path;
echo;
ls -p --color=always "${__cd_path}";
')"
[[ ${#dir} != 0 ]] || return 0
builtin cd "$dir" &> /dev/null
done
}
# Universal extract command
# example: extract file
extract () {
if [ ! -f "$1" ] ; then
echo "'$1' does not exist"
return 1
fi
case $1 in
*.tar.bz2) tar xjf "$1" ;;
*.tar.gz) tar xzf "$1" ;;
*.tar.xz) tar xpJf "$1" ;;
*.tar) tar xvf "$1" ;;
*.bz2) bunzip2 "$1" ;;
*.rar) unrar x "$1" ;;
*.gz) gunzip "$1" ;;
*.tbz2) tar xjf "$1" ;;
*.tbz) tar -xjvf "$1";;
*.tgz) tar xzf "$1" ;;
*.apk) unzip "$1" ;;
*.zip) unzip "$1" ;;
*.jar) unzip "$1" ;;
*.Z) uncompress $1 ;;
*.7z) 7z x "$1" ;;
*.xz) unxz "$1" ;;
*) echo "I don't know how to extract '$1'..." ;;
esac
}
# Universal packaging command
# example: pk tar file
pk () {
if [ ! -e "$2" ] ; then
echo "'$1' does not exist"
return 1
fi
case $1 in
tbz) tar cjvf "$"2.tar.bz2 "$2" ;;
tgz) tar czvf "$2".tar.gz "$2" ;;
tar) tar cpvf "$2".tar "$2" ;;
bz2) bzip "$2" ;;
gz) gzip -c -9 -n "$2" > "$2".gz ;;
zip) zip -r "$2".zip "$2" ;;
7z) 7z a "$2".7z "$2" ;;
*) echo "'$1' cannot be packed via pk()" ;;
esac
}
# Universal archive content list command
# example: list ./archive
list () {
if [ ! -f "$1" ] ; then
echo "'$1' does not exist"
return 1
fi
case $1 in
*.tar.bz2) tar tjf "$1" ;;
*.tar.gz) tar tzf "$1" ;;
*.tar.xz) tar tJf "$1" ;;
*.tar) tar tvf "$1" ;;
*.bz2) bunzip2 -tv "$1" ;;
*.rar) unrar l "$1" ;;
*.gz) gunzip -t "$1" ;;
*.tbz2) tar tjf "$1" ;;
*.tbz) tar -tjvf "$1";;
*.tgz) tar tzf "$1" ;;
*.apk) unzip -l "$1" ;;
*.zip) unzip -l "$1" ;;
*.jar) unzip -l "$1" ;;
*.Z) echo "cannot list files in compressed archive" ;;
*.7z) 7z l "$1" ;;
*.xz) unxz -l "$1" ;;
*) echo "I don't know how to list files in '$1'..." ;;
esac
}
# MODIFIED https://github.com/riobard/bash-powerline
__powerline() {
# Colorscheme
RESET='\[\033[m\]'
COLOR_CWD='\[\033[0;34m\]' # blue
COLOR_GIT='\[\033[0;36m\]' # cyan
COLOR_SUCCESS='\[\033[0;32m\]' # green
COLOR_VIRTUAL='\033[0;33m' # yellow
COLOR_FAILURE='\[\033[0;31m\]' # red
SYMBOL_GIT_BRANCH=''
#SYMBOL_GIT_MODIFIED='*'
SYMBOL_GIT_MODIFIED=' '
SYMBOL_GIT_PUSH='↑'
SYMBOL_GIT_PULL='↓'
if [[ -z "$PS_SYMBOL" ]]; then
case "$(uname)" in
Darwin) PS_SYMBOL='';;
Linux) PS_SYMBOL='\$';;
*) PS_SYMBOL='%';;
esac
fi
__git_info() {
[[ $POWERLINE_GIT = 0 ]] && return # disabled
hash git 2>/dev/null || return # git not found
local git_eng="env LANG=C git" # force git output in English to make our work easier
# get current branch name
local ref=$($git_eng symbolic-ref --short HEAD 2>/dev/null)
if [[ -n "$ref" ]]; then
# prepend branch symbol
ref=$SYMBOL_GIT_BRANCH'['$ref']'
else
# get tag name or short unique hash
ref=$($git_eng describe --tags --always 2>/dev/null)
fi
[[ -n "$ref" ]] || return # not a git repo
local marks
# scan first two lines of output from `git status`
while IFS= read -r line; do
if [[ $line =~ ^## ]]; then # header line
[[ $line =~ ahead\ ([0-9]+) ]] && marks+=" $SYMBOL_GIT_PUSH${BASH_REMATCH[1]}"
[[ $line =~ behind\ ([0-9]+) ]] && marks+=" $SYMBOL_GIT_PULL${BASH_REMATCH[1]}"
else # branch is modified if output contains more lines after the header line
marks="$SYMBOL_GIT_MODIFIED$marks"
break
fi
done < <($git_eng status --porcelain --branch 2>/dev/null) # note the space between the two <
# print the git branch segment without a trailing newline
printf " $ref$marks"
}
# Detect sshd in the current process tree
hash pstree &> /dev/null && \
[[ `pstree -s $PPID | grep sshd` ]] && export CUTE_BASH_SHOW_HOSTNAME=1
# Show hostname for tmux sessions
[[ -n "$TMUX" ]] && export CUTE_BASH_SHOW_HOSTNAME=1
ps1() {
# Check the exit code of the previous command and display different
# colors in the prompt accordingly.
if [ $? -eq 0 ]; then
local symbol="$COLOR_SUCCESS $PS_SYMBOL $RESET"
else
local symbol="$COLOR_FAILURE $PS_SYMBOL $RESET"
fi
local cwd="$COLOR_CWD\w$RESET"
# Bash by default expands the content of PS1 unless promptvars is disabled.
# We must use another layer of reference to prevent expanding any user
# provided strings, which would cause security issues.
# POC: https://github.com/njhartwell/pw3nage
# Related fix in git-bash: https://github.com/git/git/blob/9d77b0405ce6b471cb5ce3a904368fc25e55643d/contrib/completion/git-prompt.sh#L324
if shopt -q promptvars; then
__powerline_git_info="$(__git_info)"
local git="$COLOR_GIT\${__powerline_git_info}$RESET"
else
# promptvars is disabled. Avoid creating unnecessary env var.
local git="$COLOR_GIT$(__git_info)$RESET"
fi
if [ -n "$EZ_PATH" ]; then
local ez="${COLOR_VIRTUAL}{$EZ_PATH}$RESET "
fi
if [[ -z "$CUTE_BASH_SHOW_HOSTNAME" ]]; then
PS1="$ez$cwd$git$symbol"
else
PS1="\[\033[01;32m\]\u@\h\[\033[00m\]:$ez $cwd$git$symbol"
fi
# Python virtual env marker
if [ -n "$VIRTUAL_ENV" ]; then
PS1="${COLOR_VIRTUAL}($(basename "$VIRTUAL_ENV"))$RESET $PS1"
fi
}
PROMPT_COMMAND="ps1${PROMPT_COMMAND:+; $PROMPT_COMMAND}"
}
# MODIFIED https://github.com/mrzool/bash-sensible
__sensible () {
# Sensible Bash - An attempt at saner Bash defaults
# Maintainer: mrzool <http://mrzool.cc>
# Repository: https://github.com/mrzool/bash-sensible
# Version: 0.2.2
# Unique Bash version check
if ((BASH_VERSINFO[0] < 4))
then
echo "sensible.bash: Looks like you're running an older version of Bash."
echo "sensible.bash: You need at least bash-4.0 or some options will not work correctly."
echo "sensible.bash: Keep your software up-to-date!"
fi
## GENERAL OPTIONS ##
# Prevent file overwrite on stdout redirection
# Use `>|` to force redirection to an existing file
set -o noclobber
# Update window size after every command
shopt -s checkwinsize
# Automatically trim long paths in the prompt (requires Bash 4.x)
PROMPT_DIRTRIM=2
# Enable history expansion with space
# E.g. typing !!<space> will replace the !! with your last command
bind Space:magic-space
# Turn on recursive globbing (enables ** to recurse all directories)
shopt -s globstar 2> /dev/null
# Case-insensitive globbing (used in pathname expansion)
shopt -s nocaseglob;
## SMARTER TAB-COMPLETION (Readline bindings) ##
# Perform file completion in a case insensitive fashion
bind "set completion-ignore-case on"
# Treat hyphens and underscores as equivalent
bind "set completion-map-case on"
# Display matches for ambiguous patterns at first tab press
bind "set show-all-if-ambiguous on"
# Immediately add a trailing slash when autocompleting symlinks to directories
bind "set mark-symlinked-directories on"
## SANE HISTORY DEFAULTS ##
# Append to the history file, don't overwrite it
shopt -s histappend
# Save multi-line commands as one command
shopt -s cmdhist
# Record each line as it gets issued
PROMPT_COMMAND='history -a'
# Huge history. Doesn't appear to slow things down, so why not?
HISTSIZE=
HISTFILESIZE=
# Change the file location because certain bash sessions truncate .bash_history
export HISTFILE=~/.history_bash
# Avoid duplicate entries
HISTCONTROL="erasedups:ignoreboth"
# Don't record some commands
export HISTIGNORE="&:[ ]*:exit:ls:l:t:bg:fg:history:clear"
# Use standard ISO 8601 timestamp
# %F equivalent to %Y-%m-%d
# %T equivalent to %H:%M:%S (24-hours format)
HISTTIMEFORMAT='[%F %T] '
# Enable incremental history search with up/down arrows (also Readline goodness)
# Learn more about this here: http://codeinthehole.com/writing/the-most-important-command-line-tip-incremental-history-searching-with-inputrc/
bind '"\e[A": history-search-backward'
bind '"\e[B": history-search-forward'
bind '"\e[C": forward-char'
bind '"\e[D": backward-char'
bind '"\e[P": delete-char'
## BETTER DIRECTORY NAVIGATION ##
# Prepend cd to directory names automatically
shopt -s autocd 2> /dev/null
# Correct spelling errors during tab-completion
shopt -s dirspell 2> /dev/null
# Correct spelling errors in arguments supplied to cd
shopt -s cdspell 2> /dev/null
# This defines where cd looks for targets
# Add the directories you want to have fast access to, separated by colon
# Ex: CDPATH=".:~:~/projects" will look for targets in the current working directory, in home and in the ~/projects folder
# CDPATH="."
# This allows you to bookmark your favorite places across the file system
# Define a variable containing a path and you will be able to cd into it regardless of the directory you're in
shopt -s cdable_vars
# Examples:
# export dotfiles="$HOME/dotfiles"
# export projects="$HOME/projects"
# export documents="$HOME/Documents"
# export dropbox="$HOME/Dropbox"
}
check_tmux_session() {
if tmux has-session &> /dev/null; then
echo -e "\033[1m\tThere are tmux session.\033[m Type\033[0;32m\033[1m t\033[m to attach"
fi
}
# Check we're in an interactive shell
if [[ $- = *i* ]]; then
bind TAB:menu-complete
# shift tab cycles backward
__sensible
__powerline
#unset __powerline
[[ -s "/usr/share/doc/fzf/key-bindings.bash" ]] && \
source /usr/share/doc/fzf/key-bindings.bash
# Use bash-completion, if available
[ -f /usr/share/bash-completion/bash_completion ] && \
source /usr/share/bash-completion/bash_completion
# Download and use compatible bash-completion
# if [ -f /etc/bash/bash-completion-2.11 ]; then
# . /etc/bash/bash-completion-2.11
# else
# if [ ! -f $HOME/.bash-completion-2.11 ]; then
# wget "https://raw.githubusercontent.com/scop/bash-completion/2.11/bash_completion" -O ~/.bash-completion-2.11
# fi
# . $HOME/.bash-completion-2.11
# fi
# Download and use LS_COLORS
if [ -f /etc/bash/ls_colors ]; then
eval $( dircolors -b /etc/bash/ls_colors )
else
if [ ! -f $HOME/.ls_colors ]; then
wget "https://raw.githubusercontent.com/trapd00r/LS_COLORS/master/LS_COLORS" -O ~/.ls_colors
fi
eval $( dircolors -b $HOME/.ls_colors )
fi
# Download and use complete_alias
if [ -f /etc/bash/complete_alias ]; then
. /etc/bash/complete_alias
complete -F _complete_alias "${!BASH_ALIASES[@]}"
else
if [ ! -f $HOME/.bash_complete_alias ]; then
wget "https://raw.githubusercontent.com/cykerway/complete-alias/master/complete_alias" -O ~/.bash_complete_alias
fi
. $HOME/.bash_complete_alias
complete -F _complete_alias "${!BASH_ALIASES[@]}"
fi
# Notify about tmux session
if command -v tmux &> /dev/null && [[ ! "$TERM" =~ screen ]] && [[ ! "$TERM" =~ tmux ]] && [ -z "$TMUX" ]; then
# Run in background because tmux may hang
# Run in subshell to prevent display "[0] Done" message
(check_tmux_session &)
fi
fi
unset -f check_tmux_session