diff --git a/src/switch/cmd/l2sm-add-port/main.go b/src/switch/cmd/l2sm-add-port/main.go
new file mode 100644
index 0000000000000000000000000000000000000000..63a247c44a15def415f678cd1329260d94b5856c
--- /dev/null
+++ b/src/switch/cmd/l2sm-add-port/main.go
@@ -0,0 +1,98 @@
+package main
+
+import (
+	"encoding/json"
+	"errors"
+	"flag"
+	"fmt"
+	"io/ioutil"
+	"ovs-switch/pkg/ovs"
+)
+
+type Node struct {
+	Name          string `json:"name"`
+	NodeIP        string `json:"nodeIP"`
+	NeighborNodes []Node `json:"neighborNodes"`
+}
+
+// Script that takes two required arguments:
+// the first one is the name in the cluster of the node where the script is running
+// the second one is the path to the configuration file, in reference to the code.
+func main() {
+
+	portName, err := takeArguments()
+
+	bridge := ovs.FromName("brtun")
+
+	if err != nil {
+		fmt.Println("Error with the arguments. Error:", err)
+		return
+	}
+
+	bridge.AddPort(portName)
+
+	if err != nil {
+		fmt.Println("Port not added: ", err)
+		return
+	}
+}
+
+func takeArguments() (string, error) {
+
+	portName := flag.String("port_name", "", "port you want to add. Required.")
+
+	flag.Parse()
+
+	if *portName == "" {
+		return "", errors.New("port name is not defined")
+
+	}
+
+	return *portName, nil
+}
+
+func createTopology(bridge ovs.Bridge, nodes []Node, nodeName string) error {
+
+	// Search for the corresponding node in the configuration, according to the first passed parameter.
+	// Once the node is found, create a bridge for every neighbour node defined.
+	// The bridge is created with the nodeIp and neighborNodeIP and VNI. The VNI is generated in the l2sm-controller thats why its set to 'flow'.
+	for _, node := range nodes {
+		if node.Name == nodeName {
+			//nodeIP := strings.TrimSpace(node.NodeIP)
+			connectToNeighbors(bridge, node)
+		}
+	}
+	return nil
+}
+
+func readFile(configDir string) ([]Node, error) {
+	/// Read file and save in memory the JSON info
+	data, err := ioutil.ReadFile(configDir)
+	if err != nil {
+		fmt.Println("No input file was found.", err)
+		return nil, err
+	}
+
+	var nodes []Node
+	err = json.Unmarshal(data, &nodes)
+	if err != nil {
+		return nil, err
+	}
+
+	return nodes, nil
+
+}
+
+func connectToNeighbors(bridge ovs.Bridge, node Node) error {
+	for vxlanNumber, neighbor := range node.NeighborNodes {
+		vxlanId := fmt.Sprintf("vxlan%d", vxlanNumber)
+		err := bridge.CreateVxlan(ovs.Vxlan{VxlanId: vxlanId, LocalIp: node.NodeIP, RemoteIp: neighbor.NodeIP, UdpPort: "7000"})
+
+		if err != nil {
+			return fmt.Errorf("could not create vxlan between node %s and node %s", node.Name, neighbor)
+		} else {
+			fmt.Printf("Created vxlan between node %s and node %s.\n", node.Name, neighbor)
+		}
+	}
+	return nil
+}
diff --git a/src/switch/cmd/l2sm-init/main.go b/src/switch/cmd/l2sm-init/main.go
index 30cded3a1e219327e1879dfac6e2d4dfc28015fc..5c9a8212a804f34ad54314ed55f7ee85ea776e6c 100644
--- a/src/switch/cmd/l2sm-init/main.go
+++ b/src/switch/cmd/l2sm-init/main.go
@@ -5,6 +5,7 @@ import (
 	"flag"
 	"fmt"
 	"os/exec"
+	"ovs-switch/pkg/ovs"
 	"regexp"
 )
 
@@ -21,9 +22,11 @@ func main() {
 	}
 
 	fmt.Println("Initializing switch, connected to controller: ", controllerIP)
-	err = initializeSwitch(controllerIP)
+
+	bridge, err := initializeSwitch(controllerIP)
 
 	if err != nil {
+
 		fmt.Println("Could not initialize switch. Error:", err)
 		return
 	}
@@ -33,12 +36,11 @@ func main() {
 	// Set all virtual interfaces up, and connect them to the tunnel bridge:
 	for i := 1; i <= vethNumber; i++ {
 		veth := fmt.Sprintf("net%d", i)
-		cmd := exec.Command("ip", "link", "set", veth, "up") // i.e: ip link set veth1 up
-		if err := cmd.Run(); err != nil {
+		if err := bridge.AddPort(veth); err != nil {
 			fmt.Println("Error:", err)
 		}
-		exec.Command("ovs-vsctl", "add-port", "brtun", veth).Run() // i.e: ovs-vsctl add-port brtun veth1
 	}
+	fmt.Printf("Switch initialized, current state: ", bridge)
 }
 
 func takeArguments() (int, string, error) {
@@ -56,7 +58,7 @@ func takeArguments() (int, string, error) {
 	return *vethNumber, *controllerIP, nil
 }
 
-func initializeSwitch(controllerIP string) error {
+func initializeSwitch(controllerIP string) (ovs.Bridge, error) {
 
 	re := regexp.MustCompile(`\b(?:[0-9]{1,3}\.){3}[0-9]{1,3}\b`)
 	if !re.MatchString(controllerIP) {
@@ -64,32 +66,9 @@ func initializeSwitch(controllerIP string) error {
 		controllerIP = re.FindString(string(out))
 	}
 
-	var err error
-
-	err = exec.Command("ovs-vsctl", "add-br", "brtun").Run()
+	controller := fmt.Sprintf("tcp:%s:6633", controllerIP)
 
-	if err != nil {
-		return errors.New("could not create brtun interface")
-	}
-
-	err = exec.Command("ip", "link", "set", "brtun", "up").Run()
-
-	if err != nil {
-		return errors.New("could not set brtun interface up")
-	}
+	bridge, err := ovs.NewBridge(ovs.Bridge{Name: "brtun", Controller: controller, Protocol: "OpenFlow13"})
 
-	err = exec.Command("ovs-vsctl", "set", "bridge", "brtun", "protocols=OpenFlow13").Run()
-
-	if err != nil {
-		return errors.New("could not set brtun messaing protocol to OpenFlow13")
-	}
-
-	target := fmt.Sprintf("tcp:%s:6633", controllerIP)
-
-	err = exec.Command("ovs-vsctl", "set-controller", "brtun", target).Run()
-
-	if err != nil {
-		return errors.New("could not connect to controller")
-	}
-	return nil
+	return bridge, err
 }
diff --git a/src/switch/cmd/l2sm-vxlans/main.go b/src/switch/cmd/l2sm-vxlans/main.go
index 029647a90199e12bbd9a768caa388b0f96d0e04f..3c3434dcafdf3b73d0607823155dde5c44cb2229 100644
--- a/src/switch/cmd/l2sm-vxlans/main.go
+++ b/src/switch/cmd/l2sm-vxlans/main.go
@@ -7,14 +7,13 @@ import (
 	"fmt"
 	"io/ioutil"
 	"os"
-	"os/exec"
-	"strings"
+	"ovs-switch/pkg/ovs"
 )
 
 type Node struct {
-	Name          string   `json:"name"`
-	NodeIP        string   `json:"nodeIP"`
-	NeighborNodes []string `json:"neighborNodes"`
+	Name          string `json:"name"`
+	NodeIP        string `json:"nodeIP"`
+	NeighborNodes []Node `json:"neighborNodes"`
 }
 
 // Script that takes two required arguments:
@@ -22,14 +21,24 @@ type Node struct {
 // the second one is the path to the configuration file, in reference to the code.
 func main() {
 
-	configDir, nodeName, err := takeArguments()
+	configDir, nodeName, fileType, err := takeArguments()
+
+	bridge := ovs.FromName("brtun")
 
 	if err != nil {
 		fmt.Println("Error with the arguments. Error:", err)
 		return
 	}
 
-	err = createVxlans(configDir, nodeName)
+	nodes, err := readFile(configDir)
+
+	switch fileType {
+	case "topology":
+		err = createTopology(bridge, nodes, nodeName)
+
+	case "neighbors":
+		err = connectToNeighbors(bridge, nodes[0])
+	}
 
 	if err != nil {
 		fmt.Println("Vxlans not created: ", err)
@@ -37,73 +46,68 @@ func main() {
 	}
 }
 
-func takeArguments() (string, string, error) {
+func takeArguments() (string, string, string, error) {
 	configDir := os.Args[len(os.Args)-1]
 
 	nodeName := flag.String("node_name", "", "name of the node the script is executed in. Required.")
 
+	fileType := flag.String("file_type", "topology", "type of filed passed as an argument. Can either be topology or neighbors. Default: topology.")
+
 	flag.Parse()
 
 	switch {
 	case *nodeName == "":
-		return "", "", errors.New("node name is not defined")
+		return "", "", "", errors.New("node name is not defined")
+	case *fileType != "topology" || *fileType != "neighbors":
+		return "", "", "", errors.New("file type not supported. Available types: 'topology' and 'neighbors'")
 	case configDir == "":
-		return "", "", errors.New("config directory is not defined")
+		return "", "", "", errors.New("config directory is not defined")
 	}
 
-	return configDir, *nodeName, nil
+	return configDir, *nodeName, *fileType, nil
 }
 
-func createVxlans(configDir, nodeName string) error {
+func createTopology(bridge ovs.Bridge, nodes []Node, nodeName string) error {
+
+	// Search for the corresponding node in the configuration, according to the first passed parameter.
+	// Once the node is found, create a bridge for every neighbour node defined.
+	// The bridge is created with the nodeIp and neighborNodeIP and VNI. The VNI is generated in the l2sm-controller thats why its set to 'flow'.
+	for _, node := range nodes {
+		if node.Name == nodeName {
+			//nodeIP := strings.TrimSpace(node.NodeIP)
+			connectToNeighbors(bridge, node)
+		}
+	}
+	return nil
+}
 
+func readFile(configDir string) ([]Node, error) {
 	/// Read file and save in memory the JSON info
 	data, err := ioutil.ReadFile(configDir)
 	if err != nil {
 		fmt.Println("No input file was found.", err)
-		return err
+		return nil, err
 	}
 
 	var nodes []Node
 	err = json.Unmarshal(data, &nodes)
 	if err != nil {
-		return err
+		return nil, err
 	}
 
-	// Search for the corresponding node in the configuration, according to the first passed parameter.
-	// Once the node is found, create a bridge for every neighbour node defined.
-	// The bridge is created with the nodeIp and neighborNodeIP and VNI. The VNI is generated in the l2sm-controller thats why its set to 'flow'.
-	for _, node := range nodes {
-		if node.Name == nodeName {
-			nodeIP := strings.TrimSpace(node.NodeIP)
-			for _, neighbor := range node.NeighborNodes {
-				vxlanNumber := 1
-				for _, n := range nodes {
-					if n.Name == neighbor {
-						neighborIP := strings.TrimSpace(n.NodeIP)
-						commandArgs := []string{
-							"add-port",
-							"brtun",
-							fmt.Sprintf("vxlan%d", vxlanNumber),
-							"--",
-							"set", "interface",
-							fmt.Sprintf("vxlan%d", vxlanNumber),
-							"type=vxlan",
-							"options:key=flow",
-							fmt.Sprintf("options:remote_ip=%s", neighborIP),
-							fmt.Sprintf("options:local_ip=%s", nodeIP),
-							"options:dst_port=7000",
-						}
-						_, err := exec.Command("ovs-vsctl", commandArgs...).Output()
-						if err != nil {
-							return fmt.Errorf("could not create vxlan between node %s and node %s", node.Name, neighbor)
-						} else {
-							fmt.Printf("Created vxlan between node %s and node %s.\n", node.Name, neighbor)
-						}
-					}
-					vxlanNumber++
-				}
-
-			}
+	return nodes, nil
+
+}
+
+func connectToNeighbors(bridge ovs.Bridge, node Node) error {
+	for vxlanNumber, neighbor := range node.NeighborNodes {
+		vxlanId := fmt.Sprintf("vxlan%d", vxlanNumber)
+		err := bridge.CreateVxlan(ovs.Vxlan{VxlanId: vxlanId, LocalIp: node.NodeIP, RemoteIp: neighbor.NodeIP, UdpPort: "7000"})
+
+		if err != nil {
+			return fmt.Errorf("could not create vxlan between node %s and node %s", node.Name, neighbor)
+		} else {
+			fmt.Printf("Created vxlan between node %s and node %s.\n", node.Name, neighbor)
 		}
 	}
 	return nil
diff --git a/src/switch/go.mod b/src/switch/go.mod
index b16f0d7c546947c98f70a639319ede91c44120c8..3cd7acc358fbf707802eb628bdb4db0babfd8f22 100644
--- a/src/switch/go.mod
+++ b/src/switch/go.mod
@@ -1,3 +1,11 @@
-module app
+module ovs-switch
 
-go 1.18
+go 1.21.7
+
+require github.com/eclipse/paho.mqtt.golang v1.4.3
+
+require (
+	github.com/gorilla/websocket v1.5.0 // indirect
+	golang.org/x/net v0.8.0 // indirect
+	golang.org/x/sync v0.1.0 // indirect
+)
diff --git a/src/switch/go.sum b/src/switch/go.sum
new file mode 100644
index 0000000000000000000000000000000000000000..cf663d6440cb5b5e64ac086ec4dda8a5e4a7a2d8
--- /dev/null
+++ b/src/switch/go.sum
@@ -0,0 +1,8 @@
+github.com/eclipse/paho.mqtt.golang v1.4.3 h1:2kwcUGn8seMUfWndX0hGbvH8r7crgcJguQNCyp70xik=
+github.com/eclipse/paho.mqtt.golang v1.4.3/go.mod h1:CSYvoAlsMkhYOXh/oKyxa8EcBci6dVkLCbo5tTC1RIE=
+github.com/gorilla/websocket v1.5.0 h1:PPwGk2jz7EePpoHN/+ClbZu8SPxiqlu12wZP/3sWmnc=
+github.com/gorilla/websocket v1.5.0/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE=
+golang.org/x/net v0.8.0 h1:Zrh2ngAOFYneWTAIAPethzeaQLuHwhuBkuV6ZiRnUaQ=
+golang.org/x/net v0.8.0/go.mod h1:QVkue5JL9kW//ek3r6jTKnTFis1tRmNAW2P1shuFdJc=
+golang.org/x/sync v0.1.0 h1:wsuoTGHzEhffawBOhz5CYhcrV4IdKZbEyZjBMuTp12o=
+golang.org/x/sync v0.1.0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
diff --git a/src/switch/pkg/mqtt_sub/mqtt_sub.go b/src/switch/pkg/mqtt_sub/mqtt_sub.go
new file mode 100644
index 0000000000000000000000000000000000000000..4d05eb3f92e4bfbeeada32ad9218048a337eec92
--- /dev/null
+++ b/src/switch/pkg/mqtt_sub/mqtt_sub.go
@@ -0,0 +1,76 @@
+package main
+
+import (
+	"encoding/json"
+	"log"
+
+	ovs "ovs-switch/pkg/ovs"
+
+	mqtt "github.com/eclipse/paho.mqtt.golang"
+)
+
+type Command struct {
+	Action string     `json:"action"`
+	Bridge ovs.Bridge `json:"bridge,omitempty"`
+	Port   string     `json:"port,omitempty"`
+	Vxlan  ovs.Vxlan  `json:"vxlan,omitempty"`
+}
+
+var broker = "tcp://mqtt_broker_ip:1883"
+var topic = "ovs/commands"
+
+func LaunchSubscriber() {
+	opts := mqtt.NewClientOptions().AddBroker(broker).SetClientID("go_mqtt_client")
+	opts.SetDefaultPublishHandler(func(client mqtt.Client, msg mqtt.Message) {
+		handleMessage(msg.Payload())
+	})
+
+	client := mqtt.NewClient(opts)
+	if token := client.Connect(); token.Wait() && token.Error() != nil {
+		log.Fatal(token.Error())
+	}
+
+	if token := client.Subscribe(topic, 0, nil); token.Wait() && token.Error() != nil {
+		log.Fatal(token.Error())
+	}
+
+	// Block main routine forever
+	select {}
+}
+
+func handleMessage(payload []byte) {
+	var cmd Command
+	err := json.Unmarshal(payload, &cmd)
+	if err != nil {
+		log.Printf("Error decoding command: %v", err)
+		return
+	}
+
+	switch cmd.Action {
+	case "create_bridge":
+		bridge, err := ovs.NewBridge(cmd.Bridge)
+		if err != nil {
+			log.Printf("Error creating bridge: %v", err)
+		} else {
+			log.Printf("Bridge created: %+v", bridge)
+		}
+	case "add_port":
+		bridge := ovs.FromName(cmd.Bridge.Name)
+		err := bridge.AddPort(cmd.Port)
+		if err != nil {
+			log.Printf("Error adding port: %v", err)
+		} else {
+			log.Printf("Port added: %s", cmd.Port)
+		}
+	case "create_vxlan":
+		bridge := ovs.FromName(cmd.Bridge.Name)
+		err := bridge.CreateVxlan(cmd.Vxlan)
+		if err != nil {
+			log.Printf("Error creating VXLAN: %v", err)
+		} else {
+			log.Printf("VXLAN created: %+v", cmd.Vxlan)
+		}
+	default:
+		log.Printf("Unknown action: %s", cmd.Action)
+	}
+}
diff --git a/src/switch/pkg/ovs/vsctl.go b/src/switch/pkg/ovs/vsctl.go
new file mode 100644
index 0000000000000000000000000000000000000000..8df0a16597b480d4607052b486dc87d1a2a09c32
--- /dev/null
+++ b/src/switch/pkg/ovs/vsctl.go
@@ -0,0 +1,140 @@
+package ovs
+
+import (
+	"bytes"
+	"errors"
+	"fmt"
+	"os/exec"
+	"strings"
+)
+
+// TODO: Abstract exec client as a separate entity, that doesnt use CLI. The following presented hardcoded way is not clean.
+
+type Port struct {
+	Name   string
+	Status string
+}
+
+type Bridge struct {
+	Controller string
+	Name       string
+	Protocol   string
+	Ports      []Port
+}
+
+type Vxlan struct {
+	VxlanId  string
+	LocalIp  string
+	RemoteIp string
+	UdpPort  string
+}
+
+func FromName(bridgeName string) Bridge {
+
+	bridge := Bridge{Name: bridgeName}
+
+	bridge.getPorts()
+
+	return bridge
+}
+
+func NewBridge(bridgeConf Bridge) (Bridge, error) {
+
+	var err error
+
+	bridge := Bridge{}
+
+	err = exec.Command("ovs-vsctl", "add-br", bridgeConf.Name).Run()
+
+	if err != nil {
+		return bridge, errors.New("could not create brtun interface")
+	}
+
+	bridge.Name = bridgeConf.Name
+
+	err = exec.Command("ip", "link", "set", bridge.Name, "up").Run()
+
+	if err != nil {
+		return bridge, errors.New("could not set brtun interface up")
+	}
+
+	protocolString := fmt.Sprintf("protocols=%s", bridgeConf.Protocol)
+	err = exec.Command("ovs-vsctl", "set", "bridge", "brtun", protocolString).Run()
+
+	if err != nil {
+		return bridge, errors.New("could not set brtun messaing protocol to OpenFlow13")
+	}
+
+	bridge.Protocol = bridgeConf.Protocol
+
+	err = exec.Command("ovs-vsctl", "set-controller", bridge.Name, bridgeConf.Controller).Run()
+
+	if err != nil {
+		return bridge, errors.New("could not connect to controller")
+	}
+
+	bridge.Controller = bridgeConf.Name
+
+	return bridge, nil
+}
+
+func (bridge *Bridge) CreateVxlan(vxlan Vxlan) error {
+
+	commandArgs := []string{
+		"add-port",
+		bridge.Name,
+		vxlan.VxlanId,
+		"--",
+		"set", "interface",
+		vxlan.VxlanId,
+		"type=vxlan",
+		"options:key=flow",
+		fmt.Sprintf("options:remote_ip=%s", vxlan.RemoteIp),
+		fmt.Sprintf("options:local_ip=%s", vxlan.LocalIp),
+		fmt.Sprintf("options:dst_port=%s", vxlan.UdpPort),
+	}
+	_, err := exec.Command("ovs-vsctl", commandArgs...).Output()
+
+	return err
+
+}
+
+func (bridge *Bridge) AddPort(portName string) error {
+
+	cmd := exec.Command("ip", "link", "set", portName, "up") // i.e: ip link set veth1 up
+	if err := cmd.Run(); err != nil {
+		return err
+	}
+	exec.Command("ovs-vsctl", "add-port", bridge.Name, portName).Run() // i.e: ovs-vsctl add-port brtun veth1
+	bridge.Ports = append(bridge.Ports, Port{Name: portName, Status: "UP"})
+	return nil
+}
+
+func (bridge *Bridge) getPorts() error {
+	// Executes the ovs-vsctl command to list ports on the bridge
+	cmd := exec.Command("ovs-vsctl", "list-ports", bridge.Name)
+	var out bytes.Buffer
+	cmd.Stdout = &out
+	err := cmd.Run()
+	if err != nil {
+		return err
+	}
+
+	// Split the output by lines for each port name
+	portNames := strings.Split(out.String(), "\n")
+	for _, portName := range portNames {
+		if portName == "" {
+			continue
+		}
+		// TODO:, retrieve more details for each port; here we just set the name
+		port := Port{Name: portName}
+
+		// Retrieve status
+		// cmd = exec.Command("ovs-vsctl", "get", "Interface", portName, "status")
+
+		// Add the port to the Ports slice
+		bridge.Ports = append(bridge.Ports, port)
+	}
+
+	return nil
+}