shopping-list
Conversational shopping list with categories, family sharing, and purchase history. Add items, check them off, organize by category — all through natural language. Use for any shopping list, grocery list, "add to list", "what do we need", or "we need to buy" requests.
Packaged view
This page reorganizes the original catalog entry around fit, installability, and workflow context first. The original raw source lives below.
Install command
npx @skill-hub/cli install openclaw-skills-shopping-list
Repository
Skill path: skills/ajeenkya/shopping-list
Conversational shopping list with categories, family sharing, and purchase history. Add items, check them off, organize by category — all through natural language. Use for any shopping list, grocery list, "add to list", "what do we need", or "we need to buy" requests.
Open repositoryBest for
Primary workflow: Ship Full Stack.
Technical facets: Full Stack.
Target audience: everyone.
License: Unknown.
Original source
Catalog source: SkillHub Club.
Repository owner: openclaw.
This is still a mirrored public skill entry. Review the repository before installing into production workflows.
What it helps with
- Install shopping-list into Claude Code, Codex CLI, Gemini CLI, or OpenCode workflows
- Review https://github.com/openclaw/skills before adding shopping-list to shared team environments
- Use shopping-list for development workflows
Works across
Favorites: 0.
Sub-skills: 0.
Aggregator: No.
Original source / Raw SKILL.md
---
name: shopping-list
version: 1.0.1
description: >
Conversational shopping list with categories, family sharing, and purchase
history. Add items, check them off, organize by category — all through
natural language. Use for any shopping list, grocery list, "add to list",
"what do we need", or "we need to buy" requests.
---
# Shopping List
Manage the household shopping list. Add items with quantities, check them off,
organize by category. Data lives in `skills/shopping-list/data/`.
For full command reference and output formats, read `skills/shopping-list/references/commands.md`.
## Before Every Command
Run these checks before every shopping list operation, in order:
1. If `skills/shopping-list/data/` directory does not exist, create it.
2. If `data/active.json` does not exist, create it with:
```json
{
"items": [],
"categories": ["Produce", "Dairy", "Meat", "Pantry", "Frozen", "Beverages", "Household", "Personal"],
"lastModified": "<current ISO timestamp>"
}
```
3. If `data/active.json` exists but fails to parse as valid JSON, rename it to `data/active.json.corrupt` and create a fresh default file. Tell the user: "Shopping list data was corrupted. Saved backup as active.json.corrupt and started a fresh list."
4. If `data/config.json` does not exist, create it with: `{ "user": null, "snoozes": {} }`
5. If `config.json` has `"user": null`, ask the user: "What's your name? I'll use it to track who added each item." Store their answer (lowercased) in `config.json` before proceeding with the original command.
6. Run the archive process (see Archive section below).
All file paths in this document are relative to `skills/shopping-list/` unless stated otherwise.
## Data Files
### data/active.json
The live shopping list. Contains all items that have not yet been archived.
```json
{
"items": [
{
"id": "F47AC10B-58CC-4372-A567-0E02B2C3D479",
"name": "Whole Milk",
"normalizedName": "whole milk",
"quantity": 2,
"unit": "gallons",
"category": "Dairy",
"checkedOff": false,
"checkedOffDate": null,
"addedBy": "aj",
"addedDate": "2026-02-24T10:00:00Z",
"notes": null
}
],
"categories": ["Produce", "Dairy", "Meat", "Pantry", "Frozen", "Beverages", "Household", "Personal"],
"lastModified": "2026-02-24T10:00:00Z"
}
```
Field rules:
- `id` -- Generate via `uuidgen` in bash. If that command is unavailable, construct an ID from the current ISO timestamp concatenated with 4 random hex characters (e.g. `2026-02-24T10:00:00Z-a3f1`).
- `name` -- The item name as the user provided it, with leading/trailing whitespace trimmed. Preserve original casing for display purposes.
- `normalizedName` -- Always computed as `name.toLowerCase().trim()`. Recompute on every add or edit. This field is used for all matching and deduplication logic.
- `quantity` -- Optional. Defaults to `null` when the user does not specify a quantity. When present, must be a number greater than 0. Fractional values are fine (e.g. 0.5 for half a pound).
- `unit` -- Optional free text (e.g. "gallons", "lbs", "bunch", "bag"). Defaults to `null` when not specified.
- `category` -- One of the values from the `categories` array, or "Uncategorized".
- `checkedOff` -- Boolean. Starts as `false`. Set to `true` when the user checks off the item.
- `checkedOffDate` -- ISO timestamp when the item was checked off. `null` when not checked off.
- `addedBy` -- Lowercase string from `config.json` user field (e.g. "aj" or "shal").
- `addedDate` -- ISO timestamp when the item was first added.
- `notes` -- Optional free text for special instructions ("the organic one", "Costco size"). Defaults to `null`.
- `categories` (top-level array) -- The master list of known categories. Starts with 8 presets. Custom categories are appended when created.
- `lastModified` -- ISO timestamp. Updated on every write to this file.
### data/config.json
Stores session-persistent user configuration.
```json
{ "user": "aj", "snoozes": {} }
```
- `user` -- Set on first interaction, persists across sessions. Lowercase string.
- `snoozes` -- Reserved for Phase 2 restock suggestions. Ignore for now; preserve the field when writing.
### data/history-YYYY-MM.json
Monthly archive of purchased items. One file per calendar month, created on demand.
```json
{
"month": "2026-02",
"archivedItems": [
{ "...same fields as active item...", "archivedDate": "2026-02-25T08:00:00Z" }
]
}
```
The `archivedDate` field is added to each item when it moves from active to history. All original fields from the active item are preserved. A new file is created automatically when the first item is archived in a month that does not yet have a history file.
## Core Operations
### Adding Items
Parse the user's natural language into name, quantity, and unit for each item.
**Category inference:** Silently infer the category from the item name. Common mappings: milk/cheese/yogurt/butter go to Dairy, chicken/beef/fish to Meat, bananas/lettuce/onions to Produce, rice/pasta/flour to Pantry, dish soap/paper towels to Household, shampoo/toothpaste to Personal, beer/wine/juice to Beverages, frozen pizza/ice cream to Frozen. If the mapping is not obvious, assign "Uncategorized". Never prompt the user for a category. If the user wants to change it, they can say "move X to Y category".
**Multi-item input:** The user may add several items at once: "add eggs, bread, and 2 gallons milk" should produce 3 separate items. Parse commas and "and" as item separators. When the parsing is ambiguous (compound item names like "hot dog buns" near separators), ask the user rather than guessing wrong.
**Duplicate handling:** Before creating a new item, check if an item with the same `normalizedName` already exists in the active list.
- If it exists and is not checked off, update the existing item's quantity (add to it if the user gives a new quantity, or leave unchanged if not). Do not create a duplicate.
- If it exists and is checked off, uncheck it (set `checkedOff: false`, `checkedOffDate: null`) and update the quantity if specified.
**New categories:** If the user explicitly specifies a category that is not in the `categories` array, add it to the array and assign the item to it.
Set `addedBy` from `config.json` user value. Set `addedDate` to the current ISO timestamp. Set `checkedOff` to `false` and `checkedOffDate` to `null`. Update `lastModified` after writing.
### Checking Off Items
Match items by `normalizedName` -- case-insensitive, partial match is acceptable (e.g. "milk" matches "whole milk" if it is the only milk item).
- **One match:** Set `checkedOff` to `true` and `checkedOffDate` to the current ISO timestamp. Confirm to the user.
- **Multiple matches:** List the matching items and ask the user which one to check off.
- **No match:** Show the current list so the user can identify the correct item.
Checked-off items remain in `active.json` and display with strikethrough styling until the archive process moves them to history (after 24 hours).
### Removing Items
Permanently delete an item from the active list. No archive record, no history entry. This exists for "I added this by mistake" corrections.
Match by `normalizedName` using the same matching rules as checking off. Confirm removal to the user.
### Editing Items
Match the target item by `normalizedName`, then update the specified fields. Supported editable fields: name, quantity, unit, category, notes.
If the name changes, recompute `normalizedName`. If the category changes to one not in the `categories` array, add it. Update `lastModified` after writing.
### Changing User
When the user says "switch to Shal", "I'm AJ", or similar, update the `user` field in `config.json`. All subsequent item additions will use the new user value for `addedBy`.
## Archive Process
This process runs automatically at the start of every shopping list command, before handling the user's request.
1. Read `data/active.json`.
2. Find all items where `checkedOff` is `true` AND `checkedOffDate` is more than 24 hours ago.
3. If no items qualify, skip the rest of this process.
4. Determine the current month string (e.g. "2026-02"). Read `data/history-2026-02.json`. If it does not exist, create it with `{ "month": "2026-02", "archivedItems": [] }`.
5. Append each qualifying item to the history file's `archivedItems` array, adding an `archivedDate` field set to the current ISO timestamp.
6. Write the history file.
7. Remove the archived items from the `items` array in `data/active.json`.
8. Write the active file.
**Write order matters:** Always write the history file before the active file. If the process is interrupted between the two writes, the worst case is a harmless duplicate in history. Reversing the order risks data loss -- items removed from active but never written to history.
The `shopping clear` command triggers this same process immediately for all checked items, bypassing the 24-hour wait.
## Displaying the List
When showing the shopping list, sort categories in this fixed order:
1. Preset categories in their defined order: Produce, Dairy, Meat, Pantry, Frozen, Beverages, Household, Personal
2. Custom categories alphabetically after all presets
3. "Uncategorized" always appears last
Skip any category that has zero items (considering only unchecked items for this purpose). Within each category, sort items alphabetically by `normalizedName`. Do not rely on the stored array order in JSON.
Checked-off items appear with strikethrough styling within their category, visually distinct from unchecked items. They remain visible until archived.
## History Queries
When the user asks about past purchases ("what did we buy last month", "purchase history", "what did we get in January"), read the relevant `data/history-YYYY-MM.json` file(s). Present results grouped by date (most recent first) with item names and quantities. If no history file exists for the requested period, tell the user no purchase history was found for that month.
Note: history reflects the archive date, not the purchase date. An item checked off on Feb 28 but archived on March 1 appears in March's history.
---
## Referenced Files
> The following files are referenced in this skill and included for context.
### data/active.json
```json
{
"items": [
{
"id": "0D33C5A1-FC02-4A30-9862-B686AE924537",
"name": "Eggs",
"normalizedName": "eggs",
"quantity": null,
"unit": null,
"category": "Dairy",
"checkedOff": false,
"checkedOffDate": null,
"addedBy": "aj",
"addedDate": "2026-02-24T22:31:00Z",
"notes": null
},
{
"id": "BD736091-CEFA-4547-A5D4-CF345D30B310",
"name": "Bread",
"normalizedName": "bread",
"quantity": null,
"unit": null,
"category": "Pantry",
"checkedOff": false,
"checkedOffDate": null,
"addedBy": "aj",
"addedDate": "2026-02-24T22:31:00Z",
"notes": null
},
{
"id": "ED04A262-6C2E-4C3C-B435-7017CD7405C8",
"name": "Bananas",
"normalizedName": "bananas",
"quantity": null,
"unit": null,
"category": "Produce",
"checkedOff": false,
"checkedOffDate": null,
"addedBy": "aj",
"addedDate": "2026-02-24T22:31:00Z",
"notes": null
}
],
"categories": ["Produce", "Dairy", "Meat", "Pantry", "Frozen", "Beverages", "Household", "Personal"],
"lastModified": "2026-02-24T22:35:00Z"
}
```
### data/config.json
```json
{ "user": "aj", "snoozes": {} }
```
### data/history-2026-02.json
```json
{
"month": "2026-02",
"archivedItems": [
{
"id": "AF5234FD-FDD5-4F50-94A2-ECC7FBB96CC2",
"name": "Whole Milk",
"normalizedName": "whole milk",
"quantity": 2,
"unit": "gallons",
"category": "Dairy",
"checkedOff": true,
"checkedOffDate": "2026-02-24T22:33:00Z",
"addedBy": "aj",
"addedDate": "2026-02-24T22:30:00Z",
"notes": null,
"archivedDate": "2026-02-24T22:35:00Z"
}
]
}
```
---
## Skill Companion Files
> Additional files collected from the skill directory layout.
### _meta.json
```json
{
"owner": "ajeenkya",
"slug": "shopping-list",
"displayName": "Shopping List",
"latest": {
"version": "1.0.1",
"publishedAt": 1771976330161,
"commit": "https://github.com/openclaw/skills/commit/75a84b0a87224174172cde8a79572f28425388ac"
},
"history": [
{
"version": "1.0.0",
"publishedAt": 1771974570125,
"commit": "https://github.com/openclaw/skills/commit/f0984f722ac93b6146dc4e10a967d4c077c7ce71"
}
]
}
```
### references/commands.md
```markdown
# Shopping List — Command Reference
> This file is loaded on-demand when a shopping command is invoked.
> It is NOT loaded at session start. SKILL.md points here for details.
---
## 1. Commands
| Command | Description |
|---------|-------------|
| `shopping add <item>` | Add item(s) — "add 2 gallons milk" or "add eggs, bread, butter" |
| `shopping list` | Show active list grouped by category |
| `shopping check <item>` | Check off — "check milk" or "got the milk" |
| `shopping remove <item>` | Delete permanently, no archive |
| `shopping edit <item>` | Edit qty/unit/category/notes — "make milk 2 gallons" |
| `shopping categories` | Show categories with item counts |
| `shopping history` | Recent purchase history — "history" or "history february" |
| `shopping suggest` | Restock suggestions (Phase 2, not yet implemented) |
| `shopping clear` | Force-archive all checked items now |
| `shopping export` | Clean copy-paste list, no formatting |
| `shopping switch-user` | Change identity (AJ <-> Shal) |
---
## 2. Natural Language Mapping
Users will rarely type exact commands. Map these phrases to the right action.
| User says | Action |
|-----------|--------|
| "Add milk to the shopping list" | add milk |
| "We need eggs, bread, and butter" | add eggs, add bread, add butter |
| "Got the milk" / "Bought the milk" / "Cross off milk" | check milk |
| "What do we need from the store?" | list |
| "Never mind on the bananas" | remove bananas |
| "Make that 2 gallons instead of 1" | edit milk 2 gallons |
| "What have we been buying lately?" | history |
| "Anything we're running low on?" | suggest |
| "Move rice to Pantry" | edit rice (change category) |
---
## 3. Output Format Templates
### shopping list
```
Shopping List (8 items)
PRODUCE
[ ] Avocados 3
[ ] Bananas 1 bunch
DAIRY
[ ] Whole Milk 2 gallons
[x] Butter 1 lb <- archiving in 18h
MEAT
[ ] Chicken Breast 2 lbs
PANTRY
[ ] Black Beans 3 cans
[ ] Rice 1 bag
HOUSEHOLD
[ ] Paper Towels 1 pack
```
Formatting rules:
- Checked-off items show `[x]` with time until archive (24h countdown).
- Items without quantity/unit show just the name.
- Categories with zero unchecked and zero checked items are hidden entirely.
- Items are sorted alphabetically within each category.
- Category headers are UPPERCASE.
### shopping export
```
Shopping List
-------------
Produce: Avocados (3), Bananas (1 bunch)
Dairy: Whole Milk (2 gallons)
Meat: Chicken Breast (2 lbs)
Pantry: Black Beans (3 cans), Rice (1 bag)
Household: Paper Towels (1 pack)
```
Formatting rules:
- No emoji, no metadata, no checkboxes.
- Checked-off items are excluded.
- Output is plain text, copy-pasteable to Notes or Reminders.
### shopping categories
```
Categories (5 with items)
Produce 2 items
Dairy 2 items
Meat 1 item
Pantry 2 items
Household 1 item
```
Formatting rules:
- Only categories that currently have items are shown.
- Singular "item" when count is 1, plural "items" otherwise.
### shopping history
```
Recent Purchases (last 30 days)
Feb 23: Whole Milk (2 gallons), Eggs (1 dozen), Bread (1 loaf)
Feb 19: Chicken Breast (2 lbs), Rice (1 bag)
Feb 15: Whole Milk (2 gallons), Bananas (1 bunch)
```
Formatting rules:
- Groups by date, most recent first.
- If user says "history february" then filter to February only.
- Reads from `data/history-YYYY-MM.json` files.
### Confirmation Messages (mutating commands)
**shopping add (single item):**
```
Added: Whole Milk (2 gallons) — Dairy
```
**shopping add (multi-item):**
```
Added 3 items:
Eggs — Dairy
Bread — Pantry
Whole Milk (2 gallons) — Dairy
```
**shopping check:**
```
Checked off: Whole Milk — archiving in 24h
```
**shopping remove:**
```
Removed: Bananas
```
**shopping edit:**
```
Updated: Whole Milk — quantity: 1 → 2 gallons
```
**shopping clear:**
```
Archived 3 checked-off items.
```
**shopping switch-user:**
```
Switched to: shal
```
---
## 4. Edge Cases
### Fuzzy matching
- `"milk"` matches `"whole milk"` if it is the only milk on the list. Proceed without asking.
- `"milk"` matches both `"whole milk"` and `"oat milk"`. Ask: "Which one — whole milk or oat milk?"
- `"xyz"` matches nothing. Respond: "I don't see xyz on the list." Then show current items so the user can pick.
### Multi-item parsing
- `"add eggs, bread, and 2 gallons milk"` becomes 3 separate items: eggs, bread, 2 gallons milk.
- Ambiguous input like `"add 2 lbs chicken breast and thighs"` — unclear whether "2 lbs" applies to both or just chicken breast. Ask for clarification.
- Rule: when in doubt, ask. Never silently parse incorrectly.
### Quantity validation
- Quantity is optional. `"add milk"` creates an item with no quantity and no unit.
- `"add 2 milks"` sets quantity to 2 with no unit.
- If present, quantity must be > 0. Fractional values are allowed (e.g., 0.5 lbs).
- Zero or negative quantities are rejected: "Quantity must be greater than zero."
### Custom categories
- If the user specifies a category that does not exist in the preset list, create it automatically.
- Example: `"add vitamins to Supplements"` — if "Supplements" is not in the categories array, add it. Confirm: "Created new category: Supplements"
- Custom categories appear after the preset categories, sorted alphabetically among themselves.
### Force clear
- `shopping clear` archives ALL checked items immediately, regardless of the normal 24-hour auto-archive window. Use after a shopping trip to clean up the list.
- If there are no checked items: "Nothing to clear — no items are checked off."
### Empty list
- `shopping list` when the list has zero items: "Your shopping list is empty. Add something with 'shopping add <item>'."
### Phase 2 not ready
- `shopping suggest` before Phase 2 is implemented: "Restock suggestions aren't available yet. Keep using the list — I'll learn your patterns over time."
```