2011-01-13 07:40:53 +00:00
|
|
|
"""
|
|
|
|
SleekXMPP: The Sleek XMPP Library
|
|
|
|
Copyright (C) 2011 Nathanael C. Fritz, Dann Martens (TOMOTON).
|
|
|
|
This file is part of SleekXMPP.
|
|
|
|
|
|
|
|
See the file LICENSE for copying permission.
|
|
|
|
"""
|
|
|
|
|
2015-03-09 11:33:18 +00:00
|
|
|
from sleekxmpp.plugins.xep_0009.binding import py2xml, xml2py, xml2fault, fault2xml
|
2011-01-13 07:40:53 +00:00
|
|
|
from threading import RLock
|
|
|
|
import abc
|
|
|
|
import inspect
|
|
|
|
import logging
|
|
|
|
import sleekxmpp
|
|
|
|
import sys
|
|
|
|
import threading
|
|
|
|
import traceback
|
|
|
|
|
|
|
|
log = logging.getLogger(__name__)
|
|
|
|
|
2015-03-09 11:33:18 +00:00
|
|
|
# Define a function _isstr() to check if an object is a string in a way
|
|
|
|
# compatible with Python 2 and Python 3 (basestring does not exists in Python 3).
|
|
|
|
try:
|
|
|
|
basestring # This evaluation will throw an exception if basestring does not exists (Python 3).
|
|
|
|
def _isstr(obj):
|
|
|
|
return isinstance(obj, basestring)
|
|
|
|
except NameError:
|
|
|
|
def _isstr(obj):
|
|
|
|
return isinstance(obj, str)
|
|
|
|
|
|
|
|
|
|
|
|
# Class decorator to declare a metaclass to a class in a way compatible with Python 2 and 3.
|
|
|
|
# This decorator is copied from 'six' (https://bitbucket.org/gutworth/six):
|
|
|
|
#
|
|
|
|
# Copyright (c) 2010-2015 Benjamin Peterson
|
|
|
|
#
|
|
|
|
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
# of this software and associated documentation files (the "Software"), to deal
|
|
|
|
# in the Software without restriction, including without limitation the rights
|
|
|
|
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
# copies of the Software, and to permit persons to whom the Software is
|
|
|
|
# furnished to do so, subject to the following conditions:
|
|
|
|
#
|
|
|
|
# The above copyright notice and this permission notice shall be included in all
|
|
|
|
# copies or substantial portions of the Software.
|
|
|
|
def _add_metaclass(metaclass):
|
|
|
|
def wrapper(cls):
|
|
|
|
orig_vars = cls.__dict__.copy()
|
|
|
|
slots = orig_vars.get('__slots__')
|
|
|
|
if slots is not None:
|
|
|
|
if isinstance(slots, str):
|
|
|
|
slots = [slots]
|
|
|
|
for slots_var in slots:
|
|
|
|
orig_vars.pop(slots_var)
|
|
|
|
orig_vars.pop('__dict__', None)
|
|
|
|
orig_vars.pop('__weakref__', None)
|
|
|
|
return metaclass(cls.__name__, cls.__bases__, orig_vars)
|
|
|
|
return wrapper
|
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
def _intercept(method, name, public):
|
|
|
|
def _resolver(instance, *args, **kwargs):
|
2011-11-19 20:07:57 +00:00
|
|
|
log.debug("Locally calling %s.%s with arguments %s.", instance.FQN(), method.__name__, args)
|
2011-02-09 01:45:45 +00:00
|
|
|
try:
|
2011-01-13 07:40:53 +00:00
|
|
|
value = method(instance, *args, **kwargs)
|
|
|
|
if value == NotImplemented:
|
2011-02-09 01:45:45 +00:00
|
|
|
raise InvocationException("Local handler does not implement %s.%s!" % (instance.FQN(), method.__name__))
|
2011-01-13 07:40:53 +00:00
|
|
|
return value
|
|
|
|
except InvocationException:
|
|
|
|
raise
|
|
|
|
except Exception as e:
|
|
|
|
raise InvocationException("A problem occured calling %s.%s!" % (instance.FQN(), method.__name__), e)
|
|
|
|
_resolver._rpc = public
|
|
|
|
_resolver._rpc_name = method.__name__ if name is None else name
|
|
|
|
return _resolver
|
|
|
|
|
|
|
|
def remote(function_argument, public = True):
|
|
|
|
'''
|
|
|
|
Decorator for methods which are remotely callable. This decorator
|
|
|
|
works in conjunction with classes which extend ABC Endpoint.
|
|
|
|
Example:
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
@remote
|
|
|
|
def remote_method(arg1, arg2)
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
Arguments:
|
|
|
|
function_argument -- a stand-in for either the actual method
|
|
|
|
OR a new name (string) for the method. In that case the
|
|
|
|
method is considered mapped:
|
|
|
|
Example:
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
@remote("new_name")
|
|
|
|
def remote_method(arg1, arg2)
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
public -- A flag which indicates if this method should be part
|
|
|
|
of the known dictionary of remote methods. Defaults to True.
|
|
|
|
Example:
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
@remote(False)
|
|
|
|
def remote_method(arg1, arg2)
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
Note: renaming and revising (public vs. private) can be combined.
|
|
|
|
Example:
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
@remote("new_name", False)
|
|
|
|
def remote_method(arg1, arg2)
|
|
|
|
'''
|
|
|
|
if hasattr(function_argument, '__call__'):
|
|
|
|
return _intercept(function_argument, None, public)
|
2011-02-09 01:45:45 +00:00
|
|
|
else:
|
2015-03-09 11:33:18 +00:00
|
|
|
if not _isstr(function_argument):
|
2011-02-09 01:45:45 +00:00
|
|
|
if not isinstance(function_argument, bool):
|
2011-01-13 07:40:53 +00:00
|
|
|
raise Exception('Expected an RPC method name or visibility modifier!')
|
|
|
|
else:
|
|
|
|
def _wrap_revised(function):
|
|
|
|
function = _intercept(function, None, function_argument)
|
|
|
|
return function
|
2011-02-09 01:45:45 +00:00
|
|
|
return _wrap_revised
|
2011-01-13 07:40:53 +00:00
|
|
|
def _wrap_remapped(function):
|
|
|
|
function = _intercept(function, function_argument, public)
|
|
|
|
return function
|
|
|
|
return _wrap_remapped
|
2011-02-09 01:45:45 +00:00
|
|
|
|
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
class ACL:
|
|
|
|
'''
|
|
|
|
An Access Control List (ACL) is a list of rules, which are evaluated
|
|
|
|
in order until a match is found. The policy of the matching rule
|
|
|
|
is then applied.
|
|
|
|
|
|
|
|
Rules are 3-tuples, consisting of a policy enumerated type, a JID
|
|
|
|
expression and a RCP resource expression.
|
|
|
|
|
|
|
|
Examples:
|
|
|
|
[ (ACL.ALLOW, '*', '*') ] allow everyone everything, no restrictions
|
|
|
|
[ (ACL.DENY, '*', '*') ] deny everyone everything, no restrictions
|
|
|
|
[ (ACL.ALLOW, 'test@xmpp.org/unit', 'test.*'),
|
|
|
|
(ACL.DENY, '*', '*') ] deny everyone everything, except named
|
|
|
|
JID, which is allowed access to endpoint 'test' only.
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
The use of wildcards is allowed in expressions, as follows:
|
|
|
|
'*' everyone, or everything (= all endpoints and methods)
|
|
|
|
'test@xmpp.org/*' every JID regardless of JID resource
|
|
|
|
'*@xmpp.org/rpc' every JID from domain xmpp.org with JID res 'rpc'
|
|
|
|
'frank@*' every 'frank', regardless of domain or JID res
|
|
|
|
'system.*' all methods of endpoint 'system'
|
|
|
|
'*.reboot' all methods reboot regardless of endpoint
|
|
|
|
'''
|
|
|
|
ALLOW = True
|
|
|
|
DENY = False
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
@classmethod
|
|
|
|
def check(cls, rules, jid, resource):
|
|
|
|
if rules is None:
|
|
|
|
return cls.DENY # No rules means no access!
|
2011-12-15 21:58:33 +00:00
|
|
|
jid = str(jid) # Check the string representation of the JID.
|
|
|
|
if not jid:
|
|
|
|
return cls.DENY # Can't check an empty JID.
|
2011-01-13 07:40:53 +00:00
|
|
|
for rule in rules:
|
|
|
|
policy = cls._check(rule, jid, resource)
|
|
|
|
if policy is not None:
|
2011-02-09 01:45:45 +00:00
|
|
|
return policy
|
2011-01-13 07:40:53 +00:00
|
|
|
return cls.DENY # By default if not rule matches, deny access.
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
@classmethod
|
|
|
|
def _check(cls, rule, jid, resource):
|
|
|
|
if cls._match(jid, rule[1]) and cls._match(resource, rule[2]):
|
|
|
|
return rule[0]
|
|
|
|
else:
|
|
|
|
return None
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def _next_token(cls, expression, index):
|
|
|
|
new_index = expression.find('*', index)
|
|
|
|
if new_index == 0:
|
|
|
|
return ''
|
|
|
|
else:
|
|
|
|
if new_index == -1:
|
2011-02-09 01:45:45 +00:00
|
|
|
return expression[index : ]
|
2011-01-13 07:40:53 +00:00
|
|
|
else:
|
|
|
|
return expression[index : new_index]
|
|
|
|
|
|
|
|
@classmethod
|
2011-02-09 01:45:45 +00:00
|
|
|
def _match(cls, value, expression):
|
|
|
|
#! print "_match [VALUE] %s [EXPR] %s" % (value, expression)
|
2011-01-13 07:40:53 +00:00
|
|
|
index = 0
|
|
|
|
position = 0
|
|
|
|
while index < len(expression):
|
|
|
|
token = cls._next_token(expression, index)
|
|
|
|
#! print "[TOKEN] '%s'" % token
|
|
|
|
size = len(token)
|
|
|
|
if size > 0:
|
|
|
|
token_index = value.find(token, position)
|
|
|
|
if token_index == -1:
|
|
|
|
return False
|
|
|
|
else:
|
|
|
|
#! print "[INDEX-OF] %s" % token_index
|
|
|
|
position = token_index + len(token)
|
|
|
|
pass
|
|
|
|
if size == 0:
|
|
|
|
index += 1
|
|
|
|
else:
|
|
|
|
index += size
|
|
|
|
#! print "index %s position %s" % (index, position)
|
|
|
|
return True
|
|
|
|
|
|
|
|
ANY_ALL = [ (ACL.ALLOW, '*', '*') ]
|
|
|
|
|
|
|
|
|
|
|
|
class RemoteException(Exception):
|
|
|
|
'''
|
|
|
|
Base exception for RPC. This exception is raised when a problem
|
|
|
|
occurs in the network layer.
|
|
|
|
'''
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
def __init__(self, message="", cause=None):
|
|
|
|
'''
|
|
|
|
Initializes a new RemoteException.
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
Arguments:
|
|
|
|
message -- The message accompanying this exception.
|
|
|
|
cause -- The underlying cause of this exception.
|
2011-02-09 01:45:45 +00:00
|
|
|
'''
|
2011-01-13 07:40:53 +00:00
|
|
|
self._message = message
|
2011-02-09 01:45:45 +00:00
|
|
|
self._cause = cause
|
|
|
|
pass
|
2011-01-13 07:40:53 +00:00
|
|
|
|
|
|
|
def __str__(self):
|
|
|
|
return repr(self._message)
|
|
|
|
|
|
|
|
def get_message(self):
|
|
|
|
return self._message
|
|
|
|
|
|
|
|
def get_cause(self):
|
|
|
|
return self._cause
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class InvocationException(RemoteException):
|
|
|
|
'''
|
|
|
|
Exception raised when a problem occurs during the remote invocation
|
2011-02-09 01:45:45 +00:00
|
|
|
of a method.
|
2011-01-13 07:40:53 +00:00
|
|
|
'''
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class AuthorizationException(RemoteException):
|
|
|
|
'''
|
|
|
|
Exception raised when the caller is not authorized to invoke the
|
|
|
|
remote method.
|
|
|
|
'''
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
class TimeoutException(Exception):
|
|
|
|
'''
|
|
|
|
Exception raised when the synchronous execution of a method takes
|
|
|
|
longer than the given threshold because an underlying asynchronous
|
|
|
|
reply did not arrive in time.
|
|
|
|
'''
|
2011-02-09 01:45:45 +00:00
|
|
|
pass
|
2011-01-13 07:40:53 +00:00
|
|
|
|
|
|
|
|
2015-03-09 11:33:18 +00:00
|
|
|
@_add_metaclass(abc.ABCMeta)
|
2011-01-13 07:40:53 +00:00
|
|
|
class Callback(object):
|
|
|
|
'''
|
2011-02-09 01:45:45 +00:00
|
|
|
A base class for callback handlers.
|
2011-01-13 07:40:53 +00:00
|
|
|
'''
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
@abc.abstractproperty
|
|
|
|
def set_value(self, value):
|
|
|
|
return NotImplemented
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
@abc.abstractproperty
|
|
|
|
def cancel_with_error(self, exception):
|
|
|
|
return NotImplemented
|
|
|
|
|
|
|
|
|
|
|
|
class Future(Callback):
|
|
|
|
'''
|
|
|
|
Represents the result of an asynchronous computation.
|
|
|
|
'''
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
def __init__(self):
|
|
|
|
'''
|
|
|
|
Initializes a new Future.
|
|
|
|
'''
|
|
|
|
self._value = None
|
2011-02-09 01:45:45 +00:00
|
|
|
self._exception = None
|
2011-01-13 07:40:53 +00:00
|
|
|
self._event = threading.Event()
|
|
|
|
pass
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
def set_value(self, value):
|
|
|
|
'''
|
|
|
|
Sets the value of this Future. Once the value is set, a caller
|
|
|
|
blocked on get_value will be able to continue.
|
|
|
|
'''
|
|
|
|
self._value = value
|
|
|
|
self._event.set()
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
def get_value(self, timeout=None):
|
|
|
|
'''
|
|
|
|
Gets the value of this Future. This call will block until
|
|
|
|
the result is available, or until an optional timeout expires.
|
2011-02-09 01:45:45 +00:00
|
|
|
When this Future is cancelled with an error,
|
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
Arguments:
|
|
|
|
timeout -- The maximum waiting time to obtain the value.
|
|
|
|
'''
|
|
|
|
self._event.wait(timeout)
|
|
|
|
if self._exception:
|
|
|
|
raise self._exception
|
|
|
|
if not self._event.is_set():
|
|
|
|
raise TimeoutException
|
|
|
|
return self._value
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
def is_done(self):
|
|
|
|
'''
|
|
|
|
Returns true if a value has been returned.
|
|
|
|
'''
|
|
|
|
return self._event.is_set()
|
|
|
|
|
|
|
|
def cancel_with_error(self, exception):
|
|
|
|
'''
|
2011-02-09 01:45:45 +00:00
|
|
|
Cancels the Future because of an error. Once cancelled, a
|
2011-01-13 07:40:53 +00:00
|
|
|
caller blocked on get_value will be able to continue.
|
|
|
|
'''
|
2011-02-09 01:45:45 +00:00
|
|
|
self._exception = exception
|
|
|
|
self._event.set()
|
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
|
2015-03-09 11:33:18 +00:00
|
|
|
@_add_metaclass(abc.ABCMeta)
|
2011-01-13 07:40:53 +00:00
|
|
|
class Endpoint(object):
|
|
|
|
'''
|
|
|
|
The Endpoint class is an abstract base class for all objects
|
|
|
|
participating in an RPC-enabled XMPP network.
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
A user subclassing this class is required to implement the method:
|
2011-02-09 01:45:45 +00:00
|
|
|
FQN(self)
|
|
|
|
where FQN stands for Fully Qualified Name, an unambiguous name
|
|
|
|
which specifies which object an RPC call refers to. It is the
|
2011-01-13 07:40:53 +00:00
|
|
|
first part in a RPC method name '<fqn>.<method>'.
|
|
|
|
'''
|
|
|
|
|
|
|
|
def __init__(self, session, target_jid):
|
|
|
|
'''
|
|
|
|
Initialize a new Endpoint. This constructor should never be
|
|
|
|
invoked by a user, instead it will be called by the factories
|
|
|
|
which instantiate the RPC-enabled objects, of which only
|
2011-02-09 01:45:45 +00:00
|
|
|
the classes are provided by the user.
|
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
Arguments:
|
|
|
|
session -- An RPC session instance.
|
|
|
|
target_jid -- the identity of the remote XMPP entity.
|
|
|
|
'''
|
|
|
|
self.session = session
|
|
|
|
self.target_jid = target_jid
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
@abc.abstractproperty
|
|
|
|
def FQN(self):
|
|
|
|
return NotImplemented
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
def get_methods(self):
|
|
|
|
'''
|
|
|
|
Returns a dictionary of all RPC method names provided by this
|
|
|
|
class. This method returns the actual method names as found
|
|
|
|
in the class definition which have been decorated with:
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
@remote
|
|
|
|
def some_rpc_method(arg1, arg2)
|
2011-02-09 01:45:45 +00:00
|
|
|
|
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
Unless:
|
|
|
|
(1) the name has been remapped, in which case the new
|
|
|
|
name will be returned.
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
@remote("new_name")
|
|
|
|
def some_rpc_method(arg1, arg2)
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
(2) the method is set to hidden
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
@remote(False)
|
|
|
|
def some_hidden_method(arg1, arg2)
|
|
|
|
'''
|
|
|
|
result = dict()
|
|
|
|
for function in dir(self):
|
|
|
|
test_attr = getattr(self, function, None)
|
|
|
|
try:
|
|
|
|
if test_attr._rpc:
|
|
|
|
result[test_attr._rpc_name] = test_attr
|
|
|
|
except Exception:
|
|
|
|
pass
|
2011-02-09 01:45:45 +00:00
|
|
|
return result
|
2011-01-13 07:40:53 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class Proxy(Endpoint):
|
|
|
|
'''
|
|
|
|
Implementation of the Proxy pattern which is intended to wrap
|
|
|
|
around Endpoints in order to intercept calls, marshall them and
|
|
|
|
forward them to the remote object.
|
|
|
|
'''
|
|
|
|
|
|
|
|
def __init__(self, endpoint, callback = None):
|
|
|
|
'''
|
|
|
|
Initializes a new Proxy.
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
Arguments:
|
|
|
|
endpoint -- The endpoint which is proxified.
|
|
|
|
'''
|
|
|
|
self._endpoint = endpoint
|
|
|
|
self._callback = callback
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
def __getattribute__(self, name, *args):
|
|
|
|
if name in ('__dict__', '_endpoint', 'async', '_callback'):
|
|
|
|
return object.__getattribute__(self, name)
|
|
|
|
else:
|
|
|
|
attribute = self._endpoint.__getattribute__(name)
|
|
|
|
if hasattr(attribute, '__call__'):
|
|
|
|
try:
|
|
|
|
if attribute._rpc:
|
2011-02-09 01:45:45 +00:00
|
|
|
def _remote_call(*args, **kwargs):
|
2011-11-19 20:07:57 +00:00
|
|
|
log.debug("Remotely calling '%s.%s' with arguments %s.", self._endpoint.FQN(), attribute._rpc_name, args)
|
2011-01-13 07:40:53 +00:00
|
|
|
return self._endpoint.session._call_remote(self._endpoint.target_jid, "%s.%s" % (self._endpoint.FQN(), attribute._rpc_name), self._callback, *args, **kwargs)
|
|
|
|
return _remote_call
|
|
|
|
except:
|
2011-02-09 01:45:45 +00:00
|
|
|
pass # If the attribute doesn't exist, don't care!
|
2011-01-13 07:40:53 +00:00
|
|
|
return attribute
|
2011-02-09 01:45:45 +00:00
|
|
|
|
|
|
|
def async(self, callback):
|
2011-01-13 07:40:53 +00:00
|
|
|
return Proxy(self._endpoint, callback)
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
def get_endpoint(self):
|
|
|
|
'''
|
|
|
|
Returns the proxified endpoint.
|
|
|
|
'''
|
|
|
|
return self._endpoint
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
def FQN(self):
|
|
|
|
return self._endpoint.FQN()
|
|
|
|
|
|
|
|
|
|
|
|
class JabberRPCEntry(object):
|
2011-02-09 01:45:45 +00:00
|
|
|
|
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
def __init__(self, endpoint_FQN, call):
|
|
|
|
self._endpoint_FQN = endpoint_FQN
|
|
|
|
self._call = call
|
|
|
|
|
|
|
|
def call_method(self, args):
|
|
|
|
return_value = self._call(*args)
|
|
|
|
if return_value is None:
|
|
|
|
return return_value
|
|
|
|
else:
|
|
|
|
return self._return(return_value)
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
def get_endpoint_FQN(self):
|
|
|
|
return self._endpoint_FQN
|
|
|
|
|
|
|
|
def _return(self, *args):
|
|
|
|
return args
|
2011-02-09 01:45:45 +00:00
|
|
|
|
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
class RemoteSession(object):
|
|
|
|
'''
|
2011-02-09 01:45:45 +00:00
|
|
|
A context object for a Jabber-RPC session.
|
2011-01-13 07:40:53 +00:00
|
|
|
'''
|
2011-02-09 01:45:45 +00:00
|
|
|
|
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
def __init__(self, client, session_close_callback):
|
|
|
|
'''
|
|
|
|
Initializes a new RPC session.
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
Arguments:
|
|
|
|
client -- The SleekXMPP client associated with this session.
|
2011-02-09 01:45:45 +00:00
|
|
|
session_close_callback -- A callback called when the
|
2011-01-13 07:40:53 +00:00
|
|
|
session is closed.
|
|
|
|
'''
|
|
|
|
self._client = client
|
|
|
|
self._session_close_callback = session_close_callback
|
|
|
|
self._event = threading.Event()
|
|
|
|
self._entries = {}
|
|
|
|
self._callbacks = {}
|
|
|
|
self._acls = {}
|
|
|
|
self._lock = RLock()
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
def _wait(self):
|
|
|
|
self._event.wait()
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
def _notify(self, event):
|
2011-11-19 20:07:57 +00:00
|
|
|
log.debug("RPC Session as %s started.", self._client.boundjid.full)
|
2011-01-13 07:40:53 +00:00
|
|
|
self._client.sendPresence()
|
|
|
|
self._event.set()
|
|
|
|
pass
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
def _register_call(self, endpoint, method, name=None):
|
|
|
|
'''
|
|
|
|
Registers a method from an endpoint as remotely callable.
|
|
|
|
'''
|
|
|
|
if name is None:
|
|
|
|
name = method.__name__
|
|
|
|
key = "%s.%s" % (endpoint, name)
|
2011-11-19 20:07:57 +00:00
|
|
|
log.debug("Registering call handler for %s (%s).", key, method)
|
2011-01-13 07:40:53 +00:00
|
|
|
with self._lock:
|
2011-08-05 05:34:34 +00:00
|
|
|
if key in self._entries:
|
2011-01-13 07:40:53 +00:00
|
|
|
raise KeyError("A handler for %s has already been regisered!" % endpoint)
|
|
|
|
self._entries[key] = JabberRPCEntry(endpoint, method)
|
|
|
|
return key
|
|
|
|
|
|
|
|
def _register_acl(self, endpoint, acl):
|
2011-11-19 20:07:57 +00:00
|
|
|
log.debug("Registering ACL %s for endpoint %s.", repr(acl), endpoint)
|
2011-01-13 07:40:53 +00:00
|
|
|
with self._lock:
|
|
|
|
self._acls[endpoint] = acl
|
|
|
|
|
|
|
|
def _register_callback(self, pid, callback):
|
|
|
|
with self._lock:
|
|
|
|
self._callbacks[pid] = callback
|
2011-02-09 01:45:45 +00:00
|
|
|
|
|
|
|
def forget_callback(self, callback):
|
2011-01-13 07:40:53 +00:00
|
|
|
with self._lock:
|
|
|
|
pid = self._find_key(self._callbacks, callback)
|
|
|
|
if pid is not None:
|
|
|
|
del self._callback[pid]
|
|
|
|
else:
|
|
|
|
raise ValueError("Unknown callback!")
|
|
|
|
pass
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
def _find_key(self, dict, value):
|
|
|
|
"""return the key of dictionary dic given the value"""
|
2015-03-09 11:33:18 +00:00
|
|
|
search = [k for k, v in dict.items() if v == value]
|
2011-01-13 07:40:53 +00:00
|
|
|
if len(search) == 0:
|
|
|
|
return None
|
|
|
|
else:
|
|
|
|
return search[0]
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
def _unregister_call(self, key):
|
|
|
|
#removes the registered call
|
|
|
|
with self._lock:
|
|
|
|
if self._entries[key]:
|
|
|
|
del self._entries[key]
|
|
|
|
else:
|
|
|
|
raise ValueError()
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
def new_proxy(self, target_jid, endpoint_cls):
|
|
|
|
'''
|
|
|
|
Instantiates a new proxy object, which proxies to a remote
|
2011-02-09 01:45:45 +00:00
|
|
|
endpoint. This method uses a class reference without
|
2011-01-13 07:40:53 +00:00
|
|
|
constructor arguments to instantiate the proxy.
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
Arguments:
|
|
|
|
target_jid -- the XMPP entity ID hosting the endpoint.
|
|
|
|
endpoint_cls -- The remote (duck) type.
|
|
|
|
'''
|
2011-02-09 01:45:45 +00:00
|
|
|
try:
|
2011-01-13 07:40:53 +00:00
|
|
|
argspec = inspect.getargspec(endpoint_cls.__init__)
|
|
|
|
args = [None] * (len(argspec[0]) - 1)
|
|
|
|
result = endpoint_cls(*args)
|
|
|
|
Endpoint.__init__(result, self, target_jid)
|
|
|
|
return Proxy(result)
|
|
|
|
except:
|
|
|
|
traceback.print_exc(file=sys.stdout)
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
def new_handler(self, acl, handler_cls, *args, **kwargs):
|
|
|
|
'''
|
|
|
|
Instantiates a new handler object, which is called remotely
|
|
|
|
by others. The user can control the effect of the call by
|
|
|
|
implementing the remote method in the local endpoint class. The
|
|
|
|
returned reference can be called locally and will behave as a
|
|
|
|
regular instance.
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
Arguments:
|
|
|
|
acl -- Access control list (see ACL class)
|
|
|
|
handler_clss -- The local (duck) type.
|
|
|
|
*args -- Constructor arguments for the local type.
|
|
|
|
**kwargs -- Constructor keyworded arguments for the local
|
|
|
|
type.
|
|
|
|
'''
|
|
|
|
argspec = inspect.getargspec(handler_cls.__init__)
|
|
|
|
base_argspec = inspect.getargspec(Endpoint.__init__)
|
|
|
|
if(argspec == base_argspec):
|
|
|
|
result = handler_cls(self, self._client.boundjid.full)
|
|
|
|
else:
|
|
|
|
result = handler_cls(*args, **kwargs)
|
|
|
|
Endpoint.__init__(result, self, self._client.boundjid.full)
|
|
|
|
method_dict = result.get_methods()
|
2015-03-09 11:33:18 +00:00
|
|
|
for method_name, method in method_dict.items():
|
2011-02-09 01:45:45 +00:00
|
|
|
#!!! self._client.plugin['xep_0009'].register_call(result.FQN(), method, method_name)
|
2011-01-13 07:40:53 +00:00
|
|
|
self._register_call(result.FQN(), method, method_name)
|
|
|
|
self._register_acl(result.FQN(), acl)
|
|
|
|
return result
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
# def is_available(self, targetCls, pto):
|
|
|
|
# return self._client.is_available(pto)
|
|
|
|
|
|
|
|
def _call_remote(self, pto, pmethod, callback, *arguments):
|
|
|
|
iq = self._client.plugin['xep_0009'].make_iq_method_call(pto, pmethod, py2xml(*arguments))
|
|
|
|
pid = iq['id']
|
|
|
|
if callback is None:
|
|
|
|
future = Future()
|
|
|
|
self._register_callback(pid, future)
|
|
|
|
iq.send()
|
2011-02-09 01:45:45 +00:00
|
|
|
return future.get_value(30)
|
2011-01-13 07:40:53 +00:00
|
|
|
else:
|
2011-11-19 20:07:57 +00:00
|
|
|
log.debug("[RemoteSession] _call_remote %s", callback)
|
2011-01-13 07:40:53 +00:00
|
|
|
self._register_callback(pid, callback)
|
|
|
|
iq.send()
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2014-08-10 23:02:10 +00:00
|
|
|
def close(self, wait=False):
|
2011-01-13 07:40:53 +00:00
|
|
|
'''
|
|
|
|
Closes this session.
|
|
|
|
'''
|
2014-08-10 23:02:10 +00:00
|
|
|
self._client.disconnect(wait=wait)
|
2011-01-13 07:40:53 +00:00
|
|
|
self._session_close_callback()
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
def _on_jabber_rpc_method_call(self, iq):
|
|
|
|
iq.enable('rpc_query')
|
|
|
|
params = iq['rpc_query']['method_call']['params']
|
|
|
|
args = xml2py(params)
|
|
|
|
pmethod = iq['rpc_query']['method_call']['method_name']
|
|
|
|
try:
|
|
|
|
with self._lock:
|
|
|
|
entry = self._entries[pmethod]
|
|
|
|
rules = self._acls[entry.get_endpoint_FQN()]
|
|
|
|
if ACL.check(rules, iq['from'], pmethod):
|
|
|
|
return_value = entry.call_method(args)
|
|
|
|
else:
|
|
|
|
raise AuthorizationException("Unauthorized access to %s from %s!" % (pmethod, iq['from']))
|
|
|
|
if return_value is None:
|
|
|
|
return_value = ()
|
|
|
|
response = self._client.plugin['xep_0009'].make_iq_method_response(iq['id'], iq['from'], py2xml(*return_value))
|
|
|
|
response.send()
|
|
|
|
except InvocationException as ie:
|
|
|
|
fault = dict()
|
|
|
|
fault['code'] = 500
|
|
|
|
fault['string'] = ie.get_message()
|
|
|
|
self._client.plugin['xep_0009']._send_fault(iq, fault2xml(fault))
|
|
|
|
except AuthorizationException as ae:
|
|
|
|
log.error(ae.get_message())
|
|
|
|
error = self._client.plugin['xep_0009']._forbidden(iq)
|
2011-02-09 01:45:45 +00:00
|
|
|
error.send()
|
|
|
|
except Exception as e:
|
2011-01-13 07:40:53 +00:00
|
|
|
if isinstance(e, KeyError):
|
2011-11-19 20:07:57 +00:00
|
|
|
log.error("No handler available for %s!", pmethod)
|
2011-01-13 07:40:53 +00:00
|
|
|
error = self._client.plugin['xep_0009']._item_not_found(iq)
|
|
|
|
else:
|
|
|
|
traceback.print_exc(file=sys.stderr)
|
2011-11-19 20:07:57 +00:00
|
|
|
log.error("An unexpected problem occurred invoking method %s!", pmethod)
|
2011-02-09 01:45:45 +00:00
|
|
|
error = self._client.plugin['xep_0009']._undefined_condition(iq)
|
2011-01-13 07:40:53 +00:00
|
|
|
#! print "[REMOTE.PY] _handle_remote_procedure_call AN ERROR SHOULD BE SENT NOW %s " % e
|
|
|
|
error.send()
|
|
|
|
|
|
|
|
def _on_jabber_rpc_method_response(self, iq):
|
|
|
|
iq.enable('rpc_query')
|
|
|
|
args = xml2py(iq['rpc_query']['method_response']['params'])
|
|
|
|
pid = iq['id']
|
|
|
|
with self._lock:
|
|
|
|
callback = self._callbacks[pid]
|
|
|
|
del self._callbacks[pid]
|
|
|
|
if(len(args) > 0):
|
|
|
|
callback.set_value(args[0])
|
|
|
|
else:
|
|
|
|
callback.set_value(None)
|
2011-02-09 01:45:45 +00:00
|
|
|
pass
|
2011-01-13 07:40:53 +00:00
|
|
|
|
|
|
|
def _on_jabber_rpc_method_response2(self, iq):
|
|
|
|
iq.enable('rpc_query')
|
|
|
|
if iq['rpc_query']['method_response']['fault'] is not None:
|
|
|
|
self._on_jabber_rpc_method_fault(iq)
|
|
|
|
else:
|
|
|
|
args = xml2py(iq['rpc_query']['method_response']['params'])
|
|
|
|
pid = iq['id']
|
|
|
|
with self._lock:
|
|
|
|
callback = self._callbacks[pid]
|
|
|
|
del self._callbacks[pid]
|
|
|
|
if(len(args) > 0):
|
|
|
|
callback.set_value(args[0])
|
|
|
|
else:
|
|
|
|
callback.set_value(None)
|
2011-02-09 01:45:45 +00:00
|
|
|
pass
|
2011-01-13 07:40:53 +00:00
|
|
|
|
|
|
|
def _on_jabber_rpc_method_fault(self, iq):
|
|
|
|
iq.enable('rpc_query')
|
|
|
|
fault = xml2fault(iq['rpc_query']['method_response']['fault'])
|
2011-02-09 01:45:45 +00:00
|
|
|
pid = iq['id']
|
2011-01-13 07:40:53 +00:00
|
|
|
with self._lock:
|
|
|
|
callback = self._callbacks[pid]
|
|
|
|
del self._callbacks[pid]
|
|
|
|
e = {
|
2011-02-09 01:45:45 +00:00
|
|
|
500: InvocationException
|
2011-01-13 07:40:53 +00:00
|
|
|
}[fault['code']](fault['string'])
|
2011-02-09 01:45:45 +00:00
|
|
|
callback.cancel_with_error(e)
|
2011-01-13 07:40:53 +00:00
|
|
|
|
|
|
|
def _on_jabber_rpc_error(self, iq):
|
|
|
|
pid = iq['id']
|
|
|
|
pmethod = self._client.plugin['xep_0009']._extract_method(iq['rpc_query'])
|
2011-02-09 01:45:45 +00:00
|
|
|
code = iq['error']['code']
|
2011-01-13 07:40:53 +00:00
|
|
|
type = iq['error']['type']
|
|
|
|
condition = iq['error']['condition']
|
|
|
|
#! print("['REMOTE.PY']._BINDING_handle_remote_procedure_error -> ERROR! ERROR! ERROR! Condition is '%s'" % condition)
|
|
|
|
with self._lock:
|
|
|
|
callback = self._callbacks[pid]
|
|
|
|
del self._callbacks[pid]
|
2011-02-09 01:45:45 +00:00
|
|
|
e = {
|
2011-01-13 07:40:53 +00:00
|
|
|
'item-not-found': RemoteException("No remote handler available for %s at %s!" % (pmethod, iq['from'])),
|
|
|
|
'forbidden': AuthorizationException("Forbidden to invoke remote handler for %s at %s!" % (pmethod, iq['from'])),
|
2011-02-09 01:45:45 +00:00
|
|
|
'undefined-condition': RemoteException("An unexpected problem occured trying to invoke %s at %s!" % (pmethod, iq['from'])),
|
2011-01-13 07:40:53 +00:00
|
|
|
}[condition]
|
|
|
|
if e is None:
|
|
|
|
RemoteException("An unexpected exception occurred at %s!" % iq['from'])
|
|
|
|
callback.cancel_with_error(e)
|
|
|
|
|
|
|
|
|
|
|
|
class Remote(object):
|
|
|
|
'''
|
|
|
|
Bootstrap class for Jabber-RPC sessions. New sessions are openend
|
|
|
|
with an existing XMPP client, or one is instantiated on demand.
|
|
|
|
'''
|
|
|
|
_instance = None
|
|
|
|
_sessions = dict()
|
|
|
|
_lock = threading.RLock()
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
@classmethod
|
|
|
|
def new_session_with_client(cls, client, callback=None):
|
|
|
|
'''
|
|
|
|
Opens a new session with a given client.
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
Arguments:
|
|
|
|
client -- An XMPP client.
|
|
|
|
callback -- An optional callback which can be used to track
|
|
|
|
the starting state of the session.
|
|
|
|
'''
|
2011-02-09 01:45:45 +00:00
|
|
|
with Remote._lock:
|
2011-01-13 07:40:53 +00:00
|
|
|
if(client.boundjid.bare in cls._sessions):
|
|
|
|
raise RemoteException("There already is a session associated with these credentials!")
|
|
|
|
else:
|
2014-08-17 22:52:24 +00:00
|
|
|
cls._sessions[client.boundjid.bare] = client
|
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
def _session_close_callback():
|
|
|
|
with Remote._lock:
|
|
|
|
del cls._sessions[client.boundjid.bare]
|
2011-02-09 01:45:45 +00:00
|
|
|
result = RemoteSession(client, _session_close_callback)
|
2011-10-03 18:32:48 +00:00
|
|
|
client.plugin['xep_0009'].xmpp.add_event_handler('jabber_rpc_method_call', result._on_jabber_rpc_method_call, threaded=True)
|
|
|
|
client.plugin['xep_0009'].xmpp.add_event_handler('jabber_rpc_method_response', result._on_jabber_rpc_method_response, threaded=True)
|
|
|
|
client.plugin['xep_0009'].xmpp.add_event_handler('jabber_rpc_method_fault', result._on_jabber_rpc_method_fault, threaded=True)
|
|
|
|
client.plugin['xep_0009'].xmpp.add_event_handler('jabber_rpc_error', result._on_jabber_rpc_error, threaded=True)
|
2011-01-13 07:40:53 +00:00
|
|
|
if callback is None:
|
2011-02-09 01:45:45 +00:00
|
|
|
start_event_handler = result._notify
|
2011-01-13 07:40:53 +00:00
|
|
|
else:
|
2011-02-09 01:45:45 +00:00
|
|
|
start_event_handler = callback
|
|
|
|
client.add_event_handler("session_start", start_event_handler)
|
2011-01-13 07:40:53 +00:00
|
|
|
if client.connect():
|
|
|
|
client.process(threaded=True)
|
|
|
|
else:
|
|
|
|
raise RemoteException("Could not connect to XMPP server!")
|
2011-02-09 01:45:45 +00:00
|
|
|
pass
|
2011-01-13 07:40:53 +00:00
|
|
|
if callback is None:
|
|
|
|
result._wait()
|
|
|
|
return result
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def new_session(cls, jid, password, callback=None):
|
|
|
|
'''
|
|
|
|
Opens a new session and instantiates a new XMPP client.
|
2011-02-09 01:45:45 +00:00
|
|
|
|
2011-01-13 07:40:53 +00:00
|
|
|
Arguments:
|
|
|
|
jid -- The XMPP JID for logging in.
|
|
|
|
password -- The password for logging in.
|
|
|
|
callback -- An optional callback which can be used to track
|
2011-02-09 01:45:45 +00:00
|
|
|
the starting state of the session.
|
|
|
|
'''
|
2011-01-13 07:40:53 +00:00
|
|
|
client = sleekxmpp.ClientXMPP(jid, password)
|
2011-02-09 01:45:45 +00:00
|
|
|
#? Register plug-ins.
|
2011-01-13 07:40:53 +00:00
|
|
|
client.registerPlugin('xep_0004') # Data Forms
|
|
|
|
client.registerPlugin('xep_0009') # Jabber-RPC
|
|
|
|
client.registerPlugin('xep_0030') # Service Discovery
|
|
|
|
client.registerPlugin('xep_0060') # PubSub
|
2011-02-09 01:45:45 +00:00
|
|
|
client.registerPlugin('xep_0199') # XMPP Ping
|
|
|
|
return cls.new_session_with_client(client, callback)
|
2011-01-13 07:40:53 +00:00
|
|
|
|