solver-sdk 1.7.4 → 1.7.6
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/README.md +25 -49
- package/dist/cjs/api/{chat-api.js → chat-api/index.js} +32 -192
- package/dist/cjs/api/chat-api/index.js.map +1 -0
- package/dist/cjs/api/chat-api/interfaces.js +3 -0
- package/dist/cjs/api/chat-api/interfaces.js.map +1 -0
- package/dist/cjs/api/chat-api/models.js +6 -0
- package/dist/cjs/api/chat-api/models.js.map +1 -0
- package/dist/cjs/api/chat-api/stream-utils.js +192 -0
- package/dist/cjs/api/chat-api/stream-utils.js.map +1 -0
- package/dist/cjs/api/chat-api/websocket-helpers.js +211 -0
- package/dist/cjs/api/chat-api/websocket-helpers.js.map +1 -0
- package/dist/cjs/api/projects-api.js +211 -0
- package/dist/cjs/api/projects-api.js.map +1 -1
- package/dist/cjs/code-solver-sdk.js +10 -15
- package/dist/cjs/code-solver-sdk.js.map +1 -1
- package/dist/cjs/constants/websocket-events.constants.js +77 -52
- package/dist/cjs/constants/websocket-events.constants.js.map +1 -1
- package/dist/cjs/utils/code-solver-websocket-client.js.map +1 -1
- package/dist/esm/api/{chat-api.js → chat-api/index.js} +29 -192
- package/dist/esm/api/chat-api/index.js.map +1 -0
- package/dist/esm/api/chat-api/interfaces.js +2 -0
- package/dist/esm/api/chat-api/interfaces.js.map +1 -0
- package/dist/esm/api/chat-api/models.js +5 -0
- package/dist/esm/api/chat-api/models.js.map +1 -0
- package/dist/esm/api/chat-api/stream-utils.js +188 -0
- package/dist/esm/api/chat-api/stream-utils.js.map +1 -0
- package/dist/esm/api/chat-api/websocket-helpers.js +205 -0
- package/dist/esm/api/chat-api/websocket-helpers.js.map +1 -0
- package/dist/esm/api/projects-api.js +211 -0
- package/dist/esm/api/projects-api.js.map +1 -1
- package/dist/esm/code-solver-sdk.js +8 -13
- package/dist/esm/code-solver-sdk.js.map +1 -1
- package/dist/esm/constants/websocket-events.constants.js +76 -51
- package/dist/esm/constants/websocket-events.constants.js.map +1 -1
- package/dist/esm/utils/code-solver-websocket-client.js.map +1 -1
- package/dist/types/api/chat-api/index.d.ts +81 -0
- package/dist/types/api/chat-api/index.d.ts.map +1 -0
- package/dist/types/api/chat-api/interfaces.d.ts +47 -0
- package/dist/types/api/chat-api/interfaces.d.ts.map +1 -0
- package/dist/types/api/{chat-api.d.ts → chat-api/models.d.ts} +8 -79
- package/dist/types/api/chat-api/models.d.ts.map +1 -0
- package/dist/types/api/chat-api/stream-utils.d.ts +31 -0
- package/dist/types/api/chat-api/stream-utils.d.ts.map +1 -0
- package/dist/types/api/chat-api/websocket-helpers.d.ts +40 -0
- package/dist/types/api/chat-api/websocket-helpers.d.ts.map +1 -0
- package/dist/types/api/projects-api.d.ts +76 -0
- package/dist/types/api/projects-api.d.ts.map +1 -1
- package/dist/types/code-solver-sdk.d.ts +1 -1
- package/dist/types/code-solver-sdk.d.ts.map +1 -1
- package/dist/types/constants/websocket-events.constants.d.ts +65 -36
- package/dist/types/constants/websocket-events.constants.d.ts.map +1 -1
- package/dist/types/types/index.d.ts +8 -0
- package/dist/types/types/index.d.ts.map +1 -1
- package/dist/types/utils/code-solver-websocket-client.d.ts +28 -34
- package/dist/types/utils/code-solver-websocket-client.d.ts.map +1 -1
- package/docs/API_REFERENCE.md +432 -0
- package/docs/INTEGRATION_EXAMPLES.md +342 -0
- package/docs/README.md +78 -53
- package/docs/WEBSOCKET.md +191 -400
- package/docs/advanced/PING_PONG.md +212 -0
- package/docs/features/THINKING.md +158 -0
- package/docs/indexing/INDEXING.md +434 -210
- package/package.json +2 -2
- package/dist/cjs/api/chat-api.js.map +0 -1
- package/dist/esm/api/chat-api.js.map +0 -1
- package/dist/types/api/chat-api.d.ts.map +0 -1
- package/docs/REGIONS.md +0 -140
- package/docs/WEBSOCKET_EVENTS.md +0 -183
- package/docs/thinking/THINKING_ARCHITECTURE.md +0 -221
- package/docs/thinking/streaming-thinking-guide.md +0 -164
- package/docs/thinking/thinking-mode.md +0 -366
|
@@ -0,0 +1,212 @@
|
|
|
1
|
+
# WebSocket Ping/Pong механизм
|
|
2
|
+
|
|
3
|
+
Данный документ описывает механизм поддержания WebSocket соединений и мониторинга их состояния через ping/pong обмен.
|
|
4
|
+
|
|
5
|
+
## Обзор функциональности
|
|
6
|
+
|
|
7
|
+
SDK поддерживает автоматический механизм отправки ping-сообщений и обработки pong-ответов для:
|
|
8
|
+
|
|
9
|
+
1. **Проверки активности соединения** - обнаружение разрывов соединения даже в отсутствие активности
|
|
10
|
+
2. **Измерения времени отклика (Round Trip Time, RTT)** - мониторинг качества соединения
|
|
11
|
+
3. **Сбора статистики соединения** - для диагностики и отладки
|
|
12
|
+
4. **Автоматического обнаружения проблем** - уведомление о потере соединения
|
|
13
|
+
|
|
14
|
+
## Включение механизма ping/pong
|
|
15
|
+
|
|
16
|
+
```typescript
|
|
17
|
+
import { CodeSolverSDK, WebSocketNamespace } from 'solver-sdk';
|
|
18
|
+
|
|
19
|
+
// Создаем экземпляр SDK
|
|
20
|
+
const sdk = new CodeSolverSDK({
|
|
21
|
+
baseURL: 'https://api.example.com',
|
|
22
|
+
apiKey: 'your-api-key'
|
|
23
|
+
});
|
|
24
|
+
|
|
25
|
+
// Получаем WebSocket клиент
|
|
26
|
+
const wsClient = sdk.getWebSocketClient();
|
|
27
|
+
|
|
28
|
+
// Подключаемся к пространствам имен
|
|
29
|
+
await wsClient.connectToReasoning();
|
|
30
|
+
await wsClient.connectToDependencies();
|
|
31
|
+
await wsClient.connectToIndexing();
|
|
32
|
+
|
|
33
|
+
// Включаем автоматический механизм ping/pong
|
|
34
|
+
// параметры: интервал отправки в мс (по умолчанию 30000) и порог таймаута (по умолчанию 3)
|
|
35
|
+
wsClient.enablePingPong(10000, 3);
|
|
36
|
+
|
|
37
|
+
// Регистрируем обработчик для события таймаута соединения
|
|
38
|
+
wsClient.onPingPongEvent('connection_timeout', (data) => {
|
|
39
|
+
console.log(`Соединение потеряно для namespace ${data.namespace}`);
|
|
40
|
+
console.log(`Socket ID: ${data.socketId}`);
|
|
41
|
+
console.log(`Количество пропущенных pong: ${data.timeouts}`);
|
|
42
|
+
|
|
43
|
+
// Здесь можно добавить логику переподключения или уведомления пользователя
|
|
44
|
+
});
|
|
45
|
+
```
|
|
46
|
+
|
|
47
|
+
## Отключение механизма ping/pong
|
|
48
|
+
|
|
49
|
+
```typescript
|
|
50
|
+
// Отключение для всех пространств имен
|
|
51
|
+
wsClient.disablePingPong();
|
|
52
|
+
|
|
53
|
+
// Отключение для конкретного пространства имен
|
|
54
|
+
wsClient.disablePingPong(WebSocketNamespace.REASONING);
|
|
55
|
+
```
|
|
56
|
+
|
|
57
|
+
## Получение статистики ping/pong
|
|
58
|
+
|
|
59
|
+
```typescript
|
|
60
|
+
// Получение статистики для всех пространств имен
|
|
61
|
+
const allStats = wsClient.getPingStats();
|
|
62
|
+
console.log('Статистика для всех соединений:', allStats);
|
|
63
|
+
|
|
64
|
+
// Получение статистики для конкретного пространства имен
|
|
65
|
+
const reasoningStats = wsClient.getPingStats(WebSocketNamespace.REASONING);
|
|
66
|
+
console.log('Статистика для reasoning:', reasoningStats);
|
|
67
|
+
|
|
68
|
+
// Пример содержимого статистики:
|
|
69
|
+
// {
|
|
70
|
+
// namespace: '/reasoning',
|
|
71
|
+
// socketId: 'socket-id-123',
|
|
72
|
+
// pingSent: 10, // Количество отправленных ping
|
|
73
|
+
// pongReceived: 10, // Количество полученных pong
|
|
74
|
+
// averageRtt: 15.5, // Среднее время отклика (мс)
|
|
75
|
+
// minRtt: 5, // Минимальное время отклика (мс)
|
|
76
|
+
// maxRtt: 45, // Максимальное время отклика (мс)
|
|
77
|
+
// lastRtt: 12, // Последнее измеренное время отклика (мс)
|
|
78
|
+
// lastPongTimestamp: 1712345678901, // Timestamp последнего полученного pong
|
|
79
|
+
// isConnected: true // Текущий статус соединения
|
|
80
|
+
// }
|
|
81
|
+
```
|
|
82
|
+
|
|
83
|
+
## Ручная отправка ping/pong
|
|
84
|
+
|
|
85
|
+
Хотя SDK обеспечивает автоматический механизм ping/pong, вы также можете вручную отправлять ping-сообщения:
|
|
86
|
+
|
|
87
|
+
```typescript
|
|
88
|
+
// Отправка ping и получение pong
|
|
89
|
+
wsClient.send(WebSocketNamespace.REASONING, 'connection_ping', {
|
|
90
|
+
timestamp: Date.now()
|
|
91
|
+
});
|
|
92
|
+
|
|
93
|
+
// Добавление обработчика для pong-ответов
|
|
94
|
+
wsClient.on('connection_pong', (data) => {
|
|
95
|
+
const rtt = Date.now() - data.echo;
|
|
96
|
+
console.log(`Получен pong, RTT: ${rtt}ms`);
|
|
97
|
+
});
|
|
98
|
+
|
|
99
|
+
// Альтернативно, можно использовать готовый обработчик
|
|
100
|
+
wsClient.on('connection_pong', wsClient.getPongHandler());
|
|
101
|
+
```
|
|
102
|
+
|
|
103
|
+
## Обработка отключений и переподключение
|
|
104
|
+
|
|
105
|
+
```typescript
|
|
106
|
+
// Обработка события таймаута соединения
|
|
107
|
+
wsClient.onPingPongEvent('connection_timeout', async (data) => {
|
|
108
|
+
console.log(`Соединение потеряно для ${data.namespace}`);
|
|
109
|
+
|
|
110
|
+
// Попытка переподключения
|
|
111
|
+
try {
|
|
112
|
+
// Отключаемся от проблемного пространства имен
|
|
113
|
+
wsClient.disconnect(data.namespace);
|
|
114
|
+
|
|
115
|
+
// Пауза перед повторным подключением
|
|
116
|
+
await new Promise(resolve => setTimeout(resolve, 1000));
|
|
117
|
+
|
|
118
|
+
// Переподключаемся
|
|
119
|
+
switch (data.namespace) {
|
|
120
|
+
case WebSocketNamespace.REASONING:
|
|
121
|
+
await wsClient.connectToReasoning();
|
|
122
|
+
break;
|
|
123
|
+
case WebSocketNamespace.DEPENDENCIES:
|
|
124
|
+
await wsClient.connectToDependencies();
|
|
125
|
+
break;
|
|
126
|
+
case WebSocketNamespace.INDEXING:
|
|
127
|
+
await wsClient.connectToIndexing();
|
|
128
|
+
break;
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
console.log(`Успешно переподключились к ${data.namespace}`);
|
|
132
|
+
} catch (error) {
|
|
133
|
+
console.error(`Ошибка при переподключении к ${data.namespace}:`, error);
|
|
134
|
+
}
|
|
135
|
+
});
|
|
136
|
+
```
|
|
137
|
+
|
|
138
|
+
## Рекомендации по использованию
|
|
139
|
+
|
|
140
|
+
1. **Интервал ping/pong**: Рекомендуемый интервал - 30 секунд для продакшн и 5-10 секунд для отладки.
|
|
141
|
+
|
|
142
|
+
2. **Порог таймаута**: Значение 2-3 пропущенных ping/pong позволяет избежать ложных срабатываний при временных задержках сети.
|
|
143
|
+
|
|
144
|
+
3. **Экономия трафика**: При частой передаче данных через WebSocket можно увеличить интервал ping/pong, так как регулярный обмен данными уже поддерживает соединение активным.
|
|
145
|
+
|
|
146
|
+
4. **Мониторинг RTT**: Используйте statsPingStats для мониторинга задержек сети и раннего обнаружения проблем с соединением.
|
|
147
|
+
|
|
148
|
+
5. **Отключение перед закрытием приложения**: Вызывайте disablePingPong() и затем disconnectAll() при закрытии приложения для корректного освобождения ресурсов.
|
|
149
|
+
|
|
150
|
+
## Пример комплексного использования
|
|
151
|
+
|
|
152
|
+
```typescript
|
|
153
|
+
async function setupWebSocketWithHealthMonitoring() {
|
|
154
|
+
const sdk = new CodeSolverSDK({
|
|
155
|
+
baseURL: 'https://api.example.com',
|
|
156
|
+
apiKey: 'your-api-key',
|
|
157
|
+
websocket: {
|
|
158
|
+
reconnect: true,
|
|
159
|
+
reconnectAttempts: 5,
|
|
160
|
+
reconnectDelay: 2000
|
|
161
|
+
}
|
|
162
|
+
});
|
|
163
|
+
|
|
164
|
+
const wsClient = sdk.getWebSocketClient();
|
|
165
|
+
|
|
166
|
+
// Подключаемся ко всем пространствам имен
|
|
167
|
+
await wsClient.connectToReasoning();
|
|
168
|
+
await wsClient.connectToDependencies();
|
|
169
|
+
await wsClient.connectToIndexing();
|
|
170
|
+
|
|
171
|
+
// Включаем механизм ping/pong для всех соединений
|
|
172
|
+
wsClient.enablePingPong(20000, 3);
|
|
173
|
+
|
|
174
|
+
// Регистрируем обработчик для события таймаута
|
|
175
|
+
wsClient.onPingPongEvent('connection_timeout', handleConnectionTimeout);
|
|
176
|
+
|
|
177
|
+
// Периодически проверяем статистику соединений
|
|
178
|
+
const statsInterval = setInterval(() => {
|
|
179
|
+
const stats = wsClient.getPingStats();
|
|
180
|
+
|
|
181
|
+
// Анализируем статистику
|
|
182
|
+
for (const stat of stats) {
|
|
183
|
+
if (stat.averageRtt > 500) {
|
|
184
|
+
console.warn(`Высокая задержка для ${stat.namespace}: ${stat.averageRtt}ms`);
|
|
185
|
+
}
|
|
186
|
+
}
|
|
187
|
+
}, 60000); // Проверка каждую минуту
|
|
188
|
+
|
|
189
|
+
// Функция для обработки таймаута соединения
|
|
190
|
+
async function handleConnectionTimeout(data) {
|
|
191
|
+
console.error(`Соединение потеряно для ${data.namespace}`);
|
|
192
|
+
|
|
193
|
+
// Логика переподключения...
|
|
194
|
+
}
|
|
195
|
+
|
|
196
|
+
// Функция для корректного закрытия соединений
|
|
197
|
+
function cleanup() {
|
|
198
|
+
clearInterval(statsInterval);
|
|
199
|
+
wsClient.disablePingPong();
|
|
200
|
+
wsClient.disconnectAll();
|
|
201
|
+
}
|
|
202
|
+
|
|
203
|
+
// Возвращаем функцию очистки
|
|
204
|
+
return cleanup;
|
|
205
|
+
}
|
|
206
|
+
|
|
207
|
+
// Использование
|
|
208
|
+
const cleanup = await setupWebSocketWithHealthMonitoring();
|
|
209
|
+
|
|
210
|
+
// При закрытии приложения
|
|
211
|
+
window.addEventListener('beforeunload', cleanup);
|
|
212
|
+
```
|
|
@@ -0,0 +1,158 @@
|
|
|
1
|
+
# Потоковая передача мышления
|
|
2
|
+
|
|
3
|
+
Метод `streamChatWithThinking()` позволяет получать не только финальный ответ модели, но и её размышления в процессе формирования ответа. Этот подход даёт более глубокое понимание логики модели и возможность видеть её работу в режиме реального времени.
|
|
4
|
+
|
|
5
|
+
## Быстрый старт
|
|
6
|
+
|
|
7
|
+
```javascript
|
|
8
|
+
const { CodeSolverSDK } = require('solver-sdk');
|
|
9
|
+
|
|
10
|
+
async function example() {
|
|
11
|
+
// Инициализация SDK
|
|
12
|
+
const sdk = new CodeSolverSDK({
|
|
13
|
+
baseURL: 'https://api.example.com',
|
|
14
|
+
apiKey: 'ваш-ключ-api'
|
|
15
|
+
});
|
|
16
|
+
|
|
17
|
+
// Сообщения для отправки
|
|
18
|
+
const messages = [
|
|
19
|
+
{ role: 'user', content: 'Объясни, как работает квантовая криптография' }
|
|
20
|
+
];
|
|
21
|
+
|
|
22
|
+
// Опции запроса
|
|
23
|
+
const options = {
|
|
24
|
+
model: 'claude-3-7-sonnet-20240229',
|
|
25
|
+
thinking: true, // Включаем передачу мышления
|
|
26
|
+
temperature: 0.7
|
|
27
|
+
};
|
|
28
|
+
|
|
29
|
+
// Обработчик потоковых данных
|
|
30
|
+
const handleStream = (eventType, data) => {
|
|
31
|
+
// Обработка мышления модели
|
|
32
|
+
if (eventType === 'content_block_delta' && data.delta?.type === 'thinking_delta') {
|
|
33
|
+
console.log('Мышление:', data.delta.thinking);
|
|
34
|
+
}
|
|
35
|
+
// Обработка текстового ответа
|
|
36
|
+
else if (eventType === 'content_block_delta' && data.delta?.type === 'text_delta') {
|
|
37
|
+
console.log('Ответ:', data.delta.text);
|
|
38
|
+
}
|
|
39
|
+
// Завершение ответа
|
|
40
|
+
else if (eventType === 'message_stop') {
|
|
41
|
+
console.log('Ответ завершён');
|
|
42
|
+
}
|
|
43
|
+
};
|
|
44
|
+
|
|
45
|
+
try {
|
|
46
|
+
// Отправляем запрос с потоковым мышлением
|
|
47
|
+
await sdk.chat.streamChatWithThinking(
|
|
48
|
+
messages,
|
|
49
|
+
options,
|
|
50
|
+
handleStream
|
|
51
|
+
);
|
|
52
|
+
} catch (error) {
|
|
53
|
+
// Обработка ошибок соединения
|
|
54
|
+
if (error.code === 'CONNECTION_ERROR') {
|
|
55
|
+
console.error('Проблема с подключением к серверу');
|
|
56
|
+
}
|
|
57
|
+
// Обработка прочих ошибок
|
|
58
|
+
else {
|
|
59
|
+
console.error('Произошла ошибка:', error.message);
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
example();
|
|
65
|
+
```
|
|
66
|
+
|
|
67
|
+
## Метод streamChatWithThinking
|
|
68
|
+
|
|
69
|
+
### Синтаксис
|
|
70
|
+
|
|
71
|
+
```javascript
|
|
72
|
+
sdk.chat.streamChatWithThinking(messages, options, callback)
|
|
73
|
+
```
|
|
74
|
+
|
|
75
|
+
### Параметры
|
|
76
|
+
|
|
77
|
+
| Параметр | Тип | Описание |
|
|
78
|
+
|----------|-----|----------|
|
|
79
|
+
| `messages` | Array | Массив сообщений для отправки модели |
|
|
80
|
+
| `options` | Object | Параметры запроса, включая модель и флаг `thinking: true` |
|
|
81
|
+
| `callback` | Function | Функция обратного вызова для обработки событий потока |
|
|
82
|
+
|
|
83
|
+
### Возвращаемое значение
|
|
84
|
+
|
|
85
|
+
Промис, который разрешается объектом с информацией о соединении:
|
|
86
|
+
|
|
87
|
+
```javascript
|
|
88
|
+
{
|
|
89
|
+
status: 'streaming',
|
|
90
|
+
socketId: 'socket_12345678',
|
|
91
|
+
provider: 'AnthropicProviderAdapter',
|
|
92
|
+
model: 'claude-3-7-sonnet-20240229'
|
|
93
|
+
}
|
|
94
|
+
```
|
|
95
|
+
|
|
96
|
+
## Примеры использования
|
|
97
|
+
|
|
98
|
+
### Обработка мышления и ответа с выводом в UI
|
|
99
|
+
|
|
100
|
+
```javascript
|
|
101
|
+
// Предполагаем наличие элементов UI для вывода мышления и ответа
|
|
102
|
+
const thinkingElement = document.getElementById('thinking-output');
|
|
103
|
+
const responseElement = document.getElementById('response-output');
|
|
104
|
+
|
|
105
|
+
// Обработчик потока
|
|
106
|
+
const handleStream = (eventType, data) => {
|
|
107
|
+
if (eventType === 'content_block_delta') {
|
|
108
|
+
// Обработка мышления
|
|
109
|
+
if (data.delta?.type === 'thinking_delta') {
|
|
110
|
+
thinkingElement.textContent += data.delta.thinking;
|
|
111
|
+
}
|
|
112
|
+
// Обработка ответа
|
|
113
|
+
else if (data.delta?.type === 'text_delta') {
|
|
114
|
+
responseElement.textContent += data.delta.text;
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
};
|
|
118
|
+
|
|
119
|
+
// Отправка запроса
|
|
120
|
+
sdk.chat.streamChatWithThinking(
|
|
121
|
+
[{ role: 'user', content: userInput }],
|
|
122
|
+
{ model: 'claude-3-7-sonnet-20240229', thinking: true },
|
|
123
|
+
handleStream
|
|
124
|
+
);
|
|
125
|
+
```
|
|
126
|
+
|
|
127
|
+
### Обработка ошибок
|
|
128
|
+
|
|
129
|
+
```javascript
|
|
130
|
+
try {
|
|
131
|
+
await sdk.chat.streamChatWithThinking(
|
|
132
|
+
messages,
|
|
133
|
+
options,
|
|
134
|
+
handleStream
|
|
135
|
+
);
|
|
136
|
+
} catch (error) {
|
|
137
|
+
// Обработка ошибок соединения
|
|
138
|
+
if (error.code === 'CONNECTION_ERROR') {
|
|
139
|
+
console.error('Проблема с подключением к серверу');
|
|
140
|
+
}
|
|
141
|
+
// Обработка прочих ошибок
|
|
142
|
+
else {
|
|
143
|
+
console.error('Произошла ошибка:', error.message);
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
```
|
|
147
|
+
|
|
148
|
+
## Рекомендации по использованию
|
|
149
|
+
|
|
150
|
+
- **Включайте режим мышления** только когда это необходимо (опция `thinking: true`), так как это увеличивает объем передаваемых данных
|
|
151
|
+
- **Используйте буферизацию** при отображении потоковых данных в UI, чтобы снизить нагрузку на рендеринг
|
|
152
|
+
- **Обрабатывайте все типы событий**, особенно `error` и `message_stop`
|
|
153
|
+
- **Учитывайте объем данных** - поток мышления может быть очень большим для сложных запросов
|
|
154
|
+
|
|
155
|
+
## Ограничения
|
|
156
|
+
|
|
157
|
+
- Не все модели поддерживают режим мышления (убедитесь, что используете совместимую модель)
|
|
158
|
+
- Требуется стабильное интернет-соединение для потоковой передачи
|