mcp-proxy-adapter 3.1.3__py3-none-any.whl → 3.1.4__py3-none-any.whl

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.
@@ -0,0 +1,123 @@
1
+ #!/usr/bin/env python3
2
+ """
3
+ Скрипт для исправления проблемы с обработкой JavaScript null в сервисе VSTL.
4
+
5
+ Этот скрипт демонстрирует проблему с обработкой null значений в VSTL
6
+ и способ её решения с помощью обновленной реализации метода validate_params.
7
+ """
8
+
9
+ import sys
10
+ import json
11
+ import requests
12
+ from typing import Dict, Any, Optional
13
+
14
+ # URL и заголовки для VSTL сервиса
15
+ VSTL_URL = "http://localhost:8000/cmd"
16
+ HEADERS = {"Content-Type": "application/json"}
17
+
18
+ def call_vstl_help(params: Optional[Dict[str, Any]] = None) -> Dict[str, Any]:
19
+ """
20
+ Вызывает команду help в сервисе VSTL.
21
+
22
+ Args:
23
+ params: Параметры для команды help
24
+
25
+ Returns:
26
+ Dict[str, Any]: Ответ от сервиса
27
+ """
28
+ payload = {
29
+ "jsonrpc": "2.0",
30
+ "method": "help",
31
+ "params": params or {},
32
+ "id": 1
33
+ }
34
+
35
+ response = requests.post(VSTL_URL, json=payload, headers=HEADERS)
36
+ return response.json()
37
+
38
+ def test_validate_params_fix():
39
+ """
40
+ Демонстрирует проблему с обработкой null и решение с помощью
41
+ улучшенной реализации метода validate_params.
42
+ """
43
+ # Оригинальная реализация метода validate_params
44
+ def original_validate_params(params: Dict[str, Any]) -> Dict[str, Any]:
45
+ if params is None:
46
+ params = {}
47
+
48
+ validated_params = params.copy()
49
+
50
+ for key, value in list(validated_params.items()):
51
+ if value is None or (isinstance(value, str) and value == ""):
52
+ if key in ["cmdname"]:
53
+ pass
54
+ else:
55
+ del validated_params[key]
56
+
57
+ return validated_params
58
+
59
+ # Улучшенная реализация метода validate_params
60
+ def improved_validate_params(params: Dict[str, Any]) -> Dict[str, Any]:
61
+ if params is None:
62
+ params = {}
63
+
64
+ validated_params = params.copy()
65
+
66
+ for key, value in list(validated_params.items()):
67
+ if value is None or (isinstance(value, str) and value.lower() in ["null", "none", ""]):
68
+ if key in ["cmdname"]:
69
+ validated_params[key] = None
70
+ else:
71
+ del validated_params[key]
72
+
73
+ return validated_params
74
+
75
+ # Тестирование оригинальной реализации
76
+ original_params = {"unknown_param": "null"}
77
+
78
+ print("\n=== Оригинальные параметры ===")
79
+ print(f"Параметры: {original_params}")
80
+ print("Результат validate_params (оригинальный):")
81
+ try:
82
+ print(original_validate_params(original_params))
83
+ except Exception as e:
84
+ print(f"ОШИБКА: {e}")
85
+
86
+ # Тестирование улучшенной реализации
87
+ print("\n=== Улучшенная обработка 'null' ===")
88
+ print(f"Параметры: {original_params}")
89
+ print("Результат validate_params (улучшенный):")
90
+ try:
91
+ print(improved_validate_params(original_params))
92
+ except Exception as e:
93
+ print(f"ОШИБКА: {e}")
94
+
95
+ # Проверка запроса к VSTL с null в параметрах
96
+ print("\n=== Тестирование запроса к VSTL ===")
97
+
98
+ # Тест с null в параметрах
99
+ print("\nТест 1: Запрос с null в параметрах")
100
+ response = call_vstl_help({"unknown_param": None})
101
+ print(f"Ответ: {json.dumps(response, indent=2)}")
102
+
103
+ # Тест с строковым "null" в параметрах
104
+ print("\nТест 2: Запрос со строковым 'null' в параметрах")
105
+ response = call_vstl_help({"unknown_param": "null"})
106
+ print(f"Ответ: {json.dumps(response, indent=2)}")
107
+
108
+ # Рекомендации по исправлению
109
+ print("\n=== Рекомендации по исправлению ===")
110
+ print("""
111
+ 1. Обновите метод validate_params в файле commands/base.py на улучшенную версию:
112
+ - Добавьте проверку на строки "null" и "none" (в любом регистре)
113
+ - Для параметров, которые могут быть None, преобразуйте их в Python None
114
+
115
+ 2. Обновите метод execute команды help, чтобы игнорировать неизвестные параметры:
116
+ - Добавьте аргумент **kwargs в сигнатуру метода
117
+
118
+ Эти изменения улучшат совместимость с клиентами, отправляющими JavaScript null
119
+ и сделают API более устойчивым к различным форматам входных данных.
120
+ """)
121
+
122
+ if __name__ == "__main__":
123
+ test_validate_params_fix()
@@ -0,0 +1,177 @@
1
+ #!/usr/bin/env python3
2
+ """
3
+ Скрипт для проверки работы улучшенной обработки null в версии 3.1.4.
4
+
5
+ Этот скрипт:
6
+ 1. Проверяет, что версия пакета 3.1.4
7
+ 2. Тестирует улучшенный метод validate_params
8
+ 3. Проверяет, что команда help правильно обрабатывает null значения
9
+ """
10
+
11
+ import inspect
12
+ import sys
13
+ from typing import Dict, Any, Optional
14
+
15
+ try:
16
+ # Импортируем необходимые модули из пакета
17
+ from mcp_proxy_adapter import __version__
18
+ from mcp_proxy_adapter.commands.base import Command
19
+ from mcp_proxy_adapter.commands.help_command import HelpCommand, HelpResult
20
+ except ImportError:
21
+ print("ОШИБКА: Не удалось импортировать mcp_proxy_adapter. Убедитесь, что пакет установлен.")
22
+ sys.exit(1)
23
+
24
+
25
+ def check_version():
26
+ """
27
+ Проверяет версию установленного пакета mcp_proxy_adapter.
28
+ """
29
+ expected_version = "3.1.4"
30
+
31
+ print(f"\n=== Проверка версии ===")
32
+ print(f"Установленная версия: {__version__}")
33
+ print(f"Ожидаемая версия: {expected_version}")
34
+
35
+ if __version__ != expected_version:
36
+ print(f"ОШИБКА: Версия пакета ({__version__}) не соответствует ожидаемой ({expected_version}).")
37
+ return False
38
+
39
+ print("OK: Версия соответствует ожидаемой.")
40
+ return True
41
+
42
+
43
+ def test_validate_params():
44
+ """
45
+ Тестирует метод validate_params класса Command для обработки различных вариантов null.
46
+ """
47
+ print(f"\n=== Тестирование validate_params ===")
48
+
49
+ # Получаем исходный код метода validate_params
50
+ validate_params_source = inspect.getsource(Command.validate_params)
51
+ print(f"Исходный код метода validate_params:")
52
+ print(validate_params_source)
53
+
54
+ # Проверяем, содержит ли код улучшения для обработки null
55
+ required_improvements = [
56
+ "value.lower() in",
57
+ "null",
58
+ "none",
59
+ "validated_params[key] = None"
60
+ ]
61
+
62
+ for improvement in required_improvements:
63
+ if improvement not in validate_params_source:
64
+ print(f"ОШИБКА: Не найдено улучшение: {improvement}")
65
+ return False
66
+
67
+ # Тестируем различные значения null
68
+ test_cases = [
69
+ {"unknown_param": None},
70
+ {"unknown_param": "null"},
71
+ {"unknown_param": "NULL"},
72
+ {"unknown_param": "Null"},
73
+ {"unknown_param": "none"},
74
+ {"unknown_param": "NONE"},
75
+ {"cmdname": None},
76
+ {"cmdname": "null"},
77
+ {"cmdname": "none"},
78
+ {"cmdname": ""}
79
+ ]
80
+
81
+ print("\nРезультаты тестирования:")
82
+ for case in test_cases:
83
+ result = Command.validate_params(case)
84
+ print(f"Входные данные: {case}")
85
+ print(f"Результат: {result}")
86
+
87
+ # Проверяем специальный случай с параметром cmdname
88
+ if "cmdname" in case and (case["cmdname"] is None or case["cmdname"].lower() in ["null", "none", ""]):
89
+ if "cmdname" not in result or result["cmdname"] is not None:
90
+ print(f"ОШИБКА: Неверная обработка cmdname. Ожидалось: {{'cmdname': None}}, Получено: {result}")
91
+ return False
92
+
93
+ # Проверяем, что unknown_param был удален
94
+ if "unknown_param" in case and "unknown_param" in result:
95
+ print(f"ОШИБКА: Параметр unknown_param не был удален: {result}")
96
+ return False
97
+
98
+ print("OK: Все тесты validate_params пройдены успешно.")
99
+ return True
100
+
101
+
102
+ def test_help_command():
103
+ """
104
+ Тестирует работу команды help с различными вариантами null.
105
+ """
106
+ print(f"\n=== Тестирование команды help ===")
107
+
108
+ # Получаем исходный код метода execute команды help
109
+ help_execute_source = inspect.getsource(HelpCommand.execute)
110
+ print(f"Исходный код метода execute в HelpCommand:")
111
+ print(help_execute_source)
112
+
113
+ # Проверяем, содержит ли код поддержку **kwargs
114
+ if "**kwargs" not in help_execute_source:
115
+ print("ОШИБКА: Метод execute в HelpCommand не поддерживает **kwargs.")
116
+ return False
117
+
118
+ # Создаем экземпляр команды
119
+ help_cmd = HelpCommand()
120
+
121
+ # Тестируем с разными вариантами null в параметрах
122
+ test_cases = [
123
+ {"cmdname": None, "unknown_param": None},
124
+ {"cmdname": "null", "unknown_param": "null"},
125
+ {"cmdname": "none", "unknown_param": "none"},
126
+ {"cmdname": "", "unknown_param": ""},
127
+ {"unknown_param": "whatever"}
128
+ ]
129
+
130
+ print("\nРезультаты тестирования:")
131
+ for i, case in enumerate(test_cases):
132
+ print(f"\nТест {i+1}: {case}")
133
+ try:
134
+ # Используем execute напрямую, так как run преобразует параметры
135
+ result = help_cmd.execute(**case)
136
+ print(f"Тест пройден успешно, команда вернула результат типа: {type(result)}")
137
+ except Exception as e:
138
+ print(f"ОШИБКА: Тест не пройден: {e}")
139
+ return False
140
+
141
+ print("OK: Все тесты команды help пройдены успешно.")
142
+ return True
143
+
144
+
145
+ def main():
146
+ """
147
+ Основная функция для запуска тестов.
148
+ """
149
+ print("=== Проверка MCP Proxy Adapter 3.1.4 ===")
150
+
151
+ # Проверяем версию пакета
152
+ version_ok = check_version()
153
+
154
+ # Если версия не соответствует ожидаемой, прекращаем выполнение
155
+ if not version_ok:
156
+ print("ОШИБКА: Версия пакета не соответствует требуемой.")
157
+ return False
158
+
159
+ # Тестируем validate_params
160
+ validate_params_ok = test_validate_params()
161
+
162
+ # Тестируем команду help
163
+ help_command_ok = test_help_command()
164
+
165
+ # Выводим общий результат
166
+ print("\n=== Итоговый результат ===")
167
+ if version_ok and validate_params_ok and help_command_ok:
168
+ print("УСПЕХ: Все тесты пройдены успешно!")
169
+ return True
170
+ else:
171
+ print("ОШИБКА: Не все тесты пройдены.")
172
+ return False
173
+
174
+
175
+ if __name__ == "__main__":
176
+ success = main()
177
+ sys.exit(0 if success else 1)
@@ -91,10 +91,11 @@ class Command(ABC):
91
91
  # Handle None values and empty strings in parameters
92
92
  for key, value in list(validated_params.items()):
93
93
  # Process None values or empty strings - this helps with JavaScript null/undefined conversions
94
- if value is None or (isinstance(value, str) and value == ""):
95
- # For commands that specifically handle None values (like help), keep the parameter
94
+ if value is None or (isinstance(value, str) and value.lower() in ["null", "none", ""]):
95
+ # For commands that specifically handle None values, keep the parameter
96
+ # (like help), keep the parameter but ensure it's a proper Python None
96
97
  if key in ["cmdname"]: # список параметров, для которых None является допустимым значением
97
- pass
98
+ validated_params[key] = None
98
99
  else:
99
100
  # For most parameters, remove None values to avoid issues
100
101
  del validated_params[key]
@@ -1,3 +1,3 @@
1
1
  """Version information for MCP Microservice."""
2
2
 
3
- __version__ = "3.1.3"
3
+ __version__ = "3.1.4"
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: mcp-proxy-adapter
3
- Version: 3.1.3
3
+ Version: 3.1.4
4
4
  Summary: Reliable microservice with unified JSON-RPC endpoint
5
5
  Home-page: https://github.com/yourusername/mcp-proxy-adapter
6
6
  Author: MCP Team
@@ -1,6 +1,8 @@
1
1
  examples/__init__.py,sha256=sLYNpeoiE-X5q7fmJb7NFMmhiIn0543mgJj16q1qmk0,593
2
+ examples/fix_vstl_help.py,sha256=GvFepKbCD-a2O2LEflMGeXnsgNMUNfNezPFU6QB_7tI,5208
2
3
  examples/server.py,sha256=gnRTE_k7C0A255dLyaJWyA4YU0H6Elc7osr_JQvsQhQ,2286
3
4
  examples/simple_server.py,sha256=Bkczmz5Qs473xJ0_AJjBpqWT-oWctwED98A067z05zQ,3768
5
+ examples/test_package_3.1.4.py,sha256=OnEC4N4QOfqrh7DPsQrUV9L9TEG240CwkdM9NY2HBXU,7180
4
6
  examples/test_server.py,sha256=cKWJ4tlHqZsRKyeuXbZ1dQ7TU9riJWcDan__wK7YH_Y,3729
5
7
  examples/tool_description_example.py,sha256=blamrx_1oHCG4NnvIiYnQxphAEDqb7-TALPALJFj51s,3280
6
8
  examples/anti_patterns/README.md,sha256=1-Hby6Wf3kAC0XOV_jOvuHL-kmTypWOUfE_rEU3Knu8,2045
@@ -50,7 +52,7 @@ mcp_proxy_adapter/config.py,sha256=MjgZAld6TiD0F5oCyEaJhYhfEXVZxc5G5ke2SLKCV9A,5
50
52
  mcp_proxy_adapter/custom_openapi.py,sha256=tAE289B76nUdd2tjbiyow2Jftj0Yd-A8I2ndTD6R-5c,11706
51
53
  mcp_proxy_adapter/openapi.py,sha256=jyl5EPXcFhzFKEEMXxHeqF1U-SsYvtdlaKGU2QrekpU,13889
52
54
  mcp_proxy_adapter/py.typed,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
53
- mcp_proxy_adapter/version.py,sha256=0qNMWytNAXlNNfpN1BZMoMhsK8wp4zfWXOOFOViwLZU,71
55
+ mcp_proxy_adapter/version.py,sha256=WXry8RiuhtfzpY1KvYxY0nYxtli-yU85Nax9ZjK3axY,71
54
56
  mcp_proxy_adapter/api/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
55
57
  mcp_proxy_adapter/api/app.py,sha256=q3yYsNEBFXYr2TOJFxfh6bbF53NkmmgfzSFOtVq7xdc,14353
56
58
  mcp_proxy_adapter/api/handlers.py,sha256=lc_4eakQgQVlnGjnVkOY-mIMkfyLk4iRfwdrWTyvuvM,7194
@@ -65,7 +67,7 @@ mcp_proxy_adapter/api/middleware/logging.py,sha256=wGtw4BqKMLgn5zqYd84DnVPtO3evf
65
67
  mcp_proxy_adapter/api/middleware/performance.py,sha256=dHBxTF43LEGXMKHMH3A8ybKmwAWURd_zswqq_oC4xbw,2454
66
68
  mcp_proxy_adapter/api/middleware/rate_limit.py,sha256=DIv_-ZUVmL-jEo_A5BlfnasZf25zT84AiIJDUUnXkpM,5041
67
69
  mcp_proxy_adapter/commands/__init__.py,sha256=bHZZcVYkXVL9g-YZOnWkHOxSP2WzT-I4_OleYycQhbw,610
68
- mcp_proxy_adapter/commands/base.py,sha256=vgFhH5u3GwNs0c5FvJUf2csahzTyH5fKnYLWiqKix9o,13615
70
+ mcp_proxy_adapter/commands/base.py,sha256=lKKoN_9tJYIeOFKgQRGwWZHy_EvWP8bVB1EhIouTbi0,13740
69
71
  mcp_proxy_adapter/commands/command_registry.py,sha256=x55k4guzOlm7fVRolJ_iSFPRx2Snq4yBaMQdzVmsK0A,10460
70
72
  mcp_proxy_adapter/commands/config_command.py,sha256=-Z6BGaEQTf859l56zZpHYBeZFeIVdpMYybDrd7LOPIg,3553
71
73
  mcp_proxy_adapter/commands/dependency_container.py,sha256=Uz9OPRAUZN7tsVrMVgXgPQcsRD2N-e2Ixg9XarPOlnY,3410
@@ -106,8 +108,8 @@ mcp_proxy_adapter/tests/stubs/echo_command.py,sha256=Y7SA4IB5Lo_ncn78SDm9iRZvJSK
106
108
  mcp_proxy_adapter/tests/unit/__init__.py,sha256=RS-5UoSCcLKtr2jrAaZw_NG9MquA6BZmxq-P6cTw9ok,53
107
109
  mcp_proxy_adapter/tests/unit/test_base_command.py,sha256=ldDXQYk2eijbTgZioSBAhHzSAa_SuBKYqCutCEzUYTE,3924
108
110
  mcp_proxy_adapter/tests/unit/test_config.py,sha256=SZ62LXFOv_fsV0fmSIBdHWvapEyexKrioFRQo0I4pkg,5900
109
- mcp_proxy_adapter-3.1.3.dist-info/licenses/LICENSE,sha256=OkApFEwdgMCt_mbvUI-eIwKMSTe38K3XnU2DT5ub-wI,1072
110
- mcp_proxy_adapter-3.1.3.dist-info/METADATA,sha256=kRq0QbeD-BFX5Wz4BgflzuwOojpACvvBl3t85YZl8UU,7537
111
- mcp_proxy_adapter-3.1.3.dist-info/WHEEL,sha256=DnLRTWE75wApRYVsjgc6wsVswC54sMSJhAEd4xhDpBk,91
112
- mcp_proxy_adapter-3.1.3.dist-info/top_level.txt,sha256=kxq3OC7vBtsFdy9dDVse4cOl-SV_QlvcTeSkuw_jw3I,27
113
- mcp_proxy_adapter-3.1.3.dist-info/RECORD,,
111
+ mcp_proxy_adapter-3.1.4.dist-info/licenses/LICENSE,sha256=OkApFEwdgMCt_mbvUI-eIwKMSTe38K3XnU2DT5ub-wI,1072
112
+ mcp_proxy_adapter-3.1.4.dist-info/METADATA,sha256=G_zv6VyoWPW8P9AStPqByO-7ujKlx8JMgv8Yx_cxqOY,7537
113
+ mcp_proxy_adapter-3.1.4.dist-info/WHEEL,sha256=DnLRTWE75wApRYVsjgc6wsVswC54sMSJhAEd4xhDpBk,91
114
+ mcp_proxy_adapter-3.1.4.dist-info/top_level.txt,sha256=kxq3OC7vBtsFdy9dDVse4cOl-SV_QlvcTeSkuw_jw3I,27
115
+ mcp_proxy_adapter-3.1.4.dist-info/RECORD,,