Skip to content
Snippets Groups Projects
topologyStore.go 5.14 KiB
Newer Older
  • Learn to ignore specific revisions
  • package topology
    
    import (
    	"fmt"
    	"log"
    
    	"code.fbi.h-da.de/danet/gosdn/controller/nucleus/database"
    	"code.fbi.h-da.de/danet/gosdn/controller/nucleus/errors"
    	"code.fbi.h-da.de/danet/gosdn/controller/topology/links"
    
    	"code.fbi.h-da.de/danet/gosdn/controller/topology/store"
    
    	"github.com/google/uuid"
    	"go.mongodb.org/mongo-driver/bson"
    	"go.mongodb.org/mongo-driver/bson/primitive"
    	"go.mongodb.org/mongo-driver/mongo/options"
    )
    
    // Store defines a Topology store interface
    type Store interface {
    
    André Sterba's avatar
    André Sterba committed
    	Add(links.Link) error
    	Update(links.Link) error
    	Delete(links.Link) error
    
    	Get(store.Query) (links.Link, error)
    	GetAll() ([]links.Link, error)
    }
    
    // DatabaseTopologyStore is a database store for the topology
    type DatabaseTopologyStore struct {
    	storeName string
    }
    
    // NewDatabaseTopologyStore returns a TopologyStore
    func NewDatabaseTopologyStore() Store {
    	return &DatabaseTopologyStore{
    		storeName: fmt.Sprintf("topology-store.json"),
    	}
    }
    
    // Get takes a link's UUID or name and returns the link.
    func (s *DatabaseTopologyStore) Get(query store.Query) (links.Link, error) {
    	var loadedTopology links.Link
    
    	if query.ID.String() != "" {
    
    André Sterba's avatar
    André Sterba committed
    		loadedTopology, err := s.getByID(query.ID)
    
    André Sterba's avatar
    André Sterba committed
    			return loadedTopology, errors.ErrCouldNotFind{ID: query.ID, Name: query.Name}
    
    André Sterba's avatar
    André Sterba committed
    		return loadedTopology, nil
    
    	}
    
    	loadedTopology, err := s.getByName(query.Name)
    	if err != nil {
    
    André Sterba's avatar
    André Sterba committed
    		return loadedTopology, errors.ErrCouldNotFind{ID: query.ID, Name: query.Name}
    
    André Sterba's avatar
    André Sterba committed
    func (s *DatabaseTopologyStore) getByID(idOfTopology uuid.UUID) (links.Link, error) {
    
    	var loadedTopology links.Link
    
    	client, ctx, cancel := database.GetMongoConnection()
    	defer cancel()
    	defer client.Disconnect(ctx)
    
    	db := client.Database(database.DatabaseName)
    	collection := db.Collection(s.storeName)
    
    André Sterba's avatar
    André Sterba committed
    	result := collection.FindOne(ctx, bson.D{primitive.E{Key: "_id", Value: idOfTopology.String()}})
    
    	if result == nil {
    
    André Sterba's avatar
    André Sterba committed
    		return loadedTopology, errors.ErrCouldNotFind{ID: idOfTopology}
    
    	}
    
    	err := result.Decode(&loadedTopology)
    	if err != nil {
    		log.Printf("Failed marshalling %v", err)
    
    André Sterba's avatar
    André Sterba committed
    		return loadedTopology, errors.ErrCouldNotFind{ID: idOfTopology}
    
    André Sterba's avatar
    André Sterba committed
    func (s *DatabaseTopologyStore) getByName(nameOfTopology string) (links.Link, error) {
    
    	var loadedTopology links.Link
    
    	client, ctx, cancel := database.GetMongoConnection()
    	defer cancel()
    	defer client.Disconnect(ctx)
    
    	db := client.Database(database.DatabaseName)
    	collection := db.Collection(s.storeName)
    
    André Sterba's avatar
    André Sterba committed
    	result := collection.FindOne(ctx, bson.D{primitive.E{Key: "name", Value: nameOfTopology}})
    
    	if result == nil {
    
    André Sterba's avatar
    André Sterba committed
    		return loadedTopology, errors.ErrCouldNotFind{Name: nameOfTopology}
    
    	}
    
    	err := result.Decode(&loadedTopology)
    	if err != nil {
    		log.Printf("Failed marshalling %v", err)
    
    André Sterba's avatar
    André Sterba committed
    		return loadedTopology, errors.ErrCouldNotFind{Name: nameOfTopology}
    
    	}
    
    	return loadedTopology, nil
    }
    
    // GetAll returns all stored links.
    func (s *DatabaseTopologyStore) GetAll() ([]links.Link, error) {
    	var loadedTopology []links.Link
    
    	client, ctx, cancel := database.GetMongoConnection()
    	defer cancel()
    	defer client.Disconnect(ctx)
    	db := client.Database(database.DatabaseName)
    	collection := db.Collection(s.storeName)
    
    	cursor, err := collection.Find(ctx, bson.D{})
    	if err != nil {
    		return loadedTopology, err
    	}
    	defer cursor.Close(ctx)
    
    	err = cursor.All(ctx, &loadedTopology)
    	if err != nil {
    		log.Printf("Failed marshalling %v", err)
    
    
    		return loadedTopology, errors.ErrCouldNotMarshall{Type: loadedTopology, Err: err}
    
    André Sterba's avatar
    André Sterba committed
    // Add adds a link to the link store.
    func (s *DatabaseTopologyStore) Add(link links.Link) error {
    
    	client, ctx, cancel := database.GetMongoConnection()
    	defer cancel()
    	defer client.Disconnect(ctx)
    
    	_, err := client.Database(database.DatabaseName).
    		Collection(s.storeName).
    		InsertOne(ctx, link)
    	if err != nil {
    		log.Printf("Could not add Link: %v", err)
    
    		return errors.ErrCouldNotCreate{Identifier: link.ID, Type: link, Err: err}
    
    André Sterba's avatar
    André Sterba committed
    // Update updates a existing link.
    func (s *DatabaseTopologyStore) Update(linkToUpdate links.Link) error {
    
    	var updatedLink links.Link
    
    	client, ctx, cancel := database.GetMongoConnection()
    	defer cancel()
    	defer client.Disconnect(ctx)
    
    	update := bson.D{primitive.E{Key: "$set", Value: linkToUpdate}}
    
    	upsert := false
    	after := options.After
    	opt := options.FindOneAndUpdateOptions{
    		Upsert:         &upsert,
    		ReturnDocument: &after,
    	}
    
    	err := client.Database(database.DatabaseName).
    		Collection(s.storeName).
    		FindOneAndUpdate(
    			ctx, bson.M{"_id": linkToUpdate.ID.String()}, update, &opt).
    		Decode(&updatedLink)
    	if err != nil {
    		log.Printf("Could not update link: %v", err)
    
    
    		return errors.ErrCouldNotUpdate{Identifier: linkToUpdate.ID, Type: linkToUpdate, Err: err}
    
    André Sterba's avatar
    André Sterba committed
    // Delete deletes a link from the link store.
    func (s *DatabaseTopologyStore) Delete(linkToDelete links.Link) error {
    
    	client, ctx, cancel := database.GetMongoConnection()
    	defer cancel()
    	defer client.Disconnect(ctx)
    
    	db := client.Database(database.DatabaseName)
    	collection := db.Collection(s.storeName)
    	_, err := collection.DeleteOne(ctx, bson.D{primitive.E{Key: linkToDelete.ID.String()}})
    	if err != nil {
    		return err
    	}
    
    	return nil
    }