~/.bashrc 파일은 대화형(interactive) 모드 쉘의 행동을 결정합니다. 이 파일을 잘 공부하면 Bash 를 더욱 잘 이해할 수 있을 것입니다.
Emmanuel Rouat 가 아주 정교한 리눅스용 .bashrc 스크립트를 제공해 주었습니다. 이 파일을 아주 자세히 공부해서 여기 나오는 코드 조각들이나 함수들을 여러분의 .bashrc 나 스크립트에서 마음껏 사용하기 바랍니다.
예 F-1. 샘플 .bashrc 파일
#=============================================================== # # bash-2.05 이후 버전을 위한 개인적 $HOME/.bashrc 파일 # # 아 파일은 대화모드 쉘을 위한 것입니다. # 별칭(alias)이나 함수, 프롬프트같은 # 대화모드용 기능들을 여기에 두면 됩니다. # # 원래는 솔라리스를 위해 디자인 되었습니다. # --> 리눅스용으로 수정 # 이 파일은 너무 많은 것을 포함하고 있지만 # 단지 예제라는 것을 기억하세요. # 여러분 필요에 따라 수정해서 쓰기 바랍니다. # #=============================================================== # --> 본 문서 저자에 의한 주석. #----------------------------------- # 필요하다면 전역 정의를 source #----------------------------------- if [ -f /etc/bashrc ]; then . /etc/bashrc # --> 있다면 /etc/bashrc 를 읽음. fi #------------------------------------------------------------- # 아직 세트되지 않았다면 $DISPLAY 를 자동으로 세팅 # 이 부분은 리눅스와 솔라리스용입니다 - 필요한대로 고쳐 쓰세요... #------------------------------------------------------------- if [ -z ${DISPLAY:=""} ]; then DISPLAY=$(who am i) DISPLAY=${DISPLAY%%\!*} if [ -n "$DISPLAY" ]; then export DISPLAY=$DISPLAY:0.0 else export DISPLAY=":0.0" # 실패할 경우를 대비(fallback) fi fi #--------------- # 몇 가지 세팅 #--------------- set -o notify set -o noclobber set -o ignoreeof set -o nounset #set -o xtrace # 디버깅용 shopt -s cdspell shopt -s cdable_vars shopt -s checkhash shopt -s checkwinsize shopt -s mailwarn shopt -s sourcepath shopt -s no_empty_cmd_completion shopt -s histappend histreedit shopt -s extglob # programmable completion에 유용 #----------------------- # 인사말, motd 등등... #----------------------- # 먼저 색깔을 몇 개 정의: red='\e[0;31m' RED='\e[1;31m' blue='\e[0;34m' BLUE='\e[1;34m' cyan='\e[0;36m' CYAN='\e[1;36m' NC='\e[0m' # No Color # --> 좋군요. 도스에서 "ansi.sys"를 쓰는 것과 똑같은 효과가 있네요. # 검정색 백그라운드에서 가장 좋게 보입니다..... echo -e "${CYAN}This is BASH ${RED}${BASH_VERSION%.*}${CYAN} - DISPLAY on ${RED}$DISPLAY${NC}\n" date if [ -x /usr/games/fortune ]; then /usr/games/fortune -s # 하루를 즐겁게.... :-) fi function _exit() # 쉘에서 종료시 실행할 함수 { echo -e "${RED}나중에 또 봐요${NC}" } trap _exit 0 #--------------- # 쉘 프롬프트 #--------------- function fastprompt() { unset PROMPT_COMMAND case $TERM in *term | rxvt ) PS1="[\h] \W > \[\033]0;[\u@\h] \w\007\]" ;; *) PS1="[\h] \W > " ;; esac } function powerprompt() { _powerprompt() { LOAD=$(uptime|sed -e "s/.*: \([^,]*\).*/\1/" -e "s/ //g") TIME=$(date +%H:%M) } PROMPT_COMMAND=_powerprompt case $TERM in *term | rxvt ) PS1="${cyan}[\$TIME \$LOAD]$NC\n[\h \#] \W > \[\033]0;[\u@\h] \w\007\]" ;; linux ) PS1="${cyan}[\$TIME - \$LOAD]$NC\n[\h \#] \w > " ;; * ) PS1="[\$TIME - \$LOAD]\n[\h \#] \w > " ;; esac } powerprompt # 좀 느릴지도 모를 기본 프롬프트입니다. # 너무 느리면 fastprompt 를 쓰세요.... #=============================================================== # # 별칭(alias)과 함수들 # # 논쟁의 여지가 있지만 몇몇 함수들은 조금 덩치가 큰데(즉, 'lowercase') # 제 워크스테이션은 램이 512메가거든요... # 이 파일 크기를 줄이고 싶다면 이런 함수들은 스크립트로 빼도 됩니다. # # 많은 함수들은 bash-2.04 예제에서 거의 그대로 갖다 썼습니다. # #=============================================================== #------------------------- # 개인적인 별칭들(Aliases) #------------------------- alias rm='rm -i' alias cp='cp -i' alias mv='mv -i' # -> 파일에 실수로 타격을 입히지 않게. alias h='history' alias j='jobs -l' alias r='rlogin' alias which='type -all' alias ..='cd ..' alias path='echo -e ${PATH//:/\\n}' alias print='/usr/bin/lp -o nobanner -d $LPDEST' # LPDEST 가 정의되어 있다고 가정 alias pjet='enscript -h -G -fCourier9 -d $LPDEST' # enscript 로 예쁜 출력하기(Pretty-print) alias background='xv -root -quit -max -rmode 5' # 백그라운드 배경 그림 alias vi='vim' alias du='du -h' alias df='df -kh' # 'ls' 그룹(여러분이 GNU ls 를 쓴다고 가정) alias ls='ls -hF --color' # 파일타입 인식을 위해 색깔을 추가 alias lx='ls -lXB' # 확장자별로 정렬 alias lk='ls -lSr' # 크기별로 정렬 alias la='ls -Al' # 숨겨진 파일 보기 alias lr='ls -lR' # 재귀적 ls alias lt='ls -ltr' # 날짜별로 정렬 alias lm='ls -al |more' # 'more'로 파이프 걸기 alias tree='tree -Cs' # 'ls'의 멋진 대용품 # 맞춤 'less' alias more='less' export PAGER=less export LESSCHARSET='latin1' export LESSOPEN='|/usr/bin/lesspipe.sh %s 2>&-' # lesspipe.sh 이 있다면 이걸 쓰세요 export LESS='-i -N -w -z-4 -g -e -M -X -F -R -P%t?f%f \ :stdin .?pb%pb\%:?lbLine %lb:?bbByte %bb:-...' # 스펠링 오타용 - 아주 개인적임 :-) alias xs='cd' alias vf='cd' alias moer='more' alias moew='more' alias kk='ll' #---------------- # 재밌는 거 몇 개 #---------------- function xtitle () { case $TERM in *term | rxvt) echo -n -e "\033]0;$*\007" ;; *) ;; esac } # 별칭들(aliases)... alias top='xtitle Processes on $HOST && top' alias make='xtitle Making $(basename $PWD) ; make' alias ncftp="xtitle ncFTP ; ncftp" # .. 과 함수들 function man () { xtitle The $(basename $1|tr -d .[:digit:]) manual man -a "$*" } function ll(){ ls -l "$@"| egrep "^d" ; ls -lXB "$@" 2>&-| egrep -v "^d|total "; } function xemacs() { { command xemacs -private $* 2>&- & } && disown ;} function te() # xemacs/gnuserv 래퍼 { if [ "$(gnuclient -batch -eval t 2>&-)" == "t" ]; then gnuclient -q "$@"; else ( xemacs "$@" & ); fi } #----------------------------------- # 파일 & 문자열 관련 함수들: #----------------------------------- function ff() { find . -name '*'$1'*' ; } # 파일 찾기 function fe() { find . -name '*'$1'*' -exec $2 {} \; ; } # 파일을 찾아서 $2 의 인자로 실행 function fstr() # 여러 파일중에서 문자열 찾기 { if [ "$#" -gt 2 ]; then echo "Usage: fstr \"pattern\" [files] " return; fi SMSO=$(tput smso) RMSO=$(tput rmso) find . -type f -name "${2:-*}" -print | xargs grep -sin "$1" | \ sed "s/$1/$SMSO$1$RMSO/gI" } function cuttail() # 파일에서 끝의 n 줄을 잘라냄. 기본값은 10 { nlines=${2:-10} sed -n -e :a -e "1,${nlines}!{P;N;D;};N;ba" $1 } function lowercase() # 파일이름을 소문자로 변경 { for file ; do filename=${file##*/} case "$filename" in */*) dirname==${file%/*} ;; *) dirname=.;; esac nf=$(echo $filename | tr A-Z a-z) newname="${dirname}/${nf}" if [ "$nf" != "$filename" ]; then mv "$file" "$newname" echo "lowercase: $file --> $newname" else echo "lowercase: $file not changed." fi done } function swap() # 파일이름 두개를 서로 바꿈 { local TMPFILE=tmp.$$ mv $1 $TMPFILE mv $2 $1 mv $TMPFILE $2 } #----------------------------------- # 프로세스/시스템 관련 함수들: #----------------------------------- function my_ps() { ps $@ -u $USER -o pid,%cpu,%mem,bsdtime,command ; } function pp() { my_ps f | awk '!/awk/ && $0~var' var=${1:-".*"} ; } # 이 함수는 리눅스의 'killall' 스크립트와 거의 비슷하지만 # 솔라리스에는, 제가 아는 한, 이와 비슷한 것이 없습니다. function killps() # 프로세스 이름으로 kill { local pid pname sig="-TERM" # 기본 시그널 if [ "$#" -lt 1 ] || [ "$#" -gt 2 ]; then echo "Usage: killps [-SIGNAL] pattern" return; fi if [ $# = 2 ]; then sig=$1 ; fi for pid in $(my_ps| awk '!/awk/ && $0~pat { print $1 }' pat=${!#} ) ; do pname=$(my_ps | awk '$1~var { print $5 }' var=$pid ) if ask "Kill process $pid <$pname> with signal $sig?" then kill $sig $pid fi done } function my_ip() # IP 주소 알아내기 { MY_IP=$(/sbin/ifconfig ppp0 | awk '/inet/ { print $2 } ' | sed -e s/addr://) MY_ISP=$(/sbin/ifconfig ppp0 | awk '/P-t-P/ { print $3 } ' | sed -e s/P-t-P://) } function ii() # 현재 호스트 관련 정보들 알아내기 { echo -e "\nYou are logged on ${RED}$HOST" echo -e "\nAdditionnal information:$NC " ; uname -a echo -e "\n${RED}Users logged on:$NC " ; w -h echo -e "\n${RED}Current date :$NC " ; date echo -e "\n${RED}Machine stats :$NC " ; uptime echo -e "\n${RED}Memory stats :$NC " ; free my_ip 2>&- ; echo -e "\n${RED}Local IP Address :$NC" ; echo ${MY_IP:-"Not connected"} echo -e "\n${RED}ISP Address :$NC" ; echo ${MY_ISP:-"Not connected"} echo } # 기타 유틸리티: function repeat() # 명령어를 n 번 반복 { local i max max=$1; shift; for ((i=1; i <= max ; i++)); do # --> C 형태의 문법 eval "$@"; done } function ask() { echo -n "$@" '[y/n] ' ; read ans case "$ans" in y*|Y*) return 0 ;; *) return 1 ;; esac } #========================================================================= # # PROGRAMMABLE COMPLETION - 오직 BASH 2.04 이후에서만 동작 # (거의 대부분은 bash 2.05 문서에서 가져왔습니다) # 몇 가지 기능들을 쓰려면 bash-2.05 가 필요할 겁니다. # #========================================================================= if [ "${BASH_VERSION%.*}" \< "2.05" ]; then echo "programmable completion 을 쓰려면 bash 2.05 이상으로 업그레이드가 필요합니다." return fi shopt -s extglob # 꼭 필요함 set +o nounset # 이렇게 안 하면 programmable completion 몇 가지는 실패함 # 옮긴이: 이 이후는 잘 모르겠네요. :( complete -A hostname rsh rcp telnet rlogin r ftp ping disk complete -A command nohup exec eval trace gdb complete -A command command type which complete -A export printenv complete -A variable export local readonly unset complete -A enabled builtin complete -A alias alias unalias complete -A function function complete -A user su mail finger complete -A helptopic help # currently same as builtins complete -A shopt shopt complete -A stopped -P '%' bg complete -A job -P '%' fg jobs disown complete -A directory mkdir rmdir complete -A directory -o default cd complete -f -d -X '*.gz' gzip complete -f -d -X '*.bz2' bzip2 complete -f -o default -X '!*.gz' gunzip complete -f -o default -X '!*.bz2' bunzip2 complete -f -o default -X '!*.pl' perl perl5 complete -f -o default -X '!*.ps' gs ghostview ps2pdf ps2ascii complete -f -o default -X '!*.dvi' dvips dvipdf xdvi dviselect dvitype complete -f -o default -X '!*.pdf' acroread pdf2ps complete -f -o default -X '!*.+(pdf|ps)' gv complete -f -o default -X '!*.texi*' makeinfo texi2dvi texi2html texi2pdf complete -f -o default -X '!*.tex' tex latex slitex complete -f -o default -X '!*.lyx' lyx complete -f -o default -X '!*.+(jpg|gif|xpm|png|bmp)' xv gimp complete -f -o default -X '!*.mp3' mpg123 complete -f -o default -X '!*.ogg' ogg123 # This is a 'universal' completion function - it works when commands have # a so-called 'long options' mode , ie: 'ls --all' instead of 'ls -a' _universal_func () { case "$2" in -*) ;; *) return ;; esac case "$1" in \~*) eval cmd=$1 ;; *) cmd="$1" ;; esac COMPREPLY=( $("$cmd" --help | sed -e '/--/!d' -e 's/.*--\([^ ]*\).*/--\1/'| \ grep ^"$2" |sort -u) ) } complete -o default -F _universal_func ldd wget bash id info _make_targets () { local mdef makef gcmd cur prev i COMPREPLY=() cur=${COMP_WORDS[COMP_CWORD]} prev=${COMP_WORDS[COMP_CWORD-1]} # if prev argument is -f, return possible filename completions. # we could be a little smarter here and return matches against # `makefile Makefile *.mk', whatever exists case "$prev" in -*f) COMPREPLY=( $(compgen -f $cur ) ); return 0;; esac # if we want an option, return the possible posix options case "$cur" in -) COMPREPLY=(-e -f -i -k -n -p -q -r -S -s -t); return 0;; esac # make reads `makefile' before `Makefile' if [ -f makefile ]; then mdef=makefile elif [ -f Makefile ]; then mdef=Makefile else mdef=*.mk # local convention fi # before we scan for targets, see if a makefile name was specified # with -f for (( i=0; i < ${#COMP_WORDS[@]}; i++ )); do if [[ ${COMP_WORDS[i]} == -*f ]]; then eval makef=${COMP_WORDS[i+1]} # eval for tilde expansion break fi done [ -z "$makef" ] && makef=$mdef # if we have a partial word to complete, restrict completions to # matches of that word if [ -n "$2" ]; then gcmd='grep "^$2"' ; else gcmd=cat ; fi # if we don't want to use *.mk, we can take out the cat and use # test -f $makef and input redirection COMPREPLY=( $(cat $makef 2>/dev/null | awk 'BEGIN {FS=":"} /^[^.# ][^=]*:/ {print $1}' | tr -s ' ' '\012' | sort -u | eval $gcmd ) ) } complete -F _make_targets -X '+($*|*.[cho])' make gmake pmake _configure_func () { case "$2" in -*) ;; *) return ;; esac case "$1" in \~*) eval cmd=$1 ;; *) cmd="$1" ;; esac COMPREPLY=( $("$cmd" --help | awk '{if ($1 ~ /--.*/) print $1}' | grep ^"$2" | sort -u) ) } complete -F _configure_func configure # cvs(1) completion _cvs () { local cur prev COMPREPLY=() cur=${COMP_WORDS[COMP_CWORD]} prev=${COMP_WORDS[COMP_CWORD-1]} if [ $COMP_CWORD -eq 1 ] || [ "${prev:0:1}" = "-" ]; then COMPREPLY=( $( compgen -W 'add admin checkout commit diff \ export history import log rdiff release remove rtag status \ tag update' $cur )) else COMPREPLY=( $( compgen -f $cur )) fi return 0 } complete -F _cvs cvs _killall () { local cur prev COMPREPLY=() cur=${COMP_WORDS[COMP_CWORD]} # get a list of processes (the first sed evaluation # takes care of swapped out processes, the second # takes care of getting the basename of the process) COMPREPLY=( $( /usr/bin/ps -u $USER -o comm | \ sed -e '1,1d' -e 's#[]\[]##g' -e 's#^.*/##'| \ awk '{if ($0 ~ /^'$cur'/) print $0}' )) return 0 } complete -F _killall killall killps # Local Variables: # mode:shell-script # sh-shell:bash # End: |