Проект

Общее

Профиль

Stage-2 » История » Версия 23

Александр Александров, 05.11.2019 01:08

1 1 Александр Александров
h1. План работы
2
3 3 Александр Александров
Работа над тестовым заданием:
4 2 Александр Александров
5 3 Александр Александров
* Схематично отразить работу программы, рассмотреть несколько вариантов работы, где должно быть отражено успех программы или неуспех. Это наглядно покажет как должна работать программа.
6
* Схематично отразить работу программы, рассмотреть несколько вариантов работы, где должны быть отражены успешные и не успешные запросы к сервису. Это наглядно покажет как должна работать программа.
7 1 Александр Александров
* Создание основной бизнес логики.
8 18 Александр Александров
* Создать структуру веб-приложения работающего как REST-сервис.
9 3 Александр Александров
* Покрытие тестами.
10 4 Александр Александров
11
----
12
13
h1. Структура данных.
14
15
h2. Входные данные
16
17
h3. Граф
18
19 11 Александр Александров
Входными данными будет граф G(V,E), где V - множество вершин графа (ноды), E - множество рёбер, соединяющих вершины. Граф не ориентированный. Сама вершина (нода) представляет собой объект, состоящий из уникального имени. Ребро представляет собой объект состоящий из двух ссылок на вершины, которые это ребро соединяет, в данном случае под ссылками подразумевается уникальное имя ноды. В формате JSON это будет выглядеть так:   
20 4 Александр Александров
21
<pre>
22 20 Александр Александров
{"nodes": [{"name":"unique_node_name"}, ...], "edges": [{"nodeOne":"unique_node_one_name", "nodeTwo":"unique_node_one_name"}, ...]}
23 4 Александр Александров
</pre>
24
25
, где
26
27 10 Александр Александров
_{"name":”unique_node_name"}_ - объект описывающий вершину графа (ноду)
28
_{"nodeOne":"unique_node_one_name", "nodeTwo":"unique_node_one_name"}_ - объект описывающий ребро, соединяющее две вершины графа.
29 4 Александр Александров
30 19 Александр Александров
*Графическое представление входных графов*
31 4 Александр Александров
32
{{dmsf_image(134)}}
33
34
Циклический граф
35
36
{{dmsf_image(131)}}
37
38
Ациклический граф
39
40
Входные графы могут содержать циклы, которые нужно исключить по условию задачи.
41 1 Александр Александров
42 19 Александр Александров
*В качестве дополнения (это не входило в основную задачу)*
43
44
Возможность работы с нодами отдельно, т.е. в граф можно добавить ноду или набор нод отдельным запросом
45
46
Входной одиночный объект, описывающий ноду
47
48
<pre>
49
{"name":"unique_node_name"}
50
</pre>
51
52
Входной набор нод
53
54
<pre>
55
[{"name":"unique_node_name"}, {"name":"unique_node_name"}, ...]
56
</pre>
57
58 20 Александр Александров
Возможность работы с рёбрами графа отдельно, т.е. в граф можно добавить ребро или набор рёбер отдельным запросом
59 19 Александр Александров
60
Входной, одиночный объект описывающий ребро графа
61
62
<pre>
63
{"nodeOne":"unique_node_one_name", "nodeTwo":"unique_node_one_name"}
64
</pre>
65
66 20 Александр Александров
Входной набор рёбер
67 19 Александр Александров
68
<pre>
69
[{"nodeOne":"unique_node_one_name", "nodeTwo":"unique_node_one_name"}, {"nodeOne":"unique_node_one_name", "nodeTwo":"unique_node_one_name"}, ...]
70
</pre>
71
72 4 Александр Александров
h3. Вершины графа (Ноды)
73 12 Александр Александров
74 4 Александр Александров
Так же на вход будут подаваться списки вершин графа для проверки связности вершин. Представляет собой простой перечень уникальных имён вершин.
75
В представлении JSON будет выглядеть так:
76
77 12 Александр Александров
<pre>
78 4 Александр Александров
["unique_node_name", "unique_node_name", ...]
79
</pre>
80 12 Александр Александров
81 4 Александр Александров
, где unique_node_name - уникальное имя вершины.
82
83 1 Александр Александров
h2. Структура выходных данных
84
85 19 Александр Александров
При запросе к сервису можно получить следующие данные:
86
87
*Граф целиком*
88
89
<pre>
90
{"nodes":[{"id":1, "name":"unique_name", "counter":0}, ...], "edges":[{"id":1, "nodeOne":"unique_node_name_one", "nodeOne":"unique_node_name_one"}, ...]}
91
</pre>
92
93
94
*Список вершин графа*
95 4 Александр Александров
96
<pre>
97 13 Александр Александров
[{"id":1, "name":"unique_name", "counter":0}, ...]
98 4 Александр Александров
</pre>
99 1 Александр Александров
100 13 Александр Александров
или объект отдельной вершины (информацию по отдельной вершине можно получить выполнив соответствующий запрос, либо по уникальному имени или по идентификатору вершины)
101
102
<pre>
103
{"id":1, "name":"unique_name", "counter":0}
104
</pre>
105
106
,где: id - уникальный идентификатор вершины, name - уникальное имя вершины, counter - счётчик посещения вершины.
107 4 Александр Александров
108 14 Александр Александров
Так же есть возможность получить информацию по рёбрам графа в следующем формате
109
110
Списком
111
112
<pre>
113
[{"id":1, "nodeOne":"unique_node_name_one", "nodeOne":"unique_node_name_one"}, ...]
114
</pre>
115
116
Отдельным объектом
117
118 21 Александр Александров
<pre>
119 14 Александр Александров
{"id":1, "nodeOne":"unique_node_name_one", "nodeOne":"unique_node_name_one"}
120 21 Александр Александров
</pre>
121 14 Александр Александров
122
,где: id - уникальный идентификатор ребра графа, nodeOne, nodeTwo - уникальное имя узла графа. Запись ребра графа хранит информацию о одной паре узлов.
123
124 4 Александр Александров
h2. Модель хранения данных
125
126
Класс *Node* - описывающий модель вершины графа (ноды). Класс состоит из следующих полей:
127 15 Александр Александров
128
*id* - уникальный идентификатор (нужен для хранения в БД). Присваивается автоматически при  записи в БД, его нельзя изменить из вне, данный параметр только на отдачу из базы.
129 4 Александр Александров
*name* - уникальное имя узла (получаем из json).
130 15 Александр Александров
*counter* - при каждом удачном проходе маршрута через ноду, счётчик ноды увеличивается автоматически, данный параметр только на отдачу из базы, его нельзя заменить из вне.
131 1 Александр Александров
132 4 Александр Александров
133 15 Александр Александров
Класс *Edge* - описывающий модель ребра графа. Ребро графа способно хранить информацию только о одной паре паре узлов. Класс состоит из следующих полей:
134
135
*id* - уникальный идентификатор записи (нужен для хранения в БД). Присваивается автоматически при  записи в БД, его нельзя изменить из вне, данный параметр только на отдачу из базы.
136
*nodeOne* - ссылка на первую ноду
137
*nodeTwo* - ссылка на вторую ноду
138 4 Александр Александров
139
h2. Модель хранения данных в БД
140
141
{{dmsf_image(135)}}
142
143 16 Александр Александров
Связь один-к-одному Одна запись ребра графа хранит две ссылки на разные ноды
144 5 Александр Александров
145 17 Александр Александров
SQL Schema (PostgreSQL notation)
146 4 Александр Александров
147 1 Александр Александров
<pre><code class="sql">
148 16 Александр Александров
DROP TABLE IF EXISTS nodes CASCADE;
149
DROP TABLE IF EXISTS edges;
150
DROP SEQUENCE IF EXISTS global_seq CASCADE;
151
152
CREATE SEQUENCE global_seq START 5000;
153
154
CREATE TABLE nodes (
155
    id INTEGER PRIMARY KEY DEFAULT nextval('global_seq'),
156
    name VARCHAR NOT NULL,
157
    counter INTEGER DEFAULT 0 NOT NULL
158 4 Александр Александров
);
159 16 Александр Александров
CREATE UNIQUE INDEX nodes_unique_name_idx ON nodes(name);
160 4 Александр Александров
161 16 Александр Александров
CREATE TABLE edges (
162
    id INTEGER PRIMARY KEY DEFAULT nextval('global_seq'),
163
    nodeone INT NOT NULL,
164
    nodetwo INT NOT NULL,
165
    FOREIGN KEY (nodeone) REFERENCES nodes(id) ON DELETE CASCADE,
166
    FOREIGN KEY (nodetwo) REFERENCES nodes(id) ON DELETE CASCADE,
167
    CHECK (nodeone <> nodetwo),
168
    CONSTRAINT unique_edge UNIQUE (nodeone, nodetwo)
169 1 Александр Александров
);
170
</code></pre>
171 22 Александр Александров
172
h2. Виды ошибок
173
174
При различных запросах к программе могут возникать различного вида ошибки. В программе ошибки перехватываются специальным обработчиком, который на выход формирует специальный объект. В данном объекте описывается вид ошибки, место её возникновения и текст сообщения ошибки.
175
176
Модель данных ошибки:
177
178
{{dmsf_image(407)}}
179
180
Выходные данные о возникшей ошибки в формате JSON
181
182
<pre>
183
{
184
  "url": "request_link",
185
  "type": "APP_ERROR[,DATA_NOT_FOUND,DATA_ERROR,VALIDATION_ERROR]",
186
  "place": "APP[,GRAPH,NODE,EDGE]",
187
  "messages": [
188
    "Any message 1",
189
    "Any message 2",
190
    ...
191
  ]
192
}
193
</pre>
194
195
Виды сообщения об ошибках:
196
197
*Обобщенные сообщения об ошибках*
198
199
* Argument must not be null - на вход вместо объекта был подан null
200
* Collection must not be empty - на вход поступила пустая коллекция данных
201
* Collection must not contain a null item - на вход поступила коллекция, в которой содержится null элемент
202
* Collection must have more than one element - в определённых случаях требуется что бы коллекция состояла как минимум из двух элементов (например при работе с рёбрами графа, где требуется указать две вершины графа, которые нужно связать)
203
* %s with ID = %d is not found - данные с указанным ID не обнаружены, где %s - может быть или NODE или EDGE, %d - номер id'шника.
204
205
*Сообщения об ошибках, возникающие при работе с рёбрами графа*
206
207
* Edge for nodes [%s, %s] is not found - ребро для указанных вершин (нод) не найдено, [%s, %s] - уникальные имена пары вершин.
208
* Edge for nodes ([%s, %s], [%s, %s]) already present in the graph - данное сообщение возникает при попытки добавить в граф уже существующее ребро. Где ([%s, %s], [%s, %s]) - подставляются уникальные имена вершин графа. Так как граф неориентированный, то например v1 и v2 <=> v2 и v1.
209
* Edges for node %s is not found - данное сообщение возникает при попытки извлечь информацию по рёбрам графа для заданной вершины, %s - уникальное имя вершины графа.
210
211
*Сообщения об ошибках, возникающие при работе с вершинами (нодами) графа*
212
213
* Node %s already present in the graph - данное сообщение возникает, при попытки добавить вершину, которая уже присутствует в графе.
214
* Error while update node with id = - сообщение возникает при неудачной попытки обновить данные по вершине графа, где id - номер идентификатора вершины.
215
* Node with NAME = %s is not found - сообщение возникает, при неудачном поиске вершины графа по её уникальному имени.
216
* Node %s is not found - сообщение возникает, при неудачном поиске вершины графа в режиме поиска по заданному объекту.
217
* Nodes %s and %s are not reachable to each other - сообщение возникает если один из узлов (вершин графа) не достижим до другого узла (т.е. имеются промежуточные узлы).
218
* Node %s is fault - сообщение возникает в случае генерации сбоя в узле (в вершине графа), т.е. узел оказался недоступным во время обхода по узлам.
219 23 Александр Александров
220
h2. REST-запросы
221
222
В программе доступны следующие виды HTTP запросов: *GET, POST, DELETE и OPTION*
223
Могут возвращаться следующие коды статуса: *200, 201, 204, 422, 500*
224
225
h3. HTTP запросы при работе с графом
226
227
*GET http://localhost:8080/rest/v1/graph* - возвращает объект графа
228
229
*Данные возвращаемые при запросе*
230
231
<pre><code class="javascript">
232
{
233
  "nodes": [
234
    {
235
      "name": "v1",
236
      "counter": 0,
237
      "id": 5000
238
    },
239
    {
240
      "name": "v2",
241
      "counter": 0,
242
      "id": 5001
243
    },
244
    {
245
      "name": "v3",
246
      "counter": 0,
247
      "id": 5002
248
    },
249
    {
250
      "name": "v4",
251
      "counter": 0,
252
      "id": 5003
253
    },
254
    {
255
      "name": "v5",
256
      "counter": 0,
257
      "id": 5004
258
    }
259
  ],
260
  "edges": [
261
    {
262
      "nodeOne": "v1",
263
      "nodeTwo": "v3",
264
      "id": 5006
265
    },
266
    {
267
      "nodeOne": "v1",
268
      "nodeTwo": "v2",
269
      "id": 5005
270
    },
271
    {
272
      "nodeOne": "v1",
273
      "nodeTwo": "v5",
274
      "id": 5007
275
    },
276
    {
277
      "nodeOne": "v3",
278
      "nodeTwo": "v4",
279
      "id": 5008
280
    }
281
  ]
282
}
283
</code></pre>
284
285
----
286
287
GET http://localhost:8080/rest/v1/graph/export - возвращает граф в формате GraphViz
288
289
*Данные возвращаемые при запросе*
290
291
<pre>
292
strict graph G {
293
v1_5000 [ label="v1" ];
294
v2_5001 [ label="v2" ];
295
v3_5002 [ label="v3" ];
296
v4_5003 [ label="v4" ];
297
v5_5004 [ label="v5" ];
298
v3_5002 -- v4_5003;
299
v1_5000 -- v2_5001;
300
v1_5000 -- v3_5002;
301
v1_5000 -- v5_5004;
302
}
303
</pre>
304
305
----
306
307
*DELETE http://localhost:8080/rest/v1/graph* - удаляет граф с БД, т.е. полностью удаляется информация с БД по узлам и рёбрам графа.
308
309
----
310
311
POST http://localhost:8080/rest/v1/graph - создаёт новый граф, при этом прежний граф полностью удаляется.
312
313
*Тело запроса*
314
315
<pre><code class="javascript">
316
{
317
    "nodes":[
318
      {"name":"v1"},
319
      {"name":"v2"},
320
      {"name":"v3"},
321
      {"name":"v4"},
322
      {"name":"v5"}
323
    ],
324
    "edges":[
325
      {"nodeOne":"v1","nodeTwo":"v2"},
326
      {"nodeOne":"v2","nodeTwo":"v3"},
327
      {"nodeOne":"v3","nodeTwo":"v4"},
328
      {"nodeOne":"v3","nodeTwo":"v5"},
329
      {"nodeOne":"v5","nodeTwo":"v4"},
330
      {"nodeOne":"v5","nodeTwo":"v2"}
331
    ]
332
}
333
</code></pre>
334
335
*Данные возвращаемые при запросе*
336
337
<pre><code class="javascript">
338
{
339
  "nodes": [
340
    {
341
      "name": "v1",
342
      "counter": 0,
343
      "id": 5013
344
    },
345
    {
346
      "name": "v3",
347
      "counter": 0,
348
      "id": 5009
349
    },
350
    {
351
      "name": "v5",
352
      "counter": 0,
353
      "id": 5011
354
    },
355
    {
356
      "name": "v2",
357
      "counter": 0,
358
      "id": 5010
359
    },
360
    {
361
      "name": "v4",
362
      "counter": 0,
363
      "id": 5012
364
    }
365
  ],
366
  "edges": [
367
    {
368
      "nodeOne": "v3",
369
      "nodeTwo": "v4",
370
      "id": 5015
371
    },
372
    {
373
      "nodeOne": "v1",
374
      "nodeTwo": "v2",
375
      "id": 5014
376
    },
377
    {
378
      "nodeOne": "v5",
379
      "nodeTwo": "v4",
380
      "id": 5016
381
    },
382
    {
383
      "nodeOne": "v5",
384
      "nodeTwo": "v2",
385
      "id": 5017
386
    }
387
  ]
388
}
389
</code></pre>
390
391
----
392
393
*POST http://localhost:8080/rest/v1/graph/checkroute* - проверка работоспособности заданной последовательности узлов.
394
395
*Тело запроса*
396
397
<pre>
398
["v1", "v2", "v3"]
399
</pre>
400
401
*Данные возвращаемые при запросе*
402
403
<pre>
404
Route for nodes [v1, v2, v3] is found
405
</pre>
406
407
h3. HTTP запросы при отдельной работе с узлами графа
408
409
POST http://localhost:8080/rest/v1/graph/nodes - добавляет объект узла в граф
410
411
*Тело запроса*
412
413
<pre>
414
{"name":"v9"}
415
</pre>
416
417
*Данные возвращаемые при запросе*
418
419
<pre><code class="javascript">
420
{
421
  "name": "v9",
422
  "counter": 0,
423
  "id": 5009
424
}
425
</code></pre>
426
427
----
428
429
POST http://localhost:8080/rest/v1/graph/nodes/byBatch - добавление набора узлов в граф
430
431
*Тело запроса*
432
433
<pre>
434
[{"name":"6"}, {"name":"v7"}, {"name":"v8"}]
435
</pre>
436
437
*Данные возвращаемые при запросе*
438
439
<pre><code class="javascript">
440
[
441
  {
442
    "name": "6",
443
    "counter": 0,
444
    "id": 5010
445
  },
446
  {
447
    "name": "v7",
448
    "counter": 0,
449
    "id": 5011
450
  },
451
  {
452
    "name": "v8",
453
    "counter": 0,
454
    "id": 5012
455
  }
456
]
457
</code></pre>
458
459
----
460
461
GET http://localhost:8080/rest/v1/graph/nodes - выводит список всех узлов в графе
462
463
*Данные возвращаемые при запросе*
464
465
<pre><code class="javascript">
466
[
467
  {
468
    "name": "6",
469
    "counter": 0,
470
    "id": 5010
471
  },
472
  {
473
    "name": "v1",
474
    "counter": 1,
475
    "id": 5000
476
  },
477
  {
478
    "name": "v2",
479
    "counter": 1,
480
    "id": 5001
481
  },
482
  {
483
    "name": "v3",
484
    "counter": 1,
485
    "id": 5002
486
  },
487
  {
488
    "name": "v9",
489
    "counter": 0,
490
    "id": 5009
491
  },
492
  {
493
    "name": "v4",
494
    "counter": 0,
495
    "id": 5003
496
  },
497
  {
498
    "name": "v5",
499
    "counter": 0,
500
    "id": 5004
501
  },
502
  {
503
    "name": "v7",
504
    "counter": 0,
505
    "id": 5011
506
  },
507
  {
508
    "name": "v8",
509
    "counter": 0,
510
    "id": 5012
511
  }
512
]
513
</code></pre>
514
515
----
516
517
GET http://localhost:8080/rest/v1/graph/nodes/byId/5000 - выводит объект узла по заданному ID
518
519
*Данные возвращаемые при запросе*
520
521
<pre><code class="javascript">
522
{
523
  "name": "v1",
524
  "counter": 1,
525
  "id": 5000
526
}
527
</code></pre>
528
529
----
530
531
GET http://localhost:8080/rest/v1/graph/nodes/byName/v1 - выводит объект узла по заданному имени
532
533
*Данные возвращаемые при запросе*
534
535
<pre><code class="javascript">
536
{
537
  "name": "v1",
538
  "counter": 1,
539
  "id": 5000
540
}
541
</code></pre>
542
543
----
544
545
*DELETE http://localhost:8080/rest/v1/graph/nodes* - удаляет все узлы и рёбра связанные с узлами из БД
546
*DELETE http://localhost:8080/rest/v1/graph/nodes/byId/5003* - удаляет узел по заданному ID
547
*DELETE http://localhost:8080/rest/v1/graph/nodes/byName/v4* - удаляет узел по заданному имени
548
549
*DELETE http://localhost:8080/rest/v1/graph/nodes/byObj* - удаляет узел по заданному объекту
550
551
*Тело запроса*
552
553
<pre>
554
{"id": 5003, "name": "v4"}
555
</pre>
556
557
h3. HTTP запросы при отдельной работе с рёбрами графа
558
559
*POST http://localhost:8080/rest/v1/graph/edges* - добавляет объект ребра в граф
560
561
*Тело запроса*
562
563
<pre>
564
{"nodeOne": "v3", "nodeTwo": "v4"}
565
</pre>
566
567
*Данные возвращаемые при запросе*
568
569
<pre><code class="javascript">
570
{
571
  "nodeOne": "v6",
572
  "nodeTwo": "v7",
573
  "id": 5014
574
}
575
</code></pre>
576
577
----
578
579
*POST http://localhost:8080/rest/v1/graph/edges/byBatch* - добавляет список рёбер графа
580
581
*Тело запроса*
582
583
<pre>
584
[{"nodeOne": "v2", "nodeTwo": "v7"}, {"nodeOne": "v7", "nodeTwo": "v8"}, {"nodeOne": "v8", "nodeTwo": "v9"}]
585
</pre>
586
587
*Данные возвращаемые при запросе*
588
589
<pre><code class="javascript">
590
[
591
  {
592
    "nodeOne": "v2",
593
    "nodeTwo": "v7",
594
    "id": 5015
595
  },
596
  {
597
    "nodeOne": "v8",
598
    "nodeTwo": "v9",
599
    "id": 5017
600
  },
601
  {
602
    "nodeOne": "v7",
603
    "nodeTwo": "v8",
604
    "id": 5016
605
  }
606
]
607
</code></pre>
608
609
----
610
611
*GET http://localhost:8080/rest/v1/graph/edges* - возвращает список рёбер графа
612
613
*Данные возвращаемые при запросе*
614
615
<pre><code class="javascript">
616
[
617
  {
618
    "nodeOne": "v2",
619
    "nodeTwo": "v7",
620
    "id": 5015
621
  },
622
  {
623
    "nodeOne": "v8",
624
    "nodeTwo": "v9",
625
    "id": 5017
626
  },
627
  {
628
    "nodeOne": "v7",
629
    "nodeTwo": "v8",
630
    "id": 5016
631
  },
632
  {
633
    "nodeOne": "v1",
634
    "nodeTwo": "v3",
635
    "id": 5006
636
  },
637
  {
638
    "nodeOne": "v1",
639
    "nodeTwo": "v2",
640
    "id": 5005
641
  },
642
  {
643
    "nodeOne": "v1",
644
    "nodeTwo": "v5",
645
    "id": 5007
646
  },
647
  {
648
    "nodeOne": "v3",
649
    "nodeTwo": "v4",
650
    "id": 5008
651
  },
652
  {
653
    "nodeOne": "v6",
654
    "nodeTwo": "v7",
655
    "id": 5014
656
  }
657
]
658
</code></pre>
659
660
----
661
662
*GET http://localhost:8080/rest/v1/graph/edges/byId/5007* - возвращает ребро графа по заданному id
663
664
*Данные возвращаемые при запросе*
665
666
<pre><code class="javascript">
667
{
668
  "nodeOne": "v1",
669
  "nodeTwo": "v5",
670
  "id": 5007
671
}
672
</code></pre>
673
674
----
675
676
*GET http://localhost:8080/rest/v1/graph/edges/byName/v1* - возвращает список рёбер связанный с заданным узлом, при запросе задаётся уникальное имя узла
677
678
*Данные возвращаемые при запросе*
679
680
<pre><code class="javascript">
681
[
682
  {
683
    "nodeOne": "v1",
684
    "nodeTwo": "v3",
685
    "id": 5006
686
  },
687
  {
688
    "nodeOne": "v1",
689
    "nodeTwo": "v2",
690
    "id": 5005
691
  },
692
  {
693
    "nodeOne": "v1",
694
    "nodeTwo": "v5",
695
    "id": 5007
696
  }
697
]
698
</code></pre>
699
700
----
701
702
*GET http://localhost:8080/rest/v1/graph/edges/byName?nodeOne=v2&nodeTwo=v5* - выводит объект ребра по заданным узлам графа
703
704
*Данные возвращаемые при запросе*
705
706
<pre><code class="javascript">
707
{
708
  "nodeOne": "v1",
709
  "nodeTwo": "v2",
710
  "id": 5005
711
}
712
</code></pre>
713
714
----
715
716
*DELETE http://localhost:8080/rest/v1/graph/edges* - удаляет все рёбра графа
717
*DELETE http://localhost:8080/rest/v1/graph/edges/byId/5005* - удаляет ребро по заданному id
718
*DELETE http://localhost:8080/rest/v1/graph/edges/byName/v1* - удаляет список рёбер связанный с узлом, в запросе задаётся уникальное имя узла
719
*DELETE http://localhost:8080/rest/v1/graph/edges/byName?nodeOne=v3&nodeTwo=v4* - удаляет ребро по заданным узлам графа
Go to top