Encode a String to be used as shell argument

2019-06-22 09:46发布

I need to use a name of file as an argument in linux shell command. The problem is, java gives me that name as it is, saving all that spaces and other characters, and thus shell complains. Is there a method to escape all those problematic characters before passing the string to shell?

4条回答
ら.Afraid
2楼-- · 2019-06-22 10:09

Adding quotes single or double around the file name is often sufficient, depending on the characters you have in the name it may not.

查看更多
贪生不怕死
3楼-- · 2019-06-22 10:14

Had the same problem, single quotes wasn't sufficient (as already pointed out by Robert)

Solution:

import com.google.common.escape.Escaper;
import com.google.common.escape.Escapers;

public class YourFancyClass {
    public static final Escaper SHELL_ESCAPE;
    static {
        final Escapers.Builder builder = Escapers.builder();
        builder.addEscape('\'', "'\"'\"'");
        SHELL_ESCAPE = builder.build();
    }
}

Why such an awfully complex replacement? That's why.

Use case:

System.out.format("ln -f '%s' '%s'%n", 
    SHELL_ESCAPE.escape(anyOrig.toString()),
    SHELL_ESCAPE.escape(duplicate.toString()));

Works as intended:

ln -f '/home/user/Musik/mix-2012-13/aesthesys~ I Am Free, That Is Why I'"'"'m Lost..mp3' '/home/user/Musik/youtube converted/aesthesys~ I Am Free, That Is Why I'"'"'m Lost..mp3'
查看更多
对你真心纯属浪费
4楼-- · 2019-06-22 10:14

How about using the Exec module from Apache Commons? It includes a commandline builder. Also be aware that if the filename is retrieved from user input, you should be very careful with executing commands with the user input as a program argument. Escaping improperly may lead to execution of additional commands (unless the commons module is used I guess).

查看更多
forever°为你锁心
5楼-- · 2019-06-22 10:19

You should be able to put single quotes around the argument and avoid escaping it altogether. Will that work for you?

Old: myapp -f /bad/path/to/file

New: myapp -f '/good/path/to/file'

查看更多
登录 后发表回答