/* | 
|
 * 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 javax.sql.rowset.serial;  | 
|
import java.sql.*;  | 
|
import javax.sql.*;  | 
|
import java.io.*;  | 
|
import java.math.*;  | 
|
import java.util.Arrays;  | 
|
import java.util.Map;  | 
|
import java.util.Vector;  | 
|
import javax.sql.rowset.*;  | 
|
/** | 
|
 * A serialized mapping in the Java programming language of an SQL | 
|
 * structured type. Each attribute that is not already serialized | 
|
 * is mapped to a serialized form, and if an attribute is itself | 
|
 * a structured type, each of its attributes that is not already | 
|
 * serialized is mapped to a serialized form. | 
|
 * <P> | 
|
 * In addition, the structured type is custom mapped to a class in the | 
|
 * Java programming language if there is such a mapping, as are | 
|
 * its attributes, if appropriate. | 
|
 * <P> | 
|
 * The <code>SerialStruct</code> class provides a constructor for creating | 
|
 * an instance from a <code>Struct</code> object, a method for retrieving | 
|
 * the SQL type name of the SQL structured type in the database, and methods | 
|
 * for retrieving its attribute values. | 
|
 * | 
|
 * <h3> Thread safety </h3> | 
|
 * | 
|
 * A SerialStruct is not safe for use by multiple concurrent threads.  If a | 
|
 * SerialStruct is to be used by more than one thread then access to the | 
|
 * SerialStruct should be controlled by appropriate synchronization. | 
|
 * | 
|
*/  | 
|
public class SerialStruct implements Struct, Serializable, Cloneable {  | 
|
    /** | 
|
     * The SQL type name for the structured type that this | 
|
     * <code>SerialStruct</code> object represents.  This is the name | 
|
     * used in the SQL definition of the SQL structured type. | 
|
     * | 
|
     * @serial | 
|
*/  | 
|
private String SQLTypeName;  | 
|
    /** | 
|
     * An array of <code>Object</code> instances in  which each | 
|
     * element is an attribute of the SQL structured type that this | 
|
     * <code>SerialStruct</code> object represents.  The attributes are | 
|
     * ordered according to their order in the definition of the | 
|
     * SQL structured type. | 
|
     * | 
|
     * @serial | 
|
*/  | 
|
private Object attribs[];  | 
|
    /** | 
|
     * Constructs a <code>SerialStruct</code> object from the given | 
|
     * <code>Struct</code> object, using the given <code>java.util.Map</code> | 
|
     * object for custom mapping the SQL structured type or any of its | 
|
     * attributes that are SQL structured types. | 
|
     * | 
|
     * @param in an instance of {@code Struct} | 
|
     * @param map a <code>java.util.Map</code> object in which | 
|
     *        each entry consists of 1) a <code>String</code> object | 
|
     *        giving the fully qualified name of a UDT and 2) the | 
|
     *        <code>Class</code> object for the <code>SQLData</code> implementation | 
|
     *        that defines how the UDT is to be mapped | 
|
     * @throws SerialException if an error occurs | 
|
     * @see java.sql.Struct | 
|
*/  | 
|
public SerialStruct(Struct in, Map<String,Class<?>> map)  | 
|
throws SerialException  | 
|
     { | 
|
        try { | 
|
        // get the type name | 
|
SQLTypeName = in.getSQLTypeName();  | 
|
System.out.println("SQLTypeName: " + SQLTypeName);  | 
|
        // get the attributes of the struct | 
|
attribs = in.getAttributes(map);  | 
|
        /* | 
|
         * the array may contain further Structs | 
|
         * and/or classes that have been mapped, | 
|
         * other types that we have to serialize | 
|
*/  | 
|
mapToSerial(map);  | 
|
} catch (SQLException e) {  | 
|
throw new SerialException(e.getMessage());  | 
|
}  | 
|
}  | 
|
     /** | 
|
      * Constructs a <code>SerialStruct</code> object from the | 
|
      * given <code>SQLData</code> object, using the given type | 
|
      * map to custom map it to a class in the Java programming | 
|
      * language.  The type map gives the SQL type and the class | 
|
      * to which it is mapped.  The <code>SQLData</code> object | 
|
      * defines the class to which the SQL type will be mapped. | 
|
      * | 
|
      * @param in an instance of the <code>SQLData</code> class | 
|
      *           that defines the mapping of the SQL structured | 
|
      *           type to one or more objects in the Java programming language | 
|
      * @param map a <code>java.util.Map</code> object in which | 
|
      *        each entry consists of 1) a <code>String</code> object | 
|
      *        giving the fully qualified name of a UDT and 2) the | 
|
      *        <code>Class</code> object for the <code>SQLData</code> implementation | 
|
      *        that defines how the UDT is to be mapped | 
|
      * @throws SerialException if an error occurs | 
|
*/  | 
|
public SerialStruct(SQLData in, Map<String,Class<?>> map)  | 
|
throws SerialException  | 
|
    { | 
|
        try { | 
|
        //set the type name | 
|
SQLTypeName = in.getSQLTypeName();  | 
|
Vector<Object> tmp = new Vector<>();  | 
|
in.writeSQL(new SQLOutputImpl(tmp, map));  | 
|
attribs = tmp.toArray();  | 
|
} catch (SQLException e) {  | 
|
throw new SerialException(e.getMessage());  | 
|
}  | 
|
}  | 
|
    /** | 
|
     * Retrieves the SQL type name for this <code>SerialStruct</code> | 
|
     * object. This is the name used in the SQL definition of the | 
|
     * structured type | 
|
     * | 
|
     * @return a <code>String</code> object representing the SQL | 
|
     *         type name for the SQL structured type that this | 
|
     *         <code>SerialStruct</code> object represents | 
|
     * @throws SerialException if an error occurs | 
|
*/  | 
|
public String getSQLTypeName() throws SerialException {  | 
|
return SQLTypeName;  | 
|
}  | 
|
    /** | 
|
     * Retrieves an array of <code>Object</code> values containing the | 
|
     * attributes of the SQL structured type that this | 
|
     * <code>SerialStruct</code> object represents. | 
|
     * | 
|
     * @return an array of <code>Object</code> values, with each | 
|
     *         element being an attribute of the SQL structured type | 
|
     *         that this <code>SerialStruct</code> object represents | 
|
     * @throws SerialException if an error occurs | 
|
*/  | 
|
public Object[] getAttributes() throws SerialException {  | 
|
Object[] val = this.attribs;  | 
|
return (val == null) ? null : Arrays.copyOf(val, val.length);  | 
|
}  | 
|
    /** | 
|
     * Retrieves the attributes for the SQL structured type that | 
|
     * this <code>SerialStruct</code> represents as an array of | 
|
     * <code>Object</code> values, using the given type map for | 
|
     * custom mapping if appropriate. | 
|
     * | 
|
     * @param map a <code>java.util.Map</code> object in which | 
|
     *        each entry consists of 1) a <code>String</code> object | 
|
     *        giving the fully qualified name of a UDT and 2) the | 
|
     *        <code>Class</code> object for the <code>SQLData</code> implementation | 
|
     *        that defines how the UDT is to be mapped | 
|
     * @return an array of <code>Object</code> values, with each | 
|
     *         element being an attribute of the SQL structured | 
|
     *         type that this <code>SerialStruct</code> object | 
|
     *         represents | 
|
     * @throws SerialException if an error occurs | 
|
*/  | 
|
public Object[] getAttributes(Map<String,Class<?>> map)  | 
|
throws SerialException  | 
|
    { | 
|
Object[] val = this.attribs;  | 
|
return (val == null) ? null : Arrays.copyOf(val, val.length);  | 
|
}  | 
|
    /** | 
|
     * Maps attributes of an SQL structured type that are not | 
|
     * serialized to a serialized form, using the given type map | 
|
     * for custom mapping when appropriate.  The following types | 
|
     * in the Java programming language are mapped to their | 
|
     * serialized forms:  <code>Struct</code>, <code>SQLData</code>, | 
|
     * <code>Ref</code>, <code>Blob</code>, <code>Clob</code>, and | 
|
     * <code>Array</code>. | 
|
     * <P> | 
|
     * This method is called internally and is not used by an | 
|
     * application programmer. | 
|
     * | 
|
     * @param map a <code>java.util.Map</code> object in which | 
|
     *        each entry consists of 1) a <code>String</code> object | 
|
     *        giving the fully qualified name of a UDT and 2) the | 
|
     *        <code>Class</code> object for the <code>SQLData</code> implementation | 
|
     *        that defines how the UDT is to be mapped | 
|
     * @throws SerialException if an error occurs | 
|
*/  | 
|
private void mapToSerial(Map<String,Class<?>> map) throws SerialException {  | 
|
        try { | 
|
for (int i = 0; i < attribs.length; i++) {  | 
|
if (attribs[i] instanceof Struct) {  | 
|
attribs[i] = new SerialStruct((Struct)attribs[i], map);  | 
|
} else if (attribs[i] instanceof SQLData) {  | 
|
attribs[i] = new SerialStruct((SQLData)attribs[i], map);  | 
|
} else if (attribs[i] instanceof Blob) {  | 
|
attribs[i] = new SerialBlob((Blob)attribs[i]);  | 
|
} else if (attribs[i] instanceof Clob) {  | 
|
attribs[i] = new SerialClob((Clob)attribs[i]);  | 
|
} else if (attribs[i] instanceof Ref) {  | 
|
attribs[i] = new SerialRef((Ref)attribs[i]);  | 
|
} else if (attribs[i] instanceof java.sql.Array) {  | 
|
attribs[i] = new SerialArray((java.sql.Array)attribs[i], map);  | 
|
}  | 
|
}  | 
|
} catch (SQLException e) {  | 
|
throw new SerialException(e.getMessage());  | 
|
}  | 
|
return;  | 
|
}  | 
|
    /** | 
|
     * Compares this SerialStruct to the specified object.  The result is | 
|
     * {@code true} if and only if the argument is not {@code null} and is a | 
|
     * {@code SerialStruct} object whose attributes are identical to this | 
|
     * object's attributes | 
|
     * | 
|
     * @param  obj The object to compare this {@code SerialStruct} against | 
|
     * | 
|
     * @return {@code true} if the given object represents a {@code SerialStruct} | 
|
     *          equivalent to this SerialStruct, {@code false} otherwise | 
|
     * | 
|
*/  | 
|
public boolean equals(Object obj) {  | 
|
if (this == obj) {  | 
|
return true;  | 
|
}  | 
|
if (obj instanceof SerialStruct) {  | 
|
SerialStruct ss = (SerialStruct)obj;  | 
|
return SQLTypeName.equals(ss.SQLTypeName) &&  | 
|
Arrays.equals(attribs, ss.attribs);  | 
|
}  | 
|
return false;  | 
|
}  | 
|
    /** | 
|
     * Returns a hash code for this {@code SerialStruct}. The hash code for a | 
|
     * {@code SerialStruct} object is computed using the hash codes | 
|
     * of the attributes of the {@code SerialStruct} object and its | 
|
     * {@code SQLTypeName} | 
|
     * | 
|
     * @return  a hash code value for this object. | 
|
*/  | 
|
    public int hashCode() { | 
|
return ((31 + Arrays.hashCode(attribs)) * 31) * 31  | 
|
+ SQLTypeName.hashCode();  | 
|
}  | 
|
    /** | 
|
     * Returns a clone of this {@code SerialStruct}. The copy will contain a | 
|
     * reference to a clone of the underlying attribs array, not a reference | 
|
     * to the original underlying attribs array of this {@code SerialStruct} object. | 
|
     * | 
|
     * @return  a clone of this SerialStruct | 
|
*/  | 
|
public Object clone() {  | 
|
        try { | 
|
SerialStruct ss = (SerialStruct) super.clone();  | 
|
ss.attribs = Arrays.copyOf(attribs, attribs.length);  | 
|
return ss;  | 
|
} catch (CloneNotSupportedException ex) {  | 
|
            // this shouldn't happen, since we are Cloneable | 
|
throw new InternalError();  | 
|
}  | 
|
}  | 
|
    /** | 
|
     * readObject is called to restore the state of the {@code SerialStruct} from | 
|
     * a stream. | 
|
*/  | 
|
private void readObject(ObjectInputStream s)  | 
|
throws IOException, ClassNotFoundException {  | 
|
ObjectInputStream.GetField fields = s.readFields();  | 
|
Object[] tmp = (Object[])fields.get("attribs", null);  | 
|
attribs = tmp == null ? null : tmp.clone();  | 
|
SQLTypeName = (String)fields.get("SQLTypeName", null);  | 
|
}  | 
|
    /** | 
|
     * writeObject is called to save the state of the {@code SerialStruct} | 
|
     * to a stream. | 
|
*/  | 
|
private void writeObject(ObjectOutputStream s)  | 
|
throws IOException, ClassNotFoundException {  | 
|
ObjectOutputStream.PutField fields = s.putFields();  | 
|
fields.put("attribs", attribs);  | 
|
fields.put("SQLTypeName", SQLTypeName);  | 
|
s.writeFields();  | 
|
}  | 
|
    /** | 
|
     * The identifier that assists in the serialization of this | 
|
     * <code>SerialStruct</code> object. | 
|
*/  | 
|
static final long serialVersionUID = -8322445504027483372L;  | 
|
}  |