Skip to content
Snippets Groups Projects
configurationmanagement.go 8.8 KiB
Newer Older
  • Learn to ignore specific revisions
  • package server
    
    import (
    	"context"
    	"encoding/json"
    	"time"
    
    	cmpb "code.fbi.h-da.de/danet/gosdn/api/go/gosdn/configurationmanagement"
    	spb "code.fbi.h-da.de/danet/gosdn/api/go/gosdn/southbound"
    	"code.fbi.h-da.de/danet/gosdn/controller/interfaces/networkdomain"
    	"code.fbi.h-da.de/danet/gosdn/controller/interfaces/networkelement"
    	"code.fbi.h-da.de/danet/gosdn/controller/interfaces/southbound"
    	"code.fbi.h-da.de/danet/gosdn/controller/nucleus"
    	"code.fbi.h-da.de/danet/gosdn/controller/store"
    	"code.fbi.h-da.de/danet/gosdn/controller/topology"
    	"code.fbi.h-da.de/danet/gosdn/controller/topology/links"
    	"code.fbi.h-da.de/danet/gosdn/controller/topology/nodes"
    	"code.fbi.h-da.de/danet/gosdn/controller/topology/ports"
    
    	tpb "code.fbi.h-da.de/danet/gosdn/api/go/gosdn/transport"
    	"github.com/google/uuid"
    )
    
    // ConfigurationManagementServer represents  ConfigurationManagementServer...
    type ConfigurationManagementServer struct {
    	cmpb.UnimplementedConfigurationManagementServiceServer
    	pndStore        networkdomain.PndStore
    	topologyService topology.Service
    	nodeService     nodes.Service
    	portService     ports.Service
    }
    
    // NewConfigurationManagementServer creates the ConfigurationManagementServer..
    func NewConfigurationManagementServer(
    	pndStore networkdomain.PndStore,
    	topologyService topology.Service,
    	nodeService nodes.Service,
    	portService ports.Service,
    ) *ConfigurationManagementServer {
    	return &ConfigurationManagementServer{
    		pndStore:        pndStore,
    		topologyService: topologyService,
    		nodeService:     nodeService,
    		portService:     portService}
    }
    
    // sdnConfig is used to parse the sdnConfig into JSON.
    type sdnConfig struct {
    	PndID           string                           `json:"pndID"`
    	Nodes           []nodes.Node                     `json:"nodes"`
    	Ports           []ports.Port                     `json:"ports"`
    	Links           []links.Link                     `json:"links"`
    	Sbis            []southbound.SouthboundInterface `json:"sbis"`
    	NetworkElements []networkelement.NetworkElement  `json:"networkelements"`
    }
    
    // loadedSDNConfig is used to parse the stringified JSON sdnConfig into objects.
    type loadedSDNConfig struct {
    	PndID           string                                `json:"pndID"`
    	Nodes           []nodes.Node                          `json:"nodes"`
    	Ports           []ports.Port                          `json:"ports"`
    	Links           []links.Link                          `json:"links"`
    	Sbis            []southbound.LoadedSbi                `json:"sbis"`
    	NetworkElements []networkelement.LoadedNetworkElement `json:"networkelements"`
    }
    
    // ExportSDNConfig returns the SDN configuration.
    func (c ConfigurationManagementServer) ExportSDNConfig(ctx context.Context, request *cmpb.ExportSDNConfigRequest) (*cmpb.ExportSDNConfigResponse, error) {
    	var sdnConfig = sdnConfig{}
    	sdnConfig.PndID = request.Pid
    
    	pndUUID := uuid.MustParse(request.Pid)
    	pnd, err := c.pndStore.Get(store.Query{ID: pndUUID})
    	if err != nil {
    		return nil, err
    	}
    
    	networkElements := pnd.NetworkElements()
    
    	for _, networkElement := range networkElements {
    		model, err := networkElement.GetModelAsFilteredCopy()
    		if err != nil {
    			return nil, err
    		}
    		networkElement.SetModel(model)
    	}
    
    	sdnConfig.NetworkElements = networkElements
    	sdnConfig.Sbis, err = pnd.GetSBIs()
    	if err != nil {
    		return nil, err
    	}
    	sdnConfig.Nodes, err = c.nodeService.GetAll()
    	if err != nil {
    		return nil, err
    	}
    	sdnConfig.Ports, err = c.portService.GetAll()
    	if err != nil {
    		return nil, err
    	}
    	sdnConfig.Links, err = c.topologyService.GetAll()
    	if err != nil {
    		return nil, err
    	}
    
    	jsonSDNConfig, err := json.MarshalIndent(sdnConfig, "", "   ")
    	if err != nil {
    		return nil, err
    	}
    
    	sdnConfigDataString := string(jsonSDNConfig)
    
    	return &cmpb.ExportSDNConfigResponse{
    		Timestamp:     time.Now().UnixNano(),
    		SdnConfigData: sdnConfigDataString,
    		Status:        cmpb.Status_STATUS_OK}, nil
    }
    
    // ImportSDNConfig receives an SDN configuration and imports it.
    func (c ConfigurationManagementServer) ImportSDNConfig(ctx context.Context, request *cmpb.ImportSDNConfigRequest) (*cmpb.ImportSDNConfigResponse, error) {
    	pndUUID := uuid.MustParse(request.Pid)
    	var sdnConfig = loadedSDNConfig{}
    	err := json.Unmarshal([]byte(request.SdnConfigData), &sdnConfig)
    	if err != nil {
    		return nil, err
    	}
    
    	err = c.deleteAllElementsFromDatabase(pndUUID)
    	if err != nil {
    		return nil, err
    	}
    
    	err = c.createElementsFromSDNConfig(&sdnConfig, pndUUID)
    	if err != nil {
    		return nil, err
    	}
    
    	return &cmpb.ImportSDNConfigResponse{
    		Timestamp: time.Now().UnixNano(),
    		Status:    cmpb.Status_STATUS_OK}, nil
    }
    
    func (c ConfigurationManagementServer) deleteAllElementsFromDatabase(pndUUID uuid.UUID) error {
    	err := c.deleteNetworkElementsAndSBIs(pndUUID)
    	if err != nil {
    		return err
    	}
    
    	err = c.deleteTopology()
    	if err != nil {
    		return err
    	}
    
    	return nil
    }
    
    func (c ConfigurationManagementServer) deleteTopology() error {
    	links, err := c.topologyService.GetAll()
    	if err != nil {
    		return err
    	}
    	for _, link := range links {
    		err = c.topologyService.DeleteLink(link)
    		if err != nil {
    			return err
    		}
    	}
    
    	ports, err := c.portService.GetAll()
    	if err != nil {
    		return err
    	}
    	for _, port := range ports {
    		err = c.portService.Delete(port)
    		if err != nil {
    			return err
    		}
    	}
    
    	nodes, err := c.nodeService.GetAll()
    	if err != nil {
    		return err
    	}
    	for _, node := range nodes {
    		err = c.nodeService.Delete(node)
    		if err != nil {
    			return err
    		}
    	}
    
    	return nil
    }
    
    func (c ConfigurationManagementServer) deleteNetworkElementsAndSBIs(pndUUID uuid.UUID) error {
    	pnd, err := c.pndStore.Get(store.Query{ID: pndUUID})
    	if err != nil {
    		return err
    	}
    
    	sbis, err := pnd.GetSBIs()
    	if err != nil {
    		return err
    	}
    	for _, sbi := range sbis {
    		err = pnd.RemoveSbi(sbi.ID())
    		if err != nil {
    			return err
    		}
    	}
    
    	networkElements := pnd.NetworkElements()
    	for _, networkElement := range networkElements {
    		err = pnd.RemoveNetworkElement(networkElement.ID())
    		if err != nil {
    			return err
    		}
    	}
    
    	return nil
    }
    
    func (c ConfigurationManagementServer) createElementsFromSDNConfig(sdnConfig *loadedSDNConfig, pndUUID uuid.UUID) error {
    	err := c.createTopology(sdnConfig)
    	if err != nil {
    		return err
    	}
    
    	err = c.createNetworkElementsAndSBIs(sdnConfig, pndUUID)
    	if err != nil {
    		return err
    	}
    
    	return nil
    }
    
    func (c ConfigurationManagementServer) createTopology(sdnConfig *loadedSDNConfig) error {
    	for _, inputNode := range sdnConfig.Nodes {
    		node := nodes.Node{
    			ID:   inputNode.ID,
    			Name: inputNode.Name,
    		}
    		_, err := c.nodeService.EnsureExists(node)
    		if err != nil {
    			return err
    		}
    	}
    
    	for _, inputPort := range sdnConfig.Ports {
    		port := ports.Port{
    			ID:            inputPort.ID,
    			Name:          inputPort.Name,
    			Configuration: inputPort.Configuration,
    		}
    		_, err := c.portService.EnsureExists(port)
    		if err != nil {
    			return err
    		}
    	}
    
    	for _, inputPort := range sdnConfig.Links {
    		sourceNode, err := c.nodeService.Get(store.Query{ID: inputPort.SourceNode.ID})
    		if err != nil {
    			return err
    		}
    		targetNode, err := c.nodeService.Get(store.Query{ID: inputPort.TargetNode.ID})
    		if err != nil {
    			return err
    		}
    		sourcePort, err := c.portService.Get(store.Query{ID: inputPort.SourcePort.ID})
    		if err != nil {
    			return err
    		}
    		targetPort, err := c.portService.Get(store.Query{ID: inputPort.TargetPort.ID})
    		if err != nil {
    			return err
    		}
    		link := links.Link{
    			ID:         inputPort.ID,
    			Name:       inputPort.Name,
    			SourceNode: sourceNode,
    			TargetNode: targetNode,
    			SourcePort: sourcePort,
    			TargetPort: targetPort,
    		}
    		err = c.topologyService.AddLink(link)
    		if err != nil {
    			return err
    		}
    	}
    
    	return nil
    }
    
    func (c ConfigurationManagementServer) createNetworkElementsAndSBIs(sdnConfig *loadedSDNConfig, pndUUID uuid.UUID) error {
    	pnd, err := c.pndStore.Get(store.Query{ID: pndUUID})
    	if err != nil {
    		return err
    	}
    
    	for _, inputSBI := range sdnConfig.Sbis {
    		sbi, err := nucleus.NewSBI(inputSBI.Type, uuid.MustParse(inputSBI.ID))
    		if err != nil {
    			return err
    		}
    		err = pnd.AddSbi(sbi)
    		if err != nil {
    			return err
    		}
    	}
    
    	for _, inputNetworkElement := range sdnConfig.NetworkElements {
    		transportOption := tpb.TransportOption{
    			Address:  inputNetworkElement.TransportAddress,
    			Username: inputNetworkElement.TransportUsername,
    			Password: inputNetworkElement.TransportPassword,
    			TransportOption: &tpb.TransportOption_GnmiTransportOption{
    				GnmiTransportOption: &tpb.GnmiTransportOption{},
    			},
    			Type: spb.Type_TYPE_OPENCONFIG,
    		}
    		_, err := pnd.AddNetworkElement(
    			inputNetworkElement.Name,
    			&transportOption,
    			uuid.MustParse(inputNetworkElement.SBI),
    			uuid.MustParse(inputNetworkElement.ID),
    		)
    		if err != nil {
    			return err
    		}
    
    		networkelement, err := pnd.GetNetworkElement(inputNetworkElement.ID)
    		if err != nil {
    			return err
    		}
    
    		err = pnd.UpdateNetworkElement(networkelement.ID(), inputNetworkElement.Model)
    		if err != nil {
    			return err
    		}
    	}
    	return nil
    }