Install
openclaw skills install water-coachHydration tracking and coaching skill. Use when user wants to track water intake, get reminders to drink water, log body metrics (weight, body fat, muscle %,...
openclaw skills install water-coachFollow the first setup here references/setup.md
water setup
| is_setup | What to do |
|---|---|
| false | Ask: weight OR desired goal. Also ask: "What times do you want water reminders?" (let user configure their schedule). Then use water set_body_weight 80 or water set_goal 3000. Don't assume hardcoded times! |
| true | Skip setup. Just log water or show status. |
water_coach.py <namespace> <command> [options]
Namespaces: water | body | analytics
logged_at,drank_at,date,slot,ml_drank,goal_at_time,message_id
| Column | Description |
|---|---|
| logged_at | When user told you (NOW) |
| drank_at | When user actually drank (user can specify past time) |
| date | Derived from drank_at |
| slot | morning/lunch/afternoon/evening/manual |
| ml_drank | Amount in ml |
| goal_at_time | Goal at that moment |
| message_id | Audit trail - link to conversation |
Key Rules:
Details at references/log_format.md
Every water log entry captures:
water audit <message_id> to get entry + conversation context# Check proof of a water entry
water audit msg_123
# Returns: entry data + surrounding messages for context
⚠️ Privacy Notice: The audit feature can read your conversation transcripts, but only when you explicitly run
water audit <message_id>. This is off by default (audit_auto_capture: false).# Edit water_config.json and set: "audit_auto_capture": trueHow it works:
- Water log always saves the
message_id(regardless of this setting) ✅- When you run
water audit <message_id>:
- If
false: Shows entry data only (message_id saved, but no context read)- If
true: Also reads transcript to show conversation context ("User said: I drank 500ml")Why disable it? If you discuss sensitive topics and don't need proof of intake, leave it off.
# Water
water status # Current progress (calculated from drank_at)
water log 500 # Log intake (drank_at = now)
water log 500 --drank-at=2026-02-18T18:00:00Z # Log with past time
water log 500 --drank-at=2026-02-18T18:00:00Z --message-id=msg_123
water dynamic # Check if extra notification needed
water threshold # Get expected % for current hour
water set_body_weight 80 # Update weight + logs to body_metrics
water set_body_weight 80 --update-goal # + update goal
water audit <message_id> # Get entry + conversation context
# Body
body log --weight=80 --height=1.75 --body-fat=18
body latest # Get latest metrics
body history 30 # Get history
# Analytics
analytics week # Weekly briefing (Sunday 8pm)
analytics month # Monthly briefing (2nd day 8pm)
water threshold, don't hardcodewater-coach/
├── SKILL.md ← You are here
├── scripts/
│ ├── water_coach.py ← Unified CLI
│ └── water.py ← Core functions
├── data/ ← DO NOT USE - keep skill code separate from user data
└── references/
├── setup.md
├── dynamic.md
└── log_format.md
User data is stored in the AGENT WORKSPACE, NOT in the skill folder!
| Data | Location |
|---|---|
| water_log.csv | <agent-workspace>/memory/data/water_log.csv |
| water_config.json | <agent-workspace>/memory/data/water_config.json |
| body_metrics.csv | `<agent-workspace>/memory/data/body_metrics.csv |
Example path: /home/oriel/.openclaw/workspace/memory/data/
Why? Keeps user data separate from skill code — makes backups, migrations, and skill updates easier.
| Type | When | Command |
|---|---|---|
| User Configured | Per user's schedule | water status |
| Default Suggestion | 9am, 12pm, 3pm, 6pm, 9pm | water status |
| Dynamic | Every ~30 min (heartbeat) | water dynamic |
| Weekly | Sunday 8pm | analytics week |
| Monthly | 2nd day 8pm | analytics month |
| Task | Command |
|---|---|
| Check progress | water_coach.py water status |
| Log water | water_coach.py water log 500 |
| Need extra? | water_coach.py water dynamic |
| Body metrics | water_coach.py body log --weight=80 |
| Weekly report | water_coach.py analytics week |
| Monthly report | water_coach.py analytics month |
The water dynamic command had a bug where the hourly notification counter wouldn't reset when the hour changed. This is now fixed:
last_extra_hour and resets the counter accordinglyThe analyticsPM → 5 week and analytics month commands had a bug:
cumulative_ml column in CSVml_drank per day insteadUse these functions (don't reinvent):
| Report | Script | Function |
|---|---|---|
| Weekly | water_coach.py analytics week | analytics_week() in water_coach.py |
| Monthly | water_coach.py analytics month | analytics_month() in water_coach.py |
These call get_week_stats() and get_month_stats() in water.py.
When updating analytics functions, follow these rules:
1. Include ALL days, even with 0ml
# In get_week_stats() / get_month_stats()
# Include every day in the range, not just days with data
for i in range(days):
d = (date.today() - timedelta(days=i)).strftime("%Y-%m-%d")
ml = by_date.get(d, {}).get("ml", 0) # Default to 0, not skip
2. Calculate true average
# Average = total_ml / ALL days (including zeros), not just tracked days
avg_ml = total_ml / days # e.g., 15440ml / 7 days = 2205ml/day
3. Show all days in table format
| Dia | ML | % | Status |
| Sab 22 | 2250ml | 67.7% | ⚠️ |
| Seg 17 | 0ml | 0.0% | ❌ |
This gives users an accurate picture of their habits!
python3 -m pytest skills/water-coach/scripts/test/test_water.py -v
User: "eu tomei 2 copos"
Agent: (LLM interprets: 2 copos ≈ 500ml)
Agent: exec("water_coach.py water log 500")
→ Python logs to CSV
Agent Evaluations → evaluation/AGENT.md