<?php namespace nur\b\proc; use nur\A; use nur\base; use nur\shell; abstract class AbstractCmd implements ICmd { /** @var bool */ private $needsStdin; /** @var bool */ private $needsTty; /** @var array */ protected $vars; /** @var array */ protected $cmds; function __construct() { $this->needsStdin = true; $this->needsTty = true; $this->cmds = []; } function then($cmd, ?string $input=null, ?string $output=null): Cmd { if ($this instanceof Cmd) { $this->add($cmd, $input, $output); return $this; } else { return (new Cmd($this))->add($cmd, $input, $output); } } function or($cmd, ?string $input=null, ?string $output=null): CmdOr { if ($this instanceof CmdOr) { $this->add($cmd, $input, $output); return $this; } else { return (new CmdOr($this))->add($cmd, $input, $output); } } function and($cmd, ?string $input=null, ?string $output=null): CmdAnd { if ($this instanceof CmdAnd) { $this->add($cmd, $input, $output); return $this; } else { return (new CmdAnd($this))->add($cmd, $input, $output); } } function pipe($cmd): CmdPipe { if ($this instanceof CmdPipe) { $this->add($cmd); return $this; } else { return new CmdPipe([$this, $cmd]); } } function isNeedsStdin(): bool { return $this->needsStdin; } function setNeedsStdin(bool $needsStdin): void { $this->needsStdin = $needsStdin; } function isNeedsTty(): bool { return $this->needsTty; } function setNeedsTty(bool $needsTty): void { $this->needsTty = $needsTty; } function addLiteralVars($vars, ?string $sep=null): void { if (base::z($vars)) return; if (is_array($vars)) { if ($sep === null) $sep = "\n"; $vars = implode($sep, $vars); } A::append($this->vars, strval($vars)); } function addVars(?array $vars): void { if ($vars === null) return; foreach ($vars as $name => $value) { $var = []; if (!is_array($value)) $var[] = "export "; A::merge($var, [$name, "=", shell::quote($value)]); A::append($this->vars, implode("", $var)); } } function getVars(?string $sep=null): ?string { if ($this->vars === null) return null; if ($sep === null) $sep = "\n"; return implode($sep, $this->vars); } function addPrefix($prefix): void { $count = count($this->cmds); if ($count == 0) return; $cmd =& $this->cmds[$count - 1]; if ($cmd instanceof ICmd) { $cmd->addPrefix($prefix); } elseif (is_array($prefix)) { $prefix = shell::join($prefix); $cmd = "$prefix $cmd"; } else { $cmd = "$prefix $cmd"; } } function addRedir(?string $redir, $output=null, bool $append=false, $input=null): void { $count = count($this->cmds); if ($count == 0) return; if ($output !== null) $output = escapeshellarg($output); if ($input !== null) $input = escapeshellarg($input); if ($redir === "default") $redir = null; $gt = $append? ">>": ">"; if ($redir === null) { $redirs = []; if ($input !== null) $redirs[] = "<$input"; if ($output !== null) $redirs[] = "$gt$output"; if ($redirs) $redir = implode(" ", $redir); } else { switch ($redir) { case "outonly": case "noerr": if ($output !== null) $redir = "$gt$output 2>/dev/null"; else $redir = "2>/dev/null"; break; case "erronly": case "noout": if ($output !== null) $redir = "2$gt$output >/dev/null"; else $redir = "2>&1 >/dev/null"; break; case "both": case "err2out": if ($output !== null) $redir = "$gt$output 2>&1"; else $redir = "2>&1"; break; case "none": case "null": $redir = ">/dev/null 2>&1"; break; } } if ($redir !== null) { $cmd =& $this->cmds[$count - 1]; if ($cmd instanceof ICmd) { $cmd->addRedir($redir); } else { $cmd = "$cmd $redir"; } } } abstract function getCmd(?string $sep=null): string; function passthru(int &$retcode=null): bool { passthru($this->getCmd(), $retcode); return $retcode == 0; } function system(string &$output=null, int &$retcode=null): bool { $last_line = system($this->getCmd(), $retcode); if ($last_line !== false) $output = $last_line; return $retcode == 0; } function exec(array &$output=null, int &$retcode=null): bool { exec($this->getCmd(), $output, $retcode); return $retcode == 0; } /** * retourner true s'il faut utiliser `exec` avec {@link fork_exec()} * * ne pas utiliser `exec` si des variables sont définies ou si c'est une * composition de plusieurs commandes */ protected function useExec(): bool { return $this->vars === null && count($this->cmds) == 1; } function fork_exec(int &$retcode=null): bool { $cmd = $this->getCmd(); if ($this->useExec()) $cmd = "exec $cmd"; shell::_fork_exec($cmd, $retcode); return $retcode == 0; } }