3 #########################################################################
4 # This program is free software: you can redistribute it and/or modify #
5 # it under the terms of the version 3 of the GNU General Public License #
6 # as published by the Free Software Foundation. #
8 # This program is distributed in the hope that it will be useful, but #
9 # WITHOUT ANY WARRANTY; without even the implied warranty of #
10 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU #
11 # General Public License for more details. #
13 # You should have received a copy of the GNU General Public License #
14 # along with this program. If not, see <http://www.gnu.org/licenses/>. #
16 # Written by and Copyright (C) Francois Fleuret #
17 # Contact <francois@fleuret.org> for comments & bug reports #
18 #########################################################################
20 # The site-specific and confidential settings are in another file
22 PRIVATE_BASHRC="${HOME}/private/bashrc.perso"
24 # If the MANPATH is not set, set it
26 [ "${MANPATH}" ] || MANPATH=$(manpath)
28 # If the private bashrc exists, execute it
30 [ -f "${PRIVATE_BASHRC}" ] && source "${PRIVATE_BASHRC}"
32 # !!! THIS HAS TO BE HERE EVEN IN THE NON-INTERACTIVE PART OR YOU WILL
33 # LOSE YOU PREVIOUS HISTORY !!!
35 export HISTFILESIZE=20000
36 export HISTSIZE=${HISTFILESIZE}
38 export HISTIGNORE="${HISTIGNORE}:&:[ ]*"
40 # I want to save the command time, but I do not want to see it in
43 export HISTTIMEFORMAT=""
47 # I realized that most of my settings are meaningful only in
48 # interactive mode. This should maybe be done more properly through
49 # using different .bash_profile and .bash_login
51 [[ ${TERM} == "dumb" ]] || [ ! -t 0 ] && return
53 ######################################################################
54 ## The interactive part
56 export VT_RESET=$'\033[0m'
57 export VT_BOLD=$'\033[1m'
58 export VT_UNDERLINE=$'\033[4m'
59 export VT_BLINK=$'\033[5m'
61 export VT_SET_TITLE=$'\033]0;'
62 export VT_END_TITLE=$'\007'
64 export VT_BLACK_FG=$'\033[30m'
65 export VT_RED_FG=$'\033[31m'
66 export VT_GREEN_FG=$'\033[32m'
67 export VT_YELLOW_FG=$'\033[33m'
68 export VT_BLUE_FG=$'\033[34m'
69 export VT_MAGENTA_FG=$'\033[35m'
70 export VT_CYAN_FG=$'\033[36m'
71 export VT_WHITE_FG=$'\033[37m'
73 export VT_BLACK_BG=$'\033[40m'
74 export VT_RED_BG=$'\033[41m'
75 export VT_GREEN_BG=$'\033[42m'
76 export VT_YELLOW_BG=$'\033[43m'
77 export VT_BLUE_BG=$'\033[44m'
78 export VT_MAGENTA_BG=$'\033[45m'
79 export VT_CYAN_BG=$'\033[46m'
80 export VT_WHITE_BG=$'\033[47m'
82 # This prevents ^S from freezing the shell
88 # alias chmod='chmod -v'
89 alias chmod='chmod -c'
94 alias df='df -hT --sync'
95 alias grep='grep -E --mmap'
97 alias s='screen -d -R -U && clear'
98 alias mc='echo Try mv ' # I'm fed up with midnight commander
99 alias kj="keyjnote -s -D 1000 -t Crossfade -T 100"
101 # alias fdupes='fdupes -r .'
105 if [ -e "${HOME}/.dircolors" ]; then
106 eval $(dircolors "${HOME}/.dircolors")
107 alias ls='ls --color'
108 # alias ll='ls --color -lth'
109 alias ll='ls --color -goh --time-style="+%Y %b %d %H:%M"'
110 alias l='ls --color -I "*~" -I "*.o"'
114 alias ll='ls -goh --time-style="+%Y %b %d %H:%M"'
115 alias l='ls -I "*~" -I "*.o"'
118 export EDITOR=emacsclient
119 export GIT_EDITOR=${EDITOR}
121 ######################################################################
122 # Ignored extensions when completing
126 ######################################################################
129 # Find a file containing a name
134 find $* -name "*${name}*";
137 # Create a dir and cd there
146 ######################################################################
147 # http://www.reddit.com/r/linux/comments/akt3j/a_functional_programming_style_map_function_for/
151 if [ $# -lt 2 ] || [[ ! "$@" =~ :[[:space:]] ]];then
152 echo "Invalid syntax." >&2; return 1
154 until [[ $1 =~ : ]]; do
155 command="$command $1"; shift
157 command="$command ${1%:}"; shift
159 if [[ $command =~ \{\} ]];then
160 rep="${command//\{\}/\"$i\"}"
161 eval "${rep//\\/\\\\}"
163 eval "${command//\\/\\\\} \"${i//\\/\\\\}\""
168 ######################################################################
169 ## A version of pho which stores the image numbers in environment
174 TEMP=$(mktemp /tmp/pho.XXXXXXX)
175 ${PHO_BIN} $* | tee ${TEMP}
176 PHO_NOTE_1=$(grep ^"Note 1: " ${TEMP} | sed -e "s/^[^:]*: //")
177 PHO_NOTE_2=$(grep ^"Note 2: " ${TEMP} | sed -e "s/^[^:]*: //")
178 PHO_NOTE_3=$(grep ^"Note 3: " ${TEMP} | sed -e "s/^[^:]*: //")
179 PHO_NOTE_R90=$(grep ^"Rotate 90 \(CW\): " ${TEMP} | sed -e "s/^[^:]*: //")
180 PHO_NOTE_R180=$(grep ^"Rotate 180: " ${TEMP} | sed -e "s/^[^:]*: //")
181 PHO_NOTE_R270=$(grep ^"Rotate -90 \(CCW\): " ${TEMP} | sed -e "s/^[^:]*: //")
185 # function rotjpeg () {
186 # if [ $1 == "90" ] || [ $1 == "180" ] || [ $1 == "270" ]; then
187 # TEMP=$(mktemp /tmp/rotjpeg.XXXXXX)
188 # echo jpegtran -rotate $1 -copy all $2 > ${TEMP}
189 # echo cp $2 ${2/jpg/}original.jpg
193 # echo "Can not rotate with an angle of $1 degrees."
197 ######################################################################
198 ## A version of date that shows the time at home if TZ is set
201 echo "Local: $(date)"
208 ######################################################################
209 ## ifup / ifdown with sudo and memorization of the network
211 ## When invoked without an argument this "ifup" uses the same argument
212 ## as the previous time
214 ## When invoked without an argument this "ifdown" removes the last
215 ## interface which was ifuped
217 [[ ${IFUPRC} ]] || IFUPRC="${HOME}/.ifuprc"
220 echo "${VT_BOLD}${VT_GREEN_FG}This is the bash function ifup from .bashrc${VT_RESET}"
221 if [[ ! $* ]] && [[ -s ${IFUPRC} ]]; then
222 # If we have no argument and there is a .ifuprc, use it
223 ARGS=$(cat ${IFUPRC})
225 # Otherwise uses the given arguments, and store them
227 echo ${ARGS} > ${IFUPRC}
229 echo "${VT_GREEN_FG}Running [sudo ifup ${ARGS}]${VT_RESET}"
232 # Ugly hack to remove the dsl modem dns server when we add
233 # explicitely a dns in the /etc/network/interfaces
235 REMOVE_LOCAL_DNS=/usr/local/bin/remove-local-dns.sh
237 if [[ -x ${REMOVE_LOCAL_DNS} ]]; then
238 echo "${VT_GREEN_FG}Running [sudo ${REMOVE_LOCAL_DNS} 192.168]${VT_RESET}"
239 sudo ${REMOVE_LOCAL_DNS} 192.168
244 echo "${VT_BOLD}${VT_GREEN_FG}This is the bash function ifdown from .bashrc${VT_RESET}"
245 if [[ ! $* ]] && [[ -s ${IFUPRC} ]]; then
246 # If there are no arguments and there is a .ifuprc, get the
248 ARGS=$(cat ${IFUPRC} | sed -e "s/=.*$//")
250 # Otherwise, use the standard ifdown
253 echo "${VT_GREEN_FG}Running sudo [ifdown ${ARGS}]${VT_RESET}"
257 function checkgw () {
258 ping $(route -n | grep ^0.0.0.0 | awk '{print $2}')
261 ######################################################################
262 # Show the most recent files, no scroll
265 HEIGHT=$(stty size | awk '{print $1}')
266 WIDTH=$(stty size | awk '{print $2}')
267 \ls -goth --time-style="+%Y %b %d %H:%M" $* | head -$((HEIGHT-2)) | cut -b1-${WIDTH}
268 # \ls -lth $* | head -$((HEIGHT-2)) | cut -b1-${WIDTH}
271 ######################################################################
272 # cd and ls into a directory
273 # [from http://www.oreillynet.com/onlamp/blog/2007/01/whats_in_your_bash_history.html]
275 # function c () { cd "$@" && lr; }
277 ######################################################################
278 # You can change the xterm background color on the fly!
280 function setxtermbg () {
281 echo -n $'\033]11;'$1$'\007'
284 ######################################################################
285 # Shuffle the lines from the stdin
287 function shuffle () {
289 [[ $SEED ]] || SEED=0
290 awk 'BEGIN{srand('${SEED}')} { print rand()" "$0 }' | sort -g | sed -e "s/^[0-9\.e\-]* //"
293 ######################################################################
294 # Stores the last entered command into a file
296 KEPT_COMMANDS=${HOME}/.kept_bash_commands
299 if [[ ${KEPT_COMMANDS} ]]; then
300 LINE=$(history | tail -2 | head -1 | sed -e "s/^[0-9 ]*//")
302 echo $(date)": "${LINE} >> ${KEPT_COMMANDS}
304 echo "You have to set \$KEPT_COMMANDS"
308 ######################################################################
309 # I sometime burn CDs and DVDs
315 echo "burn <iso name | dirname>" >&2
316 elif [[ -f $1 ]]; then
317 if [[ $(file $1 | grep "ISO 9660") ]]; then
318 wodim -eject -v dev=${DEVICE} $1
320 echo "Unknown type of $1" >&2
322 elif [[ -d $1 ]]; then
323 TMP=$(mktemp /tmp/cdimage.XXXXXX) && \
324 genisoimage -input-charset iso8859-1 -r -o ${TMP} $1 && \
325 wodim -eject -v dev=${DEVICE} ${TMP}
328 echo "Can not find $1" >&2
332 ######################################################################
333 # And watch DVDs too!
338 echo " ! and @ Seek to the beginning of the previous/next chapter"
339 echo " j Cycle through the available subtitles"
340 echo " o Show/hide the timing"
347 dvd_device="/dev/cdrom"
357 mplayer > /dev/null \
359 -vc ffmpeg12 -quiet \
362 -dvd-device ${dvd_device} dvd://${title}
369 mkdir -p ${HOME}/dvds
375 ######################################################################
376 # Upload the sources from the current directory to work
379 if [[ ! "${MY_WORK_MACHINE}" ]]; then
380 echo "\$MY_WORK_MACHINE undefined" 1>&2
386 scp {Makefile,*.{cc,h,sh}} ${MY_WORK_MACHINE}:${DIR}
388 echo "Uploaded to ${MY_WORK_MACHINE}:${DIR}/"
391 ######################################################################
392 # Create small images from images
394 function mksmall () {
396 PARAMS="-geometry 800x600"
398 # Auto-orient does not seem to work at all, hence the ugly hack
401 # PARAMS="-auto-orient -geometry 800x600"
403 echo "Using ${PARAMS}"
407 [[ ${DEST_DIR} ]] || DEST_DIR=./small
411 if [[ ! -d ${DEST_DIR} ]]; then
412 echo "Can not create ${DEST_DIR}" >&2
416 NB_TOTAL=$(find -maxdepth 1 -type f | wc -l)
419 for i in $(find -maxdepth 1 -type f); do
420 if [[ -e ${DEST_DIR}/$i ]]; then
421 echo "The file ${DEST_DIR}/$i already exists."
424 orientation=$(exif $i \
425 | grep ^Orientation \
427 | sed -e "s/^[^|]*|//" \
430 case ${orientation} in
436 rotation_cmd="-rotate 90"
440 rotation_cmd="-rotate 270"
445 echo "Unknown orientation \"${orientation}\" !"
449 if [[ -f ${i/JPG/MOV} ]]; then
450 CAPTION_PARAMS="-font vera-sans -pointsize 24 -fill white -annotate +10+32 Video"
455 convert ${rotation_cmd} $i ${PARAMS} ${CAPTION_PARAMS} ${DEST_DIR}/$i
457 \ls -lt ${DEST_DIR}/$i
459 echo "$((NB*100/NB_TOTAL))% (${NB}/${NB_TOTAL})"
463 ######################################################################
464 # Move a file to the ~/sources/config directory and replace it where
465 # it was by a symbolic link
467 function mvtoconfig () {
468 CONFIGDIR=${HOME}/sources/config
469 if [[ -d ${CONFIGDIR} ]]; then
470 NEWNAME=${CONFIGDIR}/$(basename $1 | sed -e "s/^\.//")
474 echo "Can not find ${CONFIGDIR}"
478 ######################################################################
479 # Track uncommited files (I presume this is very ugly from a real git
486 for i in $(find -name ".git"); do
487 NB_SUBDIR=$((NB_SUBDIR+1))
488 cd ${CURRENT_DIR}/$(dirname $i)
489 NB_MODIFIED=$(git status | grep modified | wc -l)
490 if [[ ${NB_MODIFIED} -gt 0 ]]; then
491 echo "$(dirname $i) (${NB_MODIFIED})"
492 git status | grep modified \
493 | sed -e "s/^#\t/ /" | sed -e "s/modified: *//"
499 echo "Visited ${NB_SUBDIR} directories."
502 ######################################################################
503 # Commits all directories under git
508 for d in $(find ${PWD} -name ".git" | sed -e "s/\.git$//"); do
510 NB_MODIFIED=$(git status | grep modified | wc -l)
511 if [[ ${NB_MODIFIED} -gt 0 ]]; then
512 if [[ $(pwd) =~ ${NO_AUTOMATIC_GIT_COMMIT} ]]; then
513 UNCOMMITTED="${UNCOMMITTED} $(pwd)"
515 echo $(pwd)" (${NB_MODIFIED} modified file(s))"
516 git commit -a -m "Automatic commit" | grep -v ^#
524 if [[ ${UNCOMMITTED} ]]; then
525 echo "** WARNING: Did not automatically commit${UNCOMMITTED}"
529 ######################################################################
530 # Backups all git directories into an encrypted backup file located
531 # either on the usb key or the SD card (in that order) if they can be
534 function git-backup () {
538 mount ${BACKUPDIR} 2> /dev/null
540 if [[ ! $(mount | grep ${BACKUPDIR}) ]]; then
545 if [[ $(mount | grep ${BACKUPDIR}) ]]; then
546 echo "Mounted ${BACKUPDIR}"
548 echo "Could not mount the backup directory"
552 RESULT=${BACKUPDIR}/gitbackup-$(date +%F-%H%M%S).tgz.mc
554 tar zcvf - $(find ${HOME}/ -name .git) \
555 | mcrypt -f ${HOME}/private/mcrypt.key > ${RESULT}
557 if [[ -f ${RESULT} ]]; then
560 echo "Could not create the backup!"
566 umount ${BACKUPDIR} && echo "Umounted ${BACKUPDIR}"
569 ######################################################################
570 # Downloads torrents located in ${BT_DIR}/torrents/ and puts the
571 # result in the ${BT_DIR}
574 if [[ ${BT_DIR} ]]; then
575 if [[ -d "${BT_DIR}/torrents" ]]; then
577 mv $1 ${BT_DIR}/torrents
579 if [[ "$(ps auxwww | grep btlaunchmanycurses | grep -v grep)" ]]; then
580 echo "A client is already running."
582 cd ${BT_DIR} && screen btlaunchmanycurses torrents --max_upload_rate 32
585 echo "Directory ${BT_DIR}/torrents does not exist."
588 echo "You have to set \$BT_DIR."
592 ######################################################################
593 # The complex prompt policy
597 if [ "${CONSOLE}" == "yes" ]; then
601 # If the login is a standard one (as specified in
602 # IGNORED_PROMPT_LOGIN, which is set in the private bash file), do not
603 # show it. I have IGNORED_PROMPT_LOGIN="^fleuret$".
605 if [ ! ${IGNORED_PROMPT_LOGIN} ] || [[ ! ${USER} =~ ${IGNORED_PROMPT_LOGIN} ]]; then
609 # If the display is not the main one, make the assumption that the
610 # shell is not running on the localhost, and show the hostname
612 [ "${DISPLAY}" != ":0.0" ] && IDENT="${IDENT}@\h"
614 # If there is the login or the hostname, add a ":" to the prompt
616 [ "${IDENT}" ] && IDENT="${IDENT}:"
618 # If we are root, show that in red
620 if [[ ${USER} == "root" ]]; then
621 PS1="\[${VT_RED_BG}${VT_WHITE_FG}\]${IDENT}\w\[${VT_RESET}\] "
623 PS1="\[${VT_WHITE_BG}${VT_BLACK_FG}\]${IDENT}\w\[${VT_RESET}\] "
626 # In an xterm, show the hostname and path in the title bar, highlight
629 # [ "${TERMS_WITH_BAR}" ] || TERMS_WITH_BAR="^xterm|screen$"
631 # if [[ "${TERM}" =~ "${TERMS_WITH_BAR}" ]]; then
632 # PS1="\[${VT_SET_TITLE}shell@\h (\w)${VT_END_TITLE}${VT_WHITE_BG}\]${IDENT}\w\[${VT_RESET}\] "
634 # PS1="\[${VT_WHITE_BG}\]${IDENT}\w\[${VT_RESET}\] "
639 ######################################################################
640 # This implements a local history. If we are in a directory containing
641 # a writable local history file, we add the last line of the global
644 LOCAL_HISTORY_FILE=".local_bash_history"
646 function keep_local_history () {
647 if [[ -w "${LOCAL_HISTORY_FILE}" ]]; then
648 history 1 | sed -e 's/^ *[0-9]* *//' >> ${LOCAL_HISTORY_FILE}
649 TMP=$(mktemp /tmp/lh.XXXXXX)
651 uniq < ${LOCAL_HISTORY_FILE} | tail -${HISTSIZE} > ${TMP}
652 # mv would replace a symbolic link, while cp keeps it
653 \cp ${TMP} ${LOCAL_HISTORY_FILE}
655 LOCAL_HISTORY_HINT=" LH "
657 LOCAL_HISTORY_HINT=""
661 PS1="\[${VT_WHITE_BG}\]\${LOCAL_HISTORY_HINT}\[${VT_RESET}\]${PS1}"
663 ######################################################################
664 # Switch off the history
666 alias nh=" export HISTFILE=/dev/null"
668 function histfile_cue () {
669 if [[ ! "${HISTFILE}" == "${HOME}/.bash_history" ]]; then
670 HISTORY_CUE="[${HISTFILE}]"
676 PS1="\[${VT_YELLOW_BG}\]\${HISTORY_CUE}\[${VT_RESET}\]${PS1}"
678 ######################################################################
679 # The dus command is available on my web site
681 # git clone http://fleuret.org/git/dus/
685 ######################################################################
686 # The finddup command is available on my web site
688 # git clone http://fleuret.org/git/finddup/
690 # alias finddup='finddup -p0d'
691 alias finddup='finddup -p'
693 ######################################################################
694 # This script grep messages in my mail archives
698 ######################################################################
699 # Selector based history
701 # The selector command is available on my web site
703 # git clone http://fleuret.org/git/selector/
705 function selector-history () {
706 selector -c 7,4,0,3 -q -b -i -d -v -w -l 15000 <(history)
709 # M-r puts the selected history line in place of the current one
711 bind '"\C-[r":"\C-a\C-kselector-history\C-m"'
713 # M-t appends the selected history line and the end of the current one
715 bind '"\C-[t":"\C-a\C-kselector-history\C-m\C-a\C-y\C-e"'
717 # Find pathes in the history and make a list of the existing ones
719 function selector-cd () {
720 LIST_TEMP=$(mktemp /tmp/cdlist.XXXXXX)
721 for d in $(history | \
722 grep ^" *[0-9]* *cd" | \
723 awk '{ print $3 }' | \
730 selector -v -x '!' -d -i ${LIST_TEMP}
734 bind '"\C-[c":"\C-a\C-kselector-cd\C-m"'
736 # And we avoid to put in the history the use of the selector, which we
739 HISTIGNORE="${HISTIGNORE}:selector-history"
741 ######################################################################
743 function prompt_command () {
744 # save the history after every command to avoid loosing some when
745 # multiple shells are open
747 # and the local histories system defined above
749 # and the history cue
753 PROMPT_COMMAND="prompt_command"
755 ######################################################################
757 # Displaying the timezone if it is set
760 echo "${VT_BOLD}${VT_GREEN_FG}Time zone is ${TZ}.${VT_RESET}"
763 ######################################################################