summaryrefslogtreecommitdiffstats
path: root/pym
diff options
context:
space:
mode:
authorZac Medico <zmedico@gentoo.org>2008-07-16 07:09:51 +0000
committerZac Medico <zmedico@gentoo.org>2008-07-16 07:09:51 +0000
commit874707f61f313a7eb0282475dfe6775cc99d3813 (patch)
tree47b6b63c0d21a8a2f3b6aff61241613ab59de46e /pym
parentf40d16e3dc580979b1e18bf4cef7221e56657f85 (diff)
downloadportage-874707f61f313a7eb0282475dfe6775cc99d3813.tar.gz
portage-874707f61f313a7eb0282475dfe6775cc99d3813.tar.bz2
portage-874707f61f313a7eb0282475dfe6775cc99d3813.zip
Add async support for calling functions in misc-functions.sh:
* Create a new MiscFunctionsProcess class derived from SpawnProcess. * Add "background" support to the SpawnProcess class and split out a _spawn() template function so MiscFunctionsProcess can override it. * Rename the existing EbuildPhase class to EbuildProcess. * Create a new EbuildPhase class derived from CompositeTask. This class spawns the EbuildProcess and MiscFunctionsProcess classes as sub-tasks. svn path=/main/trunk/; revision=11072
Diffstat (limited to 'pym')
-rw-r--r--pym/_emerge/__init__.py145
-rw-r--r--pym/portage/__init__.py30
2 files changed, 121 insertions, 54 deletions
diff --git a/pym/_emerge/__init__.py b/pym/_emerge/__init__.py
index 976ddcbef..bfdea3aa5 100644
--- a/pym/_emerge/__init__.py
+++ b/pym/_emerge/__init__.py
@@ -2016,7 +2016,7 @@ class SpawnProcess(SubProcess):
__slots__ = ("args",) + \
_spawn_kwarg_names
- _file_names = ("process", "out")
+ _file_names = ("log", "process", "stdout")
_files_dict = slot_dict_class(_file_names, prefix="")
_bufsize = 4096
@@ -2047,6 +2047,7 @@ class SpawnProcess(SubProcess):
fcntl.fcntl(master_fd, fcntl.F_SETFL,
fcntl.fcntl(master_fd, fcntl.F_GETFL) | os.O_NONBLOCK)
+ files.process = os.fdopen(master_fd, 'r')
if logfile is not None:
fd_pipes_orig = fd_pipes.copy()
@@ -2054,11 +2055,14 @@ class SpawnProcess(SubProcess):
fd_pipes[1] = slave_fd
fd_pipes[2] = slave_fd
- files.out = open(logfile, "a")
+ files.log = open(logfile, "a")
portage.util.apply_secpass_permissions(logfile,
uid=portage.portage_uid, gid=portage.portage_gid,
mode=0660)
+ if not self.background:
+ files.stdout = os.fdopen(os.dup(fd_pipes_orig[1]), 'w')
+
output_handler = self._output_handler
else:
@@ -2081,13 +2085,15 @@ class SpawnProcess(SubProcess):
kwargs["returnpid"] = True
kwargs.pop("logfile", None)
- retval = portage.process.spawn(self.args, **kwargs)
+ retval = self._spawn(self.args, **kwargs)
os.close(slave_fd)
if isinstance(retval, int):
# spawn failed
os.close(master_fd)
+ for f in self.files.values():
+ f.close()
self.returncode = retval
self.wait()
return
@@ -2095,12 +2101,13 @@ class SpawnProcess(SubProcess):
self.pid = retval[0]
portage.process.spawned_pids.remove(self.pid)
-
- files.process = os.fdopen(master_fd, 'r')
self._reg_id = self.scheduler.register(files.process.fileno(),
PollConstants.POLLIN, output_handler)
self._registered = True
+ def _spawn(self, args, **kwargs):
+ return portage.process.spawn(args, **kwargs)
+
def _output_handler(self, fd, event):
files = self._files
buf = array.array('B')
@@ -2109,8 +2116,11 @@ class SpawnProcess(SubProcess):
except EOFError:
pass
if buf:
- buf.tofile(files.out)
- files.out.flush()
+ if not self.background:
+ buf.tofile(files.stdout)
+ files.stdout.flush()
+ buf.tofile(files.log)
+ files.log.flush()
else:
self._unregister()
self.wait()
@@ -2135,6 +2145,38 @@ class SpawnProcess(SubProcess):
self.wait()
return self._registered
+class MiscFunctionsProcess(SpawnProcess):
+ """
+ Spawns misc-functions.sh with an existing ebuild environment.
+ """
+
+ __slots__ = ("commands", "phase", "pkg", "settings")
+
+ def _start(self):
+ settings = self.settings
+ portage_bin_path = settings["PORTAGE_BIN_PATH"]
+ misc_sh_binary = os.path.join(portage_bin_path,
+ os.path.basename(portage.const.MISC_SH_BINARY))
+
+ self.args = [portage._shell_quote(misc_sh_binary)] + self.commands
+ self.logfile = settings.get("PORTAGE_LOG_FILE")
+
+ portage._doebuild_exit_status_unlink(
+ settings.get("EBUILD_EXIT_STATUS_FILE"))
+
+ SpawnProcess._start(self)
+
+ def _spawn(self, args, **kwargs):
+ settings = self.settings
+ debug = settings.get("PORTAGE_DEBUG") == "1"
+ return portage.spawn(" ".join(args), settings,
+ debug=debug, **kwargs)
+
+ def _set_returncode(self, wait_retval):
+ SpawnProcess._set_returncode(self, wait_retval)
+ self.returncode = portage._doebuild_exit_status_check_and_log(
+ self.settings, self.phase, self.returncode)
+
class EbuildFetcher(SpawnProcess):
__slots__ = ("fetchonly", "pkg",)
@@ -2509,17 +2551,10 @@ class EbuildExecuter(CompositeTask):
# This initializes PORTAGE_LOG_FILE.
portage.prepare_build_dirs(pkg.root, settings, cleanup)
- fd_pipes = {
- 0 : sys.stdin.fileno(),
- 1 : sys.stdout.fileno(),
- 2 : sys.stderr.fileno(),
- }
-
ebuild_phases = TaskSequence(scheduler=scheduler)
for phase in self._phases:
ebuild_phases.add(EbuildPhase(background=self.background,
- fd_pipes=fd_pipes,
pkg=pkg, phase=phase, scheduler=scheduler,
settings=settings, tree=tree))
@@ -2613,7 +2648,7 @@ class EbuildMetadataPhase(SubProcess):
return self._registered
-class EbuildPhase(SubProcess):
+class EbuildProcess(SubProcess):
__slots__ = ("fd_pipes", "phase", "pkg",
"settings", "tree")
@@ -2758,41 +2793,57 @@ class EbuildPhase(SubProcess):
settings, self.phase, self.returncode)
portage._post_phase_userpriv_perms(settings)
+
+class EbuildPhase(CompositeTask):
+
+ __slots__ = ("background", "pkg", "phase",
+ "scheduler", "settings", "tree")
+
+ _post_phase_cmds = portage._post_phase_cmds
+
+ def _start(self):
+
+ ebuild_process = EbuildProcess(background=self.background,
+ pkg=self.pkg, phase=self.phase, scheduler=self.scheduler,
+ settings=self.settings, tree=self.tree)
+
+ self._start_task(ebuild_process, self._ebuild_exit)
+
+ def _ebuild_exit(self, ebuild_process):
+
if self.phase == "install":
- portage._check_build_log(settings)
- if self.returncode == os.EX_OK:
- self.returncode = portage._post_src_install_checks(settings)
+ portage._check_build_log(self.settings)
- elif self.phase == "preinst":
- if self.returncode == os.EX_OK:
- portage._doebuild_exit_status_unlink(
- settings.get("EBUILD_EXIT_STATUS_FILE"))
- phase_retval = portage.spawn(
- " ".join(portage._post_pkg_preinst_cmd(settings)),
- settings, debug=debug, free=1, logfile=log_path)
- phase_retval = portage._doebuild_exit_status_check_and_log(
- settings, self.phase, phase_retval)
- if phase_retval != os.EX_OK:
- writemsg("!!! post preinst failed; exiting.\n",
- noiselevel=-1)
- self.returncode = phase_retval
+ if self._default_exit(ebuild_process) != os.EX_OK:
+ self.wait()
+ return
- elif self.phase == "postinst":
+ settings = self.settings
- if self.returncode == os.EX_OK:
- portage._doebuild_exit_status_unlink(
- settings.get("EBUILD_EXIT_STATUS_FILE"))
- phase_retval = portage.spawn(
- " ".join(portage._post_pkg_postinst_cmd(settings)),
- settings, debug=debug, free=1, logfile=log_path)
- phase_retval = portage._doebuild_exit_status_check_and_log(
- settings, self.phase, phase_retval)
- if phase_retval != os.EX_OK:
- writemsg("!!! post postinst failed; exiting.\n",
- noiselevel=-1)
- self.returncode = phase_retval
+ if self.phase == "install":
+ portage._post_src_install_uid_fix(settings)
+
+ post_phase_cmds = self._post_phase_cmds.get(self.phase)
+ if post_phase_cmds is not None:
+ post_phase = MiscFunctionsProcess(background=self.background,
+ commands=post_phase_cmds, phase=self.phase, pkg=self.pkg,
+ scheduler=self.scheduler, settings=settings)
+ self._start_task(post_phase, self._post_phase_exit)
+ return
+
+ self.returncode = ebuild_process.returncode
+ self._current_task = None
+ self.wait()
+
+ def _post_phase_exit(self, post_phase):
+ if self._final_exit(post_phase) != os.EX_OK:
+ writemsg("!!! post %s failed; exiting.\n" % self.phase,
+ noiselevel=-1)
+ self._current_task = None
+ self.wait()
+ return
-class EbuildBinpkg(EbuildPhase):
+class EbuildBinpkg(EbuildProcess):
"""
This assumes that src_install() has successfully completed.
"""
@@ -2817,12 +2868,12 @@ class EbuildBinpkg(EbuildPhase):
settings.backup_changes("PORTAGE_BINPKG_TMPFILE")
try:
- EbuildPhase._start(self)
+ EbuildProcess._start(self)
finally:
settings.pop("PORTAGE_BINPKG_TMPFILE", None)
def _set_returncode(self, wait_retval):
- EbuildPhase._set_returncode(self, wait_retval)
+ EbuildProcess._set_returncode(self, wait_retval)
pkg = self.pkg
bintree = pkg.root_config.trees["bintree"]
diff --git a/pym/portage/__init__.py b/pym/portage/__init__.py
index ff83658bc..7e8cc30a8 100644
--- a/pym/portage/__init__.py
+++ b/pym/portage/__init__.py
@@ -4259,6 +4259,23 @@ def spawnebuild(mydo, actionmap, mysettings, debug, alwaysdep=0,
phase_retval = _post_src_install_checks(mysettings)
return phase_retval
+_post_phase_cmds = {
+
+ "install" : [
+ "install_qa_check",
+ "install_symlink_html_docs"],
+
+ "preinst" : [
+ "preinst_bsdflags",
+ "preinst_sfperms",
+ "preinst_selinux_labels",
+ "preinst_suid_scan",
+ "preinst_mask"],
+
+ "postinst" : [
+ "postinst_bsdflags"]
+}
+
def _post_phase_userpriv_perms(mysettings):
if "userpriv" in mysettings.features and secpass >= 2:
""" Privileged phases may have left files that need to be made
@@ -4269,8 +4286,8 @@ def _post_phase_userpriv_perms(mysettings):
def _post_src_install_checks(mysettings):
_post_src_install_uid_fix(mysettings)
- retval = _spawn_misc_sh(mysettings, ["install_qa_check",
- "install_symlink_html_docs"])
+ global _post_phase_cmds
+ retval = _spawn_misc_sh(mysettings, post_phase_cmds["install"])
if retval != os.EX_OK:
writemsg("!!! install_qa_check failed; exiting.\n",
noiselevel=-1)
@@ -4396,10 +4413,8 @@ def _post_pkg_preinst_cmd(mysettings):
os.path.basename(MISC_SH_BINARY))
mysettings["EBUILD_PHASE"] = ""
- myargs = [_shell_quote(misc_sh_binary),
- "preinst_bsdflags",
- "preinst_sfperms", "preinst_selinux_labels",
- "preinst_suid_scan", "preinst_mask"]
+ global _post_phase_cmds
+ myargs = [_shell_quote(misc_sh_binary)] + _post_phase_cmds["preinst"]
return myargs
@@ -4414,7 +4429,8 @@ def _post_pkg_postinst_cmd(mysettings):
os.path.basename(MISC_SH_BINARY))
mysettings["EBUILD_PHASE"] = ""
- myargs = [_shell_quote(misc_sh_binary), "postinst_bsdflags"]
+ global _post_phase_cmds
+ myargs = [_shell_quote(misc_sh_binary)] + _post_phase_cmds["postinst"]
return myargs