neo-activerecord-jdbc-adapter 5.0.pre2

Sign up to get free protection for your applications and to get access to all the features.
Files changed (187) hide show
  1. checksums.yaml +7 -0
  2. data/.gitignore +33 -0
  3. data/.travis.yml +438 -0
  4. data/.yardopts +4 -0
  5. data/Appraisals +41 -0
  6. data/CONTRIBUTING.md +44 -0
  7. data/Gemfile +62 -0
  8. data/History.md +1191 -0
  9. data/LICENSE.txt +25 -0
  10. data/README.md +266 -0
  11. data/RUNNING_TESTS.md +88 -0
  12. data/Rakefile +100 -0
  13. data/Rakefile.jdbc +20 -0
  14. data/activerecord-jdbc-adapter.gemspec +42 -0
  15. data/lib/active_record/connection_adapters/as400_adapter.rb +2 -0
  16. data/lib/active_record/connection_adapters/db2_adapter.rb +1 -0
  17. data/lib/active_record/connection_adapters/derby_adapter.rb +1 -0
  18. data/lib/active_record/connection_adapters/firebird_adapter.rb +1 -0
  19. data/lib/active_record/connection_adapters/h2_adapter.rb +1 -0
  20. data/lib/active_record/connection_adapters/hsqldb_adapter.rb +1 -0
  21. data/lib/active_record/connection_adapters/informix_adapter.rb +1 -0
  22. data/lib/active_record/connection_adapters/jdbc_adapter.rb +1 -0
  23. data/lib/active_record/connection_adapters/jndi_adapter.rb +1 -0
  24. data/lib/active_record/connection_adapters/mariadb_adapter.rb +1 -0
  25. data/lib/active_record/connection_adapters/mssql_adapter.rb +1 -0
  26. data/lib/active_record/connection_adapters/mysql2_adapter.rb +1 -0
  27. data/lib/active_record/connection_adapters/mysql_adapter.rb +1 -0
  28. data/lib/active_record/connection_adapters/oracle_adapter.rb +1 -0
  29. data/lib/active_record/connection_adapters/postgresql_adapter.rb +1 -0
  30. data/lib/active_record/connection_adapters/sqlite3_adapter.rb +1 -0
  31. data/lib/active_record/connection_adapters/sqlserver_adapter.rb +1 -0
  32. data/lib/activerecord-jdbc-adapter.rb +1 -0
  33. data/lib/arel/visitors/compat.rb +60 -0
  34. data/lib/arel/visitors/db2.rb +137 -0
  35. data/lib/arel/visitors/derby.rb +112 -0
  36. data/lib/arel/visitors/firebird.rb +79 -0
  37. data/lib/arel/visitors/h2.rb +25 -0
  38. data/lib/arel/visitors/hsqldb.rb +32 -0
  39. data/lib/arel/visitors/postgresql_jdbc.rb +6 -0
  40. data/lib/arel/visitors/sql_server.rb +225 -0
  41. data/lib/arel/visitors/sql_server/ng42.rb +293 -0
  42. data/lib/arjdbc.rb +19 -0
  43. data/lib/arjdbc/abstract/database_statements.rb +92 -0
  44. data/lib/arjdbc/abstract/transaction_support.rb +86 -0
  45. data/lib/arjdbc/common_jdbc_methods.rb +13 -0
  46. data/lib/arjdbc/db2.rb +4 -0
  47. data/lib/arjdbc/db2/adapter.rb +789 -0
  48. data/lib/arjdbc/db2/as400.rb +130 -0
  49. data/lib/arjdbc/db2/column.rb +167 -0
  50. data/lib/arjdbc/db2/connection_methods.rb +44 -0
  51. data/lib/arjdbc/derby.rb +3 -0
  52. data/lib/arjdbc/derby/active_record_patch.rb +13 -0
  53. data/lib/arjdbc/derby/adapter.rb +556 -0
  54. data/lib/arjdbc/derby/connection_methods.rb +20 -0
  55. data/lib/arjdbc/derby/schema_creation.rb +15 -0
  56. data/lib/arjdbc/discover.rb +115 -0
  57. data/lib/arjdbc/firebird.rb +4 -0
  58. data/lib/arjdbc/firebird/adapter.rb +434 -0
  59. data/lib/arjdbc/firebird/connection_methods.rb +23 -0
  60. data/lib/arjdbc/h2.rb +3 -0
  61. data/lib/arjdbc/h2/adapter.rb +303 -0
  62. data/lib/arjdbc/h2/connection_methods.rb +27 -0
  63. data/lib/arjdbc/hsqldb.rb +3 -0
  64. data/lib/arjdbc/hsqldb/adapter.rb +297 -0
  65. data/lib/arjdbc/hsqldb/connection_methods.rb +28 -0
  66. data/lib/arjdbc/hsqldb/explain_support.rb +35 -0
  67. data/lib/arjdbc/hsqldb/schema_creation.rb +11 -0
  68. data/lib/arjdbc/informix.rb +5 -0
  69. data/lib/arjdbc/informix/adapter.rb +162 -0
  70. data/lib/arjdbc/informix/connection_methods.rb +9 -0
  71. data/lib/arjdbc/jdbc.rb +59 -0
  72. data/lib/arjdbc/jdbc/adapter.rb +899 -0
  73. data/lib/arjdbc/jdbc/adapter_java.jar +0 -0
  74. data/lib/arjdbc/jdbc/adapter_require.rb +46 -0
  75. data/lib/arjdbc/jdbc/base_ext.rb +44 -0
  76. data/lib/arjdbc/jdbc/callbacks.rb +51 -0
  77. data/lib/arjdbc/jdbc/column.rb +97 -0
  78. data/lib/arjdbc/jdbc/connection.rb +133 -0
  79. data/lib/arjdbc/jdbc/connection_methods.rb +36 -0
  80. data/lib/arjdbc/jdbc/driver.rb +43 -0
  81. data/lib/arjdbc/jdbc/extension.rb +59 -0
  82. data/lib/arjdbc/jdbc/java.rb +15 -0
  83. data/lib/arjdbc/jdbc/jdbc.rake +4 -0
  84. data/lib/arjdbc/jdbc/quoted_primary_key.rb +28 -0
  85. data/lib/arjdbc/jdbc/railtie.rb +2 -0
  86. data/lib/arjdbc/jdbc/rake_tasks.rb +3 -0
  87. data/lib/arjdbc/jdbc/serialized_attributes_helper.rb +3 -0
  88. data/lib/arjdbc/jdbc/type_cast.rb +166 -0
  89. data/lib/arjdbc/jdbc/type_converter.rb +142 -0
  90. data/lib/arjdbc/mimer.rb +3 -0
  91. data/lib/arjdbc/mimer/adapter.rb +142 -0
  92. data/lib/arjdbc/mssql.rb +7 -0
  93. data/lib/arjdbc/mssql/adapter.rb +808 -0
  94. data/lib/arjdbc/mssql/column.rb +200 -0
  95. data/lib/arjdbc/mssql/connection_methods.rb +79 -0
  96. data/lib/arjdbc/mssql/explain_support.rb +99 -0
  97. data/lib/arjdbc/mssql/limit_helpers.rb +231 -0
  98. data/lib/arjdbc/mssql/lock_methods.rb +77 -0
  99. data/lib/arjdbc/mssql/types.rb +343 -0
  100. data/lib/arjdbc/mssql/utils.rb +82 -0
  101. data/lib/arjdbc/mysql.rb +3 -0
  102. data/lib/arjdbc/mysql/adapter.rb +1006 -0
  103. data/lib/arjdbc/mysql/bulk_change_table.rb +150 -0
  104. data/lib/arjdbc/mysql/column.rb +162 -0
  105. data/lib/arjdbc/mysql/connection_methods.rb +145 -0
  106. data/lib/arjdbc/mysql/explain_support.rb +82 -0
  107. data/lib/arjdbc/mysql/schema_creation.rb +58 -0
  108. data/lib/arjdbc/oracle.rb +4 -0
  109. data/lib/arjdbc/oracle/adapter.rb +952 -0
  110. data/lib/arjdbc/oracle/column.rb +126 -0
  111. data/lib/arjdbc/oracle/connection_methods.rb +21 -0
  112. data/lib/arjdbc/postgresql.rb +3 -0
  113. data/lib/arjdbc/postgresql/_bc_time_cast_patch.rb +21 -0
  114. data/lib/arjdbc/postgresql/adapter.rb +825 -0
  115. data/lib/arjdbc/postgresql/base/array_decoder.rb +26 -0
  116. data/lib/arjdbc/postgresql/base/array_encoder.rb +25 -0
  117. data/lib/arjdbc/postgresql/base/array_parser.rb +95 -0
  118. data/lib/arjdbc/postgresql/base/pgconn.rb +11 -0
  119. data/lib/arjdbc/postgresql/column.rb +51 -0
  120. data/lib/arjdbc/postgresql/connection_methods.rb +54 -0
  121. data/lib/arjdbc/postgresql/name.rb +24 -0
  122. data/lib/arjdbc/postgresql/oid_types.rb +178 -0
  123. data/lib/arjdbc/railtie.rb +11 -0
  124. data/lib/arjdbc/sqlite3.rb +3 -0
  125. data/lib/arjdbc/sqlite3/adapter.rb +703 -0
  126. data/lib/arjdbc/sqlite3/connection_methods.rb +40 -0
  127. data/lib/arjdbc/sybase.rb +2 -0
  128. data/lib/arjdbc/sybase/adapter.rb +47 -0
  129. data/lib/arjdbc/tasks.rb +13 -0
  130. data/lib/arjdbc/tasks/database_tasks.rb +54 -0
  131. data/lib/arjdbc/tasks/databases.rake +91 -0
  132. data/lib/arjdbc/tasks/databases3.rake +215 -0
  133. data/lib/arjdbc/tasks/databases4.rake +39 -0
  134. data/lib/arjdbc/tasks/db2_database_tasks.rb +104 -0
  135. data/lib/arjdbc/tasks/derby_database_tasks.rb +95 -0
  136. data/lib/arjdbc/tasks/h2_database_tasks.rb +31 -0
  137. data/lib/arjdbc/tasks/hsqldb_database_tasks.rb +70 -0
  138. data/lib/arjdbc/tasks/jdbc_database_tasks.rb +169 -0
  139. data/lib/arjdbc/tasks/mssql_database_tasks.rb +46 -0
  140. data/lib/arjdbc/tasks/oracle/enhanced_structure_dump.rb +297 -0
  141. data/lib/arjdbc/tasks/oracle_database_tasks.rb +65 -0
  142. data/lib/arjdbc/util/quoted_cache.rb +60 -0
  143. data/lib/arjdbc/util/serialized_attributes.rb +98 -0
  144. data/lib/arjdbc/util/table_copier.rb +110 -0
  145. data/lib/arjdbc/version.rb +8 -0
  146. data/lib/generators/jdbc/USAGE +9 -0
  147. data/lib/generators/jdbc/jdbc_generator.rb +17 -0
  148. data/lib/jdbc_adapter.rb +2 -0
  149. data/lib/jdbc_adapter/rake_tasks.rb +4 -0
  150. data/lib/jdbc_adapter/version.rb +4 -0
  151. data/pom.xml +114 -0
  152. data/rails_generators/jdbc_generator.rb +15 -0
  153. data/rails_generators/templates/config/initializers/jdbc.rb +10 -0
  154. data/rails_generators/templates/lib/tasks/jdbc.rake +11 -0
  155. data/rakelib/01-tomcat.rake +51 -0
  156. data/rakelib/02-test.rake +121 -0
  157. data/rakelib/bundler_ext.rb +11 -0
  158. data/rakelib/compile.rake +62 -0
  159. data/rakelib/db.rake +58 -0
  160. data/rakelib/rails.rake +75 -0
  161. data/src/java/arjdbc/ArJdbcModule.java +178 -0
  162. data/src/java/arjdbc/db2/DB2Module.java +71 -0
  163. data/src/java/arjdbc/db2/DB2RubyJdbcConnection.java +142 -0
  164. data/src/java/arjdbc/derby/DerbyModule.java +179 -0
  165. data/src/java/arjdbc/derby/DerbyRubyJdbcConnection.java +164 -0
  166. data/src/java/arjdbc/firebird/FirebirdRubyJdbcConnection.java +190 -0
  167. data/src/java/arjdbc/h2/H2Module.java +44 -0
  168. data/src/java/arjdbc/h2/H2RubyJdbcConnection.java +67 -0
  169. data/src/java/arjdbc/hsqldb/HSQLDBModule.java +68 -0
  170. data/src/java/arjdbc/informix/InformixRubyJdbcConnection.java +75 -0
  171. data/src/java/arjdbc/jdbc/AdapterJavaService.java +45 -0
  172. data/src/java/arjdbc/jdbc/Callable.java +44 -0
  173. data/src/java/arjdbc/jdbc/JdbcConnectionFactory.java +45 -0
  174. data/src/java/arjdbc/jdbc/RubyJdbcConnection.java +3616 -0
  175. data/src/java/arjdbc/jdbc/SQLBlock.java +54 -0
  176. data/src/java/arjdbc/mssql/MSSQLModule.java +102 -0
  177. data/src/java/arjdbc/mssql/MSSQLRubyJdbcConnection.java +195 -0
  178. data/src/java/arjdbc/mysql/MySQLModule.java +147 -0
  179. data/src/java/arjdbc/mysql/MySQLRubyJdbcConnection.java +397 -0
  180. data/src/java/arjdbc/oracle/OracleModule.java +75 -0
  181. data/src/java/arjdbc/oracle/OracleRubyJdbcConnection.java +465 -0
  182. data/src/java/arjdbc/postgresql/PostgreSQLRubyJdbcConnection.java +752 -0
  183. data/src/java/arjdbc/sqlite3/SQLite3Module.java +78 -0
  184. data/src/java/arjdbc/sqlite3/SQLite3RubyJdbcConnection.java +351 -0
  185. data/src/java/arjdbc/util/CallResultSet.java +826 -0
  186. data/src/java/arjdbc/util/QuotingUtils.java +111 -0
  187. metadata +255 -0
@@ -0,0 +1,46 @@
1
+ module ActiveRecord
2
+
3
+ if defined? ConnectionAdapters::ConnectionSpecification::Resolver # 4.0
4
+ ConnectionAdapters::ConnectionSpecification::Resolver
5
+ elsif defined? Base::ConnectionSpecification::Resolver # 3.2
6
+ Base::ConnectionSpecification::Resolver
7
+ else class << Base; self; end # 2.3, 3.0, 3.1 :
8
+ # def self.establish_connection ... on ActiveRecord::Base
9
+ end.class_eval do
10
+
11
+ # @private
12
+ def require(path)
13
+ # NOTE: we're inspecting resolver.spec connection resolution which does :
14
+ # `require "active_record/connection_adapters/#{spec[:adapter]}_adapter"`
15
+ # ...
16
+ # this {#require} method is only re-defined on a Resolver object and thus
17
+ # will not hurt performance - it will only be called for a few times (most
18
+ # likely once), this should still be fine for AR < 3.2 where this patch
19
+ # ends up on `class << ActiveRecord::Base` since models usually rely on
20
+ # Rails's auto-loading of (missing) constants and rarely use `require`.
21
+ #
22
+ # other alternative (to make sure we do not need to eager load AR built-in
23
+ # adapters) would be to mingle with the $LOAD_PATH which seems worse ...
24
+ case path
25
+ when 'active_record/connection_adapters/mysql_adapter'
26
+ $LOADED_FEATURES << 'active_record/connection_adapters/mysql_adapter.rb'
27
+ super('arjdbc/mysql')
28
+ when 'active_record/connection_adapters/mysql2_adapter'
29
+ $LOADED_FEATURES << 'active_record/connection_adapters/mysql2_adapter.rb'
30
+ super('arjdbc/mysql')
31
+ when 'active_record/connection_adapters/postgresql_adapter'
32
+ $LOADED_FEATURES << 'active_record/connection_adapters/postgresql_adapter.rb'
33
+ super('arjdbc/postgresql')
34
+ when 'active_record/connection_adapters/sqlite_adapter'
35
+ $LOADED_FEATURES << 'active_record/connection_adapters/sqlite_adapter.rb'
36
+ super('arjdbc/sqlite3')
37
+ when 'active_record/connection_adapters/sqlite3_adapter'
38
+ $LOADED_FEATURES << 'active_record/connection_adapters/sqlite3_adapter.rb'
39
+ super('arjdbc/sqlite3')
40
+ else super
41
+ end
42
+ end
43
+
44
+ end
45
+
46
+ end
@@ -0,0 +1,44 @@
1
+ module ActiveRecord
2
+ class << Base
3
+ m = Module.new do
4
+ # Allow adapters to provide their own {#reset_column_information} method.
5
+ # @note This only affects the current thread's connection.
6
+ def reset_column_information # :nodoc:
7
+ # invoke the adapter-specific reset_column_information method
8
+ connection.reset_column_information if connection.respond_to?(:reset_column_information)
9
+ super
10
+ end
11
+ end
12
+
13
+ self.prepend(m)
14
+ end
15
+
16
+ # Represents exceptions that have propagated up through the JDBC API.
17
+ class JDBCError < ActiveRecordError
18
+ # The vendor code or error number that came from the database.
19
+ # @note writer being used by the Java API
20
+ attr_accessor :errno
21
+ # The full Java SQLException object that was raised.
22
+ # @note writer being used by the Java API
23
+ attr_accessor :sql_exception
24
+
25
+ attr_reader :original_exception, :raw_backtrace
26
+
27
+ def initialize(message = nil, original_exception = nil) # $!
28
+ super(message)
29
+ @original_exception = original_exception
30
+ end
31
+
32
+ def set_backtrace(backtrace)
33
+ @raw_backtrace = backtrace
34
+ if nested = original_exception
35
+ backtrace = backtrace - (
36
+ nested.respond_to?(:raw_backtrace) ? nested.raw_backtrace : nested.backtrace )
37
+ backtrace << "#{nested.backtrace.first}: #{nested.message} (#{nested.class.name})"
38
+ backtrace += nested.backtrace[1..-1] || []
39
+ end
40
+ super(backtrace)
41
+ end
42
+
43
+ end
44
+ end
@@ -0,0 +1,51 @@
1
+ module ActiveRecord::ConnectionAdapters
2
+ module Jdbc
3
+ # ActiveRecord connection pool callbacks for JDBC.
4
+ # @see ActiveRecord::ConnectionAdapters::Jdbc::JndiConnectionPoolCallbacks
5
+ module ConnectionPoolCallbacks
6
+
7
+ def self.included(base)
8
+ if base.respond_to?(:set_callback) # Rails 3 callbacks
9
+ base.set_callback :checkin, :after, :on_checkin
10
+ base.set_callback :checkout, :before, :on_checkout
11
+ else
12
+ base.checkin :on_checkin
13
+ base.checkout :on_checkout
14
+ end
15
+ end
16
+
17
+ def on_checkin
18
+ # default implementation does nothing
19
+ end
20
+
21
+ def on_checkout
22
+ # default implementation does nothing
23
+ end
24
+
25
+ end
26
+ # JNDI specific connection pool callbacks that make sure the JNDI connection
27
+ # is disconnected on check-in and looked up (re-connected) on-checkout.
28
+ module JndiConnectionPoolCallbacks
29
+
30
+ def self.prepare(adapter, connection)
31
+ if adapter.is_a?(ConnectionPoolCallbacks) && connection.jndi?
32
+ adapter.extend self # extend JndiConnectionPoolCallbacks
33
+ connection.disconnect! # disconnect initial (JNDI) connection if any
34
+ end
35
+ end
36
+
37
+ def on_checkin
38
+ disconnect!
39
+ end
40
+
41
+ def on_checkout
42
+ reconnect!
43
+ end
44
+ end
45
+
46
+ end
47
+ # @deprecated use {ActiveRecord::ConnectionAdapters::Jdbc::ConnectionPoolCallbacks}
48
+ JdbcConnectionPoolCallbacks = Jdbc::ConnectionPoolCallbacks
49
+ # @deprecated use {ActiveRecord::ConnectionAdapters::Jdbc::JndiConnectionPoolCallbacks}
50
+ JndiConnectionPoolCallbacks = Jdbc::JndiConnectionPoolCallbacks
51
+ end
@@ -0,0 +1,97 @@
1
+ module ActiveRecord
2
+ module ConnectionAdapters
3
+ module Jdbc
4
+ autoload :TypeCast, 'arjdbc/jdbc/type_cast'
5
+ end
6
+ # The base class for all of {JdbcAdapter}'s returned columns.
7
+ # Instances of {JdbcColumn} will get extended with "column-spec" modules
8
+ # (similar to how {JdbcAdapter} gets spec modules in) if the adapter spec
9
+ # module provided a `column_selector` (matcher) method for it's database
10
+ # specific type.
11
+ # @see JdbcAdapter#jdbc_column_class
12
+ class JdbcColumn < Column
13
+ # @deprecated attribute writers will be removed in 1.4
14
+ attr_writer :limit, :precision # unless ArJdbc::AR42
15
+
16
+ def initialize(config, name, *args)
17
+ if self.class == JdbcColumn
18
+ # NOTE: extending classes do not want this if they do they shall call
19
+ call_discovered_column_callbacks(config) if config
20
+ default = args.shift
21
+ else # for extending classes allow ignoring first argument :
22
+ if ! config.nil? && ! config.is_a?(Hash)
23
+ default = name; name = config # initialize(name, default, *args)
24
+ else
25
+ default = args.shift
26
+ end
27
+ end
28
+
29
+ if ArJdbc::AR50
30
+ default = args[0].cast(default)
31
+
32
+ sql_type = args.delete_at(1)
33
+ type = args.delete_at(0)
34
+
35
+ args.unshift(SqlTypeMetadata.new(:sql_type => sql_type, :type => type))
36
+ elsif ArJdbc::AR42
37
+ default = args[0].type_cast_from_database(default)
38
+ else
39
+ default = default_value(default)
40
+ end
41
+
42
+ # super <= 4.1: (name, default, sql_type = nil, null = true)
43
+ # super >= 4.2: (name, default, cast_type, sql_type = nil, null = true)
44
+ # super >= 5.0: (name, default, sql_type_metadata = nil, null = true)
45
+
46
+ super(name, default, *args)
47
+ init_column(name, default, *args)
48
+ end
49
+
50
+ # Additional column initialization for sub-classes.
51
+ def init_column(*args); end
52
+
53
+ # Similar to `ActiveRecord`'s `extract_value_from_default(default)`.
54
+ # @return default value for a column (possibly extracted from driver value)
55
+ def default_value(value); value; end
56
+
57
+ protected
58
+
59
+ # @private
60
+ def call_discovered_column_callbacks(config)
61
+ dialect = (config[:dialect] || config[:driver]).to_s
62
+ for matcher, block in self.class.column_types
63
+ block.call(config, self) if matcher === dialect
64
+ end
65
+ end
66
+
67
+ public
68
+
69
+ # Returns the available column types
70
+ # @return [Hash] of (matcher, block) pairs
71
+ def self.column_types
72
+ types = {}
73
+ for mod in ::ArJdbc.modules
74
+ if mod.respond_to?(:column_selector)
75
+ sel = mod.column_selector # [ matcher, block ]
76
+ types[ sel[0] ] = sel[1]
77
+ end
78
+ end
79
+ types
80
+ end
81
+
82
+ class << self
83
+
84
+ include Jdbc::TypeCast if ::ActiveRecord::VERSION::STRING >= '4.2'
85
+
86
+ if ActiveRecord::VERSION::MAJOR > 3 && ActiveRecord::VERSION::STRING < '4.2'
87
+
88
+ # @private provides compatibility between AR 3.x/4.0 API
89
+ def string_to_date(value); value_to_date(value) end
90
+
91
+ end
92
+
93
+ end
94
+
95
+ end
96
+ end
97
+ end
@@ -0,0 +1,133 @@
1
+ module ActiveRecord
2
+ module ConnectionAdapters
3
+ # JDBC (connection) base class, custom adapters we support likely extend
4
+ # this class. For maximum performance most of this class and the sub-classes
5
+ # we ship are implemented in Java, check: *RubyJdbcConnection.java*
6
+ class JdbcConnection
7
+
8
+ # Initializer implemented in Ruby.
9
+ # @note second argument is mandatory, only optional for compatibility
10
+ def initialize(config, adapter = nil)
11
+ @config = config; @adapter = adapter
12
+ @connection = nil; @jndi = nil
13
+ # @stmts = {} # AR compatibility - statement cache not used
14
+ setup_connection_factory
15
+ init_connection # @see RubyJdbcConnection.init_connection
16
+ rescue Java::JavaSql::SQLException => e
17
+ e = e.cause if defined?(NativeException) && e.is_a?(NativeException) # JRuby-1.6.8
18
+ error = e.getMessage || e.getSQLState
19
+ error = error ? "#{e.java_class.name}: #{error}" : e.java_class.name
20
+ error = ::ActiveRecord::JDBCError.new("The driver encountered an unknown error: #{error}")
21
+ error.errno = e.getErrorCode
22
+ error.sql_exception = e
23
+ raise error
24
+ end
25
+
26
+ attr_reader :adapter, :config
27
+
28
+ # @deprecated no longer used (pass adapter into #initialize)
29
+ # @see ActiveRecord::ConnectionAdapters::JdbcAdapter#initialize
30
+ def adapter=(adapter); @adapter = adapter; end
31
+
32
+ def native_database_types
33
+ JdbcTypeConverter.new(supported_data_types).choose_best_types
34
+ end
35
+
36
+ # @deprecated no longer used - only kept for compatibility
37
+ def set_native_database_types
38
+ ArJdbc.deprecate "set_native_database_types is no longer used and does nothing override native_database_types instead"
39
+ end
40
+
41
+ def jndi?; @jndi; end
42
+ alias_method :jndi_connection?, :jndi?
43
+
44
+ # Sets the connection factory from the available configuration.
45
+ # @see #setup_jdbc_factory
46
+ # @see #setup_jndi_factory
47
+ #
48
+ # @note this has nothing to do with the configure_connection implemented
49
+ # on some of the concrete adapters (e.g. {#ArJdbc::Postgres})
50
+ def setup_connection_factory
51
+ if self.class.jndi_config?(config)
52
+ begin
53
+ setup_jndi_factory
54
+ rescue => e
55
+ warn "JNDI data source unavailable: #{e.message}; trying straight JDBC"
56
+ setup_jdbc_factory
57
+ end
58
+ else
59
+ setup_jdbc_factory
60
+ end
61
+ end
62
+
63
+ protected
64
+
65
+ def setup_jndi_factory
66
+ data_source = config[:data_source] ||
67
+ Java::JavaxNaming::InitialContext.new.lookup(config[:jndi].to_s)
68
+
69
+ @jndi = true
70
+ self.connection_factory = JndiConnectionFactoryImpl.new(data_source)
71
+ end
72
+
73
+ # @private
74
+ class JndiConnectionFactoryImpl
75
+ include JdbcConnectionFactory
76
+
77
+ def initialize(data_source)
78
+ @data_source = data_source
79
+ end
80
+
81
+ def newConnection
82
+ @data_source.connection
83
+ end
84
+ end
85
+
86
+ def setup_jdbc_factory
87
+ if ! config[:url] || ( ! config[:driver] && ! config[:driver_instance] )
88
+ msg = config[:url] ? ":url = #{config[:url]}" : ":driver = #{config[:driver]}"
89
+ raise ::ActiveRecord::ConnectionNotEstablished, "jdbc adapter requires :driver and :url (got #{msg})"
90
+ end
91
+
92
+ url = jdbc_url
93
+ username = config[:username]
94
+ password = config[:password]
95
+ driver = ( config[:driver_instance] ||=
96
+ JdbcDriver.new(config[:driver].to_s, config[:properties]) )
97
+
98
+ @jndi = false
99
+ self.connection_factory = JdbcConnectionFactoryImpl.new(url, username, password, driver)
100
+ end
101
+
102
+ # @private
103
+ class JdbcConnectionFactoryImpl
104
+ include JdbcConnectionFactory
105
+
106
+ def initialize(url, username, password, driver)
107
+ @url = url
108
+ @username = username
109
+ @password = password
110
+ @driver = driver
111
+ end
112
+
113
+ def newConnection
114
+ @driver.connection(@url, @username, @password)
115
+ end
116
+ end
117
+
118
+ private
119
+
120
+ def jdbc_url
121
+ url = config[:url].to_s
122
+ if options = config[:options]
123
+ ArJdbc.deprecate "use config[:properties] to specify connection URL properties instead of config[:options]"
124
+ options = options.map { |key, val| "#{key}=#{val}" }.join('&') if Hash === options
125
+ url = url['?'] ? "#{url}&#{options}" : "#{url}?#{options}" unless options.empty?
126
+ config[:url] = url; config[:options] = nil
127
+ end
128
+ url
129
+ end
130
+
131
+ end
132
+ end
133
+ end
@@ -0,0 +1,36 @@
1
+ module ArJdbc
2
+ if ActiveRecord.const_defined? :ConnectionHandling # 4.0
3
+ ConnectionMethods = ActiveRecord::ConnectionHandling
4
+ else # 3.x
5
+ ConnectionMethods = (class << ActiveRecord::Base; self; end)
6
+ end
7
+ ConnectionMethods.module_eval do
8
+
9
+ def jdbc_connection(config)
10
+ adapter_class = config[:adapter_class]
11
+ adapter_class ||= ::ActiveRecord::ConnectionAdapters::JdbcAdapter
12
+
13
+ # Once all adapters converted to AR5 then this rescue can be removed
14
+ begin
15
+ adapter_class.new(nil, logger, nil, config)
16
+ rescue ArgumentError
17
+ adapter_class.new(nil, logger, config)
18
+ end
19
+ end
20
+
21
+ def jndi_connection(config); jdbc_connection(config) end
22
+
23
+ def embedded_driver(config)
24
+ config[:username] ||= "sa"
25
+ config[:password] ||= ""
26
+ jdbc_connection(config)
27
+ end
28
+
29
+ private
30
+
31
+ def jndi_config?(config)
32
+ ::ActiveRecord::ConnectionAdapters::JdbcConnection.jndi_config?(config)
33
+ end
34
+
35
+ end
36
+ end
@@ -0,0 +1,43 @@
1
+ module ActiveRecord
2
+ module ConnectionAdapters
3
+ class JdbcDriver
4
+ attr_reader :name, :properties
5
+
6
+ def initialize(name, properties = {})
7
+ @name = name
8
+ @driver = driver_class.new
9
+ if properties.is_a?(Java::JavaUtil::Properties)
10
+ @properties = properties # allow programmatically set properties
11
+ else
12
+ @properties = Java::JavaUtil::Properties.new
13
+ properties.each { |key, val| @properties[key.to_s] = val.to_s } if properties
14
+ end
15
+ end
16
+
17
+ def driver_class
18
+ @driver_class ||= begin
19
+ driver_class_const = (@name[0...1].capitalize + @name[1..@name.length]).gsub(/\./, '_')
20
+ Jdbc::DriverManager.java_class.synchronized do # avoid 2 threads here
21
+ unless Jdbc.const_defined?(driver_class_const)
22
+ driver_class_name = @name
23
+ Jdbc.module_eval do
24
+ java_import(driver_class_name) { driver_class_const }
25
+ end
26
+ end
27
+ end unless Jdbc.const_defined?(driver_class_const)
28
+ driver_class = Jdbc.const_get(driver_class_const)
29
+ raise "You must specify a driver for your JDBC connection" unless driver_class
30
+ driver_class
31
+ end
32
+ end
33
+
34
+ def connection(url, user, pass)
35
+ # bypass DriverManager to get around problem with dynamically loaded jdbc drivers
36
+ properties = self.properties.clone
37
+ properties.setProperty("user", user.to_s) if user
38
+ properties.setProperty("password", pass.to_s) if pass
39
+ @driver.connect(url, properties)
40
+ end
41
+ end
42
+ end
43
+ end