how to run linux code in python
3
1
Entering edit mode
3.6 years ago
szp770 ▴ 10

I want to run this shell code using python script:

cmd8="""join -t $'\t' -1 5 -2 5 <(sort -t $'\t' -k5,5 LP.bed1) <(sort -t $'\t' -k5,5 RE.bed1) >LP_RE.bed1"""
os.system(cmd8)

however, it reports this error: /bin/sh: 1: Syntax error: "(" unexpected.

shell python sh bash linux • 3.6k views
ADD COMMENT
2
Entering edit mode
3.6 years ago

The example is complex enough that I won't debug it for you, but I write a lot of Python scripts that make system calls, and have a few pointers that can make your job easier.

  1. Have a look at the Python subprocess module. This gives you a lot more flexibility in running external processes. I have seen cases in which, for example, I can't get the external call to run using subprocess.run, but it will work with subprocess.Popen, or with os.system. The point is it helps to have a bag of tricks that you can draw upon.

  2. The < and > shell operators respectively redirect input from a file to a command, or output from a command to a file. I think what you want here is the pipe (|) operator, which pipes output from one command to be used as input for the next command.

  3. It looks like what you're trying to do is to take the output from a series of sort commands as input to the join command. Instead of parentheses, I think the shell operator you're looking for is the left quote. In shell, enclosing a command in left quotes returns the output of that command. For example, if you had a text file called acc.txt containing a list of Accession numbers

    RESULT=`cat acc.txt `

would assign to RESULT the output of cat acc.txt, that is the contents of the file would be stored in the environment variable RESULT.

(Note: On North American English keyboards, left quote (`) is on the same key as the tilda (~). This is a different character than the right quote ('), which would be on the same key as the double quote (").)

  1. If I was doing this, I might approach it in several possible ways. One is to take the series of commands and create a bash script that ran them. That way you can debug the script first and make sure it is working without the added complication of trying to get Python to call it. Also, that makes for a much simpler command string in your Python script.

The other way is to run each step as separate Python calls, saving intermediate results in temporary files and reading them back in for the next step. This sound like a lot of extra work, but is a lot easier to understand and debug.

Often a better approach is to write the process in Python. Python has a lot of sort methods, for example, and your join operation would also be easy to implement. That would save all the reading and writing to files.

Whatever you end up doing, break complex problems down into easy ones, and get each one to work before going on to the next. For example, spend the time to just get the first sort to work, and don't worry about the other steps. You'll find that this strategy is FASTER than trying to get a big complex thing to work all at once.

By the way, you get points from creating a command string as a separate step from running the command. Again, splitting up the steps makes the code more readable, and easier to debug.

ADD COMMENT
1
Entering edit mode

Also see my tutorial Basic Shell Scripting.

ADD REPLY
1
Entering edit mode
3.6 years ago

The reason os.system(...) doesn't work out of the box is that you are using a process substitution that is a function of the bash shell, but the default shell of os.system(...) is likely /bin/sh (or /bin/zsh on OS X) which does not support that feature.

Python 3.8.3 (default, May 19 2020, 13:54:14) 
[Clang 10.0.0 ] :: Anaconda, Inc. on darwin
Type "help", "copyright", "credits" or "license" for more information.
>>> cmd="""cat <(sort -k1,1 foo.txt)"""
>>> cmd
'cat <(sort -k1,1 foo.wig)'
>>> import os
>>> os.system(cmd)
sh: -c: line 0: syntax error near unexpected token `('
sh: -c: line 0: `cat <(sort -k1,1 foo.txt)'
512

The 512 at the end is the exit code. You can bitshift this value to the right by 8 and get a "true" error code of 2.

This is just a generic error code, usually indicating a missing file or file descriptor.

To resolve this, you can manually specify /bin/bash -c and pass in the command you want to run:

>>> cmd='CMD="cat <(sort -k1,1 foo.txt)"; /bin/bash -c "$CMD"'
>>> cmd
'CMD="cat <(sort -k1,1 foo.txt)"; /bin/bash -c "$CMD"'
>>> os.system(cmd)
1   1.661
2   1.649
3   1.645
4   1.644
5   1.6434
6   1.64233333333333
7   1.64128571428571
8   1.6405
9   1.63988888888889
variableStep  chrom=Chrom_1
0

The 0 at the end is the exit code, indicating no errors.

Another way to solve this is to use the Python subprocess library and specify shell=True to use your default shell to launch processes.

ADD COMMENT
0
Entering edit mode

this really helps! Thanks so much!

ADD REPLY
0
Entering edit mode
3.6 years ago
curious ▴ 810

I python and bash often and have spent a lot of time on this problem before.

I have decided personally that it is usually more efficient t run what is in <() in a separate python command.

ADD COMMENT

Login before adding your answer.

Traffic: 2529 users visited in the last hour
Help About
FAQ
Access RSS
API
Stats

Use of this site constitutes acceptance of our User Agreement and Privacy Policy.

Powered by the version 2.3.6