skills/daily-review-browser-bookmarks/SKILL.md
Process Chrome bookmarks as GTD + S3 inbox — export recent/unprocessed bookmarks, classify (action/reference/trash), apply S3 analysis to actions and clusters, update project docs, save to inbox file. Part of the daily review workflow.
npx skillsauth add razbakov/skills skills/daily-review-browser-bookmarksInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
⚠️ Migration in progress (razbakov/ikigai#73): Control Center has moved from Notion to GitHub Issues + Project v2. When this skill references creating/reading/updating Notion cards or pages, translate to GitHub equivalents:
- Create card →
gh issue create --repo <project-repo> --title "<title>" --label agent:<name> --body "<S3 body>"thengh project item-add 5 --owner razbakov --url <issue-url>- Read cards →
gh issue list --repo <repo> --label agent:<name> --state open(or across repos viagh search issues "org:razbakov label:agent:<name> state:open")- Update card status → move on board:
gh project item-editwith the Status field, or close viagh issue close- Board columns: Inbox → To do → In progress → To review → Done
- Do not call any
notion-*MCP tools — the Notion MCP is disabled.
Chrome bookmarks are a GTD inbox — links saved throughout the day for later processing. This step collects unprocessed bookmarks, classifies them, and routes actionable items to project docs. Action items and clusters get deeper analysis using Sociocracy 3.0's Tension → Driver → Requirement pattern.
Invoked as part of /daily-review or independently via /daily-review-browser-bookmarks.
~/Projects/ikigaisessions/~/Projects/~/Library/Application Support/Google/Chrome/Default/Bookmarks (JSON)inbox/.bookmarks-processed.json (list of already-processed bookmark URLs)Read the Chrome Bookmarks JSON file and extract bookmarks. Chrome stores date_added as microseconds since 1601-01-01 UTC (Chrome epoch).
python3 -c "
import json, datetime, os
CHROME_EPOCH = datetime.datetime(1601, 1, 1)
BOOKMARKS_PATH = os.path.expanduser('~/Library/Application Support/Google/Chrome/Default/Bookmarks')
PROCESSED_PATH = os.path.expanduser('~/Projects/ikigai/inbox/.bookmarks-processed.json')
# Load processed URLs
processed = set()
if os.path.exists(PROCESSED_PATH):
with open(PROCESSED_PATH) as f:
processed = set(json.load(f))
# Load bookmarks
with open(BOOKMARKS_PATH) as f:
data = json.load(f)
# Collect all bookmarks
all_bookmarks = []
def collect(node, folder=''):
if node.get('type') == 'url':
url = node.get('url', '')
if url not in processed:
dt = CHROME_EPOCH + datetime.timedelta(microseconds=int(node.get('date_added', '0')))
all_bookmarks.append({
'name': node.get('name', ''),
'url': url,
'date_added': dt.strftime('%Y-%m-%d %H:%M'),
'folder': folder,
})
for child in node.get('children', []):
collect(child, node.get('name', folder))
for key in data.get('roots', {}):
root = data['roots'][key]
if isinstance(root, dict):
collect(root)
# Sort by date descending
all_bookmarks.sort(key=lambda x: x['date_added'], reverse=True)
print(f'Total unprocessed: {len(all_bookmarks)}')
for b in all_bookmarks:
print(f'--- {b[\"date_added\"]} [{b[\"folder\"]}] ---')
print(f'{b[\"name\"]}')
print(f'{b[\"url\"]}')
print()
"
First run: All 1700+ bookmarks will be unprocessed. Handle this by focusing on recent bookmarks first (last 7 days), then ask the user if they want to process older ones in batches.
Filter strategy for daily review: By default, only process bookmarks added since the last daily review (yesterday). For the first run or catch-up, process the last 7 days.
For each unprocessed bookmark:
Hint: Use the bookmark name, URL domain, and folder to infer classification. Fetch the page title/description if the bookmark name is unclear.
S3 Deep Analysis (Action items and Clusters only):
For each item classified as Action, apply S3 Tension → Driver → Requirement:
For Reference, Trash, Someday items that are NOT part of a cluster: skip S3 analysis. GTD classification is sufficient.
After GTD classification, scan all items for shared tensions:
| # | Date | Title | URL (domain) | GTD | Project | Action | Driver (1-line) | Cluster |
|---|------|-------|--------------|-----|---------|--------|-----------------|---------|
| 1 | 03-21 | Shape Up | basecamp.com | Action | ikigai | Read & extract | Ad-hoc planning risks MVP | — |
| 2 | 03-20 | OpenFang | openfang.sh | Reference | ikigai | — | — | Agent Infra |
| 3 | 03-20 | inference.sh | inference.sh | Reference | ikigai | — | — | Agent Infra |
| 4 | 03-18 | Old Vue template | github.com | Trash | — | Remove bookmark | — | — |
Note: The Driver (1-line) column is blank for non-Action items. The Cluster column is blank for non-clustered items. Omit both columns entirely when there are no Action items or clusters in the batch.
For each actionable item:
README.md (or TODO.md, docs/backlog.yaml if they exist)Save to inbox/YYYY-MM-DD-HH-MM-bookmarks.md:
## Summary — counts with S3 note: "N actions (M with S3 analysis), N references, N clusters detected"## Bookmarks — per-bookmark blocks with GTD classification
#### S3 Analysis sub-block:
## Clusters (when detected) — grouped S3 analysis:
The daily review file (sessions/YYYY-MM-DD-daily-review.md) should only contain a link to the inbox file, not the full content.
For each processed bookmark classified as Reference or Action (read/evaluate) — create a card on the Reading Inbox Notion board.
Use the Notion MCP create-pages tool with:
40473b3d-377c-4ddf-b7c4-9407bfc65f72Name: Short descriptive titleStatus: "To Read" (for items marked Worth Reading: Yes) or "Unread" (for Maybe/Skim)Source: "Chrome Bookmark"Type: One of Tool, Article, Thread, Video, Reference, RepoSummary: 1-2 sentence summaryWhy It Matters: Driver summary — 1 sentence connecting item to mission/OKRs (for Action items, sourced from S3 Driver analysis; for Reference items, a brief relevance note)Project: JSON array of project names, e.g. ["ikigai", "WeDance"]Worth Reading: Yes / Skim / Maybe / NouserDefined:URL: The bookmark URLdate:Date Added:start: ISO date (YYYY-MM-DD)date:Date Added:is_datetime: 0For Action items, also include page content with S3 analysis:
## S3 Analysis
### Tension
[1-2 sentences — what dissonance prompted this]
### Driver
<table header-row="true">
<tr>
<td>Conditions</td>
<td>Effect</td>
<td>Relevance</td>
</tr>
<tr>
<td>[observable facts]</td>
<td>[consequences]</td>
<td>[why it matters for mission/OKRs]</td>
</tr>
</table>
### Requirement
> [who] needs [conditions] so that [outcomes]
### Response Options
- [ ] [concrete action A]
- [ ] [alternative approach B]
- [ ] [defer/skip option]
For clustered items: Each card in the cluster gets the same cluster-level S3 content, plus a callout at the top:
<callout icon="🔗" color="purple_bg">
**Cluster: [Name]** — related cards: [list sibling item names]
</callout>
For Reference items (non-clustered): Properties only, no page content. This is the current behavior.
Skip items that: are classified as Trash, are purely task-oriented, or have no reading value.
After processing, add all processed bookmark URLs to inbox/.bookmarks-processed.json:
python3 -c "
import json, os
PROCESSED_PATH = os.path.expanduser('~/Projects/ikigai/inbox/.bookmarks-processed.json')
processed = []
if os.path.exists(PROCESSED_PATH):
with open(PROCESSED_PATH) as f:
processed = json.load(f)
# Add newly processed URLs (passed as argument or read from the inbox file)
new_urls = [...] # URLs that were just processed
processed.extend(new_urls)
processed = list(set(processed)) # dedupe
with open(PROCESSED_PATH, 'w') as f:
json.dump(processed, f, indent=2)
"
Information flow: Chrome Bookmarks -> GTD classify -> S3 analysis (Action/Cluster only) -> inbox/ file (raw + processed + S3) -> Project README (source of truth) -> Notion Reading Inbox (properties + S3 page content for Actions) -> PROJECTS.md (cache, synced later) -> Daily Review (link only).
The GTD + S3 summary table, S3 analysis blocks for Action items/clusters, and a list of project docs that were updated.
development
Seed a new or empty Instagram account with a 9-post grid (3×3) so the profile looks established the moment a new visitor lands. Designed for festivals, new businesses, product launches, conferences, communities — any time an empty IG profile would hurt conversion from external traffic (QR scans, flyer drops, cross-promo). Generates assets via /image-from-gemini (per content-publishing rules — never HTML), writes captions with hashtag sets, and outputs a posting order + cadence plan. Trigger generously: phrases like '9 posts for instagram', 'fill my IG', 'starter grid', 'launch grid', 'instagram seed', '9-post grid', 'IG account not to look empty', 'first instagram posts', 'feed bootstrap', '3x3 grid', 'instagram launch content'. Even if the user mentions only one piece (just the images, just the captions, just the order), use this skill — the grid only works as an integrated bundle.
testing
Translate one English blog post into multiple target languages via parallel sub-agents, preserving frontmatter conventions, hero image, and brand voice. Use when the user shares a published English post URL or markdown path and says 'translate it', 'add other languages', 'publish in DE/ES/RU/UK', 'translate to 5 languages', or asks for localized versions of a specific post.
development
Build a complete press kit for an event, product launch, or campaign — in multiple languages — and publish it as a shareable Google Drive folder ready to send to journalists, partners, or a delegate. Produces press releases (typically DE/EN/ES, or configurable), uploads press photos and flyers, creates an Overview document for at-a-glance briefing, and creates a Handover document with pending tasks, contacts, risks, and decisions so press distribution can be delegated. Use when the user says 'I need a press release', 'create a press kit', 'press release in X languages', 'set up a Drive folder for press', 'handover doc for someone else to run press', or has an upcoming announcement that needs to be sent to media. Trigger generously: even partial requests (just a press release, just a flyer folder) typically evolve into the full kit.
development
Track ticket sales for a live event (concert, festival, conference, workshop) with daily snapshots, generate a burndown chart comparing actual sales to ideal-linear targets and tier-cumulative milestones, and report whether the event is on pace. Use when the user asks how sales are going, wants to know if their event will sell out, asks for a daily sales report, wants to set up sales tracking for an upcoming event, or asks about ticket pace / velocity / projection. Trigger generously: phrases like 'how is concert sales going', 'burndown for my event', 'are we going to sell out', 'sales velocity', 'daily ticket chart', 'how many tickets do we need to sell', or any case where the user has a ticketed event with a fixed sales window and wants visibility on pacing.