2 Custom Query class for this backend (a derivative of
3 django.db.models.sql.query.Query).
8 from django
.db
.backends
import util
10 # Cache. Maps default query class to new Oracle query class.
13 def query_class(QueryClass
, Database
):
15 Returns a custom djang.db.models.sql.query.Query subclass that is
16 appropraite for Oracle.
18 The 'Database' module (cx_Oracle) is passed in here so that all the setup
19 required to import it only needs to be done by the calling module.
23 return _classes
[QueryClass
]
27 class OracleQuery(QueryClass
):
28 def resolve_columns(self
, row
, fields
=()):
29 index_start
= len(self
.extra_select
.keys())
30 values
= [self
.convert_values(v
, None) for v
in row
[:index_start
]]
31 for value
, field
in map(None, row
[index_start
:], fields
):
32 values
.append(self
.convert_values(value
, field
))
35 def convert_values(self
, value
, field
):
36 from django
.db
.models
.fields
import DateField
, DateTimeField
, \
37 TimeField
, BooleanField
, NullBooleanField
, DecimalField
, Field
38 if isinstance(value
, Database
.LOB
):
40 # Oracle stores empty strings as null. We need to undo this in
41 # order to adhere to the Django convention of using the empty
42 # string instead of null, but only if the field accepts the
44 if value
is None and isinstance(field
, Field
) and field
.empty_strings_allowed
:
46 # Convert 1 or 0 to True or False
47 elif value
in (1, 0) and isinstance(field
, (BooleanField
, NullBooleanField
)):
49 # Convert floats to decimals
50 elif value
is not None and isinstance(field
, DecimalField
):
51 value
= util
.typecast_decimal(field
.format_number(value
))
52 # cx_Oracle always returns datetime.datetime objects for
53 # DATE and TIMESTAMP columns, but Django wants to see a
54 # python datetime.date, .time, or .datetime. We use the type
55 # of the Field to determine which to cast to, but it's not
57 # As a workaround, we cast to date if all the time-related
58 # values are 0, or to time if the date is 1/1/1900.
59 # This could be cleaned a bit by adding a method to the Field
60 # classes to normalize values from the database (the to_python
61 # method is used for validation and isn't what we want here).
62 elif isinstance(value
, Database
.Timestamp
):
63 # In Python 2.3, the cx_Oracle driver returns its own
64 # Timestamp object that we must convert to a datetime class.
65 if not isinstance(value
, datetime
.datetime
):
66 value
= datetime
.datetime(value
.year
, value
.month
,
67 value
.day
, value
.hour
, value
.minute
, value
.second
,
69 if isinstance(field
, DateTimeField
):
70 # DateTimeField subclasses DateField so must be checked
73 elif isinstance(field
, DateField
):
75 elif isinstance(field
, TimeField
) or (value
.year
== 1900 and value
.month
== value
.day
== 1):
77 elif value
.hour
== value
.minute
== value
.second
== value
.microsecond
== 0:
81 def as_sql(self
, with_limits
=True, with_col_aliases
=False):
83 Creates the SQL for this query. Returns the SQL string and list
84 of parameters. This is overriden from the original Query class
85 to accommodate Oracle's limit/offset SQL.
87 If 'with_limits' is False, any limit/offset information is not
88 included in the query.
90 # The `do_offset` flag indicates whether we need to construct
91 # the SQL needed to use limit/offset w/Oracle.
92 do_offset
= with_limits
and (self
.high_mark
or self
.low_mark
)
94 # If no offsets, just return the result of the base class
97 return super(OracleQuery
, self
).as_sql(with_limits
=False,
98 with_col_aliases
=with_col_aliases
)
100 # `get_columns` needs to be called before `get_ordering` to
101 # populate `_select_alias`.
103 out_cols
= self
.get_columns()
104 ordering
= self
.get_ordering()
106 # Getting the "ORDER BY" SQL for the ROW_NUMBER() result.
108 rn_orderby
= ', '.join(ordering
)
110 # Oracle's ROW_NUMBER() function always requires an
111 # order-by clause. So we need to define a default
112 # order-by, since none was provided.
113 qn
= self
.quote_name_unless_alias
114 opts
= self
.model
._meta
115 rn_orderby
= '%s.%s' % (qn(opts
.db_table
), qn(opts
.fields
[0].db_column
or opts
.fields
[0].column
))
117 # Getting the selection SQL and the params, which has the `rn`
118 # extra selection SQL.
119 self
.extra_select
['rn'] = 'ROW_NUMBER() OVER (ORDER BY %s )' % rn_orderby
120 sql
, params
= super(OracleQuery
, self
).as_sql(with_limits
=False,
121 with_col_aliases
=True)
123 # Constructing the result SQL, using the initial select SQL
125 result
= ['SELECT * FROM (%s)' % sql
]
127 # Place WHERE condition on `rn` for the desired range.
128 result
.append('WHERE rn > %d' % self
.low_mark
)
130 result
.append('AND rn <= %d' % self
.high_mark
)
132 # Returning the SQL w/params.
133 return ' '.join(result
), params
135 def set_limits(self
, low
=None, high
=None):
136 super(OracleQuery
, self
).set_limits(low
, high
)
138 # We need to select the row number for the LIMIT/OFFSET sql.
139 # A placeholder is added to extra_select now, because as_sql is
140 # too late to be modifying extra_select. However, the actual sql
141 # depends on the ordering, so that is generated in as_sql.
142 self
.extra_select
['rn'] = '1'
144 def clear_limits(self
):
145 super(OracleQuery
, self
).clear_limits()
146 if 'rn' in self
.extra_select
:
147 del self
.extra_select
['rn']
149 _classes
[QueryClass
] = OracleQuery