NAME
XML::Compile::SOAP::Daemon - SOAP accepting server (base class)
INHERITANCE
XML::Compile::SOAP::Daemon is extended by
XML::Compile::SOAP::Daemon::CGI
XML::Compile::SOAP::Daemon::NetServer
SYNOPSIS
#### have a look in the examples directory!
use XML::Compile::SOAP::Daemon::CGI;
my $daemon = XML::Compile::SOAP::Daemon::CGI->new;
# operation definitions from WSDL
my $wsdl = XML::Compile::WSDL11->new(...);
$wsdl->importDefinitions(...); # more schemas
$daemon->operationsFromWSDL($wsdl, callbacks => ...);
$daemon->setWsdlResponse($wsdl_fn);
# operation definitions added manually
my $soap11 = XML::Compile::SOAP11::Server->new(schemas => $wsdl->schemas);
my $handler = $soap11->compileHandler(...);
$daemon->addHandler('getInfo', $soap11, $handler);
DESCRIPTION
This base class implements the common needs between various types of SOAP daemons. Ache daemon can handle various kinds of SOAP protocols at the same time, when possible hidden from the user of this module.
The following extensions are implemented on the moment:
- .
-
XML::Compile::SOAP::Daemon::NetServer, for transport over HTTP based on Net::Server and LWP.
- .
-
XML::Compile::SOAP::Daemon::CGI, for transport over HTTP based on CGI and LWP.
If you have a WSDL describing your procedures (operations), then the only thing you have to worry about is adding callbacks for each of the defined ports. Without WSDL, you will need to do more manually, but it is still relatively simple to achieve.
Do not forget to take a look at the extensive example, enclosed in the XML::Compile::SOAP::Daemon distribution package. It is really worth the time.
METHODS
Constructors
- XML::Compile::SOAP::Daemon->new(OPTIONS)
-
-Option --Default accept_slow_select <true> output_charset 'UTF-8' soap_action_input {} wsa_action_input {} wsa_action_output {}
- accept_slow_select => BOOLEAN
-
Traditional SOAP does not have a simple way to find out which operation is being called. The only way to determine which operation is needed, is by trying all defined operations until one matches.
Later, people started to use the soapAction HTTP header (which was officially only for proxies) and then the WSA SOAP header extension. Either of them make it easy to determine the right handler one on one.
Disabling
accept_slow_select
will protect you against various forms of DoS-attacks, however this is often not possible as many WSDLs do not define soapAction or WSA action keys. - output_charset => STRING
-
The character-set to be used for the output XML document.
- soap_action_input => HASH|ARRAY
-
The keys are port names, with as value the related SOAPAction header field content (without quotes). Often, these SOAPAction fields originate from the WSDL.
- wsa_action_input => HASH|ARRAY
-
The keys are port names, the values are strings which are used by clients to indicate which server operation they want to use. Often, an WSDL contains this information in
wsaw:Action
attributes; that info is added to this HASH automatically when XML::Compile::SOAP::WSA is loaded. - wsa_action_output => HASH|ARRAY
-
The keys are port names, the values are strings which the server will add to replies to the client. Often, an WSDL contains this information in
wsaw:Action
attributes; that info is added to this HASH automatically when XML::Compile::SOAP::WSA is loaded.
Attributes
- $obj->addSoapAction(HASH|PAIRS)
-
Map SOAPAction headers only operations. You do not need to map explicitly when the info can be derived from the WSDL.
- $obj->addWsaTable(('INPUT'|'OUTPUT'), [HASH|PAIRS])
-
Map operation name onto respectively server-input or server-output messages, used for
wsa:Action
fields in the header. Usually, these values are automatically taken from the WSDL (but only if the WSA extension is loaded). - $obj->outputCharset
-
The character-set to be used for output documents.
Running the server
- $obj->process(CLIENT, XMLIN, REQUEST, ACTION)
-
The XMLIN SOAP-structured message (an XML::LibXML::Element, XML::LibXML::Document, or XML as string), was received from the CLIENT (some extension specific object).
The full REQUEST is passed in, however its format depends on the kind of server. The ACTION parameter relates to the soapAction header field which may be available in some form.
Returned is an XML document as answer or a protocol specific ready response object (usually an error object).
- $obj->run(OPTIONS)
-
How the daemon is run depends much on the extension being used.
Preparations
- $obj->addHandler(NAME, SOAP, CODE)
-
The SOAP value is
SOAP11
,SOAP12
, or a SOAP server object or and SOAP Operation object. The CODE reference is called with the incoming document (an XML::LibXML::Document) of the received input message.In case the handler does not understand the message, it should return undef. Otherwise, it must return a correct answer message as XML::LibXML::Document.
- $obj->operationsFromWSDL(WSDL, OPTIONS)
-
Compile the operations found in the WSDL object (an XML::Compile::WSDL11). You can add the operations from many different WSDLs into one server, simply by calling this method repeatedly.
-Option --Default callbacks {} default_callback <produces fault reply>
- callbacks => HASH
-
The keys are the port names, as defined in the WSDL. The values are CODE references which are called in case a message is received which is addressing the port (this is a guess). See "Operation handlers"
- default_callback => CODE
-
When a message arrives which has no explicit handler attached to it, this handler will be called. By default, an "not implemented" fault will be returned. See "Operation handlers"
- $obj->setWsdlResponse(FILENAME)
-
Many existing SOAP servers will reponse to GET queries which end on "?WSDL" by sending the WSDL in use by the service.
Helpers
- $obj->faultInvalidXML(ERROR)
- $obj->faultNotSoapMessage(NODETYPE)
- $obj->faultUnsupportedSoapVersion(ENV_NS)
-
Produces a text message, because we do not know how to produce an error in a SOAP which we do not understand.
- $obj->handlers(('SOAP11'|'SOAP12'|SOAP))
-
Returns all the handler names for a certain soap version.
example:
foreach my $version (sort $server->soapVersions) { foreach my $action (sort $server->handlers($version)) { print "$version $action\n"; } }
- $obj->printIndex([FILEHANDLE])
-
Print a table which shows the messages that the server can handle, by default to STDOUT.
- $obj->soapVersions
DETAILS
Operation handlers
Per operation, you define a callback which handles the request. There can also be a default callback for all your operations. Besides, when an operation does not have a handler defined, one is created for you.
sub my_callback($$)
{ my ($soap, $data_in) = @_;
return $data_out;
}
The $data_out
is a nested HASH which will be translated in the right XML structure. This could be a Fault, like shown in the next section.
Please take a look at the scripts in the example directory within the distribution.
Returning errors
Returning general errors
To have a handler return an error, leave the callback with something like this:
use XML::Compile::Util qw/pack_type/;
sub my_callback($$)
{ my ($soap, $data) = @_;
my $code = pack_type $my_err_ns, 'error-code';
return
+{ Fault =>
{ faultcode => $code
, faultstring => 'something is wrong'
, faultactor => $soap->role
}
, _RETURN_CODE => 404
, _RETURN_TEXT => 'sorry, not found'
};
}
Fault codes are "prefix:error-name", XML::Compile finds the right prefix based on the URI. If your error namespace is not mentioned in the WSDL or other loaded schemas, you should use XML::Compile::WSDL11::prefixes() first.
SOAP uses error codes in the SOAPENV namespace. It shows whether errors are client or server side. This is produced like:
use XML::Compile::SOAP::Util 'SOAP11ENV';
$code = pack_type SOAP11ENV, 'Server.validationFailed';
[release 2.02] Fields _RETURN_CODE
and _RETURN_TEXT
can be used to change the HTTP response (and maybe other protocol headers in the future). These can also be used with valid answers, not limited to errors. There is no clear definition how SOAP faults and HTTP return codes should work together for user errors.
Returning private errors
In a WSDL, we can specify own fault types. These defined elements descripe the detail
component of the message.
To return such an error, you have to figure-out how the fault part is named. Often, the name simply is fault
. Then, your handle has to return a Fault structure where the detail refers to a HASH with data matching the need for the fault. Example:
return
+{ fault => # the name of the fault part, often "fault"
{ faultcode => pack_type(SOAP11ENV, 'Server')
, faultstring => 'any-ns.WentWrong'
, faultactor => $soap->role
, detail => { message => 'Hello, World!' }
}
};
In WSDLs you may find explicitly defined error details types. There is only one such error structure per operation: when an operation may return different kinds of errors, they will be wrapped into one structure which contains the details. See section "Returning private errors" below.
Errors which do not return an details
record can always be reported with code and string. Let's first explain those.
SEE ALSO
This module is part of XML-Compile-SOAP-Daemon distribution version 3.00, built on April 15, 2011. Website: http://perl.overmeer.net/xml-compile/
Other distributions in this suite: XML::Compile, XML::Compile::SOAP, XML::Compile::SOAP12, XML::Compile::SOAP::Daemon, XML::Compile::SOAP::WSA, XML::Compile::SOAP::WSS, XML::Compile::Tester, XML::Compile::Cache, XML::Compile::Dumper, XML::Compile::RPC, XML::Rewrite, XML::eXistDB, and XML::LibXML::Simple.
Please post questions or ideas to the mailinglist at http://lists.scsys.co.uk/cgi-bin/mailman/listinfo/xml-compile For live contact with other developers, visit the #xml-compile
channel on irc.perl.org
.
LICENSE
Copyrights 2007-2011 by Mark Overmeer. For other contributors see ChangeLog.
This program is free software; you can redistribute it and/or modify it under the same terms as Perl itself. See http://www.perl.com/perl/misc/Artistic.html