Shell script can be run only with sh command

Solution 1:

I can't tell for sure but it looks like you have a typo on the first line where there's a stray * over to the right.

#!/bin/sh                                                                                     *

^^^ Scroll over to the right to see it.

Example

$ cat script.bash
#!/bin/sh                    *
echo hi

Run directly:

$ ./script.bash
/bin/sh: *: No such file or directory

Run via sh:

$ sh script.bash
hi

General advice

It's generally advisable to use the exact shell you're expecting with your shebang. If you suspect that you're running into problems where you suspect dash or some other shell is being used, you can always make your shebang explicit by changing it from #!/bin/sh as your shebang #!/bin/bash.

The above is based on your comment below:

but now I think it is a problem with the script itself or with dash.

Solution 2:

The error can't open * is preventing the script from running correctly when run by run-parts.  Why is this error occurring?

When you execute a file1 that is not a binary executable, but is a text file with a shebang (the first line of the file begins with #!), the kernel (with no help from any shell) builds a command that is the shebang line (the part after the #!) followed by the original, user-level command line.  For example, if doc-script begins

#!/bin/sh -x

and it is invoked

./doc-script bashful dopey

then the kernel builds and executes the following command:

/bin/sh -x ./doc-script bashful dopey

which causes the shell to read and interpret the doc-script script, with the xtrace (-x) option set, and with $1 = bashful and $2 = dopey.  (Naturally, $0 is ./doc-script.)  If the original user-supplied command is

./doc-script b* ??p* [ghs]*

then the shell that handles that command (for simplicity, let’s assume that that is the interactive shell running on the user’s terminal) might expand this to

./doc-script bashful dopey grumpy happy sleepy sneezy

and so the kernel builds and executes the following command:

/bin/sh -x ./doc-script bashful dopey grumpy happy sleepy sneezy

But remember: the processing of the shebang line is done by the kernel, not by the shell.  Therefore, if the shebang is

#!/bin/sh *

then the final, constructed command is

/bin/sh * ./doc-script bashful dopey grumpy happy sleepy sneezy

because glob expansion doesn’t happen here.  As far as the shell is concerned, this looks like the user typed

/bin/sh '*' ./doc-script bashful dopey grumpy happy sleepy sneezy

and, because * doesn’t begin with -, the shell interprets it as a filename, so it tries to run a script called * with $0 = *$1 = ./doc-script, $2 = bashful, $3 = dopey, etc.  And, since there is no script called *, this fails.
__________
1 I am assuming that the process has the permissions necessary to execute the file.