- PHP 7.4 support - Improved Webp images support with Cloudflare (Issue [#95](https://github.com/WordOps/WordOps/issues/95)). Nginx will not serve webp images alternative with Cloudflare IP ranges. - Stack upgrade for adminer - Check acme.sh installation and setup acme.sh if needed before issuing certificate - Add `--ufw` to `wo stack status` - Add Nginx directive `gzip_static on;` to serve precompressed assets with Cache-Enabler or WP-Rocket. (Issue [#207](https://github.com/WordOps/WordOps/issues/207)) - Previous `--php73` & `--php73=off` flags are replaced by `--php72`, `--php73`, `--php74` to switch site's php version - phpMyAdmin updated to v4.9.2 - Adminer updated to v4.7.5 - Replace dot and dashes by underscores in database names (Issue [#206](https://github.com/WordOps/WordOps/issues/206)) - Increased database name length to 32 characters from domain name + 8 random characters - typo error in motd-news script (Issue [#204](https://github.com/WordOps/WordOps/issues/204)) - Install Nginx before ngxblocker - WordOps install/update script text color - Issue with MySQL stack on Raspbian 9/10 - Typo error (PR [#205](https://github.com/WordOps/WordOps/pull/205)) - php version in `wo debug` (PR [#209](https://github.com/WordOps/WordOps/pull/209)) - SSL certificates expiration display with shared wildcard certificates
76 lines
3.0 KiB
Python
76 lines
3.0 KiB
Python
"""WordOps Shell Functions"""
|
|
import subprocess
|
|
|
|
from wo.core.logging import Log
|
|
|
|
|
|
class CommandExecutionError(Exception):
|
|
"""custom Exception for command execution"""
|
|
pass
|
|
|
|
|
|
class WOShellExec():
|
|
"""Method to run shell commands"""
|
|
def __init__():
|
|
pass
|
|
|
|
def cmd_exec(self, command, errormsg='', log=True):
|
|
"""Run shell command from Python"""
|
|
try:
|
|
log and Log.debug(self, "Running command: {0}".format(command))
|
|
|
|
with subprocess.Popen([command], stdout=subprocess.PIPE,
|
|
stderr=subprocess.PIPE, shell=True) as proc:
|
|
(cmd_stdout_bytes, cmd_stderr_bytes) = proc.communicate()
|
|
(cmd_stdout, cmd_stderr) = (cmd_stdout_bytes.decode('utf-8',
|
|
"replace"),
|
|
cmd_stderr_bytes.decode('utf-8',
|
|
"replace"))
|
|
|
|
Log.debug(self, "Command Output: {0}, \nCommand Error: {1}"
|
|
.format(cmd_stdout, cmd_stderr))
|
|
return bool(proc.returncode == 0)
|
|
except OSError as e:
|
|
Log.debug(self, str(e))
|
|
raise CommandExecutionError
|
|
except Exception as e:
|
|
Log.debug(self, str(e))
|
|
raise CommandExecutionError
|
|
|
|
def invoke_editor(self, filepath, errormsg=''):
|
|
"""
|
|
Open files using sensible editor
|
|
"""
|
|
try:
|
|
subprocess.call(['sensible-editor', filepath])
|
|
except OSError as e:
|
|
Log.debug(self, "{0}{1}".format(e.errno, e.strerror))
|
|
raise CommandExecutionError
|
|
|
|
def cmd_exec_stdout(self, command, errormsg='', log=True):
|
|
"""Run shell command from Python"""
|
|
try:
|
|
log and Log.debug(self, "Running command: {0}".format(command))
|
|
with subprocess.Popen([command], stdout=subprocess.PIPE,
|
|
stderr=subprocess.PIPE, shell=True) as proc:
|
|
(cmd_stdout_bytes, cmd_stderr_bytes) = proc.communicate()
|
|
(cmd_stdout, cmd_stderr) = (cmd_stdout_bytes.decode('utf-8',
|
|
"replace"),
|
|
cmd_stderr_bytes.decode('utf-8',
|
|
"replace"))
|
|
|
|
if proc.returncode == 0:
|
|
Log.debug(self, "Command Output: {0}, \nCommand Error: {1}"
|
|
.format(cmd_stdout, cmd_stderr))
|
|
return cmd_stdout
|
|
else:
|
|
Log.debug(self, "Command Output: {0}, \nCommand Error: {1}"
|
|
.format(cmd_stdout, cmd_stderr))
|
|
return cmd_stdout
|
|
except OSError as e:
|
|
Log.debug(self, str(e))
|
|
raise CommandExecutionError
|
|
except Exception as e:
|
|
Log.debug(self, str(e))
|
|
raise CommandExecutionError
|