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.
Files changed (58) hide show
  1. package/CHANGELOG.md +9 -0
  2. package/dist/korea/coupang.d.ts +46 -0
  3. package/dist/korea/coupang.js +165 -0
  4. package/dist/korea/coupang.js.map +1 -0
  5. package/dist/korea/dart.d.ts +53 -0
  6. package/dist/korea/dart.js +105 -0
  7. package/dist/korea/dart.js.map +1 -0
  8. package/dist/korea/drug-safety.d.ts +33 -0
  9. package/dist/korea/drug-safety.js +60 -0
  10. package/dist/korea/drug-safety.js.map +1 -0
  11. package/dist/korea/fine-dust.d.ts +27 -0
  12. package/dist/korea/fine-dust.js +88 -0
  13. package/dist/korea/fine-dust.js.map +1 -0
  14. package/dist/korea/food-safety.d.ts +21 -0
  15. package/dist/korea/food-safety.js +50 -0
  16. package/dist/korea/food-safety.js.map +1 -0
  17. package/dist/korea/geeknews.d.ts +31 -0
  18. package/dist/korea/geeknews.js +144 -0
  19. package/dist/korea/geeknews.js.map +1 -0
  20. package/dist/korea/index.d.ts +16 -0
  21. package/dist/korea/index.js +17 -0
  22. package/dist/korea/index.js.map +1 -0
  23. package/dist/korea/law.d.ts +41 -0
  24. package/dist/korea/law.js +85 -0
  25. package/dist/korea/law.js.map +1 -0
  26. package/dist/korea/naver-news.d.ts +38 -0
  27. package/dist/korea/naver-news.js +67 -0
  28. package/dist/korea/naver-news.js.map +1 -0
  29. package/dist/korea/naver-shopping.d.ts +39 -0
  30. package/dist/korea/naver-shopping.js +51 -0
  31. package/dist/korea/naver-shopping.js.map +1 -0
  32. package/dist/korea/proxy-client.d.ts +20 -0
  33. package/dist/korea/proxy-client.js +88 -0
  34. package/dist/korea/proxy-client.js.map +1 -0
  35. package/dist/korea/public-restroom.d.ts +44 -0
  36. package/dist/korea/public-restroom.js +184 -0
  37. package/dist/korea/public-restroom.js.map +1 -0
  38. package/dist/korea/real-estate.d.ts +56 -0
  39. package/dist/korea/real-estate.js +95 -0
  40. package/dist/korea/real-estate.js.map +1 -0
  41. package/dist/korea/spellcheck.d.ts +32 -0
  42. package/dist/korea/spellcheck.js +142 -0
  43. package/dist/korea/spellcheck.js.map +1 -0
  44. package/dist/korea/stock.d.ts +59 -0
  45. package/dist/korea/stock.js +84 -0
  46. package/dist/korea/stock.js.map +1 -0
  47. package/dist/korea/subway.d.ts +29 -0
  48. package/dist/korea/subway.js +47 -0
  49. package/dist/korea/subway.js.map +1 -0
  50. package/dist/korea/weather.d.ts +35 -0
  51. package/dist/korea/weather.js +92 -0
  52. package/dist/korea/weather.js.map +1 -0
  53. package/dist/korea/zipcode.d.ts +25 -0
  54. package/dist/korea/zipcode.js +70 -0
  55. package/dist/korea/zipcode.js.map +1 -0
  56. package/dist/mcp-server.js +386 -0
  57. package/dist/mcp-server.js.map +1 -1
  58. 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(/&amp;/g, "&").replace(/&lt;/g, "<").replace(/&gt;/g, ">").replace(/&quot;/g, '"').replace(/&nbsp;/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(/&amp;/g, "&")
10
+ .replace(/&lt;/g, "<")
11
+ .replace(/&gt;/g, ">")
12
+ .replace(/&quot;/g, '"')
13
+ .replace(/&#39;/g, "'")
14
+ .replace(/&nbsp;/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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"}