通过 Java ProcessBuilder 激活 virtualenv

Activating virtualenv via Java ProcessBuilder

尝试通过以下代码以编程方式激活 Python 的 virtualenv 时获得以下信息:

java.io.IOException: Cannot run program "." (in directory "/Users/simeon.../..../reporting"): error=13, Permission denied
    at java.lang.ProcessBuilder.start(ProcessBuilder.java:1048)
    at VirtualEnvCreateCmdTest.runCommandInDirectory(VirtualEnvCreateCmdTest.java:30)
    at VirtualEnvCreateCmdTest.createVirtEnv(VirtualEnvCreateCmdTest.java:61)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
    at ......
Caused by: java.io.IOException: error=13, Permission denied
    at java.lang.UNIXProcess.forkAndExec(Native Method)
    at java.lang.UNIXProcess.<init>(UNIXProcess.java:247)
    at java.lang.ProcessImpl.start(ProcessImpl.java:134)
    at java.lang.ProcessBuilder.start(ProcessBuilder.java:1029)
    ... 25 more

代码:

public class VirtualEnvCreateCmdTest {

    private final static Logger LOG = LoggerFactory.getLogger(VirtualEnvCreateCmdTest.class);

    private void runCommandInDirectory(String path,String ... command) throws Throwable
    {

        LOG.info("Running command '"+String.join(" ",command)+"' in path '"+path+"'");


        ProcessBuilder builder = new ProcessBuilder(command)
                .directory(new File(path))
                .inheritIO();

        Process pr = builder.start();

        final String failureMsg = format("Failed to run '%s' in path '%s'.  Got exit code: %d", join(" ",command), path, pr.exitValue());
        LOG.info("prepared message {}: ", failureMsg);

        if(!pr.waitFor(120, TimeUnit.SECONDS))
        {
            throw new Exception(failureMsg);

        }


        int output = IOUtils.copy(pr.getInputStream(), System.out);

        int exitCode=pr.exitValue();

        if(exitCode!=0)
            throw new Exception(failureMsg);

    }

    @Test
    public void createVirtEnv()  throws Throwable {

        String path = "/Users/simeon/.../reporting";
        String [] commands = new String[]{".", "activate"};
        //String [] commands = new String [] {"/bin/bash", "-c", ". /Users/simeon/..../venv2.7/bin/activate"};
        runCommandInDirectory(path, commands);

    }

更改文件的权限似乎不起作用:

chmod u+x ./bin/activate

通过 /bin/bash 执行此操作仍然无效,但出现不同的错误。

同时,以下命令在命令行上运行良好:

. /Users/simeon/.../venv2.7/bin/activate

关于如何从 Java 中调用 python virtualenv activate 命令的任何示例?

=== 什么有效:=====

以下最终对我有用:

    private void runDjangoMigrate() throws Throwable {

        final String REPORTING_PROJECT_LOCATION = "/Users/simeon.../.../reporting";
        final String UNIX_SHELL_LOCATION = "/bin/bash";
        final String PYTHON_VIRTUALENV_ACTIVATOR_COMMAND =". /Users/simeon.../.../venv2.7/bin/activate;";
        final String PYTHON_VIRTUALENV_ACTIVATOR_COMMAND =". " + PYTHON_VIRTUALENV_ACTIVATE_SCRIPT_LOCATION + ";"; 
        final String DJANGO_MANAGE_MODULE = " manage.py";

        final String[] DJANGO_MIGRATE_COMMAND = new String[] { UNIX_SHELL_LOCATION, "-c", PYTHON_VIRTUALENV_ACTIVATOR_COMMAND
                + PYTHON_INTERPRETER + DJANGO_MANAGE_MODULE + " migrate --noinput --fake-initial" };

        runCommandInDirectory(REPORTING_PROJECT_LOCATION, DJANGO_MIGRATE_COMMAND);


}


    private void runCommandInDirectory(String path, String... command) throws Throwable {

        LOG.info(format("Running '%s' command in path '%s'", join(" ",command),path));

        ProcessBuilder builder = new ProcessBuilder(command).directory(new File(path)).inheritIO();

        Process pr = null;

        pr = builder.start();

        IOUtils.copy(pr.getInputStream(), System.out);  

        boolean terminated = pr.waitFor(SPAWNED_PYTHON_PROCESS_TTL_SEC, SECONDS);

        int exitCode = -1;

        if (terminated) {

            exitCode = pr.exitValue();
        }

        if (exitCode != 0) {

            final String failureMsg = format("Failed to run '%s' in path '%s'.  Got exit code: %d", join(" ", command),
                    path, pr.exitValue());

            throw new Exception(failureMsg);
        }

    }

并产生以下预期输出:

[java] System check identified some issues:
 [java]
 [java] WARNINGS:
 [java] ?: (urls.W005) URL namespace 'admin' isn't unique. You may not be able to reverse all URLs in this namespace
 [java] Building permissions...
 [java] Operations to perform:
 [java]   Apply all migrations: admin, auth, contenttypes, sessions
 [java] Running migrations:
 [java]   No migrations to apply.
 [java] System check identified some issues:
 [java]
 [java] WARNINGS:
 [java] ?: (urls.W005) URL namespace 'admin' isn't unique. You may not be able to reverse all URLs in this namespace
 [java] Building permissions...
 [java] User exists, exiting normally due to --preserve
 [java] System check identified some issues:
 [java]

您可以在 bash(可能还有 zsh)和 Python 中激活 Python 虚拟环境,但不能在 Java 或任何其他环境中激活。要了解的主要事情是,虚拟环境激活不会做一些系统魔法 — 激活脚本只是更改 当前环境 ,并且 Python 虚拟环境已准备好更改 shell 或 Python 但没有别的。

就 Java 而言,这意味着当您调用 runCommandInDirectory() 时,它 运行 是一个新的 shell,新的 shell 会短暂激活一个虚拟环境,但随后 shell 退出并且 "activates" 虚拟环境的所有更改都消失了。

这反过来意味着如果您需要在虚拟环境中 运行 一些 shell 或 Python 命令,您必须在每次 runCommandInDirectory() 调用中激活环境:

String [] commands1 = new String [] {"/bin/bash", "-c", ". /Users/simeon/..../venv2.7/bin/activate; script1.sh"};
runCommandInDirectory(path, commands)

String [] commands2 = new String [] {"/bin/bash", "-c", ". /Users/simeon/..../venv2.7/bin/activate; script2.sh"};
runCommandInDirectory(path, commands)

对于 Python 脚本,它更简单一些,因为您可以从环境中 运行 python 并且它会自动激活环境:

String [] commands = new String [] {"/Users/simeon/..../venv2.7/bin/python", "script.py"};
runCommandInDirectory(path, commands)