smart-web-mcp 0.9.2 → 0.10.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +9 -0
- package/dist/korea/coupang.d.ts +46 -0
- package/dist/korea/coupang.js +165 -0
- package/dist/korea/coupang.js.map +1 -0
- package/dist/korea/dart.d.ts +53 -0
- package/dist/korea/dart.js +105 -0
- package/dist/korea/dart.js.map +1 -0
- package/dist/korea/drug-safety.d.ts +33 -0
- package/dist/korea/drug-safety.js +60 -0
- package/dist/korea/drug-safety.js.map +1 -0
- package/dist/korea/fine-dust.d.ts +27 -0
- package/dist/korea/fine-dust.js +88 -0
- package/dist/korea/fine-dust.js.map +1 -0
- package/dist/korea/food-safety.d.ts +21 -0
- package/dist/korea/food-safety.js +50 -0
- package/dist/korea/food-safety.js.map +1 -0
- package/dist/korea/geeknews.d.ts +31 -0
- package/dist/korea/geeknews.js +144 -0
- package/dist/korea/geeknews.js.map +1 -0
- package/dist/korea/index.d.ts +16 -0
- package/dist/korea/index.js +17 -0
- package/dist/korea/index.js.map +1 -0
- package/dist/korea/law.d.ts +41 -0
- package/dist/korea/law.js +85 -0
- package/dist/korea/law.js.map +1 -0
- package/dist/korea/naver-news.d.ts +38 -0
- package/dist/korea/naver-news.js +67 -0
- package/dist/korea/naver-news.js.map +1 -0
- package/dist/korea/naver-shopping.d.ts +39 -0
- package/dist/korea/naver-shopping.js +51 -0
- package/dist/korea/naver-shopping.js.map +1 -0
- package/dist/korea/proxy-client.d.ts +20 -0
- package/dist/korea/proxy-client.js +88 -0
- package/dist/korea/proxy-client.js.map +1 -0
- package/dist/korea/public-restroom.d.ts +44 -0
- package/dist/korea/public-restroom.js +184 -0
- package/dist/korea/public-restroom.js.map +1 -0
- package/dist/korea/real-estate.d.ts +56 -0
- package/dist/korea/real-estate.js +95 -0
- package/dist/korea/real-estate.js.map +1 -0
- package/dist/korea/spellcheck.d.ts +32 -0
- package/dist/korea/spellcheck.js +142 -0
- package/dist/korea/spellcheck.js.map +1 -0
- package/dist/korea/stock.d.ts +59 -0
- package/dist/korea/stock.js +84 -0
- package/dist/korea/stock.js.map +1 -0
- package/dist/korea/subway.d.ts +29 -0
- package/dist/korea/subway.js +47 -0
- package/dist/korea/subway.js.map +1 -0
- package/dist/korea/weather.d.ts +35 -0
- package/dist/korea/weather.js +92 -0
- package/dist/korea/weather.js.map +1 -0
- package/dist/korea/zipcode.d.ts +25 -0
- package/dist/korea/zipcode.js +70 -0
- package/dist/korea/zipcode.js.map +1 -0
- package/dist/mcp-server.js +386 -0
- package/dist/mcp-server.js.map +1 -1
- package/package.json +2 -2
|
@@ -0,0 +1,142 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Korean spell check via nara-speller.co.kr (공개 표면, 저빈도·비상업적 용도에 한함)
|
|
3
|
+
* POST https://nara-speller.co.kr/old_speller/results
|
|
4
|
+
*
|
|
5
|
+
* Policy enforced here:
|
|
6
|
+
* - Max 3000 chars per call (chunked internally)
|
|
7
|
+
* - Non-commercial / low-frequency only — this is stated in the upstream service ToS
|
|
8
|
+
* - May return 403 if Cloudflare blocks; caller should retry later
|
|
9
|
+
*/
|
|
10
|
+
import { httpPost } from "./proxy-client.js";
|
|
11
|
+
const SPELLER_URL = "https://nara-speller.co.kr/old_speller/results";
|
|
12
|
+
const MAX_CHUNK = 1500;
|
|
13
|
+
const PAYLOAD_RE = /data\s*=\s*(\[[\s\S]*?\]);\s*pageIdx\s*=/;
|
|
14
|
+
const NO_ISSUES_RE = /맞춤법과\s*문법\s*오류를\s*찾지\s*못했습니다/;
|
|
15
|
+
function splitChunks(text, maxChars) {
|
|
16
|
+
const chunks = [];
|
|
17
|
+
let pos = 0;
|
|
18
|
+
while (pos < text.length) {
|
|
19
|
+
let end = Math.min(pos + maxChars, text.length);
|
|
20
|
+
// Try to break at sentence boundary
|
|
21
|
+
if (end < text.length) {
|
|
22
|
+
const boundary = text.lastIndexOf(".", end);
|
|
23
|
+
if (boundary > pos + maxChars / 2)
|
|
24
|
+
end = boundary + 1;
|
|
25
|
+
}
|
|
26
|
+
chunks.push(text.slice(pos, end));
|
|
27
|
+
pos = end;
|
|
28
|
+
}
|
|
29
|
+
return chunks;
|
|
30
|
+
}
|
|
31
|
+
function stripTags(html) {
|
|
32
|
+
return html.replace(/<br\s*\/?>/gi, "\n").replace(/<[^>]+>/g, "").replace(/&/g, "&").replace(/</g, "<").replace(/>/g, ">").replace(/"/g, '"').replace(/ /g, " ").trim();
|
|
33
|
+
}
|
|
34
|
+
function parsePayload(html) {
|
|
35
|
+
if (NO_ISSUES_RE.test(html))
|
|
36
|
+
return { corrected: "", issues: [] };
|
|
37
|
+
const match = PAYLOAD_RE.exec(html);
|
|
38
|
+
if (!match)
|
|
39
|
+
return { corrected: "", issues: [] };
|
|
40
|
+
let pages;
|
|
41
|
+
try {
|
|
42
|
+
pages = JSON.parse(match[1] ?? "[]");
|
|
43
|
+
}
|
|
44
|
+
catch {
|
|
45
|
+
return { corrected: "", issues: [] };
|
|
46
|
+
}
|
|
47
|
+
const correctedParts = [];
|
|
48
|
+
const issues = [];
|
|
49
|
+
for (const page of pages) {
|
|
50
|
+
correctedParts.push(String(page["str"] ?? ""));
|
|
51
|
+
const errList = (page["errInfo"] ?? []);
|
|
52
|
+
for (const err of errList) {
|
|
53
|
+
const original = String(err["orgStr"] ?? "");
|
|
54
|
+
const rawCands = String(err["candWord"] ?? "");
|
|
55
|
+
const suggestions = rawCands.split("|").map((s) => s.trim()).filter(Boolean);
|
|
56
|
+
const reason = stripTags(String(err["help"] ?? err["errMsg"] ?? ""));
|
|
57
|
+
issues.push({ original, suggestions, reason });
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
return { corrected: correctedParts.join(""), issues };
|
|
61
|
+
}
|
|
62
|
+
async function checkChunk(text, timeoutMs) {
|
|
63
|
+
const body = new URLSearchParams({ text1: text, chkKey: "", btnModeChange: "on" }).toString();
|
|
64
|
+
const result = await httpPost(SPELLER_URL, body, {
|
|
65
|
+
"Content-Type": "application/x-www-form-urlencoded; charset=UTF-8",
|
|
66
|
+
accept: "text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8",
|
|
67
|
+
"accept-language": "ko,en-US;q=0.9,en;q=0.8",
|
|
68
|
+
origin: "https://nara-speller.co.kr",
|
|
69
|
+
referer: "https://nara-speller.co.kr/old_speller/",
|
|
70
|
+
}, timeoutMs);
|
|
71
|
+
if (!result.ok) {
|
|
72
|
+
if (result.code === "http_403") {
|
|
73
|
+
return { ok: false, message: "맞춤법 검사 서비스가 Cloudflare 보호를 활성화했습니다. 잠시 후 다시 시도하세요." };
|
|
74
|
+
}
|
|
75
|
+
return { ok: false, message: `Spell check request failed: ${result.message}` };
|
|
76
|
+
}
|
|
77
|
+
const html = result.data;
|
|
78
|
+
return parsePayload(html);
|
|
79
|
+
}
|
|
80
|
+
function sleep(ms) {
|
|
81
|
+
return new Promise((resolve) => setTimeout(resolve, ms));
|
|
82
|
+
}
|
|
83
|
+
export async function runSpellCheck(options) {
|
|
84
|
+
const { text, timeoutMs = 30000, throttleMs = 1200 } = options;
|
|
85
|
+
if (!text.trim())
|
|
86
|
+
return { ok: false, message: "검사할 텍스트가 없습니다." };
|
|
87
|
+
if (text.length > 10000)
|
|
88
|
+
return { ok: false, message: "한 번에 10,000자를 초과할 수 없습니다. 텍스트를 나눠서 요청하세요." };
|
|
89
|
+
const chunks = splitChunks(text, MAX_CHUNK);
|
|
90
|
+
const allIssues = [];
|
|
91
|
+
const correctedParts = [];
|
|
92
|
+
for (let i = 0; i < chunks.length; i++) {
|
|
93
|
+
if (i > 0)
|
|
94
|
+
await sleep(throttleMs);
|
|
95
|
+
const chunk = chunks[i];
|
|
96
|
+
if (!chunk)
|
|
97
|
+
continue;
|
|
98
|
+
const res = await checkChunk(chunk, timeoutMs);
|
|
99
|
+
if (!("ok" in res) || res.ok !== false) {
|
|
100
|
+
const r = res;
|
|
101
|
+
correctedParts.push(r.corrected || chunk);
|
|
102
|
+
allIssues.push(...r.issues);
|
|
103
|
+
}
|
|
104
|
+
else {
|
|
105
|
+
// On failure, use the original chunk and surface the error
|
|
106
|
+
correctedParts.push(chunk);
|
|
107
|
+
return { ok: false, message: res.message };
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
return {
|
|
111
|
+
source: "spell-check",
|
|
112
|
+
original_text: text,
|
|
113
|
+
corrected_text: correctedParts.join(""),
|
|
114
|
+
issue_count: allIssues.length,
|
|
115
|
+
issues: allIssues,
|
|
116
|
+
chunk_count: chunks.length,
|
|
117
|
+
note: "공개 바른한글 검사기(nara-speller.co.kr) 기준. 비상업적·저빈도 사용 전용.",
|
|
118
|
+
};
|
|
119
|
+
}
|
|
120
|
+
export function renderSpellCheck(result) {
|
|
121
|
+
const lines = [
|
|
122
|
+
`# 한국어 맞춤법 검사 — ${result.issue_count}개 이슈 발견`,
|
|
123
|
+
`(청크 ${result.chunk_count}개 / ${result.original_text.length}자)`,
|
|
124
|
+
"",
|
|
125
|
+
"## 교정 결과",
|
|
126
|
+
result.corrected_text,
|
|
127
|
+
"",
|
|
128
|
+
];
|
|
129
|
+
if (result.issues.length > 0) {
|
|
130
|
+
lines.push("## 주요 변경점");
|
|
131
|
+
for (const issue of result.issues.slice(0, 30)) {
|
|
132
|
+
const sugg = issue.suggestions[0] ?? "(교정안 없음)";
|
|
133
|
+
lines.push(`- **${issue.original}** → ${sugg}`);
|
|
134
|
+
if (issue.reason)
|
|
135
|
+
lines.push(` _${issue.reason}_`);
|
|
136
|
+
}
|
|
137
|
+
lines.push("");
|
|
138
|
+
}
|
|
139
|
+
lines.push(`_${result.note}_`);
|
|
140
|
+
return lines.join("\n");
|
|
141
|
+
}
|
|
142
|
+
//# sourceMappingURL=spellcheck.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"spellcheck.js","sourceRoot":"","sources":["../../src/korea/spellcheck.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,59 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Korean stock info (KRX) via k-skill-proxy
|
|
3
|
+
* Proxy endpoints: /v1/korean-stock/search, /v1/korean-stock/base-info, /v1/korean-stock/trade-info
|
|
4
|
+
*/
|
|
5
|
+
export type StockItem = {
|
|
6
|
+
market: string;
|
|
7
|
+
code: string;
|
|
8
|
+
standard_code?: string;
|
|
9
|
+
name: string;
|
|
10
|
+
short_name?: string;
|
|
11
|
+
english_name?: string;
|
|
12
|
+
listed_at?: string;
|
|
13
|
+
};
|
|
14
|
+
export type StockBaseInfo = StockItem & {
|
|
15
|
+
security_group?: string;
|
|
16
|
+
section_type?: string;
|
|
17
|
+
stock_certificate_type?: string;
|
|
18
|
+
par_value?: number;
|
|
19
|
+
listed_shares?: number;
|
|
20
|
+
};
|
|
21
|
+
export type StockTradeInfo = StockItem & {
|
|
22
|
+
base_date: string;
|
|
23
|
+
close_price?: number;
|
|
24
|
+
change_price?: number;
|
|
25
|
+
fluctuation_rate?: number;
|
|
26
|
+
open_price?: number;
|
|
27
|
+
high_price?: number;
|
|
28
|
+
low_price?: number;
|
|
29
|
+
trading_volume?: number;
|
|
30
|
+
trading_value?: number;
|
|
31
|
+
market_cap?: number;
|
|
32
|
+
};
|
|
33
|
+
export type StockResult = {
|
|
34
|
+
source: "korean-stock";
|
|
35
|
+
action: "search";
|
|
36
|
+
query: string;
|
|
37
|
+
items: StockItem[];
|
|
38
|
+
} | {
|
|
39
|
+
source: "korean-stock";
|
|
40
|
+
action: "base-info";
|
|
41
|
+
item: StockBaseInfo;
|
|
42
|
+
} | {
|
|
43
|
+
source: "korean-stock";
|
|
44
|
+
action: "trade-info";
|
|
45
|
+
item: StockTradeInfo;
|
|
46
|
+
};
|
|
47
|
+
export declare function runStock(options: {
|
|
48
|
+
action: "search" | "base-info" | "trade-info";
|
|
49
|
+
q?: string;
|
|
50
|
+
market?: string;
|
|
51
|
+
code?: string;
|
|
52
|
+
basDate?: string;
|
|
53
|
+
limit?: number;
|
|
54
|
+
timeoutMs?: number;
|
|
55
|
+
}): Promise<StockResult | {
|
|
56
|
+
ok: false;
|
|
57
|
+
message: string;
|
|
58
|
+
}>;
|
|
59
|
+
export declare function renderStock(result: StockResult): string;
|
|
@@ -0,0 +1,84 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Korean stock info (KRX) via k-skill-proxy
|
|
3
|
+
* Proxy endpoints: /v1/korean-stock/search, /v1/korean-stock/base-info, /v1/korean-stock/trade-info
|
|
4
|
+
*/
|
|
5
|
+
import { proxyGet } from "./proxy-client.js";
|
|
6
|
+
function todayKst() {
|
|
7
|
+
const now = new Date();
|
|
8
|
+
const kst = new Date(now.getTime() + 9 * 60 * 60 * 1000);
|
|
9
|
+
return kst.toISOString().slice(0, 10).replace(/-/g, "");
|
|
10
|
+
}
|
|
11
|
+
export async function runStock(options) {
|
|
12
|
+
const { action, q, market, code, basDate, limit = 10, timeoutMs = 15000 } = options;
|
|
13
|
+
const bas_dd = basDate || todayKst();
|
|
14
|
+
if (action === "search") {
|
|
15
|
+
if (!q)
|
|
16
|
+
return { ok: false, message: "검색어(q)가 필요합니다." };
|
|
17
|
+
const result = await proxyGet("/v1/korean-stock/search", { q, bas_dd, limit: String(limit) }, timeoutMs);
|
|
18
|
+
if (!result.ok)
|
|
19
|
+
return { ok: false, message: `Stock search failed: ${result.message}` };
|
|
20
|
+
const items = (result.data["items"] ?? []);
|
|
21
|
+
return { source: "korean-stock", action: "search", query: q, items };
|
|
22
|
+
}
|
|
23
|
+
if (action === "base-info") {
|
|
24
|
+
if (!market || !code)
|
|
25
|
+
return { ok: false, message: "market과 code가 필요합니다." };
|
|
26
|
+
const result = await proxyGet("/v1/korean-stock/base-info", { market, code, bas_dd }, timeoutMs);
|
|
27
|
+
if (!result.ok)
|
|
28
|
+
return { ok: false, message: `Stock base-info failed: ${result.message}` };
|
|
29
|
+
const item = result.data["item"];
|
|
30
|
+
if (!item)
|
|
31
|
+
return { ok: false, message: "종목 정보가 없습니다." };
|
|
32
|
+
return { source: "korean-stock", action: "base-info", item };
|
|
33
|
+
}
|
|
34
|
+
// trade-info
|
|
35
|
+
if (!market || !code)
|
|
36
|
+
return { ok: false, message: "market과 code가 필요합니다." };
|
|
37
|
+
const result = await proxyGet("/v1/korean-stock/trade-info", { market, code, bas_dd }, timeoutMs);
|
|
38
|
+
if (!result.ok)
|
|
39
|
+
return { ok: false, message: `Stock trade-info failed: ${result.message}` };
|
|
40
|
+
const item = result.data["item"];
|
|
41
|
+
if (!item)
|
|
42
|
+
return { ok: false, message: `해당 날짜(${bas_dd}) 거래 데이터가 없습니다.` };
|
|
43
|
+
return { source: "korean-stock", action: "trade-info", item };
|
|
44
|
+
}
|
|
45
|
+
export function renderStock(result) {
|
|
46
|
+
if (result.action === "search") {
|
|
47
|
+
const lines = [`# 종목 검색 — "${result.query}"`, ""];
|
|
48
|
+
for (const item of result.items) {
|
|
49
|
+
lines.push(`- **${item.name}** (${item.code}) — ${item.market}`);
|
|
50
|
+
}
|
|
51
|
+
lines.push("\n_KRX 공식 데이터 기준 / 투자 조언 아님_");
|
|
52
|
+
return lines.join("\n");
|
|
53
|
+
}
|
|
54
|
+
if (result.action === "base-info") {
|
|
55
|
+
const i = result.item;
|
|
56
|
+
return [
|
|
57
|
+
`# 종목 기본정보 — ${i.name} (${i.code})`,
|
|
58
|
+
`- 시장: ${i.market}`,
|
|
59
|
+
i.security_group ? `- 증권구분: ${i.security_group}` : "",
|
|
60
|
+
i.section_type ? `- 구분: ${i.section_type}` : "",
|
|
61
|
+
i.listed_shares ? `- 상장주식수: ${i.listed_shares.toLocaleString("ko")}주` : "",
|
|
62
|
+
i.listed_at ? `- 상장일: ${i.listed_at}` : "",
|
|
63
|
+
"",
|
|
64
|
+
"_KRX 공식 데이터 기준 / 투자 조언 아님_",
|
|
65
|
+
].filter((l) => l !== "").join("\n");
|
|
66
|
+
}
|
|
67
|
+
// trade-info
|
|
68
|
+
const i = result.item;
|
|
69
|
+
const sign = (i.change_price ?? 0) > 0 ? "+" : "";
|
|
70
|
+
return [
|
|
71
|
+
`# 종목 시세 — ${i.name} (${i.code}) ${i.base_date}`,
|
|
72
|
+
`- 시장: ${i.market}`,
|
|
73
|
+
i.close_price !== undefined ? `- 종가: ${i.close_price.toLocaleString("ko")}원` : "",
|
|
74
|
+
i.change_price !== undefined ? `- 등락: ${sign}${i.change_price.toLocaleString("ko")}원 (${sign}${i.fluctuation_rate ?? 0}%)` : "",
|
|
75
|
+
i.open_price !== undefined ? `- 시가: ${i.open_price.toLocaleString("ko")}원` : "",
|
|
76
|
+
i.high_price !== undefined ? `- 고가: ${i.high_price.toLocaleString("ko")}원` : "",
|
|
77
|
+
i.low_price !== undefined ? `- 저가: ${i.low_price.toLocaleString("ko")}원` : "",
|
|
78
|
+
i.trading_volume !== undefined ? `- 거래량: ${i.trading_volume.toLocaleString("ko")}주` : "",
|
|
79
|
+
i.market_cap !== undefined ? `- 시가총액: ${Math.round((i.market_cap ?? 0) / 1_0000_0000).toLocaleString("ko")}억원` : "",
|
|
80
|
+
"",
|
|
81
|
+
"_KRX 공식 데이터 기준 / 투자 조언 아님_",
|
|
82
|
+
].filter((l) => l !== "").join("\n");
|
|
83
|
+
}
|
|
84
|
+
//# sourceMappingURL=stock.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"stock.js","sourceRoot":"","sources":["../../src/korea/stock.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,29 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Seoul real-time subway arrival info via k-skill-proxy
|
|
3
|
+
* Proxy endpoint: GET /v1/seoul-subway/arrival?stationName=강남
|
|
4
|
+
*/
|
|
5
|
+
export type SubwayArrival = {
|
|
6
|
+
line: string;
|
|
7
|
+
direction: string;
|
|
8
|
+
first_arrival: string;
|
|
9
|
+
second_arrival: string;
|
|
10
|
+
arrival_seconds?: number;
|
|
11
|
+
destination?: string;
|
|
12
|
+
train_id?: string;
|
|
13
|
+
};
|
|
14
|
+
export type SubwayResult = {
|
|
15
|
+
source: "seoul-subway";
|
|
16
|
+
station: string;
|
|
17
|
+
queried_at: string;
|
|
18
|
+
total: number;
|
|
19
|
+
arrivals: SubwayArrival[];
|
|
20
|
+
};
|
|
21
|
+
export declare function runSubway(options: {
|
|
22
|
+
stationName: string;
|
|
23
|
+
limit?: number;
|
|
24
|
+
timeoutMs?: number;
|
|
25
|
+
}): Promise<SubwayResult | {
|
|
26
|
+
ok: false;
|
|
27
|
+
message: string;
|
|
28
|
+
}>;
|
|
29
|
+
export declare function renderSubway(result: SubwayResult): string;
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Seoul real-time subway arrival info via k-skill-proxy
|
|
3
|
+
* Proxy endpoint: GET /v1/seoul-subway/arrival?stationName=강남
|
|
4
|
+
*/
|
|
5
|
+
import { proxyGet } from "./proxy-client.js";
|
|
6
|
+
export async function runSubway(options) {
|
|
7
|
+
const { stationName, limit = 20, timeoutMs = 15000 } = options;
|
|
8
|
+
const result = await proxyGet("/v1/seoul-subway/arrival", { stationName }, timeoutMs);
|
|
9
|
+
if (!result.ok) {
|
|
10
|
+
return { ok: false, message: `Seoul subway lookup failed: ${result.message}` };
|
|
11
|
+
}
|
|
12
|
+
const data = result.data;
|
|
13
|
+
const rawList = (data["list"] ?? data["arrivals"] ?? data["items"] ?? []);
|
|
14
|
+
const arrivals = rawList.slice(0, limit).map((item) => ({
|
|
15
|
+
line: String(item["subwayId"] ?? item["line"] ?? item["호선"] ?? ""),
|
|
16
|
+
direction: String(item["updnLine"] ?? item["direction"] ?? item["방향"] ?? ""),
|
|
17
|
+
first_arrival: String(item["arvlMsg2"] ?? item["first_arrival"] ?? item["첫번째도착"] ?? ""),
|
|
18
|
+
second_arrival: String(item["arvlMsg3"] ?? item["second_arrival"] ?? item["두번째도착"] ?? ""),
|
|
19
|
+
...(item["barvlDt"] !== undefined ? { arrival_seconds: Number(item["barvlDt"]) } : {}),
|
|
20
|
+
...(item["bstatnNm"] !== undefined ? { destination: String(item["bstatnNm"]) } : {}),
|
|
21
|
+
...(item["btrainNo"] !== undefined ? { train_id: String(item["btrainNo"]) } : {}),
|
|
22
|
+
}));
|
|
23
|
+
return {
|
|
24
|
+
source: "seoul-subway",
|
|
25
|
+
station: stationName,
|
|
26
|
+
queried_at: new Date().toISOString(),
|
|
27
|
+
total: arrivals.length,
|
|
28
|
+
arrivals,
|
|
29
|
+
};
|
|
30
|
+
}
|
|
31
|
+
export function renderSubway(result) {
|
|
32
|
+
const lines = [`# 서울 지하철 도착정보 — ${result.station}역`, `조회 시점: ${result.queried_at}`, ""];
|
|
33
|
+
if (result.total === 0) {
|
|
34
|
+
lines.push("도착 정보가 없습니다.");
|
|
35
|
+
return lines.join("\n");
|
|
36
|
+
}
|
|
37
|
+
for (const a of result.arrivals) {
|
|
38
|
+
lines.push(`**${a.line}호선** ${a.direction}`);
|
|
39
|
+
lines.push(`- 첫번째: ${a.first_arrival}`);
|
|
40
|
+
lines.push(`- 두번째: ${a.second_arrival}`);
|
|
41
|
+
if (a.destination)
|
|
42
|
+
lines.push(`- 종착역: ${a.destination}`);
|
|
43
|
+
lines.push("");
|
|
44
|
+
}
|
|
45
|
+
return lines.join("\n");
|
|
46
|
+
}
|
|
47
|
+
//# sourceMappingURL=subway.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"subway.js","sourceRoot":"","sources":["../../src/korea/subway.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH,OAAO,EAAE,QAAQ,EAAE,MAAM,mBAAmB,CAAA;AAoB5C,MAAM,CAAC,KAAK,UAAU,SAAS,CAAC,OAI/B;IACC,MAAM,EAAE,WAAW,EAAE,KAAK,GAAG,EAAE,EAAE,SAAS,GAAG,KAAK,EAAE,GAAG,OAAO,CAAA;IAE9D,MAAM,MAAM,GAAG,MAAM,QAAQ,CAA0B,0BAA0B,EAAE,EAAE,WAAW,EAAE,EAAE,SAAS,CAAC,CAAA;IAC9G,IAAI,CAAC,MAAM,CAAC,EAAE,EAAE,CAAC;QACf,OAAO,EAAE,EAAE,EAAE,KAAK,EAAE,OAAO,EAAE,+BAA+B,MAAM,CAAC,OAAO,EAAE,EAAE,CAAA;IAChF,CAAC;IAED,MAAM,IAAI,GAAG,MAAM,CAAC,IAAI,CAAA;IACxB,MAAM,OAAO,GAAG,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,IAAI,CAAC,UAAU,CAAC,IAAI,IAAI,CAAC,OAAO,CAAC,IAAI,EAAE,CAA8B,CAAA;IACtG,MAAM,QAAQ,GAAoB,OAAO,CAAC,KAAK,CAAC,CAAC,EAAE,KAAK,CAAC,CAAC,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC;QACvE,IAAI,EAAE,MAAM,CAAC,IAAI,CAAC,UAAU,CAAC,IAAI,IAAI,CAAC,MAAM,CAAC,IAAI,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC;QAClE,SAAS,EAAE,MAAM,CAAC,IAAI,CAAC,UAAU,CAAC,IAAI,IAAI,CAAC,WAAW,CAAC,IAAI,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC;QAC5E,aAAa,EAAE,MAAM,CAAC,IAAI,CAAC,UAAU,CAAC,IAAI,IAAI,CAAC,eAAe,CAAC,IAAI,IAAI,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC;QACvF,cAAc,EAAE,MAAM,CAAC,IAAI,CAAC,UAAU,CAAC,IAAI,IAAI,CAAC,gBAAgB,CAAC,IAAI,IAAI,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC;QACzF,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,KAAK,SAAS,CAAC,CAAC,CAAC,EAAE,eAAe,EAAE,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;QACtF,GAAG,CAAC,IAAI,CAAC,UAAU,CAAC,KAAK,SAAS,CAAC,CAAC,CAAC,EAAE,WAAW,EAAE,MAAM,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;QACpF,GAAG,CAAC,IAAI,CAAC,UAAU,CAAC,KAAK,SAAS,CAAC,CAAC,CAAC,EAAE,QAAQ,EAAE,MAAM,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;KAClF,CAAC,CAAC,CAAA;IAEH,OAAO;QACL,MAAM,EAAE,cAAc;QACtB,OAAO,EAAE,WAAW;QACpB,UAAU,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;QACpC,KAAK,EAAE,QAAQ,CAAC,MAAM;QACtB,QAAQ;KACT,CAAA;AACH,CAAC;AAED,MAAM,UAAU,YAAY,CAAC,MAAoB;IAC/C,MAAM,KAAK,GAAG,CAAC,mBAAmB,MAAM,CAAC,OAAO,GAAG,EAAE,UAAU,MAAM,CAAC,UAAU,EAAE,EAAE,EAAE,CAAC,CAAA;IACvF,IAAI,MAAM,CAAC,KAAK,KAAK,CAAC,EAAE,CAAC;QACvB,KAAK,CAAC,IAAI,CAAC,cAAc,CAAC,CAAA;QAC1B,OAAO,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;IACzB,CAAC;IACD,KAAK,MAAM,CAAC,IAAI,MAAM,CAAC,QAAQ,EAAE,CAAC;QAChC,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,IAAI,QAAQ,CAAC,CAAC,SAAS,EAAE,CAAC,CAAA;QAC5C,KAAK,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,aAAa,EAAE,CAAC,CAAA;QACvC,KAAK,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,cAAc,EAAE,CAAC,CAAA;QACxC,IAAI,CAAC,CAAC,WAAW;YAAE,KAAK,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,WAAW,EAAE,CAAC,CAAA;QACxD,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAA;IAChB,CAAC;IACD,OAAO,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;AACzB,CAAC"}
|
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Korean weather forecast via k-skill-proxy
|
|
3
|
+
* Proxy endpoint: GET /v1/korea-weather/forecast
|
|
4
|
+
*/
|
|
5
|
+
export type WeatherItem = {
|
|
6
|
+
category: string;
|
|
7
|
+
forecast_date: string;
|
|
8
|
+
forecast_time: string;
|
|
9
|
+
value: string;
|
|
10
|
+
};
|
|
11
|
+
export type WeatherResult = {
|
|
12
|
+
source: "korea-weather";
|
|
13
|
+
nx?: number;
|
|
14
|
+
ny?: number;
|
|
15
|
+
lat?: number;
|
|
16
|
+
lon?: number;
|
|
17
|
+
base_date: string;
|
|
18
|
+
base_time: string;
|
|
19
|
+
total: number;
|
|
20
|
+
items: WeatherItem[];
|
|
21
|
+
summary?: Record<string, string>;
|
|
22
|
+
};
|
|
23
|
+
export declare function runWeather(options: {
|
|
24
|
+
nx?: number;
|
|
25
|
+
ny?: number;
|
|
26
|
+
lat?: number;
|
|
27
|
+
lon?: number;
|
|
28
|
+
baseDate?: string;
|
|
29
|
+
baseTime?: string;
|
|
30
|
+
timeoutMs?: number;
|
|
31
|
+
}): Promise<WeatherResult | {
|
|
32
|
+
ok: false;
|
|
33
|
+
message: string;
|
|
34
|
+
}>;
|
|
35
|
+
export declare function renderWeather(result: WeatherResult): string;
|
|
@@ -0,0 +1,92 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Korean weather forecast via k-skill-proxy
|
|
3
|
+
* Proxy endpoint: GET /v1/korea-weather/forecast
|
|
4
|
+
*/
|
|
5
|
+
import { proxyGet } from "./proxy-client.js";
|
|
6
|
+
const CATEGORY_LABELS = {
|
|
7
|
+
TMP: "기온(°C)",
|
|
8
|
+
SKY: "하늘상태",
|
|
9
|
+
PTY: "강수형태",
|
|
10
|
+
POP: "강수확률(%)",
|
|
11
|
+
WSD: "풍속(m/s)",
|
|
12
|
+
REH: "습도(%)",
|
|
13
|
+
PCP: "1시간강수량",
|
|
14
|
+
SNO: "1시간신적설",
|
|
15
|
+
TMN: "일최저기온",
|
|
16
|
+
TMX: "일최고기온",
|
|
17
|
+
};
|
|
18
|
+
const SKY_CODES = { "1": "맑음", "3": "구름많음", "4": "흐림" };
|
|
19
|
+
const PTY_CODES = { "0": "없음", "1": "비", "2": "비/눈", "3": "눈", "4": "소나기" };
|
|
20
|
+
function decodeValue(category, value) {
|
|
21
|
+
if (category === "SKY")
|
|
22
|
+
return SKY_CODES[value] ?? value;
|
|
23
|
+
if (category === "PTY")
|
|
24
|
+
return PTY_CODES[value] ?? value;
|
|
25
|
+
return value;
|
|
26
|
+
}
|
|
27
|
+
export async function runWeather(options) {
|
|
28
|
+
const { nx, ny, lat, lon, baseDate, baseTime, timeoutMs = 15000 } = options;
|
|
29
|
+
const params = {};
|
|
30
|
+
if (nx !== undefined)
|
|
31
|
+
params["nx"] = String(nx);
|
|
32
|
+
if (ny !== undefined)
|
|
33
|
+
params["ny"] = String(ny);
|
|
34
|
+
if (lat !== undefined)
|
|
35
|
+
params["lat"] = String(lat);
|
|
36
|
+
if (lon !== undefined)
|
|
37
|
+
params["lon"] = String(lon);
|
|
38
|
+
if (baseDate)
|
|
39
|
+
params["baseDate"] = baseDate;
|
|
40
|
+
if (baseTime)
|
|
41
|
+
params["baseTime"] = baseTime;
|
|
42
|
+
const result = await proxyGet("/v1/korea-weather/forecast", params, timeoutMs);
|
|
43
|
+
if (!result.ok) {
|
|
44
|
+
return { ok: false, message: `Weather lookup failed: ${result.message}` };
|
|
45
|
+
}
|
|
46
|
+
const data = result.data;
|
|
47
|
+
const rawItems = (data["items"] ?? data["list"] ?? []);
|
|
48
|
+
const items = rawItems.map((item) => {
|
|
49
|
+
const cat = String(item["category"] ?? "");
|
|
50
|
+
const val = String(item["fcstValue"] ?? item["value"] ?? "");
|
|
51
|
+
return {
|
|
52
|
+
category: cat,
|
|
53
|
+
forecast_date: String(item["fcstDate"] ?? item["forecast_date"] ?? ""),
|
|
54
|
+
forecast_time: String(item["fcstTime"] ?? item["forecast_time"] ?? ""),
|
|
55
|
+
value: decodeValue(cat, val),
|
|
56
|
+
};
|
|
57
|
+
});
|
|
58
|
+
// Build summary from latest forecast slot
|
|
59
|
+
const summary = {};
|
|
60
|
+
const latestSlot = items[0]?.forecast_date ? `${items[0].forecast_date}${items[0].forecast_time}` : "";
|
|
61
|
+
for (const item of items) {
|
|
62
|
+
const slot = `${item.forecast_date}${item.forecast_time}`;
|
|
63
|
+
if (slot === latestSlot || !summary[item.category]) {
|
|
64
|
+
summary[CATEGORY_LABELS[item.category] ?? item.category] = item.value;
|
|
65
|
+
}
|
|
66
|
+
}
|
|
67
|
+
return {
|
|
68
|
+
source: "korea-weather",
|
|
69
|
+
...(nx !== undefined ? { nx } : {}),
|
|
70
|
+
...(ny !== undefined ? { ny } : {}),
|
|
71
|
+
...(lat !== undefined ? { lat } : {}),
|
|
72
|
+
...(lon !== undefined ? { lon } : {}),
|
|
73
|
+
base_date: String(data["base_date"] ?? data["baseDate"] ?? ""),
|
|
74
|
+
base_time: String(data["base_time"] ?? data["baseTime"] ?? ""),
|
|
75
|
+
total: items.length,
|
|
76
|
+
items,
|
|
77
|
+
summary,
|
|
78
|
+
};
|
|
79
|
+
}
|
|
80
|
+
export function renderWeather(result) {
|
|
81
|
+
const loc = result.lat !== undefined ? `위도 ${result.lat}, 경도 ${result.lon}` : `격자 (${result.nx}, ${result.ny})`;
|
|
82
|
+
const lines = [`# 한국 날씨 — ${loc}`, `기준: ${result.base_date} ${result.base_time}`, ""];
|
|
83
|
+
if (result.summary && Object.keys(result.summary).length > 0) {
|
|
84
|
+
lines.push("## 요약");
|
|
85
|
+
for (const [label, val] of Object.entries(result.summary)) {
|
|
86
|
+
lines.push(`- ${label}: ${val}`);
|
|
87
|
+
}
|
|
88
|
+
lines.push("");
|
|
89
|
+
}
|
|
90
|
+
return lines.join("\n");
|
|
91
|
+
}
|
|
92
|
+
//# sourceMappingURL=weather.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"weather.js","sourceRoot":"","sources":["../../src/korea/weather.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Korean postal code + official English address lookup
|
|
3
|
+
* Scrapes ePost official integrated search: https://www.epost.kr/search.RetrieveIntegrationNewZipCdList.comm
|
|
4
|
+
*/
|
|
5
|
+
export type ZipcodeResult = {
|
|
6
|
+
source: "zipcode";
|
|
7
|
+
query: string;
|
|
8
|
+
total: number;
|
|
9
|
+
results: ZipcodeEntry[];
|
|
10
|
+
};
|
|
11
|
+
export type ZipcodeEntry = {
|
|
12
|
+
zip_code: string;
|
|
13
|
+
road_address: string;
|
|
14
|
+
english_address: string;
|
|
15
|
+
jibun_address: string;
|
|
16
|
+
};
|
|
17
|
+
export declare function runZipcode(options: {
|
|
18
|
+
query: string;
|
|
19
|
+
limit?: number;
|
|
20
|
+
timeoutMs?: number;
|
|
21
|
+
}): Promise<ZipcodeResult | {
|
|
22
|
+
ok: false;
|
|
23
|
+
message: string;
|
|
24
|
+
}>;
|
|
25
|
+
export declare function renderZipcode(result: ZipcodeResult): string;
|
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Korean postal code + official English address lookup
|
|
3
|
+
* Scrapes ePost official integrated search: https://www.epost.kr/search.RetrieveIntegrationNewZipCdList.comm
|
|
4
|
+
*/
|
|
5
|
+
import { httpPost } from "./proxy-client.js";
|
|
6
|
+
const EPOST_URL = "https://www.epost.kr/search.RetrieveIntegrationNewZipCdList.comm";
|
|
7
|
+
function decodeHtmlEntities(text) {
|
|
8
|
+
return text
|
|
9
|
+
.replace(/&/g, "&")
|
|
10
|
+
.replace(/</g, "<")
|
|
11
|
+
.replace(/>/g, ">")
|
|
12
|
+
.replace(/"/g, '"')
|
|
13
|
+
.replace(/'/g, "'")
|
|
14
|
+
.replace(/ /g, " ")
|
|
15
|
+
.replace(/&#(\d+);/g, (_m, n) => String.fromCharCode(parseInt(n, 10)));
|
|
16
|
+
}
|
|
17
|
+
function parseViewDetail(html) {
|
|
18
|
+
const results = [];
|
|
19
|
+
// viewDetail('zipCode','roadAddress','englishAddress','jibunAddress', rowIndex)
|
|
20
|
+
const pattern = /viewDetail\('([^']*?)','([^']*?)','([^']*?)','([^']*?)',\s*'[^']*?'\)/g;
|
|
21
|
+
for (const match of html.matchAll(pattern)) {
|
|
22
|
+
results.push({
|
|
23
|
+
zip_code: decodeHtmlEntities(match[1] ?? ""),
|
|
24
|
+
road_address: decodeHtmlEntities(match[2] ?? ""),
|
|
25
|
+
english_address: decodeHtmlEntities(match[3] ?? ""),
|
|
26
|
+
jibun_address: decodeHtmlEntities(match[4] ?? ""),
|
|
27
|
+
});
|
|
28
|
+
}
|
|
29
|
+
return results;
|
|
30
|
+
}
|
|
31
|
+
export async function runZipcode(options) {
|
|
32
|
+
const { query, limit = 5, timeoutMs = 20000 } = options;
|
|
33
|
+
if (!query.trim())
|
|
34
|
+
return { ok: false, message: "주소 키워드가 필요합니다." };
|
|
35
|
+
const body = new URLSearchParams({ keyword: query }).toString();
|
|
36
|
+
const result = await httpPost(EPOST_URL, body, {
|
|
37
|
+
"Content-Type": "application/x-www-form-urlencoded",
|
|
38
|
+
accept: "text/html,application/xhtml+xml,*/*",
|
|
39
|
+
"accept-language": "ko-KR,ko;q=0.9",
|
|
40
|
+
referer: "https://www.epost.kr/",
|
|
41
|
+
}, timeoutMs);
|
|
42
|
+
if (!result.ok) {
|
|
43
|
+
return { ok: false, message: `우편번호 검색 실패: ${result.message}` };
|
|
44
|
+
}
|
|
45
|
+
const html = result.data;
|
|
46
|
+
const entries = parseViewDetail(html).slice(0, limit);
|
|
47
|
+
if (entries.length === 0) {
|
|
48
|
+
// Try a shorter keyword fallback
|
|
49
|
+
const fallback = query.split(" ").slice(-2).join(" ");
|
|
50
|
+
if (fallback !== query) {
|
|
51
|
+
return runZipcode({ query: fallback, limit, timeoutMs });
|
|
52
|
+
}
|
|
53
|
+
return { ok: false, message: `"${query}"에 해당하는 우편번호 결과가 없습니다.` };
|
|
54
|
+
}
|
|
55
|
+
return { source: "zipcode", query, total: entries.length, results: entries };
|
|
56
|
+
}
|
|
57
|
+
export function renderZipcode(result) {
|
|
58
|
+
const lines = [`# 우편번호 검색 — "${result.query}"`, `${result.total}건 (상위 결과)`, ""];
|
|
59
|
+
for (const r of result.results) {
|
|
60
|
+
lines.push(`## ${r.zip_code}`);
|
|
61
|
+
lines.push(`- 국문: ${r.road_address}`);
|
|
62
|
+
lines.push(`- 영문: ${r.english_address}`);
|
|
63
|
+
if (r.jibun_address)
|
|
64
|
+
lines.push(`- 지번: ${r.jibun_address}`);
|
|
65
|
+
lines.push("");
|
|
66
|
+
}
|
|
67
|
+
lines.push("_우체국 공식 통합 우편번호 서비스(epost.kr) 기준_");
|
|
68
|
+
return lines.join("\n");
|
|
69
|
+
}
|
|
70
|
+
//# sourceMappingURL=zipcode.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"zipcode.js","sourceRoot":"","sources":["../../src/korea/zipcode.ts"],"names":[],"mappings":"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"}
|