geup-ddong-seoul-mcp
Click on "Install Server".
Wait a few minutes for the server to deploy. Once ready, it will show a "Started" state.
In the chat, type
@followed by the MCP server name and your instructions, e.g., "@geup-ddong-seoul-mcpfind nearest open toilets near Hongdae entrance"
That's it! The server will respond to your query, and you can continue using it as needed.
Here is a step-by-step guide with screenshots.
서울시 공중화장실 데이터(스마트서울맵 테마 theme_id=100106)를 조회하는 MCP 서버입니다.
급할 때 내 위치에서 가까운, 지금 열려 있는 화장실을 찾는 것이 핵심 기능입니다.
전송 방식: Streamable HTTP (
http://{HOST}:{PORT}/mcp)데이터 출처: 스마트서울맵 갤러리 (공중화장실, 약 4,500개소)
다운로드는 로그인/인증 불필요 (공개 엔드포인트)
동작 개요
POST https://map.seoul.go.kr/smgis/webs/contents/contents.do
?mode=downloadGeomFormatXLSX&down_type=xlsx&lan_type=KOR&conts_type=A&theme_id=100106
-> contents.zip -> contents.xlsx (데이터) -> 파싱하여 메모리 적재Related MCP server: dartpoint-mcp
원본 데이터 구조 (contents.xlsx)
contents.xlsx는 스마트서울맵 테마 업로드 공통 양식으로 77개 컬럼이며, 공중화장실은
약 4,538행입니다. 77컬럼 중 실제로 값이 들어있는 컬럼은 45개이고 나머지는 빈
스타일/예비 컬럼입니다. 헤더 1행은 콘텐츠 ID\n(필수 입력)처럼 안내문구가 붙어 있어
첫 줄만 컬럼명으로 사용합니다. (채움률은 실제 데이터 기준 근삿값)
고정 컬럼
원본 컬럼(첫 줄) | 내부 필드 | 채움률 | 예시 |
콘텐츠 ID |
| 100% |
|
사용유무 |
| 100% |
|
콘텐츠명 |
| 100% | 강동구민회관 |
서브카테고리 명 |
| 100% | 정시 / 상시 |
구명 |
| 100% | 강동구 |
새주소[도로명 주소] |
| 92% | 서울특별시 강동구 상암로 168 |
지번주소 |
| 99% | 서울특별시 강동구 천호동 41 |
키워드 |
| 99% | #공중화장실 |
좌표[X] |
| 99% | 127.141222548 |
좌표[Y] |
| 99% | 37.545731457 |
전화번호 |
| 99% | 02-2045-7617 |
웹URL |
| <1% | (대부분 비어있음) |
상세 속성 (제목/내용 쌍 구조)
상세 정보는 상세 제목N / 상세 내용N 컬럼 쌍으로 저장됩니다. 예를 들어 상세 제목2칸에
개방시간, 상세 내용2칸에 정시(08:00~21:00)이 들어갑니다. 서버는 컬럼 위치(N)가 아니라
제목 텍스트로 매핑하므로 순서가 바뀌어도 안전합니다. 다중값은 |로 구분되며(남자|여자|)
끝의 |는 정리합니다.
상세 제목 | 내부 필드 | 타입 | 채움률 | 예시 |
개방주체 |
| 문자열 | ~100% | 공공개방 |
개방시간 |
| 문자열 | ~100% | 정시(08:00~21:00) |
휴관일 |
| 리스트 | 11% | ["토요일","일요일"] |
화장실구분 |
| 리스트 | 99% | ["남자","여자"] |
장애인화장실구분 |
| 리스트 | 52% | ["남자","여자"] |
편의시설 |
| 리스트 | 28% | ["기저귀교환대(남)"] |
안전시설 |
| 리스트 | 64% | ["비상벨(여)"] |
건물용도 |
| 문자열 | ~100% | 공공시설 |
관리기관 |
| 문자열 | ~100% | 강동구도시관리공단 |
원문 상세 전체(제목→내용)는 get_toilet 결과의 details에 그대로 담깁니다.
coord.xlsx
zip에 함께 들어오지만 공중화장실은 점(공간객체타입=1) 데이터라 좌표가 contents.xlsx에
모두 있어 사용하지 않습니다.
프로젝트 구조
geup-ddong-seoul-mcp/
├─ seoul_toilet/
│ ├─ __init__.py
│ ├─ data.py # 다운로드/압축해제/파싱/질의 + ToiletStore(메모리·캐시)
│ └─ server.py # FastMCP 서버(tool 4종) + /refresh, /health
├─ tests/
│ └─ test_logic.py # 파싱·개방시간·거리·검색·다운로드(mock)·캐시 검증
├─ data_cache/ # 실행 시 생성: contents.zip, contents.xlsx 캐시
├─ requirements.txt
├─ pyproject.toml
└─ README.md설치
pip install -r requirements.txt
# 또는 패키지로 설치
pip install -e .요구사항: Python 3.10+, mcp, requests, openpyxl
실행
python -m seoul_toilet.server
# -> http://127.0.0.1:8000/mcp 에서 MCP 제공환경변수로 설정합니다.
변수 | 기본값 | 설명 |
|
| 바인드 호스트 |
|
| 포트 |
|
| xlsx/zip 캐시 디렉터리 |
|
| 시작 시 캐시가 없으면 1회 다운로드( |
| (없음) | 리버스 프록시 뒤 공개 도메인. 지정 시 MCP |
| (자동) | 허용 Host 헤더 목록(쉼표 구분). 비우면 localhost 계열 + |
| (자동) | 허용 Origin 목록(쉼표 구분, 브라우저 클라이언트용) |
|
| MCP DNS 리바인딩 보호( |
| (없음) | 설정 시 |
도메인(리버스 프록시) 뒤에서
/mcp를 서비스할 때SEOUL_TOILET_PUBLIC_HOST를 지정하지 않으면 MCP의 DNS 리바인딩 보호가 외부 Host 헤더를421 Invalid Host header로 막는다./health·/refresh는 영향받지 않는다.
시작 시 캐시가 있으면 캐시를 읽고(네트워크 없음), 없으면 1회 다운로드합니다.
MCP 클라이언트 등록
Streamable HTTP를 지원하는 클라이언트에서 아래처럼 URL로 등록합니다.
{
"mcpServers": {
"geup-ddong-seoul": {
"type": "streamable-http",
"url": "http://127.0.0.1:8000/mcp"
}
}
}노출 Tool (LLM이 호출)
Tool | 용도 | 주요 인자 |
| 내 위치 기준 가까운 화장실(거리순) |
|
| 구/이름/주소/키워드 검색 |
|
| content_id 상세(원문 details 포함) |
|
| 건수·갱신시각·자치구 목록 등 메타 | (없음) |
"지금 개방 중"은 별도 tool 대신
open_now필터 + 결과의open_now/open_hours필드로 통합했습니다. LLM이 거리·개방·시설 조건을 조합해 판단하기 쉽도록, 각 결과에 개방시간(원문)·개방여부· 화장실구분·장애인화장실·편의시설·안전시설·관리기관·전화번호를 함께 담아 반환합니다.
결과 필드 예시
{
"content_id": "rest2025_0448",
"name": "강동구민회관",
"district": "강동구",
"road_address": "서울특별시 강동구 상암로 168",
"latitude": 37.545731457, "longitude": 127.141222548,
"phone": "02-2045-7617",
"open_hours": "정시(08:00~21:00)", "open_now": true, "is_24h": false,
"toilet_types": ["남자", "여자"],
"disabled_toilet": ["남자", "여자"],
"amenities": ["기저귀교환대(남)", "기저귀교환대(여)"],
"safety": ["비상벨(여)"],
"manager": "강동구도시관리공단",
"distance_m": 12.3
}반환 필드 레퍼런스
원본 필드(위 데이터 구조 표) 외에 서버가 계산해 넣는 파생 필드입니다.
필드 | 타입 | 설명 |
| bool / null | 현재(KST) 개방 여부. 형식 해석 불가 시 |
| bool | 상시/24시간 개방 여부 |
| number |
|
| string / null | 요일·공휴일 차이 등 자동해석 한계 안내 |
| object |
|
빈 값은 단일 필드는 null, 리스트 필드는 []로 반환합니다.
데이터 갱신 (사용자가 직접 호출)
새로고침은 MCP tool로 노출하지 않고 운영용 HTTP 엔드포인트로만 제공합니다.
curl -X POST http://127.0.0.1:8000/refresh # 원격에서 다시 받아 캐시 갱신
curl http://127.0.0.1:8000/health # 상태 확인월 1회 자동화 예시(cron):
0 6 1 * * curl -fsS -X POST http://127.0.0.1:8000/refresh공개 배포 등에서 /refresh 남용을 막으려면 SEOUL_TOILET_REFRESH_TOKEN을 설정하고 호출 시
토큰을 함께 보냅니다(미설정이면 공개 — cron이 없는 서버에서 그대로 호출 가능).
export SEOUL_TOILET_REFRESH_TOKEN=$(openssl rand -hex 24)
curl -fsS -X POST -H "X-Refresh-Token: $SEOUL_TOILET_REFRESH_TOKEN" http://127.0.0.1:8000/refresh
# 또는: -H "Authorization: Bearer $TOKEN" / "...?token=$TOKEN"개방시간 해석 한계
개방시간 원문은 정시(08:00~21:00), 상시(00:00~24:00), 평일(09:00~18:00) 주말휴무 등
형식이 다양합니다. 서버는 시간 구간을 추출해 open_now(KST 기준)를 계산하되, 요일/공휴일별
차이는 완전히 반영하지 못할 수 있어 그런 경우 open_hours_note로 표시하고 원문(open_hours)을
함께 제공합니다. 최종 판단 시 원문을 함께 참고하세요.
왜 FastMCP인가
읽기 전용 조회 서버(다운로드→캐시→질의)라 저수준 SDK의 세밀한 제어가 불필요합니다.
공식 mcp SDK의 FastMCP는 @mcp.tool 데코레이터로 타입힌트 기반 입력 스키마를 자동
생성하고, @mcp.custom_route로 운영용 엔드포인트(/refresh)를 같은 앱에 손쉽게 붙일 수
있어 본 용도에 가장 적합합니다.
테스트
pip install pytest
python -m pytest tests/ -q실제 스키마(77컬럼, 상세 제목/내용 쌍 구조)를 본뜬 합성 데이터로 파싱·개방시간·거리·검색·
다운로드(mock)·캐시까지 검증합니다.
라이선스/출처 표기
데이터: 서울특별시 스마트서울맵 (이용허락: 저작자표시 BY). 출처를 표기해 사용하세요.
This server cannot be installed
Maintenance
Resources
Unclaimed servers have limited discoverability.
Looking for Admin?
If you are the server author, to access and configure the admin panel.
Latest Blog Posts
MCP directory API
We provide all the information about MCP servers via our MCP API.
curl -X GET 'https://glama.ai/api/mcp/v1/servers/Seo-yul/geup-ddong-seoul-mcp'
If you have feedback or need assistance with the MCP directory API, please join our Discord server