claude unstested update for moving from wp to project -> wp to task

This commit is contained in:
buenosairesam
2025-10-03 04:47:13 -03:00
parent 72f9e9d9a7
commit 71752c7d76
10 changed files with 237 additions and 146 deletions

10
dmapp/dmcore/config.json Normal file
View File

@@ -0,0 +1,10 @@
{
"work_desktop_tasks": {
"2": null,
"5": null,
"6": null,
"7": null
},
"timezone": "America/Argentina/Buenos_Aires",
"task_file": "/home/mariano/LETRAS/adm/task/main"
}

View File

@@ -2,6 +2,7 @@
import logging
from pymongo import MongoClient
from zoneinfo import ZoneInfo
# Logging configuration
logging.basicConfig(
@@ -9,7 +10,7 @@ logging.basicConfig(
format="%(asctime)s %(levelname)-8s [%(filename)s:%(lineno)d] %(message)s",
datefmt="%Y-%m-%d %H:%M:%S",
)
# 2) Get your modules logger and bump it to DEBUG
# 2) Get your module's logger and bump it to DEBUG
logger = logging.getLogger(__name__)
logger.setLevel(logging.DEBUG)
@@ -25,3 +26,9 @@ db = client.deskmeter
switches = db.switch
states = db.state
tasks = db.task
# Application configuration
desktops = ("Plan", "Think", "Work", "Other", "Away", "Work", "Work", "Work")
unlabeled = "Away"
timezone = ZoneInfo("America/Argentina/Buenos_Aires")
task_file = "/home/mariano/LETRAS/adm/task/main"

View File

@@ -1,4 +1,5 @@
import datetime
import json
import os
import subprocess
import time
@@ -10,13 +11,92 @@ from config import logger, switches
from zoneinfo import ZoneInfo
desktops = ("Plan", "Think", "Work", "Other", "Away", "Work", "Work", "Work")
work_desktops = {2: "snk", 5: "dlt", 6: "vhs", 7: "own"}
unlabeled = "Away"
config_file = "config.json"
def load_config():
"""Load configuration from JSON file"""
with open(config_file, "r") as f:
return json.load(f)
def reload_config_if_changed():
"""Check if config file changed and reload to state if needed"""
current_config_mtime = state.retrieve("current").get("config_mtime")
config_mtime = os.path.getmtime(config_file)
if current_config_mtime != config_mtime:
cfg = load_config()
work_desktop_tasks = {int(k): v for k, v in cfg["work_desktop_tasks"].items()}
state.sync_desktop_tasks(work_desktop_tasks)
state.save("current", config_mtime=config_mtime)
logger.info(f"Config reloaded: {work_desktop_tasks}")
def now():
return datetime.datetime.now(ZoneInfo("America/Argentina/Buenos_Aires"))
cfg = load_config()
return datetime.datetime.now(ZoneInfo(cfg["timezone"]))
def handle_task_file_changes(current_task):
"""Check if task file changed and update task if needed"""
current_mtime = state.retrieve("current").get("filetime")
file_mtime = task.get_file_mtime(None)
if current_mtime != file_mtime:
task_id = task.read_and_extract(None)
logger.debug(f"task_id:{task_id}")
task.file_to_db(None)
if task_id != current_task:
state.save("current", task=task_id)
current_task = task_id
return current_task
def update_workspace_state():
"""Update current workspace in state"""
current_workspace = active_workspace()
state.save("current", workspace=current_workspace)
return current_workspace
def enforce_desktop_task(current_workspace, work_desktop_tasks, current_task):
"""Enforce assigned task for work desktops"""
if current_workspace in work_desktop_tasks and work_desktop_tasks[current_workspace]:
assigned_task = work_desktop_tasks[current_workspace]
if current_task != assigned_task:
current_task = assigned_task
state.save("current", task=current_task)
task.db_to_file_as_is(None)
return current_task
def track_workspace_switch(current_workspace, current_task, last_switch_time):
"""Update or create switch record"""
last_doc = switches.find_one(sort=[("_id", -1)])
if (
last_doc["workspace"] == desktop(current_workspace)
and last_doc["task"] == current_task
):
delta = round((now() - last_switch_time).total_seconds())
switches.update_one(
{"_id": last_doc["_id"]}, {"$set": {"delta": delta, "task": current_task}}
)
return last_switch_time
else:
switch = {
"workspace": desktop(current_workspace),
"date": now(),
"delta": 0,
"task": current_task,
}
switches.insert_one(switch)
return now()
def active_workspace():
@@ -40,7 +120,6 @@ def desktop(workspace_index):
task.read_and_extract(None)
state.init_work_state(work_desktops)
current_workspace = active_workspace()
@@ -57,65 +136,23 @@ switch = {
switches.insert_one(switch)
while True:
current_mtime = state.retrieve("current").get("filetime")
file_mtime = task.get_file_mtime(None)
# Check if config changed and reload
reload_config_if_changed()
# First handle file changes
if current_mtime != file_mtime:
task_id = task.read_and_extract(None)
logger.debug(f"task_id:{task_id}")
task.file_to_db(None)
if task_id != current_task: # Only update state if different
state.save("current", task=task_id)
current_task = task_id
# Load work_desktop_tasks from state
work_desktop_tasks = state.retrieve_desktop_state()
# Handle task file changes
current_task = handle_task_file_changes(current_task)
# Update current task and workspace
current_task = state.retrieve("current").get("task")
current_workspace = active_workspace()
state.save("current", workspace=current_workspace)
current_workspace = update_workspace_state()
last_doc = switches.find_one(sort=[("_id", -1)])
# Enforce desktop task assignments
current_task = enforce_desktop_task(current_workspace, work_desktop_tasks, current_task)
# work workflow
if current_workspace in work_desktops.keys():
work_states = state.retrieve_work_state()
current_work_task = work_states[work_desktops[current_workspace]]
# Get all task IDs under current workspace path
workspace_tasks = task.get_tasks_tree(
f"work/{work_desktops[current_workspace]}"
)
work_task_ids = {t["task_id"] for t in workspace_tasks if "task_id" in t}
# if current_task in work_task_ids and current_task != current_work_task:
if current_task not in work_task_ids:
# Enforce work task if current task is not in workspace
if current_task != current_work_task:
current_task = current_work_task
state.save("current", task=current_task)
task.db_to_file_as_is(None)
elif current_task != current_work_task:
# Update work state when switching to a different valid task
state.update_work_state(work_desktops[current_workspace], current_task)
# regular flow
if (
last_doc["workspace"] == desktop(current_workspace)
and last_doc["task"] == current_task
):
delta = round((now() - last_switch_time).total_seconds())
switches.update_one(
{"_id": last_doc["_id"]}, {"$set": {"delta": delta, "task": current_task}}
)
else:
current_workspace = active_workspace()
switch = {
"workspace": desktop(current_workspace),
"date": now(),
"delta": 0,
"task": current_task,
}
switches.insert_one(switch)
last_switch_time = now()
# Track workspace switches
last_switch_time = track_workspace_switch(current_workspace, current_task, last_switch_time)
time.sleep(2)

View File

@@ -7,10 +7,11 @@ def save(
task: str | None = None,
workspace: str | None = None,
filetime: str | None = None,
config_mtime: float | None = None,
) -> None:
"""
Upsert a document with _id=doc_id, setting any of the provided fields.
Leave fields you dont pass unchanged.
Leave fields you don't pass unchanged.
"""
updates: dict = {}
if task is not None:
@@ -19,6 +20,8 @@ def save(
updates["workspace"] = workspace
if filetime is not None:
updates["filetime"] = filetime
if config_mtime is not None:
updates["config_mtime"] = config_mtime
if updates:
states.update_one(
@@ -30,40 +33,36 @@ def save(
def retrieve(doc_id: str) -> dict[str, str | None]:
"""
Fetches the document with _id=doc_id and returns its 'task' and 'workspace'.
If the document doesnt exist, both will be None.
Fetches the document with _id=doc_id and returns its fields.
If the document doesn't exist, all fields will be None.
"""
doc = states.find_one({"_id": doc_id})
return {
"task": doc.get("task") if doc else None,
"workspace": doc.get("workspace") if doc else None,
"filetime": doc.get("filetime") if doc else None,
"config_mtime": doc.get("config_mtime") if doc else None,
}
# just
def update_work_state(work: str, task_id: str):
def sync_desktop_tasks(work_desktop_tasks: dict):
"""
update work state
Sync work_desktop_tasks from config file to state
"""
states.update_one({"_id": "work"}, {"$set": {work: task_id}})
update_dict = {str(k): v for k, v in work_desktop_tasks.items()}
states.update_one(
{"_id": "work_desktop_tasks"},
{"$set": update_dict},
upsert=True,
)
def init_work_state(wd: dict):
def retrieve_desktop_state():
"""
init work states with default values
Get work_desktop_tasks mapping from state
"""
if not states.find_one({"_id": "work"}):
states.insert_one(
{
"_id": "work",
**{
wd[k]: tasks.find_one({"path": f"work/{wd[k]}"})["task_id"]
for k in wd
},
}
)
def retrieve_work_state():
return states.find_one({"_id": "work"})
doc = states.find_one({"_id": "work_desktop_tasks"})
if not doc:
return {}
# Convert string keys to int and exclude _id
return {int(k): v for k, v in doc.items() if k != "_id"}