mynode/rootfs/standard/var/www/mynode/device_info.py

767 lines
22 KiB
Python
Raw Normal View History

2019-06-15 23:02:44 +00:00
from config import *
2019-11-04 04:50:47 +00:00
from threading import Timer
2020-05-13 01:54:17 +00:00
from werkzeug.routing import RequestRedirect
from flask import flash
2019-11-14 02:32:37 +00:00
import time
2020-07-19 03:16:51 +00:00
import json
2019-06-15 23:02:44 +00:00
import os
import subprocess
2020-08-01 03:51:59 +00:00
import random
import string
2019-06-15 23:02:44 +00:00
2019-08-27 01:36:15 +00:00
# Globals
local_ip = "unknown"
cached_data = {}
2019-08-27 01:36:15 +00:00
#==================================
# Utilities
#==================================
def get_file_contents(filename):
contents = "UNKNOWN"
try:
with open(filename, "r") as f:
contents = f.read().strip()
except:
contents = "ERROR"
return contents
def set_file_contents(filename, data):
try:
with open(filename, "w") as f:
f.write(data)
os.system("sync")
return True
except:
return False
return False
#==================================
# Manage Device
#==================================
def reboot_device():
os.system("sync")
os.system("/usr/bin/mynode_stop_critical_services.sh")
os.system("reboot")
def shutdown_device():
os.system("sync")
os.system("/usr/bin/mynode_stop_critical_services.sh")
os.system("shutdown -h now")
def factory_reset():
# Reset subsystems that have local data
delete_quicksync_data()
# Delete LND data
delete_lnd_data()
# Delete Tor data
reset_tor()
# Disable services
os.system("systemctl disable electrs --no-pager")
os.system("systemctl disable lndhub --no-pager")
os.system("systemctl disable btc_rpc_explorer --no-pager")
os.system("systemctl disable vpn --no-pager")
# Trigger drive to be reformatted on reboot
os.system("rm -f /mnt/hdd/.mynode")
# Reset password
os.system("/usr/bin/mynode_chpasswd.sh bolt")
# Reboot
reboot_device()
2020-05-13 01:54:17 +00:00
def check_and_mark_reboot_action(tmp_marker):
if os.path.isfile("/tmp/{}".format(tmp_marker)):
flash(u'Refresh prevented - action already triggered', category="error")
raise RequestRedirect("/")
os.system("touch /tmp/{}".format(tmp_marker))
#==================================
# Manage Versions and Upgrades
#==================================
2019-06-15 23:02:44 +00:00
def get_current_version():
current_version = "0.0"
try:
with open("/usr/share/mynode/version", "r") as f:
current_version = f.read().strip()
except:
current_version = "error"
return current_version
2020-02-24 01:39:40 +00:00
def get_current_beta_version():
current_beta_version = "0.0"
try:
with open("/usr/share/mynode/beta_version", "r") as f:
current_beta_version = f.read().strip()
except:
2020-02-24 02:35:08 +00:00
current_beta_version = "beta_not_installed"
2020-02-24 01:39:40 +00:00
return current_beta_version
2019-06-15 23:02:44 +00:00
def update_latest_version():
2020-07-24 18:02:15 +00:00
os.system("/usr/bin/mynode_get_latest_version.sh")
2019-06-15 23:02:44 +00:00
return True
def get_latest_version():
latest_version = "0.0"
try:
with open("/usr/share/mynode/latest_version", "r") as f:
latest_version = f.read().strip()
if latest_version == "":
latest_version = get_current_version()
2019-06-15 23:02:44 +00:00
except:
latest_version = get_current_version()
return latest_version
2020-02-24 01:39:40 +00:00
def get_latest_beta_version():
beta_version = ""
try:
with open("/usr/share/mynode/latest_beta_version", "r") as f:
beta_version = f.read().strip()
except:
beta_version = ""
return beta_version
2020-04-18 04:00:48 +00:00
def mark_upgrade_started():
os.system("touch /tmp/upgrade_started")
2020-01-26 03:39:55 +00:00
os.system("sync")
2020-04-18 04:00:48 +00:00
def is_upgrade_running():
return os.path.isfile("/tmp/upgrade_started")
def reinstall_app(app):
if not is_upgrade_running():
mark_upgrade_started()
# Upgrade
os.system("mkdir -p /home/admin/upgrade_logs")
cmd = "/usr/bin/mynode_reinstall_app.sh {} > /home/admin/upgrade_logs/reinstall_{}.txt 2>&1".format(app,app)
2020-04-18 04:00:48 +00:00
subprocess.call(cmd, shell=True)
# Sync
os.system("sync")
time.sleep(1)
# Reboot
reboot_device()
2020-01-26 03:39:55 +00:00
def upgrade_device():
2020-04-18 04:00:48 +00:00
if not is_upgrade_running():
mark_upgrade_started()
2019-06-15 23:02:44 +00:00
2020-04-18 04:00:48 +00:00
# Upgrade
os.system("mkdir -p /home/admin/upgrade_logs")
cmd = "/usr/bin/mynode_upgrade.sh > /home/admin/upgrade_logs/upgrade_log_from_{}_upgrade.txt 2>&1".format(get_current_version())
subprocess.call(cmd, shell=True)
# Sync
os.system("sync")
time.sleep(1)
2020-01-10 05:41:12 +00:00
2020-04-18 04:00:48 +00:00
# Reboot
reboot_device()
2020-02-24 01:39:40 +00:00
2020-04-18 04:00:48 +00:00
def upgrade_device_beta():
if not is_upgrade_running():
mark_upgrade_started()
# Upgrade
os.system("mkdir -p /home/admin/upgrade_logs")
cmd = "/usr/bin/mynode_upgrade.sh beta > /home/admin/upgrade_logs/upgrade_log_from_{}_upgrade.txt 2>&1".format(get_current_version())
subprocess.call(cmd, shell=True)
# Sync
os.system("sync")
time.sleep(1)
# Reboot
reboot_device()
2020-02-24 01:39:40 +00:00
2019-11-16 20:46:00 +00:00
def did_upgrade_fail():
return os.path.isfile("/mnt/hdd/mynode/settings/upgrade_error")
def get_recent_upgrade_logs():
logs=""
current_version = get_current_version()
for i in range(1,6):
filename = "/home/admin/upgrade_logs/upgrade_log_{}_post_{}.txt".format(current_version, i)
try:
with open(filename, "r") as f:
logs = logs + "===========================================================\n"
logs = logs + "=== Upgrade Attempt #{}\n".format(i)
logs = logs + "===========================================================\n\n\n"
logs = logs + f.read().decode("utf8")
except:
pass
return logs
def has_checkin_error():
return os.path.isfile("/tmp/check_in_error")
2020-05-01 01:37:17 +00:00
#==================================
# Reseller Info
#==================================
def is_device_from_reseller():
return os.path.isfile("/opt/mynode/custom/reseller")
2019-11-16 20:46:00 +00:00
#==================================
# Device Info
#==================================
2019-06-29 15:53:49 +00:00
def get_system_uptime():
uptime = subprocess.check_output('awk \'{print int($1/86400)" days "int($1%86400/3600)" hour(s) "int(($1%3600)/60)" minute(s) "int($1%60)" seconds(s)"}\' /proc/uptime', shell=True)
uptime = uptime.strip()
return uptime
2020-04-05 02:25:52 +00:00
def get_system_uptime_in_seconds():
uptime = subprocess.check_output('awk \'{print $1}\' /proc/uptime', shell=True)
uptime = int(float(uptime.strip()))
return uptime
2020-02-05 02:12:52 +00:00
def get_system_date():
date = subprocess.check_output('date', shell=True)
date = date.strip()
return date
2019-06-15 23:02:44 +00:00
def get_device_serial():
global cached_data
if "serial" in cached_data:
return cached_data["serial"]
2020-07-10 03:57:17 +00:00
serial = subprocess.check_output("mynode-get-device-serial", shell=True)
2019-06-15 23:02:44 +00:00
serial = serial.strip()
cached_data["serial"] = serial
2019-06-15 23:02:44 +00:00
return serial
2019-07-16 03:30:08 +00:00
def get_device_type():
global cached_data
if "device_type" in cached_data:
return cached_data["device_type"]
device = subprocess.check_output("mynode-get-device-type", shell=True).strip()
cached_data["device_type"] = device
2019-10-30 02:22:11 +00:00
return device
2019-07-16 03:30:08 +00:00
2020-04-17 02:19:11 +00:00
def get_device_ram():
global cached_data
if "ram" in cached_data:
return cached_data["ram"]
ram = subprocess.check_output("free --giga | grep Mem | awk '{print $2}'", shell=True).strip()
cached_data["ram"] = ram
2020-04-17 02:19:11 +00:00
return ram
def get_local_ip():
local_ip = "unknown"
try:
local_ip = subprocess.check_output('/usr/bin/get_local_ip.py', shell=True).strip()
except:
local_ip = "error"
return local_ip
def get_device_changelog():
changelog = ""
try:
changelog = subprocess.check_output(["cat", "/usr/share/mynode/changelog"])
except:
changelog = "ERROR"
return changelog
def has_changed_password():
2020-03-25 01:07:25 +00:00
try:
with open("/home/bitcoin/.mynode/.hashedpw", "r") as f:
hashedpw = f.read().strip()
if hashedpw != "d0b3cba71f725563d316ea3516099328042095d10f4571be25c07f9ce31985a5":
return True
except:
return False
return False
2020-04-23 03:47:50 +00:00
def is_mount_read_only(mnt):
with open('/proc/mounts') as f:
for line in f:
device, mount_point, filesystem, flags, __, __ = line.split()
flags = flags.split(',')
if mount_point == mnt:
return 'ro' in flags
return False
def set_swap_size(size):
size_mb = int(size) * 1024
os.system("sed -i 's|CONF_SWAPSIZE=.*|CONF_SWAPSIZE={}|' /etc/dphys-swapfile".format(size_mb))
return set_file_contents("/mnt/hdd/mynode/settings/swap_size", size)
def get_swap_size():
return get_file_contents("/mnt/hdd/mynode/settings/swap_size")
2019-07-16 03:30:08 +00:00
#==================================
# Service Status, Enabled, Logs, etc...
#==================================
def is_service_enabled(service_name):
cmd = "systemctl is-enabled {}".format(service_name)
try:
subprocess.check_call(cmd, shell=True)
return True
except:
return False
return False
def get_service_status_code(service_name):
code = os.system("systemctl status {} --no-pager".format(service_name))
return code
def get_service_status_basic_text(service_name):
if not is_service_enabled(service_name):
return "Disabled"
code = os.system("systemctl status {} --no-pager".format(service_name))
if code == 0:
return "Running"
return "Error"
def get_service_status_color(service_name):
if not is_service_enabled(service_name):
return "gray"
code = os.system("systemctl status {} --no-pager".format(service_name))
if code == 0:
return "green"
return "red"
def get_journalctl_log(service_name):
try:
2020-02-23 21:55:05 +00:00
log = subprocess.check_output("journalctl -r --unit={} --no-pager | head -n 200".format(service_name), shell=True).decode("utf8")
except:
log = "ERROR"
return log
2020-07-19 03:16:51 +00:00
#==================================
# UI Functions
#==================================
def read_ui_settings():
ui_hdd_file = '/mnt/hdd/mynode/settings/ui.json'
ui_mynode_file = '/home/bitcoin/.mynode/ui.json'
# read ui.json from HDD
if os.path.isfile(ui_hdd_file):
with open(ui_hdd_file, 'r') as fp:
ui_settings = json.load(fp)
# read ui.json from mynode
elif os.path.isfile(ui_mynode_file):
with open(ui_mynode_file, 'r') as fp:
ui_settings = json.load(fp)
# if ui.json is not found anywhere, use default settings
else:
ui_settings = {'darkmode': False}
# Set reseller
ui_settings["reseller"] = is_device_from_reseller()
return ui_settings
def write_ui_settings(ui_settings):
ui_hdd_file = '/mnt/hdd/mynode/settings/ui.json'
ui_mynode_file = '/home/bitcoin/.mynode/ui.json'
try:
with open(ui_hdd_file, 'w') as fp:
json.dump(ui_settings, fp)
except:
pass
with open(ui_mynode_file, 'w') as fp:
json.dump(ui_settings, fp)
def is_darkmode_enabled():
ui_settings = read_ui_settings()
return ui_settings['darkmode']
def disable_darkmode():
ui_settings = read_ui_settings()
ui_settings['darkmode'] = False
write_ui_settings(ui_settings)
def enable_darkmode():
ui_settings = read_ui_settings()
ui_settings['darkmode'] = True
write_ui_settings(ui_settings)
def is_https_forced():
return os.path.isfile('/home/bitcoin/.mynode/https_forced')
def force_https(force):
if force:
os.system("touch /home/bitcoin/.mynode/https_forced")
else:
os.system("rm -f /home/bitcoin/.mynode/https_forced")
2020-08-01 03:51:59 +00:00
def get_flask_secret_key():
if os.path.isfile("/home/bitcoin/.mynode/flask_secret_key"):
key = get_file_contents("/home/bitcoin/.mynode/flask_secret_key")
else:
letters = string.ascii_letters
key = ''.join(random.choice(letters) for i in range(32))
set_file_contents("/home/bitcoin/.mynode/flask_secret_key", key)
return key
#==================================
# Uploader Functions
#==================================
2019-07-25 02:51:12 +00:00
def is_uploader():
2020-03-14 21:01:21 +00:00
return os.path.isfile("/mnt/hdd/mynode/settings/uploader")
def set_uploader():
os.system("touch /mnt/hdd/mynode/settings/uploader")
def unset_uploader():
os.system("rm -rf /mnt/hdd/mynode/settings/uploader")
2019-07-25 02:51:12 +00:00
#==================================
# QuickSync Functions
#==================================
def is_quicksync_enabled():
2019-10-24 04:18:46 +00:00
return not os.path.isfile("/mnt/hdd/mynode/settings/quicksync_disabled")
def disable_quicksync():
os.system("touch /mnt/hdd/mynode/settings/quicksync_disabled")
def enable_quicksync():
os.system("rm -rf /mnt/hdd/mynode/settings/quicksync_disabled")
def settings_disable_quicksync():
stop_bitcoind()
stop_quicksync()
disable_quicksync()
delete_quicksync_data()
reboot_device()
def settings_enable_quicksync():
stop_bitcoind()
stop_quicksync()
enable_quicksync()
delete_quicksync_data()
reboot_device()
def delete_quicksync_data():
os.system("rm -rf /mnt/hdd/mynode/quicksync")
os.system("rm -rf /home/bitcoin/.config/transmission") # Old dir
os.system("rm -rf /mnt/hdd/mynode/.config/transmission")
def stop_quicksync():
os.system("systemctl stop quicksync")
def restart_quicksync():
2020-08-21 18:31:47 +00:00
os.system('echo "quicksync_reset" > /tmp/.mynode_status')
stop_bitcoind()
stop_quicksync()
delete_bitcoin_data()
delete_quicksync_data()
enable_quicksync()
reboot_device()
#==================================
# Product Key Functions
#==================================
2019-06-15 23:02:44 +00:00
def set_skipped_product_key():
os.system("touch /home/bitcoin/.mynode/.product_key_skipped")
os.system("touch /mnt/hdd/mynode/settings/.product_key_skipped")
2019-06-15 23:02:44 +00:00
def unset_skipped_product_key():
os.system("rm -rf /home/bitcoin/.mynode/.product_key_skipped")
os.system("rm -rf /mnt/hdd/mynode/settings/.product_key_skipped")
2019-06-15 23:02:44 +00:00
def skipped_product_key():
return os.path.isfile("/home/bitcoin/.mynode/.product_key_skipped") or \
os.path.isfile("/mnt/hdd/mynode/settings/.product_key_skipped")
def is_community_edition():
return skipped_product_key()
2019-06-15 23:02:44 +00:00
def delete_product_key():
os.system("rm -rf /home/bitcoin/.mynode/.product_key")
os.system("rm -rf /mnt/hdd/mynode/settings/.product_key")
2019-06-15 23:02:44 +00:00
def has_product_key():
return os.path.isfile("/home/bitcoin/.mynode/.product_key")
def get_product_key():
product_key = "no_product_key"
if skipped_product_key():
return "community_edition"
if not has_product_key():
return "product_key_missing"
try:
with open("/home/bitcoin/.mynode/.product_key", "r") as f:
product_key = f.read().strip()
except:
product_key = "product_key_error"
return product_key
def is_valid_product_key():
return not os.path.isfile("/home/bitcoin/.mynode/.product_key_error")
def save_product_key(product_key):
pk = product_key.replace("-","")
os.system("echo '{}' > /home/bitcoin/.mynode/.product_key".format(pk))
os.system("echo '{}' > /mnt/hdd/mynode/settings/.product_key".format(pk))
2019-06-15 23:02:44 +00:00
def delete_product_key_error():
os.system("rm -rf /home/bitcoin/.mynode/.product_key_error")
os.system("rm -rf /mnt/hdd/mynode/settings/.product_key_error")
#==================================
# Drive Repair Functions
#==================================
2019-12-22 03:27:14 +00:00
def is_drive_being_repaired():
return os.path.isfile("/tmp/repairing_drive")
2019-12-12 01:34:34 +00:00
def has_fsck_error():
return os.path.isfile("/tmp/fsck_error")
2020-05-01 17:47:41 +00:00
def clear_fsck_error():
os.system("rm -f /tmp/fsck_error")
2019-12-12 01:34:34 +00:00
def get_fsck_results():
try:
with open("/tmp/fsck_results", "r") as f:
return f.read()
except:
return "ERROR"
return "ERROR"
2019-08-27 01:36:15 +00:00
2020-04-07 17:36:30 +00:00
def has_sd_rw_error():
return os.path.isfile("/tmp/sd_rw_error")
#==================================
# Docker Functions
#==================================
def is_installing_docker_images():
return os.path.isfile("/tmp/installing_docker_images")
def get_docker_image_build_status():
status_code = get_service_status_code("docker_images")
if status_code != 0:
return "Failed... Retrying Later"
if is_installing_docker_images():
return "Installing..."
else:
return "Installation Complete"
return "Unknown"
2019-10-20 03:20:03 +00:00
def get_docker_image_build_status_color():
status_code = get_service_status_code("docker_images")
if status_code != 0:
return "red"
return "green"
2019-10-26 16:32:06 +00:00
2020-03-24 17:19:18 +00:00
def reset_docker():
# Delete docker data
os.system("touch /home/bitcoin/reset_docker")
# Reset marker files
os.system("rm -f /mnt/hdd/mynode/settings/webssh2_url")
os.system("rm -f /mnt/hdd/mynode/settings/mempoolspace_url")
os.system("rm -f /mnt/hdd/mynode/settings/dojo_url")
2020-03-25 03:28:42 +00:00
# Delete Dojo files
os.system("rm -rf /opt/mynode/.dojo")
os.system("rm -rf /opt/mynode/dojo")
2020-03-24 17:19:18 +00:00
os.system("sync")
reboot_device()
def get_docker_running_containers():
containers = []
try:
text = subprocess.check_output("docker ps --format '{{.Names}}'", shell=True).decode("utf8")
containers = text.splitlines()
except:
containers = ["ERROR"]
return containers
#==================================
# Bitcoin Functions
#==================================
2019-10-26 16:32:06 +00:00
def get_bitcoin_rpc_password():
try:
with open("/mnt/hdd/mynode/settings/.btcrpcpw", "r") as f:
return f.read()
except:
return "ERROR"
2019-11-04 04:50:47 +00:00
return "ERROR"
def stop_bitcoind():
os.system("systemctl stop bitcoind")
def reset_bitcoin_env_file():
os.system("echo 'BTCARGS=' > "+BITCOIN_ENV_FILE)
def delete_bitcoin_data():
os.system("rm -rf /mnt/hdd/mynode/bitcoin")
os.system("rm -rf /mnt/hdd/mynode/quicksync/.quicksync_complete")
os.system("rm -rf /mnt/hdd/mynode/settings/.btcrpc_environment")
os.system("rm -rf /mnt/hdd/mynode/settings/.btcrpcpw")
def reset_blockchain():
stop_bitcoind()
delete_bitcoin_data()
reboot_device()
2019-11-04 04:50:47 +00:00
#==================================
# LND Functions
#==================================
def stop_lnd():
os.system("systemctl stop lnd")
def restart_lnd():
os.system("systemctl restart lnd")
2019-11-04 04:50:47 +00:00
def delete_lnd_data():
#os.system("rm -f "+LND_WALLET_FILE)
os.system("rm -rf "+LND_DATA_FOLDER)
os.system("rm -rf /home/bitcoin/.lnd-admin/credentials.json")
os.system("rm -rf /mnt/hdd/mynode/settings/.lndpw")
os.system("rm -rf /home/admin/.lnd/")
return True
#==================================
# Electrum Server Functions
#==================================
def stop_electrs():
os.system("systemctl stop electrs")
def delete_electrs_data():
os.system("rm -rf /mnt/hdd/mynode/electrs")
def reset_electrs():
stop_electrs()
delete_electrs_data()
reboot_device()
#==================================
# Tor Functions
#==================================
2019-11-04 04:50:47 +00:00
def reset_tor():
os.system("rm -rf /var/lib/tor/*")
os.system("rm -rf /mnt/hdd/mynode/bitcoin/onion_private_key")
os.system("rm -rf /mnt/hdd/mynode/lnd/v2_onion_private_key")
2020-03-10 02:10:37 +00:00
os.system("rm -rf /mnt/hdd/mynode/lnd/v3_onion_private_key")
2020-01-29 01:59:46 +00:00
def is_btc_lnd_tor_enabled():
return os.path.isfile("/mnt/hdd/mynode/settings/.btc_lnd_tor_enabled")
def enable_btc_lnd_tor():
2020-05-11 03:43:22 +00:00
os.system("touch /mnt/hdd/mynode/settings/.btc_lnd_tor_enabled")
2020-01-29 01:59:46 +00:00
os.system("sync")
def disable_btc_lnd_tor():
2020-05-11 03:43:22 +00:00
os.system("rm -f /mnt/hdd/mynode/settings/.btc_lnd_tor_enabled")
os.system("sync")
def is_aptget_tor_enabled():
return os.path.isfile("/mnt/hdd/mynode/settings/torify_apt_get")
def enable_aptget_tor():
os.system("touch /mnt/hdd/mynode/settings/torify_apt_get")
os.system("sync")
def disable_aptget_tor():
os.system("rm -f /mnt/hdd/mynode/settings/torify_apt_get")
2020-02-14 00:47:51 +00:00
os.system("sync")
2020-03-10 02:10:37 +00:00
def get_onion_url_ssh():
try:
if os.path.isfile("/var/lib/tor/mynode_ssh/hostname"):
with open("/var/lib/tor/mynode_ssh/hostname") as f:
return f.read()
except:
pass
return "error"
2020-03-09 04:20:45 +00:00
def get_onion_url_general():
try:
if os.path.isfile("/var/lib/tor/mynode/hostname"):
with open("/var/lib/tor/mynode/hostname") as f:
2020-03-15 22:47:15 +00:00
return f.read().strip()
2020-03-09 04:20:45 +00:00
except:
pass
return "error"
def get_onion_url_btc():
try:
if os.path.isfile("/var/lib/tor/mynode_btc/hostname"):
with open("/var/lib/tor/mynode_btc/hostname") as f:
2020-03-15 22:47:15 +00:00
return f.read().strip()
2020-03-09 04:20:45 +00:00
except:
pass
return "error"
def get_onion_url_lnd():
try:
if os.path.isfile("/var/lib/tor/mynode_lnd/hostname"):
with open("/var/lib/tor/mynode_lnd/hostname") as f:
2020-03-15 22:47:15 +00:00
return f.read().strip()
2020-03-09 04:20:45 +00:00
except:
pass
return "error"
2020-03-10 02:10:37 +00:00
def get_onion_url_electrs():
try:
if os.path.isfile("/var/lib/tor/mynode_electrs/hostname"):
with open("/var/lib/tor/mynode_electrs/hostname") as f:
2020-03-15 22:47:15 +00:00
return f.read().strip()
2020-03-10 02:10:37 +00:00
except:
pass
return "error"
2020-05-11 03:16:02 +00:00
def get_onion_url_btcpay():
try:
if os.path.isfile("/var/lib/tor/mynode_btcpay/hostname"):
with open("/var/lib/tor/mynode_btcpay/hostname") as f:
return f.read().strip()
except:
pass
return "error"
def get_onion_info_btc_v2():
info = {}
info["url"] = "unknown"
info["pass"] = "unknown"
try:
if os.path.isfile("/var/lib/tor/mynode_btc_v2/hostname"):
with open("/var/lib/tor/mynode_btc_v2/hostname") as f:
content = f.read().strip()
parts = content.split(" ")
info["url"] = parts[0]
info["pass"] = parts[1]
return info
except:
pass
return info
2020-06-10 02:10:32 +00:00
def get_tor_version():
global cached_data
if "tor_version" in cached_data:
return cached_data["tor_version"]
cached_data["tor_version"] = subprocess.check_output("tor --version | egrep -o '[0-9\\.]+'", shell=True).strip().strip(".")
return cached_data["tor_version"]
2020-02-14 00:47:51 +00:00
#==================================
# Firewall Functions
#==================================
def reload_firewall():
os.system("ufw reload")
def get_firewall_rules():
try:
rules = subprocess.check_output("ufw status", shell=True).decode("utf8")
except:
rules = "ERROR"
return rules