From 2ca2c4172c88acd37cefcf00d29600c046366113 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Maximilian=20Schm=C3=B6cker?= Date: Fri, 4 Oct 2024 10:35:30 +0200 Subject: [PATCH 1/7] feat: history cleanup --- .../thymis_controller/models/__init__.py | 2 ++ .../thymis_controller/models/history.py | 21 +++++++++++++++++++ controller/thymis_controller/project.py | 20 +++++++++++------- controller/thymis_controller/routers/api.py | 9 ++++++-- frontend/src/lib/history.ts | 13 ++++++++++++ .../(authenticated)/history/+page.svelte | 3 ++- .../routes/(authenticated)/history/+page.ts | 12 ++--------- .../history/RollbackModal.svelte | 3 ++- 8 files changed, 61 insertions(+), 22 deletions(-) create mode 100644 controller/thymis_controller/models/history.py create mode 100644 frontend/src/lib/history.ts diff --git a/controller/thymis_controller/models/__init__.py b/controller/thymis_controller/models/__init__.py index 9d8e7116..5953f513 100644 --- a/controller/thymis_controller/models/__init__.py +++ b/controller/thymis_controller/models/__init__.py @@ -1,4 +1,5 @@ from .device import * +from .history import * from .module import * from .state import * from .task import * @@ -10,6 +11,7 @@ + task.__all__ # pylint: disable=undefined-variable + web_session.__all__ # pylint: disable=undefined-variable + device.__all__ # pylint: disable=undefined-variable + + history.__all__ # pylint: disable=undefined-variable ) # See https://stackoverflow.com/questions/60440945/correct-way-to-re-export-modules-from-init-py diff --git a/controller/thymis_controller/models/history.py b/controller/thymis_controller/models/history.py new file mode 100644 index 00000000..f92444ed --- /dev/null +++ b/controller/thymis_controller/models/history.py @@ -0,0 +1,21 @@ +import datetime +from typing import List + +from pydantic import BaseModel + + +class Commit(BaseModel): + SHA: str + SHA1: str + message: str + date: datetime.datetime + author: str + state_diff: List[str] + + +class Remote(BaseModel): + name: str + url: str + + +__all__ = ["Commit", "Remote"] diff --git a/controller/thymis_controller/project.py b/controller/thymis_controller/project.py index 5c865968..18dec1bd 100644 --- a/controller/thymis_controller/project.py +++ b/controller/thymis_controller/project.py @@ -16,6 +16,7 @@ from sqlalchemy.orm import Session from thymis_controller import crud, migration, models, task from thymis_controller.config import global_settings +from thymis_controller.models import history from thymis_controller.models.state import State from thymis_controller.nix import NIX_CMD, get_input_out_path, render_flake_nix @@ -241,20 +242,20 @@ def commit(self, summary: str): def get_history(self): return [ - { - "message": commit.message, - "author": commit.author.name, - "date": commit.authored_datetime, - "SHA": commit.hexsha, - "SHA1": self.repo.git.rev_parse(commit.hexsha, short=True), - "state_diff": self.repo.git.diff( + history.Commit( + message=commit.message, + author=commit.author.name, + date=commit.authored_datetime, + SHA=commit.hexsha, + SHA1=self.repo.git.rev_parse(commit.hexsha, short=True), + state_diff=self.repo.git.diff( commit.hexsha, commit.parents[0].hexsha if len(commit.parents) > 0 else None, "-R", "state.json", unified=5, ).split("\n")[4:], - } + ) for commit in self.repo.iter_commits() ] @@ -293,6 +294,9 @@ def clone_state_device( state.devices.append(new_device) self.write_state_and_reload(state) + def get_remotes(self): + return [history.Remote(name=r.name, url=r.url) for r in self.repo.remotes] + def create_build_task(self): return task.global_task_controller.add_task(task.BuildProjectTask(self.path)) diff --git a/controller/thymis_controller/routers/api.py b/controller/thymis_controller/routers/api.py index 87acb3e9..69530702 100644 --- a/controller/thymis_controller/routers/api.py +++ b/controller/thymis_controller/routers/api.py @@ -145,12 +145,12 @@ def download_image( raise HTTPException(status_code=404, detail="Image not found") -@router.get("/history") +@router.get("/history", tags=["history"]) def get_history(project: project.Project = Depends(get_project)): return project.get_history() -@router.post("/history/revert-commit") +@router.post("/history/revert-commit", tags=["history"]) def revert_commit( commit_sha: str, project: project.Project = Depends(get_project), @@ -159,6 +159,11 @@ def revert_commit( return {"message": "reverted commit"} +@router.get("/history/remotes", tags=["history"]) +def get_remotes(project: project.Project = Depends(get_project)): + return project.get_remotes() + + @router.post("/action/update") async def update( project: project.Project = Depends(get_project), diff --git a/frontend/src/lib/history.ts b/frontend/src/lib/history.ts new file mode 100644 index 00000000..40ee0ee8 --- /dev/null +++ b/frontend/src/lib/history.ts @@ -0,0 +1,13 @@ +export type Commit = { + message: string; + author: string; + date: string; + SHA: string; + SHA1: string; + state_diff: string[]; +}; + +export type Remote = { + name: string; + url: string; +}; diff --git a/frontend/src/routes/(authenticated)/history/+page.svelte b/frontend/src/routes/(authenticated)/history/+page.svelte index bc33592c..188c93be 100644 --- a/frontend/src/routes/(authenticated)/history/+page.svelte +++ b/frontend/src/routes/(authenticated)/history/+page.svelte @@ -5,10 +5,11 @@ import DeployActions from '$lib/components/DeployActions.svelte'; import Undo from 'lucide-svelte/icons/undo-2'; import RollbackModal from './RollbackModal.svelte'; + import type { Commit } from '$lib/history'; export let data: PageData; - let revertCommit: { SHA1: string; message: string } | undefined; + let revertCommit: Commit | undefined; const lineColor = (line: string) => { if (line.startsWith('+')) { diff --git a/frontend/src/routes/(authenticated)/history/+page.ts b/frontend/src/routes/(authenticated)/history/+page.ts index e4447140..caf95666 100644 --- a/frontend/src/routes/(authenticated)/history/+page.ts +++ b/frontend/src/routes/(authenticated)/history/+page.ts @@ -1,3 +1,4 @@ +import type { Commit } from '$lib/history'; import type { PageLoad } from './$types'; export const load = (async ({ fetch }) => { @@ -8,15 +9,6 @@ export const load = (async ({ fetch }) => { } }); return { - history: response.json() as Promise< - { - message: string; - author: string; - date: string; - SHA: string; - SHA1: string; - state_diff: string[]; - }[] - > + history: response.json() as Promise }; }) satisfies PageLoad; diff --git a/frontend/src/routes/(authenticated)/history/RollbackModal.svelte b/frontend/src/routes/(authenticated)/history/RollbackModal.svelte index a28c90a3..b1a633a9 100644 --- a/frontend/src/routes/(authenticated)/history/RollbackModal.svelte +++ b/frontend/src/routes/(authenticated)/history/RollbackModal.svelte @@ -2,8 +2,9 @@ import { t } from 'svelte-i18n'; import { invalidate } from '$app/navigation'; import { Button, Modal } from 'flowbite-svelte'; + import type { Commit } from '$lib/history'; - export let commit: { SHA1: string; message: string } | undefined; + export let commit: Commit | undefined; const revertCommit = async (commitSHA: string | undefined) => { if (commitSHA === undefined) return; From 54fe869d0cef6ff2df8668cdf15446a252aa4675 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Maximilian=20Schm=C3=B6cker?= Date: Sat, 19 Oct 2024 18:02:45 +0200 Subject: [PATCH 2/7] fix: history concurency --- controller/thymis_controller/project.py | 45 +++++++++++++++---------- 1 file changed, 28 insertions(+), 17 deletions(-) diff --git a/controller/thymis_controller/project.py b/controller/thymis_controller/project.py index 18dec1bd..e24ceb24 100644 --- a/controller/thymis_controller/project.py +++ b/controller/thymis_controller/project.py @@ -8,6 +8,7 @@ import subprocess import sys import tempfile +import threading import traceback from pathlib import Path from typing import List @@ -127,6 +128,7 @@ class Project: repo: git.Repo known_hosts_path: pathlib.Path public_key: str + history_lock = threading.Lock() def __init__(self, path, db_session: Session): self.path = pathlib.Path(path) @@ -241,23 +243,32 @@ def commit(self, summary: str): self.repo.index.commit(summary) def get_history(self): - return [ - history.Commit( - message=commit.message, - author=commit.author.name, - date=commit.authored_datetime, - SHA=commit.hexsha, - SHA1=self.repo.git.rev_parse(commit.hexsha, short=True), - state_diff=self.repo.git.diff( - commit.hexsha, - commit.parents[0].hexsha if len(commit.parents) > 0 else None, - "-R", - "state.json", - unified=5, - ).split("\n")[4:], - ) - for commit in self.repo.iter_commits() - ] + try: + with self.history_lock: + return [ + history.Commit( + message=commit.message, + author=commit.author.name, + date=commit.authored_datetime, + SHA=commit.hexsha, + SHA1=self.repo.git.rev_parse(commit.hexsha, short=True), + state_diff=self.repo.git.diff( + commit.hexsha, + ( + commit.parents[0].hexsha + if len(commit.parents) > 0 + else None + ), + "-R", + "state.json", + unified=5, + ).split("\n")[4:], + ) + for commit in self.repo.iter_commits() + ] + except Exception: + traceback.print_exc() + return [] def update_known_hosts(self, db_session: Session): if not self.known_hosts_path or not self.known_hosts_path.exists(): From ab723a4c2962d7441e133bcae97a545e620463c3 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Maximilian=20Schm=C3=B6cker?= Date: Sat, 19 Oct 2024 18:04:47 +0200 Subject: [PATCH 3/7] fix: project read race condition --- controller/thymis_controller/dependencies.py | 12 ++++++++---- 1 file changed, 8 insertions(+), 4 deletions(-) diff --git a/controller/thymis_controller/dependencies.py b/controller/thymis_controller/dependencies.py index b6b1caca..a0d8411d 100644 --- a/controller/thymis_controller/dependencies.py +++ b/controller/thymis_controller/dependencies.py @@ -1,5 +1,6 @@ import datetime import logging +import threading import uuid from typing import Annotated, Generator, Optional, Union @@ -16,15 +17,18 @@ global_project = None SESSION_LIFETIME = datetime.timedelta(days=1) +project_lock = threading.Lock() def get_project(): global global_project - if global_project is None: - REPO_PATH = global_settings.REPO_PATH.resolve() - global_project = Project(REPO_PATH, next(get_db_session())) - return global_project + with project_lock: + if global_project is None: + REPO_PATH = global_settings.REPO_PATH.resolve() + + global_project = Project(REPO_PATH, next(get_db_session())) + return global_project def get_state(project: Project = Depends(get_project)): From 5d59a3b2641d5b07f75b8323f49ed3b1bba2dd7e Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Maximilian=20Schm=C3=B6cker?= Date: Sat, 26 Oct 2024 17:20:49 +0200 Subject: [PATCH 4/7] fix: history promise --- frontend/src/routes/(authenticated)/history/+page.ts | 9 +++++++-- 1 file changed, 7 insertions(+), 2 deletions(-) diff --git a/frontend/src/routes/(authenticated)/history/+page.ts b/frontend/src/routes/(authenticated)/history/+page.ts index caf95666..818257fe 100644 --- a/frontend/src/routes/(authenticated)/history/+page.ts +++ b/frontend/src/routes/(authenticated)/history/+page.ts @@ -2,13 +2,18 @@ import type { Commit } from '$lib/history'; import type { PageLoad } from './$types'; export const load = (async ({ fetch }) => { - const response = await fetch(`/api/history`, { + const history_response = fetch(`/api/history`, { method: 'GET', headers: { 'content-type': 'application/json' } + }).then((res) => { + if (!res.ok) { + throw new Error(`Failed to fetch history: ${res.status} ${res.statusText}`); + } + return res.json() as Promise; }); return { - history: response.json() as Promise + history: history_response }; }) satisfies PageLoad; From 67784f5f5bd5ca20b66737ab384e25dd0d02cb12 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Maximilian=20Schm=C3=B6cker?= Date: Tue, 26 Nov 2024 16:14:19 +0100 Subject: [PATCH 5/7] fix: deploy single device task --- controller/thymis_controller/project.py | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/controller/thymis_controller/project.py b/controller/thymis_controller/project.py index e24ceb24..95eb1d03 100644 --- a/controller/thymis_controller/project.py +++ b/controller/thymis_controller/project.py @@ -319,7 +319,11 @@ def create_deploy_device_task(self, device_identifier: str, target_host: str): ) return task.global_task_controller.add_task( task.DeployDeviceTask( - self.path, device, target_host, global_settings.SSH_KEY_PATH + self.path, + device, + global_settings.SSH_KEY_PATH, + self.known_hosts_path, + target_host, ) ) From 310438046aaad53df2bc624f2f0e3e84865943bf Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Maximilian=20Schm=C3=B6cker?= Date: Sat, 26 Oct 2024 14:34:39 +0200 Subject: [PATCH 6/7] fix: finish reading frontend error messages --- controller/thymis_controller/routers/frontend.py | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/controller/thymis_controller/routers/frontend.py b/controller/thymis_controller/routers/frontend.py index c15d9cbc..ab75b1c3 100644 --- a/controller/thymis_controller/routers/frontend.py +++ b/controller/thymis_controller/routers/frontend.py @@ -84,13 +84,12 @@ async def run(self): # read stdout and stderr in background async def read_stream(stream: asyncio.StreamReader, level=logging.INFO): - while self.process.returncode is None and not self.stopped: + while not stream.at_eof(): line = await stream.readline() if line: logger.log( level, "frontend process: %s", line.decode("utf-8").strip() ) - await asyncio.sleep(0.001) # start threads asyncio.create_task(read_stream(self.process.stdout)) From 178235c9c9f7715a8c6a782e926567c126d9d8b5 Mon Sep 17 00:00:00 2001 From: Eli Kogan-Wang Date: Mon, 2 Dec 2024 14:54:11 +0100 Subject: [PATCH 7/7] comment out revert commit button in history page --- frontend/src/routes/(authenticated)/history/+page.svelte | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/frontend/src/routes/(authenticated)/history/+page.svelte b/frontend/src/routes/(authenticated)/history/+page.svelte index 188c93be..8781d1f5 100644 --- a/frontend/src/routes/(authenticated)/history/+page.svelte +++ b/frontend/src/routes/(authenticated)/history/+page.svelte @@ -43,7 +43,7 @@ with hash {history.SHA1}
- + -->

{#if index === 0}