// package use_link -- связь между объектами
package use_link
import (
"fmt"
"gitp78su.ipnodns.ru/svi/goarch/lev0/alias"
"gitp78su.ipnodns.ru/svi/goarch/lev0/cons"
"gitp78su.ipnodns.ru/svi/goarch/lev0/types"
"gitp78su.ipnodns.ru/svi/goarch/pkg/elems/coord"
"gitp78su.ipnodns.ru/svi/goarch/pkg/elems/elem_base"
)
// UseLink -- связь между объектами
type UseLink struct {
*elem_base.ElemBase
CoordEnd_ *coord.Coord `json:"coord1"` // Координаты второй точки
Src_ alias.Id `json:"src"` // Источник координат
Dst_ alias.Id `json:"dst"` // Получатель координат
TypeLink_ string `json:"typeLink"` // Тип связи
}
// NewUseLink -- возвращает новую связь
func NewUseLink(elem map[string]interface{}) (*UseLink, error) {
elemBase, err := elem_base.NewElemBase(elem)
if err != nil {
return nil, fmt.Errorf("NewUseLink(): in create BaseSvg, err=\n\t%w", err)
}
id := elemBase.Id_
_src := elemBase.Elem_["src"]
src0, isOk := _src.(string)
if !isOk {
return nil, fmt.Errorf("NewUseLink(): id=%q, field `src` not string, type=%T, value=%v", id, _src, elemBase.StrElem_)
}
src := alias.Id(src0)
if src == "" {
return nil, fmt.Errorf("NewUseLink(): id=%q, `src` is empty", id)
}
_dst := elemBase.Elem_["dst"]
dst0, isOk := _dst.(string)
if !isOk {
return nil, fmt.Errorf("NewUseLink(): id=%q, field `dst` not string, type=%T, value=%+v
%+v", id, _dst, _dst, elemBase.StrElem_)
}
dst := alias.Id(dst0)
_typeLink, isOk := elemBase.Elem_["type_link"]
if !isOk {
return nil, fmt.Errorf("NewUseLink(): name=%q, field `type_link` not found
%+v", id, elemBase.StrElem_)
}
typeLink, isOk := _typeLink.(string)
if !isOk {
return nil, fmt.Errorf("NewUseLink(): name=%q, `type_link`(%+v) not string", id, _typeLink)
}
if typeLink == "" {
return nil, fmt.Errorf("NewUseLink(): name=%q, type_link=%q is empty", id, typeLink)
}
sf := &UseLink{
ElemBase: elemBase,
CoordEnd_: coord.NewCoord("coord1", elem),
Src_: src,
Dst_: dst,
TypeLink_: typeLink,
}
return sf, nil
}
// Links -- ссылки актора
func (sf *UseLink) Links() []alias.Id {
return []alias.Id{sf.Src_, sf.Dst_}
}
// Check -- проверяет связи между объектами
func (sf *UseLink) Check(mapDrawer map[alias.Id]types.IElemDrawer) string {
if sf.Dst_ == sf.Src_ {
return fmt.Sprintf("ВНИМАНИЕ! Источник и получатель совпадают\nsrc=%v, dst=%v\n", sf.Src_, sf.Dst_)
}
if msgErr := sf.checkSrc(mapDrawer); msgErr != "" {
return "ОШИБКА при проверке связи источника\n" + msgErr
}
if msgErr := sf.checkDst(mapDrawer); msgErr != "" {
return "ОШИБКА при проверке связи получателя\n" + msgErr
}
if sf.TypeLink_ == "" { // Проверка на правильность типа связи
return "Пустой собственный тип связи\n"
}
return ""
}
// проверить связь получателя
func (sf *UseLink) checkDst(mapDrawer map[alias.Id]types.IElemDrawer) string {
if sf.Dst_ == "" {
return "ОШИБКА получатель `dst` не задан"
}
dst, isOk := mapDrawer[sf.Dst_] // Вычислить, если такой получатель
if !isOk {
return "Отсутствует собственный получатель(dst)\n"
}
// Вычислить на допустимые типы для получателя
switch dst.Type() {
case cons.TypeUseActor:
actor, isOk := dst.(types.IActor)
if !isOk {
return fmt.Sprintf("Недопустимый тип получателя(%q)\nтип=%q\n", dst.Id(), dst.Type())
}
isOk = false
for _, id := range actor.Links() {
if id == sf.Id_ {
isOk = true
break
}
}
if !isOk {
return fmt.Sprintf("Невзаимная ссылка\nid=%q,dst=(%q)", sf.Id_, actor.Id())
}
case cons.TypeUseLink:
link, isOk := dst.(types.ILinker)
if !isOk {
return fmt.Sprintf("Недопустимый тип получателя(%q)\nтип=%q\n", dst.Id(), dst.Type())
}
isOk = sf.Id_ == link.DstId() || sf.Id_ == link.SrcId()
if !isOk {
return fmt.Sprintf("id=%q, Невзаимная ссылка(%q)\n", sf.Id_, link.Id())
}
case cons.TypeUseCase:
useCase, isOk := dst.(types.IUseCase)
if !isOk {
return fmt.Sprintf("id=%q, Недопустимый тип получателя(%q)\nтип=%q\n", sf.Id_, dst.Id(), dst.Type())
}
isOk = false
for _, id := range useCase.Links() {
if id == sf.Id_ {
isOk = true
break
}
}
if !isOk {
return fmt.Sprintf("id=%q, Невзаимная ссылка(%q)\n", sf.Id_, useCase.Id())
}
default: // Недопустимый источник
return fmt.Sprintf("id=%q, Недопустимый тип получателя(%q)\nid=%q\n", sf.Id_, dst.Type(), dst.Id())
}
return ""
}
// Проверяет связь источника
func (sf *UseLink) checkSrc(mapDrawer map[alias.Id]types.IElemDrawer) string {
if sf.Src_ == "" {
return "ВНИМАНИЕ! Источник связи не задан(src)\n"
}
src, isOk := mapDrawer[sf.Src_]
if !isOk {
return "Отсутствует источник(src)\n"
}
switch src.Type() {
case cons.TypeUseActor:
isOk = false
actor := types.IActor(src)
for _, id := range actor.Links() { // Проверка на взаимность
if id == sf.Id_ {
isOk = true
break
}
}
if !isOk {
return fmt.Sprintf("Невзаимная ссылка с актором %q\n", actor.Id())
}
case cons.TypeUseLink:
link, isOk := src.(types.ILinker)
if !isOk {
return fmt.Sprintf("Несоответствие типа источника(%q)\nтип='useLink'\n", src.Id())
}
isOk = sf.Id_ == link.DstId() || sf.Id_ == link.SrcId()
if !isOk {
return fmt.Sprintf("id=%q, Невзаимная ссылка(%q)\n", sf.Id_, link.Id())
}
case cons.TypeUseCase:
useCase, isOk := src.(types.IUseCase)
if !isOk {
return fmt.Sprintf("id=%q, Тип источника(%q) не совпадает с фактическим\nтип='useCase'\n", sf.Id_, src.Id())
}
isOk = false
for _, id := range useCase.Links() { // Проверка на взаимность
if id == sf.Id_ {
isOk = true
break
}
}
if !isOk {
return fmt.Sprintf("id=%q, Невзаимная ссылка(%q)\n", sf.Id_, useCase.Id())
}
default: // Неизвестный источник
return fmt.Sprintf("id=%q, неизвестный тип источника(%q)
id=%q\n", sf.Id_, src.Type(), src.Id())
}
return ""
}
// SrcId -- возвращает источник связи
func (sf *UseLink) SrcId() alias.Id {
return sf.Src_
}
// DstId -- возвращает получателя связи
func (sf *UseLink) DstId() alias.Id {
return sf.Dst_
}