kl-mcp-client 2.1.2__tar.gz → 2.1.4__tar.gz
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.
- {kl_mcp_client-2.1.2 → kl_mcp_client-2.1.4}/PKG-INFO +1 -1
- {kl_mcp_client-2.1.2 → kl_mcp_client-2.1.4}/kl_mcp_client/tools.py +238 -0
- {kl_mcp_client-2.1.2 → kl_mcp_client-2.1.4}/kl_mcp_client.egg-info/PKG-INFO +1 -1
- {kl_mcp_client-2.1.2 → kl_mcp_client-2.1.4}/pyproject.toml +1 -1
- {kl_mcp_client-2.1.2 → kl_mcp_client-2.1.4}/README.md +0 -0
- {kl_mcp_client-2.1.2 → kl_mcp_client-2.1.4}/kl_mcp_client/__init__.py +0 -0
- {kl_mcp_client-2.1.2 → kl_mcp_client-2.1.4}/kl_mcp_client/__version__.py +0 -0
- {kl_mcp_client-2.1.2 → kl_mcp_client-2.1.4}/kl_mcp_client/asyncio/__init__.py +0 -0
- {kl_mcp_client-2.1.2 → kl_mcp_client-2.1.4}/kl_mcp_client/asyncio/client.py +0 -0
- {kl_mcp_client-2.1.2 → kl_mcp_client-2.1.4}/kl_mcp_client/asyncio/tools.py +0 -0
- {kl_mcp_client-2.1.2 → kl_mcp_client-2.1.4}/kl_mcp_client/client.py +0 -0
- {kl_mcp_client-2.1.2 → kl_mcp_client-2.1.4}/kl_mcp_client.egg-info/SOURCES.txt +0 -0
- {kl_mcp_client-2.1.2 → kl_mcp_client-2.1.4}/kl_mcp_client.egg-info/dependency_links.txt +0 -0
- {kl_mcp_client-2.1.2 → kl_mcp_client-2.1.4}/kl_mcp_client.egg-info/requires.txt +0 -0
- {kl_mcp_client-2.1.2 → kl_mcp_client-2.1.4}/kl_mcp_client.egg-info/top_level.txt +0 -0
- {kl_mcp_client-2.1.2 → kl_mcp_client-2.1.4}/setup.cfg +0 -0
|
@@ -321,3 +321,241 @@ class MCPTools:
|
|
|
321
321
|
"getCleanText",
|
|
322
322
|
{"sessionId": sessionId},
|
|
323
323
|
).get("structuredContent", {})
|
|
324
|
+
|
|
325
|
+
@_ensure_client
|
|
326
|
+
def evaluate_stream(
|
|
327
|
+
self,
|
|
328
|
+
sessionId: str,
|
|
329
|
+
expression: str,
|
|
330
|
+
chunkSize: int = 100,
|
|
331
|
+
) -> Dict[str, Any]:
|
|
332
|
+
"""
|
|
333
|
+
Evaluate JS expression theo chế độ STREAM.
|
|
334
|
+
Dùng khi kết quả là array lớn (DOM, list, table, ...)
|
|
335
|
+
|
|
336
|
+
Returns (init response):
|
|
337
|
+
{
|
|
338
|
+
"stream_id": "...",
|
|
339
|
+
"total": 1234,
|
|
340
|
+
"chunk_size": 100
|
|
341
|
+
}
|
|
342
|
+
"""
|
|
343
|
+
return self.client.call_tool(
|
|
344
|
+
"evaluate.stream",
|
|
345
|
+
{
|
|
346
|
+
"sessionId": sessionId,
|
|
347
|
+
"expression": expression,
|
|
348
|
+
"chunkSize": int(chunkSize),
|
|
349
|
+
},
|
|
350
|
+
).get("structuredContent", {})
|
|
351
|
+
|
|
352
|
+
@_ensure_client
|
|
353
|
+
def stream_pull(
|
|
354
|
+
self,
|
|
355
|
+
stream_id: str,
|
|
356
|
+
offset: int = 0,
|
|
357
|
+
limit: int = 100,
|
|
358
|
+
) -> Dict[str, Any]:
|
|
359
|
+
"""
|
|
360
|
+
Kéo 1 chunk từ stream đã tạo bởi evaluate.stream
|
|
361
|
+
|
|
362
|
+
Returns:
|
|
363
|
+
{
|
|
364
|
+
"items": [...],
|
|
365
|
+
"offset": 0,
|
|
366
|
+
"has_more": true
|
|
367
|
+
}
|
|
368
|
+
"""
|
|
369
|
+
return self.client.call_tool(
|
|
370
|
+
"stream.pull",
|
|
371
|
+
{
|
|
372
|
+
"stream_id": stream_id,
|
|
373
|
+
"offset": int(offset),
|
|
374
|
+
"limit": int(limit),
|
|
375
|
+
},
|
|
376
|
+
).get("structuredContent", {})
|
|
377
|
+
|
|
378
|
+
@_ensure_client
|
|
379
|
+
def evaluate_stream_all(
|
|
380
|
+
self,
|
|
381
|
+
sessionId: str,
|
|
382
|
+
expression: str,
|
|
383
|
+
chunkSize: int = 100,
|
|
384
|
+
max_items: Optional[int] = None,
|
|
385
|
+
):
|
|
386
|
+
"""
|
|
387
|
+
Helper: evaluate.stream + tự động pull toàn bộ dữ liệu.
|
|
388
|
+
|
|
389
|
+
⚠️ Chỉ dùng khi bạn THỰC SỰ cần full data.
|
|
390
|
+
"""
|
|
391
|
+
init = self.evaluate_stream(
|
|
392
|
+
sessionId=sessionId,
|
|
393
|
+
expression=expression,
|
|
394
|
+
chunkSize=chunkSize,
|
|
395
|
+
)
|
|
396
|
+
|
|
397
|
+
stream_id = init.get("stream_id")
|
|
398
|
+
total = init.get("total", 0)
|
|
399
|
+
|
|
400
|
+
if not stream_id:
|
|
401
|
+
return []
|
|
402
|
+
|
|
403
|
+
items = []
|
|
404
|
+
offset = 0
|
|
405
|
+
|
|
406
|
+
while True:
|
|
407
|
+
chunk = self.stream_pull(
|
|
408
|
+
stream_id=stream_id,
|
|
409
|
+
offset=offset,
|
|
410
|
+
limit=chunkSize,
|
|
411
|
+
)
|
|
412
|
+
|
|
413
|
+
part = chunk.get("items", [])
|
|
414
|
+
items.extend(part)
|
|
415
|
+
|
|
416
|
+
if max_items and len(items) >= max_items:
|
|
417
|
+
return items[:max_items]
|
|
418
|
+
|
|
419
|
+
if not chunk.get("has_more"):
|
|
420
|
+
break
|
|
421
|
+
|
|
422
|
+
offset += chunkSize
|
|
423
|
+
|
|
424
|
+
return items
|
|
425
|
+
|
|
426
|
+
@_ensure_client
|
|
427
|
+
def wait_for_selector(
|
|
428
|
+
self, sessionId: str, selector: str, timeoutMs: Optional[int] = None
|
|
429
|
+
) -> Dict[str, Any]:
|
|
430
|
+
args = {
|
|
431
|
+
"sessionId": sessionId,
|
|
432
|
+
"selector": selector,
|
|
433
|
+
}
|
|
434
|
+
if timeoutMs is not None:
|
|
435
|
+
args["timeout"] = int(timeoutMs)
|
|
436
|
+
|
|
437
|
+
return self.client.call_tool("waitForSelector", args).get(
|
|
438
|
+
"structuredContent", {}
|
|
439
|
+
)
|
|
440
|
+
|
|
441
|
+
@_ensure_client
|
|
442
|
+
def send_keys(
|
|
443
|
+
self, sessionId: str, key: str, interval: int = 100
|
|
444
|
+
) -> Dict[str, Any]:
|
|
445
|
+
"""
|
|
446
|
+
Gửi phím: enter, tab, esc, ctrl+c, ctrl+shift+tab, ...
|
|
447
|
+
"""
|
|
448
|
+
return self.client.call_tool(
|
|
449
|
+
"sendKeys",
|
|
450
|
+
{"sessionId": sessionId, "text": key, "interval": interval},
|
|
451
|
+
).get("structuredContent", {})
|
|
452
|
+
|
|
453
|
+
@_ensure_client
|
|
454
|
+
def perform(
|
|
455
|
+
self,
|
|
456
|
+
sessionId: str,
|
|
457
|
+
action: str,
|
|
458
|
+
target: Optional[str] = None,
|
|
459
|
+
value: Optional[str] = None,
|
|
460
|
+
x: Optional[float] = None,
|
|
461
|
+
y: Optional[float] = None,
|
|
462
|
+
from_point: Optional[dict] = None,
|
|
463
|
+
to_point: Optional[dict] = None,
|
|
464
|
+
) -> Dict[str, Any]:
|
|
465
|
+
"""
|
|
466
|
+
action:
|
|
467
|
+
- click (x,y)
|
|
468
|
+
- move / hover (x,y)
|
|
469
|
+
- drag (from -> to)
|
|
470
|
+
- type (target, value)
|
|
471
|
+
"""
|
|
472
|
+
args = {
|
|
473
|
+
"sessionId": sessionId,
|
|
474
|
+
"action": action,
|
|
475
|
+
}
|
|
476
|
+
|
|
477
|
+
if target is not None:
|
|
478
|
+
args["target"] = target
|
|
479
|
+
if value is not None:
|
|
480
|
+
args["value"] = value
|
|
481
|
+
if x is not None:
|
|
482
|
+
args["x"] = float(x)
|
|
483
|
+
if y is not None:
|
|
484
|
+
args["y"] = float(y)
|
|
485
|
+
if from_point is not None:
|
|
486
|
+
args["from"] = from_point
|
|
487
|
+
if to_point is not None:
|
|
488
|
+
args["to"] = to_point
|
|
489
|
+
|
|
490
|
+
return self.client.call_tool("perform", args).get("structuredContent", {})
|
|
491
|
+
|
|
492
|
+
@_ensure_client
|
|
493
|
+
def drag_and_drop(
|
|
494
|
+
self,
|
|
495
|
+
sessionId: str,
|
|
496
|
+
from_x: float,
|
|
497
|
+
from_y: float,
|
|
498
|
+
to_x: float,
|
|
499
|
+
to_y: float,
|
|
500
|
+
):
|
|
501
|
+
return self.perform(
|
|
502
|
+
sessionId=sessionId,
|
|
503
|
+
action="drag",
|
|
504
|
+
from_point={"x": from_x, "y": from_y},
|
|
505
|
+
to_point={"x": to_x, "y": to_y},
|
|
506
|
+
)
|
|
507
|
+
|
|
508
|
+
def hover(self, sessionId: str, x: float, y: float):
|
|
509
|
+
return self.perform(
|
|
510
|
+
sessionId=sessionId,
|
|
511
|
+
action="hover",
|
|
512
|
+
x=x,
|
|
513
|
+
y=y,
|
|
514
|
+
)
|
|
515
|
+
|
|
516
|
+
@_ensure_client
|
|
517
|
+
def scroll(
|
|
518
|
+
self,
|
|
519
|
+
sessionId: str,
|
|
520
|
+
*,
|
|
521
|
+
x: Optional[float] = None,
|
|
522
|
+
y: Optional[float] = None,
|
|
523
|
+
selector: Optional[str] = None,
|
|
524
|
+
position: Optional[str] = None, # "top" | "bottom"
|
|
525
|
+
) -> Dict[str, Any]:
|
|
526
|
+
"""
|
|
527
|
+
Scroll trang web.
|
|
528
|
+
|
|
529
|
+
Cách dùng:
|
|
530
|
+
- Scroll theo pixel:
|
|
531
|
+
scroll(sessionId, y=500)
|
|
532
|
+
- Scroll tới element:
|
|
533
|
+
scroll(sessionId, selector="#footer")
|
|
534
|
+
- Scroll top / bottom:
|
|
535
|
+
scroll(sessionId, position="top")
|
|
536
|
+
scroll(sessionId, position="bottom")
|
|
537
|
+
"""
|
|
538
|
+
|
|
539
|
+
args: Dict[str, Any] = {
|
|
540
|
+
"sessionId": sessionId,
|
|
541
|
+
}
|
|
542
|
+
|
|
543
|
+
if x is not None:
|
|
544
|
+
args["x"] = float(x)
|
|
545
|
+
if y is not None:
|
|
546
|
+
args["y"] = float(y)
|
|
547
|
+
if selector is not None:
|
|
548
|
+
args["selector"] = selector
|
|
549
|
+
if position is not None:
|
|
550
|
+
args["position"] = position
|
|
551
|
+
|
|
552
|
+
if len(args) == 1:
|
|
553
|
+
return {
|
|
554
|
+
"ok": False,
|
|
555
|
+
"error": "scroll requires x/y or selector or position",
|
|
556
|
+
}
|
|
557
|
+
|
|
558
|
+
return self.client.call_tool(
|
|
559
|
+
"scroll",
|
|
560
|
+
args,
|
|
561
|
+
).get("structuredContent", {})
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|