update_manager: resolve potential rentry issues
Calls to "refresh" should not be allowed to occur while a client a current refresh is in progress. Updates will wait for a pending refresh to complete before beginning the update procedure. Signed-off-by: Eric Callahan <arksine.code@gmail.com>
This commit is contained in:
parent
ae3661b100
commit
1e81297624
|
@ -16,7 +16,7 @@ import time
|
||||||
import tornado.gen
|
import tornado.gen
|
||||||
from tornado.ioloop import IOLoop
|
from tornado.ioloop import IOLoop
|
||||||
from tornado.httpclient import AsyncHTTPClient
|
from tornado.httpclient import AsyncHTTPClient
|
||||||
from tornado.locks import Event
|
from tornado.locks import Event, Condition
|
||||||
|
|
||||||
MOONRAKER_PATH = os.path.normpath(os.path.join(
|
MOONRAKER_PATH = os.path.normpath(os.path.join(
|
||||||
os.path.dirname(__file__), "../.."))
|
os.path.dirname(__file__), "../.."))
|
||||||
|
@ -115,9 +115,11 @@ class UpdateManager:
|
||||||
async def _handle_status_request(self, web_request):
|
async def _handle_status_request(self, web_request):
|
||||||
refresh = web_request.get_boolean('refresh', False)
|
refresh = web_request.get_boolean('refresh', False)
|
||||||
vinfo = {}
|
vinfo = {}
|
||||||
for name, updater in self.updaters.items():
|
for name, updater in list(self.updaters.items()):
|
||||||
await updater.check_initialized(120.)
|
await updater.check_initialized(120.)
|
||||||
if refresh:
|
cur_update = self.current_update or ("", -1)
|
||||||
|
# Do not refresh if the current update is in progress
|
||||||
|
if refresh and name != cur_update[0]:
|
||||||
ret = updater.refresh()
|
ret = updater.refresh()
|
||||||
if asyncio.iscoroutine(ret):
|
if asyncio.iscoroutine(ret):
|
||||||
await ret
|
await ret
|
||||||
|
@ -291,6 +293,7 @@ class GitUpdater:
|
||||||
self.version = self.cur_hash = "?"
|
self.version = self.cur_hash = "?"
|
||||||
self.remote_version = self.remote_hash = "?"
|
self.remote_version = self.remote_hash = "?"
|
||||||
self.init_evt = Event()
|
self.init_evt = Event()
|
||||||
|
self.refresh_condition = None
|
||||||
self.debug = umgr.repo_debug
|
self.debug = umgr.repo_debug
|
||||||
self.remote = "origin"
|
self.remote = "origin"
|
||||||
self.branch = "master"
|
self.branch = "master"
|
||||||
|
@ -341,8 +344,19 @@ class GitUpdater:
|
||||||
await self.init_evt.wait(timeout)
|
await self.init_evt.wait(timeout)
|
||||||
|
|
||||||
async def refresh(self):
|
async def refresh(self):
|
||||||
|
if self.refresh_condition is None:
|
||||||
|
self.refresh_condition = Condition()
|
||||||
|
else:
|
||||||
|
self.refresh_condition.wait()
|
||||||
|
return
|
||||||
|
try:
|
||||||
await self._check_version()
|
await self._check_version()
|
||||||
|
except Exception:
|
||||||
|
logging.exception("Error Refreshing git state")
|
||||||
|
else:
|
||||||
self.init_evt.set()
|
self.init_evt.set()
|
||||||
|
self.refresh_condition.notify_all()
|
||||||
|
self.refresh_condition = None
|
||||||
|
|
||||||
async def _check_version(self, need_fetch=True):
|
async def _check_version(self, need_fetch=True):
|
||||||
self.is_valid = self.detached = False
|
self.is_valid = self.detached = False
|
||||||
|
@ -434,6 +448,9 @@ class GitUpdater:
|
||||||
f"{self.remote}/{self.branch}")
|
f"{self.remote}/{self.branch}")
|
||||||
|
|
||||||
async def update(self, update_deps=False):
|
async def update(self, update_deps=False):
|
||||||
|
await self.check_initialized(20.)
|
||||||
|
if self.refresh_condition is not None:
|
||||||
|
self.refresh_condition.wait()
|
||||||
if not self.is_valid:
|
if not self.is_valid:
|
||||||
raise self._log_exc("Update aborted, repo is not valid", False)
|
raise self._log_exc("Update aborted, repo is not valid", False)
|
||||||
if self.is_dirty:
|
if self.is_dirty:
|
||||||
|
@ -592,17 +609,20 @@ class PackageUpdater:
|
||||||
self.notify_update_response = umgr.notify_update_response
|
self.notify_update_response = umgr.notify_update_response
|
||||||
self.available_packages = []
|
self.available_packages = []
|
||||||
self.init_evt = Event()
|
self.init_evt = Event()
|
||||||
|
self.refresh_condition = None
|
||||||
IOLoop.current().spawn_callback(self.refresh)
|
IOLoop.current().spawn_callback(self.refresh)
|
||||||
|
|
||||||
async def refresh(self):
|
async def refresh(self):
|
||||||
# TODO: Use python-apt python lib rather than command line for updates
|
# TODO: Use python-apt python lib rather than command line for updates
|
||||||
|
if self.refresh_condition is None:
|
||||||
|
self.refresh_condition = Condition()
|
||||||
|
else:
|
||||||
|
self.refresh_condition.wait()
|
||||||
|
return
|
||||||
try:
|
try:
|
||||||
await self.execute_cmd(f"{APT_CMD} update", timeout=300.)
|
await self.execute_cmd(f"{APT_CMD} update", timeout=300.)
|
||||||
res = await self.execute_cmd_with_response(
|
res = await self.execute_cmd_with_response(
|
||||||
"apt list --upgradable")
|
"apt list --upgradable")
|
||||||
except Exception:
|
|
||||||
logging.exception("Error Refreshing System Packages")
|
|
||||||
else:
|
|
||||||
pkg_list = [p.strip() for p in res.split("\n") if p.strip()]
|
pkg_list = [p.strip() for p in res.split("\n") if p.strip()]
|
||||||
if pkg_list:
|
if pkg_list:
|
||||||
pkg_list = pkg_list[2:]
|
pkg_list = pkg_list[2:]
|
||||||
|
@ -612,7 +632,12 @@ class PackageUpdater:
|
||||||
logging.info(
|
logging.info(
|
||||||
f"Detected {len(self.available_packages)} package updates:"
|
f"Detected {len(self.available_packages)} package updates:"
|
||||||
f"\n{pkg_list}")
|
f"\n{pkg_list}")
|
||||||
|
except Exception:
|
||||||
|
logging.exception("Error Refreshing System Packages")
|
||||||
|
else:
|
||||||
self.init_evt.set()
|
self.init_evt.set()
|
||||||
|
self.refresh_condition.notify_all()
|
||||||
|
self.refresh_condition = None
|
||||||
|
|
||||||
async def check_initialized(self, timeout=None):
|
async def check_initialized(self, timeout=None):
|
||||||
if self.init_evt.is_set():
|
if self.init_evt.is_set():
|
||||||
|
@ -622,6 +647,9 @@ class PackageUpdater:
|
||||||
await self.init_evt.wait(timeout)
|
await self.init_evt.wait(timeout)
|
||||||
|
|
||||||
async def update(self, *args):
|
async def update(self, *args):
|
||||||
|
await self.check_initialized(20.)
|
||||||
|
if self.refresh_condition is not None:
|
||||||
|
self.refresh_condition.wait()
|
||||||
self.notify_update_response("Updating packages...")
|
self.notify_update_response("Updating packages...")
|
||||||
try:
|
try:
|
||||||
await self.execute_cmd(
|
await self.execute_cmd(
|
||||||
|
@ -651,6 +679,7 @@ class ClientUpdater:
|
||||||
self.version = self.remote_version = self.dl_url = "?"
|
self.version = self.remote_version = self.dl_url = "?"
|
||||||
self.etag = None
|
self.etag = None
|
||||||
self.init_evt = Event()
|
self.init_evt = Event()
|
||||||
|
self.refresh_condition = None
|
||||||
self._get_local_version()
|
self._get_local_version()
|
||||||
logging.info(f"\nInitializing Client Updater: '{self.name}',"
|
logging.info(f"\nInitializing Client Updater: '{self.name}',"
|
||||||
f"\nversion: {self.version}"
|
f"\nversion: {self.version}"
|
||||||
|
@ -672,9 +701,22 @@ class ClientUpdater:
|
||||||
await self.init_evt.wait(timeout)
|
await self.init_evt.wait(timeout)
|
||||||
|
|
||||||
async def refresh(self):
|
async def refresh(self):
|
||||||
# Local state
|
if self.refresh_condition is None:
|
||||||
|
self.refresh_condition = Condition()
|
||||||
|
else:
|
||||||
|
self.refresh_condition.wait()
|
||||||
|
return
|
||||||
|
try:
|
||||||
self._get_local_version()
|
self._get_local_version()
|
||||||
|
await self._get_remote_version()
|
||||||
|
except Exception:
|
||||||
|
logging.exception("Error Refreshing Client")
|
||||||
|
else:
|
||||||
|
self.init_evt.set()
|
||||||
|
self.refresh_condition.notify_all()
|
||||||
|
self.refresh_condition = None
|
||||||
|
|
||||||
|
async def _get_remote_version(self):
|
||||||
# Remote state
|
# Remote state
|
||||||
url = f"https://api.github.com/repos/{self.repo}/releases/latest"
|
url = f"https://api.github.com/repos/{self.repo}/releases/latest"
|
||||||
try:
|
try:
|
||||||
|
@ -684,7 +726,7 @@ class ClientUpdater:
|
||||||
result = {}
|
result = {}
|
||||||
if result is None:
|
if result is None:
|
||||||
# No change, update not necessary
|
# No change, update not necessary
|
||||||
return None
|
return
|
||||||
self.etag = result.get('etag', None)
|
self.etag = result.get('etag', None)
|
||||||
self.remote_version = result.get('name', "?")
|
self.remote_version = result.get('name', "?")
|
||||||
release_assets = result.get('assets', [{}])[0]
|
release_assets = result.get('assets', [{}])[0]
|
||||||
|
@ -694,9 +736,12 @@ class ClientUpdater:
|
||||||
f"Local Version: {self.version}\n"
|
f"Local Version: {self.version}\n"
|
||||||
f"Remote Version: {self.remote_version}\n"
|
f"Remote Version: {self.remote_version}\n"
|
||||||
f"url: {self.dl_url}")
|
f"url: {self.dl_url}")
|
||||||
self.init_evt.set()
|
|
||||||
|
|
||||||
async def update(self, *args):
|
async def update(self, *args):
|
||||||
|
await self.check_initialized(20.)
|
||||||
|
if self.refresh_condition is not None:
|
||||||
|
# wait for refresh if in progess
|
||||||
|
self.refresh_condition.wait()
|
||||||
if self.remote_version == "?":
|
if self.remote_version == "?":
|
||||||
await self.refresh()
|
await self.refresh()
|
||||||
if self.remote_version == "?":
|
if self.remote_version == "?":
|
||||||
|
|
Loading…
Reference in New Issue