ruby-oci8 2.2.10-x64-mingw-ucrt
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +7 -0
- data/.yardopts +14 -0
- data/COPYING +30 -0
- data/COPYING_old +64 -0
- data/ChangeLog +3826 -0
- data/Makefile +92 -0
- data/NEWS +1209 -0
- data/README.md +66 -0
- data/dist-files +112 -0
- data/docs/bind-array-to-in_cond.md +38 -0
- data/docs/conflicts-local-connections-and-processes.md +98 -0
- data/docs/hanging-after-inactivity.md +63 -0
- data/docs/install-binary-package.md +44 -0
- data/docs/install-full-client.md +111 -0
- data/docs/install-instant-client.md +194 -0
- data/docs/install-on-osx.md +46 -0
- data/docs/ldap-auth-and-function-interposition.md +123 -0
- data/docs/number-type-mapping.md +79 -0
- data/docs/platform-specific-issues.md +164 -0
- data/docs/report-installation-issue.md +50 -0
- data/docs/timeout-parameters.md +94 -0
- data/lib/.document +1 -0
- data/lib/dbd/OCI8.rb +591 -0
- data/lib/oci8/.document +8 -0
- data/lib/oci8/bindtype.rb +333 -0
- data/lib/oci8/check_load_error.rb +146 -0
- data/lib/oci8/compat.rb +117 -0
- data/lib/oci8/connection_pool.rb +179 -0
- data/lib/oci8/cursor.rb +605 -0
- data/lib/oci8/datetime.rb +605 -0
- data/lib/oci8/encoding-init.rb +45 -0
- data/lib/oci8/encoding.yml +537 -0
- data/lib/oci8/metadata.rb +2148 -0
- data/lib/oci8/object.rb +641 -0
- data/lib/oci8/oci8.rb +756 -0
- data/lib/oci8/ocihandle.rb +591 -0
- data/lib/oci8/oracle_version.rb +153 -0
- data/lib/oci8/properties.rb +196 -0
- data/lib/oci8/version.rb +3 -0
- data/lib/oci8.rb +190 -0
- data/lib/oci8lib_310.so +0 -0
- data/lib/ruby-oci8.rb +1 -0
- data/metaconfig +142 -0
- data/pre-distclean.rb +7 -0
- data/ruby-oci8.gemspec +85 -0
- data/setup.rb +1342 -0
- data/test/README.md +37 -0
- data/test/config.rb +201 -0
- data/test/setup_test_object.sql +199 -0
- data/test/setup_test_package.sql +59 -0
- data/test/test_all.rb +56 -0
- data/test/test_appinfo.rb +62 -0
- data/test/test_array_dml.rb +332 -0
- data/test/test_bind_array.rb +70 -0
- data/test/test_bind_boolean.rb +99 -0
- data/test/test_bind_integer.rb +47 -0
- data/test/test_bind_raw.rb +45 -0
- data/test/test_bind_string.rb +105 -0
- data/test/test_bind_time.rb +177 -0
- data/test/test_break.rb +125 -0
- data/test/test_clob.rb +85 -0
- data/test/test_connection_pool.rb +124 -0
- data/test/test_connstr.rb +220 -0
- data/test/test_datetime.rb +585 -0
- data/test/test_dbi.rb +365 -0
- data/test/test_dbi_clob.rb +53 -0
- data/test/test_encoding.rb +103 -0
- data/test/test_error.rb +87 -0
- data/test/test_metadata.rb +2674 -0
- data/test/test_object.rb +546 -0
- data/test/test_oci8.rb +624 -0
- data/test/test_oracle_version.rb +68 -0
- data/test/test_oradate.rb +255 -0
- data/test/test_oranumber.rb +792 -0
- data/test/test_package_type.rb +981 -0
- data/test/test_properties.rb +17 -0
- data/test/test_rowid.rb +32 -0
- metadata +123 -0
@@ -0,0 +1,94 @@
|
|
1
|
+
# @title Timeout Parameters
|
2
|
+
|
3
|
+
Timeout Parameters
|
4
|
+
==================
|
5
|
+
|
6
|
+
The following timeout parameters are available since ruby-oci8 2.2.2.
|
7
|
+
|
8
|
+
* tcp_connect_timeout
|
9
|
+
* connect_timeout
|
10
|
+
* send_timeout
|
11
|
+
* recv_timeout
|
12
|
+
|
13
|
+
For example:
|
14
|
+
|
15
|
+
OCI8.properties[:tcp_connect_timeout] = 10
|
16
|
+
OCI8.properties[:connect_timeout] = 15
|
17
|
+
OCI8.properties[:send_timeout] = 60
|
18
|
+
OCI8.properties[:recv_timeout] = 60
|
19
|
+
|
20
|
+
These parameters are applied only to TCP/IP connections.
|
21
|
+
|
22
|
+
The first two parameters `tcp_connect_timeout` and `connect_timeout`
|
23
|
+
are applied only to [connect descriptors][connect descriptor] using [Easy Connect Naming Method][EZCONNECT].
|
24
|
+
If you use a net service name, you should set [TRANSPORT_CONNECT_TIMEOUT][] and/or
|
25
|
+
[CONNECT_TIMEOUT][] in the address descriptor in `tnsnames.ora` instead of these parameters.
|
26
|
+
If you use easy connect naming method without any of `port`, `service_name`, `server` and `instance_name`,
|
27
|
+
you need to use `//host` to distinguish it from a net service name.
|
28
|
+
|
29
|
+
The next two parameters `send_timeout` and `recv_timeout` are available on Oracle 11g client
|
30
|
+
or upper. Use these parameters to prevent a ruby process from being blocked by poor quality network.
|
31
|
+
Otherwise, the ruby process may be blocked until TCP keepalive time (2 hours).
|
32
|
+
|
33
|
+
See {file:docs/hanging-after-inactivity.md Hanging After a Long Period of Inactivity}
|
34
|
+
for TCP keepalive time.
|
35
|
+
|
36
|
+
tcp_connect_timeout
|
37
|
+
-------------------
|
38
|
+
|
39
|
+
`tcp_connect_timeout` is equivalent to [TCP.CONNECT_TIMEOUT][] in the client-side `sqlnet.ora` and
|
40
|
+
[TRANSPORT_CONNECT_TIMEOUT][] in the address descriptor.
|
41
|
+
See description about [TCP.CONNECT_TIMEOUT][] and [TRANSPORT_CONNECT_TIMEOUT][].
|
42
|
+
|
43
|
+
connect_timeout
|
44
|
+
---------------
|
45
|
+
|
46
|
+
`connect_timeout` is equivalent to [SQLNET.OUTBOUND_CONNECT_TIMEOUT][] in the client-side `sqlnet.ora`
|
47
|
+
and [CONNECT_TIMEOUT][] in the address description.
|
48
|
+
See description about [SQLNET.OUTBOUND_CONNECT_TIMEOUT][] and [CONNECT_TIMEOUT][].
|
49
|
+
|
50
|
+
Note: this parameter isn't equivalent to login timeout. It needs the following three
|
51
|
+
steps to establish a database connection.
|
52
|
+
|
53
|
+
1. Establish a TCP/IP connection.
|
54
|
+
2. Establish an [Oracle Net][] connection on the TCP/IP connection.
|
55
|
+
3. Authenticate and authorize the database user.
|
56
|
+
|
57
|
+
`tcp_connect_timeout` sets the timeout of the first step.
|
58
|
+
`connect_timeout` sets the total timeout of the first and the second steps.
|
59
|
+
There is no timeout parameter to limit the maximum time of all three steps.
|
60
|
+
|
61
|
+
Use `send_timeout` and `recv_timeout` in case that a TCP/IP connection stalls
|
62
|
+
in the third step.
|
63
|
+
|
64
|
+
send_timeout
|
65
|
+
------------
|
66
|
+
|
67
|
+
`send_timeout` is equivalent to [SQLNET.SEND_TIMEOUT][] in the client-side `sqlnet.ora`.
|
68
|
+
See description about [SQLNET.SEND_TIMEOUT][].
|
69
|
+
|
70
|
+
Note that the connection becomes unusable on timeout.
|
71
|
+
|
72
|
+
See also {OCI8#send_timeout=}.
|
73
|
+
|
74
|
+
recv_timeout
|
75
|
+
------------
|
76
|
+
|
77
|
+
`recv_timeout` is equivalent to [SQLNET.RECV_TIMEOUT][] in the client-side `sqlnet.ora`.
|
78
|
+
See description about [SQLNET.RECV_TIMEOUT][].
|
79
|
+
|
80
|
+
Note that the connection becomes unusable on timeout.
|
81
|
+
|
82
|
+
See also {OCI8#recv_timeout=}.
|
83
|
+
|
84
|
+
Note: This parameter must be larger than the longest SQL execution time in your applications.
|
85
|
+
|
86
|
+
[TCP.CONNECT_TIMEOUT]: http://docs.oracle.com/database/121/NETRF/sqlnet.htm#BIIDDACA
|
87
|
+
[SQLNET.OUTBOUND_CONNECT_TIMEOUT]: https://docs.oracle.com/database/121/NETRF/sqlnet.htm#NETRF427
|
88
|
+
[SQLNET.SEND_TIMEOUT]: http://docs.oracle.com/database/121/NETRF/sqlnet.htm#NETRF228
|
89
|
+
[SQLNET.RECV_TIMEOUT]: http://docs.oracle.com/database/121/NETRF/sqlnet.htm#NETRF227
|
90
|
+
[connect descriptor]: https://docs.oracle.com/database/121/NETRF/glossary.htm#BGBEDFBF
|
91
|
+
[EZCONNECT]: https://docs.oracle.com/database/121/NETAG/naming.htm#NETAG255
|
92
|
+
[CONNECT_TIMEOUT]: https://docs.oracle.com/database/121/NETRF/tnsnames.htm#NETRF666
|
93
|
+
[TRANSPORT_CONNECT_TIMEOUT]: https://docs.oracle.com/database/121/NETRF/tnsnames.htm#NETRF1982
|
94
|
+
[Oracle Net]: https://en.wikipedia.org/wiki/Oracle_Net_Services#Oracle_Net
|
data/lib/.document
ADDED
@@ -0,0 +1 @@
|
|
1
|
+
oci8
|
data/lib/dbd/OCI8.rb
ADDED
@@ -0,0 +1,591 @@
|
|
1
|
+
#
|
2
|
+
# DBD::OCI8
|
3
|
+
#
|
4
|
+
# Copyright (c) 2002-2007 KUBO Takehiro <kubo@jiubao.org>
|
5
|
+
#
|
6
|
+
# copied some code from DBD::Oracle.
|
7
|
+
# DBD::Oracle's copyright is as follows:
|
8
|
+
# --------------------- begin -------------------
|
9
|
+
#
|
10
|
+
# Copyright (c) 2001, 2002, 2003, 2004 Michael Neumann <mneumann@ntecs.de>
|
11
|
+
#
|
12
|
+
# All rights reserved.
|
13
|
+
#
|
14
|
+
# Redistribution and use in source and binary forms, with or without
|
15
|
+
# modification, are permitted provided that the following conditions
|
16
|
+
# are met:
|
17
|
+
# 1. Redistributions of source code must retain the above copyright
|
18
|
+
# notice, this list of conditions and the following disclaimer.
|
19
|
+
# 2. Redistributions in binary form must reproduce the above copyright
|
20
|
+
# notice, this list of conditions and the following disclaimer in the
|
21
|
+
# documentation and/or other materials provided with the distribution.
|
22
|
+
# 3. The name of the author may not be used to endorse or promote products
|
23
|
+
# derived from this software without specific prior written permission.
|
24
|
+
#
|
25
|
+
# THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES,
|
26
|
+
# INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
27
|
+
# AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
|
28
|
+
# THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
|
29
|
+
# EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
30
|
+
# PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
|
31
|
+
# OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
|
32
|
+
# WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
|
33
|
+
# OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
|
34
|
+
# ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
35
|
+
#
|
36
|
+
# --------------------- end -------------------
|
37
|
+
|
38
|
+
require 'oci8'
|
39
|
+
|
40
|
+
module DBI # :nodoc:
|
41
|
+
module DBD # :nodoc:
|
42
|
+
module OCI8
|
43
|
+
|
44
|
+
VERSION = "0.1"
|
45
|
+
USED_DBD_VERSION = "0.4"
|
46
|
+
|
47
|
+
def self.driver_name
|
48
|
+
"OCI8"
|
49
|
+
end
|
50
|
+
|
51
|
+
# type converstion handler to bind values. (ruby-dbi 0.4)
|
52
|
+
if DBI.const_defined?(:TypeUtil)
|
53
|
+
DBI::TypeUtil.register_conversion("OCI8") do |obj|
|
54
|
+
case obj
|
55
|
+
when ::TrueClass
|
56
|
+
['1', false]
|
57
|
+
when ::FalseClass
|
58
|
+
['0', false]
|
59
|
+
else
|
60
|
+
[obj, false]
|
61
|
+
end
|
62
|
+
end
|
63
|
+
end
|
64
|
+
|
65
|
+
# no type converstion is required for result set. (ruby-dbi 0.4)
|
66
|
+
class NoTypeConversion
|
67
|
+
def self.parse(obj)
|
68
|
+
obj
|
69
|
+
end
|
70
|
+
end
|
71
|
+
|
72
|
+
module Util
|
73
|
+
|
74
|
+
ERROR_MAP = {
|
75
|
+
1 => DBI::IntegrityError, # unique constraint violated
|
76
|
+
900 => DBI::ProgrammingError, # invalid SQL statement
|
77
|
+
904 => DBI::ProgrammingError, # invalid identifier
|
78
|
+
905 => DBI::ProgrammingError, # missing keyword
|
79
|
+
923 => DBI::ProgrammingError, # FROM keyword not found where expected
|
80
|
+
936 => DBI::ProgrammingError, # missing expression
|
81
|
+
942 => DBI::ProgrammingError, # table or view does not exist
|
82
|
+
2290 => DBI::IntegrityError, # check constraint violated
|
83
|
+
2291 => DBI::IntegrityError, # parent key not found
|
84
|
+
2292 => DBI::IntegrityError, # child record found
|
85
|
+
2293 => DBI::IntegrityError, # check constraint violated
|
86
|
+
}
|
87
|
+
|
88
|
+
def raise_dbierror(err) # :nodoc:
|
89
|
+
if err.is_a? OCIError
|
90
|
+
exc = ERROR_MAP[err.code] || DBI::DatabaseError
|
91
|
+
raise exc.new(err.message, err.code)
|
92
|
+
else
|
93
|
+
raise DBI::DatabaseError.new(err.message, -1)
|
94
|
+
end
|
95
|
+
rescue DBI::DatabaseError => exc
|
96
|
+
exc.set_backtrace(err.backtrace)
|
97
|
+
raise
|
98
|
+
end
|
99
|
+
|
100
|
+
def column_metadata_to_column_info(col)
|
101
|
+
sql_type, type_name, precision, scale =
|
102
|
+
case col.data_type
|
103
|
+
when :char
|
104
|
+
[SQL_CHAR, col.charset_form == :nchar ? "NCHAR" : "CHAR", col.data_size, nil]
|
105
|
+
when :varchar2
|
106
|
+
[SQL_VARCHAR, col.charset_form == :nchar ? "NVARCHAR2" : "VARCHAR2", col.data_size, nil]
|
107
|
+
when :raw
|
108
|
+
[SQL_VARBINARY, "RAW", col.data_size, nil]
|
109
|
+
when :long
|
110
|
+
[SQL_LONGVARCHAR, "LONG", 4000, nil]
|
111
|
+
when :long_raw
|
112
|
+
[SQL_LONGVARBINARY, "LONG RAW", 4000, nil]
|
113
|
+
when :clob
|
114
|
+
[SQL_CLOB, col.charset_form == :nchar ? "NCLOB" : "CLOB", 4000, nil]
|
115
|
+
when :blob
|
116
|
+
[SQL_BLOB, "BLOB", 4000, nil]
|
117
|
+
when :bfile
|
118
|
+
[SQL_BLOB, "BFILE", 4000, nil]
|
119
|
+
when :number
|
120
|
+
if col.scale == -127 && col.precision != 0
|
121
|
+
# To convert from binary to decimal precision, multiply n by 0.30103.
|
122
|
+
[SQL_FLOAT, "FLOAT", (col.precision * 0.30103).ceil , nil]
|
123
|
+
elsif col.precision == 0
|
124
|
+
# NUMBER or calculated value (eg. col * 1.2).
|
125
|
+
[SQL_NUMERIC, "NUMBER", 38, nil]
|
126
|
+
else
|
127
|
+
[SQL_NUMERIC, "NUMBER", col.precision, col.scale]
|
128
|
+
end
|
129
|
+
when :binary_float
|
130
|
+
# (23 * 0.30103).ceil => 7
|
131
|
+
[SQL_FLOAT, "BINARY_FLOAT", 7, nil]
|
132
|
+
when :binary_double
|
133
|
+
# (52 * 0.30103).ceil => 16
|
134
|
+
[SQL_DOUBLE, "BINARY_DOUBLE", 16, nil]
|
135
|
+
when :date
|
136
|
+
# yyyy-mm-dd hh:mi:ss
|
137
|
+
[SQL_DATE, "DATE", 19, nil]
|
138
|
+
when :timestamp
|
139
|
+
# yyyy-mm-dd hh:mi:ss.SSSS
|
140
|
+
[SQL_TIMESTAMP, "TIMESTAMP", 20 + col.fsprecision, nil]
|
141
|
+
when :timestamp_tz
|
142
|
+
# yyyy-mm-dd hh:mi:ss.SSSS +HH:MM
|
143
|
+
[SQL_TIMESTAMP, "TIMESTAMP WITH TIME ZONE", 27 + col.fsprecision, nil]
|
144
|
+
when :timestamp_ltz
|
145
|
+
# yyyy-mm-dd hh:mi:ss.SSSS
|
146
|
+
[SQL_TIMESTAMP, "TIMESTAMP WITH LOCAL TIME ZONE", 20 + col.fsprecision, nil]
|
147
|
+
when :interval_ym
|
148
|
+
# yyyy-mm
|
149
|
+
[SQL_OTHER, 'INTERVAL YEAR TO MONTH', col.lfprecision + 3, nil]
|
150
|
+
when :interval_ds
|
151
|
+
# dd hh:mi:ss.SSSSS
|
152
|
+
[SQL_OTHER, 'INTERVAL DAY TO SECOND', col.lfprecision + 10 + col.fsprecision, nil]
|
153
|
+
else
|
154
|
+
[SQL_OTHER, col.data_type.to_s, nil, nil]
|
155
|
+
end
|
156
|
+
{'name' => col.name,
|
157
|
+
'sql_type' => sql_type,
|
158
|
+
'type_name' => type_name,
|
159
|
+
'nullable' => col.nullable?,
|
160
|
+
'precision' => precision,
|
161
|
+
'scale' => scale,
|
162
|
+
'dbi_type' => NoTypeConversion,
|
163
|
+
}
|
164
|
+
end
|
165
|
+
private :column_metadata_to_column_info
|
166
|
+
end
|
167
|
+
|
168
|
+
class Driver < DBI::BaseDriver # :nodoc:
|
169
|
+
include Util
|
170
|
+
|
171
|
+
def initialize
|
172
|
+
super(USED_DBD_VERSION)
|
173
|
+
end
|
174
|
+
|
175
|
+
# external OS authentication
|
176
|
+
# (contributed by Dan Fitch)
|
177
|
+
def default_user
|
178
|
+
[nil, nil]
|
179
|
+
end
|
180
|
+
|
181
|
+
def connect( dbname, user, auth, attr )
|
182
|
+
handle = ::OCI8.new(user, auth, dbname, attr['Privilege'])
|
183
|
+
handle.non_blocking = true if attr['NonBlocking']
|
184
|
+
return Database.new(handle, attr)
|
185
|
+
rescue OCIException => err
|
186
|
+
raise_dbierror(err)
|
187
|
+
end
|
188
|
+
end
|
189
|
+
|
190
|
+
class Database < DBI::BaseDatabase
|
191
|
+
include Util
|
192
|
+
|
193
|
+
def disconnect
|
194
|
+
@handle.logoff
|
195
|
+
rescue OCIException => err
|
196
|
+
raise_dbierror(err)
|
197
|
+
end
|
198
|
+
|
199
|
+
def prepare( statement )
|
200
|
+
# convert ?-style parameters to :1, :2 etc.
|
201
|
+
prep_statement = DBI::SQL::PreparedStatement.new(DummyQuoter.new, statement)
|
202
|
+
if prep_statement.unbound.size > 0
|
203
|
+
arr = (1..(prep_statement.unbound.size)).collect{|i| ":#{i}"}
|
204
|
+
statement = prep_statement.bind( arr )
|
205
|
+
end
|
206
|
+
cursor = @handle.parse(statement)
|
207
|
+
Statement.new(cursor)
|
208
|
+
rescue OCIException => err
|
209
|
+
raise_dbierror(err)
|
210
|
+
end
|
211
|
+
|
212
|
+
def ping
|
213
|
+
@handle.exec("BEGIN NULL; END;")
|
214
|
+
true
|
215
|
+
rescue
|
216
|
+
false
|
217
|
+
end
|
218
|
+
|
219
|
+
def commit
|
220
|
+
@handle.commit()
|
221
|
+
rescue OCIException => err
|
222
|
+
raise_dbierror(err)
|
223
|
+
end
|
224
|
+
|
225
|
+
def rollback
|
226
|
+
@handle.rollback()
|
227
|
+
rescue OCIException => err
|
228
|
+
raise_dbierror(err)
|
229
|
+
end
|
230
|
+
|
231
|
+
def tables
|
232
|
+
stmt = execute("SELECT object_name FROM user_objects where object_type in ('TABLE', 'VIEW')")
|
233
|
+
rows = stmt.fetch_all || []
|
234
|
+
stmt.finish
|
235
|
+
rows.collect {|row| row[0]}
|
236
|
+
end
|
237
|
+
|
238
|
+
# SQLs are copied from DBD::Oracle.
|
239
|
+
def columns(table)
|
240
|
+
tab = @handle.describe_table(table)
|
241
|
+
cols = tab.columns
|
242
|
+
cols.collect! do |col|
|
243
|
+
column_metadata_to_column_info(col)
|
244
|
+
end
|
245
|
+
|
246
|
+
dbh = DBI::DatabaseHandle.new(self)
|
247
|
+
|
248
|
+
primaries = {}
|
249
|
+
dbh.select_all(<<EOS, tab.obj_schema, tab.obj_name) do |row|
|
250
|
+
select column_name
|
251
|
+
from all_cons_columns a, all_constraints b
|
252
|
+
where a.owner = b.owner
|
253
|
+
and a.constraint_name = b.constraint_name
|
254
|
+
and a.table_name = b.table_name
|
255
|
+
and b.constraint_type = 'P'
|
256
|
+
and b.owner = :1
|
257
|
+
and b.table_name = :2
|
258
|
+
EOS
|
259
|
+
primaries[row[0]] = true
|
260
|
+
end
|
261
|
+
|
262
|
+
indices = {}
|
263
|
+
uniques = {}
|
264
|
+
dbh.select_all(<<EOS, tab.obj_schema, tab.obj_name) do |row|
|
265
|
+
select a.column_name, a.index_name, b.uniqueness
|
266
|
+
from all_ind_columns a, all_indexes b
|
267
|
+
where a.index_name = b.index_name
|
268
|
+
and a.index_owner = b.owner
|
269
|
+
and a.table_owner = :1
|
270
|
+
and a.table_name = :2
|
271
|
+
EOS
|
272
|
+
col_name, index_name, uniqueness = row
|
273
|
+
indices[col_name] = true
|
274
|
+
uniques[col_name] = true if uniqueness == 'UNIQUE'
|
275
|
+
end
|
276
|
+
|
277
|
+
dbh.select_all(<<EOS, tab.obj_schema, tab.obj_name).collect do |row|
|
278
|
+
select column_id, column_name, data_default
|
279
|
+
from all_tab_columns
|
280
|
+
where owner = :1
|
281
|
+
and table_name = :2
|
282
|
+
EOS
|
283
|
+
col_id, col_name, default = row
|
284
|
+
|
285
|
+
col = cols[col_id.to_i - 1]
|
286
|
+
col_name = col['name']
|
287
|
+
|
288
|
+
if default && default[0] == ?'
|
289
|
+
default = default[1..-2].gsub(/''/, "'")
|
290
|
+
end
|
291
|
+
|
292
|
+
col['indexed'] = indices[col_name] || false
|
293
|
+
col['primary'] = primaries[col_name] || false
|
294
|
+
col['unique'] = uniques[col_name] || false
|
295
|
+
col['default'] = default
|
296
|
+
col
|
297
|
+
end
|
298
|
+
rescue OCIException => err
|
299
|
+
raise_dbierror(err)
|
300
|
+
end
|
301
|
+
|
302
|
+
def [](attr)
|
303
|
+
case attr
|
304
|
+
when 'AutoCommit'
|
305
|
+
@handle.autocommit?
|
306
|
+
end
|
307
|
+
end
|
308
|
+
|
309
|
+
def []=(attr, value)
|
310
|
+
case attr
|
311
|
+
when 'AutoCommit'
|
312
|
+
@handle.autocommit = value
|
313
|
+
end
|
314
|
+
end
|
315
|
+
|
316
|
+
private
|
317
|
+
|
318
|
+
class DummyQuoter # :nodoc:
|
319
|
+
# dummy to substitute ?-style parameter markers by :1 :2 etc.
|
320
|
+
def quote(str)
|
321
|
+
str
|
322
|
+
end
|
323
|
+
end
|
324
|
+
end
|
325
|
+
|
326
|
+
class Statement < DBI::BaseStatement
|
327
|
+
include Util
|
328
|
+
|
329
|
+
def initialize(cursor)
|
330
|
+
@cursor = cursor
|
331
|
+
end
|
332
|
+
|
333
|
+
def bind_param( param, value, attribs)
|
334
|
+
if attribs.nil? || attribs['type'].nil?
|
335
|
+
if value.nil?
|
336
|
+
@cursor.bind_param(param, nil, String, 1)
|
337
|
+
else
|
338
|
+
@cursor.bind_param(param, value)
|
339
|
+
end
|
340
|
+
else
|
341
|
+
case attribs['type']
|
342
|
+
when SQL_BINARY
|
343
|
+
type = OCI_TYPECODE_RAW
|
344
|
+
else
|
345
|
+
type = attribs['type']
|
346
|
+
end
|
347
|
+
@cursor.bind_param(param, value, type)
|
348
|
+
end
|
349
|
+
rescue OCIException => err
|
350
|
+
raise_dbierror(err)
|
351
|
+
end
|
352
|
+
|
353
|
+
def execute
|
354
|
+
@cursor.exec
|
355
|
+
rescue OCIException => err
|
356
|
+
raise_dbierror(err)
|
357
|
+
end
|
358
|
+
|
359
|
+
def finish
|
360
|
+
@cursor.close
|
361
|
+
rescue OCIException => err
|
362
|
+
raise_dbierror(err)
|
363
|
+
end
|
364
|
+
|
365
|
+
def fetch
|
366
|
+
@cursor.fetch
|
367
|
+
rescue OCIException => err
|
368
|
+
raise_dbierror(err)
|
369
|
+
end
|
370
|
+
|
371
|
+
def column_info
|
372
|
+
# minimum implementation.
|
373
|
+
@cursor.column_metadata.collect do |md|
|
374
|
+
col = column_metadata_to_column_info(md)
|
375
|
+
col['indexed'] = nil
|
376
|
+
col['primary'] = nil
|
377
|
+
col['unique'] = nil
|
378
|
+
col['default'] = nil
|
379
|
+
col
|
380
|
+
end
|
381
|
+
rescue OCIException => err
|
382
|
+
raise_dbierror(err)
|
383
|
+
end
|
384
|
+
|
385
|
+
def rows
|
386
|
+
@cursor.row_count
|
387
|
+
rescue OCIException => err
|
388
|
+
raise_dbierror(err)
|
389
|
+
end
|
390
|
+
|
391
|
+
def __rowid
|
392
|
+
@cursor.rowid
|
393
|
+
end
|
394
|
+
|
395
|
+
def __define(pos, type, length = nil)
|
396
|
+
@cursor.define(pos, type, length)
|
397
|
+
self
|
398
|
+
end
|
399
|
+
|
400
|
+
def __bind_value(param)
|
401
|
+
@cursor[param]
|
402
|
+
end
|
403
|
+
end
|
404
|
+
|
405
|
+
# DBI_STMT_NEW_ARGS is DBI::StatementHandle.new's arguments except +handle+.
|
406
|
+
#
|
407
|
+
# FYI: DBI::StatementHandle.new method signatures are follows:
|
408
|
+
# 0.2.2: handle, fetchable=false, prepared=true
|
409
|
+
# 0.4.0: handle, fetchable=false, prepared=true, convert_types=true
|
410
|
+
# 0.4.1: handle, fetchable=false, prepared=true, convert_types=true, executed=false
|
411
|
+
begin
|
412
|
+
DBI::StatementHandle.new(nil, false, true, true, true)
|
413
|
+
# dbi 0.4.1
|
414
|
+
DBI_STMT_NEW_ARGS = [true, true, true, true] # :nodoc:
|
415
|
+
rescue ArgumentError
|
416
|
+
# dbi 0.4.0 or lower
|
417
|
+
DBI_STMT_NEW_ARGS = [true] # :nodoc:
|
418
|
+
end
|
419
|
+
|
420
|
+
if defined? ::OCI8::BindType::Base
|
421
|
+
##
|
422
|
+
## ruby-oci8 2.0 bind classes.
|
423
|
+
##
|
424
|
+
|
425
|
+
module BindType # :nodoc:
|
426
|
+
|
427
|
+
# helper class to define/bind DBI::Date.
|
428
|
+
class DBIDate < ::OCI8::BindType::OraDate
|
429
|
+
def set(val)
|
430
|
+
# convert val to an OraDate,
|
431
|
+
# then set it to the bind handle.
|
432
|
+
super(val && OraDate.new(val.year, val.month, val.day))
|
433
|
+
end
|
434
|
+
def get()
|
435
|
+
# get an Oradate from the bind handle,
|
436
|
+
# then convert it to a DBI::Date.
|
437
|
+
val = super()
|
438
|
+
return nil if val.nil?
|
439
|
+
DBI::Date.new(val.year, val.month, val.day)
|
440
|
+
end
|
441
|
+
end
|
442
|
+
|
443
|
+
# helper class to define/bind DBI::Timestamp.
|
444
|
+
#
|
445
|
+
# To fetch all Oracle's DATE columns as DBI::Timestamp:
|
446
|
+
# ::OCI8::BindType::Mapping[OCI8::SQLT_DAT] = ::DBI::DBD::OCI8::BindType::DBITimestamp
|
447
|
+
#
|
448
|
+
class DBITimestamp < ::OCI8::BindType::OraDate
|
449
|
+
def set(val)
|
450
|
+
# convert val to an OraDate,
|
451
|
+
# then set it to the bind handle.
|
452
|
+
super(val && OraDate.new(val.year, val.month, val.day,
|
453
|
+
val.respond_to?(:hour) ? val.hour : 0,
|
454
|
+
val.respond_to?(:min) ? val.min : 0,
|
455
|
+
val.respond_to?(:sec) ? val.sec : 0))
|
456
|
+
end
|
457
|
+
def get()
|
458
|
+
# get an Oradate from the bind handle,
|
459
|
+
# then convert it to a DBI::Timestamp.
|
460
|
+
val = super()
|
461
|
+
return nil if val.nil?
|
462
|
+
DBI::Timestamp.new(val.year, val.month, val.day, val.hour, val.minute, val.second)
|
463
|
+
end
|
464
|
+
end
|
465
|
+
|
466
|
+
# helper class to bind ref cursor as DBI::StatementHandle.
|
467
|
+
#
|
468
|
+
# # Create package
|
469
|
+
# dbh.execute(<<EOS)
|
470
|
+
# create or replace package test_pkg is
|
471
|
+
# type ref_cursor is ref cursor;
|
472
|
+
# procedure tab_table(csr out ref_cursor);
|
473
|
+
# end;
|
474
|
+
# EOS
|
475
|
+
#
|
476
|
+
# # Create package body
|
477
|
+
# dbh.execute(<<EOS)
|
478
|
+
# create or replace package body test_pkg is
|
479
|
+
# procedure tab_table(csr out ref_cursor) is
|
480
|
+
# begin
|
481
|
+
# open csr for select * from tab;
|
482
|
+
# end;
|
483
|
+
# end;
|
484
|
+
# EOS
|
485
|
+
#
|
486
|
+
# # Execute test_pkg.tab_table.
|
487
|
+
# # The first parameter is bound as DBI::StatementHandle.
|
488
|
+
# plsql = dbh.execute("begin test_pkg.tab_table(?); end;", DBI::StatementHandle)
|
489
|
+
#
|
490
|
+
# # Get the first parameter, which is a DBI::StatementHandle.
|
491
|
+
# sth = plsql.func(:bind_value, 1)
|
492
|
+
#
|
493
|
+
# # fetch column data.
|
494
|
+
# sth.fetch_all
|
495
|
+
#
|
496
|
+
class DBIStatementHandle < ::OCI8::BindType::Cursor
|
497
|
+
def set(val)
|
498
|
+
if val.is_a? DBI::StatementHandle
|
499
|
+
# get OCI8::Cursor
|
500
|
+
val = val.instance_eval do @handle end
|
501
|
+
val = val.instance_eval do @cursor end
|
502
|
+
end
|
503
|
+
super(val)
|
504
|
+
end
|
505
|
+
def get()
|
506
|
+
val = super
|
507
|
+
return nil if val.nil?
|
508
|
+
stmt = DBI::DBD::OCI8::Statement.new(val)
|
509
|
+
DBI::StatementHandle.new(stmt, *DBI_STMT_NEW_ARGS)
|
510
|
+
end
|
511
|
+
end
|
512
|
+
end # BindType
|
513
|
+
|
514
|
+
else
|
515
|
+
##
|
516
|
+
## ruby-oci8 1.0 bind classes.
|
517
|
+
##
|
518
|
+
|
519
|
+
module BindType # :nodoc:
|
520
|
+
DBIDate = Object.new
|
521
|
+
class << DBIDate
|
522
|
+
def fix_type(env, val, length, precision, scale)
|
523
|
+
# bind as an OraDate
|
524
|
+
[::OCI8::SQLT_DAT, val, nil]
|
525
|
+
end
|
526
|
+
def decorate(b)
|
527
|
+
def b.set(val)
|
528
|
+
# convert val to an OraDate,
|
529
|
+
# then set it to the bind handle.
|
530
|
+
super(val && OraDate.new(val.year, val.month, val.day))
|
531
|
+
end
|
532
|
+
def b.get()
|
533
|
+
# get an Oradate from the bind handle,
|
534
|
+
# then convert it to a DBI::Date.
|
535
|
+
(val = super()) && DBI::Date.new(val.year, val.month, val.day)
|
536
|
+
end
|
537
|
+
end
|
538
|
+
end
|
539
|
+
|
540
|
+
DBITimestamp = Object.new
|
541
|
+
class << DBITimestamp
|
542
|
+
def fix_type(env, val, length, precision, scale)
|
543
|
+
# bind as an OraDate
|
544
|
+
[::OCI8::SQLT_DAT, val, nil]
|
545
|
+
end
|
546
|
+
def decorate(b)
|
547
|
+
def b.set(val)
|
548
|
+
# convert val to an OraDate,
|
549
|
+
# then set it to the bind handle.
|
550
|
+
super(val && OraDate.new(val.year, val.month, val.day,
|
551
|
+
val.respond_to?(:hour) ? val.hour : 0,
|
552
|
+
val.respond_to?(:min) ? val.min : 0,
|
553
|
+
val.respond_to?(:sec) ? val.sec : 0))
|
554
|
+
end
|
555
|
+
def b.get()
|
556
|
+
# get an Oradate from the bind handle,
|
557
|
+
# then convert it to a DBI::Timestamp.
|
558
|
+
(val = super()) && DBI::Timestamp.new(val.year, val.month, val.day, val.hour, val.minute, val.second)
|
559
|
+
end
|
560
|
+
end
|
561
|
+
end
|
562
|
+
|
563
|
+
DBIStatementHandle = Object.new
|
564
|
+
class << DBIStatementHandle
|
565
|
+
def fix_type(env, val, length, precision, scale)
|
566
|
+
raise NotImplementedError unless val.nil?
|
567
|
+
[::OCI8::SQLT_RSET, nil, env.alloc(OCIStmt)]
|
568
|
+
end
|
569
|
+
def decorate(b)
|
570
|
+
def b.set(val)
|
571
|
+
raise NotImplementedError
|
572
|
+
end
|
573
|
+
def b.get()
|
574
|
+
val = super
|
575
|
+
return val if val.nil?
|
576
|
+
cur = ::OCI8::Cursor.new(@env, @svc, @ctx, val)
|
577
|
+
stmt = DBI::DBD::OCI8::Statement.new(cur)
|
578
|
+
DBI::StatementHandle.new(stmt, *DBI_STMT_NEW_ARGS)
|
579
|
+
end
|
580
|
+
end
|
581
|
+
end
|
582
|
+
end # BindType
|
583
|
+
end
|
584
|
+
|
585
|
+
::OCI8::BindType::Mapping[DBI::Date] = BindType::DBIDate
|
586
|
+
::OCI8::BindType::Mapping[DBI::Timestamp] = BindType::DBITimestamp
|
587
|
+
::OCI8::BindType::Mapping[DBI::StatementHandle] = BindType::DBIStatementHandle
|
588
|
+
|
589
|
+
end # module OCI8
|
590
|
+
end # module DBD
|
591
|
+
end # module DBI
|