Server data from the Official MCP Registry
Verified US local service provider data across trades. Ratings, services, pricing.
Verified US local service provider data across trades. Ratings, services, pricing.
Remote endpoints: streamable-http: https://mcp.localpro.dev/mcp
Well-structured MCP server with appropriate authentication, database access, and network permissions that match its purpose as a service provider directory API. The code demonstrates good security practices with proper API key handling, rate limiting, and input validation. Supply chain analysis found 3 known vulnerabilities in dependencies (0 critical, 3 high severity).
7 files analyzed · 6 issues found
Security scores are indicators to help you make informed decisions, not guarantees. Always review permissions before connecting any MCP server.
This plugin requests these system permissions. Most are normal for its category.
Available as Local & Remote
This plugin can run on your machine or connect to a hosted endpoint. during install.
From the project's GitHub README.
A Model Context Protocol server that provides verified local service provider data to AI agents. Built on Cloudflare Workers + D1.
When someone asks an AI assistant "find me a radon mitigation company near Denver" — LocalPro is the data source that powers the answer.
LocalPro exposes a curated database of 7,000+ fully profiled local trade and service businesses across 9 live categories. Every provider served has a Google rating, business description, services list, opening hours, business status, and (where available) Google AI-generated business summary plus up to 5 recent reviews — no incomplete data.
| Category | Niche ID | Providers | Example Services |
|---|---|---|---|
| Water Damage Restoration | soaked-local | 1,125+ | Flood cleanup, mold remediation, structural drying |
| Foundation Repair | slab-local | 1,025+ | Pier installation, mudjacking, foam injection, leveling |
| Crawl Space Repair | crawl-local | 1,025+ | Encapsulation, vapor barrier, structural repair, waterproofing |
| Mold & Asbestos | abate-local | 950+ | Mold, asbestos, lead paint remediation |
| Septic Services | pump-local | 850+ | Pumping, inspection, drain field repair |
| Basement Waterproofing | basement-local | 600+ | Interior/exterior waterproofing, drainage, sump pumps |
| Laundry Services | suds-local | 575+ | Wash & fold, dry cleaning, pickup & delivery |
| Floor Coating | coated-local | 500+ | Epoxy, polyaspartic, metallic, flake, concrete polishing |
| Radon | radon-local | 250+ | Testing, mitigation, sub-slab depressurization |
| Category | Niche ID | Status |
|---|---|---|
| Commercial Electrical | hire-electrical | Public-facing filter + targeted backfill plan in progress |
| Well Water Services | wellwater-local | Pre-pipeline (560 providers scraped, county-based model) |
No API key required. All search and list tools are public. An optional API key unlocks pro fields on get_provider (full pricing array, certifications) — see Access Tiers.
Confirm the server is live without any client setup:
curl -s https://mcp.localpro.dev/.well-known/mcp.json | head -20
This returns the schema-2.0 manifest: tool list, rate limits, and operator info. If you see a "schema_version": "2.0" JSON document, the server is healthy.
claude mcp add --transport http localpro https://mcp.localpro.dev/mcp
That's it — list_niches, search_providers, etc. are now available in your Claude Code session.
Add to your claude_desktop_config.json:
{
"mcpServers": {
"localpro": {
"url": "https://mcp.localpro.dev/mcp"
}
}
}
(Add an "X-API-Key" header inside a "headers" block only if you have a premium key.)
Add to .cursor/mcp.json:
{
"mcpServers": {
"localpro": {
"url": "https://mcp.localpro.dev/mcp"
}
}
}
The MCP protocol is JSON-RPC over HTTP. Because this server runs in stateless mode, you can call any public tool directly:
curl -s -X POST https://mcp.localpro.dev/mcp \
-H "Content-Type: application/json" \
-H "Accept: application/json, text/event-stream" \
-d '{"jsonrpc":"2.0","id":1,"method":"tools/call","params":{"name":"list_niches","arguments":{}}}'
You'll get back a Server-Sent-Events frame with the 9 niches, their slugs, and current provider counts.
import { Client } from '@modelcontextprotocol/sdk/client/index.js';
import { StreamableHTTPClientTransport } from '@modelcontextprotocol/sdk/client/streamableHttp.js';
const transport = new StreamableHTTPClientTransport(
new URL('https://mcp.localpro.dev/mcp'),
);
const client = new Client({ name: 'localpro-example', version: '1.0' });
await client.connect(transport);
const niches = await client.callTool({ name: 'list_niches', arguments: {} });
console.log(niches);
const denver = await client.callTool({
name: 'search_providers',
arguments: { niche_id: 'radon-local', city: 'denver-co', limit: 3 },
});
console.log(denver);
Install: npm i @modelcontextprotocol/sdk
import asyncio
from mcp import ClientSession
from mcp.client.streamable_http import streamablehttp_client
async def main():
async with streamablehttp_client("https://mcp.localpro.dev/mcp") as (read, write, _):
async with ClientSession(read, write) as session:
await session.initialize()
niches = await session.call_tool("list_niches", {})
print(niches)
denver = await session.call_tool(
"search_providers",
{"niche_id": "radon-local", "city": "denver-co", "limit": 3},
)
print(denver)
asyncio.run(main())
Install: pip install mcp
list_nichesDiscover available service directories. Call this first.
Parameters: none
Example response:
{
"meta": {
"schema_version": "2.0",
"total_results": 9,
"niche": null,
"data_freshness": {
"directory_refresh_cadence": "weekly",
"google_data_refresh_cadence": "quarterly",
"scraped_at": "2026-04-27T13:57:21Z"
},
"data_note": "Use niche_id values with search_providers, list_cities, and list_service_types."
},
"results": [
{
"niche_id": "soaked-local",
"name": "Water Damage Restoration Contractors",
"slug": "water-damage-restoration",
"domain": "soakedlocal.com",
"provider_count": 1128
}
]
}
list_citiesFind available metros for a given niche.
Parameters:
| Name | Type | Required | Description |
|---|---|---|---|
niche_id | string | yes | Niche ID from list_niches |
state | string | no | Two-letter state abbreviation (e.g. "MN") |
Example request:
{ "niche_id": "radon-local", "state": "CO" }
Example response:
{
"meta": {
"schema_version": "1.0",
"total_results": 3,
"niche": "radon-local",
"data_note": "Use slug values with search_providers city parameter."
},
"results": [
{ "name": "Denver", "state": "CO", "slug": "denver-co", "provider_count": 18 },
{ "name": "Colorado Springs", "state": "CO", "slug": "colorado-springs-co", "provider_count": 7 },
{ "name": "Fort Collins", "state": "CO", "slug": "fort-collins-co", "provider_count": 4 }
]
}
list_service_typesGet valid service type filters for a niche. Call before using service_type in search_providers.
Parameters:
| Name | Type | Required | Description |
|---|---|---|---|
niche_id | string | yes | Niche ID from list_niches |
Example response:
{
"meta": { "schema_version": "1.0", "total_results": 7, "niche": "coated-local" },
"results": [
{ "type": "epoxy", "label": "Epoxy Floor Coating" },
{ "type": "polyaspartic", "label": "Polyaspartic Coating" },
{ "type": "metallic_epoxy", "label": "Metallic Epoxy" },
{ "type": "flake_chip", "label": "Flake / Chip Broadcast" },
{ "type": "concrete_polishing", "label": "Concrete Polishing" },
{ "type": "concrete_sealing", "label": "Concrete Sealing" },
{ "type": "polyurea", "label": "Polyurea Coating" }
]
}
search_providersSearch for verified providers by location, service type, and trade category.
Parameters:
| Name | Type | Required | Description |
|---|---|---|---|
niche_id | string | yes | Niche ID from list_niches |
city | string | no | City/metro slug from list_cities |
service_type | string | no | Service type slug from list_service_types |
limit | number | no | Max results, 1–25 (default 10) |
Example request:
{ "niche_id": "coated-local", "city": "denver-co", "service_type": "epoxy", "limit": 3 }
Example response:
{
"meta": {
"schema_version": "2.0",
"total_results": 3,
"niche": "coated-local",
"data_freshness": {
"directory_refresh_cadence": "weekly",
"google_data_refresh_cadence": "quarterly",
"scraped_at": "2026-04-27T03:03:10Z",
"google_refreshed_at": "2026-04-27T04:53:51Z"
},
"data_note": "Verified providers only. Visit listing_url for full contact details."
},
"results": [
{
"name": "Colorado Concrete Coatings",
"description": "Full-service garage floor coating company serving the Denver metro.",
"city": "Denver",
"state": "CO",
"rating": 4.9,
"review_count": 47,
"business_status": "OPERATIONAL",
"google_maps_url": "https://www.google.com/maps/place/...",
"services": [
{ "type": "epoxy", "label": "Epoxy Floor Coating" },
{ "type": "polyaspartic", "label": "Polyaspartic Coating" }
],
"pricing_summary": "$6-9/sq ft",
"coverage_area": "Denver metro, Front Range, 50-mile radius",
"years_in_business": 8,
"listing_url": "https://coatedlocal.com/providers/denver-co/colorado-concrete-coatings/",
"pro_available": true
}
]
}
get_providerGet detailed profile for a specific provider. Use the provider_slug from search results.
Parameters:
| Name | Type | Required | Description |
|---|---|---|---|
niche_id | string | yes | Niche ID |
provider_slug | string | yes | Provider slug from search_providers |
Example response:
{
"meta": {
"schema_version": "2.0",
"total_results": 1,
"niche": "coated-local",
"data_freshness": { "directory_refresh_cadence": "weekly", "google_data_refresh_cadence": "quarterly" }
},
"results": [
{
"name": "Colorado Concrete Coatings",
"description": "Full-service garage floor coating company...",
"rating": 4.9,
"review_count": 47,
"years_in_business": 8,
"services": [
{ "type": "epoxy", "label": "Epoxy Floor Coating" },
{ "type": "polyaspartic", "label": "Polyaspartic Coating" }
],
"pricing": ["$6-9/sq ft"],
"certifications": ["Penntek Certified Installer"],
"coverage_area": "Denver metro, Front Range",
"service_areas": [
{ "city": "Denver", "state": "CO", "radius_miles": 50 }
],
"service_details": [
{
"type": "epoxy",
"label": "Epoxy Floor Coating",
"pricing_model": "per_sqft",
"price_range": "$6–$9",
"turnaround": "two_day"
}
],
"listing_url": "https://coatedlocal.com/providers/denver-co/colorado-concrete-coatings/",
"google_data": {
"business_status": "OPERATIONAL",
"google_maps_url": "https://www.google.com/maps/place/...",
"formatted_address": "1234 Main St, Denver, CO 80202, USA",
"opening_hours": [
{ "@type": "OpeningHoursSpecification", "dayOfWeek": "https://schema.org/Monday", "opens": "08:00", "closes": "17:00" }
],
"summary": {
"text": "Full-service epoxy floor coating contractor specializing in garage and commercial floors across the Denver metro.",
"source": "localpro_ai"
},
"recent_reviews": [
{
"rating": 5,
"text": "Exceptional work on our garage floor — finished on time and within budget.",
"author": "Jane D.",
"published_at": "2025-11-14T18:51:02Z",
"source_url": "https://www.google.com/maps/reviews/..."
}
]
},
"json_ld": { "@context": "https://schema.org", "@type": "LocalBusiness", "...": "..." },
"credibility": { "verified": true, "listing_tier": "free", "data_sources": ["..."] },
"citation": { "display_name": "Colorado Concrete Coatings — Denver, CO", "...": "..." }
}
]
}
Every response is wrapped in a consistent envelope:
{
meta: {
schema_version: string // Currently "2.0"
total_results: number // Count of items in results array
niche: string | null // Niche ID if applicable
data_freshness: {
directory_refresh_cadence: string // "weekly"
google_data_refresh_cadence: string // "quarterly"
scraped_at?: string // ISO datetime — most recent directory write
google_refreshed_at?: string // ISO datetime — most recent Google Places refresh
}
data_note: string // Context about the data returned
}
results: Array<T> // Tool-specific result objects
}
Cadence framing. The directory layer (provider names, services, websites, descriptions) refreshes weekly via a scraping pipeline. The Google Places layer (rating, reviews, opening hours, business status, AI summary) refreshes quarterly via Google Places Text Search Enterprise. Two cadences, both deliberate. AI agents that need real-time data should call back periodically rather than caching responses indefinitely.
Errors use the same envelope with an error object:
{
meta: { schema_version: string }
error: {
code: string // "NOT_FOUND" | "INTERNAL_ERROR" | "UNAUTHORIZED" | "FORBIDDEN"
message: string // Human-readable error description
}
}
| Field | Type | Nullable | Description |
|---|---|---|---|
name | string | no | Business name (always present) |
description | string | no | Business description (always present) |
city | string | no | City name (always present) |
state | string | no | Two-letter state abbreviation (always present) |
rating | number | no | Google rating 1.0–5.0 (always present) |
review_count | number | yes | Number of Google reviews |
business_status | string | yes | OPERATIONAL / CLOSED_TEMPORARILY (CLOSED_PERMANENTLY filtered automatically) |
google_maps_url | string | yes | Direct link to Google Maps listing |
services | array | no | [{ type: string, label: string }] (always present, non-empty) |
pricing_summary | string | yes | Pricing info (public access) |
coverage_area | string | yes | Geographic coverage description |
years_in_business | number | yes | Years operating |
listing_url | string | no | Full profile URL with contact details |
get_provider adds:
| Field | Type | Description |
|---|---|---|
service_areas | array | [{ city, state, radius_miles }] |
service_details | array | [{ type, label, pricing_model, price_range, turnaround }] |
google_data | object | Structured Google Places data — see below |
json_ld | object | Schema.org LocalBusiness JSON-LD with AggregateRating, OpeningHoursSpecification, GeoCoordinates, telephone, sameAs |
credibility | object | { verified, listing_tier, verification_date, data_sources } |
citation | object | Pre-formatted strings: { display_name, in_text, attribution } |
google_data block (present on get_provider when Google data is available):
| Field | Type | Description |
|---|---|---|
business_status | string | OPERATIONAL / CLOSED_TEMPORARILY |
google_maps_url | string | Direct Google Maps link |
formatted_address | string | Google's canonical address |
opening_hours | array | Schema.org OpeningHoursSpecification[] |
summary | object | { text, source } — source is localpro_ai (LocalPro-generated, no disclosure required) or google (with required disclosure field) |
recent_reviews | array | Up to 5 Google review bodies with author, rating, publish time, source URL |
Fields marked nullable return null when data is unavailable — they are never omitted from the response. Arrays return [] when empty, never null.
All search and list tools work without an API key:
list_niches, list_cities, list_service_types, search_providersget_provider returns basic data (name, description, rating, services, pricing summary, listing URL)Include an X-API-Key header to unlock additional data on get_provider:
X-API-Key: your-api-key
Request an API key at localpro.dev or email will@localpro.dev.
AI agents can self-discover this server via standard well-known endpoints:
GET /.well-known/llms.txt — Plain text description of the server and its toolsGET /.well-known/mcp.json — Structured JSON with tool list, auth info, and operator detailslisting_url). This protects provider data while driving traffic to the directory.CLOSED_PERMANENTLY providers are filtered automatically.| Access | Limit |
|---|---|
| Public (no key) | 30 requests/minute per IP |
| Premium (API key) | 30 requests/minute per key |
Higher limits available for partners — contact will@localpro.dev.
Every provider returned by the API has been verified and meets a minimum completeness threshold:
| Category | Providers | Coverage |
|---|---|---|
| Water Damage Restoration | 1,125+ | 49 states |
| Foundation Repair | 1,025+ | 27 states |
| Crawl Space Repair | 1,025+ | 41 states |
| Mold & Asbestos | 950+ | 21 states |
| Septic Services | 850+ | 36 states |
| Basement Waterproofing | 600+ | 26 states |
| Laundry Services | 575+ | 39 states |
| Floor Coating | 500+ | 42 states |
| Radon | 250+ | 15 states |
Additional fields (pricing, certifications, coverage area, years in business, opening hours, recent reviews, AI summary) are available on most providers but not guaranteed. Fields without data return explicit null — never omitted, never empty strings.
Directory data is refreshed weekly via the scraping pipeline. Google Places data (ratings, reviews, opening hours, business status) is refreshed quarterly via Text Search Enterprise. Two additional categories are being prepared for launch.
LocalPro runs as a Cloudflare Worker with a D1 database binding. To deploy your own instance:
npm install
npx wrangler secret put API_KEY # Set your production API key
npx wrangler deploy
Requires a Cloudflare account with a D1 database named laced-directory.
LocalPro is built and operated by Laced Labs LLC.
MIT
Be the first to review this server!
by Modelcontextprotocol · Developer Tools
Read, search, and manipulate Git repositories programmatically
by Modelcontextprotocol · Developer Tools
Web content fetching and conversion for efficient LLM usage
by Toleno · Developer Tools
Toleno Network MCP Server — Manage your Toleno mining account with Claude AI using natural language.