26 Stimmen

Wie erhält man die Umgebung von einem Unterprozess?

Ich möchte einen Prozess über ein Python-Programm aufrufen, aber dieser Prozess benötigt einige spezifische Umgebungsvariablen, die von einem anderen Prozess gesetzt werden. Wie kann ich die Umgebungsvariablen des ersten Prozesses erhalten, um sie an den zweiten zu übergeben?

So sieht das Programm aus:

import subprocess

subprocess.call(['proc1']) # this set env. variables for proc2
subprocess.call(['proc2']) # this must have env. variables set by proc1 to work

aber die beiden Prozesse haben nicht dieselbe Umgebung. Beachten Sie, dass diese Programme nicht von mir sind (das erste ist eine große und hässliche .bat-Datei und das zweite eine proprietäre Software), so dass ich sie nicht ändern kann (ok, ich kann alles, was ich brauche, aus der .bat-Datei extrahieren, aber es ist sehr umständlich).

N.B.: Ich benutze Windows, aber ich bevorzuge eine plattformübergreifende Lösung (aber mein Problem würde nicht auf einem Unix-ähnlichen System auftreten ...)

31voto

Jason R. Coombs Punkte 38667

Hier ist ein Beispiel, wie Sie Umgebungsvariablen aus einer Batch- oder Cmd-Datei extrahieren können, ohne ein Wrapper-Skript zu erstellen. Viel Spaß!

from __future__ import print_function
import sys
import subprocess
import itertools

def validate_pair(ob):
    try:
        if not (len(ob) == 2):
            print("Unexpected result:", ob, file=sys.stderr)
            raise ValueError
    except:
        return False
    return True

def consume(iter):
    try:
        while True: next(iter)
    except StopIteration:
        pass

def get_environment_from_batch_command(env_cmd, initial=None):
    """
    Take a command (either a single command or list of arguments)
    and return the environment created after running that command.
    Note that if the command must be a batch file or .cmd file, or the
    changes to the environment will not be captured.

    If initial is supplied, it is used as the initial environment passed
    to the child process.
    """
    if not isinstance(env_cmd, (list, tuple)):
        env_cmd = [env_cmd]
    # construct the command that will alter the environment
    env_cmd = subprocess.list2cmdline(env_cmd)
    # create a tag so we can tell in the output when the proc is done
    tag = 'Done running command'
    # construct a cmd.exe command to do accomplish this
    cmd = 'cmd.exe /s /c "{env_cmd} && echo "{tag}" && set"'.format(**vars())
    # launch the process
    proc = subprocess.Popen(cmd, stdout=subprocess.PIPE, env=initial)
    # parse the output sent to stdout
    lines = proc.stdout
    # consume whatever output occurs until the tag is reached
    consume(itertools.takewhile(lambda l: tag not in l, lines))
    # define a way to handle each KEY=VALUE line
    handle_line = lambda l: l.rstrip().split('=',1)
    # parse key/values into pairs
    pairs = map(handle_line, lines)
    # make sure the pairs are valid
    valid_pairs = filter(validate_pair, pairs)
    # construct a dictionary of the pairs
    result = dict(valid_pairs)
    # let the process finish
    proc.communicate()
    return result

Um Ihre Frage zu beantworten, würden Sie also eine .py-Datei erstellen, die Folgendes tut:

env = get_environment_from_batch_command('proc1')
subprocess.Popen('proc2', env=env)

5voto

Martin v. Löwis Punkte 120025

Wie Sie sagen, teilen sich Prozesse nicht die Umgebung - was Sie wörtlich fordern, ist also nicht möglich, nicht nur in Python, sondern mit jeder Programmiersprache.

Was Sie puede ist es, die Umgebungsvariablen in einer Datei oder in einer Pipe zu speichern und entweder

  • der übergeordnete Prozess soll sie lesen und an proc2 weitergeben, bevor proc2 erstellt wird, oder
  • proc2 soll sie lesen und lokal setzen

Letzteres würde die Mitarbeit von proc2 erfordern; ersteres setzt voraus, dass die Variablen bekannt werden, bevor proc2 gestartet wird.

2voto

Glenn Maynard Punkte 53282

Da Sie offensichtlich unter Windows arbeiten, benötigen Sie eine Windows-Antwort.

Erstellen Sie eine Wrapper-Batch-Datei, z. B. "run_program.bat", und führen Sie beide Programme aus:

@echo off
call proc1.bat
proc2

Das Skript wird ausgeführt und setzt seine Umgebungsvariablen. Beide Skripte laufen im selben Interpreter (cmd.exe-Instanz), so dass die Variablen prog1.bat wird bei der Ausführung von prog2 gesetzt werden.

Nicht besonders schön, aber es wird funktionieren.

(Unix-Benutzer können dasselbe mit einem Bash-Skript tun: "source file.sh").

2voto

huagang Punkte 111

Sie können verwenden Process en psutil um die Umgebungsvariablen für diesen Prozess zu erhalten.

Wenn Sie es selbst implementieren wollen, können Sie sich auf die interne Implementierung von psutil . Es passt sich an einige Betriebssysteme an.

Derzeit werden die folgenden Betriebssysteme unterstützt:

  • AIX
  • FreeBSD, OpenBSD, NetBSD
  • Linux
  • macOS
  • Sonne Solaris
  • Windows

Beispiel: Auf der Linux-Plattform finden Sie eine Umgebungsvariable pid 7877 in der Datei /proc/7877/environ öffnen Sie einfach mit rt Modus, um sie zu lesen.

Am besten ist es natürlich, wenn Sie das tun:

import os
from typing import Dict
from psutil import Process

process = Process(pid=os.getpid())
process_env: Dict = process.environ()

print(process_env)

Weitere Plattformimplementierungen finden Sie unter Quellcode

Ich hoffe, ich kann Ihnen helfen.

1voto

thomas Punkte 1795

Das Python-Standardmodul Multiprocessing verfügen über ein Warteschlangensystem, das es Ihnen ermöglicht, belegbare Objekte an Prozesse weiterzugeben. Außerdem können Prozesse Nachrichten (ein gepickeltes Objekt) mit os.pipe austauschen. Denken Sie daran, dass Ressourcen (z.B. Datenbankverbindungen) und Handles (z.B. Datei-Handles) nicht gepickelt werden können.

Dieser Link könnte für Sie interessant sein: Kommunikation zwischen Prozessen mit Multiprocessing

Auch die PyMOTw über Multiprocessing ist erwähnenswert: Multiprocessing Grundlagen

Entschuldigung für meine Rechtschreibung

CodeJaeger.com

CodeJaeger ist eine Gemeinschaft für Programmierer, die täglich Hilfe erhalten..
Wir haben viele Inhalte, und Sie können auch Ihre eigenen Fragen stellen oder die Fragen anderer Leute lösen.

Powered by:

X