Is there a command for running a script according to its shebang line?

6,075

Solution 1

Yep. It is called perl. Some examples, with the corresponding interpreter in the shebang line of the file (the actual file extension doesn't matter):

perl foo.bash    # works
perl foo.lua     # works
perl foo.clisp   # works
perl foo.csh     # works
perl foo.php     # works
perl foo.gnuplot # works (no arguments)
perl foo.pl      # works (obviously)
perl foo.py      # works
perl foo.sh      # works
perl foo.tcl     # works
perl foo.rb      # works
perl foo.nodejs  # works
perl foo.r       # works
perl foo.oct     # works
perl foo.csharp  # works (no arguments)

This is mentioned in Perl's documentation:

If the #! line does not contain the word "perl" nor the word "indir", the program named after the #! is executed instead of the Perl interpreter. This is slightly bizarre, but it helps people on machines that don't do #! , because they can tell a program that their SHELL is /usr/bin/perl, and Perl will then dispatch the program to the correct interpreter for them.

Solution 2

Scripts do not necessarily have a shebang

If the script was run from the interpreter, You cannot be sure it has the shebang at all. Scripts, run from the interpreter do not need the shebang, if you call the interpreter to run the code.

The answer is therefore no, there is no command that will find out for sure what is the language (interpreter) to run the script with. You can however always look inside the script and see if it has the shebang to find out.

The rules in short:

  1. When you run the script, calling the interpreter always overrules possible shebangs, executable or not, shebang or not.
  2. If not executable and run from the interpreter, the script needs no shebang.
  3. If the script is run without calling the interpreter first, it needs (and uses) the shebang to find out what interpreter to call, and it needs to be executable to have the "permission" to call the interpreter from its shebang.

If the script has no shebang however, there is no (direct*) information inside the script to tell what interpreter to use.

Having said that

You could of course always write a wrapper script to try to find out if the script has the shebang and read the interpreter from that, subsequently run it from the found interpreter.

An example

#!/usr/bin/env python3
import subprocess
import sys

args = sys.argv[1:]; script = args[0]

try:
    lang = open(script).readlines()[0].replace("#!", "").strip().split()[-1]
    cmd = [lang, script]+args[1:]
    subprocess.call(cmd)
except (PermissionError, FileNotFoundError, IndexError):
    print("No valid shebang found")
  • Save it as tryrun in $PATH (e.g. ~/bin, make the directory if it does not exist, log out and back in), make it executable. Then running:

    tryrun /path/to/nonexecutablescript
    

    calls (tested) the correct interpreter on my non-executable python and bash scripts.

Explanation

  • The script simply reads the first line of the script, removes the #! and uses the rest to call the interpreter.
  • If it fails to call a valid interpreter, it will raise either a PermissionError or a FileNotFoundError.

Note

The extension (.sh, .py etc) plays no role whatsoever in determining the appropriate interpreter on Linux.


(*It is of course possible to develop a "smart" guess- algorithm to determine the syntax from the code.)

Solution 3

You can achieve this with a script like this:

#!/bin/bash

copy=/tmp/runner.$$
cp $1 ${copy}
chmod u+x ${copy}
${copy}
rm ${copy}

Thus:

$ echo "echo hello" > myscript
$ ./myscript
bash: ./myscript: Permission denied
$ ./runscript myscript 
hello

I recommend against doing this. Permissions are there for a reason. This is a program for subverting permissions.

Note that shebang handling is a kernel function (in the Linux source code - fs/binfmt_script.c). Fundamentally the process invoking a script directly doesn't know about the #! -- the kernel uses it to work out that it needs to launch an interpreter.

Share:
6,075

Related videos on Youtube

Aivar
Author by

Aivar

Teaching assistant in University of Tartu, Department of CS

Updated on September 18, 2022

Comments

  • Aivar
    Aivar over 1 year

    If I want to execute a bash script which doesn't have its execution permission set, I can do:

    bash script.sh
    

    What should I use instead of bash if the script isn't executable and I don't know the correct interpreter? Is there a command that looks up the interpreter from shebang line and executes the script with it?

  • Aivar
    Aivar over 7 years
    OK, so it means although Linux has shebang extraction implemented somewhere (so that it can select correct interpreter for executable srcipts), it's not provided as a standalone standard command.
  • Jacob Vlijm
    Jacob Vlijm over 7 years
    @Aivar extracting the shebang isn' t the issue, but running code without it is perfectly possible.
  • Jacob Vlijm
    Jacob Vlijm over 7 years
    @Aivar Ah, I see what you mean. If the script is executable and run without language in the command, the script calls the interpreter, not the other way around.
  • slim
    slim over 7 years
    Well, that's just dirty.
  • Paŭlo Ebermann
    Paŭlo Ebermann over 7 years
    @JacobVlijm I wouldn't say "the script calls the interpreter", more like "the Linux kernel takes the shebang line for figuring out which interpreter to call when executing the script".
  • Jacob Vlijm
    Jacob Vlijm over 7 years
    @PaŭloEbermann Thanks! True of course. The kernel takes care of the whole procedure either way, but figuratively, and I think better for understanding, is to say the script is "allowed" to take care of what interpreter to call (and actually do it). Not sure about the wording, but I' d like to describe it as if the initiative is on the script, while the kernel actually does the job.
  • Jacob Vlijm
    Jacob Vlijm over 7 years
    @OleTange surprising. Obviously Perl is smart enough to read the shebang and not pushes itself upfront without looking. Without the shebang, also Perl is helpless however.
  • simbabque
    simbabque over 7 years
    @JacobVlijm that's true, but then again the question specifically says there is always a shebang
  • Pysis
    Pysis over 7 years
    Does the file extension .js also work?
  • Pysis
    Pysis over 7 years
    Also, what machines don't do #!. I've seem a few to more now, and have not experienced this problem.
  • boatcoder
    boatcoder over 7 years
    I had always assumed that was a function of the shell, NOT the kernel. Learned something new today.
  • kaitosenpai
    kaitosenpai over 7 years
    @Pysis the file extension does not matter. It is only the #! in the first line that matters. I just included it as a short way of explaining all the script languages that I tested.
  • Pysis
    Pysis over 7 years
    Ok, it's just your example seemed to highlight a lot of file extensions, rather than showcase several shebang lines from files, and I guess I assumed that's how its prediction worked.
  • kaitosenpai
    kaitosenpai over 7 years
    @Pysis No modern OSes. But I seem to remember the early versions of Ultrix had the problem with #!.
  • slim
    slim over 7 years
    @boatcoder -- Since you're interested, added a link to where the Linux source code handles it.
  • slim
    slim over 7 years
    I like how man perlrun sheepishly admits that it's "slightly bizarre" :). I think this should be treated as a curiosity aimed at non-UNIX environments and very very old versions of UNIX.
  • Kevin
    Kevin over 7 years
    @OleTange: I'm pretty sure Windows doesn't do #! out of the box. But YMMV on whether Windows is a "modern" OS.