Stage-2 » История » Версия 30
Александр Александров, 20.11.2019 01:29
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 | 27 | Александр Александров | Ниже представлены примеры HTTP запросов |
226 | |||
227 | 26 | Александр Александров | h3. HTTP запросы при работе с графом |
228 | 23 | Александр Александров | |
229 | 29 | Александр Александров | * *POST http://localhost:8080/rest/v1/graph* - создаёт новый граф, при этом прежний граф полностью удаляется. |
230 | 30 | Александр Александров | ** Тело запроса: *{"nodes": [{"name":"unique_node_name"}, ...], "edges": [{"nodeOne":"unique_node_one_name", "nodeTwo":"unique_node_one_name"}, ...]}* |
231 | ** Ответ: *{"nodes":[{"id":1, "name":"unique_name", "counter":0}, ...], "edges":[{"id":1, "nodeOne":"unique_node_name_one", "nodeOne":"unique_node_name_one"}, ...]}* |
||
232 | 29 | Александр Александров | * *POST http://localhost:8080/rest/v1/graph/checkroute* - проверка работоспособности заданной последовательности узлов. |
233 | ** Тело запроса вида: *["unique_node_name", "unique_node_name", ...]* |
||
234 | ** Ответ вида: при корректном выполнении *Route for nodes ["unique_node_name", "unique_node_name", ...] is found* |
||
235 | 23 | Александр Александров | |
236 | ---- |
||
237 | |||
238 | 28 | Александр Александров | * *GET http://localhost:8080/rest/v1/graph* - возвращает объект графа |
239 | * *GET http://localhost:8080/rest/v1/graph/export* - возвращает граф в формате GraphViz |
||
240 | 23 | Александр Александров | |
241 | ---- |
||
242 | |||
243 | 28 | Александр Александров | * *DELETE http://localhost:8080/rest/v1/graph* - удаляет граф с БД, т.е. полностью удаляется информация с БД по узлам и рёбрам графа. |
244 | 23 | Александр Александров | |
245 | h3. HTTP запросы при отдельной работе с узлами графа |
||
246 | 1 | Александр Александров | |
247 | * *POST http://localhost:8080/rest/v1/graph/nodes* - добавляет объект узла в граф |
||
248 | 30 | Александр Александров | ** Тело запроса : *{"name":"unique_node_name"}* |
249 | ** Ответ : *{"id":1, "name":"unique_name", "counter":0}* |
||
250 | 1 | Александр Александров | * *POST http://localhost:8080/rest/v1/graph/nodes/byBatch* - добавление набора узлов в граф |
251 | 30 | Александр Александров | ** Тело запроса : *[{"name":"unique_node_name"}, {"name":"unique_node_name"}, ...]* |
252 | ** Ответ : *[{"id":1, "name":"unique_name", "counter":0}, {"id":2, "name":"unique_name", "counter":0}, ...]* |
||
253 | 23 | Александр Александров | |
254 | ---- |
||
255 | |||
256 | 28 | Александр Александров | * *GET http://localhost:8080/rest/v1/graph/nodes* - выводит список всех узлов в графе |
257 | * *GET http://localhost:8080/rest/v1/graph/nodes/byId/{id}* - выводит объект узла по заданному *{id}* |
||
258 | * *GET http://localhost:8080/rest/v1/graph/nodes/byName/{name}* - выводит объект узла по заданному имени *{name}* |
||
259 | 23 | Александр Александров | |
260 | ---- |
||
261 | 28 | Александр Александров | |
262 | * *DELETE http://localhost:8080/rest/v1/graph/nodes* - удаляет все узлы и рёбра связанные с узлами из БД |
||
263 | * *DELETE http://localhost:8080/rest/v1/graph/nodes/byId/{id}* - удаляет узел по заданному *{id}* |
||
264 | 1 | Александр Александров | * *DELETE http://localhost:8080/rest/v1/graph/nodes/byName/{name}* - удаляет узел по заданному имени *{name}* |
265 | 28 | Александр Александров | * *DELETE http://localhost:8080/rest/v1/graph/nodes/byObj* - удаляет узел по заданному объекту |
266 | 30 | Александр Александров | ** Тело запроса : *{"id": 1, "name": "unique_name"}* |
267 | 23 | Александр Александров | |
268 | 1 | Александр Александров | h3. HTTP запросы при отдельной работе с рёбрами графа |
269 | |||
270 | 23 | Александр Александров | * *POST http://localhost:8080/rest/v1/graph/edges* - добавляет объект ребра в граф |
271 | 30 | Александр Александров | ** Тело запроса : *{"nodeOne":"unique_node_one_name", "nodeTwo":"unique_node_one_name"}* |
272 | ** Ответ : *{"id":1, "nodeOne":"unique_node_name_one", "nodeOne":"unique_node_name_one"}* |
||
273 | 23 | Александр Александров | * *POST http://localhost:8080/rest/v1/graph/edges/byBatch* - добавляет список рёбер графа |
274 | 30 | Александр Александров | ** Тело запроса : *[{"nodeOne":"unique_node_one_name", "nodeTwo":"unique_node_one_name"}, {"nodeOne":"unique_node_one_name", "nodeTwo":"unique_node_one_name"}, ...]* |
275 | ** Ответ : *[{"id":1, "nodeOne":"unique_node_name_one", "nodeOne":"unique_node_name_one"}, {"id":2, "nodeOne":"unique_node_name_one", "nodeOne":"unique_node_name_one"}, ...]* |
||
276 | 23 | Александр Александров | |
277 | ---- |
||
278 | |||
279 | 28 | Александр Александров | * *GET http://localhost:8080/rest/v1/graph/edges* - возвращает список рёбер графа |
280 | * *GET http://localhost:8080/rest/v1/graph/edges/byId/{id}* - возвращает ребро графа по заданному *{id}* |
||
281 | * *GET http://localhost:8080/rest/v1/graph/edges/byName/{name}* - возвращает список рёбер связанный с заданным узлом *{name}* , при запросе задаётся уникальное имя узла |
||
282 | * *GET http://localhost:8080/rest/v1/graph/edges/byName?nodeOne={name1}&nodeTwo={name2}* - выводит объект ребра по заданным узлам графа *{name1}* и *{name2}* |
||
283 | 23 | Александр Александров | |
284 | ---- |
||
285 | |||
286 | 28 | Александр Александров | * *DELETE http://localhost:8080/rest/v1/graph/edges* - удаляет все рёбра графа |
287 | * *DELETE http://localhost:8080/rest/v1/graph/edges/byId/{id}* - удаляет ребро по заданному *{id}* |
||
288 | * *DELETE http://localhost:8080/rest/v1/graph/edges/byName/{name}* - удаляет список рёбер связанный с узлом *{name}* , в запросе задаётся уникальное имя узла |
||
289 | * *DELETE http://localhost:8080/rest/v1/graph/edges/byName?nodeOne={name1}&nodeTwo={name2}* - удаляет ребро по заданным узлам графа *{name1}* и *{name2}* |