You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
438 lines
20 KiB
438 lines
20 KiB
/*
|
|
* ============================================================================
|
|
* The Apache Software License, Version 1.1
|
|
* ============================================================================
|
|
*
|
|
* Copyright (C) 2002 The Apache Software Foundation. All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without modifica-
|
|
* tion, are permitted provided that the following conditions are met:
|
|
*
|
|
* 1. Redistributions of source code must retain the above copyright notice,
|
|
* this list of conditions and the following disclaimer.
|
|
*
|
|
* 2. Redistributions in binary form must reproduce the above copyright notice,
|
|
* this list of conditions and the following disclaimer in the documentation
|
|
* and/or other materials provided with the distribution.
|
|
*
|
|
* 3. The end-user documentation included with the redistribution, if any, must
|
|
* include the following acknowledgment: "This product includes software
|
|
* developed by SuperBonBon Industries (http://www.sbbi.net/)."
|
|
* Alternately, this acknowledgment may appear in the software itself, if
|
|
* and wherever such third-party acknowledgments normally appear.
|
|
*
|
|
* 4. The names "UPNPLib" and "SuperBonBon Industries" must not be
|
|
* used to endorse or promote products derived from this software without
|
|
* prior written permission. For written permission, please contact
|
|
* info@sbbi.net.
|
|
*
|
|
* 5. Products derived from this software may not be called
|
|
* "SuperBonBon Industries", nor may "SBBI" appear in their name,
|
|
* without prior written permission of SuperBonBon Industries.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED WARRANTIES,
|
|
* INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
|
|
* FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
|
|
* APACHE SOFTWARE FOUNDATION OR ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
|
|
* INDIRECT,INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLU-
|
|
* DING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
|
|
* OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
|
|
* ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
|
|
* THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*
|
|
* This software consists of voluntary contributions made by many individuals
|
|
* on behalf of SuperBonBon Industries. For more information on
|
|
* SuperBonBon Industries, please see <http://www.sbbi.net/>.
|
|
*/
|
|
package net.sbbi.upnp.messages;
|
|
|
|
import java.util.*;
|
|
import java.io.*;
|
|
import java.net.HttpURLConnection;
|
|
import java.net.URL;
|
|
|
|
import org.apache.commons.logging.Log;
|
|
import org.apache.commons.logging.LogFactory;
|
|
import org.xml.sax.*;
|
|
import net.sbbi.upnp.services.*;
|
|
|
|
import javax.xml.parsers.*;
|
|
|
|
/**
|
|
* Message object for an UPNP action, simply call setInputParameter() to add
|
|
* the required action message params and then service() to receive the ActionResponse
|
|
* built with the parsed UPNP device SOAP xml response.
|
|
* @author <a href="mailto:superbonbon@sbbi.net">SuperBonBon</a>
|
|
* @version 1.0
|
|
*/
|
|
public class ActionMessage {
|
|
|
|
private final static Log log = LogFactory.getLog( ActionMessage.class );
|
|
|
|
private UPNPService service;
|
|
private ServiceAction serviceAction;
|
|
private List inputParameters;
|
|
|
|
/**
|
|
* Protected constuctor so that only messages factories can build it
|
|
* @param service the service for which the
|
|
* @param serviceAction
|
|
*/
|
|
protected ActionMessage( UPNPService service, ServiceAction serviceAction ) {
|
|
this.service = service;
|
|
this.serviceAction = serviceAction;
|
|
if ( serviceAction.getInputActionArguments() != null ) {
|
|
inputParameters = new ArrayList();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Method to clear all set input parameters so that
|
|
* this object can be reused
|
|
*/
|
|
public void clearInputParameters() {
|
|
inputParameters.clear();
|
|
}
|
|
|
|
/**
|
|
* Executes the message and retuns the UPNP device response, according to the UPNP specs,
|
|
* this method could take up to 30 secs to process ( time allowed for a device to respond to a request )
|
|
* @return a response object containing the UPNP parsed response
|
|
* @throws IOException if some IOException occurs during message send and reception process
|
|
* @throws UPNPResponseException if an UPNP error message is returned from the server
|
|
* or if some parsing exception occurs ( detailErrorCode = 899, detailErrorDescription = SAXException message )
|
|
*/
|
|
public ActionResponse service() throws IOException, UPNPResponseException {
|
|
ActionResponse rtrVal = null;
|
|
UPNPResponseException upnpEx = null;
|
|
IOException ioEx = null;
|
|
StringBuffer body = new StringBuffer( 256 );
|
|
|
|
body.append( "<?xml version=\"1.0\"?>\r\n" );
|
|
body.append( "<s:Envelope xmlns:s=\"http://schemas.xmlsoap.org/soap/envelope/\"" );
|
|
body.append( " s:encodingStyle=\"http://schemas.xmlsoap.org/soap/encoding/\">" );
|
|
body.append( "<s:Body>" );
|
|
body.append( "<u:" ).append( serviceAction.getName() ).append( " xmlns:u=\"" ).append( service.getServiceType() ).append( "\">" );
|
|
|
|
if ( serviceAction.getInputActionArguments() != null ) {
|
|
// this action requires params so we just set them...
|
|
for ( Iterator itr = inputParameters.iterator(); itr.hasNext(); ) {
|
|
InputParamContainer container = (InputParamContainer)itr.next();
|
|
body.append( "<" ).append( container.name ).append( ">" ).append( container.value );
|
|
body.append( "</" ).append( container.name ).append( ">" );
|
|
}
|
|
}
|
|
body.append( "</u:" ).append( serviceAction.getName() ).append( ">" );
|
|
body.append( "</s:Body>" );
|
|
body.append( "</s:Envelope>" );
|
|
|
|
if ( log.isDebugEnabled() ) log.debug( "POST prepared for URL " + service.getControlURL() );
|
|
URL url = new URL( service.getControlURL().toString() );
|
|
HttpURLConnection conn = (HttpURLConnection)url.openConnection();
|
|
conn.setDoInput( true );
|
|
conn.setDoOutput( true );
|
|
conn.setUseCaches( false );
|
|
conn.setRequestMethod( "POST" );
|
|
HttpURLConnection.setFollowRedirects( false );
|
|
//conn.setConnectTimeout( 30000 );
|
|
conn.setRequestProperty( "HOST", url.getHost() + ":" + url.getPort() );
|
|
conn.setRequestProperty( "CONTENT-TYPE", "text/xml; charset=\"utf-8\"" );
|
|
conn.setRequestProperty( "CONTENT-LENGTH", Integer.toString( body.length() ) );
|
|
conn.setRequestProperty( "SOAPACTION", "\"" + service.getServiceType() + "#" + serviceAction.getName() + "\"" );
|
|
OutputStream out = conn.getOutputStream();
|
|
out.write( body.toString().getBytes() );
|
|
out.flush();
|
|
out.close();
|
|
conn.connect();
|
|
InputStream input = null;
|
|
|
|
if ( log.isDebugEnabled() ) log.debug( "executing query :\n" + body );
|
|
try {
|
|
input = conn.getInputStream();
|
|
} catch ( IOException ex ) {
|
|
// java can throw an exception if he error code is 500 or 404 or something else than 200
|
|
// but the device sends 500 error message with content that is required
|
|
// this content is accessible with the getErrorStream
|
|
input = conn.getErrorStream();
|
|
}
|
|
|
|
if ( input != null ) {
|
|
int response = conn.getResponseCode();
|
|
String responseBody = getResponseBody( input );
|
|
if ( log.isDebugEnabled() ) log.debug( "received response :\n" + responseBody );
|
|
SAXParserFactory saxParFact = SAXParserFactory.newInstance();
|
|
saxParFact.setValidating( false );
|
|
saxParFact.setNamespaceAware( true );
|
|
ActionMessageResponseParser msgParser = new ActionMessageResponseParser( serviceAction );
|
|
StringReader stringReader = new StringReader( responseBody );
|
|
InputSource src = new InputSource( stringReader );
|
|
try {
|
|
SAXParser parser = saxParFact.newSAXParser();
|
|
parser.parse( src, msgParser );
|
|
} catch ( ParserConfigurationException confEx ) {
|
|
// should never happen
|
|
// we throw a runtimeException to notify the env problem
|
|
throw new RuntimeException( "ParserConfigurationException during SAX parser creation, please check your env settings:" + confEx.getMessage() );
|
|
} catch ( SAXException saxEx ) {
|
|
// kind of tricky but better than nothing..
|
|
upnpEx = new UPNPResponseException( 899, saxEx.getMessage() );
|
|
} finally {
|
|
try {
|
|
input.close();
|
|
} catch ( IOException ex ) {
|
|
// ignore
|
|
}
|
|
}
|
|
if ( upnpEx == null ) {
|
|
if ( response == HttpURLConnection.HTTP_OK ) {
|
|
rtrVal = msgParser.getActionResponse();
|
|
} else if ( response == HttpURLConnection.HTTP_INTERNAL_ERROR ) {
|
|
upnpEx = msgParser.getUPNPResponseException();
|
|
} else {
|
|
ioEx = new IOException( "Unexpected server HTTP response:" + response );
|
|
}
|
|
}
|
|
}
|
|
try {
|
|
out.close();
|
|
} catch ( IOException ex ) {
|
|
// ignore
|
|
}
|
|
conn.disconnect();
|
|
if ( upnpEx != null ) {
|
|
throw upnpEx;
|
|
}
|
|
if ( rtrVal == null && ioEx == null ) {
|
|
ioEx = new IOException( "Unable to receive a response from the UPNP device" );
|
|
}
|
|
if ( ioEx != null ) {
|
|
throw ioEx;
|
|
}
|
|
return rtrVal;
|
|
}
|
|
|
|
private String getResponseBody( InputStream in ) throws IOException {
|
|
byte[] buffer = new byte[256];
|
|
int readen = 0;
|
|
StringBuffer content = new StringBuffer( 256 );
|
|
while ( ( readen = in.read( buffer ) ) != -1 ) {
|
|
content.append( new String( buffer, 0 , readen ) );
|
|
}
|
|
// some devices add \0 chars at XML message end
|
|
// which causes XML parsing errors...
|
|
int len = content.length();
|
|
while ( content.charAt( len-1 ) == '\0' ) {
|
|
len--;
|
|
content.setLength( len );
|
|
}
|
|
return content.toString().trim();
|
|
}
|
|
|
|
/**
|
|
* The list of input parameters that should be accepted by the device service for this message
|
|
* @return a list of required input parameters ServiceActionArgument objects for this message
|
|
* or null if the message does not require any input params
|
|
*/
|
|
public List getInputParameterNames() {
|
|
return serviceAction.getInputActionArgumentsNames();
|
|
}
|
|
|
|
/**
|
|
* The list of output parameters that should be returned by the device service
|
|
* @return a list of output parameters ServiceActionArgument objects for this message
|
|
* or null if the message does not contains any output params.
|
|
*/
|
|
public List getOutputParameterNames() {
|
|
return serviceAction.getOutputActionArgumentsNames();
|
|
}
|
|
|
|
/**
|
|
* Set the value of an input parameter before a message service call. If the param name already
|
|
* exists, the param value will be overwritten with the new value provided.
|
|
* @param parameterName the parameter name
|
|
* @param parameterValue the parameter value as an object, primitive object are handled, all other object
|
|
* will be assigned with a call to their toString() method call
|
|
* @return the current ActionMessage object instance
|
|
* @throws IllegalArgumentException if the provided parameterName is not valid for this message
|
|
* or if no input parameters are required for this message
|
|
*/
|
|
public ActionMessage setInputParameter( String parameterName, Object parameterValue ) throws IllegalArgumentException {
|
|
if ( parameterValue == null ) {
|
|
return setInputParameter( parameterName, "" );
|
|
} else if ( parameterValue instanceof Date ) {
|
|
return setInputParameter( parameterName, (Date)parameterValue );
|
|
} else if ( parameterValue instanceof Boolean ) {
|
|
return setInputParameter( parameterName, ((Boolean)parameterValue).booleanValue() );
|
|
} else if ( parameterValue instanceof Integer ) {
|
|
return setInputParameter( parameterName, ((Integer)parameterValue).intValue() );
|
|
} else if ( parameterValue instanceof Byte ) {
|
|
return setInputParameter( parameterName, ((Byte)parameterValue).byteValue() );
|
|
} else if ( parameterValue instanceof Short ) {
|
|
return setInputParameter( parameterName, ((Short)parameterValue).shortValue() );
|
|
} else if ( parameterValue instanceof Float ) {
|
|
return setInputParameter( parameterName, ((Float)parameterValue).floatValue() );
|
|
} else if ( parameterValue instanceof Double ) {
|
|
return setInputParameter( parameterName, ((Double)parameterValue).doubleValue() );
|
|
} else if ( parameterValue instanceof Long ) {
|
|
return setInputParameter( parameterName, ((Long)parameterValue).longValue() );
|
|
}
|
|
return setInputParameter( parameterName, parameterValue.toString() );
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
* Set the value of an input parameter before a message service call. If the param name already
|
|
* exists, the param value will be overwritten with the new value provided
|
|
* @param parameterName the parameter name
|
|
* @param parameterValue the string parameter value
|
|
* @return the current ActionMessage object instance
|
|
* @throws IllegalArgumentException if the provided parameterName is not valid for this message
|
|
* or if no input parameters are required for this message
|
|
*/
|
|
public ActionMessage setInputParameter( String parameterName, String parameterValue ) throws IllegalArgumentException {
|
|
if ( serviceAction.getInputActionArguments() == null ) throw new IllegalArgumentException( "No input parameters required for this message" );
|
|
ServiceActionArgument arg = serviceAction.getInputActionArgument( parameterName );
|
|
if ( arg == null ) throw new IllegalArgumentException( "Wrong input argument name for this action:" + parameterName + " available parameters are : " + getInputParameterNames() );
|
|
for ( Iterator i = inputParameters.iterator(); i.hasNext(); ) {
|
|
InputParamContainer container = (InputParamContainer)i.next();
|
|
if ( container.name.equals( parameterName ) ) {
|
|
container.value = parameterValue;
|
|
return this;
|
|
}
|
|
}
|
|
// nothing found add the new value
|
|
InputParamContainer container = new InputParamContainer();
|
|
container.name = parameterName;
|
|
container.value = parameterValue;
|
|
inputParameters.add( container );
|
|
return this;
|
|
}
|
|
|
|
/**
|
|
* Set the value of an input parameter before a message service call
|
|
* @param parameterName the parameter name
|
|
* @param parameterValue the date parameter value, will be automatically translated to the correct
|
|
* ISO 8601 date format for the given action input param related state variable
|
|
* @return the current ActionMessage object instance
|
|
* @throws IllegalArgumentException if the provided parameterName is not valid for this message
|
|
* or if no input parameters are required for this message
|
|
*/
|
|
public ActionMessage setInputParameter( String parameterName, Date parameterValue ) throws IllegalArgumentException {
|
|
if ( serviceAction.getInputActionArguments() == null ) throw new IllegalArgumentException( "No input parameters required for this message" );
|
|
ServiceActionArgument arg = serviceAction.getInputActionArgument( parameterName );
|
|
if ( arg == null ) throw new IllegalArgumentException( "Wrong input argument name for this action:" + parameterName + " available parameters are : " + getInputParameterNames() );
|
|
ServiceStateVariable linkedVar = arg.getRelatedStateVariable();
|
|
if ( linkedVar.getDataType().equals( ServiceStateVariable.TIME ) ) {
|
|
return setInputParameter( parameterName, ISO8601Date.getIsoTime( parameterValue ) );
|
|
} else if ( linkedVar.getDataType().equals( ServiceStateVariable.TIME_TZ ) ) {
|
|
return setInputParameter( parameterName, ISO8601Date.getIsoTimeZone( parameterValue ) );
|
|
} else if ( linkedVar.getDataType().equals( ServiceStateVariable.DATE ) ) {
|
|
return setInputParameter( parameterName, ISO8601Date.getIsoDate( parameterValue ) );
|
|
} else if ( linkedVar.getDataType().equals( ServiceStateVariable.DATETIME ) ) {
|
|
return setInputParameter( parameterName, ISO8601Date.getIsoDateTime( parameterValue ) );
|
|
} else if ( linkedVar.getDataType().equals( ServiceStateVariable.DATETIME_TZ ) ) {
|
|
return setInputParameter( parameterName, ISO8601Date.getIsoDateTimeZone( parameterValue ) );
|
|
} else {
|
|
throw new IllegalArgumentException( "Related input state variable " + linkedVar.getName() + " is not of an date type" );
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* Set the value of an input parameter before a message service call
|
|
* @param parameterName the parameter name
|
|
* @param parameterValue the boolean parameter value
|
|
* @return the current ActionMessage object instance
|
|
* @throws IllegalArgumentException if the provided parameterName is not valid for this message
|
|
* or if no input parameters are required for this message
|
|
*/
|
|
public ActionMessage setInputParameter( String parameterName, boolean parameterValue ) throws IllegalArgumentException {
|
|
return setInputParameter( parameterName, parameterValue ? "1" : "0" );
|
|
}
|
|
|
|
/**
|
|
* Set the value of an input parameter before a message service call
|
|
* @param parameterName the parameter name
|
|
* @param parameterValue the byte parameter value
|
|
* @return the current ActionMessage object instance
|
|
* @throws IllegalArgumentException if the provided parameterName is not valid for this message
|
|
* or if no input parameters are required for this message
|
|
*/
|
|
public ActionMessage setInputParameter( String parameterName, byte parameterValue ) throws IllegalArgumentException {
|
|
return setInputParameter( parameterName, Byte.toString( parameterValue ) );
|
|
}
|
|
|
|
/**
|
|
* Set the value of an input parameter before a message service call
|
|
* @param parameterName the parameter name
|
|
* @param parameterValue the short parameter value
|
|
* @return the current ActionMessage object instance
|
|
* @throws IllegalArgumentException if the provided parameterName is not valid for this message
|
|
* or if no input parameters are required for this message
|
|
*/
|
|
public ActionMessage setInputParameter( String parameterName, short parameterValue ) throws IllegalArgumentException {
|
|
return setInputParameter( parameterName, Short.toString( parameterValue ) );
|
|
}
|
|
|
|
/**
|
|
* Set the value of an input parameter before a message service call
|
|
* @param parameterName the parameter name
|
|
* @param parameterValue the integer parameter value
|
|
* @return the current ActionMessage object instance
|
|
* @throws IllegalArgumentException if the provided parameterName is not valid for this message
|
|
* or if no input parameters are required for this message
|
|
*/
|
|
public ActionMessage setInputParameter( String parameterName, int parameterValue ) throws IllegalArgumentException {
|
|
return setInputParameter( parameterName, Integer.toString( parameterValue ) );
|
|
}
|
|
|
|
/**
|
|
* Set the value of an input parameter before a message service call
|
|
* @param parameterName the parameter name
|
|
* @param parameterValue the long parameter value
|
|
* @return the current ActionMessage object instance
|
|
* @throws IllegalArgumentException if the provided parameterName is not valid for this message
|
|
* or if no input parameters are required for this message
|
|
*/
|
|
public ActionMessage setInputParameter( String parameterName, long parameterValue ) throws IllegalArgumentException {
|
|
return setInputParameter( parameterName, Long.toString( parameterValue ) );
|
|
}
|
|
|
|
/**
|
|
* Set the value of an input parameter before a message service call
|
|
* @param parameterName the parameter name
|
|
* @param parameterValue the float parameter value
|
|
* @return the current ActionMessage object instance
|
|
* @throws IllegalArgumentException if the provided parameterName is not valid for this message
|
|
* or if no input parameters are required for this message
|
|
*/
|
|
public ActionMessage setInputParameter( String parameterName, float parameterValue ) throws IllegalArgumentException {
|
|
return setInputParameter( parameterName, Float.toString( parameterValue ) );
|
|
}
|
|
|
|
/**
|
|
* Set the value of an input parameter before a message service call
|
|
* @param parameterName the parameter name
|
|
* @param parameterValue the double parameter value
|
|
* @return the current ActionMessage object instance
|
|
* @throws IllegalArgumentException if the provided parameterName is not valid for this message
|
|
* or if no input parameters are required for this message
|
|
*/
|
|
public ActionMessage setInputParameter( String parameterName, double parameterValue ) throws IllegalArgumentException {
|
|
return setInputParameter( parameterName, Double.toString( parameterValue ) );
|
|
}
|
|
|
|
/**
|
|
* Input params class container
|
|
*/
|
|
private class InputParamContainer {
|
|
|
|
private String name;
|
|
private String value;
|
|
|
|
}
|
|
|
|
}
|