goalplanner-share-mcp
An MCP server that generates RuneLite Goal Planner import strings from natural language goal descriptions, enabling assistants to create paste-ready share codes for Old School RuneScape goal planning.
README
goalplanner-share-mcp
An MCP server that crafts RuneLite Goal Planner
import strings (GPSHARE1: codes) from a structured, natural-language-friendly goal
spec — so an assistant can turn "plan my Inferno prep: 90 Ranged, 70 Defence, then beat
the Inferno" into a paste-ready code, confirming your intent before it emits anything.
Pure string generator — no plugin changes required. Verified byte-compatible with the
plugin's own com.goalplanner.share.ShareCodec in both directions (the plugin decodes
codes this tool produces, and this tool decodes codes the plugin produces).
What it does
-
Two modes: import as a new named section, or as loose goals (which land in a "Shared goals" section).
-
Multi-section codes (GPSHARE2): pass
sections[]and ONE code carries several sections — each imports as its own section, in one undo. A section withtargetDefault: truelands in the recipient's Default plan instead, REUSING existing equivalent goals (the in-game add dedup), so re-importing never duplicates. Single-section codes still emit theGPSHARE1:wire, which every plugin build imports; multi-section/default-target codes need a recent plugin build. -
Cross-section dependencies: in the
sections[]form, a goal mayrequires/orRequiresa goal in a different section by its explicitid. The edge rides the bundle-levelcrossEdgeswire field (mirroring the plugin'sCrossEdgeDto) and is rewired on import. Section-local ids always win; an id found in several other sections is ambiguous and dropped with a warning; cycles are checked across the whole bundle. The preview marks the link on the dependent goal:═══ Section 2/2: "Inferno prep" ═══ Ranged - Level 90 [Skill · Level 90 (5,346,332 xp)] ✓ auto-tracks ↪ needs "Imbued heart" — from section 1 "Slayer" TzKal-Zuk [Boss · TzKal-Zuk · 1 KC] ✓ auto-tracks ◀ final goal -
Simple goals or complex trees: goals are wired into prerequisite trees (AND via
requires, OR viaorRequires) by stableid. Diamonds and OR-groups are supported. -
Hybrid typing: recognized kinds become typed, auto-tracking goals; anything else falls back to a CUSTOM goal (imports fine, manual check-off). Every fallback, dropped edge, or cycle is reported as a warning.
-
Confirm-first, preview by default:
craft_import_stringrenders the goal list as it will import — section header, each goal with its type/target, prerequisites nested as a guide tree, and per-goal tracking badges — with no code emitted. The user eyeballs it and adjusts; the code is produced only on a follow-up call withconfirm: true.The list is rendered in the same order and nesting the plugin shows — do-first prerequisites flush-left at the top; the dependent goal indented beneath them, with the final goal at the bottom:
┌─ Goal Planner import preview ───────────────── │ Section: "Inferno prep" (created fresh on import; completed goals kept inline) │ 4 goal(s) · 4 auto-track · 0 manual/unverified │ Order: do-first at top → final goal at bottom (as shown in-game) └─────────────────────────────────────────────── Ranged - Level 90 [Skill · Level 90 (5,346,332 xp)] ✓ auto-tracks Defence - Level 70 [Skill · Level 70 (737,627 xp)] ✓ auto-tracks Beat the Fight Caves [Boss · TzTok-Jad · 1 KC] ✓ auto-tracks Beat the Inferno [Boss · TzKal-Zuk · 1 KC] ✓ auto-tracks ◀ final goal
Goal coverage
| Status | Types | Notes |
|---|---|---|
| ✅ Typed core (auto-tracks) | SKILL, BOSS, ITEM_GRIND, DIARY, QUEST, ACCOUNT, COMBAT_ACHIEVEMENT, CUSTOM |
SKILL by level or XP (all 23 skills); BOSS by name (all 89 tracked bosses + aliases), KC target defaults to 1; ITEM_GRIND by item name against the full OSRS item table (or explicit itemId); DIARY by "<Area> <Tier>" name across 12 areas × 4 tiers (or explicit known varbitId); QUEST by display name/abbreviation (209 quests+miniquests, wire carries the RuneLite Quest enum constant); ACCOUNT by metric name/shorthand (16 plugin AccountMetrics incl. Collection Log Slots and Diary Tiers; phrases like "maintain quest cape" imply metric AND milestone — quest-point max is 335 as of The Red Reef; out-of-range targets warn but emit, matching the plugin's allow-over-max behaviour; missing target = max); COMBAT_ACHIEVEMENT by exact task name (637 tasks, wire carries caTaskId 0–639, tier sprite + description match in-game-created goals) |
| 🧩 Group expansion (one phrase → many goals) | item sets/loadouts, boss & diary groups | full torva → 3, maxed melee setup → 9; GWD → 4 bosses, Dagannoth Kings → 3, all bosses → 89; all elite diaries → 12, all Ardougne diaries → 4, all diaries → 48 |
| 🔶 Passthrough (unverified) | unknown questName / accountMetric / caTaskId / varbitId / itemId identifiers |
emitted as supplied with an UNVERIFIED warning; unresolvable names fall back to CUSTOM with did-you-mean suggestions |
| 🗺️ Roadmap | CA tier groups (all easy CAs), quest groups (all f2p quests) |
Boss names are generated from the plugin's BossKillData via npm run gen:bosses
(reads $GOAL_PLANNER_REPO). The item table is generated from the OSRS cache
objtypes.txt (JayArrowz mcp-osrs) via npm run gen:items (auto-discovers the
mcp-osrs data dir, or set $OSRS_DATA_DIR) — placeholder_/cert_ variants filtered
out since the plugin tracks an exact itemId. Item names that diverge from their
internal codename (potions, Cannonball, …) resolve via a curated alias map or by you
passing an itemId you looked up on the OSRS Wiki. A second generated layer
(npm run gen:item-names, wiki prices mapping) adds authoritative display names for ~4.5k
tradeables — so the codename-divergent tail (Armadyl crossbow, Amulet of torture, Voidwaker
pieces) resolves without curation. Community nicknames (tbow, bp,
shadow, scythe, zcb, dhcb, fero, rancour, blorva, fang kit, …) resolve by NAME
REFERENCE through the generated tables (never hand-typed ids) and armour sets (full torva, fortified masori) are recognised too;
loadout presets (maxed melee setup, maxed ranged, maxed mage) expand to a full BiS-ish
kit; and a +/and-joined phrase (full masori + tbow, maxed melee + shadow) fans out into
one auto-tracking item goal per piece (visible in the preview before you confirm).
The diary table is generated via npm run gen:diaries, which joins two sources: the
plugin's AchievementDiaryData (area/tier structure + required values) with the numeric
varbit ids from the OSRS cache varbittypes.txt — the symbolic VarbitID.<AREA>_DIARY_<TIER>_COMPLETE
constants are matched by name to their cache ids (the runtime varbit the recipient reads).
The loadout presets are generated via npm run gen:loadouts (needs network), a hybrid: armour
slots come from the OSRS Wiki Armour/Highest bonuses tables (so they stay current — e.g. Amulet
of rancour), resolved to ids via the wiki's prices-mapping API; the weapon + cape are curated
because the wiki ranks weapons by raw bonus, which picks slow non-DPS weapons (Zombie axe, Kodai
wand). Loadout member ids can be newer than the objtypes snapshot, so they're treated as known.
The quest table is generated via npm run gen:quests, which runs the real RuneLite Quest
enum (values()/name()/getName()) from the version-matched runelite-api jar in the local
gradle cache — the recipient's QuestTracker does Quest.valueOf(questName), so the wire must carry
the enum constant (DRAGON_SLAYER_II), and running the enum keeps the constant↔display pairing
from ever drifting. The account-metric table (npm run gen:accounts) parses the plugin's
AccountMetric.java (the tracker's AccountMetric.valueOf constants + each metric's sensible
target range + leagues flags). The CA table (npm run gen:cas, needs network) fetches the OSRS
Wiki combat_achievement bucket — the same table the plugin's WikiCaRepository loads —
where the bucket id is the bit index (0–639) into the CA_TASK_COMPLETED varplayers.
Cross-language parity for all three Phase-2 types is proven the same way as Phase 1: the plugin's
real ShareCodec decoded a TS-crafted code and Quest.valueOf / AccountMetric.valueOf /
the caTaskId range check resolved on the Java side (throwaway JUnit test, removed after the run).
Real-world corpus test
test/clan-corpus.test.ts runs ~140 labeled goals collected from a clan Discord
(test/fixtures/clan-discord-goals-raw.txt, verbatim) through the full builder — community
shorthand, KC goals, account milestones ("Elite CAs" → CA points @ 1064), and the lines that
legitimately fall to CUSTOM (greenlogs, outfits, minigames). Add new community examples there;
the test names each line so a regression reads as English.
Tools
-
craft_import_string—{ mode?, sectionName?, sectionColorRgb?, sharedBy?, goals[]?, sections[]?, confirm? }(sections[]= multi-section/default-target form; each entry is{ name?, sectionColorRgb?, targetDefault?, goals[] }). Withoutconfirm: human-readable preview + warnings, no code. Withconfirm: true: the paste-ready code (GPSHARE1:single-section,GPSHARE2:multi-section). -
decode_import_string—{ code }. Decodes anyGPSHARE1:/GPSHARE2:code (even embedded in surrounding text) into a readable breakdown for verification — sections, identifiers, prerequisite tree, per-goaldesc:/tooltip:echo, and a Cross-section dependencies list with resolved goal names:── Cross-section dependencies (1) ── • "TzKal-Zuk" (section 2 "Inferno prep") requires "Imbued heart" (section 1 "Slayer") -
list_supported_goals— what auto-tracks vs. falls back, plus the skill names.
Goal spec fields
id, type ("skill" / "custom" / a GoalType name), name, description, requires[],
orRequires[], and per kind: skills use skill + level or xp; CUSTOM uses colorRgb,
tooltip; ITEM_GRIND uses name (resolved to an itemId) or an explicit itemId, plus
targetValue (quantity); QUEST uses name (resolved) or explicit questName (enum constant);
ACCOUNT uses name or explicit accountMetric plus targetValue; COMBAT_ACHIEVEMENT uses name
(exact task) or explicit caTaskId.
Develop
npm install
npm test # vitest unit + codec/build tests
npm run build # tsc → dist/
node test/smoke.mjs # end-to-end MCP stdio smoke test (after build)
Register in Claude
After npm run build, register the server. Project scope — a .mcp.json at a project
root (committable; activates when Claude Code runs in that project):
{
"mcpServers": {
"goalplanner-share": { "type": "stdio", "command": "node", "args": ["dist/index.js"] }
}
}
Use a relative dist/index.js only in this repo's own .mcp.json (cwd = repo root). For a
.mcp.json in any other project, or for user scope (~/.claude.json → top-level
mcpServers, available everywhere), use the absolute path:
{
"mcpServers": {
"goalplanner-share": {
"type": "stdio",
"command": "node",
"args": ["/absolute/path/to/goalplanner-share-mcp/dist/index.js"]
}
}
}
The server loads at Claude Code startup, so a newly-registered server appears in the next
session. Claude Desktop uses the same block in claude_desktop_config.json.
Format
GPSHARE1:<base64url-nopad( gzip( JSON of ShareBundle ) )>
Mirrors the plugin's ShareBundle / GoalShareDto / TagShareDto. The importer is
tolerant: unknown goal types are skipped, strings are length-clamped, edges pointing
outside the bundle are dropped, and every import lands in a new user section.
Recommended Servers
playwright-mcp
A Model Context Protocol server that enables LLMs to interact with web pages through structured accessibility snapshots without requiring vision models or screenshots.
Magic Component Platform (MCP)
An AI-powered tool that generates modern UI components from natural language descriptions, integrating with popular IDEs to streamline UI development workflow.
Audiense Insights MCP Server
Enables interaction with Audiense Insights accounts via the Model Context Protocol, facilitating the extraction and analysis of marketing insights and audience data including demographics, behavior, and influencer engagement.
VeyraX MCP
Single MCP tool to connect all your favorite tools: Gmail, Calendar and 40 more.
graphlit-mcp-server
The Model Context Protocol (MCP) Server enables integration between MCP clients and the Graphlit service. Ingest anything from Slack to Gmail to podcast feeds, in addition to web crawling, into a Graphlit project - and then retrieve relevant contents from the MCP client.
Kagi MCP Server
An MCP server that integrates Kagi search capabilities with Claude AI, enabling Claude to perform real-time web searches when answering questions that require up-to-date information.
E2B
Using MCP to run code via e2b.
Neon Database
MCP server for interacting with Neon Management API and databases
Exa Search
A Model Context Protocol (MCP) server lets AI assistants like Claude use the Exa AI Search API for web searches. This setup allows AI models to get real-time web information in a safe and controlled way.
Qdrant Server
This repository is an example of how to create a MCP server for Qdrant, a vector search engine.