ksh - Get last character from string - Bad Substitution error


 
Thread Tools Search this Thread
Top Forums Shell Programming and Scripting ksh - Get last character from string - Bad Substitution error
# 1  
Old 05-02-2014
ksh - Get last character from string - Bad Substitution error

I want to get the last character from my machine name using the following code, the default shell is bash, the script runs in ksh.
I get 'bad' substitution error on running the script, but works fine if run using dot and space.

Why?

Code:
[ysrini@linuxapp01a bin]$ echo $0
bash

[ysrini@linuxapp01a bin]$ cat -n myenv.sh 
     1  #!/usr/bin/ksh
     2
     3  export SERVER_NAME=`hostname`
     4  echo "SERVER_NAME : ${SERVER_NAME}"
     5
     6  export SERVER_NODE=`echo ${SERVER_NAME:${#SERVER_NAME} - 1}`
     7  echo "SERVER_NODE : ${SERVER_NODE}"
     8
     9  export SERVER_ROOT_NAME=`echo "${SERVER_NAME%?}"`
    10  echo "SERVER_ROOT_NAME : ${SERVER_ROOT_NAME}"


[ysrini@linuxapp01a bin]$ myenv.sh 
SERVER_NAME : linuxapp01a
./myenv.sh[6]: : bad substitution
SERVER_NODE : 
SERVER_ROOT_NAME : linuxapp01

[ysrini@linuxapp01a bin]$ . myenv.sh 
SERVER_NAME : linuxapp01a
SERVER_NODE : a
SERVER_ROOT_NAME : linuxapp01

# 2  
Old 05-02-2014
You need to understand how "variable expansion" works: lets assume we have a variable assigned (i suggest you try the examples yourself at the shell prompt and play around a bit with them to get familiar):

Code:
var="abc/def/ghi"

The point of shell variables is that you cannot use them directly, like in other programming languages:

Code:
x="acd"
y="def"
z=x+y
print z

This (or similar constructs) would work in other languages, but in the shell you use the variable "indirectly", once you have assigned it, through a set of quasi-functions. This is "variable expansion". The most basic expansion is:

Code:
${var}

which will expand to the content of the variable "var". This content is replaced at the command line and then the command line is executed. For instance:

Code:
# print - "${var}"         # your command
# print - "abc/def/ghi"    # the shell first expands the expression
abc/def/ghi                # excuting the print command

Keep in mind this mechanism when we discuss more complicated expansions. The next in the list are these:

Code:
${var#<regex>}    ${var##<regex>}
${var%<regex>}    ${var%%<regex>}

The first one ("#") takes the content of the variable, then takes the regexp, expands that and if it matches the beginning of the content, the matching part is cut off. Sounds complicated? OK, here is an example with our variable from above:

Code:
# print - "${var#?}"
bc/def/ghi

The complete content would be "abc/def/ghi". The regexp ("?") means "any one single character", which is deleted from the beginning, therefore leaving the first character out. Notice, that this DOES NOT CHANGE the variable at all:

Code:
# print - "${var#?}"
bc/def/ghi
# print - "${var}"
abc/def/ghi

The opposite of "#" is "%", which works the same, but takes away from the content at the end instead of the beginning. Also notice the "*", which means "any number of any characters". In case you wonder: yes, these are the same characters you can use as filemasks when issuing a "ls -l <mask>". Whatever you can use there you can use here:

Code:
# print - "${var%?}"
abc/def/gh
# print - "${var%/*}"
abc/def

You might wonder what the difference between "#" and "##" and "%" and "%%" respectively is. Try out the following and notice the difference:

Code:
# print - "${var%/*}"
# print - "${var%%/*}"
# print - "${var#*/}"
# print - "${var##*/}"

The one is always the shortest possible match the other the longest possible match. For matches which only occur once there is no difference.

There are a lot of other interesting and powerful expansions: you can replace one substring with another:

Code:
${var/<search>/<replace>}

and a lot of other things. Check out "man ksh" for reference.

I'd like to show you another trick: nested expansions. You can use an expansion inside another expansion, even if it uses the same variable (because - you know already - the variables content itself is not changed!). This, finally, will do what you look for:

Remember what this gives:

Code:
${var%?}

Correct: everything save for the last character. Now, let us use this as the regexp we want to take away from the beginning of the content. Obviously this will match most of the content and only leave the last character, yes?

Code:
# print - "${var#${var%?}}"
i

And because this will always be true you can use this regexp every time, regardless of what the content of "var" is, for the last characters - or more characters, if you modify it a bit:

Code:
# print - "${var#${var%??}}"       # the last 2 characters
# print - "${var%${var#??}}"       # the first 2 characters

I hope this helps.

bakunin
These 3 Users Gave Thanks to bakunin For This Post:
# 3  
Old 05-02-2014
Thanks Bakunin, not only did you provide the answer but a good explanation of understanding the variable expansion. The 'solution' is always more useful than direct 'answer' !
Thanks again
-srinivas y.
# 4  
Old 05-02-2014
You still should be careful about what shell you are using. The shebang and the thread title say ksh, but your echo $0 says bash. Though it doesn't matter here, there might be differences in other places that lead to surprising resilts.
# 5  
Old 05-02-2014
Quote:
Originally Posted by bakunin
Code:
# print - "${var#${var%??}}"       # the last 2 characters
# print - "${var%${var#??}}"       # the first 2 characters

That code is vulnerable to pattern matching metacharacters. For this approach to work with arbitrary text, it is necessary to double-quote the nested parameter expansion.
Code:
$ s=*****a
$ echo "$s"
*****a
$ echo "${s#${s%?}}"
*****a
$ echo "${s##${s%?}}"

$ echo "${s#"${s%?}"}"
a

A minor nit: You refer to shell pattern matching as regular expressions. I'm sure you know that those are two distinct grammars, but a novice may become confused.

Regards,
Alister
These 2 Users Gave Thanks to alister For This Post:
# 6  
Old 05-02-2014
Quote:
Originally Posted by alister
That code is vulnerable to pattern matching metacharacters. For this approach to work with arbitrary text, it is necessary to double-quote the nested parameter expansion.
You are right and your example is legitimate. I left that part out purposefully to avoid complicating matters. I should have probably mentioned it.

Quote:
Originally Posted by alister
A minor nit: You refer to shell pattern matching as regular expressions. I'm sure you know that those are two distinct grammars, but a novice may become confused.
Yes - and no. "regular expressions" is (in a very theoretical sense) any type-3 language in the Chomsky hierarchy: a device where some characters and some metacharacters describe a text pattern. This is the case for shell regexps (aka "file globs") as well as for "Unix Basic Regular Expressions" (what awk, grep and sed use) or "Extended Regular Expressions" (i.e. perl and some GNU variants of grep, sed, ...). These are all different flavours of Regexps (and i should have mentioned that too, probably), but still Regexps nevertheless.

You are right, though, that in UNIX environments, the term "regexp" particularily describes BREs as used in sed, awk and grep. Every other use of the term, even if technically correct, might be confusing.

bakunin
This User Gave Thanks to bakunin For This Post:
# 7  
Old 05-03-2014
Quote:
Originally Posted by bakunin
Yes - and no. "regular expressions" is (in a very theoretical sense) any type-3 language in the Chomsky hierarchy: a device where some characters and some metacharacters describe a text pattern. This is the case for shell regexps (aka "file globs") as well as for "Unix Basic Regular Expressions" (what awk, grep and sed use) or "Extended Regular Expressions" (i.e. perl and some GNU variants of grep, sed, ...). These are all different flavours of Regexps (and i should have mentioned that too, probably), but still Regexps nevertheless.
If we are going to be precise with regard to formal language theory, then you are mistaken. Neither POSIX Basic Regular Expressions, nor the "extended" dialects implemented in perl, python, php, java, et al, are regular expressions. Any grammar that supports backreferences cannot be implemented with a [non-]deterministic finite automaton (a defining characteristic of a regular language). sh pattern matching and POSIX Extended Regular Expressions, however, are formally regular languages.

As you noted, I was simply using conventional, informal nomenclature. The ksh/bash man pages make a concerted effort to not use the term 'regular expression' when discussing pattern matching notation.

Minor nit: AWK uses POSIX Extended Regular Expressions, not Basic.

Regards,
Alister
This User Gave Thanks to alister For This Post:
Login or Register to Ask a Question

Previous Thread | Next Thread

10 More Discussions You Might Find Interesting

1. Shell Programming and Scripting

Bad substitution

Cant undestand :) why i have an error on line 2.it is working on my other boxes #!/bin/bash ret=$(echo Q | timeout 5 openssl s_client connect "${1`hostname`}:${2-443}" -ssl3 2> /dev/null) if echo "${ret}" | grep -q 'Protocol.*SSLv3'; then if echo "${ret}" | grep -q 'Cipher.*0000'; then ... (7 Replies)
Discussion started by: kenshinhimura
7 Replies

2. Shell Programming and Scripting

Bad substitution error while working with substring

Hi I'm using ksh. And i'm trying to get the substring like below. but giving the following error #!/bin/ksh foo=teststring bar=${foo:0:5} echo $bar And the error is ./sbstr_test.sh: bar=${foo:0:5}: bad substitution what is wrong in this script. Please correct me ... (3 Replies)
Discussion started by: smile689
3 Replies

3. Shell Programming and Scripting

Bad substitution error in shell script

I have script data.sh which has following error. Script Name : data.sh #!/bin/sh infile=$1 len=${#infile} echo $len texfile=${infile:0:$len-4} echo $texfile run command ./data.sh acb.xml I get following error message: (5 Replies)
Discussion started by: man4ish
5 Replies

4. Shell Programming and Scripting

Why I get bad bad substitution when using eval?

Why I get bad replace when using eval? $ map0=( "0" "0000" "0") $ i=0 $ eval echo \${map$i} 0000 $ a=`eval echo \${map$i}` !!!error happens!!! bash: ${map$i}: bad substitution How to resolve it ? Thanks! (5 Replies)
Discussion started by: 915086731
5 Replies

5. Shell Programming and Scripting

bad substitution error!

Hi All, I'm building a new shell script but i'm facing a problem with one line which is giving "bad substitution" error. Please assist script lines: #!/bin/sh printf "%s: " "Occurrence DATE (YYYYMMDD)"; read DATE shortdate=${DATE#??} o/p: ./test1: bad substitution This command is... (2 Replies)
Discussion started by: Dendany83
2 Replies

6. Shell Programming and Scripting

KSH: substitution character, AWK or SED?

Hi Gurus, I am working with a korn shell script. I should replace in a very great file the character ";" with a space. Example: 2750;~ 2734;~ 2778;~ 2751;~ 2751;~ 2752;~ what the fastest method is? Sed? Awk? Speed is dead main point, Seen the dimensions of the files Thanks (6 Replies)
Discussion started by: GERMANICO
6 Replies

7. UNIX for Dummies Questions & Answers

bad substitution

#!/bin/bash a1=( win 12,01,02,03,04 ) a2=( pre 04,05,06 ) a3=( msn 06,07,08,09 ) Given the above arrays, I want the script to return/echo the following in a loop; win 12,01,02,03,04 pre 04,05,06,07 msn 06,07,08,09 But I can't get it to do as such. I've tried; (2 Replies)
Discussion started by: Muhammad Rahiz
2 Replies

8. Shell Programming and Scripting

bad substitution error in ksh

Hello, In bash I can use the following: TMP=12345 MID=${TMP:1:1} the expected result is: 2 but when using KSH I'm getting a ''bad substitution" error. What is the correct syntaxin ksh? Thanks (2 Replies)
Discussion started by: LiorAmitai
2 Replies

9. Shell Programming and Scripting

bad substitution error in ksh

hi, i created a shell script having the following content: #! /usr/bin/ksh FROM="myemail@domain.com" MAILTO="someemail@domain" SUBJECT="TEST" BODY="/export/home/adshocker/body.txt" ATTACH="/export/home/adshocker/attach.prog" echo $ATTACH ATTACH_NAME="${ATTACH##*/}" echo $ATTACH_NAME... (5 Replies)
Discussion started by: adshocker
5 Replies

10. Shell Programming and Scripting

bad substitution Error while renaming Extension

Hi All, We are in the process of Migrating from AIX 4 to Solaris 10 and getting a Few Errors. I have been programming in shell but could never establish muself as an expert, hence please need you help. I am Getting Bad Substitution error in my script, I have isolated the issue and its... (6 Replies)
Discussion started by: paragkhanore
6 Replies
Login or Register to Ask a Question