2021-11-11 18:31:28 +00:00
|
|
|
#!/usr/bin/env python3
|
2021-11-16 16:14:01 +00:00
|
|
|
"""System management module"""
|
|
|
|
import subprocess
|
2021-11-22 16:50:50 +00:00
|
|
|
import pytz
|
2021-11-11 18:31:28 +00:00
|
|
|
from flask import Blueprint
|
2021-11-22 16:50:50 +00:00
|
|
|
from flask_restful import Resource, Api, reqparse
|
|
|
|
|
|
|
|
from selfprivacy_api.utils import WriteUserData, ReadUserData
|
2021-11-11 18:31:28 +00:00
|
|
|
|
|
|
|
api_system = Blueprint("system", __name__, url_prefix="/system")
|
|
|
|
api = Api(api_system)
|
|
|
|
|
2021-11-16 16:14:01 +00:00
|
|
|
|
2021-11-22 16:50:50 +00:00
|
|
|
class Timezone(Resource):
|
|
|
|
"""Change timezone of NixOS"""
|
|
|
|
|
|
|
|
def get(self):
|
|
|
|
"""
|
|
|
|
Get current system timezone
|
|
|
|
---
|
|
|
|
tags:
|
|
|
|
- System
|
|
|
|
security:
|
|
|
|
- bearerAuth: []
|
|
|
|
responses:
|
|
|
|
200:
|
|
|
|
description: Timezone
|
|
|
|
400:
|
|
|
|
description: Bad request
|
|
|
|
"""
|
|
|
|
with ReadUserData() as data:
|
|
|
|
if "timezone" not in data:
|
|
|
|
return "Europe/Uzhgorod"
|
|
|
|
return data["timezone"]
|
|
|
|
|
|
|
|
def put(self):
|
|
|
|
"""
|
|
|
|
Change system timezone
|
|
|
|
---
|
|
|
|
tags:
|
|
|
|
- System
|
|
|
|
security:
|
|
|
|
- bearerAuth: []
|
|
|
|
parameters:
|
|
|
|
- name: timezone
|
|
|
|
in: body
|
|
|
|
required: true
|
|
|
|
description: Timezone to set
|
|
|
|
schema:
|
|
|
|
type: object
|
|
|
|
required:
|
|
|
|
- timezone
|
|
|
|
properties:
|
|
|
|
timezone:
|
|
|
|
type: string
|
|
|
|
responses:
|
|
|
|
200:
|
|
|
|
description: Timezone changed
|
|
|
|
400:
|
|
|
|
description: Bad request
|
|
|
|
"""
|
|
|
|
parser = reqparse.RequestParser()
|
|
|
|
parser.add_argument("timezone", type=str, required=True)
|
|
|
|
timezone = parser.parse_args()["timezone"]
|
|
|
|
|
|
|
|
# Check if timezone is a valid tzdata string
|
|
|
|
if timezone not in pytz.all_timezones:
|
|
|
|
return {"error": "Invalid timezone"}, 400
|
|
|
|
|
|
|
|
with WriteUserData() as data:
|
|
|
|
data["timezone"] = timezone
|
|
|
|
return "Timezone changed"
|
|
|
|
|
|
|
|
|
|
|
|
class AutoUpgrade(Resource):
|
|
|
|
"""Enable/disable automatic upgrades and reboots"""
|
|
|
|
|
|
|
|
def get(self):
|
|
|
|
"""
|
|
|
|
Get current system autoupgrade settings
|
|
|
|
---
|
|
|
|
tags:
|
|
|
|
- System
|
|
|
|
security:
|
|
|
|
- bearerAuth: []
|
|
|
|
responses:
|
|
|
|
200:
|
|
|
|
description: Auto-upgrade settings
|
|
|
|
400:
|
|
|
|
description: Bad request
|
|
|
|
"""
|
|
|
|
with ReadUserData() as data:
|
|
|
|
if "autoUpgrade" not in data:
|
|
|
|
return {"enable": True, "allowReboot": False}
|
|
|
|
if "enable" not in data["autoUpgrade"]:
|
|
|
|
data["autoUpgrade"]["enable"] = True
|
|
|
|
if "allowReboot" not in data["autoUpgrade"]:
|
|
|
|
data["autoUpgrade"]["allowReboot"] = False
|
|
|
|
return data["autoUpgrade"]
|
|
|
|
|
|
|
|
def put(self):
|
|
|
|
"""
|
|
|
|
Change system auto upgrade settings
|
|
|
|
---
|
|
|
|
tags:
|
|
|
|
- System
|
|
|
|
security:
|
|
|
|
- bearerAuth: []
|
|
|
|
parameters:
|
|
|
|
- name: autoUpgrade
|
|
|
|
in: body
|
|
|
|
required: true
|
|
|
|
description: Auto upgrade settings
|
|
|
|
schema:
|
|
|
|
type: object
|
|
|
|
required:
|
|
|
|
- enable
|
|
|
|
- allowReboot
|
|
|
|
properties:
|
|
|
|
enable:
|
|
|
|
type: boolean
|
|
|
|
allowReboot:
|
|
|
|
type: boolean
|
|
|
|
responses:
|
|
|
|
200:
|
|
|
|
description: New settings saved
|
|
|
|
400:
|
|
|
|
description: Bad request
|
|
|
|
"""
|
|
|
|
parser = reqparse.RequestParser()
|
|
|
|
parser.add_argument("enable", type=bool, required=False)
|
|
|
|
parser.add_argument("allowReboot", type=bool, required=False)
|
|
|
|
args = parser.parse_args()
|
|
|
|
enable = args["enable"]
|
|
|
|
allow_reboot = args["allowReboot"]
|
|
|
|
|
|
|
|
with WriteUserData() as data:
|
|
|
|
if "autoUpgrade" not in data:
|
|
|
|
data["autoUpgrade"] = {}
|
|
|
|
if enable is not None:
|
|
|
|
data["autoUpgrade"]["enable"] = enable
|
|
|
|
if allow_reboot is not None:
|
|
|
|
data["autoUpgrade"]["allowReboot"] = allow_reboot
|
|
|
|
return "Auto-upgrade settings changed"
|
|
|
|
|
|
|
|
|
2021-11-11 18:31:28 +00:00
|
|
|
class RebuildSystem(Resource):
|
2021-11-16 16:14:01 +00:00
|
|
|
"""Rebuild NixOS"""
|
|
|
|
|
2021-11-11 18:31:28 +00:00
|
|
|
def get(self):
|
2021-11-16 16:14:01 +00:00
|
|
|
"""
|
|
|
|
Rebuild NixOS with nixos-rebuild switch
|
|
|
|
---
|
|
|
|
tags:
|
|
|
|
- System
|
|
|
|
security:
|
|
|
|
- bearerAuth: []
|
|
|
|
responses:
|
|
|
|
200:
|
|
|
|
description: System rebuild has started
|
|
|
|
401:
|
|
|
|
description: Unauthorized
|
|
|
|
"""
|
2021-11-17 10:36:26 +00:00
|
|
|
rebuild_result = subprocess.Popen(
|
|
|
|
["systemctl", "start", "sp-nixos-rebuild.service"], start_new_session=True
|
|
|
|
)
|
2021-11-16 16:14:01 +00:00
|
|
|
rebuild_result.communicate()[0]
|
|
|
|
return rebuild_result.returncode
|
2021-11-11 18:31:28 +00:00
|
|
|
|
|
|
|
|
|
|
|
class RollbackSystem(Resource):
|
2021-11-16 16:14:01 +00:00
|
|
|
"""Rollback NixOS"""
|
|
|
|
|
2021-11-11 18:31:28 +00:00
|
|
|
def get(self):
|
2021-11-16 16:14:01 +00:00
|
|
|
"""
|
|
|
|
Rollback NixOS with nixos-rebuild switch --rollback
|
|
|
|
---
|
|
|
|
tags:
|
|
|
|
- System
|
|
|
|
security:
|
|
|
|
- bearerAuth: []
|
|
|
|
responses:
|
|
|
|
200:
|
|
|
|
description: System rollback has started
|
|
|
|
401:
|
|
|
|
description: Unauthorized
|
|
|
|
"""
|
2021-11-17 10:36:26 +00:00
|
|
|
rollback_result = subprocess.Popen(
|
|
|
|
["systemctl", "start", "sp-nixos-rollback.service"], start_new_session=True
|
|
|
|
)
|
2021-11-16 16:14:01 +00:00
|
|
|
rollback_result.communicate()[0]
|
|
|
|
return rollback_result.returncode
|
2021-11-11 18:31:28 +00:00
|
|
|
|
|
|
|
|
|
|
|
class UpgradeSystem(Resource):
|
2021-11-16 16:14:01 +00:00
|
|
|
"""Upgrade NixOS"""
|
|
|
|
|
2021-11-11 18:31:28 +00:00
|
|
|
def get(self):
|
2021-11-16 16:14:01 +00:00
|
|
|
"""
|
|
|
|
Upgrade NixOS with nixos-rebuild switch --upgrade
|
|
|
|
---
|
|
|
|
tags:
|
|
|
|
- System
|
|
|
|
security:
|
|
|
|
- bearerAuth: []
|
|
|
|
responses:
|
|
|
|
200:
|
|
|
|
description: System upgrade has started
|
|
|
|
401:
|
|
|
|
description: Unauthorized
|
|
|
|
"""
|
2021-11-17 10:36:26 +00:00
|
|
|
upgrade_result = subprocess.Popen(
|
|
|
|
["systemctl", "start", "sp-nixos-upgrade.service"], start_new_session=True
|
|
|
|
)
|
2021-11-16 16:14:01 +00:00
|
|
|
upgrade_result.communicate()[0]
|
|
|
|
return upgrade_result.returncode
|
2021-11-11 18:31:28 +00:00
|
|
|
|
|
|
|
|
2021-11-17 10:36:26 +00:00
|
|
|
class RebootSystem(Resource):
|
|
|
|
"""Reboot the system"""
|
|
|
|
|
|
|
|
def get(self):
|
|
|
|
"""
|
|
|
|
Reboot the system
|
|
|
|
---
|
|
|
|
tags:
|
|
|
|
- System
|
|
|
|
security:
|
|
|
|
- bearerAuth: []
|
|
|
|
responses:
|
|
|
|
200:
|
|
|
|
description: System reboot has started
|
|
|
|
401:
|
|
|
|
description: Unauthorized
|
|
|
|
"""
|
|
|
|
subprocess.Popen(["reboot"], start_new_session=True)
|
|
|
|
return "System reboot has started"
|
|
|
|
|
|
|
|
|
2021-11-11 18:31:28 +00:00
|
|
|
class SystemVersion(Resource):
|
2021-11-16 16:14:01 +00:00
|
|
|
"""Get system version from uname"""
|
|
|
|
|
2021-11-11 18:31:28 +00:00
|
|
|
def get(self):
|
2021-11-16 16:14:01 +00:00
|
|
|
"""
|
|
|
|
Get system version from uname -a
|
|
|
|
---
|
|
|
|
tags:
|
|
|
|
- System
|
|
|
|
security:
|
|
|
|
- bearerAuth: []
|
|
|
|
responses:
|
|
|
|
200:
|
|
|
|
description: OK
|
|
|
|
401:
|
|
|
|
description: Unauthorized
|
|
|
|
"""
|
2021-11-11 18:31:28 +00:00
|
|
|
return {
|
|
|
|
"system_version": subprocess.check_output(["uname", "-a"])
|
|
|
|
.decode("utf-8")
|
|
|
|
.strip()
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
class PythonVersion(Resource):
|
2021-11-16 16:14:01 +00:00
|
|
|
"""Get python version"""
|
|
|
|
|
2021-11-11 18:31:28 +00:00
|
|
|
def get(self):
|
2021-11-16 16:14:01 +00:00
|
|
|
"""
|
|
|
|
Get python version used by this API
|
|
|
|
---
|
|
|
|
tags:
|
|
|
|
- System
|
|
|
|
security:
|
|
|
|
- bearerAuth: []
|
|
|
|
responses:
|
|
|
|
200:
|
|
|
|
description: OK
|
|
|
|
401:
|
|
|
|
description: Unauthorized
|
|
|
|
"""
|
2021-11-11 18:31:28 +00:00
|
|
|
return subprocess.check_output(["python", "-V"]).decode("utf-8").strip()
|
|
|
|
|
|
|
|
|
2021-11-22 16:50:50 +00:00
|
|
|
api.add_resource(Timezone, "/configuration/timezone")
|
|
|
|
api.add_resource(AutoUpgrade, "/configuration/autoUpgrade")
|
2021-11-11 18:31:28 +00:00
|
|
|
api.add_resource(RebuildSystem, "/configuration/apply")
|
|
|
|
api.add_resource(RollbackSystem, "/configuration/rollback")
|
2021-11-17 09:36:39 +00:00
|
|
|
api.add_resource(UpgradeSystem, "/configuration/upgrade")
|
2021-11-17 10:36:26 +00:00
|
|
|
api.add_resource(RebootSystem, "/reboot")
|
2021-11-11 18:31:28 +00:00
|
|
|
api.add_resource(SystemVersion, "/version")
|
|
|
|
api.add_resource(PythonVersion, "/pythonVersion")
|