代码拉取完成,页面将自动刷新
/*
wsaapi.c
WS-Addressing plugin.
gSOAP XML Web services tools
Copyright (C) 2000-2015, Robert van Engelen, Genivia Inc., All Rights Reserved.
This part of the software is released under one of the following licenses:
GPL, the gSOAP public license, or Genivia's license for commercial use.
--------------------------------------------------------------------------------
gSOAP public license.
The contents of this file are subject to the gSOAP Public License Version 1.3
(the "License"); you may not use this file except in compliance with the
License. You may obtain a copy of the License at
http://www.cs.fsu.edu/~engelen/soaplicense.html
Software distributed under the License is distributed on an "AS IS" basis,
WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
for the specific language governing rights and limitations under the License.
The Initial Developer of the Original Code is Robert A. van Engelen.
Copyright (C) 2000-2015, Robert van Engelen, Genivia Inc., All Rights Reserved.
--------------------------------------------------------------------------------
GPL license.
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 2 of the License, or (at your option) any later
version.
This program is distributed in the hope that it will be useful, but WITHOUT ANY
WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A
PARTICULAR PURPOSE. See the GNU General Public License for more details.
You should have received a copy of the GNU General Public License along with
this program; if not, write to the Free Software Foundation, Inc., 59 Temple
Place, Suite 330, Boston, MA 02111-1307 USA
Author contact information:
engelen@genivia.com / engelen@acm.org
This program is released under the GPL with the additional exemption that
compiling, linking, and/or using OpenSSL is allowed.
--------------------------------------------------------------------------------
A commercial use license is available from Genivia, Inc., contact@genivia.com
--------------------------------------------------------------------------------
*/
/**
@mainpage
- @ref wsa_0 documents the wsa plugin for WS-Addressing (2003/2004/2005
standards) support.
*/
/**
@page wsa_0 The WS-Addressing Plugin
[TOC]
@section wsa_1 WS-Addressing Setup
The material in this section relates to the WS-Addressing specification.
To use the wsa plugin:
-# Run wsdl2h -t typemap.dat on a WSDL of a service that requires WS-Addressing
headers. The typemap.dat file included in the gSOAP package is used to
recognize and translate Addressing header blocks.
-# Run soapcpp2 -a on the header file produced by wsdl2h. To enable
addressing-based service operation selection, you MUST use soapcpp2 option
-a. This allows the service to dispatch methods based on the WS-Addressing
action information header value (when the wsa plugin is registered).
-# (Re-)compile and link stdsoap2.c/pp or libgsoap, (dom.c/pp when needed),
wsaapi.c and the soapcpp2-generated source files.
-# Use the wsa plugin API functions described below.
An example wsa client/server application can be found in `gsoap/samples/wsa`.
A gSOAP service definitions header file with a `#import "wsa.h"` to support
WS-Addressing is automatically generated by wsdl2h for a set of WSDLs that use
WS-Addressing. The wsdl2h-generated header file should be further processed by
soapcpp2 to generate the binding code. The wsaapi.h and wsaapi.c implement the
WS-Addressing API described in this document.
A wsdl2h-generated service definitions header file might include the following
imports:
@code
#import "soap12.h"
#import "wsa.h" // or wsa3.h (2003/03), wsa4.h (2004/03), wsa5.h (2005/03)
@endcode
The wsa.h header file is imported from import/wsa.h when soapcpp2 is run on
this file. The wsa.h import can be manually added to enable WS-Addressing when
needed. The gSOAP service definitions header file is processed with soapcpp2 to
generate the client-side and/or server-side binding code.
Note that the wsa.h, wsa3.h, wsa4.h, and wsa5.h header files are located in the
import directory of the gSOAP package. These files define the WS-Addressing
information header elements and types. The soap12.h header file enables SOAP
1.2 messaging.
For developers: the WS-Addressing header blocks in wsa.h (and others) were
generated from the WS-Addressing schema with the wsdl2h tool and
WS/WS-typemap.dat as follows:
wsdl2h -cegy -o wsa.h -t WS/WS-typemap.dat WS/WS-Addressing.xsd
Refer to wsa.h for more details.
@section wsa_2 Client-side Usage
@subsection wsa_2_1 Constructing WS-Addressing Information Headers
To associate WS-Addressing information headers with service operations, the
SOAP Header struct `SOAP_ENV__Header` must have been defined and for each service
operation that uses WS-Addressing method-header-part directives should be used
in the gSOAP service definitions header file as follows:
@code
#import "wsa.h"
//gsoap ns service method-header-part: example wsa__MessageID
//gsoap ns service method-header-part: example wsa__RelatesTo
//gsoap ns service method-header-part: example wsa__From
//gsoap ns service method-header-part: example wsa__ReplyTo
//gsoap ns service method-header-part: example wsa__FaultTo
//gsoap ns service method-header-part: example wsa__To
//gsoap ns service method-header-part: example wsa__Action
//gsoap ns service method-action: example urn:example/examplePort/example
int ns__example(char *in, struct ns__exampleResponse *out);
@endcode
Note that the use of wsa versions determines the wsa prefix, e.g. use `wsa5`
for the latest WS-Addressing as in `wsa5__MessageID`.
In the client-side code, the WS-Addressing information headers are set with
`soap_wsa_request` by passing an optional message UUID string, a mandatory
destination address URI string, and a mandatory request action URI string. The
wsa plugin should be registered with the current soap struct context. An
optional source address information header can be added with
`soap_wsa_add_From` (must be invoked after the `soap_wsa_request` call).
For example:
@code
#include "wsaapi.h"
soap_register_plugin(soap, soap_wsa);
if (soap_wsa_request(soap, RequestMessageID, ToAddress, RequestAction))
|| soap_wsa_add_From(soap, FromAddress)) // optional: add a 'From' address
... // error: out of memory
if (soap_call_ns__example(soap, ToAddress, NULL, ...))
soap_print_fault(soap, stderr); // an error occurred
else
// process the response
@endcode
To generate a UUID for the RequestMessageID, use:
@code
const char *RequestMessageID = soap_wsa_rand_uuid(soap);
@endcode
@subsection wsa_2_2 Information Headers for Relaying Server Responses
To relay the response to another destination, the WS-Addressing ReplyTo
information header is added with `soap_wsa_add_ReplyTo` by passing a reply
address URI string. The service returns "HTTP 202 ACCEPTED" to the client when
the response message relay was successful.
For example:
@code
#include "wsaapi.h"
soap_register_plugin(soap, soap_wsa);
if (soap_wsa_request(soap, RequestMessageID, ToAddress, RequestAction)
|| soap_wsa_add_From(soap, FromAddress) // optional: add a 'From' address
|| soap_wsa_add_ReplyTo(soap, ReplyToAddress))
... // error: out of memory
if (soap_call_ns__example(soap, ToAddress, NULL, ...))
{
if (soap->error == 202) // HTTP ACCEPTED
printf("Request was accepted and results were forwarded\n");
else
soap_print_fault(soap, stderr); // an error occurred
}
else
// unexpected OK: for some reason the response was not relayed
@endcode
Note: the response message will be relayed when the From address is absent or
different than the ReplyTo address
@subsection wsa_2_3 Information Headers for Relaying Server Faults
To relay a server fault message to another destination, the WS-Addressing
FaultTo information header is added with `soap_wsa_add_FaultTo` by passing a
relay address URI string. The service returns "HTTP 202 ACCEPTED" to the client
when the fault was relayed.
For example:
@code
#include "wsaapi.h"
soap_register_plugin(soap, soap_wsa);
if (soap_wsa_request(soap, RequestMessageID, ToAddress, RequestAction)
|| soap_wsa_add_From(soap, FromAddress) // optional: add a 'From' address
|| soap_wsa_add_FaultTo(soap, FaultToAddress))
... // error: out of memory
if (soap_call_ns__example(soap, ToAddress, NULL, ...))
{
if (soap->error == 202) // HTTP ACCEPTED
printf("A fault occurred and the fault details were forwarded\n");
else
soap_print_fault(soap, stderr); // a connection error occurred
}
else
... // process response
@endcode
Note that the call can still return a fault, such as a connection error when
the service is not responding. In addition to the fault relay, the responses
can be relayed with `soap_wsa_add_ReplyTo`.
@subsection wsa_2_4 Error Handling
SOAP and HTTP errors set the soap->error attribute, as shown in this example:
@code
if (soap_call_ns__example(soap, ToAddress, NULL, ...))
{
if (soap->error == 202) // HTTP ACCEPTED
printf("A fault occurred and the fault details were forwarded\n");
else
soap_print_fault(soap, stderr); // a connection error occurred
}
else
... // process response
@endcode
When a WS-Addressing error occurred, the wsa error code is stored in the SOAP
Fault Subcode field. This information can be retrieved with:
@code
wsa__FaultSubcodeValues fault;
if (soap_wsa_check_fault(soap, &fault))
{
switch (fault)
{
case wsa__InvalidMessageInformationHeader: ...
case wsa__MessageInformationHeaderRequired: ...
case wsa__DestinationUreachable: ...
case wsa__ActionNotSupported: ...
case wsa__EndpointUnavailable: ...
}
}
@endcode
When using wsa5.h, please refer to the standards and fault codes for this
implementation. For the wsa5.h 2005/03 standard, several faults have an
additional parameter (SOAP Fault detail):
@code
wsa5__FaultCodesType fault;
char *info;
if (soap_wsa_check_fault(soap, &fault, &info))
{
switch (fault)
{
case wsa5__InvalidAddressingHeader:
if (info)
printf("The invalid addressing header element is %s\n", info);
...
}
}
@endcode
@subsection wsa_2_5 Combining WS-Addressing with WS-Security
WS-Security can be combined with WS-Addressing. To sign WS-Addressing header
blocks, use the `soap_wsse_set_wsu_id` WSSE-plugin call to set the wsu:Id
attribute and signing of these attributed elements. For example, suppose we use
WS-Addressing 2005 headers (which are activated with an `#import "wsa5.h"` in
the header file for soapcpp2):
@code
#include "wsaapi.h"
#include "wsseapi.h"
soap_register_plugin(soap, soap_wsa);
soap_register_plugin(soap, soap_wsse);
soap_wsse_set_wsu_id(soap, "wsa5:From wsa5:To wsa5:ReplyTo wsa5:FaultTo wsa5:Action wsa5:MessageID");
if (soap_wsa_request(soap, RequestMessageID, ToAddress, RequestAction)
|| soap_wsa_add_From(soap, FromAddress) // optional: add a 'From' address
|| soap_wsa_add_FaultTo(soap, FaultToAddress))
... // error: out of memory
if (soap_call_ns__example(soap, ToAddress, NULL, ...))
... // error
@endcode
If your are using WS-Addressing 2004 (which is activated with an
`#import "wsa.h"` in the header file for soapcpp2) then change one line:
@code
soap_wsse_set_wsu_id(soap, "wsa:From wsa:To wsa:ReplyTo wsa:FaultTo wsa:Action wsa:MessageID");
@endcode
Note: `soap_wsse_set_wsu_id` should only be set once. Each new call overrides
the previous.
For more details on WS-Security, please see the
[WS-Security plugin documentation](../../wsse/html/index.html).
@section wsa_3 Server-side Usage
The wsa plugin should be registered with:
@code
soap_register_plugin(soap, soap_wsa);
@endcode
Once the plugin is registered, the `soap_bind`, `soap_accept`, and `soap_serve`
functions can be called to process requests and semi-automatically handle the
WS-Addressing header blocks.
Important: to dispatch service operations based on the WS-Addressing wsa:Action
information header, you must use soapcpp2 option `-a`. The generates a new
dispatcher (in soapServer.c) based on the action value.
A service operation implementation should use `soap_wsa_check` at the start of
its execution to verify the validity of the WS-Addressing information headers
in the SOAP request message. To allow response message to be automatically
relayed based on the ReplyTo information header, the service operation should
return `soap_wsa_reply` with an optional message UUID string and a mandatory
response action string. The response action string is documented in the
wsdl2h-generated .h file for this service operation.
For example:
@code
int ns__example(struct soap *soap, char *in, struct ns__exampleResponse *out)
{
if (soap_wsa_check(soap))
return soap->error;
// ... service logic
return soap_wsa_reply(soap, ResponseMessageID, ResponseAction);
}
@endcode
To return a SOAP fault that is automatically relayed to a fault service based
on the FaultTo information header, the `soap_wsa_sender_fault`,
`soap_wsa_receiver_fault`, `soap_wsa_sender_fault_subcode`, and
`soap_wsa_receiver_fault_subcode` functions should be used instead of the
usual `soap_sender_fault`, `soap_receiver_fault`, `soap_sender_fault_subcode`,
and `soap_receiver_fault_subcode`, respectively:
In case a Action must be associated with a SOAP Fault, use the
`soap_wsa_sender_fault_subcode_action` and
`soap_wsa_receiver_fault_subcode_action` functions to set the WS-Addressing
Action (and HTTP SOAP Action header as well).
For example, the following service operation illustrates the use of
`soap_wsa_check` to verify and process WS-Addressing header blocks and
`soap_wsa_reply` to enable responses to be relayed as per ReplyTo address in
the WS-Addressing header:
@code
int ns__example(struct soap *soap, char *in, struct ns__exampleResponse *out)
{
if (soap_wsa_check(soap))
return soap->error;
// ... service logic
// ... an error occurred, need to return fault possibly to fault service:
return soap_wsa_sender_fault(soap, "Exception in service operation", NULL);
// ... normal execution continues
return soap_wsa_reply(soap, ResponseMessageID, ResponseAction);
}
@endcode
@section wsa_4 HTTPS Server-side Usage
To enable HTTPS (SSL/TSL) servers, compile the sources with -DWITH_OPENSSL (and
link with libgsoapssl, libssl, and libcrypto). Because WS-Addressing may relay
messages over HTTPS as a sender (client), you must initialize the SSL context
for server and client uses. Therefore, the context must have access to all the
certificates need to verify the authenticity of the ReplyTo and FaultTo HTTPS
servers. To do so, use the following SSL initialization before `soap_bind`:
@code
struct soap *soap = soap_new();
if (soap_ssl_server_context(soap,
SOAP_SSL_DEFAULT,
"server.pem", // the keyfile (server should authenticate)
"password", // password to read the key file
"cacert.pem", // cacert file to store trusted certificates (role as client)
NULL, // optional capath
NULL, // DH file name or DH param key len bits, when NULL use RSA
NULL, // file with random data to seed randomness
"myserver" // unique server identification for SSL session cache
))
{
soap_print_fault(soap, stderr);
...
}
soap->bind_flags = SO_REUSEADDR;
if (!soap_valid_socket(soap_bind(soap, NULL, port, 100)))
{
soap_print_fault(soap, stderr);
...
}
@endcode
@section wsa_5 Implementing a Server for Handling ReplyTo Response Messages
To implement a separate server for handling relayed SOAP response messages
based on the ReplyTo information header in the request message, the gSOAP
header file should include a one-way service operation for the response
message. These one-way response service operations are automatically generated
with wsdl2h option `-b`.
For example, suppose a service operation returns an exampleResponse message. We
declare the one-way exampleResponse operation as follows:
@code
#import "wsa.h"
//gsoap ns service method-header-part: exampleResult wsa__MessageID
//gsoap ns service method-header-part: exampleResult wsa__RelatesTo
//gsoap ns service method-header-part: exampleResult wsa__From
//gsoap ns service method-header-part: exampleResult wsa__ReplyTo
//gsoap ns service method-header-part: exampleResult wsa__FaultTo
//gsoap ns service method-header-part: exampleResult wsa__To
//gsoap ns service method-header-part: exampleResult wsa__Action
//gsoap ns service method-action: exampleResult urn:example/examplePort/exampleResponse
int ns__exampleResponse(char *out, void);
@endcode
Note that the action information is important, because it is used by the
service dispatcher (assuming soapcpp2 option `-a` is used).
The implementation in the server code uses soap_wsa_check() to check the
presense and validity of the WS-Addressing information header in the message.
The `soap_send_empty_response` function should be used to return an
acknowledgment HTTP header with "HTTP 202 ACCEPTED" to the sender:
@code
int ns__exampleResponse(struct soap *soap, char *out)
{
if (soap_wsa_check(soap))
return soap_send_empty_response(soap, 500); // HTTP 500 Internal Server Error
// ... service logic
return soap_send_empty_response(soap, SOAP_OK); // HTTP 202 ACCEPTED
}
@endcode
@section wsa_6 Implementing a Server for Handling FaultTo Fault Messages
To implement a separate server for handling relayed SOAP fault messages based
on the FaultTo information header in the request message, the gSOAP header file
for soapcpp2 should include a SOAP fault service operation. This operation
accepts fault messages that are relayed by other services.
Basically, we use a trick to generate the SOAP-ENV:Fault struct via a one-way
service operation. This allows us both to implement a one-way service operation
that accepts faults and to automatically generate the fault struct for fault
data storage and manipulation.
The fault operation in the WS-Addressing files (wsa5.h etc.) is declared as
follows (here shown for the 2004/08 standard):
@code
//gsoap SOAP_ENV service method-action: Fault http://schemas.xmlsoap.org/ws/2004/08/addressing/fault
int SOAP_ENV__Fault
(
_QName faultcode, // SOAP 1.1
char *faultstring, // SOAP 1.1
char *faultactor, // SOAP 1.1
struct SOAP_ENV__Detail *detail, // SOAP 1.1
struct SOAP_ENV__Code *SOAP_ENV__Code, // SOAP 1.2
struct SOAP_ENV__Reason *SOAP_ENV__Reason, // SOAP 1.2
char *SOAP_ENV__Node, // SOAP 1.2
char *SOAP_ENV__Role, // SOAP 1.2
struct SOAP_ENV__Detail *SOAP_ENV__Detail, // SOAP 1.2
void
);
@endcode
Because each service operation has a struct to hold its input parameters, we
automatically generate the (original) `SOAP_ENV__Fault` struct on the fly!
It is important to associate the wsa fault action with this operation as shown
above.
The implementation of the Fault service operation in your server code should be
similar to:
@code
int SOAP_ENV__Fault(struct soap *soap, char *faultcode, char *faultstring, char *faultactor, struct SOAP_ENV__Detail *detail, struct SOAP_ENV__Code *SOAP_ENV__Code, struct SOAP_ENV__Reason *SOAP_ENV__Reason, char *SOAP_ENV__Node, char *SOAP_ENV__Role, struct SOAP_ENV__Detail *SOAP_ENV__Detail)
{
... = faultcode; // SOAP 1.1 fault code string (QName)
... = faultstring; // SOAP 1.1 fault string
... = faultactor; // SOAP 1.1 fault actor string
... = detail; // SOAP 1.1 fault detail struct
... = SOAP_ENV__Code; // SOAP 1.2 fault code struct
... = SOAP_ENV__Reason; // SOAP 1.2 reason struct
... = SOAP_ENV__Node; // SOAP 1.2 node string
... = SOAP_ENV__Role; // SOAP 1.2 role string
... = SOAP_ENV__Detail; // SOAP 1.2 detail struct
return SOAP_OK;
}
@endcode
Note that SOAP 1.1 or SOAP 1.2 parameters are set based on the 1.1/1.2
messaging requirements.
*/
#include "wsaapi.h"
#ifdef __cplusplus
extern "C" {
#endif
/** Plugin identification for plugin registry */
const char soap_wsa_id[] = SOAP_WSA_ID;
#if defined(SOAP_WSA_2003)
/** Anonymous Reply/To endpoint address */
const char *soap_wsa_anonymousURI = "http://schemas.xmlsoap.org/ws/2003/03/addressing/role/anonymous";
/** Specifies no Reply endpoint address (no reply) */
const char *soap_wsa_noneURI = "addressing/none not supported";
const char *soap_wsa_faultAction = "http://schemas.xmlsoap.org/ws/2003/03/addressing/fault";
#elif defined(SOAP_WSA_2004)
/** Anonymous Reply/To endpoint address */
const char *soap_wsa_anonymousURI = "http://schemas.xmlsoap.org/ws/2004/03/addressing/role/anonymous";
/** Specifies no Reply endpoint address (no reply) */
const char *soap_wsa_noneURI = "addressing/none not supported";
const char *soap_wsa_faultAction = "http://schemas.xmlsoap.org/ws/2004/03/addressing/fault";
#elif defined(SOAP_WSA_2005)
/** Anonymous Reply/To endpoint address */
const char *soap_wsa_anonymousURI = "http://www.w3.org/2005/08/addressing/anonymous";
/** Specifies no Reply endpoint address (no reply) */
const char *soap_wsa_noneURI = "http://www.w3.org/2005/08/addressing/none";
const char *soap_wsa_faultAction = "http://www.w3.org/2005/08/addressing/soap/fault";
#else
/** Anonymous Reply/To endpoint address */
const char *soap_wsa_anonymousURI = "http://schemas.xmlsoap.org/ws/2004/08/addressing/role/anonymous";
/** Specifies no Reply endpoint address (no reply) */
const char *soap_wsa_noneURI = "addressing/none not supported";
const char *soap_wsa_faultAction = "http://schemas.xmlsoap.org/ws/2004/08/addressing/fault";
#endif
/** anonymous URI of 2004 and 2005 schemas */
const char *soap_wsa_allAnonymousURI = "http://schemas.xmlsoap.org/ws/2004/03/addressing/role/anonymous http://schemas.xmlsoap.org/ws/2004/08/addressing/role/anonymous http://www.w3.org/2005/08/addressing/anonymous";
/******************************************************************************\
*
* Static protos
*
\******************************************************************************/
static int soap_wsa_init(struct soap *soap, struct soap_wsa_data *data);
static void soap_wsa_delete(struct soap *soap, struct soap_plugin *p);
static int soap_wsa_header(struct soap *soap);
static void soap_wsa_set_error(struct soap *soap, const char **c, const char **s);
static int soap_wsa_response(struct soap *soap, int status, ULONG64 count);
static int soap_wsa_disconnect(struct soap *soap);
static int soap_wsa_alloc_header(struct soap *soap);
/******************************************************************************\
*
* UUID
*
\******************************************************************************/
/**
@fn const char *soap_wsa_rand_uuid(struct soap *soap)
@brief Generates a random UUID (UUID algorithm version 4). Compile all source
codes with -DWITH_OPENSSL for better randomness.
@param soap context
@return UUID "urn:uuid:xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx" or NULL if out of memory
*/
SOAP_FMAC1
const char*
SOAP_FMAC2
soap_wsa_rand_uuid(struct soap *soap)
{
return soap_strdup(soap, soap_rand_uuid(soap, "urn:uuid:"));
}
/******************************************************************************\
*
* Client-side Request
*
\******************************************************************************/
/**
@fn int soap_wsa_request(struct soap *soap, const char *id, const char *to, const char *action)
@brief Sets the WS-Addressing information header for the next request message
with MessageID (optional), To (optional), and Action (required).
@param soap context
@param[in] id is the message ID (optional)
@param[in] to is the target endpoint (optional, set to anonymous when NULL)
@param[in] action is the target action (required)
@return SOAP_OK or error
Note: use soap_wsa_add_From, soap_wsa_add_ReplyTo, soap_wsa_add_FaultTo to add
other addressing fields following this function call.
*/
SOAP_FMAC1
int
SOAP_FMAC2
soap_wsa_request(struct soap *soap, const char *id, const char *to, const char *action)
{
DBGFUN3("soap_wsa_request", "id=%s", id?id:"(null)", "to=%s", to?to:"(null)", "action=%s", action?action:"(null)");
if (soap_wsa_alloc_header(soap))
return soap->error;
soap->header->SOAP_WSA(MessageID) = soap_strdup(soap, id);
if (to)
soap->header->SOAP_WSA(To) = soap_strdup(soap, to);
else /* this is optional */
soap->header->SOAP_WSA(To) = (char*)soap_wsa_anonymousURI;
soap->header->SOAP_WSA(Action) = soap_strdup(soap, action);
soap->header->SOAP_WSA(RelatesTo) = NULL;
soap->header->SOAP_WSA(From) = NULL;
soap->header->SOAP_WSA(FaultTo) = NULL;
soap_wsa_add_ReplyTo(soap, NULL);
return soap_wsa_check(soap);
}
/******************************************************************************/
/**
@fn int soap_wsa_add_From(struct soap *soap, const char *from)
@brief Sets WS-Addressing From header for request message.
@param soap context
@param[in] from endpoint URI
@return SOAP_OK or SOAP_ERR
Use soap_wsa_request to populate the WS-Addressing header first.
*/
SOAP_FMAC1
int
SOAP_FMAC2
soap_wsa_add_From(struct soap *soap, const char *from)
{
if (!soap->header)
return SOAP_ERR;
soap->header->SOAP_WSA(From) = (SOAP_WSA_(,From)*)soap_malloc(soap, sizeof(SOAP_WSA_(,From)));
SOAP_WSA_(soap_default,EndpointReferenceType)(soap, soap->header->SOAP_WSA(From));
soap->header->SOAP_WSA(From)->Address = soap_strdup(soap, from);
return SOAP_OK;
}
/******************************************************************************/
/**
@fn int soap_wsa_add_NoReply(struct soap *soap)
@brief Sets WS-Addressing ReplyTo header to 'none' (no reply)
@param soap context
@return SOAP_OK or SOAP_ERR
Note: WS-Addressing 2005/08 standard.
Use soap_wsa_request to populate the WS-Addressing header.
*/
SOAP_FMAC1
int
SOAP_FMAC2
soap_wsa_add_NoReply(struct soap *soap)
{
return soap_wsa_add_ReplyTo(soap, soap_wsa_noneURI);
}
/******************************************************************************/
/**
@fn int soap_wsa_add_ReplyTo(struct soap *soap, const char *replyTo)
@brief Sets WS-Addressing ReplyTo header for request message.
@param soap context
@param[in] replyTo endpoint URI or NULL for anonymous
@return SOAP_OK or SOAP_ERR
Use soap_wsa_request to populate the WS-Addressing header.
*/
SOAP_FMAC1
int
SOAP_FMAC2
soap_wsa_add_ReplyTo(struct soap *soap, const char *replyTo)
{
if (!soap->header)
return SOAP_ERR;
#ifndef SOAP_WSA_2005
if (!replyTo)
replyTo = soap_wsa_anonymousURI;
#endif
if (replyTo)
{
soap->header->SOAP_WSA(ReplyTo) = (SOAP_WSA_(,ReplyTo)*)soap_malloc(soap, sizeof(SOAP_WSA_(,ReplyTo)));
SOAP_WSA_(soap_default,EndpointReferenceType)(soap, soap->header->SOAP_WSA(ReplyTo));
soap->header->SOAP_WSA(ReplyTo)->Address = soap_strdup(soap, replyTo);
}
else
soap->header->SOAP_WSA(ReplyTo) = NULL;
return SOAP_OK;
}
/******************************************************************************/
/**
@fn int soap_wsa_add_FaultTo(struct soap *soap, const char *faultTo)
@brief Sets WS-Addressing FaultTo header for request message.
@param soap context
@param[in] faultTo endpoint URI or NULL for remove faultTo
@return SOAP_OK or SOAP_ERR
Use soap_wsa_request to populate the WS-Addressing header first.
*/
SOAP_FMAC1
int
SOAP_FMAC2
soap_wsa_add_FaultTo(struct soap *soap, const char *faultTo)
{
if (!soap->header)
return SOAP_ERR;
if (faultTo)
{
soap->header->SOAP_WSA(FaultTo) = (SOAP_WSA_(,FaultTo)*)soap_malloc(soap, sizeof(SOAP_WSA_(,FaultTo)));
SOAP_WSA_(soap_default,EndpointReferenceType)(soap, soap->header->SOAP_WSA(FaultTo));
soap->header->SOAP_WSA(FaultTo)->Address = soap_strdup(soap, faultTo);
}
else
soap->header->SOAP_WSA(FaultTo) = NULL;
return SOAP_OK;
}
/******************************************************************************/
/**
@fn int soap_wsa_add_RelatesTo(struct soap *soap, const char *relatesTo)
@brief Sets WS-Addressing RelatesTo header.
@param soap context
@param[in] relatesTo endpoint URI
@return SOAP_OK or SOAP_ERR
Use soap_wsa_request to populate the WS-Addressing header.
*/
SOAP_FMAC1
int
SOAP_FMAC2
soap_wsa_add_RelatesTo(struct soap *soap, const char *relatesTo)
{
if (!soap->header)
return SOAP_ERR;
if (relatesTo)
{
soap->header->SOAP_WSA(RelatesTo) = (SOAP_WSA_(,RelatesTo)*)soap_malloc(soap, sizeof(SOAP_WSA_(,RelatesTo)));
SOAP_WSA_(soap_default_,RelatesTo)(soap, soap->header->SOAP_WSA(RelatesTo));
soap->header->SOAP_WSA(RelatesTo)->__item = soap_strdup(soap, relatesTo);
}
return SOAP_OK;
}
/******************************************************************************/
/**
@fn const char *soap_wsa_From(struct soap *soap)
@brief Returns WS-Addressing From header.
@param soap context
@return From string or NULL
*/
SOAP_FMAC1
const char*
SOAP_FMAC2
soap_wsa_From(struct soap *soap)
{
if (!soap->header || !soap->header->SOAP_WSA(From))
return NULL;
return soap->header->SOAP_WSA(From)->Address;
}
/******************************************************************************/
/**
@fn const char *soap_wsa_ReplyTo(struct soap *soap)
@brief Returns WS-Addressing ReplyTo header.
@param soap context
@return From string or NULL
*/
SOAP_FMAC1
const char*
SOAP_FMAC2
soap_wsa_ReplyTo(struct soap *soap)
{
if (!soap->header || !soap->header->SOAP_WSA(ReplyTo))
return NULL;
return soap->header->SOAP_WSA(ReplyTo)->Address;
}
/******************************************************************************/
/**
@fn const char *soap_wsa_FaultTo(struct soap *soap)
@brief Returns WS-Addressing FaultTo header.
@param soap context
@return From string or NULL
*/
SOAP_FMAC1
const char*
SOAP_FMAC2
soap_wsa_FaultTo(struct soap *soap)
{
if (!soap->header || !soap->header->SOAP_WSA(FaultTo))
return NULL;
return soap->header->SOAP_WSA(FaultTo)->Address;
}
/******************************************************************************/
/**
@fn const char *soap_wsa_RelatesTo(struct soap *soap)
@brief Returns WS-Addressing RelatesTo header.
@param soap context
@return From string or NULL
*/
SOAP_FMAC1
const char*
SOAP_FMAC2
soap_wsa_RelatesTo(struct soap *soap)
{
if (!soap->header || !soap->header->SOAP_WSA(RelatesTo))
return NULL;
return soap->header->SOAP_WSA(RelatesTo)->__item;
}
/******************************************************************************\
*
* Server-side Check and Reply
*
\******************************************************************************/
/**
@fn int soap_wsa_check(struct soap *soap)
@brief Checks the presence and validity of WS-Addressing information headers.
@param soap context
@return SOAP_OK or fault
*/
SOAP_FMAC1
int
SOAP_FMAC2
soap_wsa_check(struct soap *soap)
{
DBGFUN("soap_wsa_check");
if (!soap->header || !soap->header->SOAP_WSA(Action))
#if defined(SOAP_WSA_2005)
return soap_wsa_error(soap, wsa5__MessageAddressingHeaderRequired, NULL);
#elif defined(SOAP_WSA_2003)
return soap_wsa_error(soap, "WS-Addressing header missing");
#else
return soap_wsa_error(soap, SOAP_WSA(MessageInformationHeaderRequired));
#endif
return SOAP_OK;
}
/******************************************************************************/
/**
@fn int soap_wsa_reply(struct soap *soap, const char *id, const char *action)
@brief Sets WS-Addressing header fields for server response. Automatically
relays the response to the ReplyTo address (when ReplyTo != to From and ReplyTo
!= 'none') and returns HTTP 202 Accept to sender when relay was successful.
@param soap context
@param[in] id is the messageID (optional)
@param[in] action is the target action (required)
@return SOAP_OK or fault
*/
SOAP_FMAC1
int
SOAP_FMAC2
soap_wsa_reply(struct soap *soap, const char *id, const char *action)
{
struct soap_wsa_data *data = (struct soap_wsa_data*)soap_lookup_plugin(soap, soap_wsa_id);
struct SOAP_ENV__Header *oldheader, *newheader;
DBGFUN1("soap_wsa_reply", "action=%s", action?action:"(null)");
if (!data)
return soap->error = SOAP_PLUGIN_ERROR;
oldheader = soap->header;
soap->header = NULL;
/* if endpoint address for reply is 'none' return immediately and STOP engine */
if (oldheader && oldheader->SOAP_WSA(ReplyTo) && oldheader->SOAP_WSA(ReplyTo)->Address && !strcmp(oldheader->SOAP_WSA(ReplyTo)->Address, soap_wsa_noneURI))
return soap_send_empty_response(soap, SOAP_OK);
/* allocate a new header */
if (soap_wsa_alloc_header(soap))
return soap->error;
newheader = soap->header;
/* copy members to new header, except WSA data */
if (oldheader)
*newheader = *oldheader;
newheader->SOAP_WSA(MessageID) = soap_strdup(soap, id);
newheader->SOAP_WSA(Action) = soap_strdup(soap, action);
newheader->SOAP_WSA(RelatesTo) = NULL;
newheader->SOAP_WSA(From) = NULL;
newheader->SOAP_WSA(To) = NULL;
newheader->SOAP_WSA(ReplyTo) = NULL;
newheader->SOAP_WSA(FaultTo) = NULL;
/* check current header content */
if (oldheader && oldheader->SOAP_WSA(MessageID))
{
newheader->SOAP_WSA(RelatesTo) = (SOAP_WSA_(,RelatesTo)*)soap_malloc(soap, sizeof(SOAP_WSA_(,RelatesTo)));
SOAP_WSA_(soap_default_,RelatesTo)(soap, newheader->SOAP_WSA(RelatesTo));
newheader->SOAP_WSA(RelatesTo)->__item = oldheader->SOAP_WSA(MessageID);
}
#ifdef SOAP_WSA_2005
/* WCF Interoperability:
ChannelInstance is required when the WCF Application hosts multiple
Callback Channels within the same application. The
ReferenceParameters->ChannelInstance element serves as a hint to the WCF
Client dispatcher, as to which WCF callback instance a received SOAP
Envelope belongs to ChannelInstance is declared as a pointer, so it is
essentially an optional element. Tests with Applications not requiring
ChannelInstance have also been done for the following fix.
*/
if (oldheader && oldheader->SOAP_WSA(ReplyTo) && oldheader->SOAP_WSA(ReplyTo)->ReferenceParameters && oldheader->SOAP_WSA(ReplyTo)->ReferenceParameters->chan__ChannelInstance)
{
if (!newheader->chan__ChannelInstance)
{
newheader->chan__ChannelInstance = (struct chan__ChannelInstanceType*)soap_malloc(soap, sizeof(struct chan__ChannelInstanceType));
if (newheader->chan__ChannelInstance)
{
soap_default_chan__ChannelInstanceType(soap, newheader->chan__ChannelInstance);
newheader->chan__ChannelInstance->__item = *(oldheader->SOAP_WSA(ReplyTo)->ReferenceParameters->chan__ChannelInstance);
newheader->chan__ChannelInstance->wsa5__IsReferenceParameter = _wsa5__IsReferenceParameter__true;
}
}
else
{
newheader->chan__ChannelInstance->__item = *(oldheader->SOAP_WSA(ReplyTo)->ReferenceParameters->chan__ChannelInstance);
newheader->chan__ChannelInstance->wsa5__IsReferenceParameter = _wsa5__IsReferenceParameter__true;
}
}
#endif
if (oldheader && oldheader->SOAP_WSA(ReplyTo) && oldheader->SOAP_WSA(ReplyTo)->Address && !soap_tagsearch(soap_wsa_allAnonymousURI, oldheader->SOAP_WSA(ReplyTo)->Address))
{
newheader->SOAP_WSA(To) = oldheader->SOAP_WSA(ReplyTo)->Address;
/* (re)connect to ReplyTo endpoint if From != ReplyTo */
if (!oldheader->SOAP_WSA(From) || !oldheader->SOAP_WSA(From)->Address || strcmp(oldheader->SOAP_WSA(From)->Address, oldheader->SOAP_WSA(ReplyTo)->Address))
{
struct soap *reply_soap = soap_copy(soap);
if (reply_soap)
{
soap_copy_stream(reply_soap, soap);
soap_free_stream(soap); /* prevents close in soap_connect() below */
soap->omode |= SOAP_ENC_PLAIN; /* omit HTTP header ("encode XML body only") */
if (soap_connect(soap, newheader->SOAP_WSA(To), newheader->SOAP_WSA(Action)))
{
int err;
soap_copy_stream(soap, reply_soap);
soap_free_stream(reply_soap);
soap_end(reply_soap);
soap_free(reply_soap);
soap->header = oldheader;
#if defined(SOAP_WSA_2005)
err = soap_wsa_error(soap, SOAP_WSA(DestinationUnreachable), newheader->SOAP_WSA(To));
#elif defined(SOAP_WSA_2003)
err = soap_wsa_error(soap, "WS-Addessing destination unreachable");
#else
err = soap_wsa_error(soap, SOAP_WSA(DestinationUnreachable));
#endif
soap->header = NULL;
return err;
}
if (soap_valid_socket(reply_soap->socket))
soap_send_empty_response(reply_soap, SOAP_OK); /* HTTP ACCEPTED */
soap->header = newheader;
soap->omode &= ~SOAP_ENC_PLAIN; /* HTTP header required */
soap_end(reply_soap);
soap_free(reply_soap);
data->fresponse = soap->fresponse;
soap->fresponse = soap_wsa_response; /* response will be a POST */
}
}
}
else if (oldheader && oldheader->SOAP_WSA(From))
newheader->SOAP_WSA(To) = oldheader->SOAP_WSA(From)->Address;
else
newheader->SOAP_WSA(To) = (char*)soap_wsa_anonymousURI;
soap->header = newheader;
soap->action = newheader->SOAP_WSA(Action);
return SOAP_OK;
}
/******************************************************************************\
*
* Server-side SOAP Fault
*
\******************************************************************************/
/**
@fn int soap_wsa_fault_subcode(struct soap *soap, int flag, const char *faultsubcode, const char *faultstring, const char *faultdetail)
@brief Sets sender/receiver SOAP Fault (sub)code for server fault response.
@param soap context
@param[in] flag 0=receiver, 1=sender
@param[in] faultsubcode sub code string
@param[in] faultstring fault string
@param[in] faultdetail detail string
@return SOAP_FAULT
*/
SOAP_FMAC1
int
SOAP_FMAC2
soap_wsa_fault_subcode(struct soap *soap, int flag, const char *faultsubcode, const char *faultstring, const char *faultdetail)
{
return soap_wsa_fault_subcode_action(soap, flag, faultsubcode, faultstring, faultdetail, NULL);
}
/******************************************************************************/
/**
@fn int soap_wsa_fault_subcode_action(struct soap *soap, int flag, const char *faultsubcode, const char *faultstring, const char *faultdetail, const char *action)
@brief Sets sender/receiver SOAP Fault (sub)code and action for server fault response.
@param soap context
@param[in] flag 0=receiver, 1=sender
@param[in] faultsubcode sub code string
@param[in] faultstring fault string
@param[in] faultdetail detail string
@param[in] action WS-Addressing action string
@return SOAP_FAULT
*/
SOAP_FMAC1
int
SOAP_FMAC2
soap_wsa_fault_subcode_action(struct soap *soap, int flag, const char *faultsubcode, const char *faultstring, const char *faultdetail, const char *action)
{
struct SOAP_ENV__Header *oldheader, *newheader;
DBGFUN2("soap_wsa_fault_subcode_action", "faultsubcode=%s", faultsubcode ? faultsubcode : "(null)", "faultstring=%s", faultstring ? faultstring : "(null)");
oldheader = soap->header;
/* no FaultTo: use ReplyTo */
if (oldheader && oldheader->SOAP_WSA(ReplyTo) && (!oldheader->SOAP_WSA(FaultTo) || soap_tagsearch(soap_wsa_allAnonymousURI, oldheader->SOAP_WSA(FaultTo)->Address)))
{
if (!oldheader->SOAP_WSA(FaultTo))
{
oldheader->SOAP_WSA(FaultTo) = (SOAP_WSA_(,FaultTo)*)soap_malloc(soap, sizeof(SOAP_WSA_(,FaultTo)));
if (oldheader->SOAP_WSA(FaultTo))
SOAP_WSA_(soap_default,EndpointReferenceType)(soap, soap->header->SOAP_WSA(FaultTo));
}
if (oldheader->SOAP_WSA(FaultTo))
oldheader->SOAP_WSA(FaultTo)->Address = oldheader->SOAP_WSA(ReplyTo)->Address;
}
/* use FaultTo */
if (oldheader && oldheader->SOAP_WSA(FaultTo) && !strcmp(oldheader->SOAP_WSA(FaultTo)->Address, soap_wsa_noneURI))
return soap_send_empty_response(soap, SOAP_OK); /* HTTP ACCEPTED */
soap->header = NULL;
/* allocate a new header */
soap_header(soap);
newheader = soap->header;
if (newheader)
{
soap_default_SOAP_ENV__Header(soap, newheader); /* remove/clear SOAP Header */
/* check header */
if (oldheader && oldheader->SOAP_WSA(MessageID))
{
newheader->SOAP_WSA(RelatesTo) = (SOAP_WSA_(,RelatesTo)*)soap_malloc(soap, sizeof(SOAP_WSA_(,RelatesTo)));
SOAP_WSA_(soap_default_,RelatesTo)(soap, newheader->SOAP_WSA(RelatesTo));
newheader->SOAP_WSA(RelatesTo)->__item = oldheader->SOAP_WSA(MessageID);
}
/* header->wsa__MessageID = "..."; */
newheader->SOAP_WSA(Action) = (char*)soap_wsa_faultAction;
if (oldheader && oldheader->SOAP_WSA(FaultTo) && oldheader->SOAP_WSA(FaultTo)->Address && !soap_tagsearch(soap_wsa_allAnonymousURI, oldheader->SOAP_WSA(FaultTo)->Address))
{
newheader->SOAP_WSA(To) = oldheader->SOAP_WSA(FaultTo)->Address;
/* (re)connect to FaultTo endpoint if From != FaultTo */
if (!oldheader->SOAP_WSA(From) || !oldheader->SOAP_WSA(From)->Address || strcmp(oldheader->SOAP_WSA(From)->Address, oldheader->SOAP_WSA(FaultTo)->Address))
{
struct soap_wsa_data *data = (struct soap_wsa_data*)soap_lookup_plugin(soap, soap_wsa_id);
if (!data)
return soap->error = SOAP_PLUGIN_ERROR;
soap->keep_alive = 0;
soap_send_empty_response(soap, SOAP_OK); /* HTTP ACCEPTED */
if (soap_connect(soap, newheader->SOAP_WSA(To), newheader->SOAP_WSA(Action)))
return soap->error = SOAP_STOP; /* nowhere to go */
soap_set_endpoint(soap, newheader->SOAP_WSA(To));
if (action)
soap->action = (char*)action;
else
soap->action = newheader->SOAP_WSA(Action);
data->fresponse = soap->fresponse;
soap->fresponse = soap_wsa_response; /* response will be a POST */
}
}
else if (oldheader && oldheader->SOAP_WSA(From))
newheader->SOAP_WSA(To) = oldheader->SOAP_WSA(From)->Address;
else
newheader->SOAP_WSA(To) = (char*)soap_wsa_anonymousURI;
soap->header = newheader;
}
if (flag)
return soap_sender_fault_subcode(soap, faultsubcode, faultstring, faultdetail);
return soap_receiver_fault_subcode(soap, faultsubcode, faultstring, faultdetail);
}
/******************************************************************************/
/**
@fn int soap_wsa_sender_fault_subcode(struct soap *soap, const char *faultsubcode, const char *faultstring, const char *faultdetail)
@brief Sets sender SOAP Fault (sub)code for server fault response.
@param soap context
@param[in] faultsubcode sub code string
@param[in] faultstring fault string
@param[in] faultdetail detail string
@return SOAP_FAULT
*/
SOAP_FMAC1
int
SOAP_FMAC2
soap_wsa_sender_fault_subcode(struct soap *soap, const char *faultsubcode, const char *faultstring, const char *faultdetail)
{
return soap_wsa_fault_subcode(soap, 1, faultsubcode, faultstring, faultdetail);
}
/******************************************************************************/
/**
@fn int soap_wsa_sender_fault_subcode_action(struct soap *soap, const char *faultsubcode, const char *faultstring, const char *faultdetail, const char *action)
@brief Sets sender SOAP Fault (sub)code for server fault response.
@param soap context
@param[in] faultsubcode sub code string
@param[in] faultstring fault string
@param[in] faultdetail detail string
@param[in] action WS-Addressing action string
@return SOAP_FAULT
*/
SOAP_FMAC1
int
SOAP_FMAC2
soap_wsa_sender_fault_subcode_action(struct soap *soap, const char *faultsubcode, const char *faultstring, const char *faultdetail, const char *action)
{
return soap_wsa_fault_subcode_action(soap, 1, faultsubcode, faultstring, faultdetail, action);
}
/******************************************************************************/
/**
@fn int soap_wsa_receiver_fault_subcode(struct soap *soap, const char *faultsubcode, const char *faultstring, const char *faultdetail)
@brief Sets receiver SOAP Fault (sub)code for server fault response.
@param soap context
@param[in] faultsubcode sub code string
@param[in] faultstring fault string
@param[in] faultdetail detail string
@return SOAP_FAULT
*/
SOAP_FMAC1
int
SOAP_FMAC2
soap_wsa_receiver_fault_subcode(struct soap *soap, const char *faultsubcode, const char *faultstring, const char *faultdetail)
{
return soap_wsa_fault_subcode(soap, 0, faultsubcode, faultstring, faultdetail);
}
/******************************************************************************/
/**
@fn int soap_wsa_receiver_fault_subcode_action(struct soap *soap, const char *faultsubcode, const char *faultstring, const char *faultdetail, const char *action)
@brief Sets receiver SOAP Fault (sub)code for server fault response.
@param soap context
@param[in] faultsubcode sub code string
@param[in] faultstring fault string
@param[in] faultdetail detail string
@param[in] action WS-Addressing action string
@return SOAP_FAULT
*/
SOAP_FMAC1
int
SOAP_FMAC2
soap_wsa_receiver_fault_subcode_action(struct soap *soap, const char *faultsubcode, const char *faultstring, const char *faultdetail, const char *action)
{
return soap_wsa_fault_subcode_action(soap, 0, faultsubcode, faultstring, faultdetail, action);
}
/******************************************************************************/
/**
@fn int soap_wsa_sender_fault(struct soap *soap, const char *faultstring, const char *faultdetail)
@brief Sets sender SOAP Fault for server fault response.
@param soap context
@param[in] faultstring fault string
@param[in] faultdetail detail string
@return SOAP_FAULT
*/
SOAP_FMAC1
int
SOAP_FMAC2
soap_wsa_sender_fault(struct soap *soap, const char *faultstring, const char *faultdetail)
{
return soap_wsa_fault_subcode(soap, 1, NULL, faultstring, faultdetail);
}
/******************************************************************************/
/**
@fn int soap_wsa_receiver_fault(struct soap *soap, const char *faultstring, const char *faultdetail)
@brief Sets receiver SOAP Fault for server fault response.
@param soap context
@param[in] faultstring fault string
@param[in] faultdetail detail string
@return SOAP_FAULT
*/
SOAP_FMAC1
int
SOAP_FMAC2
soap_wsa_receiver_fault(struct soap *soap, const char *faultstring, const char *faultdetail)
{
return soap_wsa_fault_subcode(soap, 0, NULL, faultstring, faultdetail);
}
/******************************************************************************\
*
* WS-Addressing Faults
*
\******************************************************************************/
#if defined(SOAP_WSA_2005)
/**
@fn int soap_wsa_check_fault(struct soap *soap, SOAP_WSA(FaultCodesType) *fault, const char **info)
@brief Checks the presence of a WS-Addressing fault
@param soap context
@param[out] fault code
@param[out] info string pointer related to the wsa fault (or set to NULL)
@return SOAP_OK (no fault) or fault code
*/
SOAP_FMAC1
int
SOAP_FMAC2
soap_wsa_check_fault(struct soap *soap, SOAP_WSA(FaultCodesType) *fault, const char **info)
{
if (soap->error && soap->fault && soap->fault->SOAP_ENV__Code)
{
const char *code = soap_fault_subcode(soap);
if (code)
{
SOAP_WSA__(soap_s2,FaultCodesType)(soap, code, fault);
if (info)
{
struct SOAP_ENV__Detail *detail;
*info = NULL;
if (soap->fault->detail)
detail = soap->fault->detail;
else
detail = soap->fault->SOAP_ENV__Detail;
if (detail)
{
switch (detail->__type)
{
case SOAP_WSA_(SOAP_TYPE_,ProblemHeaderQName):
case SOAP_WSA_(SOAP_TYPE_,ProblemIRI):
*info = (char*)detail->fault;
break;
case SOAP_WSA_(SOAP_TYPE_,ProblemAction):
*info = ((SOAP_WSA_(,ProblemAction)*)detail->fault)->Action;
break;
default:
break;
}
}
}
return soap->error;
}
}
return SOAP_OK;
}
#elif defined(SOAP_WSA_2003)
SOAP_FMAC1
int
SOAP_FMAC2
soap_wsa_check_fault(struct soap *soap, char **fault)
{
struct SOAP_ENV__Detail detail;
*fault = NULL;
if (soap->error && soap->fault)
{
if (soap->fault->detail)
detail = soap->fault->detail;
else
detail = soap->fault->SOAP_ENV__Detail;
}
if (detail)
{
*fault = detail->__any;
if (*fault)
return soap->error;
}
return SOAP_OK;
}
#else
/**
@fn int soap_wsa_check_fault(struct soap *soap, SOAP_WSA(FaultSubcodeValues) *fault)
@brief Checks the presence of a WS-Addressing fault
@param soap context
@param[out] fault code
@return SOAP_OK (no fault) or fault code
*/
SOAP_FMAC1
int
SOAP_FMAC2
soap_wsa_check_fault(struct soap *soap, SOAP_WSA(FaultSubcodeValues) *fault)
{
if (soap->error && soap->fault && soap->fault->SOAP_ENV__Code)
{
const char *code = soap_fault_subcode(soap);
if (code)
{
SOAP_WSA__(soap_s2,FaultSubcodeValues)(soap, code, fault);
return soap->error;
}
}
return SOAP_OK;
}
#endif
#if defined(SOAP_WSA_2005)
/**
@fn int soap_wsa_error(struct soap *soap, SOAP_WSA(FaultCodesType) fault, const char *info)
@brief Sets SOAP Fault (sub)code for server WS-Addressing fault response.
@param soap context
@param[in] fault is one of wsa:FaultCodesType enumeration values
@param[in] info is the value of the element in the Fault detail field
@return SOAP_FAULT
*/
SOAP_FMAC1
int
SOAP_FMAC2
soap_wsa_error(struct soap *soap, SOAP_WSA(FaultCodesType) fault, const char *info)
{
const char *code = SOAP_WSA_(soap,FaultCodesType2s)(soap, fault);
/* populate the SOAP Fault as per WS-Addressing spec */
switch (fault)
{
case SOAP_WSA(InvalidAddressingHeader):
soap_faultdetail(soap);
if (soap->version == 1)
{
soap->fault->detail->__type = SOAP_WSA_(SOAP_TYPE_,ProblemHeaderQName);
soap->fault->detail->fault = (void*)info;
}
else if (soap->version == 2)
{
soap->fault->SOAP_ENV__Detail->__type = SOAP_WSA_(SOAP_TYPE_,ProblemHeaderQName);
soap->fault->SOAP_ENV__Detail->fault = (void*)info;
}
soap_wsa_sender_fault_subcode(soap, code, "A header representing a Message Addressing Property is not valid and the message cannot be processed.", NULL);
break;
case SOAP_WSA(InvalidAddress):
soap_wsa_sender_fault_subcode(soap, code, "Invalid address.", NULL);
break;
case SOAP_WSA(InvalidEPR):
soap_wsa_sender_fault_subcode(soap, code, "Invalid EPR.", NULL);
break;
case SOAP_WSA(InvalidCardinality):
soap_wsa_sender_fault_subcode(soap, code, "Invalid cardinality of headers.", NULL);
break;
case SOAP_WSA(MissingAddressInEPR):
soap_wsa_sender_fault_subcode(soap, code, "Missing EPR address.", NULL);
break;
case SOAP_WSA(DuplicateMessageID):
soap_wsa_sender_fault_subcode(soap, code, "Message contains the message ID of a message already received.", NULL);
break;
case SOAP_WSA(ActionMismatch):
soap_wsa_sender_fault_subcode(soap, code, "Action and SOAP action of the message do not match.", NULL);
break;
case SOAP_WSA(MessageAddressingHeaderRequired):
soap_faultdetail(soap);
if (soap->version == 1)
{
soap->fault->detail->__type = SOAP_WSA_(SOAP_TYPE_,ProblemHeaderQName);
soap->fault->detail->fault = (void*)info;
}
else if (soap->version == 2)
{
soap->fault->SOAP_ENV__Detail->__type = SOAP_WSA_(SOAP_TYPE_,ProblemHeaderQName);
soap->fault->SOAP_ENV__Detail->fault = (void*)info;
}
soap_wsa_sender_fault_subcode(soap, code, "A required header representing a Message Addressing Property is not present.", NULL);
break;
case SOAP_WSA(DestinationUnreachable):
soap_faultdetail(soap);
if (soap->version == 1)
{
soap->fault->detail->__type = SOAP_WSA_(SOAP_TYPE_,ProblemIRI);
soap->fault->detail->fault = (void*)info;
}
else if (soap->version == 2)
{
soap->fault->SOAP_ENV__Detail->__type = SOAP_WSA_(SOAP_TYPE_,ProblemIRI);
soap->fault->SOAP_ENV__Detail->fault = (void*)info;
}
soap_wsa_sender_fault_subcode(soap, code, "No route can be determined to reach [destination]", NULL);
break;
case SOAP_WSA(ActionNotSupported):
soap_faultdetail(soap);
if (soap->version == 1)
{
soap->fault->detail->__type = SOAP_WSA_(SOAP_TYPE_,ProblemAction);
soap->fault->detail->fault = (void*)soap_malloc(soap, sizeof(SOAP_WSA_(,ProblemAction)));
SOAP_WSA_(soap_default_,ProblemAction)(soap, (SOAP_WSA_(,ProblemAction)*)soap->fault->detail->fault);
((SOAP_WSA_(,ProblemAction)*)soap->fault->detail->fault)->Action = (char*)info;
}
else if (soap->version == 2)
{
soap->fault->SOAP_ENV__Detail->__type = SOAP_WSA_(SOAP_TYPE_,ProblemAction);
soap->fault->SOAP_ENV__Detail->fault = (void*)soap_malloc(soap, sizeof(SOAP_WSA_(,ProblemAction)));
SOAP_WSA_(soap_default_,ProblemAction)(soap, (SOAP_WSA_(,ProblemAction)*)soap->fault->SOAP_ENV__Detail->fault);
((SOAP_WSA_(,ProblemAction)*)soap->fault->SOAP_ENV__Detail->fault)->Action = (char*)info;
}
soap_wsa_sender_fault_subcode(soap, code, "The [action] cannot be processed at the receiver.", NULL);
break;
case SOAP_WSA(EndpointUnavailable):
soap_faultdetail(soap);
if (soap->version == 1)
{
soap->fault->detail->__type = SOAP_WSA_(SOAP_TYPE_,ProblemIRI);
soap->fault->detail->fault = (void*)info;
}
else if (soap->version == 2)
{
soap->fault->SOAP_ENV__Detail->__type = SOAP_WSA_(SOAP_TYPE_,ProblemIRI);
soap->fault->SOAP_ENV__Detail->fault = (void*)info;
}
soap_wsa_receiver_fault_subcode(soap, code, "The endpoint is unable to process the message at this time.", NULL);
break;
default:
break;
}
return SOAP_FAULT;
}
#elif defined(SOAP_WSA_2003)
SOAP_FMAC1
int
SOAP_FMAC2
soap_wsa_error(struct soap *soap, const char *fault)
{
return soap_wsa_sender_fault_subcode(soap, NULL, fault, NULL);
}
#else
/**
@fn int soap_wsa_error(struct soap *soap, SOAP_WSA(FaultSubcodeValues) fault)
@brief Sets SOAP Fault (sub)code for server WS-Addressing fault response.
@param soap context
@param[in] fault is one of wsa:FaultSubcodeValues
@return SOAP_FAULT
*/
SOAP_FMAC1
int
SOAP_FMAC2
soap_wsa_error(struct soap *soap, SOAP_WSA(FaultSubcodeValues) fault)
{
const char *code = SOAP_WSA_(soap,FaultSubcodeValues2s)(soap, fault);
/* populate the SOAP Fault as per WS-Addressing spec */
switch (fault)
{
case SOAP_WSA(InvalidMessageInformationHeader):
return soap_wsa_sender_fault_subcode(soap, code, "A message information header is not valid and the message cannot be processed. The validity failure can be either structural or semantic, e.g. a [destination] that is not a URI or a [relationship] to a [message id] that was never issued.", "Invalid header");
case SOAP_WSA(MessageInformationHeaderRequired):
return soap_wsa_sender_fault_subcode(soap, code, "A required message information header, To, MessageID, or Action, is not present.", "Missing Header QName");
case SOAP_WSA(DestinationUnreachable):
return soap_wsa_sender_fault_subcode(soap, code, "No route can be determined to reach the destination role defined by the WS-Addressing To.", NULL);
case SOAP_WSA(ActionNotSupported):
return soap_wsa_sender_fault_subcode(soap, code, "The [action] cannot be processed at the receiver.", soap->action);
case SOAP_WSA(EndpointUnavailable):
return soap_wsa_receiver_fault_subcode(soap, code, "The endpoint is unable to process the message at this time.", NULL);
default:
break;
}
return SOAP_FAULT;
}
#endif
/******************************************************************************\
*
* Plugin registry functions
*
\******************************************************************************/
/**
@fn int soap_wsa(struct soap *soap, struct soap_plugin *p, void *arg)
@brief Plugin registry function, used with soap_register_plugin and soap_register_plugin_arg.
@param soap context
@param[in,out] p plugin created in registry
@param[in] arg passed from soap_register_plugin_arg
@return SOAP_OK
*/
SOAP_FMAC1
int
SOAP_FMAC2
soap_wsa(struct soap *soap, struct soap_plugin *p, void *arg)
{
(void)arg;
DBGFUN("soap_wsa");
p->id = soap_wsa_id;
p->data = (void*)SOAP_MALLOC(soap, sizeof(struct soap_wsa_data));
p->fcopy = NULL;
p->fdelete = soap_wsa_delete;
if (!p->data)
return SOAP_EOM;
if (soap_wsa_init(soap, (struct soap_wsa_data*)p->data))
{
SOAP_FREE(soap, p->data);
return SOAP_EOM;
}
return SOAP_OK;
}
/******************************************************************************/
/**
@fn int soap_wsa_init(struct soap *soap, struct soap_wsa_data *data)
@brief Initializes plugin data.
@param soap context
@param[in,out] data plugin data
@return SOAP_OK
*/
static int
soap_wsa_init(struct soap *soap, struct soap_wsa_data *data)
{
DBGFUN("soap_wsa_init");
data->fheader = soap->fheader;
data->fseterror = soap->fseterror;
soap->fheader = soap_wsa_header;
soap->fseterror = soap_wsa_set_error;
data->fresponse = NULL;
data->fdisconnect = NULL;
return SOAP_OK;
}
/******************************************************************************/
/**
@fn void soap_wsa_delete(struct soap *soap, struct soap_plugin *p)
@brief Deletes plugin data.
@param soap context
@param[in,out] p plugin
@return SOAP_OK
*/
static void
soap_wsa_delete(struct soap *soap, struct soap_plugin *p)
{
(void)soap;
DBGFUN("soap_wsa_delete");
SOAP_FREE(soap, p->data);
}
/******************************************************************************\
*
* Callbacks registered by plugin
*
\******************************************************************************/
/**
@fn int soap_wsa_header(struct soap *soap)
@brief Copies WS-Addressing action to SOAP action
@param soap context
@return SOAP_OK or fault
This callback is invoked to copy the WS-Addressing action to the SOAP action
before invoking the service operation.
*/
static int
soap_wsa_header(struct soap *soap)
{
struct soap_wsa_data *data = (struct soap_wsa_data*)soap_lookup_plugin(soap, soap_wsa_id);
DBGFUN("soap_wsa_header");
if (!data)
return soap->error = SOAP_PLUGIN_ERROR;
if (data->fheader && data->fheader(soap))
return soap->error;
if (soap->header && soap->header->SOAP_WSA(Action))
{
soap->action = soap->header->SOAP_WSA(Action);
DBGLOG(TEST, SOAP_MESSAGE(fdebug, "WSA action='%s'\n", soap->action));
}
return SOAP_OK;
}
/******************************************************************************/
/**
@fn void soap_wsa_set_error(struct soap *soap, const char **c, const char **s)
@brief Copies WS-Addressing action to SOAP action
@param soap context
@param c fault code
@param s fault string
*/
static void
soap_wsa_set_error(struct soap *soap, const char **c, const char **s)
{
struct soap_wsa_data *data = (struct soap_wsa_data*)soap_lookup_plugin(soap, soap_wsa_id);
DBGFUN2("soap_wsa_set_error", "code=%s", c && *c ? *c : "(null)", "string=%s", s && *s ? *s : "(null)");
if (!data)
return;
if (data->fseterror)
data->fseterror(soap, c, s);
if (soap->error == SOAP_NO_METHOD || (soap->error == SOAP_TAG_MISMATCH && soap->level == 2))
{
#if defined(SOAP_WSA_2005)
soap->error = soap_wsa_error(soap, SOAP_WSA(ActionNotSupported), soap->action);
#elif defined(SOAP_WSA_2003)
soap->error = soap_wsa_error(soap, "Action not supported");
#else
soap->error = soap_wsa_error(soap, SOAP_WSA(ActionNotSupported));
#endif
}
}
/******************************************************************************/
/**
@fn int soap_wsa_response(struct soap *soap, int status, ULONG64 count)
@brief Overrides the HTTP response operations to send an HTTP POST
@param soap context
@param status code
@param count message length (if non-chunked)
*/
static int
soap_wsa_response(struct soap *soap, int status, ULONG64 count)
{
struct soap_wsa_data *data = (struct soap_wsa_data*)soap_lookup_plugin(soap, soap_wsa_id);
(void)status;
DBGFUN2("soap_wsa_response", "status=%d", status, "count=%lu", (unsigned long)count);
if (!data)
return SOAP_PLUGIN_ERROR;
soap->fresponse = data->fresponse; /* reset (HTTP response) */
data->fdisconnect = soap->fdisconnect;
soap->fdisconnect = soap_wsa_disconnect; /* to accept HTTP 202 */
return soap->fpost(soap, soap_strdup(soap, soap->endpoint), soap->host, soap->port, soap->path, soap->action, count);
}
/******************************************************************************/
/**
@fn int soap_wsa_disconnect(struct soap *soap)
@brief Accepts HTTP 202 response upon HTTP POST response relay
@param soap context
*/
static int
soap_wsa_disconnect(struct soap *soap)
{
struct soap_wsa_data *data = (struct soap_wsa_data*)soap_lookup_plugin(soap, soap_wsa_id);
DBGFUN("soap_wsa_disconnect");
if (!data)
return SOAP_PLUGIN_ERROR;
soap->fdisconnect = data->fdisconnect; /* reset */
return soap_recv_empty_response(soap);
}
/******************************************************************************\
*
* Misc.
*
\******************************************************************************/
/**
@fn int soap_wsa_alloc_header(struct soap *soap)
@brief Adds SOAP Header if not present.
@param soap context
@return SOAP_OK
*/
static int
soap_wsa_alloc_header(struct soap *soap)
{
if (soap->header)
return SOAP_OK;
soap_header(soap);
if (soap->header)
return SOAP_OK;
return soap->error = SOAP_EOM;
}
/******************************************************************************/
#ifdef __cplusplus
}
#endif
此处可能存在不合适展示的内容,页面不予展示。您可通过相关编辑功能自查并修改。
如您确认内容无涉及 不当用语 / 纯广告导流 / 暴力 / 低俗色情 / 侵权 / 盗版 / 虚假 / 无价值内容或违法国家有关法律法规的内容,可点击提交进行申诉,我们将尽快为您处理。