Changes per Jaybird 4 release. See also What's new in Jaybird 4. For known issues, consult Known Issues.
Jaybird 4.0.10 is the last release of Jaybird 4, and Jaybird 4 is now end-of-life. We recommend upgrading to Jaybird 5.
The following has been changed or fixed since Jaybird 4.0.9
CallableStatement.getXXX(String)
could return value from wrong column — backported from Jaybird 6 (jaybird#773)The following has been changed or fixed since Jaybird 4.0.8
FBClob
threw NullPointerException
when attempting to use after implicit or explicit free()
(jaybird#719)GMT-08:00
, configure sessionTimeZone
with offset only, e.g. -08:00
(jaybird#720)CallableStatement.getString
would incorrectly trim string values (jaybird#729)ResultSetMetaData.getPrecision(int)
of a connectionless result set could throw a NullPointerException
if the column was of type FLOAT
or DOUBLE PRECISION
(jaybird#730)The following has been changed or fixed since Jaybird 4.0.7
antlr-runtime-4.7.2.jar
, if you don't need it yourself, you can remove this library from the classpath. See New parser for generated keys handling for more information.The following has been changed or fixed since Jaybird 4.0.6
(VAR)CHAR
is now sent to the server with blr_varying2
or blr_text2
which includes the character set information (jaybird#692)String.toUpperCase
and String.toLowerCase
now use Locale.ROOT
to prevent locale-sensitivity issues (jaybird#697)org.firebirdsql.management.NBackupManager
interface has three new methods: setCleanHistory(boolean)
to enable (or disable) cleaning of history during backup, and setKeepDays(int)
and setKeepRows(int)
to specify the number of days or rows to keep history. These options require Firebird 4.0.3 or higher.PreparedStatement.setClob
or PreparedStatement.setBlob
with a null
Clob
, Reader
, Blob
, or InputStream
would result in a NullPointerException
(jaybird#712)setClob
methods accepting a Reader
was changed to be identical to setCharacterStream
, and setBlob
accepting an InputStream
to setBinaryStream
. The end result before and after this change is identical, but it can result in different memory and performance characteristics, as the stream is now consumed on execute, and not on set.The following has been changed or fixed since Jaybird 4.0.5
Connection.isValid(int)
now uses the timeout as a networkSQLWarning
is registered).NativeResourceUnloadWebListener
would fail in Servlet containers with a NoClassDefFoundError
if JNA wasn't on the classpath (jaybird#686)isBeforeFirst()
, isAfterLast()
, isFirst()
, or isLast()
on a closed result set resulted in a NullPointerException
instead of a SQLException
(jaybird#689)The following has been changed or fixed since Jaybird 4.0.4
JnaService
implementation call to isc_service_query
incorrectly includes type (jaybird#678)The following has been changed or fixed since Jaybird 4.0.3
ResultSet.updateRow()
sets fields to null
in result set only (jaybird#37)updateRow()
method did not correctly update local data of the result set. As a result, calling getXXX
for that row after the update would return null
for the updated rows. The update was correctly persisted to the database.org.firebirdsql.management.NBackupManager
interface has two new methods: setBackupGuid(String)
expecting the brace-enclosed GUID of a previous backup to use as the starting point for this backup, and setInPlaceRestore(boolean)
to enable (or disable) in-place restore. These options require Firebird 4.0 or higher.The following has been changed or fixed since Jaybird 4.0.2
The following has been changed or fixed since Jaybird 4.0.1
ServiceConfigurationError
while loading plugins could prevent Jaybird from loading (JDBC-636)The following has been changed or fixed since Jaybird 4.0.0
setTransactionParameters(int, TransactionParameterBuffer)
in FBManagedConnection
can now throw ResourceException
where previously it did not. Under the assumption that most users of Jaybird are not directly using this class, the change should not break anything.NativeLibrary
(JDBC-620)NULL
(JDBC-623)INT128
(reported as JDBC type NUMERIC
) (JDBC-624)FBDriver.normalizeProperties
which, given a JDBC url and a Properties
object, returns a Map<String, String>
containing the merged properties normalized to common property name. (JDBC-627)isc_dpb
name, and removes the database
property. These are both implementation details that might change in future versions.isc_dpb_no_db_triggers
no longer logs a warning (JDBC-628)org.firebirdsql.gds.ng.tz.TimeZoneDatatypeCoder
was made almost entirely private. This should not affect normal user code.org.firebirdsql.gds.ng
package and sub-packages.TIME WITH TIME ZONE
now use 2020-01-01 as base date for named zones (JDBC-629)OffsetTime
and OffsetDateTime
for named zones. For OffsetTime
, we will always use the offset as it was on 2020-01-01, while for OffsetDateTime
we will first rebase the time in the named zone to the current date, and then derive the offset.java.time.ZonedDateTime
for the WITH TIME ZONE
types (JDBC-630)TIME WITH TIME ZONE
and TIMESTAMP WITH TIME ZONE
as a ZonedDateTime
.TIME WITH TIME ZONE
, the returned value is rebased on the current date.Connection.setNetworkTimeout
incorrectly used the provided Executor
to set the timeout (JDBC-631)connection.close()
was called immediately after, this could trigger a NullPointerException
that would bubble up into the executor.The following has been changed or fixed since Jaybird 4.0.0-beta-2
jna-4.4.0.jar
(or jna-5.3.0.jar
when coming from an earlier test version of Jaybird 4) with jna-5.5.0.jar
.EXTENDED TIME(STAMP) WITH TIME ZONE
types introduced in Firebird 4.0.0.1795 (JDBC-611)SQLException
with message "Unexpected tag type: 3" (JDBC-612)The following has been changed or fixed since Jaybird 4.0.0-beta-1
DatabaseMetaData.getFunctions
(JDBC-552)DatabaseMetaData.getFunctionColumns
(JDBC-552)defaultIsolation
/isolation
did not work through DriverManager
, but only on DataSource
implementations. (JDBC-584)NullPointerException
or just work depending on whether the connection had a new transaction. (JDBC-587)Connection.createBlob()
- to the transaction that populated it. Attempts to use the blob after the transaction ends will now throw a SQLException
with message "Blob is invalid. Blob was freed, or closed by transaction end." This restriction does not apply to cached blobs, see also next item.java.sql.Blob
and java.sql.Clob
obtained from a result set were freed after calls to ResultSet.next()
. (JDBC-588)Blob.free()
if you want to invalidate them earlier.Blob.free()
or rely on the garbage collector.Connection.setNetworkTimeout
and getNetworkTimeout
(JDBC-589)setNetworkTimeout
will override the timeout set with the soTimeout
connection property. When a timeout occurs, the connection will be closed.DatabaseMetaData.getProcedures
and getProcedureColumns
(JDBC-590)FLOAT
and DOUBLE PRECISION
are reported using binary precision (24 and 53 respectively), instead of decimal precision (7 and 15 respectively) (JBC-591)x'
) and suffix ('
) to DatabaseMetaData.getTypeInfo
for LONGVARBINARY
, VARBINARY
and BINARY
(JDBC-593)FBEventManager.createFor(Connection)
to create an EventManager
for an existing connection. Back-ported to Jaybird 3.0.7. (JDBC-594)waitTimeout
). It is still required to use connect()
and disconnect()
, to start respectively stop listening for events.WireCrypt = Enabled
, the connection could hang or throw exceptions like "Unsupported or unexpected operation code". (JDBC-599)InputStream.skip
, which in CipherInputStream
never skips beyond its current buffer. If that buffer was at (or 1 or 2 bytes from) the end, Jaybird was reading less bytes than it should. This caused subsequent reads to read wrong data, reading too little or too much data.timeZoneBind
and decfloatBind
connection properties. This feature requires Firebird 4 beta 2 or snapshot Firebird 4.0.0.1683 or higher.java.sql.Statement.setQueryTimeout
for the v16 protocol (Firebird 4 and higher) (JDBC-602)DAYNAME
, will always return day names in English (JDBC-607)MONTHNAME
, will always return month names in English (JDBC-608)DATABASE
(JDBC-609)Using a native connection with a Firebird 3.0 or higher client library to a Firebird 2.5 or older server may be slow to connect. The workaround is to specify the IPv4 address instead of the host name in the connection string, or to use a Firebird 2.5 or earlier fbclient.dll
.
This is caused by CORE-4658
If you need support with Jaybird, join the Firebird-Java Google Group and mailing list. You can subscribe by sending an email to [email protected].
Looking for professional support of Jaybird? Jaybird is now part of the Tidelift subscription.
See also Where to get help
Jaybird is a JDBC driver suite to connect to Firebird database servers from Java and other Java Virtual Machine (JVM) languages.
Jaybird 4 is - compared to Jaybird 3 - an incremental release that builds on the foundations of Jaybird 3. The focus of this release has been on further improving JDBC support and adding support for the new data types and features of Firebird 4.
The main new features are:
Upgrading from Jaybird 3 to 4 should be simple, but please make sure to read Compatibility changes before using Jaybird 4. See also Upgrading from Jaybird 3 to Jaybird 4.
Bug reports about undocumented changes in behavior are appreciated. Feedback can be sent to the Firebird-java Google Group or reported on the issue tracker https://github.com/FirebirdSQL/jaybird/issues/.
Jaybird 4.0.10 was tested against Firebird 2.5.9, 3.0.11, and 4.0.4, and a recent snapshot of Firebird 5.0, but should also support other Firebird versions from 2.5 and up.
Formal support for Firebird 2.0 and 2.1 has been dropped (although in general we expect the driver to work). The Type 2 and embedded server JDBC drivers use JNA to access the Firebird client or embedded library.
This driver does not support InterBase servers due to Firebird-specific changes in the protocol and database attachment parameters that are sent to the server.
Jaybird 4 adds support for the Firebird 3 zlib compression in the pure Java wire protocol. The compression is disabled by default.
Jaybird 4 supports the protocol improvements of Firebird 4 for statement timeouts, but does not implement the new batch protocol.
Jaybird time zone support uses functionality added after Firebird 4 beta 1 (4.0.0.1436), you will need version 4.0.0.1683 or later for the dataTypeBind
connection property.
Jaybird 4 supports the extended numeric precision types introduced after Firebird 4 beta 1 (4.0.0.1436), you will need version 4.0.0.1604 to be able to use NUMERIC
or DECIMAL
with a precision higher than 18.
Jaybird does not support the ChaCha and ChaCha64 wire encryption plugins. This may be added in a future major or point release.
Jaybird 4 does not provide full support for Firebird 5.0. In general, it offers support at the Firebird 4.0 feature set, but there are some caveats.
Most notably, the generated-keys support will not work for DML statements that now use multi-row RETURNING
(all DML except insert ... values (..) returning ...
). Full support for multi-row RETURNING
with generated-keys is provided in Jaybird 5. As a workaround, use executeQuery()
with an explicit RETURNING
clause.
Jaybird 4 supports Java 7 (JDBC 4.1), Java 8 (JDBC 4.2), and Java 9 and higher (JDBC 4.3). Support for earlier Java versions has been dropped.
Given the limited support period for Java 9 and higher versions, we limit support to Java 7, 8, 11, 17, and 211.
Jaybird 4 provides libraries for Java 7, Java 8 and Java 11. The Java 8 builds have the same source and all JDBC 4.3 related functionality as the Java 11 build and can be used on Java 9 and higher as well.
Jaybird 4 is not modularized, but all versions declare the automatic module name org.firebirdsql.jaybird
.
See als Java support in What's new in Jaybird 4.
Jaybird supports the following specifications:
Specification | Notes |
---|---|
JDBC 4.3 | All JDBC 4.3 methods for features supported by Firebird; Java 9 and higher supported using the Java 8 or Java 11 driver. |
JDBC 4.2 | All JDBC 4.2 methods for features supported by Firebird. |
JDBC 4.1 | All JDBC 4.1 methods for features supported by Firebird. |
JTA 1.0.1 | Implementation of javax.transaction.xa.XAResource interface via XADataSource implementation. |
Jaybird 4.0.10 is available from Maven central:
Groupid: org.firebirdsql.jdbc
,
Artifactid: jaybird
,
Version: 4.0.10.javaXX
(where XX
is 7
, 8
or 11
).
For ease of transition to the new artifact naming, we also provide a Maven relocation artifact with artifact id jaybird-jdkXX
(with XX
is 17
or 18
). However, we recommend switching to the jaybird
artifact id.
NOTE: SNAPSHOT releases are only available from the Sonatype snapshot repository, https://oss.sonatype.org/content/repositories/snapshots
For example:
<dependency>
<groupId>org.firebirdsql.jdbc</groupId>
<artifactId>jaybird</artifactId>
<version>4.0.10.java8</version>
</dependency>
If your application is deployed to a Java EE or Jakarta EE 8 or lower application server, you will need to exclude the javax.resource:connector-api
dependency, and add it as a provided dependency:
<dependency>
<groupId>org.firebirdsql.jdbc</groupId>
<artifactId>jaybird</artifactId>
<version>4.0.10.java8</version>
<exclusions>
<exclusion>
<groupId>javax.resource</groupId>
<artifactId>connector-api</artifactId>
</exclusion>
</exclusions>
</dependency>
<dependency>
<groupId>javax.resource</groupId>
<artifactId>connector-api</artifactId>
<version>1.5</version>
<scope>provided</scope>
</dependency>
If you want to use Type 2 support (native, local or embedded), you need to explicitly include JNA 5.5.0 as a dependency:
<dependency>
<groupId>net.java.dev.jna</groupId>
<artifactId>jna</artifactId>
<version>5.5.0</version>
</dependency>
For Windows and Linux, you can add the org.firebirdsql.jdbc:fbclient
dependency on your classpath to provide the native libraries for the native
and local
protocol. Be aware that this dependency does not support embedded
.
See also Type 2 (native) and embedded driver.
You can download the latest versions from https://www.firebirdsql.org/en/jdbc-driver/
At minimum Jaybird 4 requires jaybird-4.0.10.javaXX.jar
(where XX
is 7
, 8
or 11
) and connector-api-1.5.jar
. You can also use jaybird-full-4.0.10.javaXX.jar
which includes the connector-api files.
If you deploy your application to a Java EE or Jakarta EE 8 or lower application server, then you must use jaybird-4.0.10.javaXX.jar
(not -full
!), and not include connector-api-1.5.jar
as this dependency will be provided by your application server.
For native, local or embedded support, you will need to include jna-5.5.0.jar
on your classpath. See also Type 2 (native) and embedded driver.
Please make sure to read Compatibility changes and Changes in artifact and library names before upgrading to Jaybird 4.
Change the artifact id from jaybird-jdkXX
to jaybird
, and change the version of the dependency to 4.0.10.javaXX
(where XX
is your Java version, 7
for Java 7, 8
for Java 8 and 11
for Java 11). If you use native or embedded verify that you upgrade JNA (net.java.dev.jna:jna
) from 4.4.0 to 5.5.0.
For more detailed instructions, see also the information on Maven in Getting Jaybird 4.
If you manage your dependencies manually, you need to do the following:
Replace the Jaybird 3 library with the Jaybird 4 version
jaybird-3.0.x.jar
with jaybird-4.0.10.javaXX.jar
(where XX
is 7
, 8
or 11
)jaybird-full-3.0.x.jar
with jaybird-full-4.0.10.javaXX.jar
If installed, remove antlr-runtime-4.7.jar
or antlr-runtime-4.7.2.jar
(unless of course your application itself needs these libraries). Since Jaybird 4.0.8, Jaybird no longer relies on ANTLR.
If installed, remove jna-4.4.0.jar
and replace it with jna-5.5.0.jar
. This library is only necessary for native, local or embedded connections. If you use pure-java connections (the default), you don't need JNA.
If you find a problem while upgrading, or other bugs: please report it on https://github.com/FirebirdSQL/jaybird/issues/.
For known issues, consult Known Issues.
For a full list of changes, see Firebird tracker for Jaybird 4.
Historically, the naming of Jaybird artifacts and libraries has been a bit inconsistent. With the rapid release cycle of Java, naming collisions are imminent with the old naming convention. For example, the Maven artifact jaybird-jdk15
was used for Java 1.5 and with this naming convention, this would be reused for Java 15.
Forced by this issue, we have overhauled the naming convention entirely to bring more consistency between Maven artifacts and the Jaybird zip distribution. The full naming convention is documented in jdp-2019-04: Version Number and Naming Scheme.
This new naming convention has been changed compared to the one from Jaybird 4.0.0-beta-1.
This new naming convention has the following effects:
jaybird
for all Java versions. We provide a relocation artifact for jaybird-jdk17
and jaybird-jdk18
for backwards compatibility.4.0.0.java11
)As a result of these new naming conventions, the following has been changed:
jaybird
(for all Java versions) (was jaybird-jdk18
)4.0.0.java8
(was 3.0.5
)jaybird-4.0.0.java8.zip
(was Jaybird-3.0.5_JDK1.8.zip
)jaybird-4.0.0.java8.jar
(was jaybird-3.0.5.jar
in zip distribution)jaybird-full-4.0.0.java8.jar
(was jaybird-full-3.0.5.jar
)jaybird-4.0.0.java8-sources.jar
(was jaybird-3.0.5-sources.jar
in zip distribution)jaybird-4.0.0.java8-javadoc.jar
(was jaybird-3.0.5-javadoc.jar
in zip distribution)Furthermore, the client name reported to Firebird 2.5 and higher has been changed from Jaybird 3.0.5-JDK_1.8
to Jaybird jaybird-4.0.0.java8
The driver supports Java 7 with caveats.
Firebird 4 time zone types are not supported under Java 7, see also Firebird 4 time zone support.
Under Java 7, Jaybird requires JAXB (javax.xml.bind
), this will work in standard Java, but may require additional configuration in certain environments, for example JBoss/Wildfly.
Some libraries used for testing Jaybird have upped their minimum version to Java 8, while we need those library versions to test - for example - Java 11. When we can no longer work around these issues, we will sacrifice Java 7 test coverage to maintain Java 7 support.
The driver supports Java 8.
Jaybird 4 supports Java 9 and higher (JDBC 4.3) with the Java 8 and 11 version of the driver. Most of the JDBC 4.3 features have been implemented (in as far as they are supported by Firebird).
You can use the Java 8 driver under Java 9 and higher. For Java 11 or higher we recommend using the Java 11 driver, though its sources are identical to the Java 8 driver.
We recommend not to use the Java 7 version of Jaybird with Java 9 or higher. The Java 7 version doesn't implement all JDBC 4.3 features that are implemented in the Java 8 version. In addition, since Jaybird 3.0.4, the Java 7 version of Jaybird needs the java.xml.bind
module, where the Java 8 and higher versions do not need that module.
Given the limited support period for Java 9 and higher versions, we limit support to Java 7, 8, 11, 17, and 212.
For compatibility with Java 9 modules, Jaybird defines the automatic module name org.firebirdsql.jaybird
. This guarantees a stable module name for Jaybird, and allows for future modularization of Jaybird.
Support for Firebird 2.0 and 2.1 has been dropped. See Firebird 2.0 and 2.1 no longer supported for details.
Firebird versions 2.5, 3.0 and 4.0 are supported.
Jaybird 4 adds support for the Firebird 3 ARC4 wire encryption. This feature has been back-ported to Jaybird 3.0.4. The encryption is configured using the connection property wireCrypt
, with the following (case-insensitive) values:
DEFAULT
: default (value used when wireCrypt
is not specified; you'd normally not specify this explicitly)ENABLED
: enable, but not require, wire encryptionREQUIRED
: require wire encryption (only if Firebird version is 3.0 or higher)DISABLED
: disable wire encryptionThe default value acts as ENABLED
for pure Java connections, for JNA (native) connections this wil use the fbclient default (either Enabled
or the configured value of WireCrypt
from a firebird.conf
read by the native library).
Connection property wireCrypt=REQUIRED
will not reject unencrypted connections when connecting to Firebird 2.5 or lower. This behavior matches the Firebird 3 client library behavior. The value will also be ignored when using native connections with a Firebird 2.5 client library.
Using wireCrypt=DISABLED
when Firebird 3 or higher uses setting WireCrypt = Required
(or vice versa) will yield error "Incompatible wire encryption levels requested on client and server" (error: isc_wirecrypt_incompatible / 335545064).
The same error is raised when connecting to Firebird 3 and higher with a legacy authentication user with connection property wireCrypt=REQUIRED
.
Alternative wire encryption plugins are currently not supported, although we made some preparations to support this. If you want to develop such a plugin, contact us on the Firebird-Java mailing list so we can work out the details of adding plugin support.
The new ChaCha and ChaCha64 wire encryption introduced in Firebird 4 is not yet supported.
WARNING
The implementation comes with a number of caveats:
Jaybird 4 (and 3.0.4) adds support for Firebird 3 database encryption callbacks in the pure Java implementation of the version 13 protocol. This feature was sponsored by IBPhoenix.
In addition, version 15 of the protocol (Firebird 3.0.2 and higher) was also implemented, supporting encryption callbacks during authentication for use with encrypted security databases.
The current implementation is simple and only supports replying with a static value from a connection property. Be aware that a static value response for database encryption is not very secure as it can easily lead to replay attacks or unintended key exposure.
Future versions of Jaybird may introduce plugin support for database encryption plugins that require a more complex callback.
The static response value of the encryption callback can be set through the dbCryptConfig
connection property. DataSource
and ServiceManager
implementations have an equivalent property with the same name. This property can be set as follows:
base64:
: rest of the string is decoded as base64 to bytes. The =
padding characters are optional, but when present they must be valid (that is: if you use padding, you must use the right number of padding characters for the length).+
, it must be escaped as %2B
in the JDBC URL. For backwards compatibility with Jaybird 3, we can't switch to the URL-safe variant of base64.:
in the plain string value.Because of the limitation of connection URL parsing, we strongly suggest avoiding plain string values with &
or ;
. Likewise, avoid :
so that we can support other prefixes similar to base64:
in the future. If you need these characters, consider using a base64 encoded value instead, or ensure these characters are URL encoded: &
: %26
, ;
: %3B
, :
: %3A
, %
: %25
, +
: %2B
.
For service operations, as implemented in the org.firebirdsql.management
package, Firebird requires the KeyHolderPlugin
configuration to be globally defined in firebird.conf
. Database-specific configuration in databases.conf
will be ignored for service operations. Be aware that some service operations on encrypted databases are not supported by Firebird 3 (eg gstat
equivalents other than gstat -h
or gstat -e
).
Other warnings and limitations
wireCrypt=REQUIRED
to force encryption (caveat: see the next point).dbCryptConfig
value cannot be obtained by someone with access to your application or the machine hosting your application (although that in itself would already imply a severe security breach).Support for zlib wire compression was added in the pure Java wire protocol. Compression can be enabled using boolean connection property wireCompression
.
The connection property only affects the pure Java wire protocol connections on Firebird 3 and higher, if the server has the zlib library. Native connections will follow the WireCompression
configuration in the firebird.conf
read by the client library, if the zlib library is on the search path.
Compression is currently disabled by default. We may change this in future versions of Jaybird to be enabled by default.
The wireCompression
property is also available on data sources and the management classes in org.firebirdsql.management
.
Jaybird 4 has added support for the new SrpNNN
(with NNN is 224, 256, 384 and 512) authentication plugins added in Firebird 4 (back-ported to Firebird 3.0.4 for Srp256 only).
The original Srp
plugin uses SHA-1, the new Srp-variants use SHA-224, SHA-256, SHA-384 and SHA-512 respectively3.
Be aware, support for these plugins depends on support of these hash algorithms in the JVM. For example, SHA-224 is not supported in Oracle Java 7 by default and may require additional JCE libraries.
The default plugins applied by Jaybird are now - in order - Srp256
, Srp
. This applies only for the pure Java protocol and only when connecting to Firebird 3 or higher. The native implementation will use its own default, or the value configured through its firebird.conf
.
When connecting to Firebird 3 versions earlier than 3.0.4, or if Srp256
has been removed from the AuthServer
setting in Firebird, this might result in slower authentication because more roundtrips to the server are needed. After the attempt to use Srp256
fails, authentication continues with Srp
.
To avoid this, consider explicitly configuring the authentication plugins to use, see Configure authentication plugins for details.
When connecting to Firebird 3 or higher, the pure Java protocol in Jaybird will no longer try the Legacy_Auth
plugin by default as it is an unsafe authentication mechanism. We strongly suggest to use SRP users only, but if you really need to use legacy authentication, you can specify connection property authPlugins=Legacy_Auth
, see Configure authentication plugins for details.
Firebird 2.5 and earlier are not affected and will always use legacy authentication.
Jaybird 4 introduces the connection property authPlugins
(alias auth_plugin_list
) to specify the authentication plugins to try when connecting. The value of this property is a comma-separated4 list with the plugin names.
Unknown or unsupported plugins will be logged and skipped. When no known plugins are specified, Jaybird will throw an exception with:
For pure Java
Cannot authenticate. No known authentication plugins, requested plugins: [<plugin-names>] [SQLState:28000, ISC error code:337248287]
For native
Error occurred during login, please check server firebird.log for details [SQLState:08006, ISC error code:335545106]
The authPlugins
property only affects connecting to Firebird 3 or later. It will be ignored when connecting to Firebird 2.5 or earlier. The setting will also be ignored for native connections when using a fbclient library of version 2.5 or earlier.
Examples:
JDBC URL to connect using Srp256
only:
jdbc:firebirdsql://localhost/employee?authPlugins=Srp256
JDBC URL to connect using Legacy_Auth
only (this is unsafe!)
jdbc:firebirdsql://localhost/employee?authPlugins=Legacy_Auth
JDBC URL to try Legacy_Auth
before Srp512
(this order is unsafe!)
jdbc:firebirdsql://localhost/employee?authPlugins=Legacy_Auth,Srp512
The property is also supported by the data sources, service managers and event manager.
If you develop your own Firebird authentication plugin (or use a third-party authentication plugin), it is possible - for pure Java only - to add your own authentication plugin by implementing the interfaces
org.firebirdsql.gds.ng.wire.auth.AuthenticationPluginSpi
org.firebirdsql.gds.ng.wire.auth.AuthenticationPlugin
The SPI implementation needs to be listed in META-INF/services/org.firebirdsql.gds.ng.wire.auth.AuthenticationPluginSpi
in your jar.
This support is experimental and comes with a number of caveats:
If you implement a custom authentication plugin and run into problems, contact us on the Firebird-Java mailing list.
If you use a native connection, check the Firebird documentation how to add third-party authentication plugins to fbclient.
Firebird 4 (build 4.0.0.1683 or later) introduced the SET BIND
statement and isc_dpb_set_bind
DPB item. This allows you to define data type conversion rules for compatibility or ease of processing data.
This feature is specifically necessary for using the WITH TIME ZONE
types under Java 7. See also Time zone bind configuration.
In Jaybird this feature is exposed as connection property dataTypeBind
(alias set_bind
). The value of this connection property is a semicolon-separated list of data type bind definitions.
A data type bind definition is of the form <from-type> TO <to-type>
. A definition is the same as the second half of a SET BIND
statement after the OF
. See the Firebird documentation of SET BIND
for more information. Invalid values or impossible mappings will result in an error on connect.
When using the dataTypeBind
connection property in a JDBC URL, the semicolons of the list need to be encoded as %3B
, as semicolons in the JDBC URL are an alternative to &
as the separator between properties.
For example:
String jdbcUrl = "jdbc:firebirdsql://localhost/database?charSet=utf-8"
+ "&dataTypeBind=decfloat to varchar%3Btimestamp with time zone to legacy
When the property is set through a Properties
object or a DataSource
configuration, encoding the semicolon is not necessary and will result in errors.
For example:
Properties props = new Properties();
props.setProperty("dataTypeBind",
"decfloat to varchar;timestamp with time zone to legacy"
Values set through this connection property will be the session default configuration, which means that they are retained (or reverted to) when executing ALTER SESSION RESET
.
This feature replaces the connection properties timeZoneBind
and decfloatBind
from earlier Jaybird 4 beta or snapshot versions. The timeZoneBind
and decfloatBind
properties are no longer supported.
Firebird 4 introduces the SQL:2016 DECFLOAT
datatype, a decimal floating point with a precision of 16 or 34 digits (backed by an IEEE-754 Decimal64 or Decimal128). See the Firebird 4 release notes for details on this datatype.
Jaybird 4 adds support for this datatype. The 'default' object type for DECFLOAT
is a java.math.BigDecimal
, but conversion from and to the following datatypes is supported:
java.math.BigDecimal
(see note 1)byte
(valid range -128 to 127(!); see notes 2, 3)short
(valid range -32768 to 32767; see note 3)int
(valid range -231 to 231-1; see note 3)long
(valid range -263 to 263-1; see notes 3, 4)float
(valid range -1 * Float.MAX_VALUE to Float.MAX_VALUE; see notes 5-9)double
(valid range -1 * Double.MAX_VALUE to Double.MAX_VALUE; see notes 6-9)boolean
(see notes 10, 11)java.lang.String
(see notes 12-14)java.math.BigInteger
(see notes 15, 16)org.firebirdsql.extern.decimal.Decimal32/64/128
(see notes 17, 18)The DECFLOAT
type is not yet defined in the JDBC specification. For the time being, we have defined a Jaybird specific type code with value -6001
. This value is available through constant org.firebirdsql.jdbc.JaybirdTypeCodes.DECFLOAT
, or - for JDBC 4.2 and higher - org.firebirdsql.jdbc.JaybirdType.DECFLOAT
, which is an enum implementing java.sql.SQLType
.
If you need to use the type code, we suggest you use these constants. If a DECFLOAT
type constant gets added to the JDBC standard, we will update the value. The enum value will be deprecated when that version of JDBC has been released.
Jaybird uses a local copy of the FirebirdSQL/decimal-java library, with a custom package org.firebirdsql.extern.decimal
. This to avoid additional dependencies.
The DECFLOAT
datatype supports values with a precision of 16 or 34 decimal digits, and an exponent5 between -398 and 369 (DECFLOAT(16)
), or between -6176 and 6111 (DECFLOAT(34)
), so the minimum and maximum values are:
Type | Min/max value | Smallest (non-zero) value |
---|---|---|
DECFLOAT(16) |
+/-9.9..9E+384 (16 digits) | +/-1E-398 (1 digit) |
DECFLOAT(34) |
+/-9.9..9E+6144 (34 digits) | +/-1E-6176 (1 digit) |
When converting values from Java types to DECFLOAT
and retrieving DECFLOAT
values as Decimal32
or Decimal64
, the following rules are applied:
Zero values can have a non-zero exponent, and if the exponent is out of range, the exponent value is 'clamped' to the minimum or maximum exponent supported. This behavior is subject to change, and future releases may 'round' to exact 0
(or 0E0
)
Values with a precision larger than the target precision are rounded to the target precision using RoundingMode.HALF_EVEN
If the magnitude (or exponent) is too low, then the following steps are applied:
Precision is reduced applying RoundingMode.HALF_EVEN
, increasing the exponent by the reduction of precision.
An example: a DECFLOAT(16)
stores values as an integral coefficient of 16 digits and an exponent between -398
and +369
. The value 1.234567890123456E-394
or 1234567890123456E-409
is coefficient 1234567890123456
and exponent -409
. The coefficient is 16 digits, but the exponent is too low by 11.
If we sacrifice least-significant digits, we can increase the exponent, this is achieved by dividing the coefficient by 1011 (and rounding) and increasing the exponent by 11. We get exponent = round(1234567890123456 / 1011) = 12346 and exponent = -409 + 11 = -398.
The resulting value is now 12346E-398
or 1.2346E-394
, or in other words, we sacrificed precision to make the value fit.
If after the previous step, the magnitude is still too low, we have what is called an underflow, and the value is truncated to 0 with the minimum exponent and preserving sign, e.g. for DECFLOAT(16)
, the value will become +0E+398 or -0E-398 (see note 19). Technically, this is just a special case of the previous step.
If the magnitude (or exponent) is too high, then the following steps are applied:
If the precision is less than maximum precision, and the difference between maximum precision and actual precision is larger than or equal to the difference between the actual exponent and the maximum exponent, then the precision is increased by adding zeroes as least-significant digits and decreasing the exponent by the number of zeroes added.
An example: a DECFLOAT(16)
stores values as an integral coefficient of 16 digits and an exponent between -398
and +369
. The value 1E+384
is coefficient 1
with exponent 384
. This is too large for the maximum exponent, however, we have a value with a single digit, leaving us with 15 'unused' most-significant digits.
If we multiply the coefficient by 1015 and subtract 15 from the exponent we get: coefficient = 1 * 1015 = 1000000000000000 and exponent = 384 - 15 = 369, and these values for coefficient and exponent are in range of the storage requirements.
The resulting value is now 1000000000000000E+369
or 1.000000000000000E+384
, or in other words, we 'increased' precision by adding zeroes as least-significant digits to make the value fit.
Otherwise, we have what is called an overflow, and an SQLException
is thrown as the value is out of range.
If you need other rounding and overflow behavior, make sure you round the values appropriately before you set them.
To configure the server-side(!) error and rounding behaviour of the DECFLOAT
data types, you can configure use the following connection properties:
decfloatRound
(alias: decfloat_round
)ceiling
, up
, half_up
(default), half_even
, half_down
, down
, floor
, reround
decfloatTraps
(alias: decfloat_traps
)Division_by_zero
(default), Inexact
, Invalid_operation
(default), Overflow
(default), Underflow
Configuring these options does not change driver behaviour, only server-side behaviour.
java.math.BigDecimal
is capable of representing numbers with larger precisions than DECFLOAT
, and numbers that are out of range (too large or too small). When performing calculations in Java, use MathContext.DECIMAL64
(for DECFLOAT(16)
) or MathContext.DECIMAL128
(for DECFLOAT(34)
) to achieve similar results in calculations as in Firebird. Be aware there might still be differences in rounding, and the result of calculations may be out of range.
SQLException
, with a DecimalInconvertibleException
cause with details on the special. The support for these special values is currently under discussion and may be removed in future Firebird 4 snapshots, or may be disabled by default.byte
in Java is signed, and historically Jaybird has preserved sign when storing byte values, and it considers values outside -128 and +127 out of range.
All integral values are - if within range - first converted to long
using BigDecimal.longValue
, which discards any fractional parts (rounding by truncation).
When storing a long
in DECFLOAT(16)
, rounding will be applied using RoundingMode.HALF_EVEN
for values larger than 9999999999999999L
or smaller than -9999999999999999L
.
float
values are first converted to (or from) double, this may lead to small rounding differences
float
and double
can be fully stored in DECFLOAT(16)
and DECLOAT(34)
, with minor rounding differences.
When reading DECFLOAT
values as double
or float
, rounding will be applied as binary floating point types are inexact, and have a smaller precision.
If the magnitude of the DECFLOAT
value is too great to be represented in float
or double
, +Infinity or -Infinity may be returned (see BigDecimal.doubleValue()
). This behavior is subject to change, future releases may throw a SQLException
instead, see also related note 9.
Storing and retrieving values NaN, +Infinity and -Infinity are currently supported, but this may change as this doesn't seem to be allowed by the SQL:2016 standard.
It is possible that Jaybird or Firebird will disallow storing and retrieving NaN and Infinity values in future releases, causing Jaybird to throw an SQLException
instead. We strongly suggest not to rely on this support for special values.
DECFLOAT
currently discerns four different NaNs (+/-NaN and +/-signaling-NaN). These are all mapped to Double.NaN
(or Float.NaN
), Java NaN values are mapped to +NaN in Firebird.Setting boolean
values will set 0
(or 0E+0
) for false
and 1
(or 1E+0
) for true
.
Retrieving as boolean
will return true
for 1
(exactly 1E+0
) and false
for all other values. Be aware that this means that 1.0E+0
(or 10E-1
) etc will be false
.
This behavior may change in the future and only allow 0
for false
and exactly 1
for true
and throw an SQLException
for all other values, or maybe true
for everything other than 0
. In general, we advise to not use numerical types for boolean values, and especially not to retrieve the result of a calculation as a boolean value. Instead, use a real BOOLEAN
.
Setting values as String
is supported following the format rules of new BigDecimal(String)
, with extra support for special values +NaN
, -NaN
, +sNaN
, -sNaN
, +Infinity
and -Infinity
(case-insensitive). Other non-numerical strings throw an SQLException
with a NumberFormatException
as cause. Out of range values are handled as described in Precision and range.
Getting values as String
will be equivalent to BigDecimal.toString()
, with extra support for the special values mentioned in the previous note.
As mentioned in earlier notes, support for the special values is under discussion, and may be removed or change in future versions of Jaybird and/or Firebird.
Getting as BigInteger
will behave as BigDecimal.toBigInteger()
, which discards the fractional part (rounding by truncation), and may add (-1 * scale - precision)
least-significant zeroes if the scale exceeds precision. Be aware that use of BigInteger
for large values may result in significant memory consumption.
Setting as BigInteger
will lose precision for values with more digits than the target type. It applies the rules described in Precision and range.
Values can also be set and retrieved as types Decimal32
, Decimal64
and Decimal128
from the org.firebirdsql.extern.decimal
package. Where Decimal64
exactly matches the DECFLOAT(16)
protocol format, and Decimal128
the DECFLOAT(34)
protocol format. Be aware that this is an implementation detail that might change in future Jaybird versions (both in terms of support for these types, and in terms of the interface (API) of these types).
Setting a Decimal128
on a DECFLOAT(16)
, or a Decimal32
on a DECFLOAT(16)
or DECFLOAT(34)
, or retrieving a Decimal32
from a DECFLOAT(16)
or DECFLOAT(34)
, or a Decimal64
from a DECFLOAT(34)
will apply the rules described in Precision and range.
Zero values can have a sign (eg -0
vs 0
(+0
)), this can only be set or retrieved using String
or the DecimalXX
types, or the result of rounding. This behaviour is subject to change, and future releases may 'round' to 0
(aka +0
).
Added support for the extended precision for NUMERIC
and DECIMAL
introduced in Firebird 4, increasing the maximum precision to 38. In the implementation in Firebird, this extended precision is backed by an Int128.
Any NUMERIC
or DECIMAL
with a precision between 19 and 38 will allow storage up to a precision of 38 (technically even 39, but not full range).
Values set on a field or parameter will be rounded to the target scale of the field using RoundingMode.HALF_UP
. Values exceeding that do not fit in an Int128 after rounding will be rejected with a TypeConversionException
.
The implementation of extended numeric precision was changed after Firebird 4.0.0-beta-1 (in build 1604) from a maximum precision of 34 backed by a Decimal128 to a maximum precision of 38 backed by an Int128.
Support for the old 'DEC_FIXED' format backed by a Decimal128 was removed in Jaybird 4.0.0-beta-2.
Since: Jaybird 4.0.1
After Firebird 4 beta 2 (in build 4.0.0.2076), the Int128 type used to back the extended precision numerics was exposed as SQL type INT128
. As JDBC does not provide this type, we map it to JDBC type NUMERIC
, with a reported precision of 38 and a scale of 0.
We think this mapping will offer the best support in tool and libraries. For further details, see jdp-2020-08 Int128 support
Added support for the Firebird 4 TIME WITH TIME ZONE
and TIMESTAMP WITH TIME ZONE
types. See the Firebird 4 release notes and doc/sql.extensions/README.time_zone.md
in the Firebird installation for details on these types.
The time zone types are supported under Java 8 and higher, using the Java 8 (or higher) version of Jaybird.
Time zone types are not supported under Java 7, you will need to enable legacy time zone bind. With legacy time zone bind, Firebird will convert to the equivalent TIME
and TIMESTAMP
(WITHOUT TIME ZONE
) types using the session time zone. Time zone bind can be configured with connection property dataTypeBind
, for more information see Time zone bind configuration.
See also jdp-2019-03: Time Zone Support
JDBC 4.2 introduced support for time zones, and maps these types to java.time.OffsetTime
and java.time.OffsetDateTime
. JDBC does not define explicit setters for these types. Use setObject(index, value)
, updateObject(index, value)
, getObject(index/name)
or getObject(index/name, classType)
.
Firebird 4 supports both offset and named time zones. Given the definition in JDBC, Jaybird defaults to offset time zones. On retrieval of a value with a named zone, Jaybird will make a best-effort attempt to convert to the equivalent offset using Java's time zone information. If no mapping is available, the time will be returned at UTC (offset zero).
Since Jaybird 4.0.1, it is also possible to get and set java.time.ZonedDateTime
, which preserves the named zone information.
Jaybird 4 supports the following Java types on fields of time zone types (those marked with * are not defined in JDBC)
TIME WITH TIME ZONE
:
java.time.OffsetTime
(default for getObject
)
OffsetDateTime
for the same value.java.time.OffsetDateTime
OffsetTime
.java.time.ZonedDateTime
(*)
java.lang.String
OffsetTime.toString()
(eg 13:25:13.1+01:00
)OffsetTime
or OffsetDateTime
(eg 13:25:13.1+01:00
or 2019-03-10T13:25:13+01:00
) and then sets as that typejava.sql.Time
(*)
java.time.OffsetDateTime
, converts this to epoch milliseconds and uses new java.sql.Time(millis)
toLocalTime()
, combines this with LocalDate.now()
and then derives the offset time for the default JVM time zonejava.sql.Timestamp
(*)
java.time.OffsetDateTime
, converts this to epoch milliseconds and uses new java.sql.Timestamp(millis)
toLocalDateTime()
and derives the offset time for the default JVM time zoneTIMESTAMP WITH TIME ZONE
:
java.time.OffsetDateTime
(default for getObject
)java.time.OffsetTime
(*)
java.time.ZonedDateTime
(*)java.lang.String
OffsetDateTime.toString()
(eg 2019-03-10T13:25:13.1+01:00
)OffsetTime
or OffsetDateTime
(eg 13:25:13.1+01:00
or 2019-03-10T13:25:13+01:00
) and then sets as that typejava.sql.Time
(*)
java.time.OffsetDateTime
, converts this to epoch milliseconds and uses new java.sql.Time(millis)
toLocalTime()
, combines this with LocalDate.now()
and then derives the offset date time for the default JVM time zonejava.sql.Timestamp
(*)
java.time.OffsetDateTime
, converts this to epoch milliseconds and uses new java.sql.Timestamp(millis)
toLocalDateTime()
and derives the offset date time for the default JVM time zonejava.sql.Date
(*)
java.time.OffsetDateTime
, converts this to epoch milliseconds and uses new java.sql.Date(millis)
toLocalDate()
at start of day and derives the offset date time for the default JVM time zoneIn addition, Firebird 4 has 'bind-only' data types EXTENDED TIME/TIMESTAMP WITH TIME ZONE
. These data types can be set through the data type bind configuration and include an extra offset in its data so clients without access to ICU or other time zone data can use the offset as determined by Firebird.
Jaybird provides minimal support for these types by handling them the same as the normal WITH TIME ZONE
types. That means the extra offset information is ignored and Jaybird will always use the Java time zone information to calculate the offset of a named zone, and if a zone is unknown in Java, Jaybird will fallback to UTC even when the actual offset is available in the 'extended' time zone type.
See also:
For the WITH TIME ZONE
types, JDBC does not define support for the legacy JDBC types (java.sql.Time
, java.sql.Timestamp
and java.sql.Date
). To ease the transition and potential compatibility with tools and libraries, Jaybird does provide support. However, we strongly recommend avoiding using these types.
Compared to the WITHOUT TIME ZONE
types, there may be small discrepancies in values as Jaybird uses 1970-01-01 for WITHOUT TIME ZONE
, while for WITH TIME ZONE
it uses the current date. If this is problematic, then either apply the necessary conversions yourself, enable legacy time zone bind, or define or cast your columns to TIME
or TIMESTAMP
.
The types java.time.LocalTime
, java.time.LocalDateTime
and java.time.LocalDate
are not supported. Supporting these types would be ambiguous. If you need to use these, then either apply the necessary conversions yourself, enable legacy time zone bind, or define or cast your columns as TIME
or TIMESTAMP
.
Jaybird also does not support non-standard extensions like java.time.Instant
. If there is interest, we may add them in the future.
If you are using Java 7 and need to handle the WITH TIME ZONE
types, you will need to redefine the data type binding for the time zone types as the necessary java.time
types do not exist in Java 7. Redefining the binding can also be used for tools or applications that expect java.sql.Time
/Timestamp
types and cannot use the java.time.OffsetTime
/OffsetDateTime
types returned for the WITH TIME ZONE
types.
To redefine the data type binding, you can use the connection property dataTypeBind
. See Firebird 4 data type bind configuration support for details.
You will need to map the time zone types, either to legacy
(which uses time
and timestamp
(without time zone)) or to a desired target data type (for example time
or varchar
). You can map per type or for both types together:
Properties props = new Properties();
props.setProperty("dataTypeBind", "time zone to legacy");
The TIME ZONE TO EXTENDED
binds (including type-specific variants) is only supported under Java 8 and higher using the Java 8 or higher version of Jaybird. As mentioned earlier, the support for 'extended' time zone types will behave identical to the normal time zone types.
Important: These features requires Firebird 4 beta 2 or higher (or a snapshot build version 4.0.0.1683 or later). It will be ignored in builds before 1481 as the necessary database parameter buffer item does not exist, and it will raise an error in versions between 1481 and 1682 as there the DPB item points to the removed DPB item isc_time_zone_bind
.
The connection property sessionTimeZone
(alias session_time_zone
) does two things:
By default, Jaybird will use the JVM default time zone as reported by java.util.TimeZone.getDefault().getID()
. Using the JVM default time zone as the default is the best option in the light of JDBC requirements with regard to java.sql.Time
and java.sql.Timestamp
using the JVM default time zone.
Valid values are time zone names known by Firebird, we recommend using the long names (eg Europe/Amsterdam
), and not the ambiguous short IDs (eg CET
). Although not required, we recommend to use time zone names that are known by Firebird and Java (see Session time zone for conversion for caveats).
To use the default server time zone and the old behaviour to use the JVM default time zone, set the connection property to server
. This will result in the conversion behaviour of Jaybird 3 and earlier. Be aware that this is inconsistent if Firebird and Java are in different time zones.
The session time zone is used for conversion between WITH TIME ZONE
values and WITHOUT TIME ZONE
values (ie using cast or with legacy time zone bind), and for the value of LOCALTIME
, LOCALTIMESTAMP
, CURRENT_TIME
and CURRENT_TIMESTAMP
, and other uses of the session time zone as documented in the Firebird 4 documentation.
The value of sessionTimeZone
must be supported by Firebird 4. It is possible that time zone identifiers used by Java are not supported by Firebird. If Firebird does not know the session time zone, error (Invalid time zone region: <zone name>
) is reported on connect.
The use of the JVM default time zone as the default session time zone will result in subtly different behaviour compared to previous versions of Jaybird and - even with Jaybird 4 - Firebird 3 or earlier, as current time values like LOCALTIMESTAMP
(etc) will now reflect the time in the JVM time zone, and not the server time zone rebased on the JVM default time zone.
As an example, with a Firebird in Europe/London and a Java application in Europe/Amsterdam with Firebird time 12:00, in Jaybird 3, the Java application will report this time as 12:00, in Jaybird 4 with Firebird 4, this will now report 13:00, as that is the time in Amsterdam if it is 12:00 in London (ignoring potential DST start/end differences).
Other examples include values generated in triggers and default value clauses.
For WITHOUT TIME ZONE
types, the session time zone will be used to derive the java.sql.Time
, java.sql.Timestamp
and java.sql.Date
values. This is also done for Firebird 3 and earlier.
If Java does not know the session time zone, no error is reported, but when retrieving java.sql.Time
, java.sql.Timestamp
or java.sql.Date
a warning is logged and conversion will happen in GMT, which might yield unexpected values.
We strongly suggest that you use java.time.LocalTime
, java.time.LocalDateTime
and java.time.LocalDate
types instead of these legacy datetime types.
For WITH TIME ZONE
types, the session time zone has no effect on the conversion to the legacy JDBC date/time types: the offset date/time is converted to epoch milliseconds and used to construct these legacy types directly.
Executing SET TIME ZONE <zone name>
statements after connect will change the session time zone on the server, but Jaybird will continue to use the session time zone set in the connection property for these conversions.
Although not defined in JDBC (or ODBC), Jaybird has added a non-standard extension to the CONVERT
JDBC escape to allow conversion to the time zone types.
In addition to the standard-defined types, it also supports the type names TIME_WITH_TIME_ZONE
, TIME_WITH_TIMEZONE
, TIMESTAMP_WITH_TIME_ZONE
and TIMESTAMP_WITH_TIMEZONE
(and the same with the SQL_
prefix).
Time zone fields do not support java.time.LocalDate
, java.time.LocalTime
, java.time.LocalDateTime
.
Firebird 4 redefines CURRENT_TIME
and CURRENT_TIMESTAMP
to return a WITH TIME ZONE
type. Use LOCALTIME
and LOCALTIMESTAMP
(introduced in Firebird 3.0.4) if you want to ensure a WITHOUT TIME ZONE
type is used.
The database metadata will always return JDBC 4.2 compatible information on time zone types, even on Java 7, and even when legacy time zone bind is set. For Java 7 compatibility the JDBC 4.2 java.sql.Types
constants TIME_WITH_TIMEZONE
and TIMESTAMP_WITH_TIMEZONE
are also defined in org.firebirdsql.jdbc.JaybirdTypeCodes
.
The default sessionTimeZone
is set to the JVM default time zone, this may result in different application behavior for DATE
, TIME
and TIMESTAMP
, including values generated in triggers and default value clauses. To prevent this, either switch those types to a WITH TIME ZONE
type, or set the sessionTimeZone
to server
or to the actual time zone of the Firebird server.
As CURRENT_TIME
uses the session time zone, which usually is a named zone, use in combination with java.time.OffsetTime
can yield confusing results. For example, if the current date and time is '2020-07-01T14:51:00 Europe/Amsterdam', then retrieving CURRENT_TIME
as an OffsetTime
will return the value '14:51:00+01:00', and not '14:51:00+02:00'.
It is recommended to avoid CURRENT_TIME
and use CURRENT_TIMESTAMP
instead.
Overall, using TIME WITH TIME ZONE
with named zones is rather fragile and prone to interpretation errors. This is a result of how this is implemented in Firebird: values are stored at UTC with their offset or named zones, where derivation of the time in the named zone needs to use 2020-01-01 as the date for the time zone rules to apply.
We recommend avoiding TIME WITH TIME ZONE
where possible.
Support was added for Firebird 4 statement timeouts through java.sql.setQueryTimeout
. On Firebird 3 and earlier or a native connection with a Firebird 3 or earlier client library, the timeout is silently ignored.
This implementation supports a maximum timeout of 4294967 seconds. Larger values will be handled as if 0
(unlimited) was set. Firebird also has attachment level and global statement timeouts. This configuration governs the statement level statement timeout only. In practice, a more stringent timeout might be applied by this attachment level or global statement timeout.
Important: Query timeouts in Firebird 4 and higher have an important caveat: for result set producing statements, the timeout covers the time from execution start until the cursor is closed. This includes the time that Firebird waits for the application to fetch more rows. This means that if you execute a SELECT and take your time processing the results, the statement may be cancelled even when Firebird itself returns rows quickly.
See Firebird 4 release notes and documentation for more information.
Columns of type RDB$DB_KEY
are now identified as java.sql.Types.ROWID
, and getObject
on these columns will now return a java.sql.RowId
.
The getObject(int/String, Class)
methods support retrieval as java.sql.RowId
and org.firebirdsql.jdbc.FBRowId
; the object returned is the same type (org.firebirdsql.jdbc.FBRowId
) in both cases.
Updating row ids is not possible, so attempts to call updateRowId
or updateObject
on a RDB$DB_KEY
in an updatable result set will throw an SQLFeatureNotSupportedException
.
Unfortunately, this support does not extend to parameters, as parameters (e.g. in where RDB$DB_KEY = ?
) cannot be distinguished from parameters of a normal binary field (char character set octets
). To address this, binary fields now also accept values of type java.sql.RowId
on setRowId
and setObject
.
Support has also been added to DatabaseMetaData
:
getBestRowIdentifier
returns RDB$DB_KEY
if there is no primary key (existing functionality)getRowIdLifetime
now returns RowIdLifetime.ROWID_VALID_TRANSACTION
(even if dbkey_scope=1
has been specified!)getPseudoColumns
now returns RDB$DB_KEY
Other database metadata (eg getColumns
) will not list the RDB$DB_KEY
column, as it is a pseudo-column.
In result sets, Jaybird will now also automatically map request for columns by name RDB$DB_KEY
(case-insensitive) to DB_KEY
as Firebird automatically applies this alias for the RDB$DB_KEY
column(s) in a select-list.
Be aware that result set metadata will still report DB_KEY
as the column name and label.
The DatabaseMetaData.getPseudoColumns
method (introduced in JDBC 4.1) has now been implemented.
The JDBC API specifies this method as:
Retrieves a description of the pseudo or hidden columns available in a given table within the specified catalog and schema. Pseudo or hidden columns may not always be stored within a table and are not visible in a
ResultSet
unless they are specified in the query's outermostSELECT
list. Pseudo or hidden columns may not necessarily be able to be modified. If there are no pseudo or hidden columns, an emptyResultSet
is returned.
For Firebird 2.5 and earlier it will only report on RDB$DB_KEY
, for Firebird 3 and higher it will also report on RDB$RECORD_VERSION
.
The pseudo-column RDB$RECORD_VERSION
was introduced in Firebird 3, its value is the transaction that last updated the row.
The DatabaseMetaData.getVersionColumns
method has now been implemented.
The JDBC API specifies this method as:
Retrieves a description of a table's columns that are automatically updated when any value in a row is updated. They are unordered.
For Firebird 2.5 and earlier it will only report on RDB$DB_KEY
, for Firebird 3 and higher it will also report on RDB$RECORD_VERSION
.
The pseudo-column RDB$RECORD_VERSION
was introduced in Firebird 3, its value is the transaction that last updated the row.
Jaybird only returns pseudo-column as version columns, so 'last updated' columns updated by a trigger, calculated columns, or other forms of change tracking are not reported by this method.
The DatabaseMetaData.getFunctions
method has now been implemented.
The JDBC API specifies this method as:
Retrieves a description of the system and user functions available in the given catalog.
The implementation only returns functions that are available from the RDB$FUNCTIONS
table. This means that the built-in functions are not included in the result of this method.
For Firebird 3 and higher, the result includes native UDF, PSQL and UDR functions. The result does not include functions defined in packages as JDBC does not provide support for packages.
Jaybird provides additional columns with Firebird specific information. As these columns are not defined by JDBC, they may change position when JDBC adds new columns. We recommend retrieving these columns by name.
The additional columns are:
JB_FUNCTION_SOURCE
- Source of Firebird 3+ PSQL function, this is the part after the AS
clause.JB_FUNCTION_KIND
- Kind of function, one of "UDF"
, "PSQL"
(Firebird 3+) or "UDR"
(Firebird 3+)JB_MODULE_NAME
- Value of RDB$MODULE_NAME
, is null
for PSQL functionsJB_ENTRYPOINT
- Value of RDB$ENTRYPOINT
, is null
for PSQL functionsJB_ENGINE_NAME
- Value of RDB$ENGINE_NAME
, is null
for UDF and PSQL functionsThe DatabaseMetaData.getFunctionColumns
method has now been implemented.
The JDBC API specifies this method as:
Retrieves a description of the given catalog's system or user function parameters and return type.
The implementation only returns columns of functions that are available from the RDB$FUNCTIONS
table. This means that the built-in functions are not included in the result of this method.
For Firebird 3 and higher, the result includes native UDF, PSQL and UDR functions. The result does not include functions defined in packages as JDBC does not provide support for packages.
Where Firebird provides no column name, Jaybird generates one by combining the string PARAM_
with the value of RDB$ARGUMENT_POSITION
. Names are not available for the parameters of legacy UDF functions, and for the return value of any function.
Revised support for JDBC function escapes with optional parameters, and added support for a number of previously unsupported functions or options.
If you only target Firebird, then we suggest you do not use JDBC function escapes.
DEGREES(number)
- Degrees in number radians; implemented as ((number)*180.0/PI())
RADIANS(number)
- Radians in number degrees; implemented as ((number)*PI()/180.0)
QUARTER(date)
- Quarter of year for date; implemented as (1+(EXTRACT(MONTH FROM date)-1)/3)
TIMESTAMPADD(interval, count, timestamp)
- Implemented using DATEADD
with the following caveats:
SQL_TSI_FRAC_SECOND
unit is nanoseconds and will be simulated by using MILLISECOND
and the count multiplied by 1.0e-6
to convert the value to milliseconds.SQL_TSI_QUARTER
will be simulated by using MONTH
and count multiplied by 3.DATEADD
.TIMESTAMPDIFF(interval, timestamp1, timestamp2)
- Implemented using DATEDIFF
with the following caveats:
SQL_TSI_FRAC_SECOND
unit is nanoseconds and will be simulated by using MILLISECOND
and the result multiplied by 1.0e6
and cast to BIGINT
to convert the value to nanoseconds.SQL_TSI_QUARTER
will be simulated by using MONTH
and the result divided by 3.BIGINT
, not INTEGER
.DATEDIFF
.DAYNAME(date)
- A character string representing the day component of date; implemented to always return English day names (ie Sunday, Monday, etc)
MONTHNAME(date)
- A character string representing the month component of date; implemented to always return English month names (ie January, February, etc)
DATABASE()
Name of the database; implemented as RDB$GET_CONTEXT('SYSTEM', 'DB_NAME')
CHAR_LENGTH(string[, CHARACTERS|OCTETS])
- The optional second parameter with CHARACTERS
or OCTETS
is now supported.
Absence of second parameter, or CHARACTERS
maps to CHAR_LENGTH
, OCTETS
maps to OCTET_LENGTH
CHARACTER_LENGTH(string[, CHARACTERS|OCTETS])
- see CHAR_LENGTH
CONCAT(string1, string2)
- Added parentheses around expression to prevent ambiguity or incorrect evaluation order.
LENGTH(string[, CHARACTERS|OCTETS])
- The optional second parameter with CHARACTERS
or OCTETS
is now supported.
The JDBC specification specifies Number of characters in string, excluding trailing blanks, we right-trim (TRIM(TRAILING FROM value)
) the string before passing the value to either CHAR_LENGTH
or OCTETS_LENGTH
. As a result, the interpretation of what is a blank depends on the type of value. Is the value a normal (VAR)CHAR
(non-octets), then the blank is space (0x20), for a VAR(CHAR)CHARACTER SET OCTETS / (VAR)BINARY
the blank is NUL (0x00). This means that the optional CHARACTERS|OCTETS
parameter has no influence on which blanks are trimmed, but only whether we count characters or bytes after trimming.
LOCATE(string1, string2[, start])
- The third parameter start is now optional.
POSITION(substring IN string[, CHARACTERS|OCTETS])
- The optional second parameter is now supported if CHARACTERS
. OCTETS
is not supported.
CONVERT(value, SQLtype)
- See Improved CONVERT support.
In Jaybird 3, CONVERT(value, SQLtype)
would map directly to CAST(value as SQLtype)
, we have improved support to better conform to the JDBC requirements, with some caveats:
SQL_<datatype>
and <datatype>
mapping is now supported(SQL_)VARCHAR
, (SQL_)NVARCHAR
(and value not a parameter (?
)) without explicit length is converted using TRIM(TRAILING FROM value)
, which means the result is VARCHAR
except for blobs where this will result in a blob; national character set will be lost. If value is a parameter (?
), and no length is specified, then a length of 50 will be applied (cast to (N)VARCHAR(50)
).(SQL_)CHAR
, (SQL_)NCHAR
without explicit length will be cast to (N)CHAR(50)
(SQL_)BINARY
, and (SQL_)VARBINARY
without explicit length will be cast to (VAR)CHAR(50) CHARACTER SET OCTETS
. With explicit length, CHARACTER SET OCTETS
is appended.(SQL_)LONGVARCHAR
, (SQL_)LONGNVARCHAR
, (SQL_)CLOB
, (SQL_)NCLOB
will be cast to BLOB SUB_TYPE TEXT
, national character set will be lost(SQL_)LONGVARBINARY
, (SQL_)BLOB
will be cast to BLOB SUB_TYPE BINARY
(SQL_)TINYINT
is mapped to SMALLINT
(SQL_)ROWID
is not supported as length of DB_KEY
values depend on the context(SQL_)DECIMAL
and (SQL_)NUMERIC
without precision and scale are passed as is, in current Firebird versions, this means the value will be equivalent to DECIMAL(9,0)
(which is equivalent to INTEGER
)Historically, the JDBC protocols supported by Jaybird have used the prefix jdbc:firebirdsql:
. We have now added support for jdbc:firebird:
as an alternative prefix. This prefix was previously only supported in the OpenOffice.org/LibreOffice pure Java variant.
Jaybird now supports the following URL prefixes (or JDBC protocols):
jdbc:firebirdsql:
jdbc:firebirdsql:java
jdbc:firebird:
(new)jdbc:firebird:java:
(new)jdbc:firebirdsql:native:
jdbc:firebird:native:
(new)jdbc:firebirdsql:embedded:
jdbc:firebird:embedded:
(new)jdbc:firebirdsql:local:
jdbc:firebird:local:
(new)jdbc:firebird:oo:
jdbc:firebirdsql:oo:
Jaybird now supports UTF-8 URL encoded values (and keys) in the query part of the JDBC URL.
As a result of this change, the following previously unsupported characters can be used in a connection property value when escaped:
;
escaped as %3B
&
escaped as %26
URL encoding can also be used to encode any unicode character in the query string. Jaybird will always use UTF-8 for decoding.
This change introduces the following backwards incompatibilities:
+
in the query part now means space (0x20), so occurrences of +
(plus) need to be escaped as %2B
; make sure to do this for base64 encoded values of dbCryptConfig
%
in the query part now introduces an escape, so occurrences of %
(percent) need to be escaped as %25
Invalid URL encoded values will now throw a SQLNonTransientConnectionException
.
The reason for this change is that the new dataTypeBind
connection property requires semicolon-separated values, but Jaybird supports semicolon-separated key/value connection properties in the query part. To be able to support this new property in the connection string, we had to introduce URL encoding.
This change only applies to the JDBC URL part after the first ?
. This change does not apply to connection properties set through java.util.Properties
or on a javax.sql.DataSource
.
Support for generated keys generation was improved with the following changes.
A new connection property generatedKeysEnabled
(alias generated_keys_enabled
) has been added that allows the behaviour of generated keys support to be configured. Also available on data sources.
This property supports the following values (case-insensitive):
default
: default behaviour to enable generated keys for statement types with RETURNING
clause in the connected Firebird version (absence of this property, null
or empty string implies default
). This corresponds to the existing behaviour.disabled
: disable support. Attempts to use generated keys methods other than using Statement.NO_GENERATED_KEYS
will throw a SQLFeatureNotSupportedException
.ignored
: ignore generated keys support. Attempts to use generated keys methods will not attempt to detect generated keys support and execute as if the statement generates no keys. The Statement.getGeneratedKeys()
method will always return an empty result set. This behaviour is equivalent to using the non-generated keys methods.For disabled
and ignored
, DatabaseMetaData.supportsGetGeneratedKeys
will report false
.
Because of the behaviour specified in the next section, typos in property values will behave as ignored
(e.g. using generatedKeysEnabled=disable
instead of disabled
will behave as ignored
).
This last option allows you to selectively enable support for generated keys. For example, generatedKeysEnabled=insert
will only enable it for insert
while ignoring it for all other statement types. Statement types that are not enabled will behave as if they generate no keys and will execute normally. For these statement types, Statement.getGeneratedKeys()
will return an empty result set.
Possible statement type values (case-insensitive) are:
insert
update
delete
update_or_insert
merge
Invalid values will be ignored. If none of the specified statement types are supported by Firebird, it will behave as ignored
6.
Some examples:
jdbc:firebird://localhost/testdb?generatedKeysEnabled=insert
will only enable insert supportjdbc:firebird://localhost/testdb?generatedKeysEnabled=merge
will only enable merge support, but only on Firebird 3 and higher, for Firebird 2.5 this will behave as ignored
given the lack of RETURNING
support for merge in that version.jdbc:firebird://localhost/testdb?generatedKeysEnabled=insert,update
will only enable insert and update supportThis feature can be used to circumvent issues with frameworks or tools that always use generated keys methods for prepare or execution. For example, with UPDATE
statements that touch multiple records and - given the Firebird limitations for RETURNING
- produce the error "multiple rows in singleton select".
Firebird 3 added RETURNING
support for MERGE
, this support is now available in Jaybird.
Firebird 4 added a RETURNING *
('returning all') clause which returns all columns of the row affected by a DML statement. When connected to Firebird 4, the Statement.RETURN_GENERATED_KEYS
methods will no longer query the database metadata for the column names, but instead append a RETURNING *
clause.
Artificial testing by repeatedly executing the same insert statement using Statement.execute(insert, Statement.RETURN_GENERATED_KEYS)
shows a performance improvement of roughly 200% (even 400% on localhost).
The grammar used by the generated keys support has been simplified to avoid issues with complex statements not being identified as types that generate keys, and to reduce maintenance.
The downside is that this may introduce different behaviour, as statements previously not identified as generated keys types, now could be identified as generated keys. Especially with DML other than INSERT
, or INSERT .. SELECT ..
this could result in error "multiple rows in singleton select" as the RETURNING
clause is currently only supported for statements that modify a single row.
You will either need to change the execution of these statements to use the normal execute/prepare or use Statement.NO_GENERATED_KEYS
. Alternatively ignore or only selectively enable generated keys support, see Configuration of generated keys behaviour above.
See Changes to behaviour of generated keys in Stricter JDBC compliance.
Experimental feature
Operation monitoring is an experimental feature that allows an application to monitor and - in a limited fashion - control driver operations. This feature is exposed as the interfaces org.firebirdsql.gds.ng.monitor.OperationAware
and org.firebirdsql.gds.ng.monitor.Operation
.
An application can implement OperationAware
and register it through org.firebirdsql.gds.ng.OperationMonitor.initOperationAware(OperationAware)
. When a SecurityManager
is active, registering an OperationAware
will require the SQLPermission
with name org.firebirdsql.jaybird.initOperationAware
. Only one instance can be registered at a time. Setting null
will clear the current instance.
Once registered, the OperationAware
instance will be notified of operation start and end through the methods startOperation(Operation)
and endOperation(Operation)
. These methods are called on the thread performing the operation, so implementations must complete these methods as soon as possible to prevent performance problems in the driver.
The Operation
instance exposes the type of operation through getType()
and allows the operation to be cancelled. Cancellation is only possible as long as the operation is active. Attempting to cancel when the operation is complete will throw an SQLException
.
This feature is experimental and its API may be removed or changed, and operation types may be added or removed in point releases.
See also jdp-2019-06: Ability to monitor driver operations
This feature was contributed by Vasiliy Yashkov.
Added in Jaybird 4.0.8, backported from Jaybird 5.
The "generated keys" parser has been replaced. This parser is used to detect statement types, the table name, and presence or absence of a RETURNING
clause. The new parser has no external dependencies, so Jaybird no longer depends on the ANTLR runtime (org.antlr:antlr4-runtime
).
As a result of this change, it is possible that detection of some statements has changed, especially detection of the presence of a RETURNING
clause. Please report any incorrect changes in detection on the firebird-java list or on https://github.com/FirebirdSQL/jaybird/issues.
If you were relying on disabling generated keys support by excluding the antlr4-runtime library from the classpath, you will now need to explicitly disable it. Disabling generated keys can be done using the connection property generatedKeysEnabled
with value disabled
, or ignored
if you don't want an exception thrown when calling a generated-keys-related execute or prepare method.
This backport was prompted by the fact that Jaybird 4 still used ANTLR 4.7.2, while ANTLR 4.10 has introduced a backwards incompatible change, which, for example, causes problems when using Jaybird with Hibernate 6. Instead of upgrading ANTLR, we decided that replacing the parser was less invasive as it was less likely to introduce compatibility issues for users depending on older versions of ANTLR.
Jaybird 4 contains a number of changes that might break existing applications.
See also Compatibility changes for details.
The distribution zip no longer includes the jaybird-4.0.10.javaXX.rar. This file was an example JCA Resource Archive.
JCA support will be removed entirely in Jaybird 5. See also Dropping JCA support.
Added support for Firebird 4 page size 32768 (32KB) in FBManager
and backup managers (back-ported to Jaybird 3.0.5) (JDBC-468)
Changed: The value returned by ResultSetMetaData.getColumnDisplaySize
was revised for REAL
/FLOAT
and DOUBLE PRECISION
to take scientific notation into account (JDBC-514)
Fixed: Database metadata pattern parameters now allow the pattern escape character (\
) to occur unescaped, this means that patterns A\B
and A\\B
will both match a value of A\B
. This complies with the (undocumented) JDBC expectation that patterns follow the ODBC requirements for pattern value arguments (JDBC-562)
Upgraded antlr-runtime used for generated keys support from 4.7 to 4.7.2.
The grammar generated for version 4.7.2 should still run on 4.7, but we suggest that you upgrade this dependency.
Improvement: Added FBManager.setDefaultCharacterSet
to set default database character set during database creation (JDBC-541)
New Feature: Support for Firebird 3 'explained' (detailed) execution plan (JDBC-574)
Adds FirebirdStatement.getLastExplainedExecutionPlan()
, FirebirdPreparedStatement.getExplainedExecutionPlan()
, and Firebird ResultSet.getExplainedExecutionPlan()
.
This feature was contributed by Vasiliy Yashkov.
Upgraded jna library used for native/embedded from 4.4 to 5.5 (JDBC-509
The pull request to upgrade (from 4.4 to 5.2) was contributed by Julien Nabet.
Native libraries will now be disposed on application exit (JDBC-519)
On JVM exit or - if deployed inside a WAR - servlet context destroy (tested on Tomcat), Jaybird will call fb_shutdown
on any loaded native libraries and dispose the JNA handle to the native library. This should prevent crashes (e.g. access violation / 0xc0000005 error on Windows) on library unload if there were still embedded connections open.
Given the potential for bugs or timing issues with this feature, it can be disabled with system property org.firebirdsql.nativeResourceShutdownDisabled
set to true
. This property must be set before Jaybird is loaded, preferably on the Java command line.
See Removal of deprecated classes, packages and methods in Compatibility changes for more details.
Jaybird 4 introduces some changes in compatibility and announces future breaking changes.
The list might not be complete, if you notice a difference in behavior that is not listed, please report it as bug. It might have been a change we forgot to document, but it could just as well be an implementation bug.
Support for Firebird 2.0 and 2.1 has been dropped in Jaybird 4. In general, we expect the driver to remain functional, but chances are certain metadata (eg DatabaseMetaData
) will break if we use features introduced in newer versions.
In general, we will no longer fix issues that only occur with Firebird 2.1 or earlier.
As a result of changes in FBDatabaseMetaData
, most result set producing methods will no longer work with Firebird 1.5 or earlier (unsupported since Jaybird 3).
The pure Java protocol in Jaybird will - by default - no longer try the Legacy_Auth
plugin when connecting to Firebird 3 or higher.
See Default authentication plugins for more information.
Using Jaybird 4 with Firebird 4 on a machine with a different time zone than the JVM, values of CURRENT_TIMESTAMP
, LOCALTIMESTAMP
, CURRENT_TIME
and LOCALTIME
will result in different values compared to Jaybird 4 with Firebird 3 (or Jaybird 3 or earlier with Firebird 4). This is caused by Jaybird 4 setting the session time zone to the JVM default time zone. As a result values will be in the time zone of the JVM and not the Firebird server time zone.
To revert to the previous behaviour, explicitly set sessionTimeZone=server
. See Connection property sessionTimeZone for more information.
With the introduction of JDBC RowId support, RDB$DB_KEY
columns are no longer identified as java.sql.Types.BINARY
, but as java.sql.Types.ROWID
. The column will behave in a backwards-compatible manner as a binary field, except getObject
, which will return a java.sql.RowId
instead.
Unfortunately this does not apply to parameters, see also JDBC RowId support.
Due to the method of identification, real columns of type char character set octets
with the name DB_KEY
will also be identified as a ROWID
column.
Previously, the Jaybird implementation of DatabaseMetaData.getBestRowIdentifier
used the scope
parameter to populate the SCOPE
column of its result set, instead of using it to filter on the required scope.
This has been changed to instead filter on scope
. In this implementation, the columns of the primary key are considered the best row identifier, with scope bestRowSession
. It will be returned for all values of scope
.
If a table does not have a primary key, the RDB$DB_KEY
is considered the second-best alternative, with scope bestRowTransaction
. It will only be returned for scopes bestRowTemporary
and bestRowTransaction
. See also JDBC RowId support.
If you are currently using DatabaseMetaData.getBestRowIdentifier
with scope
value DatabaseMetaData.bestRowSession
, consider if you need to use bestRowTransaction
instead.
If you are relying on the SCOPE
column containing the value for the requested scope, change your logic to remove that dependency.
Firebird 4 introduced support for SQL standard FLOAT(p)
with 1 <= p <= 24 a synonym of FLOAT
, and 25 <= p <= 53 as synonym of DOUBLE PRECISION
. This precision is expressed in binary digits (or radix 2).
To bring the metadata reported by Jaybird in line for this change, on Firebird 4 and higher, we now report the precision for FLOAT
and DOUBLE PRECISION
in binary digits instead of decimal digits.
For example for FLOAT
precision, on Firebird 3 and lower Jaybird 4 returns 7
, and 24
on Firebird 4 and higher. For DOUBLE PRECISION
precision, on Firebird 3 and lower Jaybird 4 returns 15
, and 53
on Firebird 4 and higher. In addition, the radix reported in the metadata for these types will be 2 instead of 10 on Firebird 4 and higher.
This change affects
DatabaseMetaData.getColumns
(columns COLUMN_SIZE
and NUM_PREC_RADIX
),DatabaseMetaData.getProcedureColumns
(columns PRECISION
and RADIX
),DatabaseMetaData.getTypeInfo
(columns PRECISION
and NUM_PREC_RADIX
),ParameterMetaData.getPrecison
,ResultSetMetaData.getPrecision
.With the introduction of URL encoding for the query part of the JDBC URL, the use of characters +
and %
in the query part of a JDBC URL now have different meaning and can lead to errors or unexpected results. This is especially relevant for base64 encoded values of dbCryptConfig
.
See URL encoding in query part of JDBC URL for more information.
In Jaybird 4 a number of changes were made for stricter compliance to the JDBC specification.
In previous versions of Jaybird, the column indexes (passed to Connection.prepareStatement
and Statement.executeXXX
methods accepting an int[]
) where sorted. The columns in the generated RETURNING
clause where in ascending ordinal order.
In Jaybird 4 this sort is no longer applied, so columns will be in the order specified by the array. If you were previously relying on this behaviour, you will need to sort the array yourself or correct the indexes used in ResultSet.getXXX(int)
.
The various generated keys Connection.prepareStatement
and Statement.executeXXX
methods accepting an int[]
or String[]
array no longer accept a null or empty array if the statement is a statement that generates keys. Instead, an exception is thrown with message "Generated keys array (columnIndexes|columnNames) was empty or null. A non-empty array is required."
This change does not apply for statements that already explicitly include a RETURNING
clause or for non-generated keys statements. In those cases, the array is ignored.
In addition, the methods accepting an int[]
array no longer ignore invalid column indexes, and instead throw an exception with message "Generated keys column position <position> does not exist for table <tablename>. Check DatabaseMetaData.getColumns (column ORDINAL_POSITION) for valid values."
If you were previously relying on this behaviour, you will need to remove invalid column indexes from the array.
This change does not apply for statements that already explicitly include a RETURNING
clause or for non-generated keys statements. In those cases, the array is ignored.
If generated keys methods using Statement.RETURN_GENERATED_KEYS
or int[]
cannot find any columns for a table, an exception is now thrown with message "No columns were found for table <tablename> to build RETURNING clause. The table does not exist.". Previously this executed as if the statement generated no keys and deferred to Firebird to return a "Table unknown" error.
On Firebird 4, using Statement.RETURN_GENERATED_KEYS
will continue to produce a "Table unknown" error as it does not need to query the metadata, and instead defers this to Firebird using RETURNING *
.
This change does not apply for statements that already explicitly include a RETURNING
clause or for non-generated keys statements.
The generated keys grammar was changed, this may in some cases change the detection of statement types and execute statements previously generated as normal statement to be enhanced with a RETURNING
clause.
This is probably only a theoretical concern (we don't know of actual cases where detection changed).
See also Generated keys grammar simplification.
The definition of getProcedures
and getProcedureColumns
does not offer a way to return information on procedures in packages without breaking tools or applications that rely on the JDBC standard behaviour. To avoid these problems, stored procedures in packages are no longer returned from these methods.
The way Jaybird handled this previously was 'by accident', and the information returned was not enough to correctly call the procedure as the package name was not included.
Character mapping has been removed. Character mapping (also known as translation) was a feature that allowed to remap certain characters when encoding/decoding strings. This could be used to address incompatibilities or incorrect mappings in the character set implementation of certain platforms (apparently HPUX was affected).
If you do need this feature, then you will need to use a custom encoding implementation.
Connection property useTranslation
(and it's alias mapping_path
) will no longer be available. As part of this change, the following parts of the implementation have been removed (note that most are internal to Jaybird):
org.firebirdsql.encodings.CharacterTranslator
will be removed entirelyDatatypeCoder#encodeString(String value, String javaEncoding, String mappingPath)
DatatypeCoder#encodeString(String value, Encoding encoding, String mappingPath)
DatatypeCoder#decodeString(byte[] value, String javaEncoding, String mappingPath)
DatatypeCoder#decodeString(byte[] value, Encoding encoding, String mappingPath)
Encoding#withTranslation(CharacterTranslator translator)
EncodingFactory#getEncoding(String encoding, String mappingPath)
EncodingFactory#getEncoding(Charset charset, String mappingPath)
FirebirdConnectionProperties#setUseTranslation(String translationPath)
(and on data sources)FirebirdConnectionProperties#getUseTranslation
(and on data sources)IEncodingFactory#getCharacterTranslator(String mappingPath)
Since Jaybird 4.0.8.
Previously, support for generated keys depended on the presence of the antlr4-runtime library on the classpath. With New parser for generated keys handling, generated keys support is now always available.
See New parser for generated keys handling for information on disabling or ignoring generated keys support if you relied on this behaviour.
The following array constants in FBDatabaseMetaData
have been made private or have been removed to avoid unintended side effects of modification:
ALL_TYPES_2_5
ALL_TYPES_2_1
ALL_TYPES
Instead, use DatabaseMetaData.getTableTypes()
(which returns a ResultSet
), or FirebirdDatabaseMetaData.getTableTypeNames()
(which returns a String[]
).
To access getTableTypeNames()
, the DatabaseMetaData
needs to be unwrapped to FirebirdDatabaseMetaData
The following connection properties (and equivalent data source properties) have been removed:
useTranslation
: See Removal of character mappingoctetsAsBytes
: Since Jaybird 3 octets is always handled as BINARY
noResultSetTracking
: Option does nothing since Jaybird 3paranoia_mode
: Option does nothing since Jaybird 2.2 (maybe earlier)The following deprecated methods have been removed in Jaybird 4:
GDSHelper.iscVaxInteger(byte[] buffer, int pos, int length)
use VaxEncoding.iscVaxInteger(byte[] buffer, int startPosition, int length)
instead.GDSHelper.iscVaxLong(byte[] buffer, int pos, int length)
use VaxEncoding.iscVaxLong(byte[] buffer, int startPosition, int length)
instead.MaintenanceManager.commitTransaction(int transactionId)
, use MaintenanceManager.commitTransaction(long transactionId)
instead.MaintenanceManager.rollbackTransaction(int transactionId)
, use MaintenanceManager.rollbackTransaction(long transactionId)
instead.ServiceRequestBufferImp#ServiceRequestBufferImp()
ServiceRequestBufferImp#ServiceRequestBufferImp(int taskIdentifier)
FBBlob#copyCharacterStream(Reader reader, long length, String encoding)
FBBlob#copyCharacterStream(Reader reader, String encoding)
The following classes have been removed in Jaybird 4:
org.firebirdsql.gds.ExceptionListener
, use org.firebirdsql.gds.ng.listeners.ExceptionListener
org.firebirdsql.pool.FBSimpleDataSource
, use org.firebirdsql.ds.FBSimpleDataSource
The following constants have been removed in Jaybird 4:
SQL_STATE_*
constants in FBSQLException
, FBResourceTransactionException
, FBResourceException
, and FBDriverNotCapableException
have been removed. Use equivalent constants in org.firebirdsql.jdbc.SQLStateConstants
.With Jaybird 5 the following breaking changes will be introduced.
Jaybird 5 will drop support for Java 7.
Jaybird 5 will continue to support Java 8.
(This section exists because in previous versions it announced tentative removal of Java 8 support in Jaybird 5.)
Jaybird is currently built around a JCA (Java Connector Architecture) implementation. As such, it is both a JDBC driver and a JCA driver. The current structure requires a dependency on JCA for non-JCA usage.
We will remove support for JCA entirely in Jaybird 5 to simplify the implementation. The package org.firebirdsql.jca
will be removed entirely.
If you are currently using Jaybird as a JCA driver, please let us know on the Firebird-Java mailing list. We may reconsider this decision and instead restructure Jaybird so the dependency on JCA is only needed when Jaybird is used as a JCA driver.
The following methods will be removed in Jaybird 5:
MaintenanceManager.listLimboTransactions()
, use MaintenanceManager.limboTransactionsAsList()
or MaintenanceManager.getLimboTransactions()
instead.TraceManager.loadConfigurationFromFile(String)
, use standard Java functionality like new String(Files.readAllBytes(Paths.get(fileName)), <charset>)
FBDatabaseMetaData.hasNoWildcards(String pattern)
FBDatabaseMetaData.stripEscape(String pattern)
StatementParser.parseInsertStatement(String sql)
, use StatementParser.parseStatement(String sql)
FbStatement.getFieldDescriptor()
, use FbStatement.getRowDescriptor()
AbstractFbStatement.setFieldDescriptor(RowDescriptor fieldDescriptor)
, use AbstractFbStatement.setRowDescriptor(RowDescriptor rowDescriptor)
FBField.isType(FieldDescriptor, int)
, use JdbcTypeConverter.isJdbcType(FieldDescriptor, int)
The following classes will be removed in Jaybird 5:
FBMissingParameterException
, exception is no longer used.The following constants will be removed in Jaybird 5:
SQL_STATE_*
constants in FBSQLParseException
will be removed. Use equivalent constants in org.firebirdsql.jdbc.SQLStateConstants
.DatabaseParameterBufferExtension.EXTENSION_PARAMETERS
will be removed. There is no official replacement as this should be considered an implementation detail. It is possible that DatabaseParameterBufferExtension
will be removed entirely.Jaybird 4 and earlier have support to map JDBC function escapes to UDFs from ib_udf
instead of built-in function using the boolean connection property useStandarUdf
[sic].
Given recent Firebird versions have significantly improved support for built-in functions, and UDFs are now deprecated, this option will be removed in Jaybird 5.
Jaybird uses JNA to access the client library. If you want to use the Type 2 driver, or Firebird embedded, then you need to include jna-5.5.0.jar
on the classpath.
When using Maven, you need to specify the dependency on JNA yourself, as we don't depend on it by default (it is specified as an optional dependency):
<dependency>
<groupId>net.java.dev.jna</groupId>
<artifactId>jna</artifactId>
<version>5.5.0</artifactId>
</dependency>
The fbclient.dll
, fbembed.dll
, libfbclient.so
, or libfbembed.so
need to be on the path, or the location needs to be specified in the system property jna.library.path
(as an absolute or relative path to the directory/directories containing the library file(s)).
For Windows and Linux, you can add the org.firebirdsql.jdbc:fbclient
dependency on your classpath to provide the native libraries for the native
and local
protocol. Be aware that this dependency does not support embedded
.
<dependency>
<groupId>org.firebirdsql.jdbc</groupId>
<artifactId>fbclient</artifactId>
<version>4.0.3.0</artifactId>
</dependency>
In the future we will move the Type 2 support to a separate library and provide JNA-compatible jars that provide the embedded libraries of a specific Firebird version.
The support for Java 21 comes with one caveat: some tests weren't run due to lack of support for Java 21 in test libraries used for testing Jaybird, and we did not want to invest time to upgrade or replace those libraries.↩︎
The support for Java 21 comes with one caveat: some tests weren't run due to lack of support for Java 21 in test libraries used for testing Jaybird, and we did not want to invest time to upgrade or replace those libraries.↩︎
Internally SrpNNN
continues to use SHA-1, only the client-proof applies the SHA-NNN hash. See also firebird#6051/CORE-5788.↩︎
The authPlugins
values can be separated by commas, spaces, tabs, or semi-colons. The semi-colon should not be used in a JDBC URL as there the semi-colon is a separator between connection properties.↩︎
The DECFLOAT
decimal format stores values as sign, integral number with 16 or 34 digits, and an exponent. This is similar to
java.math.BigDecimal
, but instead of an exponent, that uses the concept scale
, where scale = -1 * exponent
.↩︎
This is not the case for the unsupported Firebird 1.0 and 1.5 versions. There this will behave similar to disabled
, and you will need to explicitly specify ignored
instead to get this behaviour.↩︎