For AI agents
Connect your AI
Every listing on SPACEXPLORATION is searchable from Claude, ChatGPT, Cursor, or any client that speaks the Model Context Protocol. Searching needs no account and no API key — paste one URL and ask.
The server
https://spacexploration.com/mcp Streamable-HTTP MCP endpoint. Read tools work anonymously; account features use OAuth that your client handles for you.
Set it up
Claude (claude.ai & Desktop)
Settings → Connectors → Add custom connector, then paste the server URL.
https://spacexploration.com/mcp Claude Code
One command in your terminal:
claude mcp add --transport http spacexploration https://spacexploration.com/mcp Cursor
Add the server to .cursor/mcp.json (project) or ~/.cursor/mcp.json (global):
{
"mcpServers": {
"spacexploration": {
"url": "https://spacexploration.com/mcp"
}
}
} ChatGPT
Settings → Connectors → Advanced → Developer mode, then add the server URL as a custom connector.
https://spacexploration.com/mcp Every tool, live
This catalog is generated from the running server — 20 tools, each with the exact argument schema your agent receives. Sign-in is standard OAuth where marked; your client walks you through it. Listing changes always require a verified email and your ownership of the listing.
No account needed
› get_registry read-only
Returns the full attribute registry — every searchable and filterable field, its type, allowed values, and display label. Read this first to understand the schema before calling search_listings with attribute filters.
{
"type": "object",
"properties": {}
} › search_listings read-only
Search commercial real estate listings. Returns paginated hits with facet counts. For AI-driven search, call interpret_search first to convert a natural-language query into structured filters, then pass those filters — and its bounds, when present — here.
{
"properties": {
"query": {
"description": "Free-text search query (street address, city, or keyword). Omit or pass null for no text filter.",
"type": [
"string",
"null"
]
},
"filters": {
"description": "Structured filter map keyed by Typesense field name. Ranges as {min?, max?}, multi-select as string[], booleans as true/false. Call get_registry to see valid field names and allowed values.",
"type": [
"object",
"null"
]
},
"sort": {
"description": "Sort key. Valid values from get_registry sort_options (e.g. \"listed_at:desc\", \"price:asc\").",
"type": [
"string",
"null"
]
},
"page": {
"description": "Page number (1-based). Defaults to 1.",
"type": [
"integer",
"null"
]
},
"bounds": {
"description": "Geographic bounding box (degrees) limiting hits to an area. Forward the bounds returned by interpret_search verbatim to honour \"near <place>\" intent; omit for a nationwide search.",
"properties": {
"sw_lat": {
"minimum": -90,
"maximum": 90,
"type": "number"
},
"sw_lng": {
"minimum": -180,
"maximum": 180,
"type": "number"
},
"ne_lat": {
"minimum": -90,
"maximum": 90,
"type": "number"
},
"ne_lng": {
"minimum": -180,
"maximum": 180,
"type": "number"
}
},
"type": [
"object",
"null"
]
}
},
"type": "object"
} › get_listing read-only
Retrieve the full details of a single listing by its hashid. Returns the complete ListingResource projection including parcel data, all attribute details, and attachments. Broker contact is included only when the requesting user is authorised.
{
"properties": {
"hashid": {
"description": "The short hashid of the listing (e.g. \"a1b2c3d4\"). Found in listing URLs and search hits.",
"type": "string"
}
},
"type": "object",
"required": [
"hashid"
]
} › interpret_search read-only
Translate a natural-language property-search sentence into a structured filter payload compatible with search_listings. Use this as a transparent intermediate step: pass the user's raw query here, then forward the returned filters — and the returned bounds, when present (they carry the "near <place>" intent) — to search_listings.
{
"properties": {
"prompt": {
"description": "A natural-language property search sentence from the end user. Example: \"industrial with rail access in Denver under $5m\".",
"type": "string"
}
},
"type": "object",
"required": [
"prompt"
]
} › send_feedback
File feedback or a support ticket with the SPACEXPLORATION team: a bug report, feature request, question, or anything you would improve about the app or the MCP experience. Works anonymously — include a contact email if you want a reply.
{
"properties": {
"message": {
"description": "The feedback itself — what happened, or what you would improve about the app or the MCP experience. Plain text, max 5000 characters.",
"type": "string"
},
"category": {
"description": "What kind of feedback this is. Defaults to \"other\".",
"enum": [
"bug",
"feature_request",
"question",
"other"
],
"type": "string"
},
"email": {
"description": "Optional contact email if you or your human want a reply. Authenticated callers default to their account email.",
"type": [
"string",
"null"
]
}
},
"type": "object",
"required": [
"message"
]
} With your account
› save_search
Save the current search as a named alert. The user will receive email digests when new matching listings appear. Requires authentication.
{
"properties": {
"name": {
"description": "A human-readable name for the saved search. If omitted, a name is derived from the filters.",
"type": [
"string",
"null"
]
},
"query": {
"description": "The free-text query string, if any.",
"type": [
"string",
"null"
]
},
"filters": {
"description": "The structured filter map to persist (same shape as search_listings filters).",
"type": [
"object",
"null"
]
}
},
"type": "object"
} › list_my_saved_searches read-only
List the authenticated user's saved searches, including their filters and alert schedule. Requires authentication.
{
"type": "object",
"properties": {}
} › delete_saved_search destructive
Delete a saved search by its ID. Only the owner may delete their own searches. Requires authentication.
{
"properties": {
"id": {
"description": "The numeric ID of the saved search to delete (from list_my_saved_searches).",
"type": "integer"
}
},
"type": "object",
"required": [
"id"
]
} › list_my_listings read-only
List all listings owned or brokered by the authenticated user. Optionally filter by status. Requires authentication.
{
"properties": {
"status": {
"description": "Filter by listing status. One of: draft, pending_payment, active, suspended, sold, leased, withdrawn. Omit to return all statuses.",
"type": [
"string",
"null"
]
}
},
"type": "object"
} › geocode_address read-only
Geocode a free-form US address into coordinates and structured components (street_address, city, county, state, postal_code) plus a formatted address and place_id — exactly the identity fields create_listing requires. Use this first, then pass the result straight into create_listing instead of supplying your own coordinates.
{
"properties": {
"address": {
"description": "Free-form address to geocode, e.g. \"3763 Imperial St, Frederick, CO 80516\".",
"type": "string"
}
},
"type": "object",
"required": [
"address"
]
} › create_listing
Create a new draft listing. Returns the full created listing exactly as stored, including its hashid and status.
{
"properties": {
"street_address": {
"description": "Street address of the property.",
"type": "string"
},
"city": {
"description": "City.",
"type": "string"
},
"county": {
"description": "County.",
"type": "string"
},
"state": {
"description": "State (2-letter abbreviation).",
"type": "string"
},
"postal_code": {
"description": "ZIP/postal code.",
"type": "string"
},
"latitude": {
"description": "Latitude coordinate.",
"type": "number"
},
"longitude": {
"description": "Longitude coordinate.",
"type": "number"
},
"property_type": {
"description": "Property type. Allowed values are listed under the property_type attribute returned by get_registry (e.g. industrial, office, retail, land).",
"type": "string"
},
"listing_type": {
"description": "Listing type: sale or lease.",
"enum": [
"sale",
"lease"
],
"type": "string"
},
"unit": {
"description": "Unit or suite number, if applicable.",
"type": [
"string",
"null"
]
},
"formatted_address": {
"description": "Full formatted address string from geocoder.",
"type": [
"string",
"null"
]
},
"place_id": {
"description": "Place ID from geocoder.",
"type": [
"string",
"null"
]
}
},
"type": "object",
"required": [
"street_address",
"city",
"county",
"state",
"postal_code",
"latitude",
"longitude",
"property_type",
"listing_type"
]
} › update_listing
Update a listing's parcel, pricing, or marketing data. Pass exactly one group (parcel, pricing, or marketing). Returns the full updated listing exactly as stored.
{
"properties": {
"hashid": {
"description": "Hashid of the listing to update.",
"type": "string"
},
"parcel": {
"description": "Parcel/address fields to update: street_address, city, county, state, postal_code, latitude, longitude, and optionally unit, lot_acres, year_built, property_type.",
"type": [
"object",
"null"
]
},
"pricing": {
"description": "Pricing fields to update: listing_type (sale|lease); for a sale set sale_price (dollars), for a lease set lease_rate (per the lease_rate_period basis, default per_sqft_year) and optionally lease_rate_period (per_sqft_year|per_sqft_month|per_month|per_year). Omit the price and set price_on_request=true for \"price upon request\". Optionally expires_at (ISO date).",
"type": [
"object",
"null"
]
},
"marketing": {
"description": "Marketing copy to update: headline, summary, title_override, highlights (string[]).",
"type": [
"object",
"null"
]
}
},
"type": "object",
"required": [
"hashid"
]
} › set_listing_attributes
Set broker-authored attributes on a listing you own. Pass `attributes` as a map of registry field name to value (the field names get_registry returns, e.g. {"attr_parking_spaces": 40, "attr_office_class": "Class A"}). Only fields get_registry marks "writable":true can be set; publish-time computed fields and hot columns (use update_listing for those) are rejected with guidance. Enum fields take the allowed value (or an array for multiselect); send null to clear a field. Returns the full updated listing.
{
"properties": {
"hashid": {
"description": "Hashid of the listing to update.",
"type": "string"
},
"attributes": {
"description": "Map of registry field name to value, e.g. {\"attr_parking_spaces\": 40, \"attr_office_class\": \"Class A\", \"attr_industrial_rail_access\": true}. Only fields get_registry marks \"writable\":true are accepted. Multiselect enum fields take an array of allowed values. Send null to clear a field.",
"type": "object"
}
},
"type": "object",
"required": [
"hashid",
"attributes"
]
} › attach_listing_photo
Attach a photo to a listing you own directly from its public URL — one call, no separate sign/upload/confirm. The server fetches the image and ingests it with auto-generated thumbnail/hero/full variants. Only https image URLs whose host is publicly routable are accepted. The photo is content-moderated (must be real-estate related and safe) before it can appear publicly — the returned snapshot includes the moderation_status (approved / rejected / escalated) and moderation_reason. A rejected or escalated photo will not be publicly visible and will block publishing until removed or replaced.
{
"properties": {
"hashid": {
"description": "Hashid of the listing to attach the photo to.",
"type": "string"
},
"source_url": {
"description": "Public https URL of the image to fetch and attach. Must point directly at an image (jpeg, png, webp, gif, avif, heic/heif) on a publicly-routable host.",
"type": "string"
}
},
"type": "object",
"required": [
"hashid",
"source_url"
]
} › sign_attachment_upload
Begin an out-of-band file upload. Returns a presigned R2 PUT URL and the headers to use; the file bytes are uploaded directly to storage, never through this channel. After PUTting the file, call confirm_attachment_upload with the returned attachment_id. Supports listings (photos, documents), users (avatar), and pages (cover, photos).
{
"properties": {
"attachable_type": {
"description": "Morph alias of the resource to attach to: \"listing\", \"user\", or \"page\".",
"type": "string"
},
"attachable_id": {
"description": "Identifier of the attachable resource. For \"listing\" pass the listing hashid (e.g. \"d5yqqgoa\") — the same id every read tool returns. For \"user\" pass \"me\" to attach to yourself. For \"page\" pass its numeric id.",
"type": "string"
},
"filename": {
"description": "Original filename including extension, e.g. \"front-elevation.jpg\". Used to derive the storage key and extension.",
"type": "string"
},
"content_type": {
"description": "MIME type of the file, e.g. \"image/jpeg\" or \"application/pdf\". Must match the collection whitelist.",
"type": "string"
},
"size": {
"description": "Exact byte size of the file. Verified against the uploaded object at confirm time, so it must be accurate.",
"type": "integer"
},
"collection": {
"description": "Optional explicit collection: \"photos\", \"documents\", \"avatar\", \"logo\", or \"cover\". Omit to auto-route by MIME (images→photos, docs→documents). Singleton slots (avatar/cover/logo) must be set explicitly.",
"type": [
"string",
"null"
]
}
},
"type": "object",
"required": [
"attachable_type",
"attachable_id",
"filename",
"content_type",
"size"
]
} › confirm_attachment_upload
Finalize an out-of-band upload after the file bytes have been PUT to the presigned URL. Verifies the stored object matches what was signed (size and content-type) and marks the attachment ready. Returns the attachment snapshot with its public variant URLs. Idempotent: confirming an already-confirmed upload returns the same snapshot.
{
"properties": {
"attachment_id": {
"description": "The attachment_id (ULID) returned by sign_attachment_upload. Confirm only after the file bytes have been successfully PUT to the presigned URL.",
"type": "string"
}
},
"type": "object",
"required": [
"attachment_id"
]
} › delete_attachment destructive
Delete an attachment you own by its ULID. Soft-deletes the row so it no longer appears on the listing. Use this to clean up an abandoned upload (one you signed but never finished) or a photo you no longer want. Idempotent: deleting an already-deleted attachment succeeds.
{
"properties": {
"attachment_id": {
"description": "The attachment_id (ULID) of the attachment to delete, as returned by sign_attachment_upload / confirm_attachment_upload or listed in get_listing.",
"type": "string"
}
},
"type": "object",
"required": [
"attachment_id"
]
} › publish_listing
Kick off the publish pipeline for a draft listing. Enriches with AI, gathers market data, then transitions the listing to Active. Returns the pipeline ID to track progress. Fails with the unmet readiness checklist if the listing is not ready (including any photo that has not cleared content moderation).
{
"properties": {
"hashid": {
"description": "Hashid of the listing to publish.",
"type": "string"
}
},
"type": "object",
"required": [
"hashid"
]
} › get_publish_status read-only
Get the status of your listing publishes. Pass a listing hashid for the detailed status of that listing's latest publish run — overall state (pending/running/completed/failed), current phase, per-step progress, and any failure reason. Omit the hashid to list your recent publish runs across all your listings. Use this to track a publish_listing call to completion.
{
"properties": {
"hashid": {
"description": "Optional. Hashid of a listing to get its latest publish run in detail. Omit to list your recent publish runs across all your listings.",
"type": [
"string",
"null"
]
}
},
"type": "object"
} › withdraw_listing destructive
Withdraw a listing, removing it from public search. This is a terminal status — the listing cannot be re-activated; create a new listing instead.
{
"properties": {
"hashid": {
"description": "Hashid of the listing to withdraw.",
"type": "string"
}
},
"type": "object",
"required": [
"hashid"
]
} Recipes
The common journeys, end to end — from a pasted URL to a published listing.
› Connect Claude to SPACEXPLORATION
Wire Claude (or any MCP client) to the live listings server. Searching works the moment you connect — no account, no API key.
Claude on the web or Desktop
- Open Settings → Connectors → Add custom connector.
- Paste the server URL shown at the top of this page (it ends in
/mcp). - Save. The SPACEXPLORATION tools appear in Claude's tool picker immediately.
Claude Code
One command in your terminal:
claude mcp add --transport http spacexploration https://spacexploration.com/mcp
Verify it works
Ask your client:
Find industrial listings over 50,000 sq ft in Texas.
A connected agent will call get_registry to learn the filter vocabulary, then search_listings — anonymously. If the tools never appear, re-check the URL: the endpoint speaks streamable-HTTP MCP, not SSE.
When sign-in happens
Saved searches, geocoding, and listing authoring need an account. You don't provision anything: the server advertises its OAuth metadata, your client registers itself (Dynamic Client Registration) and walks you through a standard browser sign-in the first time an account-bound tool runs. Headless agents that can't open a browser can follow the claim ceremony in auth.md instead.
› Build a saved-search agent from scratch
Turn a plain-English brief into a persistent, alerting search in three tool calls.
1. Interpret the brief
Hand the user's words to interpret_search:
{ "query": "owner-user buildings near Denver under $4M with a dock door" }
It returns a structured filters map (keyed by registry field names), plus bounds when the query names a place. Forward both to the next call — dropping bounds loses the "near Denver" intent.
2. Run it
Call search_listings with the interpreted filters and bounds and show the user what came back. Adjust filters by hand if the user refines — get_registry lists every legal field, its type, and its allowed values.
3. Persist it
Once the user is happy, call save_search (this is the first account-bound call, so OAuth kicks in here):
{
"name": "Denver owner-user under $4M",
"query": "dock door",
"filters": { "listing_type": ["sale"], "price": { "max": 4000000 } }
}
The platform emails the user a digest when new matching listings publish — no polling loop to build.
Manage the portfolio of searches
list_my_saved_searches returns everything saved; delete_saved_search retires one by id. A good agent reconciles: before saving, list what exists and update rather than duplicate.
› Pull all my listings into a spreadsheet
Export your whole inventory — drafts and published — into CSV with two tools.
1. Enumerate
list_my_listings (account-bound — your client handles the OAuth sign-in) returns every listing you broker or own, each with its hashid, title, status, and price. The hashid is the only key you need; numeric ids never appear on the agent surface.
2. Hydrate
For each hashid, get_listing returns the full projection: address, pricing, status, and every populated registry attribute with its label and unit.
{ "hashid": "a1b2c3d4" }
3. Shape the rows
Ask your agent to flatten what it fetched:
Make a CSV with one row per listing: title, status, city, state, price, building sq ft, lot acres, year built. Use empty cells where a listing has no value.
Attribute coverage varies by asset type — a land listing has no building square footage. Emitting blanks (not zeros) keeps the spreadsheet honest.
Keeping it fresh
Re-running the two calls is idempotent and cheap. For a recurring report, schedule the same conversation — the hashids are stable across the listing's life.
› Author a listing from a flyer
Two paths from a PDF flyer or offering memorandum to a published listing. Both require a verified email on your account.
The short path: upload it in the editor
Signed in on the web, choose Import listing from document when creating a new listing and upload the flyer. The extraction service reads it synchronously and prefills a draft — headline, pricing, property facts — which you review and publish from the editor. This is the right path when a human is at the keyboard.
The agent path: full MCP authoring
When an agent holds the flyer (or its text), it can author end-to-end:
geocode_address— turn the flyer's address line into the coordinates and county thatcreate_listingneeds.create_listing— returns the new draft, including itshashid; every later call keys on it.update_listing— pricing, marketing copy, parcel facts.set_listing_attributes— the rich registry fields, as a map of field name to value. Only fieldsget_registrymarkswritable: trueare accepted; the rest are computed at publish.attach_listing_photo— one call per image URL; the server fetches and ingests it. (For local files, use thesign_attachment_upload→ upload →confirm_attachment_uploadpair instead.)publish_listing— kicks off the asynchronous publish pipeline. Pollget_publish_statuswith the listing hashid until its state iscompleted.
Honesty guardrails
Extract only what the document states. Leave unknown fields empty rather than guessing — publish-time enrichment fills the computed facts (flood zone, demographics, market data) from authoritative sources, and a wrong guess in a broker-asserted field is worse than a blank.
› Add an attribute and watch every AI surface inherit it
How the registry-as-contract works, from the inside. The attribute registry — one database table — is the single schema every surface derives from. Nothing on this page is hand-maintained.
One row, every surface
When an administrator adds a row to the registry (say industrial.dock_doors, a number range applying to industrial assets), each surface picks it up from the same snapshot, with no deploy:
get_registryand theregistry://attributesresource expose it to agents on the next call, typed, with its range hint.search_listingsacceptsattr_industrial_dock_doorsas a filter key immediately.interpret_searchfolds it into the natural-language vocabulary, so "at least 4 dock doors" starts resolving to a structured filter.- The search UI grows a sidebar facet where the row's sidebar position says.
set_listing_attributesaccepts it from authoring agents once the row is marked agent-writable.- Document extraction adds it to what the flyer reader looks for.
- This page — the attribute table above re-renders from the live registry on every request.
The contract
The field name (attr_industrial_dock_doors) is the one identity shared by every query surface. What an agent needs to construct a query — type, control, allowed values, range bounds, writability — is exactly what the registry projection carries; storage internals deliberately never leave the server.
Why this matters to you
If you build against get_registry instead of a hardcoded field list, your agent inherits every future attribute the day it ships. The recipe is no recipe: read the registry, build the query, stay current forever.
The attribute registry
Every filterable field, straight from the live registry —
the same schema get_registry returns. Filter search_listings by the field name; fields marked writable accept values via set_listing_attributes.
All assets
| Field | Label | Type | Values | Writable |
|---|---|---|---|---|
property_type | Property Type | enum (multi) | Industrial, Retail, Office, Multifamily, Land | — |
price | Price | number range | 0 – 50000000 $ | — |
listing_type | Listing Type | enum | For sale, For lease | — |
lot_acres | Lot Size | number range | 0 – 200 acres | — |
year_built | Year Built | number range | 1900 – 2026 | — |
lease_type | Lease Type | enum (multi) | Triple Net (NNN), Full Service Gross, Modified Gross, Gross | — |
space_type | Space Use | enum (multi) | Office, Retail, Industrial, Flex | — |
attr_construction_type | Construction Type | enum | Steel Frame, Masonry, Tilt-up Concrete, Wood Frame, Pre-engineered Metal | Yes |
attr_construction_condition | Condition | enum | New, Excellent, Good, Fair, Needs Renovation | Yes |
attr_construction_stories | Stories | number range | 1 – 40 | Yes |
attr_parking_spaces | Parking Spaces | number range | 0 – 2000 | Yes |
attr_parking_ratio | Parking Ratio | number range | 0 – 10 per 1k sqft | Yes |
attr_utilities_power_phase | Power Phase | enum | Single-phase, Three-phase | Yes |
attr_utilities_power_amps | Service Amperage | number range | 100 – 4000 A | Yes |
attr_utilities_power_voltage | Voltage | enum | 120/240V, 208V, 277/480V, 480V | Yes |
attr_utilities_power_generator | Backup Generator | boolean | — | Yes |
attr_utilities_fiber | Fiber Internet | boolean | — | Yes |
attr_utilities_water | Water | enum | Municipal, Well | Yes |
attr_utilities_sewer | Sewer | enum | Municipal, Septic | Yes |
county | County | enum (multi) | — | — |
city | City | enum (multi) | — | — |
attr_zoning | Zoning | enum (multi) | C-1, C-2, C-3, I-1, I-2, +3 more | Yes |
attr_flood_zone | FEMA Flood Zone | enum | X — minimal hazard, Shaded X — 0.2% annual chance, A — 1% annual chance (no BFE), AE — 1% annual chance (with BFE), AH — shallow ponding, +6 more | — |
attr_opportunity_zone | Opportunity Zone | boolean | — | — |
attr_lease_structure | Lease Structure | enum | NNN, Modified Gross, Full Service Gross | Yes |
attr_lease_term_months | Lease Term | number range | 0 – 240 months | Yes |
attr_lease_available_date | Available Date | date range | — | Yes |
attr_lease_escalations_pct | Annual Escalation | number range | 0 – 10 %/yr | Yes |
attr_lease_ti_allowance | TI Allowance | number range | 0 – 200 $/sqft | Yes |
attr_lease_exclusive_use | Exclusive Use | text | — | Yes |
attr_accessibility_ada_compliant | ADA Compliant | boolean | — | Yes |
attr_environmental_phase_1_completed | Phase I ESA Completed | boolean | — | Yes |
attr_environmental_contamination_known | Known Contamination | boolean | — | Yes |
attr_census_median_income | Median Household Income | number range | 0 – 250000 $ | — |
attr_census_population_density | Population Density | number range | 0 – 50000 people/sq mi | — |
attr_census_median_age | Median Age | number range | 0 – 100 yrs | — |
attr_census_education_bachelor_pct | Bachelor's Degree or Higher | number range | 0 – 100 % | — |
attr_census_median_home_value | Median Home Value | number range | 0 – 5000000 $ | — |
attr_census_population | Population | number range | 0 – 200000 people | — |
attr_census_households | Households | number range | 0 – 100000 | — |
attr_census_median_gross_rent | Median Gross Rent | number range | 0 – 6000 $/mo | — |
attr_census_unemployment_rate | Unemployment Rate | number range | 0 – 100 % | — |
attr_census_poverty_rate | Poverty Rate | number range | 0 – 100 % | — |
attr_walk_score | Walk Score | number range | 0 – 100 | — |
attr_transit_score | Transit Score | number range | 0 – 100 | — |
attr_proximity_nearest_highway_miles | Nearest Highway | number range | 0 – 50 mi | — |
attr_proximity_nearest_airport_miles | Nearest Airport | number range | 0 – 100 mi | — |
attr_amenities_transit_within_quarter_mile | Transit Within ¼ Mile | boolean | — | — |
attr_amenities_schools_within_mile | Schools Within 1 Mile | number range | 0 – 30 | — |
attr_amenities_hospitals_within_5mi | Hospitals Within 5 Miles | number range | 0 – 20 | — |
attr_amenities_restaurants_within_mile | Restaurants Within 1 Mile | number range | 0 – 500 | — |
attr_amenities_grocery_within_mile | Grocery Stores Within 1 Mile | number range | 0 – 50 | — |
attr_amenities_retail_within_mile | Retail Shops Within 1 Mile | number range | 0 – 500 | — |
attr_amenities_parks_within_mile | Parks Within 1 Mile | number range | 0 – 50 | — |
attr_seismic_zone | Seismic Zone | enum | 0, 1, 2A, 2B, 3, +1 more | — |
attr_environmental_epa_facilities_within_mile | EPA-Tracked Facilities Within 1 Mile | number range | 0 – 50 | — |
attr_environmental_tri_facilities_within_mile | Toxic-Release Facilities Within 1 Mile | number range | 0 – 50 | — |
attr_environmental_epa_severe_violators_within_mile | EPA Severe Violators Within 1 Mile | number range | 0 – 50 | — |
attr_seismic_risk_category | Seismic Risk Category | enum | Very Low, Low, Moderate, High, Very High | — |
attr_seismic_pga_g | Peak Ground Acceleration | number range | 0 – 2 g | — |
attr_seismic_design_category | Seismic Design Category | enum | A, B, C, D, E, +1 more | — |
attr_flood_in_sfha | In FEMA Special Flood Hazard Area | boolean | — | — |
attr_flood_base_flood_elevation | Base Flood Elevation | number range | 0 – 500 ft | — |
attr_market_median_sale_price_5mi | Median Sale Price (5mi) | number range | 0 – 50000000 $ | — |
attr_market_median_lease_rate_5mi | Median Lease Rate (5mi) | number range | 0 – 100 $/sqft/yr | — |
attr_lihtc_properties_within_2mi | LIHTC Properties Within 2 Miles | number range | 0 – 200 | — |
attr_lihtc_units_within_2mi | LIHTC Units Within 2 Miles | number range | 0 – 20000 | — |
industrial
| Field | Label | Type | Values | Writable |
|---|---|---|---|---|
attr_building_sqft | Building Size | number range | 0 – 500000 sqft | Yes |
available_sqft | Available Space | number range | 0 – 500000 sqft | — |
attr_year_renovated | Year Renovated | number range | 1900 – 2026 | Yes |
attr_building_class | Building Class | enum | Class A, Class B, Class C | Yes |
attr_utilities_hvac_type | HVAC Type | enum | Central, VAV, Packaged Units, Split System, Forced Air, +1 more | Yes |
attr_industrial_clear_height | Clear Height | number range | 10 – 50 ft | Yes |
attr_industrial_dock_doors | Dock Doors | number range | 0 – 100 | Yes |
attr_industrial_drive_in_doors | Drive-in Doors | number range | 0 – 20 | Yes |
attr_industrial_sprinkler | Sprinkler System | enum | ESFR, Wet, Dry, None | Yes |
attr_industrial_rail_access | Rail Access | boolean | — | Yes |
attr_industrial_crane_capacity | Crane Capacity | number range | 0 – 50 tons | Yes |
attr_industrial_hazmat_permitted | Hazmat Permitted | boolean | — | Yes |
land
| Field | Label | Type | Values | Writable |
|---|---|---|---|---|
attr_land_topography | Topography | enum | Flat, Gently Rolling, Sloped | Yes |
attr_land_utilities_at_site | Utilities at Site | boolean | — | Yes |
attr_land_road_frontage_ft | Road Frontage | number range | 0 – 3000 ft | Yes |
attr_land_cleared | Cleared | boolean | — | Yes |
attr_land_entitlement_status | Entitlement Status | enum | Raw, Zoned, Permits Submitted, Permits In Hand, Shovel Ready | Yes |
multifamily
| Field | Label | Type | Values | Writable |
|---|---|---|---|---|
attr_multifamily_units | Number of Units | number range | 2 – 500 | Yes |
attr_multifamily_occupancy_pct | Occupancy | number range | 0 – 100 % | Yes |
attr_multifamily_avg_rent | Average Rent | number range | 0 – 4000 $/mo | Yes |
attr_multifamily_unit_mix | Unit Mix | text | — | Yes |
attr_multifamily_laundry_type | Laundry | enum | In-unit, Shared on-site, None | Yes |
office
| Field | Label | Type | Values | Writable |
|---|---|---|---|---|
attr_office_class | Office Class | enum | Class A, Class B, Class C | Yes |
attr_office_elevators | Elevators | number range | 0 – 12 | Yes |
attr_office_hvac | HVAC System | enum | Central, Packaged Units, VAV | Yes |
attr_office_move_in_ready | Move-in Ready | boolean | — | Yes |
attr_office_layout | Layout | enum | Open, Private offices, Mixed, Coworking | Yes |
retail
| Field | Label | Type | Values | Writable |
|---|---|---|---|---|
attr_cap_rate | Cap Rate | number range | 3 – 12 % | Yes |
attr_noi | Net Operating Income | number range | 0 – 50000000 $/yr | Yes |
attr_retail_frontage_ft | Frontage | number range | 0 – 500 ft | Yes |
attr_retail_anchored | Anchored Center | boolean | — | Yes |
attr_retail_drive_thru | Drive-thru | boolean | — | Yes |
attr_retail_traffic_count | Daily Traffic Count | number range | 0 – 80000 vehicles/day | Yes |
attr_retail_corner_lot | Corner Lot | boolean | — | Yes |
attr_retail_end_cap | End-cap Unit | boolean | — | Yes |
attr_retail_signage_exclusive_rights | Exclusive Signage Rights | boolean | — | Yes |
Sort keys for search_listings: listed_at:desc, price:asc, price:desc, attr_building_sqft:desc, year_built:desc, attr_cap_rate:desc, attr_noi:desc, views_30d:desc
Building an agent? The machine-readable surface: llms.txt (tool catalog and recommended flows), server.json (MCP registry manifest), and auth.md (self-serve OAuth registration, step by step).