# # NEPI, a framework to manage network experiments # Copyright (C) 2013 INRIA # # This program is free software: you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation, either version 3 of the License, or # (at your option) any later version. # # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License # along with this program. If not, see . # # Author: Alina Quereilhac from nepi.execution.attribute import Attribute, Flags, Types from nepi.execution.resource import clsinit_copy, ResourceState, \ reschedule_delay from nepi.resources.linux.tunnel import LinuxTunnel from nepi.util.sshfuncs import ProcStatus from nepi.util.timefuncs import tnow, tdiffsec import re import socket import time import os @clsinit_copy class LinuxGRETunnel(LinuxTunnel): _rtype = "LinuxGRETunnel" _help = "Constructs a tunnel between two Linux endpoints using a UDP connection " _backend = "linux" def log_message(self, msg): return " guid %d - GRE tunnel %s - %s - %s " % (self.guid, self.endpoint1.node.get("hostname"), self.endpoint2.node.get("hostname"), msg) def get_endpoints(self): """ Returns the list of RM that are endpoints to the tunnel """ connected = [] for guid in self.connections: rm = self.ec.get_resource(guid) if hasattr(rm, "gre_connect_command"): connected.append(rm) return connected def initiate_connection(self, endpoint, remote_endpoint): # Return the command to execute to initiate the connection to the # other endpoint connection_run_home = self.run_home(endpoint) gre_connect_command = endpoint.gre_connect_command( remote_endpoint, connection_run_home) # upload command to connect.sh script shfile = os.path.join(self.app_home(endpoint), "gre-connect.sh") endpoint.node.upload(gre_connect_command, shfile, text = True, overwrite = False) # invoke connect script cmd = "bash %s" % shfile (out, err), proc = endpoint.node.run(cmd, self.run_home(endpoint)) # check if execution errors occurred msg = " Failed to connect endpoints " if proc.poll() or err: self.error(msg, out, err) raise RuntimeError, msg # Wait for pid file to be generated pid, ppid = endpoint.node.wait_pid(self.run_home(endpoint)) # If the process is not running, check for error information # on the remote machine if not pid or not ppid: (out, err), proc = endpoint.node.check_errors(self.run_home(endpoint)) # Out is what was written in the stderr file if err: msg = " Failed to start command '%s' " % command self.error(msg, out, err) raise RuntimeError, msg # After creating the TAP, the pl-vif-create.py script # will write the name of the TAP to a file. We wait until # we can read the interface name from the file. vif_name = endpoint.wait_vif_name() endpoint.set("deviceName", vif_name) # Wait if name return True def establish_connection(self, endpoint, remote_endpoint, data): pass def verify_connection(self, endpoint, remote_endpoint): remote_ip = socket.gethostbyname(remote_endpoint.node.get("hostname")) command = "ping -c 4 %s" % remote_ip (out, err), proc = endpoint.node.execute(command, blocking = True) m = re.search("(\d+)% packet loss", str(out)) if not m or int(m.groups()[0]) == 100: msg = " Erroro establishing GRE Tunnel" self.error(msg, out, err) raise RuntimeError, msg def terminate_connection(self, endpoint, remote_endpoint): pass def check_state_connection(self): pass def valid_connection(self, guid): # TODO: Validate! return True