Meeran
Meeran

Reputation: 91

Infinite while-loop in BASH script

I'm really struggling to see why this while-loop never ends, when the loop starts, my variable LOC is set to Testing/, which is a directory I created to test this program, it has the following layout:

Layout of my Testing Directory

I want the loop to end once all Directories have had the "count" function applied to them.
Here are the things I have tried;

I've checked my count function, and it doesn't produce an infinite loop

I've tried running through the algorithm by hand

PARSE=1
LOC=$LOC/
count
AVAILABLEDIR=$(ls $LOC -AFl | sed "1 d" |  grep "/$" | awk '{ print $9 }')
while [ $PARSE = "1" ]
do

if [[ ${AVAILABLEDIR[@]} == '' ]]; then
    PARSE=0
fi

DIRBASE=$LOC

for a in ${AVAILABLEDIR[@]}; do
LOC="${DIRBASE}${a}"
LOCLIST="$LOCLIST $LOC"
count
done

for a in ${LOCLIST[@]}; do
TMPAVAILABLEDIR=$(ls $a -AFl | sed "1 d" |  grep "/$" | awk '{ print $9 }')
PREPEND=$a
if [[ ${TMPAVAILABLEDIR[@]} == '' ]]; then
    continue
fi

for a in ${TMPAVAILABLEDIR[@]}; do
    TMPAVAILABLEDIR2="$TMPAVAILABLEDIR2 ${PREPEND[@]}${a}"
done

NEWAVAILABLEDIR="$NEWAVAILABLEDIR $TMPAVAILABLEDIR2"
done

AVAILABLEDIR=$NEWAVAILABLEDIR
NEWAVAILABLEDIR=''
LOC=''
done

I am really struggling, and any input would be greatly appreciated, I've been trying to figure this out for the last couple of hours.

Upvotes: 0

Views: 924

Answers (4)

Walter A
Walter A

Reputation: 20002

You wrote you want to perform "count" on all directories. Look at the options of find:

find $LOC -type d | while read dir; do
   cd $LOC
   cd ${dir}
   count
done

Or shorter (when your function count accepts a directory as parameter 1):

find $LOC -type d | xargs count

I now see you do not want to use find or ls -R (recursive function). Then you should make your own recursive function, something like

function parseDir {
   ls -d */ $1 | while read dir; do
      count
      parseDir $1/$dir
   done
}

Upvotes: 1

Hans Klünder
Hans Klünder

Reputation: 2292

If you really must avoid recursion, try this. It completely recursion-free:

#!/bin/bash

count() {
   echo counting "$1"
}

todo=(Testing)

while test ${#todo[@]} != 0
do
   doit=("${todo[@]}")
   todo=()
   for dir in "${doit[@]}"
   do
      for entry in "$dir"/*   # If the directory is empty, this shows an entry named "*"
      do
         test -e "$entry" || continue   # Skip the entry "*" of an empty directory
         count "$entry"
         test -d "$entry" || continue
         todo+=("$entry")
      done
   done
done

Upvotes: 1

Southie75
Southie75

Reputation: 1

I have no idea if this will work, but it’s an interesting question I couldn't stop thinking about.

while true ; do
    for word in "$(echo *)" ; do
        if [[ -d "$word" ]] ; then
            d[$((i++))]="$PWD"/"$word"
        elif [[ -f "$word" ]] ;then
            f[$((j++))]="$PWD"/"$word"
        fi
    done

    [[ $k -gt $i ]] && cd ..
    cd "$d[$((k++))]" || break
done

Upvotes: 0

Hans Klünder
Hans Klünder

Reputation: 2292

You should try to run the script with argument -x, or write it into the first line:

#!/bin/bash -x

Then it tells you everything it does.

In that case, you might notice two errors:

  1. You never reset TMPAVAILABLEDIR2

  2. You do ls on regular files as well.

Upvotes: 1

Related Questions