package org.apache.xmlrpc.applet;
/*
* The Apache Software License, Version 1.1
*
*
* Copyright (c) 2001 The Apache Software Foundation. All rights
* reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, 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 the
* Apache Software Foundation (http://www.apache.org/)."
* Alternately, this acknowledgment may appear in the software itself,
* if and wherever such third-party acknowledgments normally appear.
*
* 4. The names "XML-RPC" and "Apache Software Foundation" must
* not be used to endorse or promote products derived from this
* software without prior written permission. For written
* permission, please contact apache@apache.org.
*
* 5. Products derived from this software may not be called "Apache",
* nor may "Apache" appear in their name, without prior written
* permission of the Apache Software Foundation.
*
* 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 (INCLUDING, 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 the Apache Software Foundation. For more
* information on the Apache Software Foundation, please see
* <http://www.apache.org/>.
*/
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.UnsupportedEncodingException;
import java.net.MalformedURLException;
import java.net.URL;
import java.net.URLConnection;
import java.text.DateFormat;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.Enumeration;
import java.util.Hashtable;
import java.util.Stack;
import java.util.Vector;
import org.apache.xmlrpc.Base64;
import org.apache.xmlrpc.XmlRpc;
import org.xml.sax.AttributeList;
import org.xml.sax.HandlerBase;
import org.xml.sax.InputSource;
import org.xml.sax.SAXException;
import org.xml.sax.SAXParseException;
import uk.co.wilson.xml.MinML;
/**
* A simple XML-RPC client.
*
* FIXME: This code is VERY out of date with the rest of the package.
*
* @version $Id: SimpleXmlRpcClient.java,v 1.5.2.3 2003/02/08 05:09:28 rhoegg Exp $
*/
public class SimpleXmlRpcClient
{
URL url;
/**
* Construct a XML-RPC client with this URL.
*/
public SimpleXmlRpcClient(URL url)
{
this.url = url;
}
/**
* Construct a XML-RPC client for the URL represented by this String.
*/
public SimpleXmlRpcClient(String url) throws MalformedURLException
{
this.url = new URL(url);
}
/**
* Construct a XML-RPC client for the specified hostname and port.
*/
public SimpleXmlRpcClient(String hostname, int port)
throws MalformedURLException
{
this.url = new URL("http://" + hostname + ":" + port + "/RPC2");
}
/**
*
* @param method
* @param params
* @return
* @throws XmlRpcException
* @throws IOException
*/
public Object execute(String method, Vector params)
throws XmlRpcException, IOException
{
return new XmlRpcSupport (url).execute (method, params);
}
}
/**
* FIXME: Leverage the XmlRpc class.
*/
class XmlRpcSupport extends HandlerBase
{
URL url;
String methodName;
boolean fault = false;
Object result = null;
// the stack we're parsing our values into.
Stack values;
Value currentValue;
boolean readCdata;
// formats for parsing and generating dateTime values
static final DateFormat format = new SimpleDateFormat("yyyyMMdd'T'HH:mm:ss");
// used to collect character data of parameter values
StringBuffer cdata = new StringBuffer ();
// XML RPC parameter types used for dataMode
static final int STRING = 0;
static final int INTEGER = 1;
static final int BOOLEAN = 2;
static final int DOUBLE = 3;
static final int DATE = 4;
static final int BASE64 = 5;
static final int STRUCT = 6;
static final int ARRAY = 7;
// for debugging output
public static boolean debug = false;
final static String types[] = {"String", "Integer", "Boolean", "Double",
"Date", "Base64", "Struct", "Array"};
/**
*
* @param url
*/
public XmlRpcSupport(URL url)
{
this.url = url;
}
/**
* Switch debugging output on/off.
*/
public static void setDebug(boolean val)
{
debug = val;
}
/**
* Parse the input stream. For each root level object, method
* <code>objectParsed</code> is called.
*/
synchronized void parse(InputStream is) throws Exception
{
values = new Stack();
long now = System.currentTimeMillis();
MinML parser = new MinML();
parser.setDocumentHandler(this);
parser.setErrorHandler(this);
parser.parse(new InputSource(is));
if (debug)
{
System.out.println("Spent " + (System.currentTimeMillis() - now)
+ " parsing");
}
}
/**
* Writes the XML representation of a supported Java object to the XML writer.
*/
void writeObject (Object what, XmlWriter writer) throws IOException
{
writer.startElement("value");
if (what instanceof String)
{
writer.write(what.toString());
}
else if (what instanceof Integer)
{
writer.startElement("int");
writer.write (what.toString());
writer.endElement("int");
}
else if (what instanceof Boolean)
{
writer.startElement("boolean");
writer.write(((Boolean) what).booleanValue() ? "1" : "0");
writer.endElement("boolean");
}
else if (what instanceof Double)
{
writer.startElement("double");
writer.write (what.toString());
writer.endElement("double");
}
else if (what instanceof Date)
{
writer.startElement("dateTime.iso8601");
Date d = (Date) what;
writer.write(format.format(d));
writer.endElement("dateTime.iso8601");
}
else if (what instanceof byte[])
{
writer.startElement("base64");
writer.write(Base64.encode((byte[]) what));
writer.endElement("base64");
}
else if (what instanceof Vector)
{
writer.startElement("array");
writer.startElement("data");
Vector v = (Vector) what;
int l2 = v.size();
for (int i2 = 0; i2 < l2; i2++)
{
writeObject(v.elementAt(i2), writer);
}
writer.endElement("data");
writer.endElement("array");
}
else if (what instanceof Hashtable)
{
writer.startElement("struct");
Hashtable h = (Hashtable) what;
for (Enumeration e = h.keys (); e.hasMoreElements (); )
{
String nextkey = (String) e.nextElement ();
Object nextval = h.get(nextkey);
writer.startElement("member");
writer.startElement("name");
writer.write(nextkey);
writer.endElement("name");
writeObject(nextval, writer);
writer.endElement("member");
}
writer.endElement("struct");
}
else
{
String unsupportedType = what == null ? "null"
: what.getClass().toString();
throw new IOException("unsupported Java type: " + unsupportedType);
}
writer.endElement("value");
}
/**
* Generate an XML-RPC request and send it to the server. Parse the result
* and return the corresponding Java object.
*
* @exception XmlRpcException If the remote host returned a fault message.
* @exception IOException If the call could not be made for lower level
* problems.
*/
public Object execute(String method, Vector arguments)
throws XmlRpcException, IOException
{
fault = false;
long now = System.currentTimeMillis();
try
{
StringBuffer strbuf = new StringBuffer();
XmlWriter writer = new XmlWriter(strbuf);
writeRequest(writer, method, arguments);
byte[] request = strbuf.toString().getBytes();
URLConnection con = url.openConnection();
con.setDoOutput(true);
con.setDoInput(true);
con.setUseCaches(false);
con.setAllowUserInteraction(false);
con.setRequestProperty("Content-Length",
Integer.toString(request.length));
con.setRequestProperty("Content-Type", "text/xml");
// con.connect ();
OutputStream out = con.getOutputStream();
out.write(request);
out.flush();
InputStream in = con.getInputStream();
parse(in);
System.out.println("result = " + result);
}
catch (Exception x)
{
x.printStackTrace();
throw new IOException(x.getMessage());
}
if (fault)
{
// generate an XmlRpcException
XmlRpcException exception = null;
try
{
Hashtable f = (Hashtable) result;
String faultString = (String) f.get("faultString");
int faultCode = Integer.parseInt(f.get("faultCode").toString());
exception = new XmlRpcException(faultCode, faultString.trim());
}
catch (Exception x)
{
throw new XmlRpcException(0, "Invalid fault response");
}
throw exception;
}
System.out.println("Spent " + (System.currentTimeMillis() - now)
+ " in request");
return result;
}
/**
* Called when the return value has been parsed.
*/
void objectParsed(Object what)
{
result = what;
}
/**
* Generate an XML-RPC request from a method name and a parameter vector.
*/
void writeRequest (XmlWriter writer, String method, Vector params)
throws IOException
{
writer.startElement("methodCall");
writer.startElement("methodName");
writer.write(method);
writer.endElement("methodName");
writer.startElement("params");
int l = params.size();
for (int i = 0; i < l; i++)
{
writer.startElement("param");
writeObject(params.elementAt (i), writer);
writer.endElement("param");
}
writer.endElement("params");
writer.endElement("methodCall");
}
////////////////////////////////////////////////////////////////
// methods called by XML parser
/**
* Method called by SAX driver.
*/
public void characters(char ch[], int start, int length)
throws SAXException
{
if (! readCdata)
{
return;
}
cdata.append (ch, start, length);
}
/**
* Method called by SAX driver.
*/
public void endElement(String name) throws SAXException
{
if (debug)
{
System.err.println("endElement: " + name);
}
// finalize character data, if appropriate
if (currentValue != null && readCdata)
{
currentValue.characterData(cdata.toString());
cdata.setLength(0);
readCdata = false;
}
if ("value".equals(name))
{
int depth = values.size();
// Only handle top level objects or objects contained in arrays here.
// For objects contained in structs, wait for </member> (see code below).
if (depth < 2 || values.elementAt (depth - 2).hashCode () != STRUCT)
{
Value v = currentValue;
values.pop();
if (depth < 2)
{
// This is a top-level object
objectParsed(v.value);
currentValue = null;
}
else
{
// add object to sub-array; if current container is a struct, add later (at </member>)
currentValue = (Value) values.peek();
currentValue.endElement(v);
}
}
}
// Handle objects contained in structs.
if ("member".equals(name))
{
Value v = currentValue;
values.pop();
currentValue = (Value) values.peek();
currentValue.endElement(v);
}
else if ("methodName".equals(name))
{
methodName = cdata.toString();
cdata.setLength(0);
readCdata = false;
}
}
/**
* Method called by SAX driver.
*/
public void startElement (String name, AttributeList atts)
throws SAXException
{
if (debug)
{
System.err.println("startElement: " + name);
}
if( "fault".equals(name))
{
fault = true;
}
else if ("value".equals(name))
{
// System.err.println ("starting value");
Value v = new Value();
values.push(v);
currentValue = v;
// cdata object is reused
cdata.setLength(0);
readCdata = true;
}
else if ("methodName".equals(name))
{
cdata.setLength(0);
readCdata = true;
}
else if ("name".equals(name))
{
cdata.setLength(0);
readCdata = true;
}
else if ("string".equals(name))
{
// currentValue.setType (STRING);
cdata.setLength(0);
readCdata = true;
}
else if ("i4".equals(name) || "int".equals(name))
{
currentValue.setType(INTEGER);
cdata.setLength(0);
readCdata = true;
}
else if ("boolean".equals(name))
{
currentValue.setType(BOOLEAN);
cdata.setLength(0);
readCdata = true;
}
else if ("double".equals(name))
{
currentValue.setType(DOUBLE);
cdata.setLength(0);
readCdata = true;
}
else if ("dateTime.iso8601".equals(name))
{
currentValue.setType(DATE);
cdata.setLength(0);
readCdata = true;
}
else if ("base64".equals(name))
{
currentValue.setType(BASE64);
cdata.setLength(0);
readCdata = true;
}
else if ("struct".equals(name))
{
currentValue.setType(STRUCT);
}
else if ("array".equals(name))
{
currentValue.setType(ARRAY);
}
}
/**
*
* @param e
* @throws SAXException
*/
public void error(SAXParseException e) throws SAXException
{
System.err.println("Error parsing XML: " + e);
// errorLevel = RECOVERABLE;
// errorMsg = e.toString ();
}
/**
*
* @param e
* @throws SAXException
*/
public void fatalError(SAXParseException e) throws SAXException
{
System.err.println("Fatal error parsing XML: " + e);
// errorLevel = FATAL;
// errorMsg = e.toString ();
}
/**
* This represents an XML-RPC Value while the request is being parsed.
*/
class Value
{
int type;
Object value;
// the name to use for the next member of struct values
String nextMemberName;
Hashtable struct;
Vector array;
/**
* Constructor.
*/
public Value()
{
this.type = STRING;
}
/**
* Notification that a new child element has been parsed.
*/
public void endElement(Value child)
{
if (type == ARRAY)
{
array.addElement(child.value);
}
else if (type == STRUCT)
{
struct.put(nextMemberName, child.value);
}
}
/**
* Set the type of this value. If it's a container, create the
* corresponding java container.
*/
public void setType(int type)
{
// System.err.println ("setting type to "+types[type]);
this.type = type;
if (type == ARRAY)
{
value = new Vector();
array = new Vector();
}
if (type == STRUCT)
{
value = new Hashtable();
struct = new Hashtable();
}
}
/**
* Set the character data for the element and interpret it according to
* the element type
*/
public void characterData (String cdata)
{
switch (type)
{
case INTEGER:
value = new Integer(cdata.trim());
break;
case BOOLEAN:
value = new Boolean("1".equals(cdata.trim()));
break;
case DOUBLE:
value = new Double(cdata.trim());
break;
case DATE:
try
{
value = format.parse(cdata.trim());
}
catch (ParseException p)
{
// System.err.println ("Exception while parsing date: "+p);
throw new RuntimeException(p.getMessage());
}
break;
case BASE64:
value = Base64.decode(cdata.getBytes());
break;
case STRING:
value = cdata;
break;
case STRUCT:
// this is the name to use for the next member of this struct
nextMemberName = cdata;
break;
}
}
/**
* This is a performance hack to get the type of a value without casting
* the Object. It breaks the contract of method hashCode, but it doesn't
* matter since Value objects are never used as keys in Hashtables.
*/
public int hashCode ()
{
return type;
}
/**
*
* @return
*/
public String toString ()
{
return (types[type] + " element " + value);
}
}
/**
* A quick and dirty XML writer.
* TODO: Replace with core package's XmlWriter class.
*/
class XmlWriter
{
StringBuffer buf;
String enc;
/**
*
* @param buf
*/
public XmlWriter(StringBuffer buf)
{
this.buf = buf;
buf.append("<?xml version=\"1.0\" encoding=\"ISO-8859-1\"?>");
}
/**
*
* @param elem
*/
public void startElement(String elem)
{
buf.append("<");
buf.append(elem);
buf.append(">");
}
/**
*
* @param elem
*/
public void endElement(String elem)
{
buf.append("</");
buf.append(elem);
buf.append(">");
}
/**
*
* @param elem
*/
public void emptyElement(String elem)
{
buf.append("<");
buf.append(elem);
buf.append("/>");
}
/**
*
* @param text
*/
public void chardata(String text)
{
int l = text.length();
for (int i = 0; i < l; i++)
{
char c = text.charAt(i);
switch (c)
{
case '<' :
buf.append("<");
break;
case '>' :
buf.append(">");
break;
case '&' :
buf.append("&");
break;
default :
buf.append(c);
}
}
}
/**
*
* @param text
*/
public void write(byte[] text)
{
buf.append(text);
}
/**
*
* @param text
*/
public void write(char[] text)
{
chardata(new String(text));
}
/**
*
* @param text
*/
public void write(String text)
{
chardata(text);
}
/**
*
* @return
*/
public String toString()
{
return buf.toString();
}
/**
*
* @return
* @throws UnsupportedEncodingException
*/
public byte[] getBytes() throws UnsupportedEncodingException
{
return buf.toString().getBytes();
}
}
}