skills/anysite-mcp-migration/SKILL.md
Migrate anysite MCP skills, prompts, and agent instructions from v1 (individual tools like search_linkedin_users, get_linkedin_profile) to v2 (universal meta-tools execute, discover, get_page, query_cache, export_data). Automatically rewrites tool references, adds pagination/filtering/export capabilities, and validates migrated output. Use when users need to update existing skills or prompts for the new anysite MCP v2 API, convert old tool calls to execute() format, or adapt workflows to use new v2 features like server-side filtering and data export.
npx skillsauth add anysiteio/agent-skills anysite-mcp-migrationInstall 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.
Migrate your anysite MCP skills, prompts, and agent instructions from v1 (individual tools) to v2 (universal meta-tools).
Anysite MCP v2 replaces 70+ individual tools with 5 universal meta-tools. This skill helps you:
search_linkedin_users, get_linkedin_profile) to new execute() callssearch_*, get_*, find_*)Ask the user for one of:
references/Scan the input for any of these v1 tool name patterns:
| Pattern | Example |
|---------|---------|
| search_linkedin_* | search_linkedin_users, search_linkedin_companies, search_linkedin_jobs, search_linkedin_posts |
| get_linkedin_* | get_linkedin_profile, get_linkedin_company, get_linkedin_user_posts |
| find_linkedin_* | find_linkedin_email, find_linkedin_user_email |
| google_linkedin_* | google_linkedin_search |
| search_twitter_* / get_twitter_* | search_twitter_users, get_twitter_user, get_twitter_user_tweets |
| search_instagram_* / get_instagram_* | search_instagram_users, get_instagram_user, get_instagram_post |
| search_youtube_* / get_youtube_* | search_youtube, get_youtube_channel, get_youtube_video |
| search_reddit_* / get_reddit_* | search_reddit, get_reddit_user, get_reddit_posts |
| search_yc_* / get_yc_* | search_yc_companies, get_yc_company |
| search_sec_* / get_sec_* | search_sec_filings, get_sec_document |
| scrape_webpage | scrape_webpage |
| mcp__anysite__* | Any tool with the MCP prefix — strip prefix and match above |
Also look for references to Crunchbase — this source is disabled in v2 and must be removed.
Replace each old tool call using this mapping:
| Old tool | New execute() call |
|----------|-------------------|
| search_linkedin_users(keywords, location, count, ...) | execute("linkedin", "search", "search_users", {"keywords": ..., "location": ..., "count": ...}) |
| get_linkedin_profile(user) | execute("linkedin", "user", "get", {"user": ...}) |
| get_linkedin_company(company) | execute("linkedin", "company", "get", {"company": ...}) |
| search_linkedin_companies(keywords, count) | execute("linkedin", "search", "search_companies", {"keywords": ..., "count": ...}) |
| search_linkedin_jobs(keywords, location, count) | execute("linkedin", "job_search", "search_jobs", {"keywords": ..., "count": ...}) |
| search_linkedin_posts(keywords, count) | execute("linkedin", "post", "search_posts", {"keywords": ..., "count": ...}) |
| get_linkedin_user_posts(user) | execute("linkedin", "post", "get_user_posts", {"user": ...}) |
| find_linkedin_email(user) | execute("linkedin", "email", "find", {"user": ...}) |
| google_linkedin_search(query, count) | execute("linkedin", "google", "search", {"query": ..., "count": ...}) |
| Old tool | New execute() call |
|----------|-------------------|
| search_twitter_users(query) | execute("twitter", "search", "search_users", {"query": ...}) |
| get_twitter_user(username) | execute("twitter", "user", "get", {"username": ...}) |
| get_twitter_user_tweets(username) | execute("twitter", "user_tweets", "get", {"username": ...}) |
| Old tool | New execute() call |
|----------|-------------------|
| search_instagram_users(query) | execute("instagram", "search", "search_users", {"query": ...}) |
| get_instagram_user(username) | execute("instagram", "user", "get", {"username": ...}) |
| get_instagram_post(url) | execute("instagram", "post", "get", {"url": ...}) |
| Old tool | New execute() call |
|----------|-------------------|
| search_youtube(query, count) | execute("youtube", "search", "search_videos", {"query": ..., "count": ...}) |
| get_youtube_channel(channel_id) | execute("youtube", "channel", "get", {"channel_id": ...}) |
| get_youtube_video(video_id) | execute("youtube", "video", "get", {"video_id": ...}) |
| Old tool | New execute() call |
|----------|-------------------|
| search_reddit(query) | execute("reddit", "search", "search", {"query": ...}) |
| get_reddit_user(username) | execute("reddit", "user", "get", {"username": ...}) |
| get_reddit_posts(subreddit) | execute("reddit", "posts", "get", {"subreddit": ...}) |
| Old tool | New execute() call |
|----------|-------------------|
| search_yc_companies(query) | execute("yc", "search", "search", {"query": ...}) |
| get_yc_company(slug) | execute("yc", "company", "get", {"slug": ...}) |
| search_sec_filings(query) | execute("sec", "search", "search", {"query": ...}) |
| get_sec_document(url) | execute("sec", "document", "get", {"url": ...}) |
| scrape_webpage(url) | execute("webparser", "parse", "parse", {"url": ...}) |
If the input references a tool name not in the mapping above:
get_instagram_user_friendships → source "instagram", category "user" or "friendship")discover() yourself right now — do NOT leave placeholder {endpoint} in the migrated output. Run discover("{source}", "{category}") via the MCP tool to get the real endpoint names and parameter schemas."friendship" fails, try "user" — the endpoint may be nested under a different category)discover(), use the exact endpoint name and params in the migrated execute() callExample — resolving an unknown tool:
Old tool: get_instagram_user_friendships(user, type, count)
→ Not in mapping table
→ You call: discover("instagram", "friendship") → error "Category not found"
→ You call: discover("instagram", "user") → returns endpoints including "user_friendships"
→ Migrated: execute("instagram", "user", "user_friendships", {"user": "...", "count": 100, "type": "followers"})
Rules:
discover() as a placeholder instruction in the final migrated skill. The migrated output must contain exact execute() calls with real endpoint names and params.discover() in the migrated skill text if the skill's workflow genuinely needs runtime discovery (e.g., the skill works with user-specified sources where the endpoint can't be known at migration time).execute() directly — no discover needed.Review the migrated skill for opportunities to add new v2 features:
When the workflow processes large result sets or needs "more results":
Results from execute() include cache_key. If more data exists, use:
get_page(cache_key="{cache_key}", offset=10, limit=10)
When the workflow filters results after fetching (e.g., "only show people in SF"):
After execute(), filter without consuming context tokens:
query_cache(cache_key="{cache_key}", conditions=[{"field": "location", "op": "contains", "value": "San Francisco"}])
When the workflow computes statistics or groups data:
query_cache(cache_key="{cache_key}", aggregate={"field": "followers", "op": "avg"}, group_by="industry")
When the workflow outputs structured data (CSV, JSON) or the user needs downloadable results:
export_data(cache_key="{cache_key}", output_format="csv")
→ returns download URL
Replace old-style error handling:
Before:
If search_linkedin_users returns an error, try with different keywords.
After:
If execute() returns an error with "llm_hint", follow the hint.
If execute() returns {"error": "Source not found", "available_sources": [...]}, check source name.
If execute() returns {"error": "Endpoint not found", "available_endpoints": [...]}, call discover() to find correct endpoint names.
Remove any references to Crunchbase — this source is disabled in v2. If the skill relies on Crunchbase data, suggest alternatives:
Run through this checklist on the migrated output:
search_linkedin_*, get_linkedin_*, find_linkedin_* referencessearch_twitter_*, get_twitter_* referencessearch_instagram_*, get_instagram_* referencessearch_youtube_*, get_youtube_* referencessearch_reddit_*, get_reddit_* referencessearch_yc_*, get_yc_* referencessearch_sec_*, get_sec_* referencesscrape_webpage referencesmcp__anysite__ prefixed tool names (old MCP format)discover() added only where endpoint/params are genuinely unknownget_page / query_cache / export_data added where beneficialPresent the migrated skill in the same format as the input:
Always show a migration summary after the output:
## Migration Summary
- Tool calls replaced: N
- discover() calls added: N
- New v2 features added: [list]
- Crunchbase references removed: N
- Warnings: [any issues found]
Input:
Use search_linkedin_users to find CTOs in Berlin, then get_linkedin_profile for each result.
Output:
Use execute("linkedin", "search", "search_users", {"title": "CTO", "location": "Berlin", "count": 10}) to find CTOs in Berlin, then execute("linkedin", "user", "get", {"user": "{alias}"}) for each result.
Input:
1. Use search_linkedin_users to find the person
2. Use get_linkedin_profile to get their full profile
3. Use find_linkedin_email to get their email
4. Use get_twitter_user to check their Twitter
Output:
1. Use execute("linkedin", "search", "search_users", {"first_name": ..., "last_name": ..., "count": 5}) to find the person
2. Use execute("linkedin", "user", "get", {"user": "{alias from step 1}"}) to get their full profile
3. Use execute("linkedin", "email", "find", {"user": "{alias from step 1}"}) to get their email
4. Use execute("twitter", "user", "get", {"username": "..."}) to check their Twitter
Input:
Search for 50 marketing managers and filter by location.
Output:
1. Use execute("linkedin", "search", "search_users", {"title": "Marketing Manager", "count": 50}) to search
2. If more results exist, use get_page(cache_key="{cache_key}", offset=10, limit=10) to load additional pages
3. Use query_cache(cache_key="{cache_key}", conditions=[{"field": "location", "op": "contains", "value": "..."}]) to filter by location server-side
4. Use export_data(cache_key="{cache_key}", output_format="csv") to download the filtered list
| Tool | Purpose | When to use |
|------|---------|-------------|
| discover(source, category) | Learn available endpoints and params | Before execute() when endpoint name or params are unknown |
| execute(source, category, endpoint, params) | Fetch data from any source | Every data retrieval — replaces all v1 tools |
| get_page(cache_key, offset, limit) | Load more items from previous execute() | When execute() returned next_offset |
| query_cache(cache_key, conditions, sort_by, aggregate, group_by) | Filter/sort/aggregate cached data | When you need to slice results without re-fetching |
| export_data(cache_key, format) | Save dataset as downloadable file | When user needs CSV/JSON/JSONL export |
tools
Read-only static security audit of Claude Code skills, commands, and plugins. Analyzes SKILL.md frontmatter, body content, supporting scripts, and hooks for security risks. Use this skill when the user asks to "audit a skill", "review skill security", "check SKILL.md for risks", "scan a plugin for dangerous patterns", "verify skill safety", "check skill permissions", "analyze skill hooks", "audit a skill from GitHub", "review a remote skill", "check a skill by URL", or needs a security assessment of any Claude Code skill, command, or plugin before enabling it.
tools
Build a positioning map for 3–5 competitors and identify the empty quadrant the founder could own. Use when a founder asks "where's the positioning gap?", "how do I position against X?", "what's the competitive landscape look like on hero / pricing / hiring / customers?", or needs a structured comparison before a launch, repositioning, or fundraise. Combines Anysite MCP (LinkedIn company entity + post search + jobs search; SEC for late-stage) with Exa MCP (fetch JS-heavy SaaS marketing pages, find case studies and changelogs). Returns a comparison table across 5 axes (hero / pricing / specialities / recent shipping / hiring), 3 candidate positioning moves with explicit choice criteria, and a one-sentence positioning statement. Requires customer pain themes as input — positioning without pain context is just rearranging marketing copy. Run competitor-discovery and customer-pain-mining first if you don't have a curated competitor list + pain themes.
tools
Extract verbatim customer complaints about competitors — the exact wording the founder's product copy should steal, in customers' own words. Use when a founder asks "what do users hate about X?", "what's broken about [category]?", "what's the white space?", or needs raw customer language for landing-page copy, custdev prep, ad copy, or product strategy. Combines Anysite MCP (Reddit broad sweeps, LinkedIn issue-level pain search, YouTube comments under review videos, Twitter for viral pain quotes) with Exa MCP (semantic search for review blog posts, comparison articles, "why I left X" Medium posts). Returns 3–5 pain clusters with 2–3 verbatim quotes each plus a white-space section listing features customers ask for that no competitor ships. Run competitor-discovery first if there's no validated competitor list.
tools
--- name: competitor-discovery description: Find a startup's REAL competitors — the alternatives customers actually compare them to, not just the names on the pitch deck. Use when a founder asks "who are my real competitors?", asks "what does the choice landscape look like for X?", wants to validate or expand a pitch-deck competitor slide, or needs to map alternatives before pain mining or positioning work. Combines Anysite MCP (LinkedIn company/search, LinkedIn posts `mentioned[]` extraction +