# Channel Management (Path Based)
## Resource
Ресурс является унифицированной структурой данных, адресующие сущности в базе, через которые могут проходить каналы.
Состоит из 3 частей:
* Модель (в терминах NOC)
* id - идентификатор в модели
* path - опциональный путь внутри ресурса
Для кодирования ресурса можно использовать любое обратимое преобразование, которое позволяет:
* От записи в базе (или ее части) перейти к идентификатору ресурса
* От идентификатора ресурса перейти к записи в базе
Наиболее простым в работе будет кодирование в виде строки:
```
<код модели>:<id>[:path]
```
Коды модели берутся из таблицы:
| Код | Модель |
| --- | --- |
| `o` | inv.Object |
Так как инвентарная база иерархическая, некоторые типы ресурсов могут также быть иерархическими, например, точка присутсвия включает в себя порты всех устройств в ней.
Примеры:
| Resource | Пример |
| --- | --- |
| `o:12345` | Объект с id 12345 |
| `o:12345:4` | Connection `4` объекта с id 12345 |
## Канальный ресурс
Ресурсы, которые могут использоваться для организации каналов называются "Канальными ресурсами"
## Python API
### Получение ресурса для модели/документа
В моделях/документах, которые могут использоваться как ресурсы реализуется протокол `AsResource`:
``` python
class AsResource(Protocol):
def as_resource(self, path: Optional[str] = None) -> str: ...
```
Используется:
``` python
Object.get_by_id("xxxxx").as_resource()
Object.get_by_id("xxxxx").as_resource("3")
```
### Получение объекта
`noc.core.resource`
```
def from_resource(id:str) -> Tuple[Model|Document, Optional[str]]:...
```
## TechDomain
Технологический домен, представляет собой технологию для организации связи и тесно связан с контроллером - технической реализацией процесса автоматизации.
Модель данных:
* id
* name
* code - короткий символьный код (`group`, ...)
* uuid
* description
* controller_handler
* allow_parent - разрешать родительские каналы
* allow_children - разрешать дочерние каналы
* allow_p2p - разрешить топологию p2p
* allow_up2p
* allow_bunch - ... bunch
* allow_p2mp - ... p2mp
* allow_up2mp
* allow_star - ... star
* channel_discriminator - опционально - дискриминатор на уровне канала
* endpoint_discriminator - опицонально - дискриминатор для подключения к каналу
Из коробки система поставляется со следующими доменами:
* group - группировка канала
* optical_sm - оптика одномод
* optical_mm - оптика многомод
* otn_otu - Уровень OTN OTU
* otn_pdu - Уровень OTN ODU
## Channel
Канал обеспечивает связность между канальными ресурсами. Каналы могут образовывать иерархию. В случае иерархической схемы каналы-листья организовывают связность в пределах одного технологического домена, в то время как каналы-узлы их просто группируют. В таком случае у каналов-узлов технологический домен всегда равен `group`.
:::info
Вопрос о целесообразности более двух уровней иерархии каналов остается открытым.
:::
Каждый канал строится в рамках отдельной задачи, так что дробление может быть вызвано не только технологическими, но и организационными причинами.
Каналы реализуют связность по одной из нескольких топологий:
* `p2p` - point to point, точка-точка
* `up2p` - unidirectional point to point, однонаправленная точка-точка
* `bunch` - множественная точка-точка. Состоит из независимых и связанных между собой пар ресурсов. Фактически является группировкой нескольких `p2p` каналов (пучок), без необходимости создания отдельного канала на каждую пару
* `ubunch` - однонаправленный `bunch`. Возможно движение строго от корней к листьям.
* `p2mp` - point to multipoint. дерево. Связность между листьями отсутсвует. Возможно движение от корня к листу или от листа к корню
* `up2mp` - unidirectional point to multipoint, однонаправленное дерево. Возможно движение строго от корня к листу.
* `star` - полносвязная топология.
Модель данных:
* id
* name
* parent - родитель (если есть)
* tech_domain - ссылка на TechDomain
* description
* topology - тип топологии (см. выше)
* p2p
* up2p
* bunch
* ubunch
* p2mp
* up2mp
* up2mp
* star
* kind - тип канала
* l1
* l2
* l3
* internet
* state
* labels
* effective_labels
* project
* subscriber
* supplier
* remote_system
* remote_id
* bi_id
* context - контекст контроллера
* constraints - клиентские настройки, ограничивающие возможные варианты построения. Список:
* type - тип ограничения:
* `i` - include
* `e` - exclude
* strict - boolean. Если `True`, то ограничение всегда должно соблюдаться, если `False`, то соблюдение условия желательно, но невозможность соблюдения не является фатальной.
* resource - Resource, который задействован в ограничении.
:::info
При задании ограничений можно и нужно использовать иерархичность ресурсов
:::
## Endpoint
По определению каналы начинаются и заканчиваются на каких-то ресурсах. Эти ресурсы отличаются от промежуточных ресурсов по пути. Во первых, они являются границами канала, то есть являются конечной целью всего процесса управлениия каналами. Во вторых - они позволяют другим каналам пользоваться связностью. Назовем их точками терминации или конечными точками.
В отдельных случаях каналы требуют раздельных физических сред для приема и передачи. В таком случае подключение возникает в 4 точках, которые должны быть разделены на 2 пары. Это решается кодом направления. Так как понятие "направление" весьма условно, используется "правило большого пальца":
* 0 - исходящее направление от того места, с которого мы начали строить канал
* 1 - входящее направление
Модель:
* id
* channel - указатель на канал, связывающий точки подключения
* resource - Resource
* is_root - корневой узел дерева для топологии p2mp
* pair - Для топологии `bunch` задает номер пары. Допускается не более 2 TP в канале с одной парой.
* used_by - коналы, использующие связность.
* channel - указатель на channel
* discriminator - опциональный дискриминатор, в случае, если точка подключения используется не монопольно.
* direction - опциональный код направления.
## ResourcePath
Для организации связности канальные ресурсы должны быть связаны между собой, образовывая граф. Граф просчитываеся заранее и связывает между собой все возможные ресурсы. Узлами графа являются ресурсы, дугами - возможная связность между ними.
В случае, если ресурсы - объекты и их connection, дуги формируются из ObjectConnection и кроссировок объекта.
Аттрибутика для особых ресурсов уже размечена в структуре Endpoint.
Модель данных для дуг:
* x - resorce
* y - resource
* channels - список каналов, которые проходят через дугу
* channel - id канала
## Пример OTN
``` mermaid
%%{}%%
graph LR
1((1))
2((2))
subgraph ADM1
ADM1::Client1[Client1]
subgraph ADM1::SFP[Line1]
ADM1::sfp_tx[tx]
ADM1::sfp_rx[rx]
end
ADM1::Client1 -- odu::ODU2:ODU1-1 --- ADM1::SFP
end
subgraph OM1
OM1::C14[C14]
OM1::Line[Line]
OM1::C14 --> OM1::Line
end
subgraph OD1
OD1::C14[C14]
OD1::Line[Line]
OD1::Line -- lambda::c14 --> OD1::C14
end
subgraph Patch1
Patch1::1[1]
Patch1::2[2]
Patch1::c[Cable]
Patch1::1 -- fiber::1 --- Patch1::c
Patch1::2 -- fiber::2 --- Patch1::c
end
subgraph Patch2
Patch2::1[1]
Patch2::2[2]
Patch2::c[Cable]
Patch2::1 -- fiber1::1 --- Patch2::c
Patch2::2 -- fiber::2 --- Patch2::c
end
Patch1::c --- Patch2::c
subgraph ADM2
ADM2::Client5[Client5]
subgraph ADM2::SFP[Line2]
ADM2::sfp_tx[tx]
ADM2::sfp_rx[rx]
end
ADM2::Client5 -- odu::ODU2:ODU1-1 --- ADM2::SFP
end
subgraph OM2
OM2::C14[C14]
OM2::Line[Line]
OM2::C14 --> OM2::Line
end
subgraph OD2
OD2::C14[C14]
OD2::Line[Line]
OD2::Line -- lambda:c14 --> OD2::C14
end
1 --- ADM1::Client1
2 --- ADM2::Client5
ADM1::sfp_tx --> OM1::C14
OM1::Line --> Patch1::1
Patch2::1 --> OD2::Line
OD2::C14 --> ADM2::sfp_rx
ADM2::sfp_tx --> OM2::C14
OM2::Line --> Patch2::2
Patch1::2 --> OD1::Line
OD1::C14 --> ADM1::sfp_rx
```
```graphviz
graph {
graph [rankdir = LR]
e1 [shape = point]
e2 [shape = point]
subgraph cluster_ADM1 {
graph [ label="ADM1"]
ADM1_Client1 [shape = box label = "Client 1"]
ADM1_Line1 [shape = record label="<rx>rx|<out>Line1|<tx>tx"]
}
subgraph cluster_ADM2 {
graph [ label="ADM2"]
ADM2_Client5 [shape = box label = "Client 5"]
ADM2_Line2 [shape = record label="<rx>rx|<out>Line2|<tx>tx"]
}
e1 -- ADM1_Client1
e2 -- ADM2_Client5
ADM1_Client1 -- ADM1_Line1:out [label="odu::ODU2:ODU1-1" style=dashed]
ADM2_Client5 -- ADM2_Line2:out [label="odu::ODU2:ODU1-1" style=dashed]
ADM1_Line1:tx -- ADM2_Line2:rx [label="C14" dir=forward]
ADM2_Line2:tx -- ADM1_Line1:rx [label="C14" dir=forward]
}
```
Рассмотрим процесс организации канала по сети OTN от точки 1 к точке 2.
Процесс разбивается на 3 уровня, причем результат предыдущих уровней может повторно использоваться для организации каналов следующего уровня:
* Оптическая коммутация (Och)
* OTU Trail
* ODU Trail
**Och**
На нашей схеме изображены две патч-панели, соединеные кабелем. Для простоты мы используем только первых 2 порта, остальные опускаем, чтобы не загромождать схему.
Пользователь выбирает первый порт Patch1 и жмет на кнопку "Create channel"
Создаем оптический канал. Для оптических каналов идеальным типом топологии является `bunch`, так как его endpoint'ы образуют связанные пары. Это избавляет нас от необходимости создавать отдельные каналы на каждое волокно. Весь сегмент оптической сети после разварки может быть представлен как один канал.
Добавляем канал в справочник:
| id | tech_domain | kind | topology |
| --- | --- | --- | --- |
| ==Och1== | ==optical_sm== | ==L1== | ==bunch== |
Порты патч-панелей являются точками выхода из каналов.
Мы вызываем метод `prepare_channel` контроллера `optical_sm`, передавая ему как параметр первый порт патч-панели Patch1.
Контроллер проводит трассировку связности, используя object connection и, по необходимости, дискриминатор fiber. Дискриминатор fiber задает номер волокна.
В результате в таблице Endpoint добавляются записи
| channel | resource | pair | used_by |
| --- | --- | --- | --- |
| ==Och1== | ==o:Patch1:1== | ==1== | |
| ==Och1== | ==o:Patch1:2== | ==2== | |
| ==Och1== | ==o:Patch2:1== | ==1== | |
| ==Och1== | ==o:Patch2:2== | ==2== | |
Мы видим 4 endpoint'а, объединенные в 2 пары.
На схеме так же присутсвуют пассивные оптические устройства:
* Оптические мультиплексоры (OM) - собирают сигнал с клиентских портов в один линейный
* Оптические демультиплексоры (DM) - раскладывают входящий по линейному порту сигнал по клиентским портам по длинам волны.
Комбинация OM и OD позволяет использовать оптическое уплотнение.
Пара устройств OM и OD образуют однонаправленный канал. Для таких устройств удобнее всего использовать топологию `ubunch`, которая представляет собой однонаправленный вариант `bunch`.
Пользователь выбирает клиентский порт мультиплексора или демультиплексора и нажимает кнопку "Создать канал".
Контроллер проводит трассировку связности, используя object connection и, по необходимости, дискриминатор fiber. Дискриминатор fiber задает номер волокна. Если по пути встречаются endpoint'ы процедура проходит сразу к парному endpoint'у.
Так как у нас две пары мультиплексоров мы создаем два канала:
| id | tech_domain | kind | topology |
| --- | --- | --- | --- |
| Och1 | optical_sm | L1 | bunch |
| ==Och2== | ==optical_sm== | ==L1== | ==ubunch== |
| ==Och3== | ==optical_sm== | ==L1== | ==ubunch== |
Также создаются endpoint'ы для каждого из оптических каналов. Чтобы не загромождать таблицу мы показываем только канал 14 (пара 14).
| channel | resource | pair | is_root | used_by |
| --- | --- | --- | --- | --- |
| Och1 | o:Patch1:1 | 1 | | ==Och2== |
| Och1 | o:Patch1:2 | 2 | | ==Och3== |
| Och1 | o:Patch2:1 | 1 | | ==Och2== |
| Och1 | o:Patch2:2 | 2 | | ==Och2== |
| ==Och2== | ==o:OM1:C14== | ==14== | ==True== | |
| ==Och2== | ==o:OD2:C14== | ==14== | | |
| ==Och3== | ==o:OM2:C14== | ==14== | ==True== | |
| ==Och3== | ==o:OD1:C14== | ==14== | | |
Так как мы использовали промежуточный канал Och1, мы также заполняем поля used_by на его соответствующих endpoint.
**OTU Trail**
* Пользователь выбирает карту ADM1
* Контроллер находим список свободных линейных портов по следующему алгоритму:
* поддерживаются протоколы OTU
* resource порта не содержится в поле Endpoint.resource
* если порт имеет direction in, проверяем, что он соединен.
* Пользователь выбирает порт/для всех портов и длину волны
* Если connection имеет тип inner, спускаемся к трансиверу
* Находим выходной порт (tx)
* Проходим по object connection, по необходимости используя дискриминаторы lambda и fiber пока не дойдем до слота, который поддерживает протоколы OTU или имеет connection type типа outer и парный слот поддерживает протоколы OTU.
* Повторяем процесс в обратную сторону пока не придем в исходный порт
* Берем пересечение списка протоколов OTU на портах с двух сторон
* Показываем пользователю парный порт и типы OTU
* Пользователь выбирает порт и тип OTU
Создаем канал:
| id | tech_domain | kind | topology | channel_discriminator |
| --- | --- | --- | --- | --- |
| Och1 | optical_sm | L1 | bunch | |
| Och2 | optical_sm | L1 | ubunch | |
| Och3 | optical_sm | L1 | ubunch | |
| ==OTU1== | ==otn_otu== | ==L1== | ==p2p== | ==otu:OTU2, lambda:c14== |
Мы запомнили выбранный тип OTU и длину волны в канальном дискриминаторе.
Добавляем в endpoint'ы выбранные порты:
| channel | resource | pair | is_root | used_by |
| --- | --- | --- | --- | --- |
| Och1 | o:Patch1:1 | 1 | | Och2 |
| Och1 | o:Patch1:2 | 2 | | Och3 |
| Och1 | o:Patch2:1 | 1 | | Och2 |
| Och1 | o:Patch2:2 | 2 | | Och2 |
| Och2 | o:OM1:C14 | 14 | True | =={OTU1, direction=0}== |
| Och2 | o:OD2:C14 | 14 | | =={OTU1, direction=0}== |
| Och3 | o:OM2:C14 | 14 | True | =={OTU1, direction=1}== |
| Och3 | o:OD1:C14 | 14 | | =={OTU1, direction=1}== |
| ==OTU1== | ==o:ADM1:Line1== | | | |
| ==OTU1== | ==o:ADM2:Line2== | | | |
Мы используем индикатор direction для того, чтобы разделить входящее и исходящее направление.
Контроллер создает два задания на настройку OTU на линейных картах
**ODU**
* Клиент выбирает клиентский порт
* Контроллер находит все каналы с типом otn_otu, которые выходят с линейных портов карты
* Для каждого из каналов контроллер находит:
* список не занятых портов с протоколами ODU с другой стороны, на которые физически можно осуществить коммутацию
* Свободные ODU
* Клиент выбирает ответный порт, тип ODU и место ODU в общем контейнере
Создаем новый канал:
| id | tech_domain | kind | topology | channel_discriminator |
| --- | --- | --- | --- | --- |
| Och1 | optical_sm | L1 | bunch | |
| Och2 | optical_sm | L1 | ubunch | |
| Och3 | optical_sm | L1 | ubunch | |
| OTU1 | otn_otu | L1 | p2p | otu:OTU2, lambda:c14 |
| ==ODU1== | ==otn_odu== | ==L1== | ==p2p== | ==odu:ODU2:ODU1-1== |
Добавляем endpoint'ы:
| channel | resource | pair | is_root | used_by |
| --- | --- | --- | --- | --- |
| Och1 | o:Patch1:1 | 1 | | Och2 |
| Och1 | o:Patch1:2 | 2 | | Och3 |
| Och1 | o:Patch2:1 | 1 | | Och2 |
| Och1 | o:Patch2:2 | 2 | | Och2 |
| Och2 | o:OM1:C14 | 14 | True | {OTU1, direction=0} |
| Och2 | o:OD2:C14 | 14 | | {OTU1, direction=0} |
| Och3 | o:OM2:C14 | 14 | True | {OTU1, direction=1} |
| Och3 | o:OD1:C14 | 14 | | {OTU1, direction=1} |
| OTU1 | o:ADM1:Line1 | | | =={ODU1, disc=odu:ODU2:ODU1-1}== |
| OTU1 | o:ADM2:Line2 | | | =={ODU1, disc=odu:ODU2:ODU1-1}== |
| ==ODU1== | ==o:ADM1:Client1== |
| ==ODU1== | ==o:ADM2:Client5== |
Контроллер создает 2 кроссировки на линенйных картах
Контроллер отправляет задание на настройку кроссировок на ADM картах