com.ibm.as400.access.AS400JDBCDriver Maven / Gradle / Ivy
Show all versions of jt400 Show documentation
///////////////////////////////////////////////////////////////////////////////
//
// JTOpen (IBM Toolbox for Java - OSS version)
//
// Filename: AS400JDBCDriver.java
//
// The source code contained herein is licensed under the IBM Public License
// Version 1.0, which has been approved by the Open Source Initiative.
// Copyright (C) 1997-2010 International Business Machines Corporation and
// others. All rights reserved.
//
///////////////////////////////////////////////////////////////////////////////
package com.ibm.as400.access;
import java.io.IOException;
import java.net.InetAddress;
import java.sql.Connection;
import java.sql.Driver;
import java.sql.DriverManager;
import java.sql.DriverPropertyInfo;
import java.sql.ResultSet;
import java.sql.SQLException;
/* ifdef JDBC40 */
import java.sql.SQLFeatureNotSupportedException;
import java.sql.Statement;
import java.util.logging.Logger;
/* endif */
import java.util.Properties;
import java.util.MissingResourceException;
import java.util.ResourceBundle;
/**
A JDBC 3.0/4.0/4.2/4.3 driver that accesses DB2 for IBM i databases.
To use this driver, the application or caller must register
the driver with the JDBC DriverManager prior to JDBC 4.0.
This class also registers
itself automatically when it is loaded.
When using the JDBC 4.0 or later versions, the driver is
automatically registered.
After registering the driver, applications make connection
requests to the DriverManager, which dispatches them to the
appropriate driver. This driver accepts connection requests
for databases specified by the URLs that match the following syntax:
jdbc:as400://system-name/default-schema;properties
The driver uses the specified system name to connect
to a corresponding IBM i system.
If an IPV6 address is used as the system name, it must be enclosed within braces,
i.e. [fd13:ac12:18:17::16].
If a system name is not specified, then the user will be prompted.
The default SQL schema is optional and the driver uses it to resolve
unqualified names in SQL statements. If no default SQL schema is set, then
the driver resolves unqualified names based on the naming convention
for the connection. If SQL naming is being used, and no default SQL schema
is set, then the driver resolves unqualified names using the schema with
the same name as the user. If system naming is being used, and no
default SQL schema is set, then the driver resolves unqualified names using
the server job's library list. See
JDBC properties
for more details on how to set the naming convention
and library list.
Several properties can optionally be set within the URL. They are
separated by semicolons and are in the form:
name1=value1;name2=value2;...
See
JDBC properties
for a complete list of properties supported by this driver.
The following example URL specifies a connection to the
database on system mysystem.helloworld.com with
mylibrary as the default SQL schema. The connection will
use the system naming convention and return full error messages:
jdbc:as400://mysystem.helloworld.com/mylibrary;naming=system;errors=full
**/
//
// Implementation note:
//
// 1. A goal stated in the JDBC specification is to keep
// the Driver class as small and standalone as possible,
// so that it can be quickly loaded when choosing a
// driver for a particular database.
//
// 2. It was proposed that we also accept URLs with the
// "db2" subprotocol. This would make us consistent with
// other IBM drivers. In addition, it would also allow
// developers to hardcode URLs in programs and they would
// run as-is with both this driver and the "native" driver.
//
// We realized, though, that if running on a client with
// both this driver and another DB2 client for that platform,
// how do the drivers differentiate themselves? Therefore
// we are chosing NOT to recognized the "db2" subprotocol.
// Instead, suggest to developers to externalize the URL
// to users, rather than hardcoding it.
//
//
// Note: Change log is now in Copyright.java
//
public class AS400JDBCDriver
implements java.sql.Driver
{
// Constants.
static final int MAJOR_VERSION_ = Copyright.MAJOR_VERSION;
static final int MINOR_VERSION_ = Copyright.MINOR_VERSION;
static final String DATABASE_PRODUCT_NAME_ = "DB2 UDB for AS/400"; // @D0A
static final String DRIVER_NAME_ = "AS/400 Toolbox for Java JDBC Driver"; // @D0C @C5C @C6C
static final String DRIVER_LEVEL_ = Copyright.DRIVER_LEVEL;
/* ifdef JDBC40 */
public static final int JDBC_MAJOR_VERSION_ = 4; // JDBC spec version: 4.0
/* endif */
/* ifndef JDBC40
public static final int JDBC_MAJOR_VERSION_ = 3; // JDBC spec version: 3.0
endif */
/* ifdef JAVA9
public static final int JDBC_MINOR_VERSION_ = 3;
endif JAVA9 */
/* ifndef JAVA9 */
/* ifdef JDBC42
public static final int JDBC_MINOR_VERSION_ = 2;
endif */
/* ifndef JDBC42 */
public static final int JDBC_MINOR_VERSION_ = 0;
/* endif */
/* endif JAVA9 */
// This string "9999:9999" is returned when resource
// bundle errors occur. No significance to this string,
// except that Client Access used to use it. It would
// probably be more helpful to return some other string.
//
private static final String MRI_NOT_FOUND_ = "9999:9999";
// Private data.
// Toolbox resources needed in proxy jar file. @A1C
private static ResourceBundle resources_;
// Toolbox resources NOT needed in proxy jar file. @A1A
private static ResourceBundle resources2_;
private static final String CLASSNAME = "com.ibm.as400.access.AS400JDBCDriver";
private static Driver nativeDriver = null;
/**
Static initializer. Registers the JDBC driver with the JDBC
driver manager and loads the appropriate resource bundle for
the current locale.
**/
static {
try
{
// Log where the toolbox is loaded from @B1A
if (Trace.traceOn_) Trace.logLoadPath(CLASSNAME, Trace.JDBC);
DriverManager.registerDriver (new AS400JDBCDriver ());
resources_ = ResourceBundle.getBundle ("com.ibm.as400.access.JDMRI");
resources2_ = ResourceBundle.getBundle ("com.ibm.as400.access.JDMRI2");
// Note: When using the proxy jar file, we do not expect to find JDMRI2.
}
catch (MissingResourceException e)
{
// Catch the exception. This is because exceptions
// thrown from static initializers are hard to debug.
// Instead, we will handle the error when the
// driver needs to get at particular methods.
// See getResource().
}
catch (SQLException e)
{
// Ignore.
}
}
/**
Indicates if the driver understands how to connect
to the database named by the URL.
@param url The URL for the database.
@return true if the driver understands how
to connect to the database; false
otherwise.
@exception SQLException If an error occurs.
**/
public boolean acceptsURL (String url)
throws SQLException
{
JDDataSourceURL dataSourceUrl = new JDDataSourceURL (url);
return dataSourceUrl.isValid ();
}
/**
Connects to the database named by the specified URL using the
specified userid and password.
@param url The URL for the database.
@param userid The userid for the connection
@param password The password for the connection. The caller should clear the
password from the array after the method returns.
@return The connection to the database or null if
the driver does not understand how to connect
to the database.
@exception SQLException If the driver is unable to make the connection.
**/
public java.sql.Connection connect (String url, String userid, char[] password)
throws SQLException
{
return connect(url, userid, password, (char[])null);
}
/**
Connects to the database named by the specified URL using the
specified userid, password, and additional authentication factor.
@param url The URL for the database.
@param userid The userid for the connection
@param password The password for the connection. The caller should clear the
password from the array after the method returns.
@param additionalAuthenticationFactor The additional authentication factor, or null
if not providing one
@return The connection to the database or null if
the driver does not understand how to connect
to the database.
@exception SQLException If the driver is unable to make the connection.
**/
public java.sql.Connection connect (String url, String userid, char[] password, char[] additionalAuthenticationFactor)
throws SQLException
{
Properties properties = new Properties();
properties.put("user", userid);
return connect(url, properties, password, additionalAuthenticationFactor);
}
/**
Connects to the database named by the specified URL.
There are many optional properties that can be specified.
Properties can be specified either as part of the URL or in
a java.util.Properties object. See
JDBC properties
for a complete list of properties
supported by this driver.
@param url The URL for the database.
@param info The connection properties.
@return The connection to the database or null if
the driver does not understand how to connect
to the database.
@exception SQLException If the driver is unable to make the connection.
**/
public java.sql.Connection connect (String url,
Properties info)
throws SQLException
{
return connect(url, info, null);
}
/**
Connects to the database named by the specified URL.
There are many optional properties that can be specified.
Properties can be specified either as part of the URL or in
a java.util.Properties object. See
JDBC properties
for a complete list of properties
supported by this driver.
@param url The URL for the database.
@param info The connection properties.
@param password The password as a char array. The caller should clear the
char array after returning.
@return The connection to the database or null if
the driver does not understand how to connect
to the database.
@exception SQLException If the driver is unable to make the connection.
**/
public java.sql.Connection connect (String url,
Properties info,
char[] password)
throws SQLException
{
return connect(url, info, password, (char[])null);
}
/**
Connects to the database named by the specified URL.
There are many optional properties that can be specified.
Properties can be specified either as part of the URL or in
a java.util.Properties object. See
JDBC properties
for a complete list of properties
supported by this driver.
@param url The URL for the database.
@param info The connection properties.
@param password The password as a char array. The caller should clear the
char array after returning.
@param additionalAuthenticationFactor The additional authentication factor (or null
if not providing one)
@return The connection to the database or null if
the driver does not understand how to connect
to the database.
@exception SQLException If the driver is unable to make the connection.
**/
public java.sql.Connection connect (String url,
Properties info,
char[] password,
char[] additionalAuthenticationFactor)
throws SQLException
{
// Check first thing to see if the trace property is
// turned on. This way we can trace everything, including
// the important stuff like loading the properties.
JDDataSourceURL dataSourceUrl = new JDDataSourceURL (url);
Properties urlProperties = dataSourceUrl.getProperties ();
// If trace property was set to true, turn on tracing. If trace property was set to false,
// turn off tracing. If trace property was not set, do not change.
if (JDProperties.isTraceSet (urlProperties, info) == JDProperties.TRACE_SET_ON)
{ //@B5C
if (! JDTrace.isTraceOn ())
JDTrace.setTraceOn (true);
}
else if (JDProperties.isTraceSet (urlProperties, info) == JDProperties.TRACE_SET_OFF) //@B5A
{
//@B5A
if (JDTrace.isTraceOn ()) //@B5A
JDTrace.setTraceOn (false); //@B5A
} //@B5A
//@B4D Deleted lines because trace should not be set off just because property
//@B4D not specified.
//@B4D else
//@B4D JDTrace.setTraceOn (false);
// If toolbox trace is set to datastream. Turn on datastream tracing.
if (JDProperties.isToolboxTraceSet (urlProperties, info) == JDProperties.TRACE_TOOLBOX_DATASTREAM)
{
//@K1A
if (! Trace.isTraceOn())
{
Trace.setTraceOn(true);
}
Trace.setTraceDatastreamOn(true);
}
// If toolbox trace is set to diagnostic. Turn on diagnostic tracing.
else if (JDProperties.isToolboxTraceSet (urlProperties, info) == JDProperties.TRACE_TOOLBOX_DIAGNOSTIC)
{
//@K1A
if (! Trace.isTraceOn())
{
Trace.setTraceOn(true);
}
Trace.setTraceDiagnosticOn(true);
}
// If toolbox trace is set to error. Turn on error tracing.
else if (JDProperties.isToolboxTraceSet (urlProperties, info) == JDProperties.TRACE_TOOLBOX_ERROR)
{
//@K1A
if (! Trace.isTraceOn())
{
Trace.setTraceOn(true);
}
Trace.setTraceErrorOn(true);
}
// If toolbox trace is set to information. Turn on information tracing.
else if (JDProperties.isToolboxTraceSet (urlProperties, info) == JDProperties.TRACE_TOOLBOX_INFORMATION)
{
//@K1A
if (! Trace.isTraceOn())
{
Trace.setTraceOn(true);
}
Trace.setTraceInformationOn(true);
}
// If toolbox trace is set to warning. Turn on warning tracing.
else if (JDProperties.isToolboxTraceSet (urlProperties, info) == JDProperties.TRACE_TOOLBOX_WARNING)
{
//@K1A
if (! Trace.isTraceOn())
{
Trace.setTraceOn(true);
}
Trace.setTraceWarningOn(true);
}
// If toolbox trace is set to conversion. Turn on conversion tracing.
else if (JDProperties.isToolboxTraceSet (urlProperties, info) == JDProperties.TRACE_TOOLBOX_CONVERSION)
{
//@K1A
if (! Trace.isTraceOn())
{
Trace.setTraceOn(true);
}
Trace.setTraceConversionOn(true);
}
// If toolbox trace is set to proxy. Turn on proxy tracing.
else if (JDProperties.isToolboxTraceSet (urlProperties, info) == JDProperties.TRACE_TOOLBOX_PROXY)
{
//@K1A
if (! Trace.isTraceOn())
{
Trace.setTraceOn(true);
}
Trace.setTraceProxyOn(true);
}
// If toolbox trace is set to pcml. Turn on pcml tracing.
else if (JDProperties.isToolboxTraceSet (urlProperties, info) == JDProperties.TRACE_TOOLBOX_PCML)
{
//@K1A
if (! Trace.isTraceOn())
{
Trace.setTraceOn(true);
}
Trace.setTracePCMLOn(true);
}
// If toolbox trace is set to jdbc. Turn on jdbc tracing.
else if (JDProperties.isToolboxTraceSet (urlProperties, info) == JDProperties.TRACE_TOOLBOX_JDBC)
{
//@K1A
if (! Trace.isTraceOn())
{
Trace.setTraceOn(true);
}
Trace.setTraceJDBCOn(true);
}
// If toolbox trace is set to all. Turn on tracing for all categories.
else if (JDProperties.isToolboxTraceSet (urlProperties, info) == JDProperties.TRACE_TOOLBOX_ALL)
{
//@K1A
if (! Trace.isTraceOn())
{
Trace.setTraceOn(true);
}
Trace.setTraceAllOn(true);
}
// If toolbox trace is set to thread. Turn on thread tracing.
else if (JDProperties.isToolboxTraceSet (urlProperties, info) == JDProperties.TRACE_TOOLBOX_THREAD)
{
//@K1A
if (! Trace.isTraceOn())
{
Trace.setTraceOn(true);
}
Trace.setTraceThreadOn(true);
}
// If toolbox trace is set to none. Turn off tracing.
else if (JDProperties.isToolboxTraceSet (urlProperties, info) == JDProperties.TRACE_TOOLBOX_NONE)
{
//@K1A
if (Trace.isTraceOn())
{
Trace.setTraceOn(false);
}
}
if (JDTrace.isTraceOn()) {
String traceUrl = url;
int passwordIndex = url.indexOf("password=");
if (passwordIndex >= 0) {
int semicolonIndex = url.indexOf(";",passwordIndex);
if (semicolonIndex < 0) {
traceUrl = url.substring(0,passwordIndex)+"password=******";
} else {
traceUrl = url.substring(0,passwordIndex)+"password=******"+url.substring(semicolonIndex);
}
}
JDTrace.logInformation (this,"connect called with URL: "+traceUrl);
}
JDProperties jdProperties = new JDProperties (urlProperties, info, password, additionalAuthenticationFactor);
// Initialize the connection if the URL is valid.
Connection connection = null; //@A0C
if (dataSourceUrl.isValid ())
connection = initializeConnection (dataSourceUrl, jdProperties); //@A0C
return connection;
}
//@B5A
/**
Connects to the database on the specified system.
Note: Since this method is not defined in the JDBC Driver interface,
you typically need to create a Driver object in order
to call this method:
AS400JDBCDriver d = new AS400JDBCDriver();
AS400 o = new AS400(myAS400, myUserId, myPwd);
Connection c = d.connect (o);
@param system The IBM i system to connect.
@return The connection to the database or null if
the driver does not understand how to connect
to the database.
@exception SQLException If the driver is unable to make the connection.
**/
public java.sql.Connection connect (AS400 system)
throws SQLException
{
if (system == null)
throw new NullPointerException("system");
return initializeConnection(AS400.newInstance(system.isSecure(), system));
// Initialize the connection.
//@B7D Connection connection = null;
//@B7D connection = initializeConnection (o);
//@B7D return connection;
}
//@KKB
/**
Connects to the database on the specified system.
Note: Since this method is not defined in the JDBC Driver interface,
you typically need to create a Driver object in order
to call this method:
AS400JDBCDriver d = new AS400JDBCDriver();
AS400 o = new AS400(myAS400, myUserId, myPwd);
Connection c = d.connect (o, false);
@param system The IBM i system to connect.
@param clone True if the AS400 object should be cloned, false otherwise.
@return The connection to the database or null if
the driver does not understand how to connect
to the database.
@exception SQLException If the driver is unable to make the connection.
**/
public java.sql.Connection connect (AS400 system, boolean clone)
throws SQLException
{
if (system == null)
throw new NullPointerException("system");
if(!clone) //Do not clone the AS400 object, use the one passed in
return initializeConnection(system);
else //clone the AS400 object
return connect(system);
}
//@D4A
/**
Connects to the database on the specified system.
Note: Since this method is not defined in the JDBC Driver interface,
you typically need to create a Driver object in order
to call this method:
AS400JDBCDriver d = new AS400JDBCDriver();
AS400 o = new AS400(myAS400, myUserId, myPwd);
String mySchema = "defaultSchema";
Properties prop = new Properties();
Connection c = d.connect (o, prop, mySchema, false);
@param system The IBM i system to connect.
@param info The connection properties.
@param schema The default SQL schema or null meaning no default SQL schema specified.
@param clone True if the AS400 object should be cloned, false otherwise.
@return The connection to the database or null if
the driver does not understand how to connect
to the database.
@exception SQLException If the driver is unable to make the connection.
**/
public java.sql.Connection connect (AS400 system, Properties info, String schema, boolean clone)
throws SQLException
{
if (system == null)
throw new NullPointerException("system");
if (info == null)
throw new NullPointerException("properties");
//@PDD not needed, just pass in null to isXTraceSet(null, info) below
//Properties urlProperties = new Properties();
// Check first thing to see if the trace property is
// turned on. This way we can trace everything, including
// the important stuff like loading the properties.
// If trace property was set to true, turn on tracing. If trace property was set to false,
// turn off tracing. If trace property was not set, do not change.
if (JDProperties.isTraceSet (null, info) == JDProperties.TRACE_SET_ON)
{
if (! JDTrace.isTraceOn ())
JDTrace.setTraceOn (true);
}
else if (JDProperties.isTraceSet (null, info) == JDProperties.TRACE_SET_OFF)
{
if (JDTrace.isTraceOn ())
JDTrace.setTraceOn (false);
}
// If toolbox trace is set to datastream. Turn on datastream tracing.
if (JDProperties.isToolboxTraceSet (null, info) == JDProperties.TRACE_TOOLBOX_DATASTREAM)
{
if (! Trace.isTraceOn())
{
Trace.setTraceOn(true);
}
Trace.setTraceDatastreamOn(true);
}
// If toolbox trace is set to diagnostic. Turn on diagnostic tracing.
else if (JDProperties.isToolboxTraceSet (null, info) == JDProperties.TRACE_TOOLBOX_DIAGNOSTIC)
{
if (! Trace.isTraceOn())
{
Trace.setTraceOn(true);
}
Trace.setTraceDiagnosticOn(true);
}
// If toolbox trace is set to error. Turn on error tracing.
else if (JDProperties.isToolboxTraceSet (null, info) == JDProperties.TRACE_TOOLBOX_ERROR)
{
if (! Trace.isTraceOn())
{
Trace.setTraceOn(true);
}
Trace.setTraceErrorOn(true);
}
// If toolbox trace is set to information. Turn on information tracing.
else if (JDProperties.isToolboxTraceSet (null, info) == JDProperties.TRACE_TOOLBOX_INFORMATION)
{
if (! Trace.isTraceOn())
{
Trace.setTraceOn(true);
}
Trace.setTraceInformationOn(true);
}
// If toolbox trace is set to warning. Turn on warning tracing.
else if (JDProperties.isToolboxTraceSet (null, info) == JDProperties.TRACE_TOOLBOX_WARNING)
{
if (! Trace.isTraceOn())
{
Trace.setTraceOn(true);
}
Trace.setTraceWarningOn(true);
}
// If toolbox trace is set to conversion. Turn on conversion tracing.
else if (JDProperties.isToolboxTraceSet (null, info) == JDProperties.TRACE_TOOLBOX_CONVERSION)
{
if (! Trace.isTraceOn())
{
Trace.setTraceOn(true);
}
Trace.setTraceConversionOn(true);
}
// If toolbox trace is set to proxy. Turn on proxy tracing.
else if (JDProperties.isToolboxTraceSet (null, info) == JDProperties.TRACE_TOOLBOX_PROXY)
{
if (! Trace.isTraceOn())
{
Trace.setTraceOn(true);
}
Trace.setTraceProxyOn(true);
}
// If toolbox trace is set to pcml. Turn on pcml tracing.
else if (JDProperties.isToolboxTraceSet (null, info) == JDProperties.TRACE_TOOLBOX_PCML)
{
if (! Trace.isTraceOn())
{
Trace.setTraceOn(true);
}
Trace.setTracePCMLOn(true);
}
// If toolbox trace is set to jdbc. Turn on jdbc tracing.
else if (JDProperties.isToolboxTraceSet (null, info) == JDProperties.TRACE_TOOLBOX_JDBC)
{
if (! Trace.isTraceOn())
{
Trace.setTraceOn(true);
}
Trace.setTraceJDBCOn(true);
}
// If toolbox trace is set to all. Turn on tracing for all categories.
else if (JDProperties.isToolboxTraceSet (null, info) == JDProperties.TRACE_TOOLBOX_ALL)
{
if (! Trace.isTraceOn())
{
Trace.setTraceOn(true);
}
Trace.setTraceAllOn(true);
}
// If toolbox trace is set to thread. Turn on thread tracing.
else if (JDProperties.isToolboxTraceSet (null, info) == JDProperties.TRACE_TOOLBOX_THREAD)
{
if (! Trace.isTraceOn())
{
Trace.setTraceOn(true);
}
Trace.setTraceThreadOn(true);
}
// If toolbox trace is set to none. Turn off tracing.
else if (JDProperties.isToolboxTraceSet (null, info) == JDProperties.TRACE_TOOLBOX_NONE)
{
if (Trace.isTraceOn())
{
Trace.setTraceOn(false);
}
}
if(!clone) //Do not clone the AS400 object, use the one passed in
return initializeConnection(schema, info, system);
else //clone the AS400 object
return initializeConnection(schema, info, AS400.newInstance(system.isSecure(), system));
}
//@B5A
/**
Connects to the database on the specified system.
There are many optional properties that can be specified.
Properties can be specified in
a java.util.Properties object. See
JDBC properties
for a complete list of properties
supported by this driver.
Note: Since this method is not defined in the JDBC Driver interface,
you typically need to create a Driver object in order
to call this method:
AS400JDBCDriver d = new AS400JDBCDriver();
String mySchema = "defaultSchema";
Properties p = new Properties();
AS400 o = new AS400(myAS400, myUserId, myPwd);
Connection c = d.connect (o, p, mySchema);
@param system The IBM i system to connect.
@param info The connection properties.
@param schema The default SQL schema or null meaning no default SQL schema specified.
@return The connection to the database or null if
the driver does not understand how to connect
to the database.
@exception SQLException If the driver is unable to make the connection.
**/
public java.sql.Connection connect (AS400 system, Properties info, String schema)
throws SQLException
{
if (system == null)
throw new NullPointerException("system");
if (info == null)
throw new NullPointerException("properties");
//@B7D AS400 o = new AS400(system);
//@PDD not needed, just pass in null to isXTraceSet(null, info) below
//Properties urlProperties = new Properties();
// Check first thing to see if the trace property is
// turned on. This way we can trace everything, including
// the important stuff like loading the properties.
// If trace property was set to true, turn on tracing. If trace property was set to false,
// turn off tracing. If trace property was not set, do not change.
if (JDProperties.isTraceSet (null, info) == JDProperties.TRACE_SET_ON)
{
if (! JDTrace.isTraceOn ())
JDTrace.setTraceOn (true);
}
else if (JDProperties.isTraceSet (null, info) == JDProperties.TRACE_SET_OFF)
{
if (JDTrace.isTraceOn ())
JDTrace.setTraceOn (false);
}
// If toolbox trace is set to datastream. Turn on datastream tracing.
if (JDProperties.isToolboxTraceSet (null, info) == JDProperties.TRACE_TOOLBOX_DATASTREAM)
{
//@K1A
if (! Trace.isTraceOn())
{
Trace.setTraceOn(true);
}
Trace.setTraceDatastreamOn(true);
}
// If toolbox trace is set to diagnostic. Turn on diagnostic tracing.
else if (JDProperties.isToolboxTraceSet (null, info) == JDProperties.TRACE_TOOLBOX_DIAGNOSTIC)
{
//@K1A
if (! Trace.isTraceOn())
{
Trace.setTraceOn(true);
}
Trace.setTraceDiagnosticOn(true);
}
// If toolbox trace is set to error. Turn on error tracing.
else if (JDProperties.isToolboxTraceSet (null, info) == JDProperties.TRACE_TOOLBOX_ERROR)
{
//@K1A
if (! Trace.isTraceOn())
{
Trace.setTraceOn(true);
}
Trace.setTraceErrorOn(true);
}
// If toolbox trace is set to information. Turn on information tracing.
else if (JDProperties.isToolboxTraceSet (null, info) == JDProperties.TRACE_TOOLBOX_INFORMATION)
{
//@K1A
if (! Trace.isTraceOn())
{
Trace.setTraceOn(true);
}
Trace.setTraceInformationOn(true);
}
// If toolbox trace is set to warning. Turn on warning tracing.
else if (JDProperties.isToolboxTraceSet (null, info) == JDProperties.TRACE_TOOLBOX_WARNING)
{
//@K1A
if (! Trace.isTraceOn())
{
Trace.setTraceOn(true);
}
Trace.setTraceWarningOn(true);
}
// If toolbox trace is set to conversion. Turn on conversion tracing.
else if (JDProperties.isToolboxTraceSet (null, info) == JDProperties.TRACE_TOOLBOX_CONVERSION)
{
//@K1A
if (! Trace.isTraceOn())
{
Trace.setTraceOn(true);
}
Trace.setTraceConversionOn(true);
}
// If toolbox trace is set to proxy. Turn on proxy tracing.
else if (JDProperties.isToolboxTraceSet (null, info) == JDProperties.TRACE_TOOLBOX_PROXY)
{
//@K1A
if (! Trace.isTraceOn())
{
Trace.setTraceOn(true);
}
Trace.setTraceProxyOn(true);
}
// If toolbox trace is set to pcml. Turn on pcml tracing.
else if (JDProperties.isToolboxTraceSet (null, info) == JDProperties.TRACE_TOOLBOX_PCML)
{
//@K1A
if (! Trace.isTraceOn())
{
Trace.setTraceOn(true);
}
Trace.setTracePCMLOn(true);
}
// If toolbox trace is set to jdbc. Turn on jdbc tracing.
else if (JDProperties.isToolboxTraceSet (null, info) == JDProperties.TRACE_TOOLBOX_JDBC)
{
//@K1A
if (! Trace.isTraceOn())
{
Trace.setTraceOn(true);
}
Trace.setTraceJDBCOn(true);
}
// If toolbox trace is set to all. Turn on tracing for all categories.
else if (JDProperties.isToolboxTraceSet (null, info) == JDProperties.TRACE_TOOLBOX_ALL)
{
//@K1A
if (! Trace.isTraceOn())
{
Trace.setTraceOn(true);
}
Trace.setTraceAllOn(true);
}
// If toolbox trace is set to thread. Turn on thread tracing.
else if (JDProperties.isToolboxTraceSet (null, info) == JDProperties.TRACE_TOOLBOX_THREAD)
{
//@K1A
if (! Trace.isTraceOn())
{
Trace.setTraceOn(true);
}
Trace.setTraceThreadOn(true);
}
// If toolbox trace is set to none. Turn off tracing.
else if (JDProperties.isToolboxTraceSet (null, info) == JDProperties.TRACE_TOOLBOX_NONE)
{
//@K1A
if (Trace.isTraceOn())
{
Trace.setTraceOn(false);
}
}
//@PDD not used JDProperties jdProperties = new JDProperties (null, info);
return initializeConnection(schema, info, AS400.newInstance(system.isSecure(), system));
// Initialize the connection if the URL is valid.
//@B7D Connection connection = null;
//@B7D connection = initializeConnection (schema, info, o);
//@B7D return connection;
}
/**
Returns the driver's major version number.
@return The major version number.
**/
public int getMajorVersion ()
{
return MAJOR_VERSION_;
}
/**
Returns the driver's minor version number.
@return The minor version number.
**/
public int getMinorVersion ()
{
return MINOR_VERSION_;
}
/**
Returns an array of DriverPropertyInfo objects that
describe the properties that are supported by this
driver.
@param url The URL for the database.
@param info The connection properties.
@return The descriptions of all possible properties or null if
the driver does not understand how to connect to the
database.
@exception SQLException If an error occurs.
**/
public DriverPropertyInfo[] getPropertyInfo (String url,
Properties info)
throws SQLException
{
JDDataSourceURL dataSourceUrl = new JDDataSourceURL (url);
DriverPropertyInfo[] dpi = null;
if (dataSourceUrl.isValid ())
{
JDProperties properties = new JDProperties (dataSourceUrl.getProperties(), info, null, null);
dpi = properties.getInfo ();
}
return dpi;
}
/**
Returns a resource from the resource bundle.
@param key The resource key.
@param replacementVariables -- replacement variables for the message
@return The resource String.
**/
static String getResource (String key, String[] replacementVariables)
{
// If the resource bundle or resource is not found,
// do not thrown an exception. Instead, return a
// default string. This is because some JVMs will
// not recover quite right from such errors, and
// claim a security exception (e.g. Netscape starts
// looking in the client class path, which is
// not allowed.)
//
String resource;
if (resources_ == null)
resource = MRI_NOT_FOUND_;
else
{
try
{
resource = resources_.getString (key);
}
catch (MissingResourceException e)
{
if (resources2_ == null) //@A1A
resource = MRI_NOT_FOUND_; //@A1A
else
{ //@A1A
try
{ //@A1A
resource = resources2_.getString (key); //@A1A
} //@A1A
catch (MissingResourceException e1)
{ //@A1A
JDTrace.logInformation (AS400JDBCDriver.class,
"Missing resource [" + key + "]"); //@A1A
resource = MRI_NOT_FOUND_;
}
}
}
}
if (replacementVariables != null) {
resource = substitute(resource, replacementVariables);
}
return resource;
}
// Replaces substitution variables in a string.
// @param text The text string, with substitution variables (e.g. "Error &0 in table &1.")
// @param values The replacement values.
// @return The text string with all substitution variables replaced.
static String substitute (String text, Object[] values)
{
String result = text;
if (values != null) {
for (int i = 0; i < values.length; ++i) {
String variable = "&" + i;
int j = result.indexOf (variable);
if (j >= 0) {
StringBuffer buffer = new StringBuffer();
buffer.append(result.substring(0, j));
buffer.append(values[i].toString ());
buffer.append(result.substring(j + variable.length ()));
result = buffer.toString ();
}
}
}
return result;
}
//@B3A - This logic was formerly in the initializeConnection() method.
static AS400 initializeAS400(JDDataSourceURL dataSourceUrl,
JDProperties jdProperties)
throws SQLException //@pw1
{
// We must handle the different combinations of input
// user names and passwords.
String serverName = dataSourceUrl.getServerName();
String userName = jdProperties.getString (JDProperties.USER);
char[] clearPassword = jdProperties.getClearPassword();
char[] additionalAuthenticationFactor = jdProperties.getAdditionalAuthenticationFactor();
String prompt = jdProperties.getString (JDProperties.PROMPT); // @B8C
boolean secure = jdProperties.getBoolean (JDProperties.SECURE);
boolean useThreads = jdProperties.getBoolean(JDProperties.THREAD_USED);
// Updated 2023 to not pass old Properties information.
// Everything should be in the JDProperties object
// The JDProperties object was updated to also allow the use of null values.
//
//@pw1 Decided to leave connections via AS400() as-is and just implement to mimic Native JDBC
//@pw1 info contains args from DriverMangager.getConnection(args)
//@pw1 jdProperties does not represent null values. Both null and "" have a value of "".
//@pw1 if info contains id/pass of "" then they must not be "" in jdProperties (allowing jdProperties to override)
//@pw1 throw exception if info id/pass == "" and change info id/pass to "" if they are null
//@pw3 Add way to get old behavior allowing "" (!but also need to allow new behavior of allowing null is/passwd so customers can slowly migrate)
//check if "".
String secureCurrentUser = SystemProperties.getProperty (SystemProperties.JDBC_SECURE_CURRENT_USER); //@pw3
boolean isSecureCurrentUser = true; //@pw3
//if system property or jdbc property is set to false then secure current user code is not used
//null value for system property means not specified...so true by default
if(((secureCurrentUser != null) && (Boolean.valueOf(secureCurrentUser).booleanValue() == false)) || !jdProperties.getBoolean(JDProperties.SECURE_CURRENT_USER)) //@pw3
isSecureCurrentUser = false; //@pw3
boolean forcePrompt = false; //@prompt
if ("".equals(userName)) //@pw1 //@pw2
{ //@pw1
if(isSecureCurrentUser)//@pw3
{ //@pw3
if (JDTrace.isTraceOn()) //jdbc category trace //@pw1
JDTrace.logInformation (AS400JDBCDriver.class, "Userid/password cannot be \"\" or *CURRENT due to security constraints. Use null instead"); //@pw1
//JDError.throwSQLException(JDError.EXC_CONNECTION_REJECTED); //@pw1 //@prompt
forcePrompt = true; //@prompt
} //@pw3
} //@pw1
if (clearPassword != null && clearPassword.length==0) //@pw1 //@pw2
{ //@pw1
if(isSecureCurrentUser)//@pw3
{ //@pw3
if (JDTrace.isTraceOn()) //jdbc category trace //@pw1
JDTrace.logInformation (AS400JDBCDriver.class, "Userid/password cannot be \"\" or *CURRENT due to security constraints. Use null instead"); //@pw1
//JDError.throwSQLException(JDError.EXC_CONNECTION_REJECTED); //@pw1 //@prompt
forcePrompt = true; //@prompt
} //@pw3
} //@pw1
if(userName != null) //@pw1
{ //@pw1
//check for *current //@pw1
if (userName.compareToIgnoreCase("*CURRENT") == 0) //@pw1
{ //@pw1
if(isSecureCurrentUser)//@pw3
{ //@pw3
if (JDTrace.isTraceOn()) //jdbc category trace //@pw1
JDTrace.logInformation (AS400JDBCDriver.class, "Userid/password cannot be \"\" or *CURRENT due to security constraints. Use null instead"); //@pw1
//JDError.throwSQLException(JDError.EXC_CONNECTION_REJECTED); //@pw1
forcePrompt = true; //@prompt
} //@pw3
} //@pw1
} //@pw1
if(clearPassword!= null) //@pw1
{ //@pw1
/* check for *CURRENT. Be sure to check the length */
if (clearPassword.length == 8 &&
clearPassword[0] == '*' &&
(clearPassword[1] == 'C' || clearPassword[1] == 'c') &&
(clearPassword[2] == 'U' || clearPassword[2] == 'u') &&
(clearPassword[3] == 'R' || clearPassword[3] == 'r') &&
(clearPassword[4] == 'R' || clearPassword[4] == 'r') &&
(clearPassword[5] == 'E' || clearPassword[5] == 'e') &&
(clearPassword[6] == 'N' || clearPassword[6] == 'n') &&
(clearPassword[7] == 'T' || clearPassword[7] == 't') ) //@pw1
{ //@pw1
if(isSecureCurrentUser)//@pw3
{ //@pw3
if (JDTrace.isTraceOn()) //jdbc category trace //@pw1
JDTrace.logInformation (AS400JDBCDriver.class, "Userid/password cannot be \"\" or *CURRENT due to security constraints. Use null instead"); //@pw1
//JDError.throwSQLException(JDError.EXC_CONNECTION_REJECTED); //@pw1
forcePrompt = true; //@prompt
} //@pw3
} //@pw1
} //@pw1
// Create the AS400 object, so we can create a Connection via loadImpl2.
AS400 as400 = null;
try
{
if (serverName.length() == 0)
as400 = AS400.newInstance(secure);
else if ((userName == null) || (userName.length() == 0))
as400 = AS400.newInstance(secure, serverName);
else if (clearPassword == null)
as400 = AS400.newInstance(secure, serverName, userName);
else
as400 = AS400.newInstance(secure, serverName, userName, clearPassword, additionalAuthenticationFactor);
}
catch (AS400SecurityException e)
{
JDError.throwSQLException (as400, JDError.EXC_CONNECTION_REJECTED, e);
}
catch (IOException e)
{
JDError.throwSQLException (as400, JDError.EXC_CONNECTION_UNABLE, e);
}
if (clearPassword != null) {
CredentialVault.clearArray(clearPassword);
}
// Determine when the signon GUI can be presented..
try
{
if (!prompt.equals(JDProperties.NOT_SPECIFIED)) // @B8A
as400.setGuiAvailable(jdProperties.getBoolean(JDProperties.PROMPT)); // @B8C
}
catch (java.beans.PropertyVetoException e)
{
// This will never happen, as there are no listeners.
}
//Determine if threads should be used in communication with the host servers
try{
if(!useThreads)
as400.setThreadUsed(useThreads);
}
catch(java.beans.PropertyVetoException e){
}
if(forcePrompt) //@prompt
as400.forcePrompt(); //@prompt
return as400;
}
//@A0A - This logic was formerly in the AS400JDBCConnection ctor and open() method.
private Connection initializeConnection (JDDataSourceURL dataSourceUrl,
JDProperties jdProperties)
throws SQLException
{
//@B7D Connection connection = null;
AS400 as400 = null;
boolean proxyServerWasSpecifiedInUrl = false;
boolean proxyServerWasSpecifiedInProperties = false;
boolean proxyServerWasSpecified = false;
//@C4A Check for native driver only if driver property is not set to Toolbox
String driverImplementation = jdProperties.getString(JDProperties.DRIVER); //@C4M
if (!driverImplementation.equals(JDProperties.DRIVER_TOOLBOX)) //@C4A
{ //@C4A
// @B2A
// Determine whether the native driver is available.
if (nativeDriver == null) {
try {
nativeDriver = (Driver) Class.forName(
"com.ibm.db2.jdbc.app.DB2Driver").newInstance();
if (JDTrace.isTraceOn()) // @C2A
JDTrace
.logInformation(this,
"Native IBM Developer Kit for Java JDBC driver implementation was loaded"); // @C2A
} catch (Throwable e) {
nativeDriver = null;
}
}
// @B2A
// Decide which JDBC driver implementation to use. If the
// native driver is available AND there is no secondary URL
// available AND the "driver" property was not set to "toolbox",
// then use the native driver implementation.
//@C4M String driverImplementation = jdProperties.getString(JDProperties.DRIVER);
if ((nativeDriver != null)
&& (dataSourceUrl.getSecondaryURL().length() == 0))
//@C4D Already checked above && (!driverImplementation.equals(JDProperties.DRIVER_TOOLBOX)))
{
//@C3M
boolean isLocal = false; // @C2A
String serverName = dataSourceUrl.getServerName(); // @C2A
if (serverName.length() == 0 || serverName.equalsIgnoreCase("localhost")) // @C2A //@locala
isLocal = true; // @C2A
else { // @C2A
try { // @C2A
InetAddress localInet = InetAddress.getLocalHost(); // @C2A
InetAddress[] remoteInet = InetAddress.getAllByName(serverName); // @C2A
for (int i = 0; i < remoteInet.length; ++i) { // @C2A
if (localInet.equals(remoteInet[i])) { // @C2A
isLocal = true; // @C2A
} // @C2A
} // @C2A
} // @C2A
catch (Throwable e) { // @C2A
// Ignore. We will just assume that we are not local. @C2A
} // @C2A
}
if (isLocal) { // @C2A
if (JDTrace.isTraceOn()) // @C2A
JDTrace.logInformation(this, "Connection is local"); // @C2A
String nativeURL = dataSourceUrl.getNativeURL();
if (JDTrace.isTraceOn())
JDTrace.logInformation(this, "Using native IBM Developer Kit for Java JDBC driver implementation");//@native don't print passwd
return nativeDriver.connect(nativeURL, jdProperties.getOriginalInfo());
} // @C2A
}
}//@C4A
// @C2D else
// @C2D {
if (JDTrace.isTraceOn())
JDTrace.logInformation (this, "Using IBM Toolbox for Java JDBC driver implementation");
// @C2D }
// @A0A
// See if a proxy server was specified.
//if (jdProperties.getIndex (JDProperties.PROXY_SERVER) != -1) //@A3D
if (jdProperties.getString(JDProperties.PROXY_SERVER).length() != 0) //@A3C
proxyServerWasSpecifiedInUrl = true;
if (SystemProperties.getProperty (SystemProperties.AS400_PROXY_SERVER) != null)
proxyServerWasSpecifiedInProperties = true;
if (proxyServerWasSpecifiedInUrl || proxyServerWasSpecifiedInProperties)
proxyServerWasSpecified = true;
// If no proxy server was specified, and there is a secondary URL,
// simply pass the secondary URL to the DriverManager and ask it for
// an appropriate Connection object.
if (!proxyServerWasSpecified)
{
String secondaryUrl = dataSourceUrl.getSecondaryURL ();
if (secondaryUrl.length() != 0)
{
if (JDTrace.isTraceOn())
JDTrace.logInformation (this,
"Secondary URL [" + secondaryUrl + "]");
return DriverManager.getConnection (secondaryUrl, jdProperties.getOriginalInfo());
}
}
as400 = initializeAS400(dataSourceUrl, jdProperties); // @B3C
if (proxyServerWasSpecifiedInUrl)
{
// A proxy server was specified in URL,
// so we need to inform the AS400 object.
//boolean proxyServerSecure = jdProperties.getBoolean (JDProperties.PROXY_SERVER_SECURE); // TBD
String proxyServerNameAndPort = jdProperties.getString (JDProperties.PROXY_SERVER);
// Note: The PROXY_SERVER property is of the form:
// hostName[:portNumber]
// where portNumber is optional.
try
{
as400.setProxyServer (proxyServerNameAndPort);
//as400.setProxyServerSecure (proxyServerSecure); // TBD
}
catch (java.beans.PropertyVetoException e)
{
} // Will never happen.
}
//@B6C Moved common code to prepareConnection.
return prepareConnection(as400, dataSourceUrl, jdProperties);
}
//@B5A
private Connection initializeConnection (AS400 as400)
throws SQLException
{
JDDataSourceURL dataSourceUrl = new JDDataSourceURL(null);
Properties info = new Properties();
JDProperties jdProperties = new JDProperties(null, info, null, null);
//@B6C Moved common code to prepareConnection.
return prepareConnection(as400, dataSourceUrl, jdProperties);
}
//@B5A
private Connection initializeConnection (String schema, Properties info, AS400 as400)
throws SQLException
{
boolean proxyServerWasSpecifiedInUrl = false;
String url = null;
if (schema != null) //@B6A
url = "jdbc:as400://" + as400.getSystemName() + "/" + schema;
else //@B6A
url = "jdbc:as400://" + as400.getSystemName(); //@B6A
JDDataSourceURL dataSourceUrl = new JDDataSourceURL(url);
JDProperties jdProperties = new JDProperties(null, info, null, null);
if (JDTrace.isTraceOn())
JDTrace.logInformation (this, "Using IBM Toolbox for Java JDBC driver implementation");
// See if a proxy server was specified.
if (jdProperties.getString(JDProperties.PROXY_SERVER).length() != 0)
proxyServerWasSpecifiedInUrl = true;
if (proxyServerWasSpecifiedInUrl)
{
// A proxy server was specified in URL, so we need to inform the AS400 object.
//boolean proxyServerSecure = jdProperties.getBoolean (JDProperties.PROXY_SERVER_SECURE);// TBD
String proxyServerNameAndPort = jdProperties.getString (JDProperties.PROXY_SERVER);
// Note: The PROXY_SERVER property is of the form:
// hostName[:portNumber]
// where portNumber is optional.
try
{
as400.setProxyServer (proxyServerNameAndPort);
//as400.setProxyServerSecure (proxyServerSecure); // TBD
}
catch (java.beans.PropertyVetoException e)
{
} // Will never happen.
}
//@B6C Moved common code to prepareConnection.
return prepareConnection(as400, dataSourceUrl, jdProperties);
}
/**
Indicates if the driver is a genuine JDBC compliant driver.
@return Always true.
**/
public boolean jdbcCompliant ()
{
return true;
}
//@B6A -- This logic was formerly in the initializeConnection() method.
private Connection prepareConnection(AS400 as400, JDDataSourceURL dataSourceUrl,
JDProperties jdProperties) throws SQLException {
return prepareConnection(as400,dataSourceUrl, jdProperties, false);
}
private Connection prepareConnection(AS400 as400, JDDataSourceURL dataSourceUrl,
JDProperties jdProperties,
boolean vrmSet)
throws SQLException
{
// set socket properties
SocketProperties sockProps = null;
//if == "", then take platform defaults...do not set
//only get/set properties is one is updated
if( jdProperties.getString(JDProperties.KEEP_ALIVE).equals("") == false)
{
if(sockProps == null)
sockProps = as400.getSocketProperties();
sockProps.setKeepAlive(jdProperties.getBoolean(JDProperties.KEEP_ALIVE));
}
if( jdProperties.getString(JDProperties.RECEIVE_BUFFER_SIZE).equals("") == false)
{
if(sockProps == null)
sockProps = as400.getSocketProperties();
sockProps.setReceiveBufferSize( jdProperties.getInt(JDProperties.RECEIVE_BUFFER_SIZE));
}
if( jdProperties.getString(JDProperties.SEND_BUFFER_SIZE).equals("") == false)
{
if(sockProps == null)
sockProps = as400.getSocketProperties();
sockProps.setSendBufferSize(jdProperties.getInt(JDProperties.SEND_BUFFER_SIZE));
}
//@timeout2
//First get setting from DriverManager, then override with property updates
if(!as400.arePropertiesFrozen()) //@timeout3 AS400JDBCDriver.connect(clone=false) cannot update props. We don't know if DriverManager.setLoginTimeout() has been updated.
{
if(sockProps == null)
sockProps = as400.getSocketProperties();
sockProps.setLoginTimeout(DriverManager.getLoginTimeout() * 1000); //@STIMEOUT
}
//@timeout
if( jdProperties.getString(JDProperties.LOGIN_TIMEOUT).equals("") == false)
{
if(sockProps == null)
sockProps = as400.getSocketProperties();
sockProps.setLoginTimeout(jdProperties.getInt(JDProperties.LOGIN_TIMEOUT) * 1000); //@STIMEOUT
}
//@STIMEOUT
if( jdProperties.getString(JDProperties.SOCKET_TIMEOUT).equals("") == false)
{
if(sockProps == null)
sockProps = as400.getSocketProperties();
sockProps.setSoTimeout(jdProperties.getInt(JDProperties.SOCKET_TIMEOUT)); //@STIMEOUT already in milliseconds
}
if( jdProperties.getBoolean(JDProperties.TCP_NO_DELAY) == true)
{
if(sockProps == null)
sockProps = as400.getSocketProperties();
sockProps.setTcpNoDelay(true);
if (JDTrace.isTraceOn())
JDTrace.logInformation(this, "Setting sockProps.setTcpNoDelay(true)");
}
String defaultImpl = "com.ibm.as400.access.AS400JDBCConnectionImpl";
if (jdProperties.getInt(JDProperties.ENABLE_CLIENT_AFFINITIES_LIST) == 1) {
defaultImpl = "com.ibm.as400.access.AS400JDBCConnectionRedirect";
}
if(sockProps != null)
as400.setSocketProperties(sockProps);
// Create the appropriate kind of Connection object.
Connection connection = (Connection) as400.loadImpl2 (
defaultImpl,
"com.ibm.as400.access.JDConnectionProxy");
// Set the properties on the Connection object.
if (connection != null)
{
// If we get an exception, make sure the connection is closed.
// The common case is when an exit program prevents access to the system.
// @AB1A
try {
if (connection instanceof JDConnectionProxy) {
((JDConnectionProxy)connection).setSystem(as400);
((JDConnectionProxy)connection).setProperties(dataSourceUrl, jdProperties, as400);
} else {
((AS400JDBCConnection)connection).setSystem(as400);
((AS400JDBCConnection)connection).setProperties(dataSourceUrl, jdProperties, as400);
}
} catch (SQLException sqlex) {
try {
connection.close();
} catch (Exception e) {
// Just ignore
}
throw sqlex;
}
}
//
// If the signon server was skipped, we need to manually determine the release
// This is important for boolean support
//
if (as400.skipSignonServer_ && ! vrmSet) {
try {
Statement s = connection.createStatement();
ResultSet rs = s.executeQuery("SELECT OS_VERSION,OS_RELEASE FROM SYSIBMADM.ENVSYSINFO");
if (rs.next()) {
int version = rs.getInt(1);
int release = rs.getInt(2);
as400.setVRM(version,release,0);
}
rs.close();
s.close();
} catch (SQLException sqlex) {
// Log and ignore
}
//
// Connect again to get the correct settings
// This didn't work!!!! TODO;
//
connection.close();
return prepareConnection(as400,dataSourceUrl, jdProperties, true);
}
return connection;
}
/**
Returns the name of the driver.
@return The driver name.
**/
public String toString ()
{
return DRIVER_NAME_; // @D0C
}
/* ifdef JDBC40 */
public Logger getParentLogger() throws SQLFeatureNotSupportedException {
throw new SQLFeatureNotSupportedException();
}
/* endif */
}