Variable not expanding inside another variable bash

20,004

Solution 1

You can use eval, but the next maintainer will curse your name. Have you tried just appending the variable when using the parsed data, or expanding the variable when creating the data?

If the file is created that way, can't you just use some [redacted] techniques to convince the originator to change their wicked ways?

If change is literally not possible, then you must have control over which variables are possible. Otherwise your script is vulnerable to all sorts of injection attacks, such as inputs like 3|sample1.txt|$(rm --arr --eff /)/operation/. Since you obviously have that under control, you can do some literal replacements of variables with their values on a case by case basis:

IFS='/' read -a pathnames <<< "$fpath"
for pathname in "${pathnames[@]}"
do
    if [ "${pathname::1}" = '$' ]
    then
        variable_name="${pathname:1}"
        printf '%s' "${!variable_name}"
    else
        printf '%s' "$pathname"
    fi
done

With some additional boilerplate to add slashes between pathnames.

Solution 2

You can also use indirect variable expansion

$fpath=`echo $resource/operation`
echo ${!fpath}

This will expand the value of resource inside fpath.

Solution 3

Variable expansion happens just once and isn't recursive.

You can expand the value yourself:

resource=/data/bscs
fpath='$resource/operation'
echo ${fpath/'$resource'/$resource}

Note that this is much safer than eval, because all the possible variables to substitute must be listed in the code.

Share:
20,004

Related videos on Youtube

Menon
Author by

Menon

Hi, I am a telecom engineer working in a software based company. Here, I work in billing software which is based on Unix(scripting), SQL and java. Additionally, I have studied web technologies(php,j2ee and asp.net(C#)). I am here to gain knowledge as well as help others whenever I can.

Updated on September 18, 2022

Comments

  • Menon
    Menon over 1 year

    I am reading a csv file in bash script as follows:

    resource='/data/bscs/'
    while IFS='|'read seqid fname fpath 
    do
        echo "FILENO:         $seqid"
        echo "FILENAME:       $fname"
        echo "FILE_PATH:    $fpath"                    
    done < "report.csv"
    

    My csv file has following values :

    3|sample1.txt|$resource/operation/

    I want the $resource to expand inside $fname. Instead, I am getting this output:

    FILENO:         3
    FILENAME:       sample1.txt
    SOURCE PATH:    $resource/operation
    

    I have tried the following:

    "${fpath}" $(echo $fpath)

    How can I achieve this?

  • Menon
    Menon about 9 years
    Unfortunately, I have to read values from file and there are different variables in file and it is not possible for me to expand all of them this way.
  • Menon
    Menon about 9 years
    I cannot expand variable in file because the file does not belong to me. The file is created that way.
  • Menon
    Menon about 9 years
    I cannot do this because there are other variables in file, so I will have to expand all of them. Besides, I dont know the input values.
  • rahul
    rahul about 9 years
    @Menon you've shot down all the ideas presented above. Could you probably add more detail to your question on what do you mean by other variables in file? Some examples perhaps?
  • Menon
    Menon about 9 years
    @I0b0, eval works for me and has not caused any problems till now. I think I will stick to that for the meantime. Thanks for helping.
  • AdminBee
    AdminBee over 2 years
    For the sake of the argument, one link to why eval is considered "evil"... ;)