2 # SFA XML-RPC and SOAP interfaces
14 from sfa.trust.auth import Auth
15 from sfa.util.config import *
16 from sfa.util.faults import *
17 from sfa.util.debug import *
18 from sfa.trust.credential import *
19 from sfa.trust.certificate import *
20 from sfa.util.namespace import *
21 from sfa.util.sfalogging import *
23 # See "2.2 Characters" in the XML specification:
25 # #x9 | #xA | #xD | [#x20-#xD7FF] | [#xE000-#xFFFD]
27 # [#x7F-#x84], [#x86-#x9F], [#xFDD0-#xFDDF]
29 invalid_xml_ascii = map(chr, range(0x0, 0x8) + [0xB, 0xC] + range(0xE, 0x1F))
30 xml_escape_table = string.maketrans("".join(invalid_xml_ascii), "?" * len(invalid_xml_ascii))
32 def xmlrpclib_escape(s, replace = string.replace):
34 xmlrpclib does not handle invalid 7-bit control characters. This
35 function augments xmlrpclib.escape, which by default only replaces
36 '&', '<', and '>' with entities.
39 # This is the standard xmlrpclib.escape function
40 s = replace(s, "&", "&")
41 s = replace(s, "<", "<")
42 s = replace(s, ">", ">",)
44 # Replace invalid 7-bit control characters with '?'
45 return s.translate(xml_escape_table)
47 def xmlrpclib_dump(self, value, write):
49 xmlrpclib cannot marshal instances of subclasses of built-in
50 types. This function overrides xmlrpclib.Marshaller.__dump so that
51 any value that is an instance of one of its acceptable types is
52 marshalled as that type.
54 xmlrpclib also cannot handle invalid 7-bit control characters. See
58 # Use our escape function
59 args = [self, value, write]
60 if isinstance(value, (str, unicode)):
61 args.append(xmlrpclib_escape)
64 # Try for an exact match first
65 f = self.dispatch[type(value)]
68 # Try for an isinstance() match
69 for Type, f in self.dispatch.iteritems():
70 if isinstance(value, Type):
73 raise TypeError, "cannot marshal %s objects" % type(value)
77 # You can't hide from me!
78 xmlrpclib.Marshaller._Marshaller__dump = xmlrpclib_dump
80 # SOAP support is optional
83 from SOAPpy.Parser import parseSOAPRPC
84 from SOAPpy.Types import faultType
85 from SOAPpy.NS import NS
86 from SOAPpy.SOAPBuilder import buildSOAP
91 def import_deep(name):
92 mod = __import__(name)
93 components = name.split('.')
94 for comp in components[1:]:
95 mod = getattr(mod, comp)
100 def __init__(self, config = "/etc/sfa/sfa_config.py", encoding = "utf-8", methods='sfa.methods',
102 peer_cert = None, interface = None, key_file = None, cert_file = None):
104 self.encoding = encoding
106 # flat list of method names
108 self.methods_module = methods_module = __import__(methods, fromlist=[methods])
109 self.methods = methods_module.all
111 # Better just be documenting the API
116 self.config = Config(config)
117 self.auth = Auth(peer_cert)
118 self.interface = interface
119 self.key_file = key_file
120 self.key = Keypair(filename=self.key_file)
121 self.cert_file = cert_file
122 self.cert = Certificate(filename=self.cert_file)
123 self.credential = None
125 self.time_format = "%Y-%m-%d %H:%M:%S"
126 self.logger=get_sfa_logger()
129 def callable(self, method):
131 Return a new instance of the specified method.
134 if method not in self.methods:
135 raise SfaInvalidAPIMethod, method
137 # Get new instance of method
139 classname = method.split(".")[-1]
140 module = __import__(self.methods_module.__name__ + "." + method, globals(), locals(), [classname])
141 callablemethod = getattr(module, classname)(self)
142 return getattr(module, classname)(self)
143 except ImportError, AttributeError:
144 raise SfaInvalidAPIMethod, method
146 def call(self, source, method, *args):
148 Call the named method from the specified source with the
151 function = self.callable(method)
152 function.source = source
153 return function(*args)
155 def handle(self, source, data):
157 Handle an XML-RPC or SOAP request from the specified source.
159 # Parse request into method name and arguments
161 interface = xmlrpclib
162 (args, method) = xmlrpclib.loads(data)
163 methodresponse = True
165 if SOAPpy is not None:
167 (r, header, body, attrs) = parseSOAPRPC(data, header = 1, body = 1, attrs = 1)
170 # XXX Support named arguments
175 result = self.call(source, method, *args)
176 except Exception, fault:
177 traceback.print_exc(file = log)
178 # Handle expected faults
179 if interface == xmlrpclib:
181 methodresponse = None
182 elif interface == SOAPpy:
183 result = faultParameter(NS.ENV_T + ":Server", "Method Failed", method)
184 result._setDetail("Fault %d: %s" % (fault.faultCode, fault.faultString))
189 if interface == xmlrpclib:
190 if not isinstance(result, SfaFault):
193 data = xmlrpclib.dumps(result, methodresponse = True, encoding = self.encoding, allow_none = 1)
194 elif interface == SOAPpy:
195 data = buildSOAP(kw = {'%sResponse' % method: {'Result': result}}, encoding = self.encoding)