mirror of
https://github.com/marcoallegretti/WEFT_OS.git
synced 2026-03-26 17:03:09 +00:00
feat(appd): session persistence across clean restarts
On clean shutdown (SIGINT/SIGTERM), save the list of running app IDs to /weft/last-session.json before calling shutdown_all(). On the next startup, load_session() reads and immediately deletes that file, then dispatches a LaunchApp for each saved app ID. This restores the previous session after a system restart or orderly service stop. If the file does not exist (crash, first boot, or deliberate reset) no auto-launch occurs. Duplicate detection relies on the existing fact that the saved processes are gone before appd starts again.
This commit is contained in:
parent
92920984bd
commit
5061310d63
1 changed files with 44 additions and 0 deletions
|
|
@ -80,6 +80,14 @@ impl SessionRegistry {
|
|||
.collect()
|
||||
}
|
||||
|
||||
fn running_app_ids(&self) -> Vec<String> {
|
||||
self.sessions
|
||||
.values()
|
||||
.filter(|e| !matches!(e.state, AppStateKind::Stopped))
|
||||
.map(|e| e.app_id.clone())
|
||||
.collect()
|
||||
}
|
||||
|
||||
fn state(&self, session_id: u64) -> AppStateKind {
|
||||
self.sessions
|
||||
.get(&session_id)
|
||||
|
|
@ -151,6 +159,20 @@ async fn run() -> anyhow::Result<()> {
|
|||
|
||||
let _ = sd_notify::notify(false, &[sd_notify::NotifyState::Ready]);
|
||||
|
||||
if let Some(app_ids) = load_session() {
|
||||
tracing::info!(count = app_ids.len(), "restoring previous session");
|
||||
for app_id in app_ids {
|
||||
let _ = dispatch(
|
||||
crate::ipc::Request::LaunchApp {
|
||||
app_id,
|
||||
surface_id: 0,
|
||||
},
|
||||
®istry,
|
||||
)
|
||||
.await;
|
||||
}
|
||||
}
|
||||
|
||||
#[cfg(unix)]
|
||||
let mut sigterm = {
|
||||
use tokio::signal::unix::{SignalKind, signal};
|
||||
|
|
@ -194,12 +216,34 @@ async fn run() -> anyhow::Result<()> {
|
|||
}
|
||||
}
|
||||
|
||||
save_session(registry.lock().await.running_app_ids()).await;
|
||||
registry.lock().await.shutdown_all();
|
||||
tokio::time::sleep(tokio::time::Duration::from_millis(200)).await;
|
||||
let _ = std::fs::remove_file(&socket_path);
|
||||
Ok(())
|
||||
}
|
||||
|
||||
fn session_file_path() -> Option<std::path::PathBuf> {
|
||||
let runtime_dir = std::env::var("XDG_RUNTIME_DIR").ok()?;
|
||||
Some(PathBuf::from(runtime_dir).join("weft/last-session.json"))
|
||||
}
|
||||
|
||||
async fn save_session(app_ids: Vec<String>) {
|
||||
let Some(path) = session_file_path() else {
|
||||
return;
|
||||
};
|
||||
if let Ok(json) = serde_json::to_string(&app_ids) {
|
||||
let _ = std::fs::write(&path, json);
|
||||
}
|
||||
}
|
||||
|
||||
fn load_session() -> Option<Vec<String>> {
|
||||
let path = session_file_path()?;
|
||||
let json = std::fs::read_to_string(&path).ok()?;
|
||||
let _ = std::fs::remove_file(&path);
|
||||
serde_json::from_str(&json).ok()
|
||||
}
|
||||
|
||||
fn ws_port() -> u16 {
|
||||
std::env::var("WEFT_APPD_WS_PORT")
|
||||
.ok()
|
||||
|
|
|
|||
Loading…
Reference in a new issue