Opendaylight does not see the topology

64 Views Asked by At

I have a big issue that I can not understand. I have two programs, one with Java 17 using Spring Boot 3.1.2 and the other using Python 2.7 on Ubuntu 14.04, and the VM is working on VMware. Unfortunately, I can not change the Python and the Ubuntu, so I must work with them. My main problem is stability. I am trying to understand why one day it was working perfectly, and the other day I had a lot of issues.

I am using Miniet for the first machine that has the issue, and for the other machine for the controller, I am using Opendaylight.

I will start with the Java code and then with the Python code.

The Java program:

Write Topology class:

public class Write_Topology {

    public String write_topology(HashMap<Integer,Node> nodes,
                                 String Topology_name ){
        ObjectMapper mapper = new ObjectMapper();
//        Path path = Paths.get(System.getProperty("userhome"), "api", Topology_name + ".json");
//        String path = System.getProperty("userhome") + "/api/" + Topology_name + ".json";
        String path = "/home/mininet/api/" + Topology_name + ".json";
        try {
            File file = new File(path);
            FileWriter fileWriter = new FileWriter(path);
            fileWriter.write(mapper.writeValueAsString(nodes));
            fileWriter.close();
        } catch (IOException e) {
            throw new RuntimeException(e);
        }
        return path;
    }
}

The service class:

@Service
public class Create_Topology_Service {

    private final Create_Topology create_topology = new Create_Topology();
    private final Check_host_range checkHostRange = new Check_host_range();
    private final Write_Topology writeTopology = new Write_Topology();

    public Optional create_topology_service_method(Create_Topology_Model createTopologyModel){

        HashMap<Integer,Node> nodes;

        try {
            if(checkHostRange.check_host_range_method(
                    createTopologyModel.ipBase,
                    createTopologyModel.number_of_hosts_per_switch,
                    createTopologyModel.number_of_switches
                    )) {

                nodes = create_topology
                        .create_topology(createTopologyModel.number_of_switches,
                                createTopologyModel.number_of_hosts_per_switch);

                String path = writeTopology.write_topology(nodes, createTopologyModel.topology_name);

                String[] cmd = getStrings(createTopologyModel,path);

                IndexController.process = Runtime.getRuntime().exec(cmd);
                System.out.println(IndexController.process.pid());
                Read_Output read_output = new Read_Output();
                read_output.read_output();

            }else{
                return Optional.of(getResponse());
            }

        }catch (Exception e){
            throw new RuntimeException(e);
        }
        return Optional.of(nodes);
    }

    private String[] getStrings(Create_Topology_Model createTopologyModel, String path) {

        if(createTopologyModel.controller_port!=0) {
            String command = "nohup sudo python ~/api/API_TEST.py " +
                    path +
                    " " +
                    createTopologyModel.ipBase +
                    " " +
                    createTopologyModel.controller_ip +
                    " " +
                    createTopologyModel.controller_port +
                    " " +
                    createTopologyModel.switches_name_prefix +
                    " " +
                    createTopologyModel.hosts_name_prefix +
                    " " +
                    createTopologyModel.number_of_switches +
                    " " +
                    createTopologyModel.number_of_hosts_per_switch +
                    " " +
                    "&";

            System.out.println(command);

            return new String[]{"/bin/bash", "-c", command};
        }

        String command = "nohup sudo python ~/api/API_TEST.py " +
                path +
                " " +
                createTopologyModel.ipBase +
                " " +
                createTopologyModel.controller_ip +
                " " +
                6633 +
                createTopologyModel.switches_name_prefix +
                " " +
                createTopologyModel.hosts_name_prefix +
                " " +
                createTopologyModel.number_of_switches +
                " " +
                createTopologyModel.number_of_hosts_per_switch +
                " " +
                "&";

        System.out.println(command);

        return new String[]{"/bin/bash", "-c", command};
    }

    private List<String> getResponse(){
        List<String> response = new ArrayList<>();
        response.add("Too many hosts the specified base Ip");
        response.add("1) You can reduce number of hosts per device");
        response.add("2) You can increase the number of hosts in the network");

     return response;
    }
}

The endpoint:

@PostMapping(path = {"CreateTopology"})
public Optional create_topology(@RequestBody Create_Topology_Model createTopologyModel) {
    return this.createTopologyService.create_topology_service_method(createTopologyModel);
}

The python code:

The main file:

#!/bin/python

import sys
import json
from Create_Topology2 import Create_Topology

if __name__ == '__main__':
    
    
    print(sys.argv[1])
    print(sys.argv[2],
        sys.argv[3],
        int(sys.argv[4]),
        sys.argv[5],
        sys.argv[6])
        
    with open(sys.argv[1], 'r') as f:
        data = json.load(f)
        print("2: ",sys.argv[2],
            "3: ",sys.argv[3],
            "4: ",int(sys.argv[4]),
            "5: ",sys.argv[5],
            "6: ",sys.argv[6],
            "data: ",data)
        
        topology = Create_Topology(sys.argv[2],
                                    sys.argv[3],
                                    int(sys.argv[4]),
                                    sys.argv[5],
                                    sys.argv[6],
                                    data)
        topology.create_topology()

The Create_Topology:

#!/bin/python


from mininet.net import Mininet
from mininet.node import Controller, RemoteController, OVSController
from mininet.node import CPULimitedHost, Host, Node
from mininet.node import OVSKernelSwitch, UserSwitch
from mininet.node import IVSSwitch
from mininet.cli import CLI
from mininet.log import setLogLevel, info
from mininet.link import TCLink, Intf

from Random_nodes_service import random_nodes_service


class Create_Topology():
       
    def __init__(self,
                 ipBase,
                 controller_ip,
                 controller_port,
                 switches_name_prefix,
                 hosts_name_prefix,
                 topology
                ):
        self.ipBase = ipBase
        self.controller_ip = controller_ip
        self.controller_port = controller_port
        self.switches_name_prefix = switches_name_prefix
        self.hosts_name_prefix = hosts_name_prefix
        self.topology = topology
        self.copy_ipBase = ipBase
        self.host_number=0
        
        
    def get_host(self):

        ip = self.copy_ipBase.split('/')[0]
        
        a = ip.split('.')
        
        if 254 < (int(a[3]) + 1):
            if 254 < (int(a[2])+1):
                c = int(a[2]) +1
                host_ip = ""
                for b in range(0,1):
                    host_ip += a[b]+"."
                    
                host_ip += str(c) + "." + "0"+"."+"1"
                
                self.copy_ipBase = host_ip
                
                return host_ip
                
            else:
                c = int(a[2]) +1
                host_ip = ""
                for b in range(0,2):
                    host_ip += a[b]+"."
                    
                host_ip += str(c) + "." + "1"
                
                self.copy_ipBase = host_ip
                
                return host_ip
    
        c = int(a[3]) +1
        host_ip = ""
        for b in range(0,3):
            host_ip += a[b]+"."
            
        host_ip += str(c)
        
        self.copy_ipBase = host_ip
        
        return host_ip
        
            
    def create_topology(self):
        
        net = Mininet(topo=None,
                    build=False,
                    ipBase=self.ipBase)
        
        c0 = net.addController(name='c0',
                      controller=RemoteController,
                      ip=self.controller_ip, # variable that I will pass in the 
                      protocol='tcp',
                      port=self.controller_port) # vatiable with the default value of 6633
        

        switches_list = []
        
        for i in self.topology:
            print(i)
            switches_list.append(net.addSwitch((self.switches_name_prefix+str(i)), 
                                               cls=OVSKernelSwitch))
            

        print(switches_list)
        
        a=0
        
        for i in self.topology:
            
            for j in range(self.topology[i]["hosts"]):
                net.addLink(switches_list[int(i)],
                            net.addHost(
                                self.hosts_name_prefix+str(a+j), 
                                cls=Host,
                                ip=self.get_host(),
                                defaultRoute=None))
                a=a+1
            
            for j in self.topology[i]["list"]:
                
                net.addLink(switches_list[int(i)],switches_list[int(j)])
        
        
        info( '*** Starting network\n')
        net.build()
        
        info( '*** Starting controllers\n')
        for controller in net.controllers:
            controller.start()

        info( '*** Starting switches\n')
        for i in switches_list:
            net.get(str(i)).start([c0])

        info( '*** Post configure switches and hosts\n')
        CLI(net)
        net.stop()

Now let me explain a bit the flow:

The Java code will get a request with the details and create the topology. It will write it on a JSON file, pass everything in a command line to the Python script, and do the work of the program I want to run in the background.

The output is:

nohup sudo python ~/api/API_TEST.py /home/mininet/api/topology3.json 10.0.0.0/8 192.168.241.129 6633 sw h 10 10 &
nohup sudo python ~/api/API_TEST.py /home/mininet/api/topology3.json 10.0.0.0/8 192.168.241.129 6633 sw h 10 10 &
nohup sudo python ~/api/API_TEST.py /home/mininet/api/topology3.json 10.0.0.0/8 192.168.241.129 6633 sw h 10 10 &
/home/mininet/api/topology3.json
('10.0.0.0/8', '192.168.241.129', 6633, 'sw', 'h')
('2: ', '10.0.0.0/8', '3: ', '192.168.241.129', '4: ', 6633, '5: ', 'sw', '6: ', 'h', 'data: ', {u'1': {u'hosts': 3, u'list': [3, 4, 6, 7, 8, 9]}, u'0': {u'hosts': 3, u'list': [1, 3, 4, 5, 6, 7, 9]}, u'3': {u'hosts': 8, u'list': [4, 5, 6, 7, 8, 9]}, u'2': {u'hosts': 4, u'list': [3, 5, 6, 8, 9]}, u'5': {u'hosts': 8, u'list': [6, 7, 8]}, u'4': {u'hosts': 4, u'list': [5, 7]}, u'7': {u'hosts': 2, u'list': [8]}, u'6': {u'hosts': 7, u'list': [7, 8]}, u'9': {u'hosts': 0, u'list': []}, u'8': {u'hosts': 9, u'list': [9]}})
1
0
3
2
5
4
7
6
9
8
[<OVSSwitch sw1: lo:127.0.0.1 pid=9709> , <OVSSwitch sw0: lo:127.0.0.1 pid=9712> , <OVSSwitch sw3: lo:127.0.0.1 pid=9715> , <OVSSwitch sw2: lo:127.0.0.1 pid=9718> , <OVSSwitch sw5: lo:127.0.0.1 pid=9721> , <OVSSwitch sw4: lo:127.0.0.1 pid=9724> , <OVSSwitch sw7: lo:127.0.0.1 pid=9727> , <OVSSwitch sw6: lo:127.0.0.1 pid=9730> , <OVSSwitch sw9: lo:127.0.0.1 pid=9733> , <OVSSwitch sw8: lo:127.0.0.1 pid=9736> ]
mininet> 

In this case opendaylight is not showing anything, as you can see in the next picture: enter image description here

If I will put the same command to call directly the python script:

sudo python ~/api/API_TEST.py /home/mininet/api/topology3.json 10.0.0.0/8 192.168.241.129 6633 sw h 10 10

I will get the right output:

enter image description here

Moreover, when I am running the scrip with the nohup:

nohup sudo python ~/api/API_TEST.py /home/mininet/api/topology3.json 10.0.0.0/8 192.168.241.129 6633 sw h 10 10 &

I will get the right output: enter image description here

The error in the opendaylight is:

Exception in thread "Thread-421" java.lang.IllegalStateException: Expected state: HANDSHAKING, actual state:RIP
    at org.opendaylight.openflowplugin.openflow.md.core.ConnectionConductorImpl.checkState(ConnectionConductorImpl.java:386)
    at org.opendaylight.openflowplugin.openflow.md.core.ConnectionConductorImpl.onConnectionReady(ConnectionConductorImpl.java:448)
    at org.opendaylight.openflowjava.protocol.impl.core.connection.ConnectionAdapterImpl$1.run(ConnectionAdapterImpl.java:185)
    at java.lang.Thread.run(Thread.java:750)

At this point, I am still trying to figure out what I should do more; if you have any suggestions, please feel free to comment on them.

0

There are 0 best solutions below