diff --git a/python/vyos/config_mgmt.py b/python/vyos/config_mgmt.py index 70b6ea203..d518737ca 100644 --- a/python/vyos/config_mgmt.py +++ b/python/vyos/config_mgmt.py @@ -1,759 +1,761 @@ # Copyright 2023-2024 VyOS maintainers and contributors <maintainers@vyos.io> # # This library is free software; you can redistribute it and/or # modify it under the terms of the GNU Lesser General Public # License as published by the Free Software Foundation; either # version 2.1 of the License, or (at your option) any later version. # # This library is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU # Lesser General Public License for more details. # # You should have received a copy of the GNU Lesser General Public License # along with this library. If not, see <http://www.gnu.org/licenses/>. import os import re import sys import gzip import logging from typing import Optional from typing import Tuple from filecmp import cmp from datetime import datetime from textwrap import dedent from pathlib import Path from tabulate import tabulate from shutil import copy, chown from urllib.parse import urlsplit from urllib.parse import urlunsplit from vyos.config import Config from vyos.configtree import ConfigTree from vyos.configtree import ConfigTreeError from vyos.configtree import show_diff from vyos.load_config import load from vyos.load_config import LoadConfigError from vyos.defaults import directories from vyos.version import get_full_version_data from vyos.utils.io import ask_yes_no from vyos.utils.boot import boot_configuration_complete from vyos.utils.process import is_systemd_service_active from vyos.utils.process import rc_cmd SAVE_CONFIG = '/usr/libexec/vyos/vyos-save-config.py' config_json = '/run/vyatta/config/config.json' # created by vyatta-cfg-postinst commit_post_hook_dir = '/etc/commit/post-hooks.d' commit_hooks = {'commit_revision': '01vyos-commit-revision', 'commit_archive': '02vyos-commit-archive'} DEFAULT_TIME_MINUTES = 10 timer_name = 'commit-confirm' config_file = os.path.join(directories['config'], 'config.boot') archive_dir = os.path.join(directories['config'], 'archive') archive_config_file = os.path.join(archive_dir, 'config.boot') commit_log_file = os.path.join(archive_dir, 'commits') logrotate_conf = os.path.join(archive_dir, 'lr.conf') logrotate_state = os.path.join(archive_dir, 'lr.state') rollback_config = os.path.join(archive_dir, 'config.boot-rollback') prerollback_config = os.path.join(archive_dir, 'config.boot-prerollback') tmp_log_entry = '/tmp/commit-rev-entry' logger = logging.getLogger('config_mgmt') logger.setLevel(logging.INFO) ch = logging.StreamHandler() formatter = logging.Formatter('%(funcName)s: %(levelname)s:%(message)s') ch.setFormatter(formatter) logger.addHandler(ch) def save_config(target, json_out=None): if json_out is None: cmd = f'{SAVE_CONFIG} {target}' else: cmd = f'{SAVE_CONFIG} {target} --write-json-file {json_out}' rc, out = rc_cmd(cmd) if rc != 0: logger.critical(f'save config failed: {out}') -def unsaved_commits() -> bool: +def unsaved_commits(allow_missing_config=False) -> bool: if get_full_version_data()['boot_via'] == 'livecd': return False + if allow_missing_config and not os.path.exists(config_file): + return True tmp_save = '/tmp/config.running' save_config(tmp_save) ret = not cmp(tmp_save, config_file, shallow=False) os.unlink(tmp_save) return ret def get_file_revision(rev: int): revision = os.path.join(archive_dir, f'config.boot.{rev}.gz') try: with gzip.open(revision) as f: r = f.read().decode() except FileNotFoundError: logger.warning(f'commit revision {rev} not available') return '' return r def get_config_tree_revision(rev: int): c = get_file_revision(rev) return ConfigTree(c) def is_node_revised(path: list = [], rev1: int = 1, rev2: int = 0) -> bool: from vyos.configtree import DiffTree left = get_config_tree_revision(rev1) right = get_config_tree_revision(rev2) diff_tree = DiffTree(left, right) if diff_tree.add.exists(path) or diff_tree.sub.exists(path): return True return False class ConfigMgmtError(Exception): pass class ConfigMgmt: def __init__(self, session_env=None, config=None): if session_env: self._session_env = session_env else: self._session_env = None if config is None: config = Config() d = config.get_config_dict(['system', 'config-management'], key_mangling=('-', '_'), get_first_key=True) self.max_revisions = int(d.get('commit_revisions', 0)) self.num_revisions = 0 self.locations = d.get('commit_archive', {}).get('location', []) self.source_address = d.get('commit_archive', {}).get('source_address', '') if config.exists(['system', 'host-name']): self.hostname = config.return_value(['system', 'host-name']) if config.exists(['system', 'domain-name']): tmp = config.return_value(['system', 'domain-name']) self.hostname += f'.{tmp}' else: self.hostname = 'vyos' # upload only on existence of effective values, notably, on boot. # one still needs session self.locations (above) for setting # post-commit hook in conf_mode script path = ['system', 'config-management', 'commit-archive', 'location'] if config.exists_effective(path): self.effective_locations = config.return_effective_values(path) else: self.effective_locations = [] # a call to compare without args is edit_level aware edit_level = os.getenv('VYATTA_EDIT_LEVEL', '') self.edit_path = [l for l in edit_level.split('/') if l] self.active_config = config._running_config self.working_config = config._session_config # Console script functions # def commit_confirm(self, minutes: int=DEFAULT_TIME_MINUTES, no_prompt: bool=False) -> Tuple[str,int]: """Commit with reboot to saved config in 'minutes' minutes if 'confirm' call is not issued. """ if is_systemd_service_active(f'{timer_name}.timer'): msg = 'Another confirm is pending' return msg, 1 if unsaved_commits(): W = '\nYou should save previous commits before commit-confirm !\n' else: W = '' prompt_str = f''' commit-confirm will automatically reboot in {minutes} minutes unless changes are confirmed.\n Proceed ?''' prompt_str = W + prompt_str if not no_prompt and not ask_yes_no(prompt_str, default=True): msg = 'commit-confirm canceled' return msg, 1 action = 'sg vyattacfg "/usr/bin/config-mgmt revert"' cmd = f'sudo systemd-run --quiet --on-active={minutes}m --unit={timer_name} {action}' rc, out = rc_cmd(cmd) if rc != 0: raise ConfigMgmtError(out) # start notify cmd = f'sudo -b /usr/libexec/vyos/commit-confirm-notify.py {minutes}' os.system(cmd) msg = f'Initialized commit-confirm; {minutes} minutes to confirm before reboot' return msg, 0 def confirm(self) -> Tuple[str,int]: """Do not reboot to saved config following 'commit-confirm'. Update commit log and archive. """ if not is_systemd_service_active(f'{timer_name}.timer'): msg = 'No confirm pending' return msg, 0 cmd = f'sudo systemctl stop --quiet {timer_name}.timer' rc, out = rc_cmd(cmd) if rc != 0: raise ConfigMgmtError(out) # kill notify cmd = 'sudo pkill -f commit-confirm-notify.py' rc, out = rc_cmd(cmd) if rc != 0: raise ConfigMgmtError(out) entry = self._read_tmp_log_entry() if self._archive_active_config(): self._add_log_entry(**entry) self._update_archive() msg = 'Reboot timer stopped' return msg, 0 def revert(self) -> Tuple[str,int]: """Reboot to saved config, dropping commits from 'commit-confirm'. """ _ = self._read_tmp_log_entry() # archived config will be reverted on boot rc, out = rc_cmd('sudo systemctl reboot') if rc != 0: raise ConfigMgmtError(out) return '', 0 def rollback(self, rev: int, no_prompt: bool=False) -> Tuple[str,int]: """Reboot to config revision 'rev'. """ msg = '' if not self._check_revision_number(rev): msg = f'Invalid revision number {rev}: must be 0 < rev < {self.num_revisions}' return msg, 1 prompt_str = 'Proceed with reboot ?' if not no_prompt and not ask_yes_no(prompt_str, default=True): msg = 'Canceling rollback' return msg, 0 rc, out = rc_cmd(f'sudo cp {archive_config_file} {prerollback_config}') if rc != 0: raise ConfigMgmtError(out) path = os.path.join(archive_dir, f'config.boot.{rev}.gz') with gzip.open(path) as f: config = f.read() try: with open(rollback_config, 'wb') as f: f.write(config) copy(rollback_config, config_file) except OSError as e: raise ConfigMgmtError from e rc, out = rc_cmd('sudo systemctl reboot') if rc != 0: raise ConfigMgmtError(out) return msg, 0 def rollback_soft(self, rev: int): """Rollback without reboot (rollback-soft) """ msg = '' if not self._check_revision_number(rev): msg = f'Invalid revision number {rev}: must be 0 < rev < {self.num_revisions}' return msg, 1 rollback_ct = self._get_config_tree_revision(rev) try: load(rollback_ct, switch='explicit') print('Rollback diff has been applied.') print('Use "compare" to review the changes or "commit" to apply them.') except LoadConfigError as e: raise ConfigMgmtError(e) from e return msg, 0 def compare(self, saved: bool=False, commands: bool=False, rev1: Optional[int]=None, rev2: Optional[int]=None) -> Tuple[str,int]: """General compare function for config file revisions: revision n vs. revision m; working version vs. active version; or working version vs. saved version. """ ct1 = self.active_config ct2 = self.working_config msg = 'No changes between working and active configurations.\n' if saved: ct1 = self._get_saved_config_tree() ct2 = self.working_config msg = 'No changes between working and saved configurations.\n' if rev1 is not None: if not self._check_revision_number(rev1): return f'Invalid revision number {rev1}', 1 ct1 = self._get_config_tree_revision(rev1) ct2 = self.working_config msg = f'No changes between working and revision {rev1} configurations.\n' if rev2 is not None: if not self._check_revision_number(rev2): return f'Invalid revision number {rev2}', 1 # compare older to newer ct2 = ct1 ct1 = self._get_config_tree_revision(rev2) msg = f'No changes between revisions {rev2} and {rev1} configurations.\n' out = '' path = [] if commands else self.edit_path try: if commands: out = show_diff(ct1, ct2, path=path, commands=True) else: out = show_diff(ct1, ct2, path=path) except ConfigTreeError as e: return e, 1 if out: msg = out return msg, 0 def wrap_compare(self, options) -> Tuple[str,int]: """Interface to vyatta-cfg-run: args collected as 'options' to parse for compare. """ cmnds = False r1 = None r2 = None if 'commands' in options: cmnds=True options.remove('commands') for i in options: if not i.isnumeric(): options.remove(i) if len(options) > 0: r1 = int(options[0]) if len(options) > 1: r2 = int(options[1]) return self.compare(commands=cmnds, rev1=r1, rev2=r2) # Initialization and post-commit hooks for conf-mode # def initialize_revision(self): """Initialize config archive, logrotate conf, and commit log. """ mask = os.umask(0o002) os.makedirs(archive_dir, exist_ok=True) json_dir = os.path.dirname(config_json) try: os.makedirs(json_dir, exist_ok=True) chown(json_dir, group='vyattacfg') except OSError as e: logger.warning(f'cannot create {json_dir}: {e}') self._add_logrotate_conf() if (not os.path.exists(commit_log_file) or self._get_number_of_revisions() == 0): user = self._get_user() via = 'init' comment = '' # add empty init config before boot-config load for revision # and diff consistency if self._archive_active_config(): self._add_log_entry(user, via, comment) self._update_archive() os.umask(mask) def commit_revision(self): """Update commit log and rotate archived config.boot. commit_revision is called in post-commit-hooks, if ['commit-archive', 'commit-revisions'] is configured. """ if os.getenv('IN_COMMIT_CONFIRM', ''): self._new_log_entry(tmp_file=tmp_log_entry) return if self._archive_active_config(): self._add_log_entry() self._update_archive() def commit_archive(self): """Upload config to remote archive. """ from vyos.remote import upload hostname = self.hostname t = datetime.now() timestamp = t.strftime('%Y%m%d_%H%M%S') remote_file = f'config.boot-{hostname}.{timestamp}' source_address = self.source_address if self.effective_locations: print("Archiving config...") for location in self.effective_locations: url = urlsplit(location) _, _, netloc = url.netloc.rpartition("@") redacted_location = urlunsplit(url._replace(netloc=netloc)) print(f" {redacted_location}", end=" ", flush=True) upload(archive_config_file, f'{location}/{remote_file}', source_host=source_address) # op-mode functions # def get_raw_log_data(self) -> list: """Return list of dicts of log data: keys: [timestamp, user, commit_via, commit_comment] """ log = self._get_log_entries() res_l = [] for line in log: d = self._get_log_entry(line) res_l.append(d) return res_l @staticmethod def format_log_data(data: list) -> str: """Return formatted log data as str. """ res_l = [] for l_no, l in enumerate(data): time_d = datetime.fromtimestamp(int(l['timestamp'])) time_str = time_d.strftime("%Y-%m-%d %H:%M:%S") res_l.append([l_no, time_str, f"by {l['user']}", f"via {l['commit_via']}"]) if l['commit_comment'] != 'commit': # default comment res_l.append([None, l['commit_comment']]) ret = tabulate(res_l, tablefmt="plain") return ret @staticmethod def format_log_data_brief(data: list) -> str: """Return 'brief' form of log data as str. Slightly compacted format used in completion help for 'rollback'. """ res_l = [] for l_no, l in enumerate(data): time_d = datetime.fromtimestamp(int(l['timestamp'])) time_str = time_d.strftime("%Y-%m-%d %H:%M:%S") res_l.append(['\t', l_no, time_str, f"{l['user']}", f"by {l['commit_via']}"]) ret = tabulate(res_l, tablefmt="plain") return ret def show_commit_diff(self, rev: int, rev2: Optional[int]=None, commands: bool=False) -> str: """Show commit diff at revision number, compared to previous revision, or to another revision. """ if rev2 is None: out, _ = self.compare(commands=commands, rev1=rev, rev2=(rev+1)) return out out, _ = self.compare(commands=commands, rev1=rev, rev2=rev2) return out def show_commit_file(self, rev: int) -> str: return self._get_file_revision(rev) # utility functions # def _get_saved_config_tree(self): with open(config_file) as f: c = f.read() return ConfigTree(c) def _get_file_revision(self, rev: int): if rev not in range(0, self._get_number_of_revisions()): raise ConfigMgmtError('revision not available') revision = os.path.join(archive_dir, f'config.boot.{rev}.gz') with gzip.open(revision) as f: r = f.read().decode() return r def _get_config_tree_revision(self, rev: int): c = self._get_file_revision(rev) return ConfigTree(c) def _add_logrotate_conf(self): conf: str = dedent(f"""\ {archive_config_file} {{ su root vyattacfg rotate {self.max_revisions} start 0 compress copy }} """) conf_file = Path(logrotate_conf) conf_file.write_text(conf) conf_file.chmod(0o644) def _archive_active_config(self) -> bool: save_to_tmp = (boot_configuration_complete() or not os.path.isfile(archive_config_file)) mask = os.umask(0o113) ext = os.getpid() cmp_saved = f'/tmp/config.boot.{ext}' if save_to_tmp: save_config(cmp_saved, json_out=config_json) else: copy(config_file, cmp_saved) # on boot, we need to manually create the config.json file; after # boot, it is written by save_config, above if not os.path.exists(config_json): ct = self._get_saved_config_tree() try: with open(config_json, 'w') as f: f.write(ct.to_json()) chown(config_json, group='vyattacfg') except OSError as e: logger.warning(f'cannot create {config_json}: {e}') try: if cmp(cmp_saved, archive_config_file, shallow=False): os.unlink(cmp_saved) os.umask(mask) return False except FileNotFoundError: pass rc, out = rc_cmd(f'sudo mv {cmp_saved} {archive_config_file}') os.umask(mask) if rc != 0: logger.critical(f'mv file to archive failed: {out}') return False return True @staticmethod def _update_archive(): cmd = f"sudo logrotate -f -s {logrotate_state} {logrotate_conf}" rc, out = rc_cmd(cmd) if rc != 0: logger.critical(f'logrotate failure: {out}') @staticmethod def _get_log_entries() -> list: """Return lines of commit log as list of strings """ entries = [] if os.path.exists(commit_log_file): with open(commit_log_file) as f: entries = f.readlines() return entries def _get_number_of_revisions(self) -> int: l = self._get_log_entries() return len(l) def _check_revision_number(self, rev: int) -> bool: self.num_revisions = self._get_number_of_revisions() if not 0 <= rev < self.num_revisions: return False return True @staticmethod def _get_user() -> str: import pwd try: user = os.getlogin() except OSError: try: user = pwd.getpwuid(os.geteuid())[0] except KeyError: user = 'unknown' return user def _new_log_entry(self, user: str='', commit_via: str='', commit_comment: str='', timestamp: Optional[int]=None, tmp_file: str=None) -> Optional[str]: # Format log entry and return str or write to file. # # Usage is within a post-commit hook, using env values. In case of # commit-confirm, it can be written to a temporary file for # inclusion on 'confirm'. from time import time if timestamp is None: timestamp = int(time()) if not user: user = self._get_user() if not commit_via: commit_via = os.getenv('COMMIT_VIA', 'other') if not commit_comment: commit_comment = os.getenv('COMMIT_COMMENT', 'commit') # the commit log reserves '|' as field demarcation, so replace in # comment if present; undo this in _get_log_entry, below if re.search(r'\|', commit_comment): commit_comment = commit_comment.replace('|', '%%') entry = f'|{timestamp}|{user}|{commit_via}|{commit_comment}|\n' mask = os.umask(0o113) if tmp_file is not None: try: with open(tmp_file, 'w') as f: f.write(entry) except OSError as e: logger.critical(f'write to {tmp_file} failed: {e}') os.umask(mask) return None os.umask(mask) return entry @staticmethod def _get_log_entry(line: str) -> dict: log_fmt = re.compile(r'\|.*\|\n?$') keys = ['user', 'commit_via', 'commit_comment', 'timestamp'] if not log_fmt.match(line): logger.critical(f'Invalid log format {line}') return {} timestamp, user, commit_via, commit_comment = ( tuple(line.strip().strip('|').split('|'))) commit_comment = commit_comment.replace('%%', '|') d = dict(zip(keys, [user, commit_via, commit_comment, timestamp])) return d def _read_tmp_log_entry(self) -> dict: try: with open(tmp_log_entry) as f: entry = f.read() os.unlink(tmp_log_entry) except OSError as e: logger.critical(f'error on file {tmp_log_entry}: {e}') return self._get_log_entry(entry) def _add_log_entry(self, user: str='', commit_via: str='', commit_comment: str='', timestamp: Optional[int]=None): mask = os.umask(0o113) entry = self._new_log_entry(user=user, commit_via=commit_via, commit_comment=commit_comment, timestamp=timestamp) log_entries = self._get_log_entries() log_entries.insert(0, entry) if len(log_entries) > self.max_revisions: log_entries = log_entries[:-1] try: with open(commit_log_file, 'w') as f: f.writelines(log_entries) except OSError as e: logger.critical(e) os.umask(mask) # entry_point for console script # def run(): from argparse import ArgumentParser, REMAINDER config_mgmt = ConfigMgmt() for s in list(commit_hooks): if sys.argv[0].replace('-', '_').endswith(s): func = getattr(config_mgmt, s) try: func() except Exception as e: print(f'{s}: {e}') sys.exit(0) parser = ArgumentParser() subparsers = parser.add_subparsers(dest='subcommand') commit_confirm = subparsers.add_parser('commit_confirm', help="Commit with opt-out reboot to saved config") commit_confirm.add_argument('-t', dest='minutes', type=int, default=DEFAULT_TIME_MINUTES, help="Minutes until reboot, unless 'confirm'") commit_confirm.add_argument('-y', dest='no_prompt', action='store_true', help="Execute without prompt") subparsers.add_parser('confirm', help="Confirm commit") subparsers.add_parser('revert', help="Revert commit-confirm") rollback = subparsers.add_parser('rollback', help="Rollback to earlier config") rollback.add_argument('--rev', type=int, help="Revision number for rollback") rollback.add_argument('-y', dest='no_prompt', action='store_true', help="Excute without prompt") rollback_soft = subparsers.add_parser('rollback_soft', help="Rollback to earlier config") rollback_soft.add_argument('--rev', type=int, help="Revision number for rollback") compare = subparsers.add_parser('compare', help="Compare config files") compare.add_argument('--saved', action='store_true', help="Compare session config with saved config") compare.add_argument('--commands', action='store_true', help="Show difference between commands") compare.add_argument('--rev1', type=int, default=None, help="Compare revision with session config or other revision") compare.add_argument('--rev2', type=int, default=None, help="Compare revisions") wrap_compare = subparsers.add_parser('wrap_compare', help="Wrapper interface for vyatta-cfg-run") wrap_compare.add_argument('--options', nargs=REMAINDER) args = vars(parser.parse_args()) func = getattr(config_mgmt, args['subcommand']) del args['subcommand'] res = '' try: res, rc = func(**args) except ConfigMgmtError as e: print(e) sys.exit(1) if res: print(res) sys.exit(rc) diff --git a/src/op_mode/powerctrl.py b/src/op_mode/powerctrl.py index 6c8f802b5..cb4a175dd 100755 --- a/src/op_mode/powerctrl.py +++ b/src/op_mode/powerctrl.py @@ -1,236 +1,236 @@ #!/usr/bin/env python3 # # Copyright (C) 2023-2024 VyOS maintainers and contributors # # This program is free software; you can redistribute it and/or modify # it under the terms of the GNU General Public License version 2 or later as # published by the Free Software Foundation. # # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License # along with this program. If not, see <http://www.gnu.org/licenses/>. import os import re from argparse import ArgumentParser from datetime import datetime from sys import exit from time import time from vyos.utils.io import ask_yes_no from vyos.utils.process import call from vyos.utils.process import cmd from vyos.utils.process import run from vyos.utils.process import STDOUT systemd_sched_file = "/run/systemd/shutdown/scheduled" def utc2local(datetime): now = time() offs = datetime.fromtimestamp(now) - datetime.utcfromtimestamp(now) return datetime + offs def parse_time(s): try: if re.match(r'^\d{1,9999}$', s): if (int(s) > 59) and (int(s) < 1440): s = str(int(s)//60) + ":" + str(int(s)%60) return datetime.strptime(s, "%H:%M").time() if (int(s) >= 1440): return s.split() else: return datetime.strptime(s, "%M").time() else: return datetime.strptime(s, "%H:%M").time() except ValueError: return None def parse_date(s): for fmt in ["%d%m%Y", "%d/%m/%Y", "%d.%m.%Y", "%d:%m:%Y", "%Y-%m-%d"]: try: return datetime.strptime(s, fmt).date() except ValueError: continue # If nothing matched... return None def get_shutdown_status(): if os.path.exists(systemd_sched_file): # Get scheduled from systemd file with open(systemd_sched_file, 'r') as f: data = f.read().rstrip('\n') r_data = {} for line in data.splitlines(): tmp_split = line.split("=") if tmp_split[0] == "USEC": # Convert USEC to human readable format r_data['DATETIME'] = datetime.utcfromtimestamp( int(tmp_split[1])/1000000).strftime('%Y-%m-%d %H:%M:%S') else: r_data[tmp_split[0]] = tmp_split[1] return r_data return None def check_shutdown(): output = get_shutdown_status() if output and 'MODE' in output: dt = datetime.strptime(output['DATETIME'], '%Y-%m-%d %H:%M:%S') if output['MODE'] == 'reboot': print("Reboot is scheduled", utc2local(dt)) elif output['MODE'] == 'poweroff': print("Poweroff is scheduled", utc2local(dt)) else: print("Reboot or poweroff is not scheduled") def cancel_shutdown(): output = get_shutdown_status() if output and 'MODE' in output: timenow = datetime.now().strftime('%Y-%m-%d %H:%M:%S') try: run('/sbin/shutdown -c --no-wall') except OSError as e: exit(f'Could not cancel a reboot or poweroff: {e}') mode = output['MODE'] message = f'Scheduled {mode} has been cancelled {timenow}' run(f'wall {message} > /dev/null 2>&1') else: print("Reboot or poweroff is not scheduled") def check_unsaved_config(): from vyos.config_mgmt import unsaved_commits from vyos.utils.boot import boot_configuration_success - if unsaved_commits() and boot_configuration_success(): + if unsaved_commits(allow_missing_config=True) and boot_configuration_success(): print("Warning: there are unsaved configuration changes!") print("Run 'save' command if you do not want to lose those changes after reboot/shutdown.") else: pass def execute_shutdown(time, reboot=True, ask=True): check_unsaved_config() action = "reboot" if reboot else "poweroff" if not ask: if not ask_yes_no(f"Are you sure you want to {action} this system?"): exit(0) action_cmd = "-r" if reboot else "-P" if len(time) == 0: # T870 legacy reboot job support chk_vyatta_based_reboots() ### out = cmd(f'/sbin/shutdown {action_cmd} now', stderr=STDOUT) print(out.split(",", 1)[0]) return elif len(time) == 1: # Assume the argument is just time ts = parse_time(time[0]) if ts: cmd(f'/sbin/shutdown {action_cmd} {time[0]}', stderr=STDOUT) # Inform all other logged in users about the reboot/shutdown wall_msg = f'System {action} is scheduled {time[0]}' cmd(f'/usr/bin/wall "{wall_msg}"') else: exit(f'Invalid time "{time[0]}". The valid format is HH:MM') elif len(time) == 2: # Assume it's date and time ts = parse_time(time[0]) ds = parse_date(time[1]) if ts and ds: t = datetime.combine(ds, ts) td = t - datetime.now() t2 = 1 + int(td.total_seconds())//60 # Get total minutes cmd(f'/sbin/shutdown {action_cmd} {t2}', stderr=STDOUT) # Inform all other logged in users about the reboot/shutdown wall_msg = f'System {action} is scheduled {time[1]} {time[0]}' cmd(f'/usr/bin/wall "{wall_msg}"') else: if not ts: exit(f'Invalid time "{time[0]}". Uses 24 Hour Clock format') else: exit(f'Invalid date "{time[1]}". A valid format is YYYY-MM-DD [HH:MM]') else: exit('Could not decode date and time. Valids formats are HH:MM or YYYY-MM-DD HH:MM') check_shutdown() def chk_vyatta_based_reboots(): # T870 commit-confirm is still using the vyatta code base, once gone, the code below can be removed # legacy scheduled reboot s are using at and store the is as /var/run/<name>.job # name is the node of scheduled the job, commit-confirm checks for that f = r'/var/run/confirm.job' if os.path.exists(f): jid = open(f).read().strip() if jid != 0: call(f'sudo atrm {jid}') os.remove(f) def main(): parser = ArgumentParser() parser.add_argument("--yes", "-y", help="Do not ask for confirmation", action="store_true", dest="yes") action = parser.add_mutually_exclusive_group(required=True) action.add_argument("--reboot", "-r", help="Reboot the system", nargs="*", metavar="HH:MM") action.add_argument("--reboot-in", "-i", help="Reboot the system", nargs="*", metavar="Minutes") action.add_argument("--poweroff", "-p", help="Poweroff the system", nargs="*", metavar="Minutes|HH:MM") action.add_argument("--cancel", "-c", help="Cancel pending shutdown", action="store_true") action.add_argument("--check", help="Check pending shutdown", action="store_true") args = parser.parse_args() try: if args.reboot is not None: for r in args.reboot: if ':' not in r and '/' not in r and '.' not in r: print("Incorrect format! Use HH:MM") exit(1) execute_shutdown(args.reboot, reboot=True, ask=args.yes) if args.reboot_in is not None: for i in args.reboot_in: if ':' in i: print("Incorrect format! Use Minutes") exit(1) execute_shutdown(args.reboot_in, reboot=True, ask=args.yes) if args.poweroff is not None: execute_shutdown(args.poweroff, reboot=False, ask=args.yes) if args.cancel: cancel_shutdown() if args.check: check_shutdown() except KeyboardInterrupt: exit("Interrupted") if __name__ == "__main__": main()