]> git.sur5r.net Git - bacula/bacula/blobdiff - bacula/scripts/disk-changer.in
tweak debug
[bacula/bacula] / bacula / scripts / disk-changer.in
index c66799b50e0e8b13e8aeaee71b5a4ee1c1936c8c..53e884aae8f44f995fa7618c7467661f527fe15b 100644 (file)
@@ -2,7 +2,32 @@
 #
 # Bacula interface to virtual autoloader using disk storage
 #
-#  $Id$
+#  Written by Kern Sibbald
+#
+#  Copyright (C) 2000-2010 Free Software Foundation Europe e.V.
+#
+#  The main author of Bacula is Kern Sibbald, with contributions from
+#  many others, a complete list can be found in the file AUTHORS.
+#  This program is Free Software; you can redistribute it and/or
+#  modify it under the terms of version three of the GNU Affero General Public
+#  License as published by the Free Software Foundation, which is 
+#  listed in the file LICENSE.
+#
+#  This program is distributed in the hope that it will be useful, but
+#  WITHOUT ANY WARRANTY; without even the implied warranty of
+#  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+#  General Public License for more details.
+#
+#  You should have received a copy of the GNU Affero General Public License
+#  along with this program; if not, write to the Free Software
+#  Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
+#  02110-1301, USA.
+#
+#  Bacula® is a registered trademark of Kern Sibbald.
+#  The licensor of Bacula is the Free Software Foundation Europe
+#  (FSFE), Fiduciary Program, Sumatrastrasse 25, 8006 Zürich,
+#  Switzerland, email:ftf@fsfeurope.org.
+#
 #
 #  If you set in your Device resource
 #
 #    you will have the following input to this script:
 #
 #  So Bacula will always call with all the following arguments, even though
-#    in come cases, not all are used.
+#    in come cases, not all are used. Note, the Volume name is not always
+#    included.
 #
-#  disk-changer "changer-device" "command" "slot" "archive-device" "drive-index"
-#                  $1              $2       $3        $4               $5
+#  disk-changer "changer-device" "command" "slot" "archive-device" "drive-index" "volume"
+#                   $1              $2       $3        $4               $5         $6
 #
 # By default the autochanger has 10 Volumes and 1 Drive.
 #
@@ -23,8 +49,8 @@
 #
 # changer-device is the name of a file that overrides the default
 #   volumes and drives.  It may have:
-#       maxslot=n   where n is one based (default 10)
-#       maxdrive=m  where m is zero based (default 1 -- i.e. 2 drives)
+#        maxslot=n   where n is one based (default 10)
+#        maxdrive=m  where m is zero based (default 1 -- i.e. 2 drives)
 #  
 #   This code can also simulate barcodes. You simply put
 #   a list of the slots and barcodes in the "base" directory/barcodes.
@@ -43,7 +69,7 @@
 #  any other part of the directory name. These restrictions could be
 #  easily removed by any clever script jockey.
 #
-#  Full example: disk-changer /var/bacula/conf load 1 /var/bacula/drive0 0
+#  Full example: disk-changer /var/bacula/conf load 1 /var/bacula/drive0 0 TestVol001
 #
 # The Volumes will be created with names slot1, slot2, slot3, ... maxslot in the
 #  base directory. In the above example the base directory is /var/bacula.
@@ -52,6 +78,9 @@
 #  /var/bacula/slot3, ...) this script will create a /var/bacula/loadedn
 #  file to keep track of what Slot is loaded. You should not change this file.
 #
+# Modified 8 June 2010 to accept Volume names from the calling program as arg 6.
+#  In this case, rather than storing the data in slotn, it is stored in the
+#  Volume name.  Note: for this to work, Volume names may not include spaces.
 #
 
 wd=@working_dir@
@@ -64,8 +93,8 @@ wd=@working_dir@
 #
 dbgfile="$wd/disk-changer.log"
 debug() {
-    if test -e $dbgfile; then
-       echo "`date +\"%Y%m%d-%H:%M:%S\"` $*" >> $dbgfile
+    if test -f $dbgfile; then
+        echo "`date +\"%Y%m%d-%H:%M:%S\"` $*" >> $dbgfile
     fi
 }
 
@@ -78,8 +107,8 @@ make_temp_file() {
   if test x${TMPFILE} = x; then
      TMPFILE="$wd/disk-changer.$$"
      if test -f ${TMPFILE}; then
-       echo "Temp file security problem on: ${TMPFILE}"
-       exit 1
+        echo "Temp file security problem on: ${TMPFILE}"
+        exit 1
      fi
   fi
 }
@@ -90,14 +119,14 @@ check_parm_count() {
     pCount=$1
     pCountNeed=$2
     if test $pCount -lt $pCountNeed; then
-       echo "usage: disk-changer ctl-device command [slot archive-device drive-index]"
-       echo "  Insufficient number of arguments arguments given."
-       if test $pCount -lt 2; then
-           echo "  Mimimum usage is first two arguments ..."
-       else
-           echo "  Command expected $pCountNeed arguments"
-       fi
-       exit 1
+        echo "usage: disk-changer ctl-device command [slot archive-device drive-index]"
+        echo "  Insufficient number of arguments arguments given."
+        if test $pCount -lt 2; then
+            echo "  Mimimum usage is first two arguments ..."
+        else
+            echo "  Command expected $pCountNeed arguments"
+        fi
+        exit 1
     fi
 }
 
@@ -107,7 +136,30 @@ check_parm_count() {
 #
 get_dir() {
    bn=`basename $device`
-   dir=`echo "$device" | sed -e s%/$bn%%g -`
+   dir=`echo "$device" | sed -e s%/$bn%%g`
+   if [ ! -d $dir ]; then
+      echo "ERROR: Autochanger directory \"$dir\" does not exist."
+      echo "       You must create it."
+      exit 1
+   fi
+}
+
+#
+# Get the Volume name from the call line, or directly from
+#  the volslotn information.
+#
+get_vol() {
+   havevol=0
+   debug "vol=$volume"
+   if test "x$volume" != x && test "x$volume" != "x*NONE*" ; then
+      debug "touching $dir/$volume"
+      touch $dir/$volume
+      echo "$volume" >$dir/volslot${slot}
+      havevol=1
+   elif [ -f $dir/volslot${slot} ]; then
+      volume=`cat $dir/volslot${slot}`
+      havevol=1
+   fi
 }
 
 
@@ -117,6 +169,7 @@ cmd="$2"
 slot=$3
 device=$4
 drive=$5
+volume=$6
 
 # set defaults
 maxdrive=1
@@ -132,48 +185,101 @@ fi
 #  all others are a minimum of 5
 #
 case $2 in
-    list)
-       check_parm_count $# 2
-       ;;
+    list|listall)
+        check_parm_count $# 2
+        ;;
     slots)
-       check_parm_count $# 2
-       ;;
+        check_parm_count $# 2
+        ;;
+    transfer)
+        check_parm_count $# 4
+        if [ $slot -gt $maxslot ]; then
+           echo "Slot ($slot) out of range (1-$maxslot)"
+           exit 1
+        fi
+        ;;
     *)
-       check_parm_count $# 5
-       if [ $drive -gt $maxdrive ]; then
-          echo "Drive ($drive) out of range (0-$maxdrive)"
-          exit 1
-       fi
-       if [ $slot -gt $maxslot ]; then
-          echo "Slot ($slot) out of range (1-$maxslot)"
-          exit 1
-       fi
-       ;;
+        check_parm_count $# 5
+        if [ $drive -gt $maxdrive ]; then
+           echo "Drive ($drive) out of range (0-$maxdrive)"
+           exit 1
+        fi
+        if [ $slot -gt $maxslot ]; then
+           echo "Slot ($slot) out of range (1-$maxslot)"
+           exit 1
+        fi
+        ;;
 esac
 
 
-
-debug "Parms: $ctl $cmd $slot $device $drive"
+debug "Parms: $ctl $cmd $slot $device $drive $volume $havevol"
 
 case $cmd in 
    unload)
-      debug "Doing disk -f $ctl unload $slot $device $drive"
+      debug "Doing disk -f $ctl unload $slot $device $drive $volume"
       get_dir
-      echo "0" >$dir/loaded${drive}
-      unlink $device 2>/dev/null >/dev/null
-      rm -f $device
+      if [ -f $dir/loaded${drive} ]; then
+         ld=`cat $dir/loaded${drive}`
+      else 
+         echo "Storage Element $slot is Already Full"
+         exit 1
+      fi
+      if [ $slot -eq $ld ]; then
+         echo "0" >$dir/loaded${drive}
+         unlink $device 2>/dev/null >/dev/null
+         rm -f $device
+      else
+         echo "Storage Element $slot is Already Full"
+         exit 1
+      fi
       ;;
 
    load)
-      debug "Doing disk $ctl load $slot $device $drive"
+      debug "Doing disk $ctl load $slot $device $drive $volume"
       get_dir
+      i=0
+      while [ $i -le $maxdrive ]; do
+         if [ -f $dir/loaded${i} ]; then
+            ld=`cat $dir/loaded${i}`
+         else    
+            ld=0
+         fi
+         if [ $ld -eq $slot ]; then
+            echo "Drive ${i} Full (Storage element ${ld} loaded)"
+            exit 1
+         fi
+         i=`expr $i + 1`
+      done
+      # Check if we have a Volume name
+      get_vol
+      if [ $havevol -eq 0 ]; then
+         # check if slot exists
+         if [ ! -f $dir/slot${slot} ] ; then
+            echo "source Element Address $slot is Empty"
+            exit 1
+         fi
+      fi
+      if [ -f $dir/loaded${drive} ]; then
+         ld=`cat $dir/loaded${drive}`
+      else
+         ld=0
+      fi
+      if [ $ld -ne 0 ]; then
+         echo "Drive ${drive} Full (Storage element ${ld} loaded)"
+         exit 1
+      fi
       echo "0" >$dir/loaded${drive}
       unlink $device 2>/dev/null >/dev/null
       rm -f $device
-      ln -s $dir/slot${slot} $device
-      rtn=$?
+      if [ $havevol -ne 0 ]; then
+         ln -s $dir/$volume $device
+         rtn=$?
+      else
+         ln -s $dir/slot${slot} $device
+         rtn=$?
+      fi
       if [ $rtn -eq 0 ]; then
-        echo $slot >$dir/loaded${drive}
+         echo $slot >$dir/loaded${drive}
       fi
       exit $rtn
       ;;
@@ -182,24 +288,97 @@ case $cmd in
       debug "Doing disk -f $ctl -- to list volumes"
       get_dir 
       if [ -f $dir/barcodes ]; then
-        cat $dir/barcodes
+         cat $dir/barcodes
       else
-        i=1
-        while [ $i -le $maxslot ]; do
-           echo "$i:"
-           i=`expr $i + 1`
-        done
+         i=1
+         while [ $i -le $maxslot ]; do
+            slot=$i
+            volume=
+            get_vol
+            if [ $havevol -eq 0 ]; then
+               echo "$i:"
+            else
+               echo "$i:$volume"
+            fi
+            i=`expr $i + 1`
+         done
       fi
       exit 0
       ;;
 
+   listall) 
+      # ***FIXME*** must add new Volume stuff
+      make_temp_file
+      debug "Doing disk -f $ctl -- to list volumes"
+      get_dir 
+      if [ ! -f $dir/barcodes ]; then
+          exit 0
+      fi
+
+      # we print drive content seen by autochanger
+      # and we also remove loaded media from the barcode list
+      i=0
+      while [ $i -le $maxdrive ]; do
+         if [ -f $dir/loaded${i} ]; then
+             ld=`cat $dir/loaded${i}`
+             v=`awk -F: "/^$ld:/"' { print $2 }' $dir/barcodes`
+             echo "D:$i:F:$ld:$v"
+             echo "^$ld:" >> $TMPFILE
+         fi
+         i=`expr $i + 1`
+      done
+
+      # Empty slots are not in barcodes file
+      # When we detect a gap, we print missing rows as empty
+      # At the end, we fill the gap between the last entry and maxslot
+      grep -v -f $TMPFILE $dir/barcodes | sort -n | \
+      perl -ne 'BEGIN { $cur=1 } 
+       if (/(\d+):(.+)?/) {
+         if ($cur == $1) { 
+           print "S:$1:F:$2\n" 
+         } else { 
+           while ($cur < $1) {
+              print "S:$cur:E\n";
+              $cur++;
+           }
+         }
+         $cur++;
+       } 
+       END { while ($cur < '"$maxslot"') { print "S:$cur:E\n"; $cur++; } } '
+
+      rm -f $TMPFILE
+      exit 0
+      ;;
+   transfer)
+      #  ***FIXME*** must add new Volume stuff
+      get_dir
+      make_temp_file
+      slotdest=$device
+      if [ -f $dir/slot{$slotdest} ]; then
+         echo "destination Element Address $slot is Full"
+         exit 1
+      fi
+      if [ ! -f $dir/slot${slot} ] ; then
+         echo "source Element Address $slot is Empty"
+         exit 1
+      fi
+
+      echo "Transfering $slot to $slotdest"
+      mv $dir/slot${slot} $dir/slot{$slotdest}
+
+      if [ -f $dir/barcodes ]; then
+         sed "s/^$slot:/$slotdest:/" >  $TMPFILE
+         sort -n $TMPFILE > $dir/barcodes
+      fi
+      exit 0
+      ;;
    loaded)
       debug "Doing disk -f $ctl $drive -- to find what is loaded"
       get_dir
       if [ -f $dir/loaded${drive} ]; then
-        cat $dir/loaded${drive}
+         cat $dir/loaded${drive}
       else
-        echo "0"
+         echo "0"
       fi
       exit
       ;;