--- /dev/null Thu Jan 01 00:00:00 1970 +0000
+++ b/src/java.sql.rowset/share/classes/com/sun/rowset/internal/XmlReaderContentHandler.java Tue Sep 12 19:03:39 2017 +0200
@@ -0,0 +1,1454 @@
+/*
+ * Copyright (c) 2003, 2013, Oracle and/or its affiliates. All rights reserved.
+ * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
+ *
+ * This code is free software; you can redistribute it and/or modify it
+ * under the terms of the GNU General Public License version 2 only, as
+ * published by the Free Software Foundation. Oracle designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Oracle in the LICENSE file that accompanied this code.
+ *
+ * This code 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
+ * version 2 for more details (a copy is included in the LICENSE file that
+ * accompanied this code).
+ *
+ * You should have received a copy of the GNU General Public License version
+ * 2 along with this work; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
+ *
+ * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
+ * or visit www.oracle.com if you need additional information or have any
+ * questions.
+ */
+
+package com.sun.rowset.internal;
+
+import java.util.*;
+
+import org.xml.sax.*;
+import org.xml.sax.helpers.*;
+
+import java.sql.*;
+import javax.sql.*;
+
+import javax.sql.rowset.*;
+import com.sun.rowset.*;
+import java.io.IOException;
+import java.text.MessageFormat;
+
+/**
+ * The document handler that receives parse events that an XML parser sends while it
+ * is parsing an XML document representing a <code>WebRowSet</code> object. The
+ * parser sends strings to this <code>XmlReaderContentHandler</code> and then uses
+ * these strings as arguments for the <code>XmlReaderContentHandler</code> methods
+ * it invokes. The final goal of the SAX parser working with an
+ * <code>XmlReaderContentHandler</code> object is to read an XML document that represents
+ * a <code>RowSet</code> object.
+ * <P>
+ * A rowset consists of its properties, metadata, and data values. An XML document
+ * representating a rowset includes the values in these three categories along with
+ * appropriate XML tags to identify them. It also includes a top-level XML tag for
+ * the rowset and three section tags identifying the three categories of values.
+ * <P>
+ * The tags in an XML document are hierarchical.
+ * This means that the top-level tag, <code>RowSet</code>, is
+ * followed by the three sections with appropriate tags, which are in turn each
+ * followed by their constituent elements. For example, the <code>properties</code>
+ * element will be followed by an element for each of the properties listed in
+ * in this <code>XmlReaderContentHandler</code> object's <code>properties</code>
+ * field. The content of the other two fields, <code>colDef</code>, which lists
+ * the rowset's metadata elements, and <code>data</code>, which lists the rowset's data
+ * elements, are handled similarly .
+ * <P>
+ * This implementation of <code>XmlReaderContentHandler</code> provides the means for the
+ * parser to determine which elements need to have a value set and then to set
+ * those values. The methods in this class are all called by the parser; an
+ * application programmer never calls them directly.
+ *
+ */
+
+public class XmlReaderContentHandler extends DefaultHandler {
+
+ private HashMap <String, Integer> propMap;
+ private HashMap <String, Integer> colDefMap;
+ private HashMap <String, Integer> dataMap;
+
+ private HashMap<String,Class<?>> typeMap;
+
+ private Vector<Object[]> updates;
+ private Vector<String> keyCols;
+
+ private String columnValue;
+ private String propertyValue;
+ private String metaDataValue;
+
+ private int tag;
+ private int state;
+
+ private WebRowSetImpl rs;
+ private boolean nullVal;
+ private boolean emptyStringVal;
+ private RowSetMetaData md;
+ private int idx;
+ private String lastval;
+ private String Key_map;
+ private String Value_map;
+ private String tempStr;
+ private String tempUpdate;
+ private String tempCommand;
+ private Object [] upd;
+
+ /**
+ * A list of the properties for a rowset. There is a constant defined to
+ * correspond to each of these properties so that a <code>HashMap</code>
+ * object can be created to map the properties, which are strings, to
+ * the constants, which are integers.
+ */
+ private String [] properties = {"command", "concurrency", "datasource",
+ "escape-processing", "fetch-direction", "fetch-size",
+ "isolation-level", "key-columns", "map",
+ "max-field-size", "max-rows", "query-timeout",
+ "read-only", "rowset-type", "show-deleted",
+ "table-name", "url", "null", "column", "type",
+ "class", "sync-provider", "sync-provider-name",
+ "sync-provider-vendor", "sync-provider-version",
+ "sync-provider-grade","data-source-lock"};
+
+ /**
+ * A constant representing the tag for the command property.
+ */
+ private final static int CommandTag = 0;
+
+ /**
+ * A constant representing the tag for the concurrency property.
+ */
+ private final static int ConcurrencyTag = 1;
+
+ /**
+ * A constant representing the tag for the datasource property.
+ */
+ private final static int DatasourceTag = 2;
+
+ /**
+ * A constant representing the tag for the escape-processing property.
+ */
+ private final static int EscapeProcessingTag = 3;
+
+ /**
+ * A constant representing the tag for the fetch-direction property.
+ */
+ private final static int FetchDirectionTag = 4;
+
+ /**
+ * A constant representing the tag for the fetch-size property.
+ */
+ private final static int FetchSizeTag = 5;
+
+ /**
+ * A constant representing the tag for the isolation-level property
+ */
+ private final static int IsolationLevelTag = 6;
+
+ /**
+ * A constant representing the tag for the key-columns property.
+ */
+ private final static int KeycolsTag = 7;
+
+ /**
+ * A constant representing the tag for the map property.
+ * This map is the type map that specifies the custom mapping
+ * for an SQL user-defined type.
+ */
+ private final static int MapTag = 8;
+
+ /**
+ * A constant representing the tag for the max-field-size property.
+ */
+ private final static int MaxFieldSizeTag = 9;
+
+ /**
+ * A constant representing the tag for the max-rows property.
+ */
+ private final static int MaxRowsTag = 10;
+
+ /**
+ * A constant representing the tag for the query-timeout property.
+ */
+ private final static int QueryTimeoutTag = 11;
+
+ /**
+ * A constant representing the tag for the read-only property.
+ */
+ private final static int ReadOnlyTag = 12;
+
+ /**
+ * A constant representing the tag for the rowset-type property.
+ */
+ private final static int RowsetTypeTag = 13;
+
+ /**
+ * A constant representing the tag for the show-deleted property.
+ */
+ private final static int ShowDeletedTag = 14;
+
+ /**
+ * A constant representing the tag for the table-name property.
+ */
+ private final static int TableNameTag = 15;
+
+ /**
+ * A constant representing the tag for the URL property.
+ */
+ private final static int UrlTag = 16;
+
+ /**
+ * A constant representing the tag for the null property.
+ */
+ private final static int PropNullTag = 17;
+
+ /**
+ * A constant representing the tag for the column property.
+ */
+ private final static int PropColumnTag = 18;
+
+ /**
+ * A constant representing the tag for the type property.
+ */
+ private final static int PropTypeTag = 19;
+
+ /**
+ * A constant representing the tag for the class property.
+ */
+ private final static int PropClassTag = 20;
+
+ /**
+ * A constant representing the tag for the sync-provider.
+ */
+ private final static int SyncProviderTag = 21;
+
+ /**
+ * A constant representing the tag for the sync-provider
+ * name
+ */
+ private final static int SyncProviderNameTag = 22;
+
+ /**
+ * A constant representing the tag for the sync-provider
+ * vendor tag.
+ */
+ private final static int SyncProviderVendorTag = 23;
+
+ /**
+ * A constant representing the tag for the sync-provider
+ * version tag.
+ */
+ private final static int SyncProviderVersionTag = 24;
+
+ /**
+ * A constant representing the tag for the sync-provider
+ * grade tag.
+ */
+ private final static int SyncProviderGradeTag = 25;
+
+ /**
+ * A constant representing the tag for the data source lock.
+ */
+ private final static int DataSourceLock = 26;
+
+ /**
+ * A listing of the kinds of metadata information available about
+ * the columns in a <code>WebRowSet</code> object.
+ */
+ private String [] colDef = {"column-count", "column-definition", "column-index",
+ "auto-increment", "case-sensitive", "currency",
+ "nullable", "signed", "searchable",
+ "column-display-size", "column-label", "column-name",
+ "schema-name", "column-precision", "column-scale",
+ "table-name", "catalog-name", "column-type",
+ "column-type-name", "null"};
+
+
+ /**
+ * A constant representing the tag for column-count.
+ */
+ private final static int ColumnCountTag = 0;
+
+ /**
+ * A constant representing the tag for column-definition.
+ */
+ private final static int ColumnDefinitionTag = 1;
+
+ /**
+ * A constant representing the tag for column-index.
+ */
+ private final static int ColumnIndexTag = 2;
+
+ /**
+ * A constant representing the tag for auto-increment.
+ */
+ private final static int AutoIncrementTag = 3;
+
+ /**
+ * A constant representing the tag for case-sensitive.
+ */
+ private final static int CaseSensitiveTag = 4;
+
+ /**
+ * A constant representing the tag for currency.
+ */
+ private final static int CurrencyTag = 5;
+
+ /**
+ * A constant representing the tag for nullable.
+ */
+ private final static int NullableTag = 6;
+
+ /**
+ * A constant representing the tag for signed.
+ */
+ private final static int SignedTag = 7;
+
+ /**
+ * A constant representing the tag for searchable.
+ */
+ private final static int SearchableTag = 8;
+
+ /**
+ * A constant representing the tag for column-display-size.
+ */
+ private final static int ColumnDisplaySizeTag = 9;
+
+ /**
+ * A constant representing the tag for column-label.
+ */
+ private final static int ColumnLabelTag = 10;
+
+ /**
+ * A constant representing the tag for column-name.
+ */
+ private final static int ColumnNameTag = 11;
+
+ /**
+ * A constant representing the tag for schema-name.
+ */
+ private final static int SchemaNameTag = 12;
+
+ /**
+ * A constant representing the tag for column-precision.
+ */
+ private final static int ColumnPrecisionTag = 13;
+
+ /**
+ * A constant representing the tag for column-scale.
+ */
+ private final static int ColumnScaleTag = 14;
+
+ /**
+ * A constant representing the tag for table-name.
+ */
+ private final static int MetaTableNameTag = 15;
+
+ /**
+ * A constant representing the tag for catalog-name.
+ */
+ private final static int CatalogNameTag = 16;
+
+ /**
+ * A constant representing the tag for column-type.
+ */
+ private final static int ColumnTypeTag = 17;
+
+ /**
+ * A constant representing the tag for column-type-name.
+ */
+ private final static int ColumnTypeNameTag = 18;
+
+ /**
+ * A constant representing the tag for null.
+ */
+ private final static int MetaNullTag = 19;
+
+ private String [] data = {"currentRow", "columnValue", "insertRow", "deleteRow", "insdel", "updateRow", "null" , "emptyString"};
+
+ private final static int RowTag = 0;
+ private final static int ColTag = 1;
+ private final static int InsTag = 2;
+ private final static int DelTag = 3;
+ private final static int InsDelTag = 4;
+ private final static int UpdTag = 5;
+ private final static int NullTag = 6;
+ private final static int EmptyStringTag = 7;
+
+ /**
+ * A constant indicating the state of this <code>XmlReaderContentHandler</code>
+ * object in which it has not yet been called by the SAX parser and therefore
+ * has no indication of what type of input to expect from the parser next.
+ * <P>
+ * The state is set to <code>INITIAL</code> at the end of each
+ * section, which allows the sections to appear in any order and
+ * still be parsed correctly (except that metadata must be
+ * set before data values can be set).
+ */
+ private final static int INITIAL = 0;
+
+ /**
+ * A constant indicating the state in which this <code>XmlReaderContentHandler</code>
+ * object expects the next input received from the
+ * SAX parser to be a string corresponding to one of the elements in
+ * <code>properties</code>.
+ */
+ private final static int PROPERTIES = 1;
+
+ /**
+ * A constant indicating the state in which this <code>XmlReaderContentHandler</code>
+ * object expects the next input received from the
+ * SAX parser to be a string corresponding to one of the elements in
+ * <code>colDef</code>.
+ */
+ private final static int METADATA = 2;
+
+ /**
+ * A constant indicating the state in which this <code>XmlReaderContentHandler</code>
+ * object expects the next input received from the
+ * SAX parser to be a string corresponding to one of the elements in
+ * <code>data</code>.
+ */
+ private final static int DATA = 3;
+
+ private JdbcRowSetResourceBundle resBundle;
+
+ /**
+ * Constructs a new <code>XmlReaderContentHandler</code> object that will
+ * assist the SAX parser in reading a <code>WebRowSet</code> object in the
+ * format of an XML document. In addition to setting some default values,
+ * this constructor creates three <code>HashMap</code> objects, one for
+ * properties, one for metadata, and one for data. These hash maps map the
+ * strings sent by the SAX parser to integer constants so that they can be
+ * compared more efficiently in <code>switch</code> statements.
+ *
+ * @param r the <code>RowSet</code> object in XML format that will be read
+ */
+ public XmlReaderContentHandler(RowSet r) {
+ // keep the rowset we've been given
+ rs = (WebRowSetImpl)r;
+
+ // set-up the token maps
+ initMaps();
+
+ // allocate the collection for the updates
+ updates = new Vector<>();
+
+ // start out with the empty string
+ columnValue = "";
+ propertyValue = "";
+ metaDataValue = "";
+
+ nullVal = false;
+ idx = 0;
+ tempStr = "";
+ tempUpdate = "";
+ tempCommand = "";
+
+ try {
+ resBundle = JdbcRowSetResourceBundle.getJdbcRowSetResourceBundle();
+ } catch(IOException ioe) {
+ throw new RuntimeException(ioe);
+ }
+ }
+
+ /**
+ * Creates and initializes three new <code>HashMap</code> objects that map
+ * the strings returned by the SAX parser to <code>Integer</code>
+ * objects. The strings returned by the parser will match the strings that
+ * are array elements in this <code>XmlReaderContentHandler</code> object's
+ * <code>properties</code>, <code>colDef</code>, or <code>data</code>
+ * fields. For each array element in these fields, there is a corresponding
+ * constant defined. It is to these constants that the strings are mapped.
+ * In the <code>HashMap</code> objects, the string is the key, and the
+ * integer is the value.
+ * <P>
+ * The purpose of the mapping is to make comparisons faster. Because comparing
+ * numbers is more efficient than comparing strings, the strings returned
+ * by the parser are mapped to integers, which can then be used in a
+ * <code>switch</code> statement.
+ */
+ private void initMaps() {
+ int items, i;
+
+ propMap = new HashMap<>();
+ items = properties.length;
+
+ for (i=0;i<items;i++) {
+ propMap.put(properties[i], Integer.valueOf(i));
+ }
+
+ colDefMap = new HashMap<>();
+ items = colDef.length;
+
+ for (i=0;i<items;i++) {
+ colDefMap.put(colDef[i], Integer.valueOf(i));
+ }
+
+ dataMap = new HashMap<>();
+ items = data.length;
+
+ for (i=0;i<items;i++) {
+ dataMap.put(data[i], Integer.valueOf(i));
+ }
+
+ //Initialize connection map here
+ typeMap = new HashMap<>();
+ }
+
+ public void startDocument() throws SAXException {
+ }
+
+ public void endDocument() throws SAXException {
+ }
+
+
+ /**
+ * Sets this <code>XmlReaderContentHandler</code> object's <code>tag</code>
+ * field if the given name is the key for a tag and this object's state
+ * is not <code>INITIAL</code>. The field is set
+ * to the constant that corresponds to the given element name.
+ * If the state is <code>INITIAL</code>, the state is set to the given
+ * name, which will be one of the sections <code>PROPERTIES</code>,
+ * <code>METADATA</code>, or <code>DATA</code>. In either case, this
+ * method puts this document handler in the proper state for calling
+ * the method <code>endElement</code>.
+ * <P>
+ * If the state is <code>DATA</code> and the tag is <code>RowTag</code>,
+ * <code>DelTag</code>, or <code>InsTag</code>, this method moves the
+ * rowset's cursor to the insert row and sets this
+ * <code>XmlReaderContentHandler</code> object's <code>idx</code>
+ * field to <code>0</code> so that it will be in the proper
+ * state when the parser calls the method <code>endElement</code>.
+ *
+ * @param lName the name of the element; either (1) one of the array
+ * elements in the fields <code>properties</code>,
+ * <code>colDef</code>, or <code>data</code> or
+ * (2) one of the <code>RowSet</code> elements
+ * <code>"properties"</code>, <code>"metadata"</code>, or
+ * <code>"data"</code>
+ * @param attributes <code>org.xml.sax.AttributeList</code> objects that are
+ * attributes of the named section element; may be <code>null</code>
+ * if there are no attributes, which is the case for
+ * <code>WebRowSet</code> objects
+ * @exception SAXException if a general SAX error occurs
+ */
+ public void startElement(String uri, String lName, String qName, Attributes attributes) throws SAXException {
+ int tag;
+ String name = "";
+
+ name = lName;
+
+ switch (getState()) {
+ case PROPERTIES:
+
+ tempCommand = "";
+ tag = propMap.get(name);
+ if (tag == PropNullTag)
+ setNullValue(true);
+ else
+ setTag(tag);
+ break;
+ case METADATA:
+ tag = colDefMap.get(name);
+
+ if (tag == MetaNullTag)
+ setNullValue(true);
+ else
+ setTag(tag);
+ break;
+ case DATA:
+
+ /**
+ * This has been added to clear out the values of the previous read
+ * so that we should not add up values of data between different tags
+ */
+ tempStr = "";
+ tempUpdate = "";
+ if(dataMap.get(name) == null) {
+ tag = NullTag;
+ } else if(dataMap.get(name) == EmptyStringTag) {
+ tag = EmptyStringTag;
+ } else {
+ tag = dataMap.get(name);
+ }
+
+ if (tag == NullTag) {
+ setNullValue(true);
+ } else if(tag == EmptyStringTag) {
+ setEmptyStringValue(true);
+ } else {
+ setTag(tag);
+
+ if (tag == RowTag || tag == DelTag || tag == InsTag) {
+ idx = 0;
+ try {
+ rs.moveToInsertRow();
+ } catch (SQLException ex) {
+ ;
+ }
+ }
+ }
+
+ break;
+ default:
+ setState(name);
+ }
+
+ }
+
+ /**
+ * Sets the value for the given element if <code>name</code> is one of
+ * the array elements in the fields <code>properties</code>,
+ * <code>colDef</code>, or <code>data</code> and this
+ * <code>XmlReaderContentHandler</code> object's state is not
+ * <code>INITIAL</code>. If the state is <code>INITIAL</code>,
+ * this method does nothing.
+ * <P>
+ * If the state is <code>METADATA</code> and
+ * the argument supplied is <code>"metadata"</code>, the rowset's
+ * metadata is set. If the state is <code>PROPERTIES</code>, the
+ * appropriate property is set using the given name to determine
+ * the appropriate value. If the state is <code>DATA</code> and
+ * the argument supplied is <code>"data"</code>, this method sets
+ * the state to <code>INITIAL</code> and returns. If the argument
+ * supplied is one of the elements in the field <code>data</code>,
+ * this method makes the appropriate changes to the rowset's data.
+ *
+ * @param lName the name of the element; either (1) one of the array
+ * elements in the fields <code>properties</code>,
+ * <code>colDef</code>, or <code>data</code> or
+ * (2) one of the <code>RowSet</code> elements
+ * <code>"properties"</code>, <code>"metadata"</code>, or
+ * <code>"data"</code>
+ *
+ * @exception SAXException if a general SAX error occurs
+ */
+ @SuppressWarnings("fallthrough")
+ public void endElement(String uri, String lName, String qName) throws SAXException {
+ int tag;
+
+ String name = "";
+ name = lName;
+
+ switch (getState()) {
+ case PROPERTIES:
+ if (name.equals("properties")) {
+ state = INITIAL;
+ break;
+ }
+
+ try {
+ tag = propMap.get(name);
+ switch (tag) {
+ case KeycolsTag:
+ if (keyCols != null) {
+ int i[] = new int[keyCols.size()];
+ for (int j = 0; j < i.length; j++)
+ i[j] = Integer.parseInt(keyCols.elementAt(j));
+ rs.setKeyColumns(i);
+ }
+ break;
+
+ case PropClassTag:
+ //Added the handling for Class tags to take care of maps
+ //Makes an entry into the map upon end of class tag
+ try{
+ typeMap.put(Key_map,sun.reflect.misc.ReflectUtil.forName(Value_map));
+
+ }catch(ClassNotFoundException ex) {
+ throw new SAXException(MessageFormat.format(resBundle.handleGetObject("xmlrch.errmap").toString(), ex.getMessage()));
+ }
+ break;
+
+ case MapTag:
+ //Added the handling for Map to take set the typeMap
+ rs.setTypeMap(typeMap);
+ break;
+
+ default:
+ break;
+ }
+
+ if (getNullValue()) {
+ setPropertyValue(null);
+ setNullValue(false);
+ } else {
+ setPropertyValue(propertyValue);
+ }
+ } catch (SQLException ex) {
+ throw new SAXException(ex.getMessage());
+ }
+
+ // propertyValue need to be reset to an empty string
+ propertyValue = "";
+ setTag(-1);
+ break;
+ case METADATA:
+ if (name.equals("metadata")) {
+ try {
+ rs.setMetaData(md);
+ state = INITIAL;
+ } catch (SQLException ex) {
+ throw new SAXException(MessageFormat.format(resBundle.handleGetObject("xmlrch.errmetadata").toString(), ex.getMessage()));
+ }
+ } else {
+ try {
+ if (getNullValue()) {
+ setMetaDataValue(null);
+ setNullValue(false);
+ } else {
+ setMetaDataValue(metaDataValue);
+ }
+ } catch (SQLException ex) {
+ throw new SAXException(MessageFormat.format(resBundle.handleGetObject("xmlrch.errmetadata").toString(), ex.getMessage()));
+
+ }
+ // metaDataValue needs to be reset to an empty string
+ metaDataValue = "";
+ }
+ setTag(-1);
+ break;
+ case DATA:
+ if (name.equals("data")) {
+ state = INITIAL;
+ return;
+ }
+
+ if(dataMap.get(name) == null) {
+ tag = NullTag;
+ } else {
+ tag = dataMap.get(name);
+ }
+ switch (tag) {
+ case ColTag:
+ try {
+ idx++;
+ if (getNullValue()) {
+ insertValue(null);
+ setNullValue(false);
+ } else {
+ insertValue(tempStr);
+ }
+ // columnValue now need to be reset to the empty string
+ columnValue = "";
+ } catch (SQLException ex) {
+ throw new SAXException(MessageFormat.format(resBundle.handleGetObject("xmlrch.errinsertval").toString(), ex.getMessage()));
+ }
+ break;
+ case RowTag:
+ try {
+ rs.insertRow();
+ rs.moveToCurrentRow();
+ rs.next();
+
+ // Making this as the original to turn off the
+ // rowInserted flagging
+ rs.setOriginalRow();
+
+ applyUpdates();
+ } catch (SQLException ex) {
+ throw new SAXException(MessageFormat.format(resBundle.handleGetObject("xmlrch.errconstr").toString(), ex.getMessage()));
+ }
+ break;
+ case DelTag:
+ try {
+ rs.insertRow();
+ rs.moveToCurrentRow();
+ rs.next();
+ rs.setOriginalRow();
+ applyUpdates();
+ rs.deleteRow();
+ } catch (SQLException ex) {
+ throw new SAXException(MessageFormat.format(resBundle.handleGetObject("xmlrch.errdel").toString() , ex.getMessage()));
+ }
+ break;
+ case InsTag:
+ try {
+ rs.insertRow();
+ rs.moveToCurrentRow();
+ rs.next();
+ applyUpdates();
+ } catch (SQLException ex) {
+ throw new SAXException(MessageFormat.format(resBundle.handleGetObject("xmlrch.errinsert").toString() , ex.getMessage()));
+ }
+ break;
+
+ case InsDelTag:
+ try {
+ rs.insertRow();
+ rs.moveToCurrentRow();
+ rs.next();
+ rs.setOriginalRow();
+ applyUpdates();
+ } catch (SQLException ex) {
+ throw new SAXException(MessageFormat.format(resBundle.handleGetObject("xmlrch.errinsdel").toString() , ex.getMessage()));
+ }
+ break;
+
+ case UpdTag:
+ try {
+ if(getNullValue())
+ {
+ insertValue(null);
+ setNullValue(false);
+ } else if(getEmptyStringValue()) {
+ insertValue("");
+ setEmptyStringValue(false);
+ } else {
+ updates.add(upd);
+ }
+ } catch(SQLException ex) {
+ throw new SAXException(MessageFormat.format(resBundle.handleGetObject("xmlrch.errupdate").toString() , ex.getMessage()));
+ }
+ break;
+
+ default:
+ break;
+ }
+ default:
+ break;
+ }
+ }
+
+ private void applyUpdates() throws SAXException {
+ // now handle any updates
+ if (updates.size() > 0) {
+ try {
+ Object upd[];
+ Iterator<?> i = updates.iterator();
+ while (i.hasNext()) {
+ upd = (Object [])i.next();
+ idx = ((Integer)upd[0]).intValue();
+
+ if(!(lastval.equals(upd[1]))){
+ insertValue((String)(upd[1]));
+ }
+ }
+
+ rs.updateRow();
+ } catch (SQLException ex) {
+ throw new SAXException(MessageFormat.format(resBundle.handleGetObject("xmlrch.errupdrow").toString() , ex.getMessage()));
+ }
+ updates.removeAllElements();
+ }
+
+
+ }
+
+ /**
+ * Sets a property, metadata, or data value with the characters in
+ * the given array of characters, starting with the array element
+ * indicated by <code>start</code> and continuing for <code>length</code>
+ * number of characters.
+ * <P>
+ * The SAX parser invokes this method and supplies
+ * the character array, start position, and length parameter values it
+ * got from parsing the XML document. An application programmer never
+ * invokes this method directly.
+ *
+ * @param ch an array of characters supplied by the SAX parser, all or part of
+ * which will be used to set a value
+ * @param start the position in the given array at which to start
+ * @param length the number of consecutive characters to use
+ */
+ public void characters(char[] ch, int start, int length) throws SAXException {
+ try {
+ switch (getState()) {
+ case PROPERTIES:
+ propertyValue = new String(ch, start, length);
+
+ /**
+ * This has been added for handling of special characters. When special
+ * characters are encountered the characters function gets called for
+ * each of the characters so we need to append the value got in the
+ * previous call as it is the same data present between the start and
+ * the end tag.
+ **/
+ tempCommand = tempCommand.concat(propertyValue);
+ propertyValue = tempCommand;
+
+ // Added the following check for handling of type tags in maps
+ if(tag == PropTypeTag)
+ {
+ Key_map = propertyValue;
+ }
+
+ // Added the following check for handling of class tags in maps
+ else if(tag == PropClassTag)
+ {
+ Value_map = propertyValue;
+ }
+ break;
+
+ case METADATA:
+
+ // The parser will come here after the endElement as there is
+ // "\n" in the after endTag is printed. This will cause a problem
+ // when the data between the tags is an empty string so adding
+ // below condition to take care of that situation.
+
+ if (tag == -1)
+ {
+ break;
+ }
+
+ metaDataValue = new String(ch, start, length);
+ break;
+ case DATA:
+ setDataValue(ch, start, length);
+ break;
+ default:
+ ;
+ }
+ } catch (SQLException ex) {
+ throw new SAXException(resBundle.handleGetObject("xmlrch.chars").toString() + ex.getMessage());
+ }
+ }
+
+ private void setState(String s) throws SAXException {
+ if (s.equals("webRowSet")) {
+ state = INITIAL;
+ } else if (s.equals("properties")) {
+ if (state != PROPERTIES)
+ state = PROPERTIES;
+ else
+ state = INITIAL;
+ } else if (s.equals("metadata")) {
+ if (state != METADATA)
+ state = METADATA;
+ else
+ state = INITIAL;
+ } else if (s.equals("data")) {
+ if (state != DATA)
+ state = DATA;
+ else
+ state = INITIAL;
+ }
+
+ }
+
+ /**
+ * Retrieves the current state of this <code>XmlReaderContentHandler</code>
+ * object's rowset, which is stored in the document handler's
+ * <code>state</code> field.
+ *
+ * @return one of the following constants:
+ * <code>XmlReaderContentHandler.PROPERTIES</code>
+ * <code>XmlReaderContentHandler.METADATA</code>
+ * <code>XmlReaderContentHandler.DATA</code>
+ * <code>XmlReaderContentHandler.INITIAL</code>
+ */
+ private int getState() {
+ return state;
+ }
+
+ private void setTag(int t) {
+ tag = t;
+ }
+
+ private int getTag() {
+ return tag;
+ }
+
+ private void setNullValue(boolean n) {
+ nullVal = n;
+ }
+
+ private boolean getNullValue() {
+ return nullVal;
+ }
+
+ private void setEmptyStringValue(boolean e) {
+ emptyStringVal = e;
+ }
+
+ private boolean getEmptyStringValue() {
+ return emptyStringVal;
+ }
+
+ private String getStringValue(String s) {
+ return s;
+ }
+
+ private int getIntegerValue(String s) {
+ return Integer.parseInt(s);
+ }
+
+ private boolean getBooleanValue(String s) {
+
+ return Boolean.valueOf(s).booleanValue();
+ }
+
+ private java.math.BigDecimal getBigDecimalValue(String s) {
+ return new java.math.BigDecimal(s);
+ }
+
+ private byte getByteValue(String s) {
+ return Byte.parseByte(s);
+ }
+
+ private short getShortValue(String s) {
+ return Short.parseShort(s);
+ }
+
+ private long getLongValue(String s) {
+ return Long.parseLong(s);
+ }
+
+ private float getFloatValue(String s) {
+ return Float.parseFloat(s);
+ }
+
+ private double getDoubleValue(String s) {
+ return Double.parseDouble(s);
+ }
+
+ private byte[] getBinaryValue(String s) {
+ return s.getBytes();
+ }
+
+ private java.sql.Date getDateValue(String s) {
+ return new java.sql.Date(getLongValue(s));
+ }
+
+ private java.sql.Time getTimeValue(String s) {
+ return new java.sql.Time(getLongValue(s));
+ }
+
+ private java.sql.Timestamp getTimestampValue(String s) {
+ return new java.sql.Timestamp(getLongValue(s));
+ }
+
+ private void setPropertyValue(String s) throws SQLException {
+ // find out if we are going to be dealing with a null
+ boolean nullValue = getNullValue();
+
+ switch(getTag()) {
+ case CommandTag:
+ if (nullValue)
+ ; //rs.setCommand(null);
+ else
+ rs.setCommand(s);
+ break;
+ case ConcurrencyTag:
+ if (nullValue)
+ throw new SQLException(resBundle.handleGetObject("xmlrch.badvalue").toString());
+ else
+ rs.setConcurrency(getIntegerValue(s));
+ break;
+ case DatasourceTag:
+ if (nullValue)
+ rs.setDataSourceName(null);
+ else
+ rs.setDataSourceName(s);
+ break;
+ case EscapeProcessingTag:
+ if (nullValue)
+ throw new SQLException(resBundle.handleGetObject("xmlrch.badvalue").toString());
+ else
+ rs.setEscapeProcessing(getBooleanValue(s));
+ break;
+ case FetchDirectionTag:
+ if (nullValue)
+ throw new SQLException(resBundle.handleGetObject("xmlrch.badvalue").toString());
+ else
+ rs.setFetchDirection(getIntegerValue(s));
+ break;
+ case FetchSizeTag:
+ if (nullValue)
+ throw new SQLException(resBundle.handleGetObject("xmlrch.badvalue").toString());
+ else
+ rs.setFetchSize(getIntegerValue(s));
+ break;
+ case IsolationLevelTag:
+ if (nullValue)
+ throw new SQLException(resBundle.handleGetObject("xmlrch.badvalue").toString());
+ else
+ rs.setTransactionIsolation(getIntegerValue(s));
+ break;
+ case KeycolsTag:
+ break;
+ case PropColumnTag:
+ if (keyCols == null)
+ keyCols = new Vector<>();
+ keyCols.add(s);
+ break;
+ case MapTag:
+ break;
+ case MaxFieldSizeTag:
+ if (nullValue)
+ throw new SQLException(resBundle.handleGetObject("xmlrch.badvalue").toString());
+ else
+ rs.setMaxFieldSize(getIntegerValue(s));
+ break;
+ case MaxRowsTag:
+ if (nullValue)
+ throw new SQLException(resBundle.handleGetObject("xmlrch.badvalue").toString());
+ else
+ rs.setMaxRows(getIntegerValue(s));
+ break;
+ case QueryTimeoutTag:
+ if (nullValue)
+ throw new SQLException(resBundle.handleGetObject("xmlrch.badvalue").toString());
+ else
+ rs.setQueryTimeout(getIntegerValue(s));
+ break;
+ case ReadOnlyTag:
+ if (nullValue)
+ throw new SQLException(resBundle.handleGetObject("xmlrch.badvalue").toString());
+ else
+ rs.setReadOnly(getBooleanValue(s));
+ break;
+ case RowsetTypeTag:
+ if (nullValue) {
+ throw new SQLException(resBundle.handleGetObject("xmlrch.badvalue").toString());
+ } else {
+ //rs.setType(getIntegerValue(s));
+ String strType = getStringValue(s);
+ int iType = 0;
+
+ if(strType.trim().equals("ResultSet.TYPE_SCROLL_INSENSITIVE")) {
+ iType = 1004;
+ } else if(strType.trim().equals("ResultSet.TYPE_SCROLL_SENSITIVE")) {
+ iType = 1005;
+ } else if(strType.trim().equals("ResultSet.TYPE_FORWARD_ONLY")) {
+ iType = 1003;
+ }
+ rs.setType(iType);
+ }
+ break;
+ case ShowDeletedTag:
+ if (nullValue)
+ throw new SQLException(resBundle.handleGetObject("xmlrch.badvalue").toString());
+ else
+ rs.setShowDeleted(getBooleanValue(s));
+ break;
+ case TableNameTag:
+ if (nullValue)
+ //rs.setTableName(null);
+ ;
+ else
+ rs.setTableName(s);
+ break;
+ case UrlTag:
+ if (nullValue)
+ rs.setUrl(null);
+ else
+ rs.setUrl(s);
+ break;
+ case SyncProviderNameTag:
+ if (nullValue) {
+ rs.setSyncProvider(null);
+ } else {
+ String str = s.substring(0,s.indexOf('@')+1);
+ rs.setSyncProvider(str);
+ }
+ break;
+ case SyncProviderVendorTag:
+ // to be implemented
+ break;
+ case SyncProviderVersionTag:
+ // to be implemented
+ break;
+ case SyncProviderGradeTag:
+ // to be implemented
+ break;
+ case DataSourceLock:
+ // to be implemented
+ break;
+ default:
+ break;
+ }
+
+ }
+
+ private void setMetaDataValue(String s) throws SQLException {
+ // find out if we are going to be dealing with a null
+ boolean nullValue = getNullValue();
+
+ switch (getTag()) {
+ case ColumnCountTag:
+ md = new RowSetMetaDataImpl();
+ idx = 0;
+
+ if (nullValue) {
+ throw new SQLException(resBundle.handleGetObject("xmlrch.badvalue1").toString());
+ } else {
+ md.setColumnCount(getIntegerValue(s));
+ }
+ break;
+ case ColumnDefinitionTag:
+ break;
+ case ColumnIndexTag:
+ idx++;
+ break;
+ case AutoIncrementTag:
+ if (nullValue)
+ throw new SQLException(resBundle.handleGetObject("xmlrch.badvalue1").toString());
+ else
+ md.setAutoIncrement(idx, getBooleanValue(s));
+ break;
+ case CaseSensitiveTag:
+ if (nullValue)
+ throw new SQLException(resBundle.handleGetObject("xmlrch.badvalue1").toString());
+ else
+ md.setCaseSensitive(idx, getBooleanValue(s));
+ break;
+ case CurrencyTag:
+ if (nullValue)
+ throw new SQLException(resBundle.handleGetObject("xmlrch.badvalue1").toString());
+ else
+ md.setCurrency(idx, getBooleanValue(s));
+ break;
+ case NullableTag:
+ if (nullValue)
+ throw new SQLException(resBundle.handleGetObject("xmlrch.badvalue1").toString());
+ else
+ md.setNullable(idx, getIntegerValue(s));
+ break;
+ case SignedTag:
+ if (nullValue)
+ throw new SQLException(resBundle.handleGetObject("xmlrch.badvalue1").toString());
+ else
+ md.setSigned(idx, getBooleanValue(s));
+ break;
+ case SearchableTag:
+ if (nullValue)
+ throw new SQLException(resBundle.handleGetObject("xmlrch.badvalue1").toString());
+ else
+ md.setSearchable(idx, getBooleanValue(s));
+ break;
+ case ColumnDisplaySizeTag:
+ if (nullValue)
+ throw new SQLException(resBundle.handleGetObject("xmlrch.badvalue1").toString());
+ else
+ md.setColumnDisplaySize(idx, getIntegerValue(s));
+ break;
+ case ColumnLabelTag:
+ if (nullValue)
+ md.setColumnLabel(idx, null);
+ else
+ md.setColumnLabel(idx, s);
+ break;
+ case ColumnNameTag:
+ if (nullValue)
+ md.setColumnName(idx, null);
+ else
+ md.setColumnName(idx, s);
+
+ break;
+ case SchemaNameTag:
+ if (nullValue) {
+ md.setSchemaName(idx, null); }
+ else
+ md.setSchemaName(idx, s);
+ break;
+ case ColumnPrecisionTag:
+ if (nullValue)
+ throw new SQLException(resBundle.handleGetObject("xmlrch.badvalue1").toString());
+ else
+ md.setPrecision(idx, getIntegerValue(s));
+ break;
+ case ColumnScaleTag:
+ if (nullValue)
+ throw new SQLException(resBundle.handleGetObject("xmlrch.badvalue1").toString());
+ else
+ md.setScale(idx, getIntegerValue(s));
+ break;
+ case MetaTableNameTag:
+ if (nullValue)
+ md.setTableName(idx, null);
+ else
+ md.setTableName(idx, s);
+ break;
+ case CatalogNameTag:
+ if (nullValue)
+ md.setCatalogName(idx, null);
+ else
+ md.setCatalogName(idx, s);
+ break;
+ case ColumnTypeTag:
+ if (nullValue)
+ throw new SQLException(resBundle.handleGetObject("xmlrch.badvalue1").toString());
+ else
+ md.setColumnType(idx, getIntegerValue(s));
+ break;
+ case ColumnTypeNameTag:
+ if (nullValue)
+ md.setColumnTypeName(idx, null);
+ else
+ md.setColumnTypeName(idx, s);
+ break;
+ default:
+ //System.out.println("MetaData: Unknown Tag: (" + getTag() + ")");
+ break;
+
+ }
+ }
+
+ private void setDataValue(char[] ch, int start, int len) throws SQLException {
+ switch (getTag()) {
+ case ColTag:
+ columnValue = new String(ch, start, len);
+ /**
+ * This has been added for handling of special characters. When special
+ * characters are encountered the characters function gets called for
+ * each of the characters so we need to append the value got in the
+ * previous call as it is the same data present between the start and
+ * the end tag.
+ **/
+ tempStr = tempStr.concat(columnValue);
+ break;
+ case UpdTag:
+ upd = new Object[2];
+
+ /**
+ * This has been added for handling of special characters. When special
+ * characters are encountered the characters function gets called for
+ * each of the characters so we need to append the value got in the
+ * previous call as it is the same data present between the start and
+ * the end tag.
+ **/
+
+ tempUpdate = tempUpdate.concat(new String(ch,start,len));
+ upd[0] = Integer.valueOf(idx);
+ upd[1] = tempUpdate;
+ //updates.add(upd);
+
+ lastval = (String)upd[1];
+ //insertValue(ch, start, len);
+ break;
+ case InsTag:
+
+ }
+ }
+
+ private void insertValue(String s) throws SQLException {
+
+ if (getNullValue()) {
+ rs.updateNull(idx);
+ return;
+ }
+
+ // no longer have to deal with those pesky nulls.
+ int type = rs.getMetaData().getColumnType(idx);
+ switch (type) {
+ case java.sql.Types.BIT:
+ rs.updateBoolean(idx, getBooleanValue(s));
+ break;
+ case java.sql.Types.BOOLEAN:
+ rs.updateBoolean(idx, getBooleanValue(s));
+ break;
+ case java.sql.Types.SMALLINT:
+ case java.sql.Types.TINYINT:
+ rs.updateShort(idx, getShortValue(s));
+ break;
+ case java.sql.Types.INTEGER:
+ rs.updateInt(idx, getIntegerValue(s));
+ break;
+ case java.sql.Types.BIGINT:
+ rs.updateLong(idx, getLongValue(s));
+ break;
+ case java.sql.Types.REAL:
+ case java.sql.Types.FLOAT:
+ rs.updateFloat(idx, getFloatValue(s));
+ break;
+ case java.sql.Types.DOUBLE:
+ rs.updateDouble(idx, getDoubleValue(s));
+ break;
+ case java.sql.Types.NUMERIC:
+ case java.sql.Types.DECIMAL:
+ rs.updateObject(idx, getBigDecimalValue(s));
+ break;
+ case java.sql.Types.BINARY:
+ case java.sql.Types.VARBINARY:
+ case java.sql.Types.LONGVARBINARY:
+ rs.updateBytes(idx, getBinaryValue(s));
+ break;
+ case java.sql.Types.DATE:
+ rs.updateDate(idx, getDateValue(s));
+ break;
+ case java.sql.Types.TIME:
+ rs.updateTime(idx, getTimeValue(s));
+ break;
+ case java.sql.Types.TIMESTAMP:
+ rs.updateTimestamp(idx, getTimestampValue(s));
+ break;
+ case java.sql.Types.CHAR:
+ case java.sql.Types.VARCHAR:
+ case java.sql.Types.LONGVARCHAR:
+ rs.updateString(idx, getStringValue(s));
+ break;
+ default:
+
+ }
+
+ }
+
+ /**
+ * Throws the given <code>SAXParseException</code> object. This
+ * exception was originally thrown by the SAX parser and is passed
+ * to the method <code>error</code> when the SAX parser invokes it.
+ *
+ * @param e the <code>SAXParseException</code> object to throw
+ */
+ public void error (SAXParseException e) throws SAXParseException {
+ throw e;
+ }
+
+ // dump warnings too
+ /**
+ * Prints a warning message to <code>System.out</code> giving the line
+ * number and uri for what caused the warning plus a message explaining
+ * the reason for the warning. This method is invoked by the SAX parser.
+ *
+ * @param err a warning generated by the SAX parser
+ */
+ public void warning (SAXParseException err) throws SAXParseException {
+ System.out.println (MessageFormat.format(resBundle.handleGetObject("xmlrch.warning").toString(), new Object[] { err.getMessage(), err.getLineNumber(), err.getSystemId() }));
+ }
+
+ /**
+ *
+ */
+ public void notationDecl(String name, String publicId, String systemId) {
+
+ }
+
+ /**
+ *
+ */
+ public void unparsedEntityDecl(String name, String publicId, String systemId, String notationName) {
+
+ }
+
+ /**
+ * Returns the current row of this <code>Rowset</code>object.
+ * The ResultSet's cursor is positioned at the Row which is needed
+ *
+ * @return the <code>Row</code> object on which the <code>RowSet</code>
+ * implementation objects's cursor is positioned
+ */
+ private Row getPresentRow(WebRowSetImpl rs) throws SQLException {
+ //rs.setOriginalRow();
+ // ResultSetMetaData rsmd = rs.getMetaData();
+ // int numCols = rsmd.getColumnCount();
+ // Object vals[] = new Object[numCols];
+ // for(int j = 1; j<= numCols ; j++){
+ // vals[j-1] = rs.getObject(j);
+ // }
+ // return(new Row(numCols, vals));
+ return null;
+ }
+
+
+
+
+}