Fix some compose panel bugs and reload workers when changed

This commit is contained in:
2026-02-16 13:39:48 +00:00
parent 9ce50a3053
commit 8ca1695fab
7 changed files with 177 additions and 25 deletions

View File

@@ -1033,7 +1033,7 @@ class XMPPComponent(ComponentXMPP):
sender="XMPP", sender="XMPP",
text=body, text=body,
ts=int(now().timestamp() * 1000), ts=int(now().timestamp() * 1000),
# outgoing=detail.is_outgoing_message, ????????? TODO: outgoing=True,
) )
self.log.info("Stored a message sent from XMPP in the history.") self.log.info("Stored a message sent from XMPP in the history.")

View File

@@ -284,6 +284,17 @@
{% endif %} {% endif %}
<p class="compose-msg-meta"> <p class="compose-msg-meta">
{{ msg.display_ts }}{% if msg.author %} · {{ msg.author }}{% endif %} {{ msg.display_ts }}{% if msg.author %} · {{ msg.author }}{% endif %}
{% if msg.read_ts %}
<span class="compose-ticks" title="Read at {{ msg.read_display }}">
<span class="icon is-small"><i class="fa-solid fa-check-double has-text-info"></i></span>
<span class="compose-tick-time">{{ msg.read_display }}</span>
</span>
{% elif msg.delivered_ts %}
<span class="compose-ticks" title="Delivered at {{ msg.delivered_display }}">
<span class="icon is-small"><i class="fa-solid fa-check-double has-text-grey"></i></span>
<span class="compose-tick-time">{{ msg.delivered_display }}</span>
</span>
{% endif %}
</p> </p>
</article> </article>
</div> </div>
@@ -538,6 +549,18 @@
#{{ panel_id }} .compose-msg-meta { #{{ panel_id }} .compose-msg-meta {
margin: 0; margin: 0;
} }
#{{ panel_id }} .compose-ticks {
display: inline-flex;
align-items: center;
gap: 0.22rem;
margin-left: 0.4rem;
color: #6b7787;
font-size: 0.72rem;
}
#{{ panel_id }} .compose-tick-time {
font-size: 0.66rem;
color: #616161;
}
#{{ panel_id }} .compose-platform-switch { #{{ panel_id }} .compose-platform-switch {
margin-top: 0.32rem; margin-top: 0.32rem;
} }
@@ -1737,6 +1760,40 @@
metaText += " · " + String(msg.author); metaText += " · " + String(msg.author);
} }
meta.textContent = metaText; meta.textContent = metaText;
// Render delivery/read ticks and a small time label when available.
if (msg.read_ts) {
const tickWrap = document.createElement("span");
tickWrap.className = "compose-ticks";
tickWrap.title = "Read at " + String(msg.read_display || msg.read_ts || "");
const icon = document.createElement("span");
icon.className = "icon is-small";
const i = document.createElement("i");
i.className = "fa-solid fa-check-double has-text-info";
icon.appendChild(i);
const timeSpan = document.createElement("span");
timeSpan.className = "compose-tick-time";
timeSpan.textContent = String(msg.read_display || "");
tickWrap.appendChild(icon);
tickWrap.appendChild(timeSpan);
meta.appendChild(document.createTextNode(" "));
meta.appendChild(tickWrap);
} else if (msg.delivered_ts) {
const tickWrap = document.createElement("span");
tickWrap.className = "compose-ticks";
tickWrap.title = "Delivered at " + String(msg.delivered_display || msg.delivered_ts || "");
const icon = document.createElement("span");
icon.className = "icon is-small";
const i = document.createElement("i");
i.className = "fa-solid fa-check-double has-text-grey";
icon.appendChild(i);
const timeSpan = document.createElement("span");
timeSpan.className = "compose-tick-time";
timeSpan.textContent = String(msg.delivered_display || "");
tickWrap.appendChild(icon);
tickWrap.appendChild(timeSpan);
meta.appendChild(document.createTextNode(" "));
meta.appendChild(tickWrap);
}
bubble.appendChild(meta); bubble.appendChild(meta);
row.appendChild(bubble); row.appendChild(bubble);

View File

@@ -324,6 +324,14 @@ def _serialize_message(msg: Message) -> dict:
) )
display_text = text_value if text_value.strip() else ("(no text)" if not image_url else "") display_text = text_value if text_value.strip() else ("(no text)" if not image_url else "")
author = str(msg.custom_author or "").strip() author = str(msg.custom_author or "").strip()
delivered_ts = int(msg.delivered_ts or 0)
read_ts = int(msg.read_ts or 0)
delivered_display = _format_ts_label(int(delivered_ts)) if delivered_ts else ""
read_display = _format_ts_label(int(read_ts)) if read_ts else ""
ts_val = int(msg.ts or 0)
delivered_delta = int(delivered_ts - ts_val) if delivered_ts and ts_val else None
read_delta = int(read_ts - ts_val) if read_ts and ts_val else None
return { return {
"id": str(msg.id), "id": str(msg.id),
"ts": int(msg.ts or 0), "ts": int(msg.ts or 0),
@@ -335,6 +343,12 @@ def _serialize_message(msg: Message) -> dict:
"hide_text": hide_text, "hide_text": hide_text,
"author": author, "author": author,
"outgoing": _is_outgoing(msg), "outgoing": _is_outgoing(msg),
"delivered_ts": delivered_ts,
"read_ts": read_ts,
"delivered_display": delivered_display,
"read_display": read_display,
"delivered_delta": delivered_delta,
"read_delta": read_delta,
} }

View File

@@ -271,6 +271,48 @@ services:
# memory: 0.25G # memory: 0.25G
#network_mode: host #network_mode: host
# Optional watcher service to restart the runtime router (UR) when core code changes.
# This runs the `docker/watch_and_restart.py` script inside the same image and
# will restart the `ur_gia` container when files under `/code/core` change.
watch_ur:
image: xf/gia:prod
container_name: watch_ur_gia
build:
context: .
args:
OPERATION: ${OPERATION}
command: sh -c '. /venv/bin/activate && python docker/watch_and_restart.py'
volumes:
- ${REPO_DIR}:/code
- ${REPO_DIR}/docker/uwsgi.ini:/conf/uwsgi.ini
- ${APP_DATABASE_FILE}:/conf/db.sqlite3
- type: bind
source: /code/vrun
target: /var/run
environment:
WATCH_PATHS: "/code/core"
TARGET_CONTAINER: "ur_gia"
# Optional watcher service to restart the scheduling process when app code changes.
watch_scheduling:
image: xf/gia:prod
container_name: watch_scheduling_gia
build:
context: .
args:
OPERATION: ${OPERATION}
command: sh -c '. /venv/bin/activate && python docker/watch_and_restart.py'
volumes:
- ${REPO_DIR}:/code
- ${REPO_DIR}/docker/uwsgi.ini:/conf/uwsgi.ini
- ${APP_DATABASE_FILE}:/conf/db.sqlite3
- type: bind
source: /code/vrun
target: /var/run
environment:
WATCH_PATHS: "/code/app"
TARGET_CONTAINER: "scheduling_gia"
redis: redis:
image: redis image: redis
container_name: redis_gia container_name: redis_gia

View File

@@ -23,5 +23,7 @@ log-level=debug
# Autoreload on code changes (graceful reload) # Autoreload on code changes (graceful reload)
py-autoreload=1 py-autoreload=1
py-autoreload-on-edit=/code/GIA/core # In the container the repository is mounted at /code, not /code/GIA
py-autoreload-on-edit=/code/GIA/app # point autoreload at the actual in-container paths
py-autoreload-on-edit=/code/core
py-autoreload-on-edit=/code/app

View File

@@ -1,6 +1,27 @@
#!/usr/bin/env python3 #!/usr/bin/env python3
""" """
Watch for code changes in core/ and app/ and restart ur_gia container. Watch for code changes and restart a target container.
This script watches one or more directories (set via the `WATCH_PATHS`
environment variable, comma-separated) and restarts the container named by
`TARGET_CONTAINER` (defaults to `ur_gia`) when a filesystem change is detected.
Typical usage in this repo (examples are provided in `docker-compose.yml`):
- To restart the runtime router (UR) when core code changes set:
WATCH_PATHS=/code/core
TARGET_CONTAINER=ur_gia
- To restart the scheduling command when app code changes set:
WATCH_PATHS=/code/app
TARGET_CONTAINER=scheduling_gia
If you need to force a restart manually (for example to refresh a running
management command), you can "touch" any file under the watched path:
docker compose exec <service> sh -c 'touch /code/core/__restart__'
The watcher ignores `__pycache__`, `.pyc` files and `.git` paths.
""" """
import os import os
import sys import sys
@@ -40,17 +61,16 @@ class ChangeHandler(FileSystemEventHandler):
self._restart_ur() self._restart_ur()
def _restart_ur(self): def _restart_ur(self):
print(f'[{time.strftime("%H:%M:%S")}] Restarting ur_gia...', flush=True) # Determine target container from environment (default `ur_gia`)
target = os.environ.get('TARGET_CONTAINER', 'ur_gia')
print(f'[{time.strftime("%H:%M:%S")}] Restarting {target}...', flush=True)
# Try podman first (preferred in this setup), then docker # Try podman first (preferred in this setup), then docker
result = subprocess.run( cmd = f"podman restart {target} 2>/dev/null || docker restart {target} 2>/dev/null"
'podman restart ur_gia 2>/dev/null || docker restart ur_gia 2>/dev/null', result = subprocess.run(cmd, shell=True, capture_output=True)
shell=True,
capture_output=True,
)
if result.returncode == 0: if result.returncode == 0:
print(f'[{time.strftime("%H:%M:%S")}] ur_gia restarted successfully', flush=True) print(f'[{time.strftime("%H:%M:%S")}] {target} restarted successfully', flush=True)
else: else:
print(f'[{time.strftime("%H:%M:%S")}] ur_gia restart failed', flush=True) print(f'[{time.strftime("%H:%M:%S")}] {target} restart failed', flush=True)
time.sleep(1) time.sleep(1)
@@ -58,12 +78,16 @@ def main():
handler = ChangeHandler() handler = ChangeHandler()
observer = Observer() observer = Observer()
# Watch both core and app directories # Allow overriding watched paths via environment variable `WATCH_PATHS`.
watch_paths = ['/code/GIA/core', '/code/GIA/app'] # Default is `/code/core,/code/app` but you can set e.g. `WATCH_PATHS=/code/core`
watch_paths_env = os.environ.get('WATCH_PATHS', '/code/core,/code/app')
watch_paths = [p.strip() for p in watch_paths_env.split(',') if p.strip()]
for path in watch_paths: for path in watch_paths:
if os.path.exists(path): if os.path.exists(path):
observer.schedule(handler, path, recursive=True) observer.schedule(handler, path, recursive=True)
print(f'Watching: {path}', flush=True) print(f'Watching: {path}', flush=True)
else:
print(f'Not found (will not watch): {path}', flush=True)
observer.start() observer.start()
print(f'[{time.strftime("%H:%M:%S")}] File watcher started. Monitoring for changes...', flush=True) print(f'[{time.strftime("%H:%M:%S")}] File watcher started. Monitoring for changes...', flush=True)

View File

@@ -1,7 +1,18 @@
#!/usr/bin/env python3 #!/usr/bin/env python3
""" """
Simple file watcher using stat() instead of watchdog (no external deps). Simple file watcher using stat() instead of watchdog (no external deps).
Watches core/ and app/ for changes and restarts ur_gia.
This lightweight watcher can be used when you don't want the `watchdog`
dependency. Configure the directories to watch using the `WATCH_PATHS`
environment variable (comma-separated). Configure which container to restart
using `TARGET_CONTAINER` (defaults to `ur_gia`).
Example:
WATCH_PATHS=/code/core TARGET_CONTAINER=ur_gia
Touching a file under the watched path will trigger a restart of the target
container; e.g. `touch /code/core/__restart__` will cause the watcher to act.
""" """
import os import os
import sys import sys
@@ -27,21 +38,23 @@ def get_mtime(path):
def restart_ur(): def restart_ur():
"""Restart ur_gia container.""" """Restart target container (defaults to `ur_gia`)."""
print(f'[{time.strftime("%H:%M:%S")}] Restarting ur_gia...', flush=True) target = os.environ.get('TARGET_CONTAINER', 'ur_gia')
result = subprocess.run( print(f'[{time.strftime("%H:%M:%S")}] Restarting {target}...', flush=True)
'podman restart ur_gia 2>/dev/null || docker restart ur_gia 2>/dev/null', cmd = f'podman restart {target} 2>/dev/null || docker restart {target} 2>/dev/null'
shell=True, result = subprocess.run(cmd, shell=True, capture_output=True)
capture_output=True,
)
if result.returncode == 0: if result.returncode == 0:
print(f'[{time.strftime("%H:%M:%S")}] ur_gia restarted', flush=True) print(f'[{time.strftime("%H:%M:%S")}] {target} restarted', flush=True)
else: else:
print(f'[{time.strftime("%H:%M:%S")}] restart failed', flush=True) print(f'[{time.strftime("%H:%M:%S")}] restart failed', flush=True)
def main(): def main():
paths = ['/code/GIA/core', '/code/GIA/app'] # In the container the repository is mounted at /code
# Allow overriding watched paths via environment variable `WATCH_PATHS`.
# Default is `/code/core,/code/app`.
paths_env = os.environ.get('WATCH_PATHS', '/code/core,/code/app')
paths = [p.strip() for p in paths_env.split(',') if p.strip()]
last_mtimes = {} last_mtimes = {}
for path in paths: for path in paths: