# Copyright (C) 2014 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.
+# it under the terms of the GNU General Public License version 2 as
+# published by the Free Software Foundation;
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# Author: Alina Quereilhac <alina.quereilhac@inria.fr>
import base64
-import cPickle
+import pickle
import errno
+import os
import socket
+import time
import weakref
+import threading
from optparse import OptionParser, SUPPRESS_HELP
def __init__(self, simulation):
super(LinuxNS3Client, self).__init__()
self._simulation = weakref.ref(simulation)
-
- self._socat_proc = None
- self.connect_client()
+ self._socket_lock = threading.Lock()
@property
def simulation(self):
return self._simulation()
- def connect_client(self):
- if self.simulation.node.get("hostname") in ['localhost', '127.0.0.1']:
- return
-
- (out, err), self._socat_proc = self.simulation.node.socat(
- self.simulation.local_socket,
- self.simulation.remote_socket)
-
def send_msg(self, msg_type, *args, **kwargs):
- sock = socket.socket(socket.AF_UNIX, socket.SOCK_STREAM)
- sock.connect(self.simulation.local_socket)
-
msg = [msg_type, args, kwargs]
def encode(item):
- item = cPickle.dumps(item)
+ item = pickle.dumps(item)
return base64.b64encode(item)
encoded = "|".join(map(encode, msg))
- sock.send("%s\n" % encoded)
-
- reply = sock.recv(1024)
- return cPickle.loads(base64.b64decode(reply))
+
+ with self._socket_lock:
+ if self.simulation.node.get("hostname") in ['localhost', '127.0.0.1']:
+ sock = socket.socket(socket.AF_UNIX, socket.SOCK_STREAM)
+ sock.connect(self.simulation.remote_socket)
+ sock.send("%s\n" % encoded)
+ reply = sock.recv(1024)
+ sock.close()
+ else:
+ command = ( "python -c 'import socket;"
+ "sock = socket.socket(socket.AF_UNIX, socket.SOCK_STREAM);"
+ "sock.connect(\"%(socket_addr)s\");"
+ "msg = \"%(encoded_message)s\\n\";"
+ "sock.send(msg);"
+ "reply = sock.recv(1024);"
+ "sock.close();"
+ "print reply'") % {
+ "encoded_message": encoded,
+ "socket_addr": self.simulation.remote_socket,
+ }
+
+ (reply, err), proc = self.simulation.node.execute(command,
+ with_lock = True)
+
+ if (err and proc.poll()) or reply.strip() == "":
+ msg = (" Couldn't connect to remote socket %s - REPLY: %s "
+ "- ERROR: %s ") % (
+ self.simulation.remote_socket, reply, err)
+ self.simulation.error(msg, reply, err)
+ raise RuntimeError(msg)
+
+ reply = pickle.loads(base64.b64decode(reply))
+
+ return reply
def create(self, *args, **kwargs):
return self.send_msg(NS3WrapperMessage.CREATE, *args, **kwargs)
return self.send_msg(NS3WrapperMessage.STOP, *args, **kwargs)
def shutdown(self, *args, **kwargs):
- ret = None
-
- try:
- ret = self.send_msg(NS3WrapperMessage.SHUTDOWN, *args, **kwargs)
- except:
- pass
-
- try:
- if self._socat_proc:
- self._socat_proc.kill()
- except:
- pass
-
try:
- os.remove(self.simulation.local_socket)
+ return self.send_msg(NS3WrapperMessage.SHUTDOWN, *args, **kwargs)
except:
pass
- return ret
+ return None