Switch branches/tags
Nothing to show
Find file Copy path
Fetching contributors…
Cannot retrieve contributors at this time
617 lines (520 sloc) 21.9 KB
This module defines the client-side classes for the Nuke command server interface.
It also functions as an executable to launch NukeCommandManager instances.
import os
import inspect
import pickle
import socket
import subprocess
import sys
import threading
import time
import traceback
from nukeExternalControl.common import *
THIS_FILE = inspect.getabsfile(lambda:0)
except TypeError:
this_mod = __import__(__name__, {}, {}, [])
THIS_FILE = getattr(this_mod, '__file__', None)
THIS_FILE = os.path.abspath(THIS_FILE)
class NukeConnection(object):
If 'port' is specified, the client will attempt to connect
to a command server on that port, raising an exception
if one is not found.
Otherwise, the standard port search routine runs.
def __init__(self, port=None, host="localhost", instance=0):
self._objects = {}
self._functions = {}
self._host = host
self.is_active = False
if not port:
start_port = DEFAULT_START_PORT + instance
self._port = self.find_connection_port(start_port, end_port)
if self._port == -1:
raise NukeConnectionError("Connection with Nuke failed")
self.is_active = True
self._port = port
if not self.test_connection():
raise NukeConnectionError("Could not connect to Nuke command server on port %d" % self._port)
self.is_active = True
if not self.authenticate_connection():
self.is_active = False
raise NukeConnectionError("Connection with Nuke denied")
def find_connection_port(self, start_port, end_port):
Find the first available open port between start_port and end_port
for port in range(start_port, end_port + 1):
self._port = port
if self.test_connection():
return port
return -1
def send(self, data):
Send some ASCII data to the server, and then wait for a response
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
s.connect((self._host, self._port))
result = s.recv(SOCKET_BUFFER_SIZE)
except socket.error:
raise NukeConnectionError("Connection with Nuke failed")
return result
def authenticate_connection(self):
Pass a message to the server identifying where the client is coming from,
and give the server the option of refusing to talk.
if self._host == "localhost":
host = "localhost"
host = os.getenv("HOST")
if self.get("initiate", parameters = host) == "accept":
return True
return False
def test_connection(self):
Test to see if the connection is working.
The 'test' action should return True
return self.get("test")
except NukeConnectionError, e:
return False
def get(self, item_type, item_id = -1, parameters = None):
Encode the action, object and parameters and pass them over the socket connection.
If the pickled data is too long, send it as a multi-part message.
Decode any returned data, joining together multiple parts as necessary,
and return (or raise, in the case of an Exception) the result.
data = {'action': item_type, 'id': item_id, 'parameters': parameters}
encoded = pickle.dumps(self.encode(data))
if len(encoded) > MAX_SOCKET_BYTES:
encodedBits = []
while encoded:
encoded = encoded[MAX_SOCKET_BYTES:]
for i in range(len(encodedBits)):
result = pickle.loads(self.send(pickle.dumps({'type': "NukeTransferPartialObject", 'part': i, 'part_count': len(encodedBits), 'data': encodedBits[i]})))
if i < (len(encodedBits) - 1):
if not (isinstance(result, dict) and 'type' in result and result['type'] == "NukeTransferPartialObjectRequest" and 'part' in result and result['part'] == i+1):
raise NukeConnectionError("Unexpected response to partial object")
result = pickle.loads(self.send(encoded))
if isinstance(result, dict) and 'type' in result and result['type'] == "NukeTransferPartialObject":
data = result['data']
nextPart = 1
while nextPart < result['part_count']:
returnData = self.send(pickle.dumps({'type': "NukeTransferPartialObjectRequest", 'part': nextPart}))
result = pickle.loads(returnData)
data += result['data']
nextPart += 1
result = pickle.loads(data)
except Exception, e:
raise e
if isinstance(result, Exception):
raise result
return result
def shutdown_server(self):
Passes the 'shutdown' keyword to the server.
This will raise a special exception in the
server's listener loop, causing it to pass
back a shutdown message, close the client,
and exit cleanly.
Returns whatever shutdown message the server
sends back as a string.
self.is_active = False
return self.get('shutdown')
def get_object_attribute(self, obj_id, property_name):
Get an attribute from an object on the server
result = object.property_name
return self.decode(self.get("getattr", obj_id, property_name))
def set_object_attribute(self, obj_id, property_name, value):
Set an attribute on an object on the server
object.property_name = value
return self.decode(self.get("setattr", obj_id, (property_name, value)))
def get_object_item(self, obj_id, property_name):
Get an item from an object on the server
result = object[property_name]
return self.decode(self.get("getitem", obj_id, property_name))
def set_object_item(self, obj_id, property_name, value):
Set an item on an object on the server
object[property_name] = value
return self.decode(self.get("setitem", obj_id, (property_name, value)))
def call_object_function(self, obj_id, parameters):
Call an object on the server
result = object(parameters)
return self.decode(self.get("call", obj_id, parameters))
def get_object_length(self, obj_id):
Get the length of an object on the server
result = len(object)
return self.decode(self.get("len", obj_id))
def get_object_string(self, obj_id):
Get the string equivalent of an object on the server
result = str(object)
return self.decode(self.get("str", obj_id))
def get_object_repr(self, obj_id):
Get the representation of an object on the server
result = `object`
return self.decode(self.get("repr", obj_id))
def delete_object(self, obj_id):
return self.decode(self.get("del", obj_id))
def get_object_isinstance(self, obj_id, instance):
return self.decode(self.get("isinstance", obj_id, instance))
def get_object_issubclass(self, obj_id, subclass):
return self.decode(self.get("issubclass", obj_id, subclass))
def import_module(self, module_name):
Import a module on the server
import module_name
return module_name
return self.decode(self.get("import", parameters = module_name))
def recode_data(self, data, recode_object_func):
Recode some data with the passed recode function
This deals with passing data both to and from the interim format,
and recursively recoding lists and dictionaries.
if type(data) in basicTypes:
return data
elif type(data) in listTypes:
newList = []
for i in data:
newList.append(self.recode_data(i, recode_object_func))
return type(data)(newList)
elif type(data) in dictTypes:
if 'type' in data and data['type'] == "NukeTransferObject":
return recode_object_func(data)
newDict = {}
for k in data:
newDict[self.recode_data(k, recode_object_func)] = self.recode_data(data[k], recode_object_func)
return newDict
return recode_object_func(data)
def encode_data(self, data):
Encode data to send to the server
return self.recode_data(data, self.encode_data_object)
def decode_data(self, data):
Decode data that the server has sent back
return self.recode_data(data, self.decode_data_object)
def encode_data_object(self, data):
Encode any NukeObject instances so that they can be
turned back into their actual objects on the server end
if isinstance(data, NukeObject):
return {'type': "NukeTransferObject", 'id': data._id}
raise TypeError("Invalid object type being passed through connection: '%s'" % data)
def decode_data_object(self, data):
Convert a dictionary representing an object on the server into
a NukeObject instance
return NukeObject(self, data['id'])
def encode(self, data):
Encode some data, and turn it into a pickled stream
return self.encode_data(data)
def decode(self, data):
Decode a pickle stream of data, ensuring that any NukeObject
instances are created
return self.decode_data(data)
def __getattr__(self, attrname):
Get a globals-level item from the server, by requesting it as an
attribute from the connection object
return self.get_object_item(-1, attrname)
def __getitem__(self, itemname):
Getting an item from the connection object works in the same way
as getting an attribute
return self.__getattr__(itemname)
def __repr__(self):
Return a string representation of the connection object
return object.__repr__(self).replace("instance object", "NukeConnection instance")
def __str__(self):
Return a string representation of the connection object
return self.__repr__()
class NukeObject(object):
The class that is used on the client to represent objects on the server
inside Nuke.
This class deals with catching anything that is called on itself and
ensuring that it is passed through to the server, and the appropriate
result is returned
def __init__(self, connection, id):
self.__dict__['_id'] = id
self.__dict__['_connection'] = connection
def __getattr__(self, attrname):
Get an attribute from the object.
If the requested attribute begins with a _, then get it from
self.__dict__, otherwise request it from the server
result = object.attrname
if attrname in self.__dict__:
return self.__dict__[attrname]
return self._connection.get_object_attribute(self._id, attrname)
def __setattr__(self, attrname, value):
Set an attribute on the object
object.attrname = value
return self._connection.set_object_attribute(self._id, attrname, value)
def __getitem__(self, itemname):
Get an item from the object
result = object[itemname]
return self._connection.get_object_item(self._id, itemname)
def __setitem__(self, itemname, value):
Set an item on the object
object[itemname] = value
return self._connection.set_object_item(self._id, itemname, value)
def __call__(self, *args, **kwargs):
Call the object with the passed arguments
result = object(*args, **kwargs)
return self._connection.call_object_function(self._id, {'args': args, 'kwargs': kwargs})
def __len__(self):
Get the length of the object
result = len(object)
return self._connection.get_object_length(self._id)
def __str__(self):
Get the string equivalent of the object
result = str(object)
return self._connection.get_object_string(self._id)
def __repr__(self):
Get the representation of the object
result = `object`
return self._connection.get_object_repr(self._id)
def __del__(self):
Delete an object
del object
return self._connection.delete_object(self._id)
def __instancecheck__(cls, inst):
Check whether the object is an instance of a specific class
result = isinstance(inst, cls)
return cls._connection.get_object_isinstance(cls._id, inst)
def __subclasscheck__(self, subclass):
Check whether the object is an subclass of a specific class
result = isinstance(object, cls)
return self._connection.get_object_issubclass(self._id, subclass)
class NukeCommandManager(object):
This class internally manages a Nuke command client-server pair.
It is designed to be instantiated as the 'as' assignment in a
'with' statement.
Example usage:
with NukeCommandManager() as conn:
nuke = conn.nuke
b = nuke.createNode('Blur')
print b.writeKnobs()
When it starts up, it establishes a manager socket on an
available OS-assigned port.
When the manager's __enter__ method is called, the server
subprocess is started. The manager then waits for the server
to call back with its status and bound port.
A NukeConnection instance is then started using the port number
returned by the managed server's callback. This instance is
attached to the manager and returned to the 'with' statement.
The body of the 'with' block is now executed,
with the client instance available via the 'as' assignment.
When the 'with' statement is complete, the client instance sends
its companion server the 'shutdown' signal. This will cause the
server to send back its shutdown message, close the connection to
the client, and exit cleanly.
def __init__(self, license_retry_count=5, license_retry_delay=5, extra_nuke_args=()):
self.manager_port = -1
self.manager_socket = None
self.server_port = -1
self.client = None
self.license_retry_count = license_retry_count
self.license_retry_delay = license_retry_delay
self.nuke_stdout, self.nuke_stderr = None, None
bound_port = False
manager = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
manager.bind(('', 0))
bound_port = True
self.manager_port = manager.getsockname()[1]
self.manager_socket = manager
if (not bound_port) or (self.manager_port == -1):
raise NukeManagerError("MANAGER: Cannot find port to bind to")
self.extra_nuke_args = extra_nuke_args
def __enter__(self):
if not self.manager_socket:
raise NukeManagerError("Manager failed to initialize socket.")
backlog = 5
# Start the server process and wait for it to call back to the
# manager with its success status and bound port
self.client = NukeConnection(self.server_port)
return self.client
def __exit__(self, type, value, traceback):
self.nuke_stdout, self.nuke_stderr = self.server_proc.communicate()
def start_server(self):
if not THIS_FILE:
raise RuntimeError("could not determine absolute path to %s module" % globals()['__name__'])
# Make sure the port number has a trailing space... this is a bug in Nuke's
# Python argument parsing (logged with The Foundry as Bug 17918)
procArgs = ([NUKE_EXEC, '-t', '-m', '1'] + list(self.extra_nuke_args) + ['--', THIS_FILE, '%d ' % self.manager_port],)
for i in xrange(self.license_retry_count+1):
self.server_proc = subprocess.Popen(stdout=subprocess.PIPE,
startTime = time.time()
timeout = startTime + 15 # Timeout after 10 seconds of waiting for server
while True:
# Times out after 15 seconds based on the socket settings
server, address = self.manager_socket.accept()
except socket.timeout:
retCode = self.server_proc.poll()
if retCode:
if retCode == 100: # License failure.
raise NukeLicenseError
else: # Nuke died with another return code
print "Nuke process died with an unexpected return code"
raise NukeManagerError("Server process failed to start. Nuke exited with code %s." % retCode)
elif retCode is None:
# Nuke is still running
print "Nuke process is still alive but hasn't responded to the Manager yet (timed out)."
raise nukeManagerError("Nuke process hasn't exited, but hasn't responded to the Manager either.")
else: # Nuke exited cleanly (0) for some reason
print "Nuke exited with code 0 (server script failed to start running)"
raise NukeManagerError("Server process failed to start properly.")
data = server.recv(SOCKET_BUFFER_SIZE)
if data:
serverData = pickle.loads(data)
if not serverData[0]:
raise NukeServerError("Server could not find port to bind to.")
self.server_port = serverData[1]
if time.time() >= timeout:
raise NukeManagerError("Manager timed out waiting for server connection.")
except NukeConnectionError, e:
except NukeServerError, se:
self.nuke_stdout = self.nuke_stderr = ""
print "Error in emergency Nuke shutdown:"
print se
retCode = self.server_proc.poll()
if retCode is not None:
self.nuke_stdout, self.nuke_stderr = self.server_proc.communicate()
print "Issuing shell kill on stalled process"
#FIXME: Kill is not cross-platform["kill", "-9",])
e.nuke_sdout, e.nuke_stderr = self.nuke_stdout, self.nuke_stderr
except NukeLicenseError:
print "License error. Retrying in %d seconds..." % self.license_retry_delay
raise NukeLicenseError("Maximum license retry count exceeded. Aborting.")
def shutdown_server(self):
Used to shut down a managed server if its
client could not be initialized.
Returns the server's shutdown message.
packet = {'action':'shutdown', 'id':-1, 'parameters':None}
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
s.connect(('', self.server_port))
result = s.recv(SOCKET_BUFFER_SIZE)
return pickle.loads(result)
except socket.error:
# Failed to connect to server port (server is dead?)
raise NukeServerError("Server failed to initialize.")
def start_managed_nuke_server(manager_port=None):
Convenience function for launching a managed Nuke command
server instance that will communicate with a NukeCommandManager
on the specified port. Must be called from within Nuke.
import nukeExternalControl.server as comServer
if __name__ == '__main__':
manager_port = None
if len(sys.argv) > 1:
manager_port = int(sys.argv[1])