4

Given this file

printf 'alpha\0bravo\0charlie' > delta.txt

I would like to read the fields into separate variables. The reason I am using a null separator is because the fields will contain file paths, which can contain any character except null. I tried these commands:

IFS= read mike november oscar < delta.txt
IFS=$'\0' read mike november oscar < delta.txt

However the fields are not being properly split

$ echo $mike
alphabravocharlie
Zombo
  • 1
  • 62
  • 391
  • 407

2 Answers2

3

The assignment IFS=$'\0' doesn't make the null character a delimiter, since Bash variables cannot hold null characters. IFS=$'\0' is equivalent to IFS=, which you can verify by doing:

bash-4.3$ IFS=$'\0'
bash-4.3$ echo ${#IFS}
0

And IFS= by definition means no word splitting at all (see Bash Reference Manual).

What you can do is read the null delimited items one by one using the -d option of the read builtin. According to linked documentation,

-d delim

The first character of delim is used to terminate the input line, rather than newline.

We can use a null string for delim to get desired behavior.

Example (I took the liberty to add whitespace to your example to demonstrate how it achieves what to want — no splitting on whitespace):

bash-4.3$ printf 'alpha with whitespace\0bravo with whitespace\0charlie with whitespace' > delta.txt
bash-4.3$ { read -r -d '' mike; IFS= read -r -d '' november; IFS= read -r -d '' oscar; echo $mike; echo $november; echo $oscar; } < delta.txt
alpha with whitespace
bravo with whitespace
charlie with whitespace

I'm also using the -r option to preserve backslashes in the input file. Of course you can replace < delta.txt with a cat delta.txt | at the beginning.

I know reading one by one is annoying, but I can't think of anything better.

4ae1e1
  • 7,228
  • 8
  • 44
  • 77
1

As a workaround, I created this function

function read_loop {
  while [ "$#" -gt 0 ]
  do
    read -d '' "$1"
    shift
  done
}

Example use

read-nul mike november oscar < delta.txt
Zombo
  • 1
  • 62
  • 391
  • 407