mirror of
				https://github.com/django/django.git
				synced 2025-10-25 06:36:07 +00:00 
			
		
		
		
	Fixed #22603 -- Reorganized classes in django.db.backends.
This commit is contained in:
		
							
								
								
									
										0
									
								
								django/contrib/gis/db/backends/base/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								django/contrib/gis/db/backends/base/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										82
									
								
								django/contrib/gis/db/backends/base/features.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										82
									
								
								django/contrib/gis/db/backends/base/features.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,82 @@ | |||||||
|  | from functools import partial | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class BaseSpatialFeatures(object): | ||||||
|  |     gis_enabled = True | ||||||
|  |  | ||||||
|  |     # Does the database contain a SpatialRefSys model to store SRID information? | ||||||
|  |     has_spatialrefsys_table = True | ||||||
|  |  | ||||||
|  |     # Does the backend support the django.contrib.gis.utils.add_srs_entry() utility? | ||||||
|  |     supports_add_srs_entry = True | ||||||
|  |     # Does the backend introspect GeometryField to its subtypes? | ||||||
|  |     supports_geometry_field_introspection = True | ||||||
|  |  | ||||||
|  |     # Reference implementation of 3D functions is: | ||||||
|  |     # http://postgis.net/docs/PostGIS_Special_Functions_Index.html#PostGIS_3D_Functions | ||||||
|  |     supports_3d_functions = False | ||||||
|  |     # Does the database support SRID transform operations? | ||||||
|  |     supports_transform = True | ||||||
|  |     # Do geometric relationship operations operate on real shapes (or only on bounding boxes)? | ||||||
|  |     supports_real_shape_operations = True | ||||||
|  |     # Can geometry fields be null? | ||||||
|  |     supports_null_geometries = True | ||||||
|  |     # Can the `distance` GeoQuerySet method be applied on geodetic coordinate systems? | ||||||
|  |     supports_distance_geodetic = True | ||||||
|  |     # Is the database able to count vertices on polygons (with `num_points`)? | ||||||
|  |     supports_num_points_poly = True | ||||||
|  |  | ||||||
|  |     # The following properties indicate if the database backend support | ||||||
|  |     # certain lookups (dwithin, left and right, relate, ...) | ||||||
|  |     supports_distances_lookups = True | ||||||
|  |     supports_left_right_lookups = False | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def supports_bbcontains_lookup(self): | ||||||
|  |         return 'bbcontains' in self.connection.ops.gis_operators | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def supports_contained_lookup(self): | ||||||
|  |         return 'contained' in self.connection.ops.gis_operators | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def supports_dwithin_lookup(self): | ||||||
|  |         return 'dwithin' in self.connection.ops.gis_operators | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def supports_relate_lookup(self): | ||||||
|  |         return 'relate' in self.connection.ops.gis_operators | ||||||
|  |  | ||||||
|  |     # For each of those methods, the class will have a property named | ||||||
|  |     # `has_<name>_method` (defined in __init__) which accesses connection.ops | ||||||
|  |     # to determine GIS method availability. | ||||||
|  |     geoqueryset_methods = ( | ||||||
|  |         'area', 'centroid', 'difference', 'distance', 'distance_spheroid', | ||||||
|  |         'envelope', 'force_rhr', 'geohash', 'gml', 'intersection', 'kml', | ||||||
|  |         'length', 'num_geom', 'perimeter', 'point_on_surface', 'reverse', | ||||||
|  |         'scale', 'snap_to_grid', 'svg', 'sym_difference', 'transform', | ||||||
|  |         'translate', 'union', 'unionagg', | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |     # Specifies whether the Collect and Extent aggregates are supported by the database | ||||||
|  |     @property | ||||||
|  |     def supports_collect_aggr(self): | ||||||
|  |         return 'Collect' in self.connection.ops.valid_aggregates | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def supports_extent_aggr(self): | ||||||
|  |         return 'Extent' in self.connection.ops.valid_aggregates | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def supports_make_line_aggr(self): | ||||||
|  |         return 'MakeLine' in self.connection.ops.valid_aggregates | ||||||
|  |  | ||||||
|  |     def __init__(self, *args): | ||||||
|  |         super(BaseSpatialFeatures, self).__init__(*args) | ||||||
|  |         for method in self.geoqueryset_methods: | ||||||
|  |             # Add dynamically properties for each GQS method, e.g. has_force_rhr_method, etc. | ||||||
|  |             setattr(self.__class__, 'has_%s_method' % method, | ||||||
|  |                     property(partial(BaseSpatialFeatures.has_ops_method, method=method))) | ||||||
|  |  | ||||||
|  |     def has_ops_method(self, method): | ||||||
|  |         return getattr(self.connection.ops, method, False) | ||||||
| @@ -1,8 +1,3 @@ | |||||||
| """ |  | ||||||
| Base/mixin classes for the spatial backend database operations and the |  | ||||||
| `<Backend>SpatialRefSys` model. |  | ||||||
| """ |  | ||||||
| from functools import partial |  | ||||||
| import re | import re | ||||||
| 
 | 
 | ||||||
| from django.contrib.gis import gdal | from django.contrib.gis import gdal | ||||||
| @@ -10,203 +5,6 @@ from django.utils import six | |||||||
| from django.utils.encoding import python_2_unicode_compatible | from django.utils.encoding import python_2_unicode_compatible | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| class BaseSpatialFeatures(object): |  | ||||||
|     gis_enabled = True |  | ||||||
| 
 |  | ||||||
|     # Does the database contain a SpatialRefSys model to store SRID information? |  | ||||||
|     has_spatialrefsys_table = True |  | ||||||
| 
 |  | ||||||
|     # Does the backend support the django.contrib.gis.utils.add_srs_entry() utility? |  | ||||||
|     supports_add_srs_entry = True |  | ||||||
|     # Does the backend introspect GeometryField to its subtypes? |  | ||||||
|     supports_geometry_field_introspection = True |  | ||||||
| 
 |  | ||||||
|     # Reference implementation of 3D functions is: |  | ||||||
|     # http://postgis.net/docs/PostGIS_Special_Functions_Index.html#PostGIS_3D_Functions |  | ||||||
|     supports_3d_functions = False |  | ||||||
|     # Does the database support SRID transform operations? |  | ||||||
|     supports_transform = True |  | ||||||
|     # Do geometric relationship operations operate on real shapes (or only on bounding boxes)? |  | ||||||
|     supports_real_shape_operations = True |  | ||||||
|     # Can geometry fields be null? |  | ||||||
|     supports_null_geometries = True |  | ||||||
|     # Can the `distance` GeoQuerySet method be applied on geodetic coordinate systems? |  | ||||||
|     supports_distance_geodetic = True |  | ||||||
|     # Is the database able to count vertices on polygons (with `num_points`)? |  | ||||||
|     supports_num_points_poly = True |  | ||||||
| 
 |  | ||||||
|     # The following properties indicate if the database backend support |  | ||||||
|     # certain lookups (dwithin, left and right, relate, ...) |  | ||||||
|     supports_distances_lookups = True |  | ||||||
|     supports_left_right_lookups = False |  | ||||||
| 
 |  | ||||||
|     @property |  | ||||||
|     def supports_bbcontains_lookup(self): |  | ||||||
|         return 'bbcontains' in self.connection.ops.gis_operators |  | ||||||
| 
 |  | ||||||
|     @property |  | ||||||
|     def supports_contained_lookup(self): |  | ||||||
|         return 'contained' in self.connection.ops.gis_operators |  | ||||||
| 
 |  | ||||||
|     @property |  | ||||||
|     def supports_dwithin_lookup(self): |  | ||||||
|         return 'dwithin' in self.connection.ops.gis_operators |  | ||||||
| 
 |  | ||||||
|     @property |  | ||||||
|     def supports_relate_lookup(self): |  | ||||||
|         return 'relate' in self.connection.ops.gis_operators |  | ||||||
| 
 |  | ||||||
|     # For each of those methods, the class will have a property named |  | ||||||
|     # `has_<name>_method` (defined in __init__) which accesses connection.ops |  | ||||||
|     # to determine GIS method availability. |  | ||||||
|     geoqueryset_methods = ( |  | ||||||
|         'area', 'centroid', 'difference', 'distance', 'distance_spheroid', |  | ||||||
|         'envelope', 'force_rhr', 'geohash', 'gml', 'intersection', 'kml', |  | ||||||
|         'length', 'num_geom', 'perimeter', 'point_on_surface', 'reverse', |  | ||||||
|         'scale', 'snap_to_grid', 'svg', 'sym_difference', 'transform', |  | ||||||
|         'translate', 'union', 'unionagg', |  | ||||||
|     ) |  | ||||||
| 
 |  | ||||||
|     # Specifies whether the Collect and Extent aggregates are supported by the database |  | ||||||
|     @property |  | ||||||
|     def supports_collect_aggr(self): |  | ||||||
|         return 'Collect' in self.connection.ops.valid_aggregates |  | ||||||
| 
 |  | ||||||
|     @property |  | ||||||
|     def supports_extent_aggr(self): |  | ||||||
|         return 'Extent' in self.connection.ops.valid_aggregates |  | ||||||
| 
 |  | ||||||
|     @property |  | ||||||
|     def supports_make_line_aggr(self): |  | ||||||
|         return 'MakeLine' in self.connection.ops.valid_aggregates |  | ||||||
| 
 |  | ||||||
|     def __init__(self, *args): |  | ||||||
|         super(BaseSpatialFeatures, self).__init__(*args) |  | ||||||
|         for method in self.geoqueryset_methods: |  | ||||||
|             # Add dynamically properties for each GQS method, e.g. has_force_rhr_method, etc. |  | ||||||
|             setattr(self.__class__, 'has_%s_method' % method, |  | ||||||
|                     property(partial(BaseSpatialFeatures.has_ops_method, method=method))) |  | ||||||
| 
 |  | ||||||
|     def has_ops_method(self, method): |  | ||||||
|         return getattr(self.connection.ops, method, False) |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| class BaseSpatialOperations(object): |  | ||||||
|     """ |  | ||||||
|     This module holds the base `BaseSpatialBackend` object, which is |  | ||||||
|     instantiated by each spatial database backend with the features |  | ||||||
|     it has. |  | ||||||
|     """ |  | ||||||
|     truncate_params = {} |  | ||||||
| 
 |  | ||||||
|     # Quick booleans for the type of this spatial backend, and |  | ||||||
|     # an attribute for the spatial database version tuple (if applicable) |  | ||||||
|     postgis = False |  | ||||||
|     spatialite = False |  | ||||||
|     mysql = False |  | ||||||
|     oracle = False |  | ||||||
|     spatial_version = None |  | ||||||
| 
 |  | ||||||
|     # How the geometry column should be selected. |  | ||||||
|     select = None |  | ||||||
| 
 |  | ||||||
|     # Does the spatial database have a geometry or geography type? |  | ||||||
|     geography = False |  | ||||||
|     geometry = False |  | ||||||
| 
 |  | ||||||
|     area = False |  | ||||||
|     centroid = False |  | ||||||
|     difference = False |  | ||||||
|     distance = False |  | ||||||
|     distance_sphere = False |  | ||||||
|     distance_spheroid = False |  | ||||||
|     envelope = False |  | ||||||
|     force_rhr = False |  | ||||||
|     mem_size = False |  | ||||||
|     bounding_circle = False |  | ||||||
|     num_geom = False |  | ||||||
|     num_points = False |  | ||||||
|     perimeter = False |  | ||||||
|     perimeter3d = False |  | ||||||
|     point_on_surface = False |  | ||||||
|     polygonize = False |  | ||||||
|     reverse = False |  | ||||||
|     scale = False |  | ||||||
|     snap_to_grid = False |  | ||||||
|     sym_difference = False |  | ||||||
|     transform = False |  | ||||||
|     translate = False |  | ||||||
|     union = False |  | ||||||
| 
 |  | ||||||
|     # Aggregates |  | ||||||
|     collect = False |  | ||||||
|     extent = False |  | ||||||
|     extent3d = False |  | ||||||
|     make_line = False |  | ||||||
|     unionagg = False |  | ||||||
| 
 |  | ||||||
|     # Serialization |  | ||||||
|     geohash = False |  | ||||||
|     geojson = False |  | ||||||
|     gml = False |  | ||||||
|     kml = False |  | ||||||
|     svg = False |  | ||||||
| 
 |  | ||||||
|     # Constructors |  | ||||||
|     from_text = False |  | ||||||
|     from_wkb = False |  | ||||||
| 
 |  | ||||||
|     # Default conversion functions for aggregates; will be overridden if implemented |  | ||||||
|     # for the spatial backend. |  | ||||||
|     def convert_extent(self, box, srid): |  | ||||||
|         raise NotImplementedError('Aggregate extent not implemented for this spatial backend.') |  | ||||||
| 
 |  | ||||||
|     def convert_extent3d(self, box, srid): |  | ||||||
|         raise NotImplementedError('Aggregate 3D extent not implemented for this spatial backend.') |  | ||||||
| 
 |  | ||||||
|     def convert_geom(self, geom_val, geom_field): |  | ||||||
|         raise NotImplementedError('Aggregate method not implemented for this spatial backend.') |  | ||||||
| 
 |  | ||||||
|     # For quoting column values, rather than columns. |  | ||||||
|     def geo_quote_name(self, name): |  | ||||||
|         return "'%s'" % name |  | ||||||
| 
 |  | ||||||
|     # GeometryField operations |  | ||||||
|     def geo_db_type(self, f): |  | ||||||
|         """ |  | ||||||
|         Returns the database column type for the geometry field on |  | ||||||
|         the spatial backend. |  | ||||||
|         """ |  | ||||||
|         raise NotImplementedError('subclasses of BaseSpatialOperations must provide a geo_db_type() method') |  | ||||||
| 
 |  | ||||||
|     def get_distance(self, f, value, lookup_type): |  | ||||||
|         """ |  | ||||||
|         Returns the distance parameters for the given geometry field, |  | ||||||
|         lookup value, and lookup type. |  | ||||||
|         """ |  | ||||||
|         raise NotImplementedError('Distance operations not available on this spatial backend.') |  | ||||||
| 
 |  | ||||||
|     def get_geom_placeholder(self, f, value, compiler): |  | ||||||
|         """ |  | ||||||
|         Returns the placeholder for the given geometry field with the given |  | ||||||
|         value.  Depending on the spatial backend, the placeholder may contain a |  | ||||||
|         stored procedure call to the transformation function of the spatial |  | ||||||
|         backend. |  | ||||||
|         """ |  | ||||||
|         raise NotImplementedError('subclasses of BaseSpatialOperations must provide a geo_db_placeholder() method') |  | ||||||
| 
 |  | ||||||
|     # Spatial SQL Construction |  | ||||||
|     def spatial_aggregate_sql(self, agg): |  | ||||||
|         raise NotImplementedError('Aggregate support not implemented for this spatial backend.') |  | ||||||
| 
 |  | ||||||
|     # Routines for getting the OGC-compliant models. |  | ||||||
|     def geometry_columns(self): |  | ||||||
|         raise NotImplementedError('subclasses of BaseSpatialOperations must a provide geometry_columns() method') |  | ||||||
| 
 |  | ||||||
|     def spatial_ref_sys(self): |  | ||||||
|         raise NotImplementedError('subclasses of BaseSpatialOperations must a provide spatial_ref_sys() method') |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| @python_2_unicode_compatible | @python_2_unicode_compatible | ||||||
| class SpatialRefSysMixin(object): | class SpatialRefSysMixin(object): | ||||||
|     """ |     """ | ||||||
							
								
								
									
										114
									
								
								django/contrib/gis/db/backends/base/operations.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										114
									
								
								django/contrib/gis/db/backends/base/operations.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,114 @@ | |||||||
|  | class BaseSpatialOperations(object): | ||||||
|  |     """ | ||||||
|  |     This module holds the base `BaseSpatialBackend` object, which is | ||||||
|  |     instantiated by each spatial database backend with the features | ||||||
|  |     it has. | ||||||
|  |     """ | ||||||
|  |     truncate_params = {} | ||||||
|  |  | ||||||
|  |     # Quick booleans for the type of this spatial backend, and | ||||||
|  |     # an attribute for the spatial database version tuple (if applicable) | ||||||
|  |     postgis = False | ||||||
|  |     spatialite = False | ||||||
|  |     mysql = False | ||||||
|  |     oracle = False | ||||||
|  |     spatial_version = None | ||||||
|  |  | ||||||
|  |     # How the geometry column should be selected. | ||||||
|  |     select = None | ||||||
|  |  | ||||||
|  |     # Does the spatial database have a geometry or geography type? | ||||||
|  |     geography = False | ||||||
|  |     geometry = False | ||||||
|  |  | ||||||
|  |     area = False | ||||||
|  |     centroid = False | ||||||
|  |     difference = False | ||||||
|  |     distance = False | ||||||
|  |     distance_sphere = False | ||||||
|  |     distance_spheroid = False | ||||||
|  |     envelope = False | ||||||
|  |     force_rhr = False | ||||||
|  |     mem_size = False | ||||||
|  |     bounding_circle = False | ||||||
|  |     num_geom = False | ||||||
|  |     num_points = False | ||||||
|  |     perimeter = False | ||||||
|  |     perimeter3d = False | ||||||
|  |     point_on_surface = False | ||||||
|  |     polygonize = False | ||||||
|  |     reverse = False | ||||||
|  |     scale = False | ||||||
|  |     snap_to_grid = False | ||||||
|  |     sym_difference = False | ||||||
|  |     transform = False | ||||||
|  |     translate = False | ||||||
|  |     union = False | ||||||
|  |  | ||||||
|  |     # Aggregates | ||||||
|  |     collect = False | ||||||
|  |     extent = False | ||||||
|  |     extent3d = False | ||||||
|  |     make_line = False | ||||||
|  |     unionagg = False | ||||||
|  |  | ||||||
|  |     # Serialization | ||||||
|  |     geohash = False | ||||||
|  |     geojson = False | ||||||
|  |     gml = False | ||||||
|  |     kml = False | ||||||
|  |     svg = False | ||||||
|  |  | ||||||
|  |     # Constructors | ||||||
|  |     from_text = False | ||||||
|  |     from_wkb = False | ||||||
|  |  | ||||||
|  |     # Default conversion functions for aggregates; will be overridden if implemented | ||||||
|  |     # for the spatial backend. | ||||||
|  |     def convert_extent(self, box, srid): | ||||||
|  |         raise NotImplementedError('Aggregate extent not implemented for this spatial backend.') | ||||||
|  |  | ||||||
|  |     def convert_extent3d(self, box, srid): | ||||||
|  |         raise NotImplementedError('Aggregate 3D extent not implemented for this spatial backend.') | ||||||
|  |  | ||||||
|  |     def convert_geom(self, geom_val, geom_field): | ||||||
|  |         raise NotImplementedError('Aggregate method not implemented for this spatial backend.') | ||||||
|  |  | ||||||
|  |     # For quoting column values, rather than columns. | ||||||
|  |     def geo_quote_name(self, name): | ||||||
|  |         return "'%s'" % name | ||||||
|  |  | ||||||
|  |     # GeometryField operations | ||||||
|  |     def geo_db_type(self, f): | ||||||
|  |         """ | ||||||
|  |         Returns the database column type for the geometry field on | ||||||
|  |         the spatial backend. | ||||||
|  |         """ | ||||||
|  |         raise NotImplementedError('subclasses of BaseSpatialOperations must provide a geo_db_type() method') | ||||||
|  |  | ||||||
|  |     def get_distance(self, f, value, lookup_type): | ||||||
|  |         """ | ||||||
|  |         Returns the distance parameters for the given geometry field, | ||||||
|  |         lookup value, and lookup type. | ||||||
|  |         """ | ||||||
|  |         raise NotImplementedError('Distance operations not available on this spatial backend.') | ||||||
|  |  | ||||||
|  |     def get_geom_placeholder(self, f, value, compiler): | ||||||
|  |         """ | ||||||
|  |         Returns the placeholder for the given geometry field with the given | ||||||
|  |         value.  Depending on the spatial backend, the placeholder may contain a | ||||||
|  |         stored procedure call to the transformation function of the spatial | ||||||
|  |         backend. | ||||||
|  |         """ | ||||||
|  |         raise NotImplementedError('subclasses of BaseSpatialOperations must provide a geo_db_placeholder() method') | ||||||
|  |  | ||||||
|  |     # Spatial SQL Construction | ||||||
|  |     def spatial_aggregate_sql(self, agg): | ||||||
|  |         raise NotImplementedError('Aggregate support not implemented for this spatial backend.') | ||||||
|  |  | ||||||
|  |     # Routines for getting the OGC-compliant models. | ||||||
|  |     def geometry_columns(self): | ||||||
|  |         raise NotImplementedError('subclasses of BaseSpatialOperations must a provide geometry_columns() method') | ||||||
|  |  | ||||||
|  |     def spatial_ref_sys(self): | ||||||
|  |         raise NotImplementedError('subclasses of BaseSpatialOperations must a provide spatial_ref_sys() method') | ||||||
| @@ -1,22 +1,10 @@ | |||||||
| from django.db.backends.mysql.base import ( | from django.db.backends.mysql.base import DatabaseWrapper as MySQLDatabaseWrapper | ||||||
|     DatabaseWrapper as MySQLDatabaseWrapper, |  | ||||||
|     DatabaseFeatures as MySQLDatabaseFeatures, |  | ||||||
| ) |  | ||||||
| from django.contrib.gis.db.backends.base import BaseSpatialFeatures |  | ||||||
| from django.contrib.gis.db.backends.mysql.creation import MySQLCreation |  | ||||||
| from django.contrib.gis.db.backends.mysql.introspection import MySQLIntrospection |  | ||||||
| from django.contrib.gis.db.backends.mysql.operations import MySQLOperations |  | ||||||
| from django.contrib.gis.db.backends.mysql.schema import MySQLGISSchemaEditor |  | ||||||
|  |  | ||||||
|  | from .creation import MySQLCreation | ||||||
| class DatabaseFeatures(BaseSpatialFeatures, MySQLDatabaseFeatures): | from .features import DatabaseFeatures | ||||||
|     has_spatialrefsys_table = False | from .introspection import MySQLIntrospection | ||||||
|     supports_add_srs_entry = False | from .operations import MySQLOperations | ||||||
|     supports_distances_lookups = False | from .schema import MySQLGISSchemaEditor | ||||||
|     supports_transform = False |  | ||||||
|     supports_real_shape_operations = False |  | ||||||
|     supports_null_geometries = False |  | ||||||
|     supports_num_points_poly = False |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class DatabaseWrapper(MySQLDatabaseWrapper): | class DatabaseWrapper(MySQLDatabaseWrapper): | ||||||
|   | |||||||
							
								
								
									
										12
									
								
								django/contrib/gis/db/backends/mysql/features.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								django/contrib/gis/db/backends/mysql/features.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | |||||||
|  | from django.contrib.gis.db.backends.base.features import BaseSpatialFeatures | ||||||
|  | from django.db.backends.mysql.features import DatabaseFeatures as MySQLDatabaseFeatures | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class DatabaseFeatures(BaseSpatialFeatures, MySQLDatabaseFeatures): | ||||||
|  |     has_spatialrefsys_table = False | ||||||
|  |     supports_add_srs_entry = False | ||||||
|  |     supports_distances_lookups = False | ||||||
|  |     supports_transform = False | ||||||
|  |     supports_real_shape_operations = False | ||||||
|  |     supports_null_geometries = False | ||||||
|  |     supports_num_points_poly = False | ||||||
| @@ -1,8 +1,7 @@ | |||||||
| from django.db.backends.mysql.base import DatabaseOperations | from django.contrib.gis.db.backends.base.adapter import WKTAdapter | ||||||
|  | from django.contrib.gis.db.backends.base.operations import BaseSpatialOperations | ||||||
| from django.contrib.gis.db.backends.adapter import WKTAdapter |  | ||||||
| from django.contrib.gis.db.backends.base import BaseSpatialOperations |  | ||||||
| from django.contrib.gis.db.backends.utils import SpatialOperator | from django.contrib.gis.db.backends.utils import SpatialOperator | ||||||
|  | from django.db.backends.mysql.operations import DatabaseOperations | ||||||
|  |  | ||||||
|  |  | ||||||
| class MySQLOperations(DatabaseOperations, BaseSpatialOperations): | class MySQLOperations(DatabaseOperations, BaseSpatialOperations): | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| from cx_Oracle import CLOB | from cx_Oracle import CLOB | ||||||
| from django.contrib.gis.db.backends.adapter import WKTAdapter | from django.contrib.gis.db.backends.base.adapter import WKTAdapter | ||||||
|  |  | ||||||
|  |  | ||||||
| class OracleSpatialAdapter(WKTAdapter): | class OracleSpatialAdapter(WKTAdapter): | ||||||
|   | |||||||
| @@ -1,17 +1,10 @@ | |||||||
| from django.db.backends.oracle.base import ( | from django.db.backends.oracle.base import DatabaseWrapper as OracleDatabaseWrapper | ||||||
|     DatabaseWrapper as OracleDatabaseWrapper, |  | ||||||
|     DatabaseFeatures as OracleDatabaseFeatures, |  | ||||||
| ) |  | ||||||
| from django.contrib.gis.db.backends.base import BaseSpatialFeatures |  | ||||||
| from django.contrib.gis.db.backends.oracle.creation import OracleCreation |  | ||||||
| from django.contrib.gis.db.backends.oracle.introspection import OracleIntrospection |  | ||||||
| from django.contrib.gis.db.backends.oracle.operations import OracleOperations |  | ||||||
| from django.contrib.gis.db.backends.oracle.schema import OracleGISSchemaEditor |  | ||||||
|  |  | ||||||
|  | from .creation import OracleCreation | ||||||
| class DatabaseFeatures(BaseSpatialFeatures, OracleDatabaseFeatures): | from .features import DatabaseFeatures | ||||||
|     supports_add_srs_entry = False | from .introspection import OracleIntrospection | ||||||
|     supports_geometry_field_introspection = False | from .operations import OracleOperations | ||||||
|  | from .schema import OracleGISSchemaEditor | ||||||
|  |  | ||||||
|  |  | ||||||
| class DatabaseWrapper(OracleDatabaseWrapper): | class DatabaseWrapper(OracleDatabaseWrapper): | ||||||
|   | |||||||
							
								
								
									
										7
									
								
								django/contrib/gis/db/backends/oracle/features.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								django/contrib/gis/db/backends/oracle/features.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | |||||||
|  | from django.contrib.gis.db.backends.base.features import BaseSpatialFeatures | ||||||
|  | from django.db.backends.oracle.features import DatabaseFeatures as OracleDatabaseFeatures | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class DatabaseFeatures(BaseSpatialFeatures, OracleDatabaseFeatures): | ||||||
|  |     supports_add_srs_entry = False | ||||||
|  |     supports_geometry_field_introspection = False | ||||||
| @@ -8,7 +8,7 @@ | |||||||
|  model and the `SDO_COORD_REF_SYS` is used for the SpatialRefSys model. |  model and the `SDO_COORD_REF_SYS` is used for the SpatialRefSys model. | ||||||
| """ | """ | ||||||
| from django.contrib.gis.db import models | from django.contrib.gis.db import models | ||||||
| from django.contrib.gis.db.backends.base import SpatialRefSysMixin | from django.contrib.gis.db.backends.base.models import SpatialRefSysMixin | ||||||
| from django.utils.encoding import python_2_unicode_compatible | from django.utils.encoding import python_2_unicode_compatible | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -9,12 +9,13 @@ | |||||||
| """ | """ | ||||||
| import re | import re | ||||||
|  |  | ||||||
| from django.db.backends.oracle.base import DatabaseOperations, Database | from django.contrib.gis.db.backends.base.operations import BaseSpatialOperations | ||||||
| from django.contrib.gis.db.backends.base import BaseSpatialOperations |  | ||||||
| from django.contrib.gis.db.backends.oracle.adapter import OracleSpatialAdapter | from django.contrib.gis.db.backends.oracle.adapter import OracleSpatialAdapter | ||||||
| from django.contrib.gis.db.backends.utils import SpatialOperator | from django.contrib.gis.db.backends.utils import SpatialOperator | ||||||
| from django.contrib.gis.geometry.backend import Geometry | from django.contrib.gis.geometry.backend import Geometry | ||||||
| from django.contrib.gis.measure import Distance | from django.contrib.gis.measure import Distance | ||||||
|  | from django.db.backends.oracle.base import Database | ||||||
|  | from django.db.backends.oracle.operations import DatabaseOperations | ||||||
| from django.utils import six | from django.utils import six | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,20 +1,15 @@ | |||||||
| from django.conf import settings | from django.conf import settings | ||||||
| from django.db.backends import NO_DB_ALIAS | from django.db.backends.base.base import NO_DB_ALIAS | ||||||
| from django.db.backends.postgresql_psycopg2.base import ( | from django.db.backends.postgresql_psycopg2.base import ( | ||||||
|     DatabaseWrapper as Psycopg2DatabaseWrapper, |     DatabaseWrapper as Psycopg2DatabaseWrapper, | ||||||
|     DatabaseFeatures as Psycopg2DatabaseFeatures |  | ||||||
| ) | ) | ||||||
| from django.contrib.gis.db.backends.base import BaseSpatialFeatures |  | ||||||
| from django.contrib.gis.db.backends.postgis.creation import PostGISCreation |  | ||||||
| from django.contrib.gis.db.backends.postgis.introspection import PostGISIntrospection |  | ||||||
| from django.contrib.gis.db.backends.postgis.operations import PostGISOperations |  | ||||||
| from django.contrib.gis.db.backends.postgis.schema import PostGISSchemaEditor |  | ||||||
| from django.utils.functional import cached_property | from django.utils.functional import cached_property | ||||||
|  |  | ||||||
|  | from .creation import PostGISCreation | ||||||
| class DatabaseFeatures(BaseSpatialFeatures, Psycopg2DatabaseFeatures): | from .features import DatabaseFeatures | ||||||
|     supports_3d_functions = True | from .introspection import PostGISIntrospection | ||||||
|     supports_left_right_lookups = True | from .operations import PostGISOperations | ||||||
|  | from .schema import PostGISSchemaEditor | ||||||
|  |  | ||||||
|  |  | ||||||
| class DatabaseWrapper(Psycopg2DatabaseWrapper): | class DatabaseWrapper(Psycopg2DatabaseWrapper): | ||||||
|   | |||||||
							
								
								
									
										9
									
								
								django/contrib/gis/db/backends/postgis/features.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								django/contrib/gis/db/backends/postgis/features.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | |||||||
|  | from django.contrib.gis.db.backends.base.features import BaseSpatialFeatures | ||||||
|  | from django.db.backends.postgresql_psycopg2.features import ( | ||||||
|  |     DatabaseFeatures as Psycopg2DatabaseFeatures, | ||||||
|  | ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class DatabaseFeatures(BaseSpatialFeatures, Psycopg2DatabaseFeatures): | ||||||
|  |     supports_3d_functions = True | ||||||
|  |     supports_left_right_lookups = True | ||||||
| @@ -2,7 +2,7 @@ | |||||||
|  The GeometryColumns and SpatialRefSys models for the PostGIS backend. |  The GeometryColumns and SpatialRefSys models for the PostGIS backend. | ||||||
| """ | """ | ||||||
| from django.db import models | from django.db import models | ||||||
| from django.contrib.gis.db.backends.base import SpatialRefSysMixin | from django.contrib.gis.db.backends.base.models import SpatialRefSysMixin | ||||||
| from django.utils.encoding import python_2_unicode_compatible | from django.utils.encoding import python_2_unicode_compatible | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,13 +1,13 @@ | |||||||
| import re | import re | ||||||
|  |  | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| from django.contrib.gis.db.backends.base import BaseSpatialOperations | from django.contrib.gis.db.backends.base.operations import BaseSpatialOperations | ||||||
| from django.contrib.gis.db.backends.postgis.adapter import PostGISAdapter | from django.contrib.gis.db.backends.postgis.adapter import PostGISAdapter | ||||||
| from django.contrib.gis.db.backends.utils import SpatialOperator | from django.contrib.gis.db.backends.utils import SpatialOperator | ||||||
| from django.contrib.gis.geometry.backend import Geometry | from django.contrib.gis.geometry.backend import Geometry | ||||||
| from django.contrib.gis.measure import Distance | from django.contrib.gis.measure import Distance | ||||||
| from django.core.exceptions import ImproperlyConfigured | from django.core.exceptions import ImproperlyConfigured | ||||||
| from django.db.backends.postgresql_psycopg2.base import DatabaseOperations | from django.db.backends.postgresql_psycopg2.operations import DatabaseOperations | ||||||
| from django.db.utils import ProgrammingError | from django.db.utils import ProgrammingError | ||||||
| from django.utils.functional import cached_property | from django.utils.functional import cached_property | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| from django.db.backends.sqlite3.base import Database | from django.db.backends.sqlite3.base import Database | ||||||
| from django.contrib.gis.db.backends.adapter import WKTAdapter | from django.contrib.gis.db.backends.base.adapter import WKTAdapter | ||||||
|  |  | ||||||
|  |  | ||||||
| class SpatiaLiteAdapter(WKTAdapter): | class SpatiaLiteAdapter(WKTAdapter): | ||||||
|   | |||||||
| @@ -1,32 +1,19 @@ | |||||||
| import sys | import sys | ||||||
| from ctypes.util import find_library | from ctypes.util import find_library | ||||||
|  |  | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
|  |  | ||||||
| from django.core.exceptions import ImproperlyConfigured | from django.core.exceptions import ImproperlyConfigured | ||||||
| from django.db.backends.sqlite3.base import (Database, | from django.db.backends.sqlite3.base import ( | ||||||
|     DatabaseWrapper as SQLiteDatabaseWrapper, |     Database, DatabaseWrapper as SQLiteDatabaseWrapper, SQLiteCursorWrapper, | ||||||
|     DatabaseFeatures as SQLiteDatabaseFeatures, SQLiteCursorWrapper) | ) | ||||||
| from django.contrib.gis.db.backends.base import BaseSpatialFeatures |  | ||||||
| from django.contrib.gis.db.backends.spatialite.client import SpatiaLiteClient |  | ||||||
| from django.contrib.gis.db.backends.spatialite.creation import SpatiaLiteCreation |  | ||||||
| from django.contrib.gis.db.backends.spatialite.introspection import SpatiaLiteIntrospection |  | ||||||
| from django.contrib.gis.db.backends.spatialite.operations import SpatiaLiteOperations |  | ||||||
| from django.contrib.gis.db.backends.spatialite.schema import SpatialiteSchemaEditor |  | ||||||
| from django.utils import six | from django.utils import six | ||||||
| from django.utils.functional import cached_property |  | ||||||
|  |  | ||||||
|  | from .client import SpatiaLiteClient | ||||||
| class DatabaseFeatures(BaseSpatialFeatures, SQLiteDatabaseFeatures): | from .creation import SpatiaLiteCreation | ||||||
|     supports_distance_geodetic = False | from .features import DatabaseFeatures | ||||||
|     # SpatiaLite can only count vertices in LineStrings | from .introspection import SpatiaLiteIntrospection | ||||||
|     supports_num_points_poly = False | from .operations import SpatiaLiteOperations | ||||||
|  | from .schema import SpatialiteSchemaEditor | ||||||
|     @cached_property |  | ||||||
|     def supports_initspatialmetadata_in_one_transaction(self): |  | ||||||
|         # SpatiaLite 4.1+ support initializing all metadata in one transaction |  | ||||||
|         # which can result in a significant performance improvement when |  | ||||||
|         # creating the database. |  | ||||||
|         return self.connection.ops.spatial_version >= (4, 1, 0) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class DatabaseWrapper(SQLiteDatabaseWrapper): | class DatabaseWrapper(SQLiteDatabaseWrapper): | ||||||
|   | |||||||
							
								
								
									
										16
									
								
								django/contrib/gis/db/backends/spatialite/features.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								django/contrib/gis/db/backends/spatialite/features.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | |||||||
|  | from django.contrib.gis.db.backends.base.features import BaseSpatialFeatures | ||||||
|  | from django.db.backends.sqlite3.features import DatabaseFeatures as SQLiteDatabaseFeatures | ||||||
|  | from django.utils.functional import cached_property | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class DatabaseFeatures(BaseSpatialFeatures, SQLiteDatabaseFeatures): | ||||||
|  |     supports_distance_geodetic = False | ||||||
|  |     # SpatiaLite can only count vertices in LineStrings | ||||||
|  |     supports_num_points_poly = False | ||||||
|  |  | ||||||
|  |     @cached_property | ||||||
|  |     def supports_initspatialmetadata_in_one_transaction(self): | ||||||
|  |         # SpatiaLite 4.1+ support initializing all metadata in one transaction | ||||||
|  |         # which can result in a significant performance improvement when | ||||||
|  |         # creating the database. | ||||||
|  |         return self.connection.ops.spatial_version >= (4, 1, 0) | ||||||
| @@ -3,7 +3,7 @@ | |||||||
| """ | """ | ||||||
| from django.db import connection, models | from django.db import connection, models | ||||||
| from django.db.backends.signals import connection_created | from django.db.backends.signals import connection_created | ||||||
| from django.contrib.gis.db.backends.base import SpatialRefSysMixin | from django.contrib.gis.db.backends.base.models import SpatialRefSysMixin | ||||||
| from django.contrib.gis.db.backends.spatialite.base import DatabaseWrapper | from django.contrib.gis.db.backends.spatialite.base import DatabaseWrapper | ||||||
| from django.utils.encoding import python_2_unicode_compatible | from django.utils.encoding import python_2_unicode_compatible | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,13 +1,13 @@ | |||||||
| import re | import re | ||||||
| import sys | import sys | ||||||
|  |  | ||||||
| from django.contrib.gis.db.backends.base import BaseSpatialOperations | from django.contrib.gis.db.backends.base.operations import BaseSpatialOperations | ||||||
| from django.contrib.gis.db.backends.utils import SpatialOperator | from django.contrib.gis.db.backends.utils import SpatialOperator | ||||||
| from django.contrib.gis.db.backends.spatialite.adapter import SpatiaLiteAdapter | from django.contrib.gis.db.backends.spatialite.adapter import SpatiaLiteAdapter | ||||||
| from django.contrib.gis.geometry.backend import Geometry | from django.contrib.gis.geometry.backend import Geometry | ||||||
| from django.contrib.gis.measure import Distance | from django.contrib.gis.measure import Distance | ||||||
| from django.core.exceptions import ImproperlyConfigured | from django.core.exceptions import ImproperlyConfigured | ||||||
| from django.db.backends.sqlite3.base import DatabaseOperations | from django.db.backends.sqlite3.operations import DatabaseOperations | ||||||
| from django.db.utils import DatabaseError | from django.db.utils import DatabaseError | ||||||
| from django.utils import six | from django.utils import six | ||||||
| from django.utils.functional import cached_property | from django.utils.functional import cached_property | ||||||
|   | |||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										0
									
								
								django/db/backends/base/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								django/db/backends/base/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										507
									
								
								django/db/backends/base/base.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										507
									
								
								django/db/backends/base/base.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,507 @@ | |||||||
|  | from collections import deque | ||||||
|  | from contextlib import contextmanager | ||||||
|  | import time | ||||||
|  | import warnings | ||||||
|  |  | ||||||
|  | from django.conf import settings | ||||||
|  | from django.db import DEFAULT_DB_ALIAS | ||||||
|  | from django.db.backends.signals import connection_created | ||||||
|  | from django.db.backends import utils | ||||||
|  | from django.db.transaction import TransactionManagementError | ||||||
|  | from django.db.utils import DatabaseError, DatabaseErrorWrapper | ||||||
|  | from django.utils.functional import cached_property | ||||||
|  | try: | ||||||
|  |     from django.utils.six.moves import _thread as thread | ||||||
|  | except ImportError: | ||||||
|  |     from django.utils.six.moves import _dummy_thread as thread | ||||||
|  |  | ||||||
|  |  | ||||||
|  | NO_DB_ALIAS = '__no_db__' | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class BaseDatabaseWrapper(object): | ||||||
|  |     """ | ||||||
|  |     Represents a database connection. | ||||||
|  |     """ | ||||||
|  |     # Mapping of Field objects to their column types. | ||||||
|  |     data_types = {} | ||||||
|  |     # Mapping of Field objects to their SQL suffix such as AUTOINCREMENT. | ||||||
|  |     data_types_suffix = {} | ||||||
|  |     # Mapping of Field objects to their SQL for CHECK constraints. | ||||||
|  |     data_type_check_constraints = {} | ||||||
|  |     ops = None | ||||||
|  |     vendor = 'unknown' | ||||||
|  |     SchemaEditorClass = None | ||||||
|  |  | ||||||
|  |     queries_limit = 9000 | ||||||
|  |  | ||||||
|  |     def __init__(self, settings_dict, alias=DEFAULT_DB_ALIAS, | ||||||
|  |                  allow_thread_sharing=False): | ||||||
|  |         # Connection related attributes. | ||||||
|  |         # The underlying database connection. | ||||||
|  |         self.connection = None | ||||||
|  |         # `settings_dict` should be a dictionary containing keys such as | ||||||
|  |         # NAME, USER, etc. It's called `settings_dict` instead of `settings` | ||||||
|  |         # to disambiguate it from Django settings modules. | ||||||
|  |         self.settings_dict = settings_dict | ||||||
|  |         self.alias = alias | ||||||
|  |         # Query logging in debug mode or when explicitly enabled. | ||||||
|  |         self.queries_log = deque(maxlen=self.queries_limit) | ||||||
|  |         self.force_debug_cursor = False | ||||||
|  |  | ||||||
|  |         # Transaction related attributes. | ||||||
|  |         # Tracks if the connection is in autocommit mode. Per PEP 249, by | ||||||
|  |         # default, it isn't. | ||||||
|  |         self.autocommit = False | ||||||
|  |         # Tracks if the connection is in a transaction managed by 'atomic'. | ||||||
|  |         self.in_atomic_block = False | ||||||
|  |         # Increment to generate unique savepoint ids. | ||||||
|  |         self.savepoint_state = 0 | ||||||
|  |         # List of savepoints created by 'atomic'. | ||||||
|  |         self.savepoint_ids = [] | ||||||
|  |         # Tracks if the outermost 'atomic' block should commit on exit, | ||||||
|  |         # ie. if autocommit was active on entry. | ||||||
|  |         self.commit_on_exit = True | ||||||
|  |         # Tracks if the transaction should be rolled back to the next | ||||||
|  |         # available savepoint because of an exception in an inner block. | ||||||
|  |         self.needs_rollback = False | ||||||
|  |  | ||||||
|  |         # Connection termination related attributes. | ||||||
|  |         self.close_at = None | ||||||
|  |         self.closed_in_transaction = False | ||||||
|  |         self.errors_occurred = False | ||||||
|  |  | ||||||
|  |         # Thread-safety related attributes. | ||||||
|  |         self.allow_thread_sharing = allow_thread_sharing | ||||||
|  |         self._thread_ident = thread.get_ident() | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def queries_logged(self): | ||||||
|  |         return self.force_debug_cursor or settings.DEBUG | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def queries(self): | ||||||
|  |         if len(self.queries_log) == self.queries_log.maxlen: | ||||||
|  |             warnings.warn( | ||||||
|  |                 "Limit for query logging exceeded, only the last {} queries " | ||||||
|  |                 "will be returned.".format(self.queries_log.maxlen)) | ||||||
|  |         return list(self.queries_log) | ||||||
|  |  | ||||||
|  |     ##### Backend-specific methods for creating connections and cursors ##### | ||||||
|  |  | ||||||
|  |     def get_connection_params(self): | ||||||
|  |         """Returns a dict of parameters suitable for get_new_connection.""" | ||||||
|  |         raise NotImplementedError('subclasses of BaseDatabaseWrapper may require a get_connection_params() method') | ||||||
|  |  | ||||||
|  |     def get_new_connection(self, conn_params): | ||||||
|  |         """Opens a connection to the database.""" | ||||||
|  |         raise NotImplementedError('subclasses of BaseDatabaseWrapper may require a get_new_connection() method') | ||||||
|  |  | ||||||
|  |     def init_connection_state(self): | ||||||
|  |         """Initializes the database connection settings.""" | ||||||
|  |         raise NotImplementedError('subclasses of BaseDatabaseWrapper may require an init_connection_state() method') | ||||||
|  |  | ||||||
|  |     def create_cursor(self): | ||||||
|  |         """Creates a cursor. Assumes that a connection is established.""" | ||||||
|  |         raise NotImplementedError('subclasses of BaseDatabaseWrapper may require a create_cursor() method') | ||||||
|  |  | ||||||
|  |     ##### Backend-specific methods for creating connections ##### | ||||||
|  |  | ||||||
|  |     def connect(self): | ||||||
|  |         """Connects to the database. Assumes that the connection is closed.""" | ||||||
|  |         # In case the previous connection was closed while in an atomic block | ||||||
|  |         self.in_atomic_block = False | ||||||
|  |         self.savepoint_ids = [] | ||||||
|  |         self.needs_rollback = False | ||||||
|  |         # Reset parameters defining when to close the connection | ||||||
|  |         max_age = self.settings_dict['CONN_MAX_AGE'] | ||||||
|  |         self.close_at = None if max_age is None else time.time() + max_age | ||||||
|  |         self.closed_in_transaction = False | ||||||
|  |         self.errors_occurred = False | ||||||
|  |         # Establish the connection | ||||||
|  |         conn_params = self.get_connection_params() | ||||||
|  |         self.connection = self.get_new_connection(conn_params) | ||||||
|  |         self.set_autocommit(self.settings_dict['AUTOCOMMIT']) | ||||||
|  |         self.init_connection_state() | ||||||
|  |         connection_created.send(sender=self.__class__, connection=self) | ||||||
|  |  | ||||||
|  |     def ensure_connection(self): | ||||||
|  |         """ | ||||||
|  |         Guarantees that a connection to the database is established. | ||||||
|  |         """ | ||||||
|  |         if self.connection is None: | ||||||
|  |             with self.wrap_database_errors: | ||||||
|  |                 self.connect() | ||||||
|  |  | ||||||
|  |     ##### Backend-specific wrappers for PEP-249 connection methods ##### | ||||||
|  |  | ||||||
|  |     def _cursor(self): | ||||||
|  |         self.ensure_connection() | ||||||
|  |         with self.wrap_database_errors: | ||||||
|  |             return self.create_cursor() | ||||||
|  |  | ||||||
|  |     def _commit(self): | ||||||
|  |         if self.connection is not None: | ||||||
|  |             with self.wrap_database_errors: | ||||||
|  |                 return self.connection.commit() | ||||||
|  |  | ||||||
|  |     def _rollback(self): | ||||||
|  |         if self.connection is not None: | ||||||
|  |             with self.wrap_database_errors: | ||||||
|  |                 return self.connection.rollback() | ||||||
|  |  | ||||||
|  |     def _close(self): | ||||||
|  |         if self.connection is not None: | ||||||
|  |             with self.wrap_database_errors: | ||||||
|  |                 return self.connection.close() | ||||||
|  |  | ||||||
|  |     ##### Generic wrappers for PEP-249 connection methods ##### | ||||||
|  |  | ||||||
|  |     def cursor(self): | ||||||
|  |         """ | ||||||
|  |         Creates a cursor, opening a connection if necessary. | ||||||
|  |         """ | ||||||
|  |         self.validate_thread_sharing() | ||||||
|  |         if self.queries_logged: | ||||||
|  |             cursor = self.make_debug_cursor(self._cursor()) | ||||||
|  |         else: | ||||||
|  |             cursor = self.make_cursor(self._cursor()) | ||||||
|  |         return cursor | ||||||
|  |  | ||||||
|  |     def commit(self): | ||||||
|  |         """ | ||||||
|  |         Commits a transaction and resets the dirty flag. | ||||||
|  |         """ | ||||||
|  |         self.validate_thread_sharing() | ||||||
|  |         self.validate_no_atomic_block() | ||||||
|  |         self._commit() | ||||||
|  |         # A successful commit means that the database connection works. | ||||||
|  |         self.errors_occurred = False | ||||||
|  |  | ||||||
|  |     def rollback(self): | ||||||
|  |         """ | ||||||
|  |         Rolls back a transaction and resets the dirty flag. | ||||||
|  |         """ | ||||||
|  |         self.validate_thread_sharing() | ||||||
|  |         self.validate_no_atomic_block() | ||||||
|  |         self._rollback() | ||||||
|  |         # A successful rollback means that the database connection works. | ||||||
|  |         self.errors_occurred = False | ||||||
|  |  | ||||||
|  |     def close(self): | ||||||
|  |         """ | ||||||
|  |         Closes the connection to the database. | ||||||
|  |         """ | ||||||
|  |         self.validate_thread_sharing() | ||||||
|  |         # Don't call validate_no_atomic_block() to avoid making it difficult | ||||||
|  |         # to get rid of a connection in an invalid state. The next connect() | ||||||
|  |         # will reset the transaction state anyway. | ||||||
|  |         if self.closed_in_transaction or self.connection is None: | ||||||
|  |             return | ||||||
|  |         try: | ||||||
|  |             self._close() | ||||||
|  |         finally: | ||||||
|  |             if self.in_atomic_block: | ||||||
|  |                 self.closed_in_transaction = True | ||||||
|  |                 self.needs_rollback = True | ||||||
|  |             else: | ||||||
|  |                 self.connection = None | ||||||
|  |  | ||||||
|  |     ##### Backend-specific savepoint management methods ##### | ||||||
|  |  | ||||||
|  |     def _savepoint(self, sid): | ||||||
|  |         with self.cursor() as cursor: | ||||||
|  |             cursor.execute(self.ops.savepoint_create_sql(sid)) | ||||||
|  |  | ||||||
|  |     def _savepoint_rollback(self, sid): | ||||||
|  |         with self.cursor() as cursor: | ||||||
|  |             cursor.execute(self.ops.savepoint_rollback_sql(sid)) | ||||||
|  |  | ||||||
|  |     def _savepoint_commit(self, sid): | ||||||
|  |         with self.cursor() as cursor: | ||||||
|  |             cursor.execute(self.ops.savepoint_commit_sql(sid)) | ||||||
|  |  | ||||||
|  |     def _savepoint_allowed(self): | ||||||
|  |         # Savepoints cannot be created outside a transaction | ||||||
|  |         return self.features.uses_savepoints and not self.get_autocommit() | ||||||
|  |  | ||||||
|  |     ##### Generic savepoint management methods ##### | ||||||
|  |  | ||||||
|  |     def savepoint(self): | ||||||
|  |         """ | ||||||
|  |         Creates a savepoint inside the current transaction. Returns an | ||||||
|  |         identifier for the savepoint that will be used for the subsequent | ||||||
|  |         rollback or commit. Does nothing if savepoints are not supported. | ||||||
|  |         """ | ||||||
|  |         if not self._savepoint_allowed(): | ||||||
|  |             return | ||||||
|  |  | ||||||
|  |         thread_ident = thread.get_ident() | ||||||
|  |         tid = str(thread_ident).replace('-', '') | ||||||
|  |  | ||||||
|  |         self.savepoint_state += 1 | ||||||
|  |         sid = "s%s_x%d" % (tid, self.savepoint_state) | ||||||
|  |  | ||||||
|  |         self.validate_thread_sharing() | ||||||
|  |         self._savepoint(sid) | ||||||
|  |  | ||||||
|  |         return sid | ||||||
|  |  | ||||||
|  |     def savepoint_rollback(self, sid): | ||||||
|  |         """ | ||||||
|  |         Rolls back to a savepoint. Does nothing if savepoints are not supported. | ||||||
|  |         """ | ||||||
|  |         if not self._savepoint_allowed(): | ||||||
|  |             return | ||||||
|  |  | ||||||
|  |         self.validate_thread_sharing() | ||||||
|  |         self._savepoint_rollback(sid) | ||||||
|  |  | ||||||
|  |     def savepoint_commit(self, sid): | ||||||
|  |         """ | ||||||
|  |         Releases a savepoint. Does nothing if savepoints are not supported. | ||||||
|  |         """ | ||||||
|  |         if not self._savepoint_allowed(): | ||||||
|  |             return | ||||||
|  |  | ||||||
|  |         self.validate_thread_sharing() | ||||||
|  |         self._savepoint_commit(sid) | ||||||
|  |  | ||||||
|  |     def clean_savepoints(self): | ||||||
|  |         """ | ||||||
|  |         Resets the counter used to generate unique savepoint ids in this thread. | ||||||
|  |         """ | ||||||
|  |         self.savepoint_state = 0 | ||||||
|  |  | ||||||
|  |     ##### Backend-specific transaction management methods ##### | ||||||
|  |  | ||||||
|  |     def _set_autocommit(self, autocommit): | ||||||
|  |         """ | ||||||
|  |         Backend-specific implementation to enable or disable autocommit. | ||||||
|  |         """ | ||||||
|  |         raise NotImplementedError('subclasses of BaseDatabaseWrapper may require a _set_autocommit() method') | ||||||
|  |  | ||||||
|  |     ##### Generic transaction management methods ##### | ||||||
|  |  | ||||||
|  |     def get_autocommit(self): | ||||||
|  |         """ | ||||||
|  |         Check the autocommit state. | ||||||
|  |         """ | ||||||
|  |         self.ensure_connection() | ||||||
|  |         return self.autocommit | ||||||
|  |  | ||||||
|  |     def set_autocommit(self, autocommit): | ||||||
|  |         """ | ||||||
|  |         Enable or disable autocommit. | ||||||
|  |         """ | ||||||
|  |         self.validate_no_atomic_block() | ||||||
|  |         self.ensure_connection() | ||||||
|  |         self._set_autocommit(autocommit) | ||||||
|  |         self.autocommit = autocommit | ||||||
|  |  | ||||||
|  |     def get_rollback(self): | ||||||
|  |         """ | ||||||
|  |         Get the "needs rollback" flag -- for *advanced use* only. | ||||||
|  |         """ | ||||||
|  |         if not self.in_atomic_block: | ||||||
|  |             raise TransactionManagementError( | ||||||
|  |                 "The rollback flag doesn't work outside of an 'atomic' block.") | ||||||
|  |         return self.needs_rollback | ||||||
|  |  | ||||||
|  |     def set_rollback(self, rollback): | ||||||
|  |         """ | ||||||
|  |         Set or unset the "needs rollback" flag -- for *advanced use* only. | ||||||
|  |         """ | ||||||
|  |         if not self.in_atomic_block: | ||||||
|  |             raise TransactionManagementError( | ||||||
|  |                 "The rollback flag doesn't work outside of an 'atomic' block.") | ||||||
|  |         self.needs_rollback = rollback | ||||||
|  |  | ||||||
|  |     def validate_no_atomic_block(self): | ||||||
|  |         """ | ||||||
|  |         Raise an error if an atomic block is active. | ||||||
|  |         """ | ||||||
|  |         if self.in_atomic_block: | ||||||
|  |             raise TransactionManagementError( | ||||||
|  |                 "This is forbidden when an 'atomic' block is active.") | ||||||
|  |  | ||||||
|  |     def validate_no_broken_transaction(self): | ||||||
|  |         if self.needs_rollback: | ||||||
|  |             raise TransactionManagementError( | ||||||
|  |                 "An error occurred in the current transaction. You can't " | ||||||
|  |                 "execute queries until the end of the 'atomic' block.") | ||||||
|  |  | ||||||
|  |     ##### Foreign key constraints checks handling ##### | ||||||
|  |  | ||||||
|  |     @contextmanager | ||||||
|  |     def constraint_checks_disabled(self): | ||||||
|  |         """ | ||||||
|  |         Context manager that disables foreign key constraint checking. | ||||||
|  |         """ | ||||||
|  |         disabled = self.disable_constraint_checking() | ||||||
|  |         try: | ||||||
|  |             yield | ||||||
|  |         finally: | ||||||
|  |             if disabled: | ||||||
|  |                 self.enable_constraint_checking() | ||||||
|  |  | ||||||
|  |     def disable_constraint_checking(self): | ||||||
|  |         """ | ||||||
|  |         Backends can implement as needed to temporarily disable foreign key | ||||||
|  |         constraint checking. Should return True if the constraints were | ||||||
|  |         disabled and will need to be reenabled. | ||||||
|  |         """ | ||||||
|  |         return False | ||||||
|  |  | ||||||
|  |     def enable_constraint_checking(self): | ||||||
|  |         """ | ||||||
|  |         Backends can implement as needed to re-enable foreign key constraint | ||||||
|  |         checking. | ||||||
|  |         """ | ||||||
|  |         pass | ||||||
|  |  | ||||||
|  |     def check_constraints(self, table_names=None): | ||||||
|  |         """ | ||||||
|  |         Backends can override this method if they can apply constraint | ||||||
|  |         checking (e.g. via "SET CONSTRAINTS ALL IMMEDIATE"). Should raise an | ||||||
|  |         IntegrityError if any invalid foreign key references are encountered. | ||||||
|  |         """ | ||||||
|  |         pass | ||||||
|  |  | ||||||
|  |     ##### Connection termination handling ##### | ||||||
|  |  | ||||||
|  |     def is_usable(self): | ||||||
|  |         """ | ||||||
|  |         Tests if the database connection is usable. | ||||||
|  |  | ||||||
|  |         This function may assume that self.connection is not None. | ||||||
|  |  | ||||||
|  |         Actual implementations should take care not to raise exceptions | ||||||
|  |         as that may prevent Django from recycling unusable connections. | ||||||
|  |         """ | ||||||
|  |         raise NotImplementedError( | ||||||
|  |             "subclasses of BaseDatabaseWrapper may require an is_usable() method") | ||||||
|  |  | ||||||
|  |     def close_if_unusable_or_obsolete(self): | ||||||
|  |         """ | ||||||
|  |         Closes the current connection if unrecoverable errors have occurred, | ||||||
|  |         or if it outlived its maximum age. | ||||||
|  |         """ | ||||||
|  |         if self.connection is not None: | ||||||
|  |             # If the application didn't restore the original autocommit setting, | ||||||
|  |             # don't take chances, drop the connection. | ||||||
|  |             if self.get_autocommit() != self.settings_dict['AUTOCOMMIT']: | ||||||
|  |                 self.close() | ||||||
|  |                 return | ||||||
|  |  | ||||||
|  |             # If an exception other than DataError or IntegrityError occurred | ||||||
|  |             # since the last commit / rollback, check if the connection works. | ||||||
|  |             if self.errors_occurred: | ||||||
|  |                 if self.is_usable(): | ||||||
|  |                     self.errors_occurred = False | ||||||
|  |                 else: | ||||||
|  |                     self.close() | ||||||
|  |                     return | ||||||
|  |  | ||||||
|  |             if self.close_at is not None and time.time() >= self.close_at: | ||||||
|  |                 self.close() | ||||||
|  |                 return | ||||||
|  |  | ||||||
|  |     ##### Thread safety handling ##### | ||||||
|  |  | ||||||
|  |     def validate_thread_sharing(self): | ||||||
|  |         """ | ||||||
|  |         Validates that the connection isn't accessed by another thread than the | ||||||
|  |         one which originally created it, unless the connection was explicitly | ||||||
|  |         authorized to be shared between threads (via the `allow_thread_sharing` | ||||||
|  |         property). Raises an exception if the validation fails. | ||||||
|  |         """ | ||||||
|  |         if not (self.allow_thread_sharing | ||||||
|  |                 or self._thread_ident == thread.get_ident()): | ||||||
|  |             raise DatabaseError("DatabaseWrapper objects created in a " | ||||||
|  |                 "thread can only be used in that same thread. The object " | ||||||
|  |                 "with alias '%s' was created in thread id %s and this is " | ||||||
|  |                 "thread id %s." | ||||||
|  |                 % (self.alias, self._thread_ident, thread.get_ident())) | ||||||
|  |  | ||||||
|  |     ##### Miscellaneous ##### | ||||||
|  |  | ||||||
|  |     def prepare_database(self): | ||||||
|  |         """ | ||||||
|  |         Hook to do any database check or preparation, generally called before | ||||||
|  |         migrating a project or an app. | ||||||
|  |         """ | ||||||
|  |         pass | ||||||
|  |  | ||||||
|  |     @cached_property | ||||||
|  |     def wrap_database_errors(self): | ||||||
|  |         """ | ||||||
|  |         Context manager and decorator that re-throws backend-specific database | ||||||
|  |         exceptions using Django's common wrappers. | ||||||
|  |         """ | ||||||
|  |         return DatabaseErrorWrapper(self) | ||||||
|  |  | ||||||
|  |     def make_debug_cursor(self, cursor): | ||||||
|  |         """ | ||||||
|  |         Creates a cursor that logs all queries in self.queries_log. | ||||||
|  |         """ | ||||||
|  |         return utils.CursorDebugWrapper(cursor, self) | ||||||
|  |  | ||||||
|  |     def make_cursor(self, cursor): | ||||||
|  |         """ | ||||||
|  |         Creates a cursor without debug logging. | ||||||
|  |         """ | ||||||
|  |         return utils.CursorWrapper(cursor, self) | ||||||
|  |  | ||||||
|  |     @contextmanager | ||||||
|  |     def temporary_connection(self): | ||||||
|  |         """ | ||||||
|  |         Context manager that ensures that a connection is established, and | ||||||
|  |         if it opened one, closes it to avoid leaving a dangling connection. | ||||||
|  |         This is useful for operations outside of the request-response cycle. | ||||||
|  |  | ||||||
|  |         Provides a cursor: with self.temporary_connection() as cursor: ... | ||||||
|  |         """ | ||||||
|  |         must_close = self.connection is None | ||||||
|  |         cursor = self.cursor() | ||||||
|  |         try: | ||||||
|  |             yield cursor | ||||||
|  |         finally: | ||||||
|  |             cursor.close() | ||||||
|  |             if must_close: | ||||||
|  |                 self.close() | ||||||
|  |  | ||||||
|  |     @cached_property | ||||||
|  |     def _nodb_connection(self): | ||||||
|  |         """ | ||||||
|  |         Alternative connection to be used when there is no need to access | ||||||
|  |         the main database, specifically for test db creation/deletion. | ||||||
|  |         This also prevents the production database from being exposed to | ||||||
|  |         potential child threads while (or after) the test database is destroyed. | ||||||
|  |         Refs #10868, #17786, #16969. | ||||||
|  |         """ | ||||||
|  |         settings_dict = self.settings_dict.copy() | ||||||
|  |         settings_dict['NAME'] = None | ||||||
|  |         nodb_connection = self.__class__( | ||||||
|  |             settings_dict, | ||||||
|  |             alias=NO_DB_ALIAS, | ||||||
|  |             allow_thread_sharing=False) | ||||||
|  |         return nodb_connection | ||||||
|  |  | ||||||
|  |     def _start_transaction_under_autocommit(self): | ||||||
|  |         """ | ||||||
|  |         Only required when autocommits_when_autocommit_is_off = True. | ||||||
|  |         """ | ||||||
|  |         raise NotImplementedError( | ||||||
|  |             'subclasses of BaseDatabaseWrapper may require a ' | ||||||
|  |             '_start_transaction_under_autocommit() method' | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     def schema_editor(self, *args, **kwargs): | ||||||
|  |         """ | ||||||
|  |         Returns a new instance of this backend's SchemaEditor. | ||||||
|  |         """ | ||||||
|  |         if self.SchemaEditorClass is None: | ||||||
|  |             raise NotImplementedError( | ||||||
|  |                 'The SchemaEditorClass attribute of this database wrapper is still None') | ||||||
|  |         return self.SchemaEditorClass(self, *args, **kwargs) | ||||||
							
								
								
									
										15
									
								
								django/db/backends/base/client.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								django/db/backends/base/client.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | |||||||
|  | class BaseDatabaseClient(object): | ||||||
|  |     """ | ||||||
|  |     This class encapsulates all backend-specific methods for opening a | ||||||
|  |     client shell. | ||||||
|  |     """ | ||||||
|  |     # This should be a string representing the name of the executable | ||||||
|  |     # (e.g., "psql"). Subclasses must override this. | ||||||
|  |     executable_name = None | ||||||
|  |  | ||||||
|  |     def __init__(self, connection): | ||||||
|  |         # connection is an instance of BaseDatabaseWrapper. | ||||||
|  |         self.connection = connection | ||||||
|  |  | ||||||
|  |     def runshell(self): | ||||||
|  |         raise NotImplementedError('subclasses of BaseDatabaseClient must provide a runshell() method') | ||||||
| @@ -3,16 +3,16 @@ import sys | |||||||
| import time | import time | ||||||
| import warnings | import warnings | ||||||
| 
 | 
 | ||||||
|  | from django.apps import apps | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
|  | from django.core import serializers | ||||||
|  | from django.db import router | ||||||
|  | from django.db.backends.utils import truncate_name | ||||||
| from django.utils.deprecation import RemovedInDjango20Warning | from django.utils.deprecation import RemovedInDjango20Warning | ||||||
| from django.utils.encoding import force_bytes | from django.utils.encoding import force_bytes | ||||||
| from django.utils.six.moves import input |  | ||||||
| from django.utils.six import StringIO | from django.utils.six import StringIO | ||||||
| from django.db import router | from django.utils.six.moves import input | ||||||
| from django.apps import apps |  | ||||||
| from django.core import serializers |  | ||||||
| 
 | 
 | ||||||
| from .utils import truncate_name |  | ||||||
| 
 | 
 | ||||||
| # The prefix to put on the default database name when creating | # The prefix to put on the default database name when creating | ||||||
| # the test database. | # the test database. | ||||||
							
								
								
									
										250
									
								
								django/db/backends/base/features.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										250
									
								
								django/db/backends/base/features.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,250 @@ | |||||||
|  | from django.db.utils import ProgrammingError | ||||||
|  | from django.utils.functional import cached_property | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class BaseDatabaseFeatures(object): | ||||||
|  |     gis_enabled = False | ||||||
|  |     allows_group_by_pk = False | ||||||
|  |     # True if django.db.backends.utils.typecast_timestamp is used on values | ||||||
|  |     # returned from dates() calls. | ||||||
|  |     needs_datetime_string_cast = True | ||||||
|  |     empty_fetchmany_value = [] | ||||||
|  |     update_can_self_select = True | ||||||
|  |  | ||||||
|  |     # Does the backend distinguish between '' and None? | ||||||
|  |     interprets_empty_strings_as_nulls = False | ||||||
|  |  | ||||||
|  |     # Does the backend allow inserting duplicate NULL rows in a nullable | ||||||
|  |     # unique field? All core backends implement this correctly, but other | ||||||
|  |     # databases such as SQL Server do not. | ||||||
|  |     supports_nullable_unique_constraints = True | ||||||
|  |  | ||||||
|  |     # Does the backend allow inserting duplicate rows when a unique_together | ||||||
|  |     # constraint exists and some fields are nullable but not all of them? | ||||||
|  |     supports_partially_nullable_unique_constraints = True | ||||||
|  |  | ||||||
|  |     can_use_chunked_reads = True | ||||||
|  |     can_return_id_from_insert = False | ||||||
|  |     has_bulk_insert = False | ||||||
|  |     uses_savepoints = False | ||||||
|  |     can_release_savepoints = False | ||||||
|  |     can_combine_inserts_with_and_without_auto_increment_pk = False | ||||||
|  |  | ||||||
|  |     # If True, don't use integer foreign keys referring to, e.g., positive | ||||||
|  |     # integer primary keys. | ||||||
|  |     related_fields_match_type = False | ||||||
|  |     allow_sliced_subqueries = True | ||||||
|  |     has_select_for_update = False | ||||||
|  |     has_select_for_update_nowait = False | ||||||
|  |  | ||||||
|  |     supports_select_related = True | ||||||
|  |  | ||||||
|  |     # Does the default test database allow multiple connections? | ||||||
|  |     # Usually an indication that the test database is in-memory | ||||||
|  |     test_db_allows_multiple_connections = True | ||||||
|  |  | ||||||
|  |     # Can an object be saved without an explicit primary key? | ||||||
|  |     supports_unspecified_pk = False | ||||||
|  |  | ||||||
|  |     # Can a fixture contain forward references? i.e., are | ||||||
|  |     # FK constraints checked at the end of transaction, or | ||||||
|  |     # at the end of each save operation? | ||||||
|  |     supports_forward_references = True | ||||||
|  |  | ||||||
|  |     # Does the backend truncate names properly when they are too long? | ||||||
|  |     truncates_names = False | ||||||
|  |  | ||||||
|  |     # Is there a REAL datatype in addition to floats/doubles? | ||||||
|  |     has_real_datatype = False | ||||||
|  |     supports_subqueries_in_group_by = True | ||||||
|  |     supports_bitwise_or = True | ||||||
|  |  | ||||||
|  |     # Is there a true datatype for timedeltas? | ||||||
|  |     has_native_duration_field = False | ||||||
|  |  | ||||||
|  |     # Does the database driver support timedeltas as arguments? | ||||||
|  |     # This is only relevant when there is a native duration field. | ||||||
|  |     # Specifically, there is a bug with cx_Oracle: | ||||||
|  |     # https://bitbucket.org/anthony_tuininga/cx_oracle/issue/7/ | ||||||
|  |     driver_supports_timedelta_args = False | ||||||
|  |  | ||||||
|  |     # Do time/datetime fields have microsecond precision? | ||||||
|  |     supports_microsecond_precision = True | ||||||
|  |  | ||||||
|  |     # Does the __regex lookup support backreferencing and grouping? | ||||||
|  |     supports_regex_backreferencing = True | ||||||
|  |  | ||||||
|  |     # Can date/datetime lookups be performed using a string? | ||||||
|  |     supports_date_lookup_using_string = True | ||||||
|  |  | ||||||
|  |     # Can datetimes with timezones be used? | ||||||
|  |     supports_timezones = True | ||||||
|  |  | ||||||
|  |     # Does the database have a copy of the zoneinfo database? | ||||||
|  |     has_zoneinfo_database = True | ||||||
|  |  | ||||||
|  |     # When performing a GROUP BY, is an ORDER BY NULL required | ||||||
|  |     # to remove any ordering? | ||||||
|  |     requires_explicit_null_ordering_when_grouping = False | ||||||
|  |  | ||||||
|  |     # Does the backend order NULL values as largest or smallest? | ||||||
|  |     nulls_order_largest = False | ||||||
|  |  | ||||||
|  |     # Is there a 1000 item limit on query parameters? | ||||||
|  |     supports_1000_query_parameters = True | ||||||
|  |  | ||||||
|  |     # Can an object have an autoincrement primary key of 0? MySQL says No. | ||||||
|  |     allows_auto_pk_0 = True | ||||||
|  |  | ||||||
|  |     # Do we need to NULL a ForeignKey out, or can the constraint check be | ||||||
|  |     # deferred | ||||||
|  |     can_defer_constraint_checks = False | ||||||
|  |  | ||||||
|  |     # date_interval_sql can properly handle mixed Date/DateTime fields and timedeltas | ||||||
|  |     supports_mixed_date_datetime_comparisons = True | ||||||
|  |  | ||||||
|  |     # Does the backend support tablespaces? Default to False because it isn't | ||||||
|  |     # in the SQL standard. | ||||||
|  |     supports_tablespaces = False | ||||||
|  |  | ||||||
|  |     # Does the backend reset sequences between tests? | ||||||
|  |     supports_sequence_reset = True | ||||||
|  |  | ||||||
|  |     # Can the backend determine reliably the length of a CharField? | ||||||
|  |     can_introspect_max_length = True | ||||||
|  |  | ||||||
|  |     # Can the backend determine reliably if a field is nullable? | ||||||
|  |     # Note that this is separate from interprets_empty_strings_as_nulls, | ||||||
|  |     # although the latter feature, when true, interferes with correct | ||||||
|  |     # setting (and introspection) of CharFields' nullability. | ||||||
|  |     # This is True for all core backends. | ||||||
|  |     can_introspect_null = True | ||||||
|  |  | ||||||
|  |     # Confirm support for introspected foreign keys | ||||||
|  |     # Every database can do this reliably, except MySQL, | ||||||
|  |     # which can't do it for MyISAM tables | ||||||
|  |     can_introspect_foreign_keys = True | ||||||
|  |  | ||||||
|  |     # Can the backend introspect an AutoField, instead of an IntegerField? | ||||||
|  |     can_introspect_autofield = False | ||||||
|  |  | ||||||
|  |     # Can the backend introspect a BigIntegerField, instead of an IntegerField? | ||||||
|  |     can_introspect_big_integer_field = True | ||||||
|  |  | ||||||
|  |     # Can the backend introspect an BinaryField, instead of an TextField? | ||||||
|  |     can_introspect_binary_field = True | ||||||
|  |  | ||||||
|  |     # Can the backend introspect an DecimalField, instead of an FloatField? | ||||||
|  |     can_introspect_decimal_field = True | ||||||
|  |  | ||||||
|  |     # Can the backend introspect an IPAddressField, instead of an CharField? | ||||||
|  |     can_introspect_ip_address_field = False | ||||||
|  |  | ||||||
|  |     # Can the backend introspect a PositiveIntegerField, instead of an IntegerField? | ||||||
|  |     can_introspect_positive_integer_field = False | ||||||
|  |  | ||||||
|  |     # Can the backend introspect a SmallIntegerField, instead of an IntegerField? | ||||||
|  |     can_introspect_small_integer_field = False | ||||||
|  |  | ||||||
|  |     # Can the backend introspect a TimeField, instead of a DateTimeField? | ||||||
|  |     can_introspect_time_field = True | ||||||
|  |  | ||||||
|  |     # Support for the DISTINCT ON clause | ||||||
|  |     can_distinct_on_fields = False | ||||||
|  |  | ||||||
|  |     # Does the backend decide to commit before SAVEPOINT statements | ||||||
|  |     # when autocommit is disabled? http://bugs.python.org/issue8145#msg109965 | ||||||
|  |     autocommits_when_autocommit_is_off = False | ||||||
|  |  | ||||||
|  |     # Does the backend prevent running SQL queries in broken transactions? | ||||||
|  |     atomic_transactions = True | ||||||
|  |  | ||||||
|  |     # Can we roll back DDL in a transaction? | ||||||
|  |     can_rollback_ddl = False | ||||||
|  |  | ||||||
|  |     # Can we issue more than one ALTER COLUMN clause in an ALTER TABLE? | ||||||
|  |     supports_combined_alters = False | ||||||
|  |  | ||||||
|  |     # Does it support foreign keys? | ||||||
|  |     supports_foreign_keys = True | ||||||
|  |  | ||||||
|  |     # Does it support CHECK constraints? | ||||||
|  |     supports_column_check_constraints = True | ||||||
|  |  | ||||||
|  |     # Does the backend support 'pyformat' style ("... %(name)s ...", {'name': value}) | ||||||
|  |     # parameter passing? Note this can be provided by the backend even if not | ||||||
|  |     # supported by the Python driver | ||||||
|  |     supports_paramstyle_pyformat = True | ||||||
|  |  | ||||||
|  |     # Does the backend require literal defaults, rather than parameterized ones? | ||||||
|  |     requires_literal_defaults = False | ||||||
|  |  | ||||||
|  |     # Does the backend require a connection reset after each material schema change? | ||||||
|  |     connection_persists_old_columns = False | ||||||
|  |  | ||||||
|  |     # What kind of error does the backend throw when accessing closed cursor? | ||||||
|  |     closed_cursor_error_class = ProgrammingError | ||||||
|  |  | ||||||
|  |     # Does 'a' LIKE 'A' match? | ||||||
|  |     has_case_insensitive_like = True | ||||||
|  |  | ||||||
|  |     # Does the backend require the sqlparse library for splitting multi-line | ||||||
|  |     # statements before executing them? | ||||||
|  |     requires_sqlparse_for_splitting = True | ||||||
|  |  | ||||||
|  |     # Suffix for backends that don't support "SELECT xxx;" queries. | ||||||
|  |     bare_select_suffix = '' | ||||||
|  |  | ||||||
|  |     # If NULL is implied on columns without needing to be explicitly specified | ||||||
|  |     implied_column_null = False | ||||||
|  |  | ||||||
|  |     uppercases_column_names = False | ||||||
|  |  | ||||||
|  |     # Does the backend support "select for update" queries with limit (and offset)? | ||||||
|  |     supports_select_for_update_with_limit = True | ||||||
|  |  | ||||||
|  |     def __init__(self, connection): | ||||||
|  |         self.connection = connection | ||||||
|  |  | ||||||
|  |     @cached_property | ||||||
|  |     def supports_transactions(self): | ||||||
|  |         """Confirm support for transactions.""" | ||||||
|  |         with self.connection.cursor() as cursor: | ||||||
|  |             cursor.execute('CREATE TABLE ROLLBACK_TEST (X INT)') | ||||||
|  |             self.connection.set_autocommit(False) | ||||||
|  |             cursor.execute('INSERT INTO ROLLBACK_TEST (X) VALUES (8)') | ||||||
|  |             self.connection.rollback() | ||||||
|  |             self.connection.set_autocommit(True) | ||||||
|  |             cursor.execute('SELECT COUNT(X) FROM ROLLBACK_TEST') | ||||||
|  |             count, = cursor.fetchone() | ||||||
|  |             cursor.execute('DROP TABLE ROLLBACK_TEST') | ||||||
|  |         return count == 0 | ||||||
|  |  | ||||||
|  |     @cached_property | ||||||
|  |     def supports_stddev(self): | ||||||
|  |         """Confirm support for STDDEV and related stats functions.""" | ||||||
|  |         class StdDevPop(object): | ||||||
|  |             sql_function = 'STDDEV_POP' | ||||||
|  |  | ||||||
|  |         try: | ||||||
|  |             self.connection.ops.check_aggregate_support(StdDevPop()) | ||||||
|  |             return True | ||||||
|  |         except NotImplementedError: | ||||||
|  |             return False | ||||||
|  |  | ||||||
|  |     def introspected_boolean_field_type(self, field=None, created_separately=False): | ||||||
|  |         """ | ||||||
|  |         What is the type returned when the backend introspects a BooleanField? | ||||||
|  |         The optional arguments may be used to give further details of the field to be | ||||||
|  |         introspected; in particular, they are provided by Django's test suite: | ||||||
|  |         field -- the field definition | ||||||
|  |         created_separately -- True if the field was added via a SchemaEditor's AddField, | ||||||
|  |                               False if the field was created with the model | ||||||
|  |  | ||||||
|  |         Note that return value from this function is compared by tests against actual | ||||||
|  |         introspection results; it should provide expectations, not run an introspection | ||||||
|  |         itself. | ||||||
|  |         """ | ||||||
|  |         if self.can_introspect_null and field and field.null: | ||||||
|  |             return 'NullBooleanField' | ||||||
|  |         return 'BooleanField' | ||||||
							
								
								
									
										178
									
								
								django/db/backends/base/introspection.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										178
									
								
								django/db/backends/base/introspection.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,178 @@ | |||||||
|  | from collections import namedtuple | ||||||
|  |  | ||||||
|  | from django.utils import six | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # Structure returned by DatabaseIntrospection.get_table_list() | ||||||
|  | TableInfo = namedtuple('TableInfo', ['name', 'type']) | ||||||
|  |  | ||||||
|  | # Structure returned by the DB-API cursor.description interface (PEP 249) | ||||||
|  | FieldInfo = namedtuple('FieldInfo', | ||||||
|  |     'name type_code display_size internal_size precision scale null_ok') | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class BaseDatabaseIntrospection(object): | ||||||
|  |     """ | ||||||
|  |     This class encapsulates all backend-specific introspection utilities | ||||||
|  |     """ | ||||||
|  |     data_types_reverse = {} | ||||||
|  |  | ||||||
|  |     def __init__(self, connection): | ||||||
|  |         self.connection = connection | ||||||
|  |  | ||||||
|  |     def get_field_type(self, data_type, description): | ||||||
|  |         """Hook for a database backend to use the cursor description to | ||||||
|  |         match a Django field type to a database column. | ||||||
|  |  | ||||||
|  |         For Oracle, the column data_type on its own is insufficient to | ||||||
|  |         distinguish between a FloatField and IntegerField, for example.""" | ||||||
|  |         return self.data_types_reverse[data_type] | ||||||
|  |  | ||||||
|  |     def table_name_converter(self, name): | ||||||
|  |         """Apply a conversion to the name for the purposes of comparison. | ||||||
|  |  | ||||||
|  |         The default table name converter is for case sensitive comparison. | ||||||
|  |         """ | ||||||
|  |         return name | ||||||
|  |  | ||||||
|  |     def column_name_converter(self, name): | ||||||
|  |         """ | ||||||
|  |         Apply a conversion to the column name for the purposes of comparison. | ||||||
|  |  | ||||||
|  |         Uses table_name_converter() by default. | ||||||
|  |         """ | ||||||
|  |         return self.table_name_converter(name) | ||||||
|  |  | ||||||
|  |     def table_names(self, cursor=None, include_views=False): | ||||||
|  |         """ | ||||||
|  |         Returns a list of names of all tables that exist in the database. | ||||||
|  |         The returned table list is sorted by Python's default sorting. We | ||||||
|  |         do NOT use database's ORDER BY here to avoid subtle differences | ||||||
|  |         in sorting order between databases. | ||||||
|  |         """ | ||||||
|  |         def get_names(cursor): | ||||||
|  |             return sorted(ti.name for ti in self.get_table_list(cursor) | ||||||
|  |                           if include_views or ti.type == 't') | ||||||
|  |         if cursor is None: | ||||||
|  |             with self.connection.cursor() as cursor: | ||||||
|  |                 return get_names(cursor) | ||||||
|  |         return get_names(cursor) | ||||||
|  |  | ||||||
|  |     def get_table_list(self, cursor): | ||||||
|  |         """ | ||||||
|  |         Returns an unsorted list of TableInfo named tuples of all tables and | ||||||
|  |         views that exist in the database. | ||||||
|  |         """ | ||||||
|  |         raise NotImplementedError('subclasses of BaseDatabaseIntrospection may require a get_table_list() method') | ||||||
|  |  | ||||||
|  |     def django_table_names(self, only_existing=False, include_views=True): | ||||||
|  |         """ | ||||||
|  |         Returns a list of all table names that have associated Django models and | ||||||
|  |         are in INSTALLED_APPS. | ||||||
|  |  | ||||||
|  |         If only_existing is True, the resulting list will only include the tables | ||||||
|  |         that actually exist in the database. | ||||||
|  |         """ | ||||||
|  |         from django.apps import apps | ||||||
|  |         from django.db import router | ||||||
|  |         tables = set() | ||||||
|  |         for app_config in apps.get_app_configs(): | ||||||
|  |             for model in router.get_migratable_models(app_config, self.connection.alias): | ||||||
|  |                 if not model._meta.managed: | ||||||
|  |                     continue | ||||||
|  |                 tables.add(model._meta.db_table) | ||||||
|  |                 tables.update(f.m2m_db_table() for f in model._meta.local_many_to_many) | ||||||
|  |         tables = list(tables) | ||||||
|  |         if only_existing: | ||||||
|  |             existing_tables = self.table_names(include_views=include_views) | ||||||
|  |             tables = [ | ||||||
|  |                 t | ||||||
|  |                 for t in tables | ||||||
|  |                 if self.table_name_converter(t) in existing_tables | ||||||
|  |             ] | ||||||
|  |         return tables | ||||||
|  |  | ||||||
|  |     def installed_models(self, tables): | ||||||
|  |         "Returns a set of all models represented by the provided list of table names." | ||||||
|  |         from django.apps import apps | ||||||
|  |         from django.db import router | ||||||
|  |         all_models = [] | ||||||
|  |         for app_config in apps.get_app_configs(): | ||||||
|  |             all_models.extend(router.get_migratable_models(app_config, self.connection.alias)) | ||||||
|  |         tables = list(map(self.table_name_converter, tables)) | ||||||
|  |         return { | ||||||
|  |             m for m in all_models | ||||||
|  |             if self.table_name_converter(m._meta.db_table) in tables | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |     def sequence_list(self): | ||||||
|  |         "Returns a list of information about all DB sequences for all models in all apps." | ||||||
|  |         from django.apps import apps | ||||||
|  |         from django.db import models, router | ||||||
|  |  | ||||||
|  |         sequence_list = [] | ||||||
|  |  | ||||||
|  |         for app_config in apps.get_app_configs(): | ||||||
|  |             for model in router.get_migratable_models(app_config, self.connection.alias): | ||||||
|  |                 if not model._meta.managed: | ||||||
|  |                     continue | ||||||
|  |                 if model._meta.swapped: | ||||||
|  |                     continue | ||||||
|  |                 for f in model._meta.local_fields: | ||||||
|  |                     if isinstance(f, models.AutoField): | ||||||
|  |                         sequence_list.append({'table': model._meta.db_table, 'column': f.column}) | ||||||
|  |                         break  # Only one AutoField is allowed per model, so don't bother continuing. | ||||||
|  |  | ||||||
|  |                 for f in model._meta.local_many_to_many: | ||||||
|  |                     # If this is an m2m using an intermediate table, | ||||||
|  |                     # we don't need to reset the sequence. | ||||||
|  |                     if f.rel.through is None: | ||||||
|  |                         sequence_list.append({'table': f.m2m_db_table(), 'column': None}) | ||||||
|  |  | ||||||
|  |         return sequence_list | ||||||
|  |  | ||||||
|  |     def get_key_columns(self, cursor, table_name): | ||||||
|  |         """ | ||||||
|  |         Backends can override this to return a list of (column_name, referenced_table_name, | ||||||
|  |         referenced_column_name) for all key columns in given table. | ||||||
|  |         """ | ||||||
|  |         raise NotImplementedError('subclasses of BaseDatabaseIntrospection may require a get_key_columns() method') | ||||||
|  |  | ||||||
|  |     def get_primary_key_column(self, cursor, table_name): | ||||||
|  |         """ | ||||||
|  |         Returns the name of the primary key column for the given table. | ||||||
|  |         """ | ||||||
|  |         for column in six.iteritems(self.get_indexes(cursor, table_name)): | ||||||
|  |             if column[1]['primary_key']: | ||||||
|  |                 return column[0] | ||||||
|  |         return None | ||||||
|  |  | ||||||
|  |     def get_indexes(self, cursor, table_name): | ||||||
|  |         """ | ||||||
|  |         Returns a dictionary of indexed fieldname -> infodict for the given | ||||||
|  |         table, where each infodict is in the format: | ||||||
|  |             {'primary_key': boolean representing whether it's the primary key, | ||||||
|  |              'unique': boolean representing whether it's a unique index} | ||||||
|  |  | ||||||
|  |         Only single-column indexes are introspected. | ||||||
|  |         """ | ||||||
|  |         raise NotImplementedError('subclasses of BaseDatabaseIntrospection may require a get_indexes() method') | ||||||
|  |  | ||||||
|  |     def get_constraints(self, cursor, table_name): | ||||||
|  |         """ | ||||||
|  |         Retrieves any constraints or keys (unique, pk, fk, check, index) | ||||||
|  |         across one or more columns. | ||||||
|  |  | ||||||
|  |         Returns a dict mapping constraint names to their attributes, | ||||||
|  |         where attributes is a dict with keys: | ||||||
|  |          * columns: List of columns this covers | ||||||
|  |          * primary_key: True if primary key, False otherwise | ||||||
|  |          * unique: True if this is a unique constraint, False otherwise | ||||||
|  |          * foreign_key: (table, column) of target, or None | ||||||
|  |          * check: True if check constraint, False otherwise | ||||||
|  |          * index: True if index, False otherwise. | ||||||
|  |  | ||||||
|  |         Some backends may return special constraint names that don't exist | ||||||
|  |         if they don't name constraints of a certain type (e.g. SQLite) | ||||||
|  |         """ | ||||||
|  |         raise NotImplementedError('subclasses of BaseDatabaseIntrospection may require a get_constraints() method') | ||||||
							
								
								
									
										555
									
								
								django/db/backends/base/operations.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										555
									
								
								django/db/backends/base/operations.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,555 @@ | |||||||
|  | import datetime | ||||||
|  | import decimal | ||||||
|  | from importlib import import_module | ||||||
|  | import warnings | ||||||
|  |  | ||||||
|  | from django.conf import settings | ||||||
|  | from django.db.backends import utils | ||||||
|  | from django.utils import six, timezone | ||||||
|  | from django.utils.dateparse import parse_duration | ||||||
|  | from django.utils.deprecation import RemovedInDjango19Warning | ||||||
|  | from django.utils.encoding import force_text | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class BaseDatabaseOperations(object): | ||||||
|  |     """ | ||||||
|  |     This class encapsulates all backend-specific differences, such as the way | ||||||
|  |     a backend performs ordering or calculates the ID of a recently-inserted | ||||||
|  |     row. | ||||||
|  |     """ | ||||||
|  |     compiler_module = "django.db.models.sql.compiler" | ||||||
|  |  | ||||||
|  |     # Integer field safe ranges by `internal_type` as documented | ||||||
|  |     # in docs/ref/models/fields.txt. | ||||||
|  |     integer_field_ranges = { | ||||||
|  |         'SmallIntegerField': (-32768, 32767), | ||||||
|  |         'IntegerField': (-2147483648, 2147483647), | ||||||
|  |         'BigIntegerField': (-9223372036854775808, 9223372036854775807), | ||||||
|  |         'PositiveSmallIntegerField': (0, 32767), | ||||||
|  |         'PositiveIntegerField': (0, 2147483647), | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     def __init__(self, connection): | ||||||
|  |         self.connection = connection | ||||||
|  |         self._cache = None | ||||||
|  |  | ||||||
|  |     def autoinc_sql(self, table, column): | ||||||
|  |         """ | ||||||
|  |         Returns any SQL needed to support auto-incrementing primary keys, or | ||||||
|  |         None if no SQL is necessary. | ||||||
|  |  | ||||||
|  |         This SQL is executed when a table is created. | ||||||
|  |         """ | ||||||
|  |         return None | ||||||
|  |  | ||||||
|  |     def bulk_batch_size(self, fields, objs): | ||||||
|  |         """ | ||||||
|  |         Returns the maximum allowed batch size for the backend. The fields | ||||||
|  |         are the fields going to be inserted in the batch, the objs contains | ||||||
|  |         all the objects to be inserted. | ||||||
|  |         """ | ||||||
|  |         return len(objs) | ||||||
|  |  | ||||||
|  |     def cache_key_culling_sql(self): | ||||||
|  |         """ | ||||||
|  |         Returns an SQL query that retrieves the first cache key greater than the | ||||||
|  |         n smallest. | ||||||
|  |  | ||||||
|  |         This is used by the 'db' cache backend to determine where to start | ||||||
|  |         culling. | ||||||
|  |         """ | ||||||
|  |         return "SELECT cache_key FROM %s ORDER BY cache_key LIMIT 1 OFFSET %%s" | ||||||
|  |  | ||||||
|  |     def unification_cast_sql(self, output_field): | ||||||
|  |         """ | ||||||
|  |         Given a field instance, returns the SQL necessary to cast the result of | ||||||
|  |         a union to that type. Note that the resulting string should contain a | ||||||
|  |         '%s' placeholder for the expression being cast. | ||||||
|  |         """ | ||||||
|  |         return '%s' | ||||||
|  |  | ||||||
|  |     def date_extract_sql(self, lookup_type, field_name): | ||||||
|  |         """ | ||||||
|  |         Given a lookup_type of 'year', 'month' or 'day', returns the SQL that | ||||||
|  |         extracts a value from the given date field field_name. | ||||||
|  |         """ | ||||||
|  |         raise NotImplementedError('subclasses of BaseDatabaseOperations may require a date_extract_sql() method') | ||||||
|  |  | ||||||
|  |     def date_interval_sql(self, sql, connector, timedelta): | ||||||
|  |         """ | ||||||
|  |         Implements the date interval functionality for expressions | ||||||
|  |         """ | ||||||
|  |         raise NotImplementedError('subclasses of BaseDatabaseOperations may require a date_interval_sql() method') | ||||||
|  |  | ||||||
|  |     def date_trunc_sql(self, lookup_type, field_name): | ||||||
|  |         """ | ||||||
|  |         Given a lookup_type of 'year', 'month' or 'day', returns the SQL that | ||||||
|  |         truncates the given date field field_name to a date object with only | ||||||
|  |         the given specificity. | ||||||
|  |         """ | ||||||
|  |         raise NotImplementedError('subclasses of BaseDatabaseOperations may require a datetrunc_sql() method') | ||||||
|  |  | ||||||
|  |     def datetime_cast_sql(self): | ||||||
|  |         """ | ||||||
|  |         Returns the SQL necessary to cast a datetime value so that it will be | ||||||
|  |         retrieved as a Python datetime object instead of a string. | ||||||
|  |  | ||||||
|  |         This SQL should include a '%s' in place of the field's name. | ||||||
|  |         """ | ||||||
|  |         return "%s" | ||||||
|  |  | ||||||
|  |     def datetime_extract_sql(self, lookup_type, field_name, tzname): | ||||||
|  |         """ | ||||||
|  |         Given a lookup_type of 'year', 'month', 'day', 'hour', 'minute' or | ||||||
|  |         'second', returns the SQL that extracts a value from the given | ||||||
|  |         datetime field field_name, and a tuple of parameters. | ||||||
|  |         """ | ||||||
|  |         raise NotImplementedError('subclasses of BaseDatabaseOperations may require a datetime_extract_sql() method') | ||||||
|  |  | ||||||
|  |     def datetime_trunc_sql(self, lookup_type, field_name, tzname): | ||||||
|  |         """ | ||||||
|  |         Given a lookup_type of 'year', 'month', 'day', 'hour', 'minute' or | ||||||
|  |         'second', returns the SQL that truncates the given datetime field | ||||||
|  |         field_name to a datetime object with only the given specificity, and | ||||||
|  |         a tuple of parameters. | ||||||
|  |         """ | ||||||
|  |         raise NotImplementedError('subclasses of BaseDatabaseOperations may require a datetime_trunk_sql() method') | ||||||
|  |  | ||||||
|  |     def deferrable_sql(self): | ||||||
|  |         """ | ||||||
|  |         Returns the SQL necessary to make a constraint "initially deferred" | ||||||
|  |         during a CREATE TABLE statement. | ||||||
|  |         """ | ||||||
|  |         return '' | ||||||
|  |  | ||||||
|  |     def distinct_sql(self, fields): | ||||||
|  |         """ | ||||||
|  |         Returns an SQL DISTINCT clause which removes duplicate rows from the | ||||||
|  |         result set. If any fields are given, only the given fields are being | ||||||
|  |         checked for duplicates. | ||||||
|  |         """ | ||||||
|  |         if fields: | ||||||
|  |             raise NotImplementedError('DISTINCT ON fields is not supported by this database backend') | ||||||
|  |         else: | ||||||
|  |             return 'DISTINCT' | ||||||
|  |  | ||||||
|  |     def drop_foreignkey_sql(self): | ||||||
|  |         """ | ||||||
|  |         Returns the SQL command that drops a foreign key. | ||||||
|  |         """ | ||||||
|  |         return "DROP CONSTRAINT" | ||||||
|  |  | ||||||
|  |     def drop_sequence_sql(self, table): | ||||||
|  |         """ | ||||||
|  |         Returns any SQL necessary to drop the sequence for the given table. | ||||||
|  |         Returns None if no SQL is necessary. | ||||||
|  |         """ | ||||||
|  |         return None | ||||||
|  |  | ||||||
|  |     def fetch_returned_insert_id(self, cursor): | ||||||
|  |         """ | ||||||
|  |         Given a cursor object that has just performed an INSERT...RETURNING | ||||||
|  |         statement into a table that has an auto-incrementing ID, returns the | ||||||
|  |         newly created ID. | ||||||
|  |         """ | ||||||
|  |         return cursor.fetchone()[0] | ||||||
|  |  | ||||||
|  |     def field_cast_sql(self, db_type, internal_type): | ||||||
|  |         """ | ||||||
|  |         Given a column type (e.g. 'BLOB', 'VARCHAR'), and an internal type | ||||||
|  |         (e.g. 'GenericIPAddressField'), returns the SQL necessary to cast it | ||||||
|  |         before using it in a WHERE statement. Note that the resulting string | ||||||
|  |         should contain a '%s' placeholder for the column being searched against. | ||||||
|  |         """ | ||||||
|  |         return '%s' | ||||||
|  |  | ||||||
|  |     def force_no_ordering(self): | ||||||
|  |         """ | ||||||
|  |         Returns a list used in the "ORDER BY" clause to force no ordering at | ||||||
|  |         all. Returning an empty list means that nothing will be included in the | ||||||
|  |         ordering. | ||||||
|  |         """ | ||||||
|  |         return [] | ||||||
|  |  | ||||||
|  |     def for_update_sql(self, nowait=False): | ||||||
|  |         """ | ||||||
|  |         Returns the FOR UPDATE SQL clause to lock rows for an update operation. | ||||||
|  |         """ | ||||||
|  |         if nowait: | ||||||
|  |             return 'FOR UPDATE NOWAIT' | ||||||
|  |         else: | ||||||
|  |             return 'FOR UPDATE' | ||||||
|  |  | ||||||
|  |     def fulltext_search_sql(self, field_name): | ||||||
|  |         """ | ||||||
|  |         Returns the SQL WHERE clause to use in order to perform a full-text | ||||||
|  |         search of the given field_name. Note that the resulting string should | ||||||
|  |         contain a '%s' placeholder for the value being searched against. | ||||||
|  |         """ | ||||||
|  |         raise NotImplementedError('Full-text search is not implemented for this database backend') | ||||||
|  |  | ||||||
|  |     def last_executed_query(self, cursor, sql, params): | ||||||
|  |         """ | ||||||
|  |         Returns a string of the query last executed by the given cursor, with | ||||||
|  |         placeholders replaced with actual values. | ||||||
|  |  | ||||||
|  |         `sql` is the raw query containing placeholders, and `params` is the | ||||||
|  |         sequence of parameters. These are used by default, but this method | ||||||
|  |         exists for database backends to provide a better implementation | ||||||
|  |         according to their own quoting schemes. | ||||||
|  |         """ | ||||||
|  |         # Convert params to contain Unicode values. | ||||||
|  |         to_unicode = lambda s: force_text(s, strings_only=True, errors='replace') | ||||||
|  |         if isinstance(params, (list, tuple)): | ||||||
|  |             u_params = tuple(to_unicode(val) for val in params) | ||||||
|  |         elif params is None: | ||||||
|  |             u_params = () | ||||||
|  |         else: | ||||||
|  |             u_params = {to_unicode(k): to_unicode(v) for k, v in params.items()} | ||||||
|  |  | ||||||
|  |         return six.text_type("QUERY = %r - PARAMS = %r") % (sql, u_params) | ||||||
|  |  | ||||||
|  |     def last_insert_id(self, cursor, table_name, pk_name): | ||||||
|  |         """ | ||||||
|  |         Given a cursor object that has just performed an INSERT statement into | ||||||
|  |         a table that has an auto-incrementing ID, returns the newly created ID. | ||||||
|  |  | ||||||
|  |         This method also receives the table name and the name of the primary-key | ||||||
|  |         column. | ||||||
|  |         """ | ||||||
|  |         return cursor.lastrowid | ||||||
|  |  | ||||||
|  |     def lookup_cast(self, lookup_type): | ||||||
|  |         """ | ||||||
|  |         Returns the string to use in a query when performing lookups | ||||||
|  |         ("contains", "like", etc). The resulting string should contain a '%s' | ||||||
|  |         placeholder for the column being searched against. | ||||||
|  |         """ | ||||||
|  |         return "%s" | ||||||
|  |  | ||||||
|  |     def max_in_list_size(self): | ||||||
|  |         """ | ||||||
|  |         Returns the maximum number of items that can be passed in a single 'IN' | ||||||
|  |         list condition, or None if the backend does not impose a limit. | ||||||
|  |         """ | ||||||
|  |         return None | ||||||
|  |  | ||||||
|  |     def max_name_length(self): | ||||||
|  |         """ | ||||||
|  |         Returns the maximum length of table and column names, or None if there | ||||||
|  |         is no limit. | ||||||
|  |         """ | ||||||
|  |         return None | ||||||
|  |  | ||||||
|  |     def no_limit_value(self): | ||||||
|  |         """ | ||||||
|  |         Returns the value to use for the LIMIT when we are wanting "LIMIT | ||||||
|  |         infinity". Returns None if the limit clause can be omitted in this case. | ||||||
|  |         """ | ||||||
|  |         raise NotImplementedError('subclasses of BaseDatabaseOperations may require a no_limit_value() method') | ||||||
|  |  | ||||||
|  |     def pk_default_value(self): | ||||||
|  |         """ | ||||||
|  |         Returns the value to use during an INSERT statement to specify that | ||||||
|  |         the field should use its default value. | ||||||
|  |         """ | ||||||
|  |         return 'DEFAULT' | ||||||
|  |  | ||||||
|  |     def prepare_sql_script(self, sql, _allow_fallback=False): | ||||||
|  |         """ | ||||||
|  |         Takes a SQL script that may contain multiple lines and returns a list | ||||||
|  |         of statements to feed to successive cursor.execute() calls. | ||||||
|  |  | ||||||
|  |         Since few databases are able to process raw SQL scripts in a single | ||||||
|  |         cursor.execute() call and PEP 249 doesn't talk about this use case, | ||||||
|  |         the default implementation is conservative. | ||||||
|  |         """ | ||||||
|  |         # Remove _allow_fallback and keep only 'return ...' in Django 1.9. | ||||||
|  |         try: | ||||||
|  |             # This import must stay inside the method because it's optional. | ||||||
|  |             import sqlparse | ||||||
|  |         except ImportError: | ||||||
|  |             if _allow_fallback: | ||||||
|  |                 # Without sqlparse, fall back to the legacy (and buggy) logic. | ||||||
|  |                 warnings.warn( | ||||||
|  |                     "Providing initial SQL data on a %s database will require " | ||||||
|  |                     "sqlparse in Django 1.9." % self.connection.vendor, | ||||||
|  |                     RemovedInDjango19Warning) | ||||||
|  |                 from django.core.management.sql import _split_statements | ||||||
|  |                 return _split_statements(sql) | ||||||
|  |             else: | ||||||
|  |                 raise | ||||||
|  |         else: | ||||||
|  |             return [sqlparse.format(statement, strip_comments=True) | ||||||
|  |                     for statement in sqlparse.split(sql) if statement] | ||||||
|  |  | ||||||
|  |     def process_clob(self, value): | ||||||
|  |         """ | ||||||
|  |         Returns the value of a CLOB column, for backends that return a locator | ||||||
|  |         object that requires additional processing. | ||||||
|  |         """ | ||||||
|  |         return value | ||||||
|  |  | ||||||
|  |     def return_insert_id(self): | ||||||
|  |         """ | ||||||
|  |         For backends that support returning the last insert ID as part | ||||||
|  |         of an insert query, this method returns the SQL and params to | ||||||
|  |         append to the INSERT query. The returned fragment should | ||||||
|  |         contain a format string to hold the appropriate column. | ||||||
|  |         """ | ||||||
|  |         pass | ||||||
|  |  | ||||||
|  |     def compiler(self, compiler_name): | ||||||
|  |         """ | ||||||
|  |         Returns the SQLCompiler class corresponding to the given name, | ||||||
|  |         in the namespace corresponding to the `compiler_module` attribute | ||||||
|  |         on this backend. | ||||||
|  |         """ | ||||||
|  |         if self._cache is None: | ||||||
|  |             self._cache = import_module(self.compiler_module) | ||||||
|  |         return getattr(self._cache, compiler_name) | ||||||
|  |  | ||||||
|  |     def quote_name(self, name): | ||||||
|  |         """ | ||||||
|  |         Returns a quoted version of the given table, index or column name. Does | ||||||
|  |         not quote the given name if it's already been quoted. | ||||||
|  |         """ | ||||||
|  |         raise NotImplementedError('subclasses of BaseDatabaseOperations may require a quote_name() method') | ||||||
|  |  | ||||||
|  |     def random_function_sql(self): | ||||||
|  |         """ | ||||||
|  |         Returns an SQL expression that returns a random value. | ||||||
|  |         """ | ||||||
|  |         return 'RANDOM()' | ||||||
|  |  | ||||||
|  |     def regex_lookup(self, lookup_type): | ||||||
|  |         """ | ||||||
|  |         Returns the string to use in a query when performing regular expression | ||||||
|  |         lookups (using "regex" or "iregex"). The resulting string should | ||||||
|  |         contain a '%s' placeholder for the column being searched against. | ||||||
|  |  | ||||||
|  |         If the feature is not supported (or part of it is not supported), a | ||||||
|  |         NotImplementedError exception can be raised. | ||||||
|  |         """ | ||||||
|  |         raise NotImplementedError('subclasses of BaseDatabaseOperations may require a regex_lookup() method') | ||||||
|  |  | ||||||
|  |     def savepoint_create_sql(self, sid): | ||||||
|  |         """ | ||||||
|  |         Returns the SQL for starting a new savepoint. Only required if the | ||||||
|  |         "uses_savepoints" feature is True. The "sid" parameter is a string | ||||||
|  |         for the savepoint id. | ||||||
|  |         """ | ||||||
|  |         return "SAVEPOINT %s" % self.quote_name(sid) | ||||||
|  |  | ||||||
|  |     def savepoint_commit_sql(self, sid): | ||||||
|  |         """ | ||||||
|  |         Returns the SQL for committing the given savepoint. | ||||||
|  |         """ | ||||||
|  |         return "RELEASE SAVEPOINT %s" % self.quote_name(sid) | ||||||
|  |  | ||||||
|  |     def savepoint_rollback_sql(self, sid): | ||||||
|  |         """ | ||||||
|  |         Returns the SQL for rolling back the given savepoint. | ||||||
|  |         """ | ||||||
|  |         return "ROLLBACK TO SAVEPOINT %s" % self.quote_name(sid) | ||||||
|  |  | ||||||
|  |     def set_time_zone_sql(self): | ||||||
|  |         """ | ||||||
|  |         Returns the SQL that will set the connection's time zone. | ||||||
|  |  | ||||||
|  |         Returns '' if the backend doesn't support time zones. | ||||||
|  |         """ | ||||||
|  |         return '' | ||||||
|  |  | ||||||
|  |     def sql_flush(self, style, tables, sequences, allow_cascade=False): | ||||||
|  |         """ | ||||||
|  |         Returns a list of SQL statements required to remove all data from | ||||||
|  |         the given database tables (without actually removing the tables | ||||||
|  |         themselves). | ||||||
|  |  | ||||||
|  |         The returned value also includes SQL statements required to reset DB | ||||||
|  |         sequences passed in :param sequences:. | ||||||
|  |  | ||||||
|  |         The `style` argument is a Style object as returned by either | ||||||
|  |         color_style() or no_style() in django.core.management.color. | ||||||
|  |  | ||||||
|  |         The `allow_cascade` argument determines whether truncation may cascade | ||||||
|  |         to tables with foreign keys pointing the tables being truncated. | ||||||
|  |         PostgreSQL requires a cascade even if these tables are empty. | ||||||
|  |         """ | ||||||
|  |         raise NotImplementedError('subclasses of BaseDatabaseOperations must provide a sql_flush() method') | ||||||
|  |  | ||||||
|  |     def sequence_reset_by_name_sql(self, style, sequences): | ||||||
|  |         """ | ||||||
|  |         Returns a list of the SQL statements required to reset sequences | ||||||
|  |         passed in :param sequences:. | ||||||
|  |  | ||||||
|  |         The `style` argument is a Style object as returned by either | ||||||
|  |         color_style() or no_style() in django.core.management.color. | ||||||
|  |         """ | ||||||
|  |         return [] | ||||||
|  |  | ||||||
|  |     def sequence_reset_sql(self, style, model_list): | ||||||
|  |         """ | ||||||
|  |         Returns a list of the SQL statements required to reset sequences for | ||||||
|  |         the given models. | ||||||
|  |  | ||||||
|  |         The `style` argument is a Style object as returned by either | ||||||
|  |         color_style() or no_style() in django.core.management.color. | ||||||
|  |         """ | ||||||
|  |         return []  # No sequence reset required by default. | ||||||
|  |  | ||||||
|  |     def start_transaction_sql(self): | ||||||
|  |         """ | ||||||
|  |         Returns the SQL statement required to start a transaction. | ||||||
|  |         """ | ||||||
|  |         return "BEGIN;" | ||||||
|  |  | ||||||
|  |     def end_transaction_sql(self, success=True): | ||||||
|  |         """ | ||||||
|  |         Returns the SQL statement required to end a transaction. | ||||||
|  |         """ | ||||||
|  |         if not success: | ||||||
|  |             return "ROLLBACK;" | ||||||
|  |         return "COMMIT;" | ||||||
|  |  | ||||||
|  |     def tablespace_sql(self, tablespace, inline=False): | ||||||
|  |         """ | ||||||
|  |         Returns the SQL that will be used in a query to define the tablespace. | ||||||
|  |  | ||||||
|  |         Returns '' if the backend doesn't support tablespaces. | ||||||
|  |  | ||||||
|  |         If inline is True, the SQL is appended to a row; otherwise it's appended | ||||||
|  |         to the entire CREATE TABLE or CREATE INDEX statement. | ||||||
|  |         """ | ||||||
|  |         return '' | ||||||
|  |  | ||||||
|  |     def prep_for_like_query(self, x): | ||||||
|  |         """Prepares a value for use in a LIKE query.""" | ||||||
|  |         return force_text(x).replace("\\", "\\\\").replace("%", "\%").replace("_", "\_") | ||||||
|  |  | ||||||
|  |     # Same as prep_for_like_query(), but called for "iexact" matches, which | ||||||
|  |     # need not necessarily be implemented using "LIKE" in the backend. | ||||||
|  |     prep_for_iexact_query = prep_for_like_query | ||||||
|  |  | ||||||
|  |     def validate_autopk_value(self, value): | ||||||
|  |         """ | ||||||
|  |         Certain backends do not accept some values for "serial" fields | ||||||
|  |         (for example zero in MySQL). This method will raise a ValueError | ||||||
|  |         if the value is invalid, otherwise returns validated value. | ||||||
|  |         """ | ||||||
|  |         return value | ||||||
|  |  | ||||||
|  |     def value_to_db_date(self, value): | ||||||
|  |         """ | ||||||
|  |         Transform a date value to an object compatible with what is expected | ||||||
|  |         by the backend driver for date columns. | ||||||
|  |         """ | ||||||
|  |         if value is None: | ||||||
|  |             return None | ||||||
|  |         return six.text_type(value) | ||||||
|  |  | ||||||
|  |     def value_to_db_datetime(self, value): | ||||||
|  |         """ | ||||||
|  |         Transform a datetime value to an object compatible with what is expected | ||||||
|  |         by the backend driver for datetime columns. | ||||||
|  |         """ | ||||||
|  |         if value is None: | ||||||
|  |             return None | ||||||
|  |         return six.text_type(value) | ||||||
|  |  | ||||||
|  |     def value_to_db_time(self, value): | ||||||
|  |         """ | ||||||
|  |         Transform a time value to an object compatible with what is expected | ||||||
|  |         by the backend driver for time columns. | ||||||
|  |         """ | ||||||
|  |         if value is None: | ||||||
|  |             return None | ||||||
|  |         if timezone.is_aware(value): | ||||||
|  |             raise ValueError("Django does not support timezone-aware times.") | ||||||
|  |         return six.text_type(value) | ||||||
|  |  | ||||||
|  |     def value_to_db_decimal(self, value, max_digits, decimal_places): | ||||||
|  |         """ | ||||||
|  |         Transform a decimal.Decimal value to an object compatible with what is | ||||||
|  |         expected by the backend driver for decimal (numeric) columns. | ||||||
|  |         """ | ||||||
|  |         return utils.format_number(value, max_digits, decimal_places) | ||||||
|  |  | ||||||
|  |     def year_lookup_bounds_for_date_field(self, value): | ||||||
|  |         """ | ||||||
|  |         Returns a two-elements list with the lower and upper bound to be used | ||||||
|  |         with a BETWEEN operator to query a DateField value using a year | ||||||
|  |         lookup. | ||||||
|  |  | ||||||
|  |         `value` is an int, containing the looked-up year. | ||||||
|  |         """ | ||||||
|  |         first = datetime.date(value, 1, 1) | ||||||
|  |         second = datetime.date(value, 12, 31) | ||||||
|  |         return [first, second] | ||||||
|  |  | ||||||
|  |     def year_lookup_bounds_for_datetime_field(self, value): | ||||||
|  |         """ | ||||||
|  |         Returns a two-elements list with the lower and upper bound to be used | ||||||
|  |         with a BETWEEN operator to query a DateTimeField value using a year | ||||||
|  |         lookup. | ||||||
|  |  | ||||||
|  |         `value` is an int, containing the looked-up year. | ||||||
|  |         """ | ||||||
|  |         first = datetime.datetime(value, 1, 1) | ||||||
|  |         second = datetime.datetime(value, 12, 31, 23, 59, 59, 999999) | ||||||
|  |         if settings.USE_TZ: | ||||||
|  |             tz = timezone.get_current_timezone() | ||||||
|  |             first = timezone.make_aware(first, tz) | ||||||
|  |             second = timezone.make_aware(second, tz) | ||||||
|  |         return [first, second] | ||||||
|  |  | ||||||
|  |     def get_db_converters(self, expression): | ||||||
|  |         """Get a list of functions needed to convert field data. | ||||||
|  |  | ||||||
|  |         Some field types on some backends do not provide data in the correct | ||||||
|  |         format, this is the hook for coverter functions. | ||||||
|  |         """ | ||||||
|  |         return [] | ||||||
|  |  | ||||||
|  |     def convert_durationfield_value(self, value, expression, context): | ||||||
|  |         if value is not None: | ||||||
|  |             value = str(decimal.Decimal(value) / decimal.Decimal(1000000)) | ||||||
|  |             value = parse_duration(value) | ||||||
|  |         return value | ||||||
|  |  | ||||||
|  |     def check_aggregate_support(self, aggregate_func): | ||||||
|  |         """Check that the backend supports the provided aggregate | ||||||
|  |  | ||||||
|  |         This is used on specific backends to rule out known aggregates | ||||||
|  |         that are known to have faulty implementations. If the named | ||||||
|  |         aggregate function has a known problem, the backend should | ||||||
|  |         raise NotImplementedError. | ||||||
|  |         """ | ||||||
|  |         pass | ||||||
|  |  | ||||||
|  |     def combine_expression(self, connector, sub_expressions): | ||||||
|  |         """Combine a list of subexpressions into a single expression, using | ||||||
|  |         the provided connecting operator. This is required because operators | ||||||
|  |         can vary between backends (e.g., Oracle with %% and &) and between | ||||||
|  |         subexpression types (e.g., date expressions) | ||||||
|  |         """ | ||||||
|  |         conn = ' %s ' % connector | ||||||
|  |         return conn.join(sub_expressions) | ||||||
|  |  | ||||||
|  |     def combine_duration_expression(self, connector, sub_expressions): | ||||||
|  |         return self.combine_expression(connector, sub_expressions) | ||||||
|  |  | ||||||
|  |     def modify_insert_params(self, placeholders, params): | ||||||
|  |         """Allow modification of insert parameters. Needed for Oracle Spatial | ||||||
|  |         backend due to #10888. | ||||||
|  |         """ | ||||||
|  |         return params | ||||||
|  |  | ||||||
|  |     def integer_field_range(self, internal_type): | ||||||
|  |         """ | ||||||
|  |         Given an integer field internal type (e.g. 'PositiveIntegerField'), | ||||||
|  |         returns a tuple of the (min_value, max_value) form representing the | ||||||
|  |         range of the column type bound to the field. | ||||||
|  |         """ | ||||||
|  |         return self.integer_field_ranges[internal_type] | ||||||
| @@ -3,9 +3,9 @@ import hashlib | |||||||
| from django.db.backends.utils import truncate_name | from django.db.backends.utils import truncate_name | ||||||
| from django.db.models.fields.related import ManyToManyField | from django.db.models.fields.related import ManyToManyField | ||||||
| from django.db.transaction import atomic | from django.db.transaction import atomic | ||||||
|  | from django.utils import six | ||||||
| from django.utils.encoding import force_bytes | from django.utils.encoding import force_bytes | ||||||
| from django.utils.log import getLogger | from django.utils.log import getLogger | ||||||
| from django.utils import six |  | ||||||
| 
 | 
 | ||||||
| logger = getLogger('django.db.backends.schema') | logger = getLogger('django.db.backends.schema') | ||||||
| 
 | 
 | ||||||
							
								
								
									
										38
									
								
								django/db/backends/base/validation.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										38
									
								
								django/db/backends/base/validation.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,38 @@ | |||||||
|  | from django.core import checks | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class BaseDatabaseValidation(object): | ||||||
|  |     """ | ||||||
|  |     This class encapsulates all backend-specific model validation. | ||||||
|  |     """ | ||||||
|  |     def __init__(self, connection): | ||||||
|  |         self.connection = connection | ||||||
|  |  | ||||||
|  |     def validate_field(self, errors, opts, f): | ||||||
|  |         """ | ||||||
|  |         By default, there is no backend-specific validation. | ||||||
|  |  | ||||||
|  |         This method has been deprecated by the new checks framework. New | ||||||
|  |         backends should implement check_field instead. | ||||||
|  |         """ | ||||||
|  |         # This is deliberately commented out. It exists as a marker to | ||||||
|  |         # remind us to remove this method, and the check_field() shim, | ||||||
|  |         # when the time comes. | ||||||
|  |         # warnings.warn('"validate_field" has been deprecated", RemovedInDjango19Warning) | ||||||
|  |         pass | ||||||
|  |  | ||||||
|  |     def check_field(self, field, **kwargs): | ||||||
|  |         class ErrorList(list): | ||||||
|  |             """A dummy list class that emulates API used by the older | ||||||
|  |             validate_field() method. When validate_field() is fully | ||||||
|  |             deprecated, this dummy can be removed too. | ||||||
|  |             """ | ||||||
|  |             def add(self, opts, error_message): | ||||||
|  |                 self.append(checks.Error(error_message, hint=None, obj=field)) | ||||||
|  |  | ||||||
|  |         errors = ErrorList() | ||||||
|  |         # Some tests create fields in isolation -- the fields are not attached | ||||||
|  |         # to any model, so they have no `model` attribute. | ||||||
|  |         opts = field.model._meta if hasattr(field, 'model') else None | ||||||
|  |         self.validate_field(errors, field, opts) | ||||||
|  |         return list(errors) | ||||||
| @@ -8,10 +8,13 @@ ImproperlyConfigured. | |||||||
| """ | """ | ||||||
|  |  | ||||||
| from django.core.exceptions import ImproperlyConfigured | from django.core.exceptions import ImproperlyConfigured | ||||||
| from django.db.backends import (BaseDatabaseOperations, BaseDatabaseClient, | from django.db.backends.base.base import BaseDatabaseWrapper | ||||||
|     BaseDatabaseIntrospection, BaseDatabaseWrapper, BaseDatabaseFeatures, | from django.db.backends.base.client import BaseDatabaseClient | ||||||
|     BaseDatabaseValidation) | from django.db.backends.base.creation import BaseDatabaseCreation | ||||||
| from django.db.backends.creation import BaseDatabaseCreation | from django.db.backends.base.features import BaseDatabaseFeatures | ||||||
|  | from django.db.backends.base.operations import BaseDatabaseOperations | ||||||
|  | from django.db.backends.base.introspection import BaseDatabaseIntrospection | ||||||
|  | from django.db.backends.base.validation import BaseDatabaseValidation | ||||||
|  |  | ||||||
|  |  | ||||||
| def complain(*args, **kwargs): | def complain(*args, **kwargs): | ||||||
|   | |||||||
| @@ -9,15 +9,35 @@ from __future__ import unicode_literals | |||||||
| import datetime | import datetime | ||||||
| import re | import re | ||||||
| import sys | import sys | ||||||
| import uuid |  | ||||||
| import warnings | import warnings | ||||||
|  |  | ||||||
|  | from django.conf import settings | ||||||
|  | from django.db import utils | ||||||
|  | from django.db.backends import utils as backend_utils | ||||||
|  | from django.db.backends.base.base import BaseDatabaseWrapper | ||||||
|  | from django.utils.encoding import force_str | ||||||
|  | from django.db.backends.mysql.schema import DatabaseSchemaEditor | ||||||
|  | from django.utils import six, timezone | ||||||
|  | from django.utils.functional import cached_property | ||||||
|  | from django.utils.safestring import SafeBytes, SafeText | ||||||
|  |  | ||||||
| try: | try: | ||||||
|     import MySQLdb as Database |     import MySQLdb as Database | ||||||
| except ImportError as e: | except ImportError as e: | ||||||
|     from django.core.exceptions import ImproperlyConfigured |     from django.core.exceptions import ImproperlyConfigured | ||||||
|     raise ImproperlyConfigured("Error loading MySQLdb module: %s" % e) |     raise ImproperlyConfigured("Error loading MySQLdb module: %s" % e) | ||||||
|  |  | ||||||
|  | from MySQLdb.converters import conversions, Thing2Literal | ||||||
|  | from MySQLdb.constants import FIELD_TYPE, CLIENT | ||||||
|  |  | ||||||
|  | # Some of these import MySQLdb, so import them after checking if it's installed. | ||||||
|  | from .client import DatabaseClient | ||||||
|  | from .creation import DatabaseCreation | ||||||
|  | from .features import DatabaseFeatures | ||||||
|  | from .introspection import DatabaseIntrospection | ||||||
|  | from .operations import DatabaseOperations | ||||||
|  | from .validation import DatabaseValidation | ||||||
|  |  | ||||||
| # We want version (1, 2, 1, 'final', 2) or later. We can't just use | # We want version (1, 2, 1, 'final', 2) or later. We can't just use | ||||||
| # lexicographic ordering in this check because then (1, 2, 1, 'gamma') | # lexicographic ordering in this check because then (1, 2, 1, 'gamma') | ||||||
| # inadvertently passes the version test. | # inadvertently passes the version test. | ||||||
| @@ -27,28 +47,6 @@ if (version < (1, 2, 1) or (version[:3] == (1, 2, 1) and | |||||||
|     from django.core.exceptions import ImproperlyConfigured |     from django.core.exceptions import ImproperlyConfigured | ||||||
|     raise ImproperlyConfigured("MySQLdb-1.2.1p2 or newer is required; you have %s" % Database.__version__) |     raise ImproperlyConfigured("MySQLdb-1.2.1p2 or newer is required; you have %s" % Database.__version__) | ||||||
|  |  | ||||||
| from MySQLdb.converters import conversions, Thing2Literal |  | ||||||
| from MySQLdb.constants import FIELD_TYPE, CLIENT |  | ||||||
|  |  | ||||||
| try: |  | ||||||
|     import pytz |  | ||||||
| except ImportError: |  | ||||||
|     pytz = None |  | ||||||
|  |  | ||||||
| from django.conf import settings |  | ||||||
| from django.db import utils |  | ||||||
| from django.db.backends import (utils as backend_utils, BaseDatabaseFeatures, |  | ||||||
|     BaseDatabaseOperations, BaseDatabaseWrapper) |  | ||||||
| from django.db.backends.mysql.client import DatabaseClient |  | ||||||
| from django.db.backends.mysql.creation import DatabaseCreation |  | ||||||
| from django.db.backends.mysql.introspection import DatabaseIntrospection |  | ||||||
| from django.db.backends.mysql.validation import DatabaseValidation |  | ||||||
| from django.utils.encoding import force_str, force_text |  | ||||||
| from django.db.backends.mysql.schema import DatabaseSchemaEditor |  | ||||||
| from django.utils.functional import cached_property |  | ||||||
| from django.utils.safestring import SafeBytes, SafeText |  | ||||||
| from django.utils import six |  | ||||||
| from django.utils import timezone |  | ||||||
|  |  | ||||||
| DatabaseError = Database.DatabaseError | DatabaseError = Database.DatabaseError | ||||||
| IntegrityError = Database.IntegrityError | IntegrityError = Database.IntegrityError | ||||||
| @@ -159,259 +157,6 @@ class CursorWrapper(object): | |||||||
|         self.close() |         self.close() | ||||||
|  |  | ||||||
|  |  | ||||||
| class DatabaseFeatures(BaseDatabaseFeatures): |  | ||||||
|     empty_fetchmany_value = () |  | ||||||
|     update_can_self_select = False |  | ||||||
|     allows_group_by_pk = True |  | ||||||
|     related_fields_match_type = True |  | ||||||
|     allow_sliced_subqueries = False |  | ||||||
|     has_bulk_insert = True |  | ||||||
|     has_select_for_update = True |  | ||||||
|     has_select_for_update_nowait = False |  | ||||||
|     supports_forward_references = False |  | ||||||
|     supports_regex_backreferencing = False |  | ||||||
|     supports_date_lookup_using_string = False |  | ||||||
|     can_introspect_autofield = True |  | ||||||
|     can_introspect_binary_field = False |  | ||||||
|     can_introspect_small_integer_field = True |  | ||||||
|     supports_timezones = False |  | ||||||
|     requires_explicit_null_ordering_when_grouping = True |  | ||||||
|     allows_auto_pk_0 = False |  | ||||||
|     uses_savepoints = True |  | ||||||
|     can_release_savepoints = True |  | ||||||
|     atomic_transactions = False |  | ||||||
|     supports_column_check_constraints = False |  | ||||||
|  |  | ||||||
|     @cached_property |  | ||||||
|     def _mysql_storage_engine(self): |  | ||||||
|         "Internal method used in Django tests. Don't rely on this from your code" |  | ||||||
|         with self.connection.cursor() as cursor: |  | ||||||
|             cursor.execute("SELECT ENGINE FROM INFORMATION_SCHEMA.ENGINES WHERE SUPPORT = 'DEFAULT'") |  | ||||||
|             result = cursor.fetchone() |  | ||||||
|         return result[0] |  | ||||||
|  |  | ||||||
|     @cached_property |  | ||||||
|     def can_introspect_foreign_keys(self): |  | ||||||
|         "Confirm support for introspected foreign keys" |  | ||||||
|         return self._mysql_storage_engine != 'MyISAM' |  | ||||||
|  |  | ||||||
|     @cached_property |  | ||||||
|     def supports_microsecond_precision(self): |  | ||||||
|         # See https://github.com/farcepest/MySQLdb1/issues/24 for the reason |  | ||||||
|         # about requiring MySQLdb 1.2.5 |  | ||||||
|         return self.connection.mysql_version >= (5, 6, 4) and Database.version_info >= (1, 2, 5) |  | ||||||
|  |  | ||||||
|     @cached_property |  | ||||||
|     def has_zoneinfo_database(self): |  | ||||||
|         # MySQL accepts full time zones names (eg. Africa/Nairobi) but rejects |  | ||||||
|         # abbreviations (eg. EAT). When pytz isn't installed and the current |  | ||||||
|         # time zone is LocalTimezone (the only sensible value in this |  | ||||||
|         # context), the current time zone name will be an abbreviation. As a |  | ||||||
|         # consequence, MySQL cannot perform time zone conversions reliably. |  | ||||||
|         if pytz is None: |  | ||||||
|             return False |  | ||||||
|  |  | ||||||
|         # Test if the time zone definitions are installed. |  | ||||||
|         with self.connection.cursor() as cursor: |  | ||||||
|             cursor.execute("SELECT 1 FROM mysql.time_zone LIMIT 1") |  | ||||||
|             return cursor.fetchone() is not None |  | ||||||
|  |  | ||||||
|     def introspected_boolean_field_type(self, *args, **kwargs): |  | ||||||
|         return 'IntegerField' |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class DatabaseOperations(BaseDatabaseOperations): |  | ||||||
|     compiler_module = "django.db.backends.mysql.compiler" |  | ||||||
|  |  | ||||||
|     # MySQL stores positive fields as UNSIGNED ints. |  | ||||||
|     integer_field_ranges = dict(BaseDatabaseOperations.integer_field_ranges, |  | ||||||
|         PositiveSmallIntegerField=(0, 4294967295), |  | ||||||
|         PositiveIntegerField=(0, 18446744073709551615), |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|     def date_extract_sql(self, lookup_type, field_name): |  | ||||||
|         # http://dev.mysql.com/doc/mysql/en/date-and-time-functions.html |  | ||||||
|         if lookup_type == 'week_day': |  | ||||||
|             # DAYOFWEEK() returns an integer, 1-7, Sunday=1. |  | ||||||
|             # Note: WEEKDAY() returns 0-6, Monday=0. |  | ||||||
|             return "DAYOFWEEK(%s)" % field_name |  | ||||||
|         else: |  | ||||||
|             return "EXTRACT(%s FROM %s)" % (lookup_type.upper(), field_name) |  | ||||||
|  |  | ||||||
|     def date_trunc_sql(self, lookup_type, field_name): |  | ||||||
|         fields = ['year', 'month', 'day', 'hour', 'minute', 'second'] |  | ||||||
|         format = ('%%Y-', '%%m', '-%%d', ' %%H:', '%%i', ':%%s')  # Use double percents to escape. |  | ||||||
|         format_def = ('0000-', '01', '-01', ' 00:', '00', ':00') |  | ||||||
|         try: |  | ||||||
|             i = fields.index(lookup_type) + 1 |  | ||||||
|         except ValueError: |  | ||||||
|             sql = field_name |  | ||||||
|         else: |  | ||||||
|             format_str = ''.join([f for f in format[:i]] + [f for f in format_def[i:]]) |  | ||||||
|             sql = "CAST(DATE_FORMAT(%s, '%s') AS DATETIME)" % (field_name, format_str) |  | ||||||
|         return sql |  | ||||||
|  |  | ||||||
|     def datetime_extract_sql(self, lookup_type, field_name, tzname): |  | ||||||
|         if settings.USE_TZ: |  | ||||||
|             field_name = "CONVERT_TZ(%s, 'UTC', %%s)" % field_name |  | ||||||
|             params = [tzname] |  | ||||||
|         else: |  | ||||||
|             params = [] |  | ||||||
|         # http://dev.mysql.com/doc/mysql/en/date-and-time-functions.html |  | ||||||
|         if lookup_type == 'week_day': |  | ||||||
|             # DAYOFWEEK() returns an integer, 1-7, Sunday=1. |  | ||||||
|             # Note: WEEKDAY() returns 0-6, Monday=0. |  | ||||||
|             sql = "DAYOFWEEK(%s)" % field_name |  | ||||||
|         else: |  | ||||||
|             sql = "EXTRACT(%s FROM %s)" % (lookup_type.upper(), field_name) |  | ||||||
|         return sql, params |  | ||||||
|  |  | ||||||
|     def datetime_trunc_sql(self, lookup_type, field_name, tzname): |  | ||||||
|         if settings.USE_TZ: |  | ||||||
|             field_name = "CONVERT_TZ(%s, 'UTC', %%s)" % field_name |  | ||||||
|             params = [tzname] |  | ||||||
|         else: |  | ||||||
|             params = [] |  | ||||||
|         fields = ['year', 'month', 'day', 'hour', 'minute', 'second'] |  | ||||||
|         format = ('%%Y-', '%%m', '-%%d', ' %%H:', '%%i', ':%%s')  # Use double percents to escape. |  | ||||||
|         format_def = ('0000-', '01', '-01', ' 00:', '00', ':00') |  | ||||||
|         try: |  | ||||||
|             i = fields.index(lookup_type) + 1 |  | ||||||
|         except ValueError: |  | ||||||
|             sql = field_name |  | ||||||
|         else: |  | ||||||
|             format_str = ''.join([f for f in format[:i]] + [f for f in format_def[i:]]) |  | ||||||
|             sql = "CAST(DATE_FORMAT(%s, '%s') AS DATETIME)" % (field_name, format_str) |  | ||||||
|         return sql, params |  | ||||||
|  |  | ||||||
|     def date_interval_sql(self, timedelta): |  | ||||||
|         return "INTERVAL '%d 0:0:%d:%d' DAY_MICROSECOND" % ( |  | ||||||
|             timedelta.days, timedelta.seconds, timedelta.microseconds), [] |  | ||||||
|  |  | ||||||
|     def format_for_duration_arithmetic(self, sql): |  | ||||||
|         return 'INTERVAL %s MICROSECOND' % sql |  | ||||||
|  |  | ||||||
|     def drop_foreignkey_sql(self): |  | ||||||
|         return "DROP FOREIGN KEY" |  | ||||||
|  |  | ||||||
|     def force_no_ordering(self): |  | ||||||
|         """ |  | ||||||
|         "ORDER BY NULL" prevents MySQL from implicitly ordering by grouped |  | ||||||
|         columns. If no ordering would otherwise be applied, we don't want any |  | ||||||
|         implicit sorting going on. |  | ||||||
|         """ |  | ||||||
|         return [(None, ("NULL", [], False))] |  | ||||||
|  |  | ||||||
|     def fulltext_search_sql(self, field_name): |  | ||||||
|         return 'MATCH (%s) AGAINST (%%s IN BOOLEAN MODE)' % field_name |  | ||||||
|  |  | ||||||
|     def last_executed_query(self, cursor, sql, params): |  | ||||||
|         # With MySQLdb, cursor objects have an (undocumented) "_last_executed" |  | ||||||
|         # attribute where the exact query sent to the database is saved. |  | ||||||
|         # See MySQLdb/cursors.py in the source distribution. |  | ||||||
|         return force_text(getattr(cursor, '_last_executed', None), errors='replace') |  | ||||||
|  |  | ||||||
|     def no_limit_value(self): |  | ||||||
|         # 2**64 - 1, as recommended by the MySQL documentation |  | ||||||
|         return 18446744073709551615 |  | ||||||
|  |  | ||||||
|     def quote_name(self, name): |  | ||||||
|         if name.startswith("`") and name.endswith("`"): |  | ||||||
|             return name  # Quoting once is enough. |  | ||||||
|         return "`%s`" % name |  | ||||||
|  |  | ||||||
|     def random_function_sql(self): |  | ||||||
|         return 'RAND()' |  | ||||||
|  |  | ||||||
|     def sql_flush(self, style, tables, sequences, allow_cascade=False): |  | ||||||
|         # NB: The generated SQL below is specific to MySQL |  | ||||||
|         # 'TRUNCATE x;', 'TRUNCATE y;', 'TRUNCATE z;'... style SQL statements |  | ||||||
|         # to clear all tables of all data |  | ||||||
|         if tables: |  | ||||||
|             sql = ['SET FOREIGN_KEY_CHECKS = 0;'] |  | ||||||
|             for table in tables: |  | ||||||
|                 sql.append('%s %s;' % ( |  | ||||||
|                     style.SQL_KEYWORD('TRUNCATE'), |  | ||||||
|                     style.SQL_FIELD(self.quote_name(table)), |  | ||||||
|                 )) |  | ||||||
|             sql.append('SET FOREIGN_KEY_CHECKS = 1;') |  | ||||||
|             sql.extend(self.sequence_reset_by_name_sql(style, sequences)) |  | ||||||
|             return sql |  | ||||||
|         else: |  | ||||||
|             return [] |  | ||||||
|  |  | ||||||
|     def validate_autopk_value(self, value): |  | ||||||
|         # MySQLism: zero in AUTO_INCREMENT field does not work. Refs #17653. |  | ||||||
|         if value == 0: |  | ||||||
|             raise ValueError('The database backend does not accept 0 as a ' |  | ||||||
|                              'value for AutoField.') |  | ||||||
|         return value |  | ||||||
|  |  | ||||||
|     def value_to_db_datetime(self, value): |  | ||||||
|         if value is None: |  | ||||||
|             return None |  | ||||||
|  |  | ||||||
|         # MySQL doesn't support tz-aware datetimes |  | ||||||
|         if timezone.is_aware(value): |  | ||||||
|             if settings.USE_TZ: |  | ||||||
|                 value = value.astimezone(timezone.utc).replace(tzinfo=None) |  | ||||||
|             else: |  | ||||||
|                 raise ValueError("MySQL backend does not support timezone-aware datetimes when USE_TZ is False.") |  | ||||||
|  |  | ||||||
|         return six.text_type(value) |  | ||||||
|  |  | ||||||
|     def value_to_db_time(self, value): |  | ||||||
|         if value is None: |  | ||||||
|             return None |  | ||||||
|  |  | ||||||
|         # MySQL doesn't support tz-aware times |  | ||||||
|         if timezone.is_aware(value): |  | ||||||
|             raise ValueError("MySQL backend does not support timezone-aware times.") |  | ||||||
|  |  | ||||||
|         return six.text_type(value) |  | ||||||
|  |  | ||||||
|     def max_name_length(self): |  | ||||||
|         return 64 |  | ||||||
|  |  | ||||||
|     def bulk_insert_sql(self, fields, num_values): |  | ||||||
|         items_sql = "(%s)" % ", ".join(["%s"] * len(fields)) |  | ||||||
|         return "VALUES " + ", ".join([items_sql] * num_values) |  | ||||||
|  |  | ||||||
|     def combine_expression(self, connector, sub_expressions): |  | ||||||
|         """ |  | ||||||
|         MySQL requires special cases for ^ operators in query expressions |  | ||||||
|         """ |  | ||||||
|         if connector == '^': |  | ||||||
|             return 'POW(%s)' % ','.join(sub_expressions) |  | ||||||
|         return super(DatabaseOperations, self).combine_expression(connector, sub_expressions) |  | ||||||
|  |  | ||||||
|     def get_db_converters(self, expression): |  | ||||||
|         converters = super(DatabaseOperations, self).get_db_converters(expression) |  | ||||||
|         internal_type = expression.output_field.get_internal_type() |  | ||||||
|         if internal_type in ['BooleanField', 'NullBooleanField']: |  | ||||||
|             converters.append(self.convert_booleanfield_value) |  | ||||||
|         if internal_type == 'UUIDField': |  | ||||||
|             converters.append(self.convert_uuidfield_value) |  | ||||||
|         if internal_type == 'TextField': |  | ||||||
|             converters.append(self.convert_textfield_value) |  | ||||||
|         return converters |  | ||||||
|  |  | ||||||
|     def convert_booleanfield_value(self, value, expression, context): |  | ||||||
|         if value in (0, 1): |  | ||||||
|             value = bool(value) |  | ||||||
|         return value |  | ||||||
|  |  | ||||||
|     def convert_uuidfield_value(self, value, expression, context): |  | ||||||
|         if value is not None: |  | ||||||
|             value = uuid.UUID(value) |  | ||||||
|         return value |  | ||||||
|  |  | ||||||
|     def convert_textfield_value(self, value, expression, context): |  | ||||||
|         if value is not None: |  | ||||||
|             value = force_text(value) |  | ||||||
|         return value |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class DatabaseWrapper(BaseDatabaseWrapper): | class DatabaseWrapper(BaseDatabaseWrapper): | ||||||
|     vendor = 'mysql' |     vendor = 'mysql' | ||||||
|     # This dictionary maps Field objects to their associated MySQL column |     # This dictionary maps Field objects to their associated MySQL column | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| import subprocess | import subprocess | ||||||
|  |  | ||||||
| from django.db.backends import BaseDatabaseClient | from django.db.backends.base.client import BaseDatabaseClient | ||||||
|  |  | ||||||
|  |  | ||||||
| class DatabaseClient(BaseDatabaseClient): | class DatabaseClient(BaseDatabaseClient): | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| from django.db.backends.creation import BaseDatabaseCreation | from django.db.backends.base.creation import BaseDatabaseCreation | ||||||
|  |  | ||||||
|  |  | ||||||
| class DatabaseCreation(BaseDatabaseCreation): | class DatabaseCreation(BaseDatabaseCreation): | ||||||
|   | |||||||
							
								
								
									
										70
									
								
								django/db/backends/mysql/features.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										70
									
								
								django/db/backends/mysql/features.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,70 @@ | |||||||
|  | from django.db.backends.base.features import BaseDatabaseFeatures | ||||||
|  | from django.utils.functional import cached_property | ||||||
|  |  | ||||||
|  | from .base import Database | ||||||
|  |  | ||||||
|  | try: | ||||||
|  |     import pytz | ||||||
|  | except ImportError: | ||||||
|  |     pytz = None | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class DatabaseFeatures(BaseDatabaseFeatures): | ||||||
|  |     empty_fetchmany_value = () | ||||||
|  |     update_can_self_select = False | ||||||
|  |     allows_group_by_pk = True | ||||||
|  |     related_fields_match_type = True | ||||||
|  |     allow_sliced_subqueries = False | ||||||
|  |     has_bulk_insert = True | ||||||
|  |     has_select_for_update = True | ||||||
|  |     has_select_for_update_nowait = False | ||||||
|  |     supports_forward_references = False | ||||||
|  |     supports_regex_backreferencing = False | ||||||
|  |     supports_date_lookup_using_string = False | ||||||
|  |     can_introspect_autofield = True | ||||||
|  |     can_introspect_binary_field = False | ||||||
|  |     can_introspect_small_integer_field = True | ||||||
|  |     supports_timezones = False | ||||||
|  |     requires_explicit_null_ordering_when_grouping = True | ||||||
|  |     allows_auto_pk_0 = False | ||||||
|  |     uses_savepoints = True | ||||||
|  |     can_release_savepoints = True | ||||||
|  |     atomic_transactions = False | ||||||
|  |     supports_column_check_constraints = False | ||||||
|  |  | ||||||
|  |     @cached_property | ||||||
|  |     def _mysql_storage_engine(self): | ||||||
|  |         "Internal method used in Django tests. Don't rely on this from your code" | ||||||
|  |         with self.connection.cursor() as cursor: | ||||||
|  |             cursor.execute("SELECT ENGINE FROM INFORMATION_SCHEMA.ENGINES WHERE SUPPORT = 'DEFAULT'") | ||||||
|  |             result = cursor.fetchone() | ||||||
|  |         return result[0] | ||||||
|  |  | ||||||
|  |     @cached_property | ||||||
|  |     def can_introspect_foreign_keys(self): | ||||||
|  |         "Confirm support for introspected foreign keys" | ||||||
|  |         return self._mysql_storage_engine != 'MyISAM' | ||||||
|  |  | ||||||
|  |     @cached_property | ||||||
|  |     def supports_microsecond_precision(self): | ||||||
|  |         # See https://github.com/farcepest/MySQLdb1/issues/24 for the reason | ||||||
|  |         # about requiring MySQLdb 1.2.5 | ||||||
|  |         return self.connection.mysql_version >= (5, 6, 4) and Database.version_info >= (1, 2, 5) | ||||||
|  |  | ||||||
|  |     @cached_property | ||||||
|  |     def has_zoneinfo_database(self): | ||||||
|  |         # MySQL accepts full time zones names (eg. Africa/Nairobi) but rejects | ||||||
|  |         # abbreviations (eg. EAT). When pytz isn't installed and the current | ||||||
|  |         # time zone is LocalTimezone (the only sensible value in this | ||||||
|  |         # context), the current time zone name will be an abbreviation. As a | ||||||
|  |         # consequence, MySQL cannot perform time zone conversions reliably. | ||||||
|  |         if pytz is None: | ||||||
|  |             return False | ||||||
|  |  | ||||||
|  |         # Test if the time zone definitions are installed. | ||||||
|  |         with self.connection.cursor() as cursor: | ||||||
|  |             cursor.execute("SELECT 1 FROM mysql.time_zone LIMIT 1") | ||||||
|  |             return cursor.fetchone() is not None | ||||||
|  |  | ||||||
|  |     def introspected_boolean_field_type(self, *args, **kwargs): | ||||||
|  |         return 'IntegerField' | ||||||
| @@ -1,10 +1,14 @@ | |||||||
| from collections import namedtuple | from collections import namedtuple | ||||||
| import re | import re | ||||||
| from .base import FIELD_TYPE |  | ||||||
| from django.utils.datastructures import OrderedSet | from django.utils.datastructures import OrderedSet | ||||||
| from django.db.backends import BaseDatabaseIntrospection, FieldInfo, TableInfo | from django.db.backends.base.introspection import ( | ||||||
|  |     BaseDatabaseIntrospection, FieldInfo, TableInfo, | ||||||
|  | ) | ||||||
| from django.utils.encoding import force_text | from django.utils.encoding import force_text | ||||||
|  |  | ||||||
|  | from MySQLdb.constants import FIELD_TYPE | ||||||
|  |  | ||||||
| FieldInfo = namedtuple('FieldInfo', FieldInfo._fields + ('extra',)) | FieldInfo = namedtuple('FieldInfo', FieldInfo._fields + ('extra',)) | ||||||
|  |  | ||||||
| foreign_key_re = re.compile(r"\sCONSTRAINT `[^`]*` FOREIGN KEY \(`([^`]*)`\) REFERENCES `([^`]*)` \(`([^`]*)`\)") | foreign_key_re = re.compile(r"\sCONSTRAINT `[^`]*` FOREIGN KEY \(`([^`]*)`\) REFERENCES `([^`]*)` \(`([^`]*)`\)") | ||||||
|   | |||||||
							
								
								
									
										200
									
								
								django/db/backends/mysql/operations.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										200
									
								
								django/db/backends/mysql/operations.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,200 @@ | |||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | import uuid | ||||||
|  |  | ||||||
|  | from django.conf import settings | ||||||
|  | from django.db.backends.base.operations import BaseDatabaseOperations | ||||||
|  | from django.utils import six, timezone | ||||||
|  | from django.utils.encoding import force_text | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class DatabaseOperations(BaseDatabaseOperations): | ||||||
|  |     compiler_module = "django.db.backends.mysql.compiler" | ||||||
|  |  | ||||||
|  |     # MySQL stores positive fields as UNSIGNED ints. | ||||||
|  |     integer_field_ranges = dict(BaseDatabaseOperations.integer_field_ranges, | ||||||
|  |         PositiveSmallIntegerField=(0, 4294967295), | ||||||
|  |         PositiveIntegerField=(0, 18446744073709551615), | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |     def date_extract_sql(self, lookup_type, field_name): | ||||||
|  |         # http://dev.mysql.com/doc/mysql/en/date-and-time-functions.html | ||||||
|  |         if lookup_type == 'week_day': | ||||||
|  |             # DAYOFWEEK() returns an integer, 1-7, Sunday=1. | ||||||
|  |             # Note: WEEKDAY() returns 0-6, Monday=0. | ||||||
|  |             return "DAYOFWEEK(%s)" % field_name | ||||||
|  |         else: | ||||||
|  |             return "EXTRACT(%s FROM %s)" % (lookup_type.upper(), field_name) | ||||||
|  |  | ||||||
|  |     def date_trunc_sql(self, lookup_type, field_name): | ||||||
|  |         fields = ['year', 'month', 'day', 'hour', 'minute', 'second'] | ||||||
|  |         format = ('%%Y-', '%%m', '-%%d', ' %%H:', '%%i', ':%%s')  # Use double percents to escape. | ||||||
|  |         format_def = ('0000-', '01', '-01', ' 00:', '00', ':00') | ||||||
|  |         try: | ||||||
|  |             i = fields.index(lookup_type) + 1 | ||||||
|  |         except ValueError: | ||||||
|  |             sql = field_name | ||||||
|  |         else: | ||||||
|  |             format_str = ''.join([f for f in format[:i]] + [f for f in format_def[i:]]) | ||||||
|  |             sql = "CAST(DATE_FORMAT(%s, '%s') AS DATETIME)" % (field_name, format_str) | ||||||
|  |         return sql | ||||||
|  |  | ||||||
|  |     def datetime_extract_sql(self, lookup_type, field_name, tzname): | ||||||
|  |         if settings.USE_TZ: | ||||||
|  |             field_name = "CONVERT_TZ(%s, 'UTC', %%s)" % field_name | ||||||
|  |             params = [tzname] | ||||||
|  |         else: | ||||||
|  |             params = [] | ||||||
|  |         # http://dev.mysql.com/doc/mysql/en/date-and-time-functions.html | ||||||
|  |         if lookup_type == 'week_day': | ||||||
|  |             # DAYOFWEEK() returns an integer, 1-7, Sunday=1. | ||||||
|  |             # Note: WEEKDAY() returns 0-6, Monday=0. | ||||||
|  |             sql = "DAYOFWEEK(%s)" % field_name | ||||||
|  |         else: | ||||||
|  |             sql = "EXTRACT(%s FROM %s)" % (lookup_type.upper(), field_name) | ||||||
|  |         return sql, params | ||||||
|  |  | ||||||
|  |     def datetime_trunc_sql(self, lookup_type, field_name, tzname): | ||||||
|  |         if settings.USE_TZ: | ||||||
|  |             field_name = "CONVERT_TZ(%s, 'UTC', %%s)" % field_name | ||||||
|  |             params = [tzname] | ||||||
|  |         else: | ||||||
|  |             params = [] | ||||||
|  |         fields = ['year', 'month', 'day', 'hour', 'minute', 'second'] | ||||||
|  |         format = ('%%Y-', '%%m', '-%%d', ' %%H:', '%%i', ':%%s')  # Use double percents to escape. | ||||||
|  |         format_def = ('0000-', '01', '-01', ' 00:', '00', ':00') | ||||||
|  |         try: | ||||||
|  |             i = fields.index(lookup_type) + 1 | ||||||
|  |         except ValueError: | ||||||
|  |             sql = field_name | ||||||
|  |         else: | ||||||
|  |             format_str = ''.join([f for f in format[:i]] + [f for f in format_def[i:]]) | ||||||
|  |             sql = "CAST(DATE_FORMAT(%s, '%s') AS DATETIME)" % (field_name, format_str) | ||||||
|  |         return sql, params | ||||||
|  |  | ||||||
|  |     def date_interval_sql(self, timedelta): | ||||||
|  |         return "INTERVAL '%d 0:0:%d:%d' DAY_MICROSECOND" % ( | ||||||
|  |             timedelta.days, timedelta.seconds, timedelta.microseconds), [] | ||||||
|  |  | ||||||
|  |     def format_for_duration_arithmetic(self, sql): | ||||||
|  |         return 'INTERVAL %s MICROSECOND' % sql | ||||||
|  |  | ||||||
|  |     def drop_foreignkey_sql(self): | ||||||
|  |         return "DROP FOREIGN KEY" | ||||||
|  |  | ||||||
|  |     def force_no_ordering(self): | ||||||
|  |         """ | ||||||
|  |         "ORDER BY NULL" prevents MySQL from implicitly ordering by grouped | ||||||
|  |         columns. If no ordering would otherwise be applied, we don't want any | ||||||
|  |         implicit sorting going on. | ||||||
|  |         """ | ||||||
|  |         return [(None, ("NULL", [], False))] | ||||||
|  |  | ||||||
|  |     def fulltext_search_sql(self, field_name): | ||||||
|  |         return 'MATCH (%s) AGAINST (%%s IN BOOLEAN MODE)' % field_name | ||||||
|  |  | ||||||
|  |     def last_executed_query(self, cursor, sql, params): | ||||||
|  |         # With MySQLdb, cursor objects have an (undocumented) "_last_executed" | ||||||
|  |         # attribute where the exact query sent to the database is saved. | ||||||
|  |         # See MySQLdb/cursors.py in the source distribution. | ||||||
|  |         return force_text(getattr(cursor, '_last_executed', None), errors='replace') | ||||||
|  |  | ||||||
|  |     def no_limit_value(self): | ||||||
|  |         # 2**64 - 1, as recommended by the MySQL documentation | ||||||
|  |         return 18446744073709551615 | ||||||
|  |  | ||||||
|  |     def quote_name(self, name): | ||||||
|  |         if name.startswith("`") and name.endswith("`"): | ||||||
|  |             return name  # Quoting once is enough. | ||||||
|  |         return "`%s`" % name | ||||||
|  |  | ||||||
|  |     def random_function_sql(self): | ||||||
|  |         return 'RAND()' | ||||||
|  |  | ||||||
|  |     def sql_flush(self, style, tables, sequences, allow_cascade=False): | ||||||
|  |         # NB: The generated SQL below is specific to MySQL | ||||||
|  |         # 'TRUNCATE x;', 'TRUNCATE y;', 'TRUNCATE z;'... style SQL statements | ||||||
|  |         # to clear all tables of all data | ||||||
|  |         if tables: | ||||||
|  |             sql = ['SET FOREIGN_KEY_CHECKS = 0;'] | ||||||
|  |             for table in tables: | ||||||
|  |                 sql.append('%s %s;' % ( | ||||||
|  |                     style.SQL_KEYWORD('TRUNCATE'), | ||||||
|  |                     style.SQL_FIELD(self.quote_name(table)), | ||||||
|  |                 )) | ||||||
|  |             sql.append('SET FOREIGN_KEY_CHECKS = 1;') | ||||||
|  |             sql.extend(self.sequence_reset_by_name_sql(style, sequences)) | ||||||
|  |             return sql | ||||||
|  |         else: | ||||||
|  |             return [] | ||||||
|  |  | ||||||
|  |     def validate_autopk_value(self, value): | ||||||
|  |         # MySQLism: zero in AUTO_INCREMENT field does not work. Refs #17653. | ||||||
|  |         if value == 0: | ||||||
|  |             raise ValueError('The database backend does not accept 0 as a ' | ||||||
|  |                              'value for AutoField.') | ||||||
|  |         return value | ||||||
|  |  | ||||||
|  |     def value_to_db_datetime(self, value): | ||||||
|  |         if value is None: | ||||||
|  |             return None | ||||||
|  |  | ||||||
|  |         # MySQL doesn't support tz-aware datetimes | ||||||
|  |         if timezone.is_aware(value): | ||||||
|  |             if settings.USE_TZ: | ||||||
|  |                 value = value.astimezone(timezone.utc).replace(tzinfo=None) | ||||||
|  |             else: | ||||||
|  |                 raise ValueError("MySQL backend does not support timezone-aware datetimes when USE_TZ is False.") | ||||||
|  |  | ||||||
|  |         return six.text_type(value) | ||||||
|  |  | ||||||
|  |     def value_to_db_time(self, value): | ||||||
|  |         if value is None: | ||||||
|  |             return None | ||||||
|  |  | ||||||
|  |         # MySQL doesn't support tz-aware times | ||||||
|  |         if timezone.is_aware(value): | ||||||
|  |             raise ValueError("MySQL backend does not support timezone-aware times.") | ||||||
|  |  | ||||||
|  |         return six.text_type(value) | ||||||
|  |  | ||||||
|  |     def max_name_length(self): | ||||||
|  |         return 64 | ||||||
|  |  | ||||||
|  |     def bulk_insert_sql(self, fields, num_values): | ||||||
|  |         items_sql = "(%s)" % ", ".join(["%s"] * len(fields)) | ||||||
|  |         return "VALUES " + ", ".join([items_sql] * num_values) | ||||||
|  |  | ||||||
|  |     def combine_expression(self, connector, sub_expressions): | ||||||
|  |         """ | ||||||
|  |         MySQL requires special cases for ^ operators in query expressions | ||||||
|  |         """ | ||||||
|  |         if connector == '^': | ||||||
|  |             return 'POW(%s)' % ','.join(sub_expressions) | ||||||
|  |         return super(DatabaseOperations, self).combine_expression(connector, sub_expressions) | ||||||
|  |  | ||||||
|  |     def get_db_converters(self, expression): | ||||||
|  |         converters = super(DatabaseOperations, self).get_db_converters(expression) | ||||||
|  |         internal_type = expression.output_field.get_internal_type() | ||||||
|  |         if internal_type in ['BooleanField', 'NullBooleanField']: | ||||||
|  |             converters.append(self.convert_booleanfield_value) | ||||||
|  |         if internal_type == 'UUIDField': | ||||||
|  |             converters.append(self.convert_uuidfield_value) | ||||||
|  |         if internal_type == 'TextField': | ||||||
|  |             converters.append(self.convert_textfield_value) | ||||||
|  |         return converters | ||||||
|  |  | ||||||
|  |     def convert_booleanfield_value(self, value, expression, context): | ||||||
|  |         if value in (0, 1): | ||||||
|  |             value = bool(value) | ||||||
|  |         return value | ||||||
|  |  | ||||||
|  |     def convert_uuidfield_value(self, value, expression, context): | ||||||
|  |         if value is not None: | ||||||
|  |             value = uuid.UUID(value) | ||||||
|  |         return value | ||||||
|  |  | ||||||
|  |     def convert_textfield_value(self, value, expression, context): | ||||||
|  |         if value is not None: | ||||||
|  |             value = force_text(value) | ||||||
|  |         return value | ||||||
| @@ -1,4 +1,4 @@ | |||||||
| from django.db.backends.schema import BaseDatabaseSchemaEditor | from django.db.backends.base.schema import BaseDatabaseSchemaEditor | ||||||
| from django.db.models import NOT_PROVIDED | from django.db.models import NOT_PROVIDED | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| from django.core import checks | from django.core import checks | ||||||
| from django.db.backends import BaseDatabaseValidation | from django.db.backends.base.validation import BaseDatabaseValidation | ||||||
|  |  | ||||||
|  |  | ||||||
| class DatabaseValidation(BaseDatabaseValidation): | class DatabaseValidation(BaseDatabaseValidation): | ||||||
|   | |||||||
| @@ -7,12 +7,20 @@ from __future__ import unicode_literals | |||||||
|  |  | ||||||
| import datetime | import datetime | ||||||
| import decimal | import decimal | ||||||
| import re | import os | ||||||
| import platform | import platform | ||||||
| import sys | import sys | ||||||
| import uuid |  | ||||||
| import warnings | import warnings | ||||||
|  |  | ||||||
|  | from django.conf import settings | ||||||
|  | from django.db import utils | ||||||
|  | from django.db.backends.base.base import BaseDatabaseWrapper | ||||||
|  | from django.db.backends.base.validation import BaseDatabaseValidation | ||||||
|  | from django.utils import six, timezone | ||||||
|  | from django.utils.duration import duration_string | ||||||
|  | from django.utils.encoding import force_bytes, force_text | ||||||
|  | from django.utils.functional import cached_property | ||||||
|  |  | ||||||
|  |  | ||||||
| def _setup_environment(environ): | def _setup_environment(environ): | ||||||
|     # Cygwin requires some special voodoo to set the environment variables |     # Cygwin requires some special voodoo to set the environment variables | ||||||
| @@ -29,7 +37,6 @@ def _setup_environment(environ): | |||||||
|         for name, value in environ: |         for name, value in environ: | ||||||
|             kernel32.SetEnvironmentVariableA(name, value) |             kernel32.SetEnvironmentVariableA(name, value) | ||||||
|     else: |     else: | ||||||
|         import os |  | ||||||
|         os.environ.update(environ) |         os.environ.update(environ) | ||||||
|  |  | ||||||
| _setup_environment([ | _setup_environment([ | ||||||
| @@ -47,520 +54,18 @@ except ImportError as e: | |||||||
|     from django.core.exceptions import ImproperlyConfigured |     from django.core.exceptions import ImproperlyConfigured | ||||||
|     raise ImproperlyConfigured("Error loading cx_Oracle module: %s" % e) |     raise ImproperlyConfigured("Error loading cx_Oracle module: %s" % e) | ||||||
|  |  | ||||||
| try: | # Some of these import cx_Oracle, so import them after checking if it's installed. | ||||||
|     import pytz | from .client import DatabaseClient | ||||||
| except ImportError: | from .creation import DatabaseCreation | ||||||
|     pytz = None | from .features import DatabaseFeatures | ||||||
|  | from .introspection import DatabaseIntrospection | ||||||
| from django.conf import settings | from .operations import DatabaseOperations | ||||||
| from django.db import utils | from .schema import DatabaseSchemaEditor | ||||||
| from django.db.backends import (BaseDatabaseFeatures, BaseDatabaseOperations, | from .utils import convert_unicode, Oracle_datetime | ||||||
|     BaseDatabaseWrapper, BaseDatabaseValidation, utils as backend_utils) |  | ||||||
| from django.db.backends.oracle.client import DatabaseClient |  | ||||||
| from django.db.backends.oracle.creation import DatabaseCreation |  | ||||||
| from django.db.backends.oracle.introspection import DatabaseIntrospection |  | ||||||
| from django.db.backends.oracle.schema import DatabaseSchemaEditor |  | ||||||
| from django.db.utils import InterfaceError |  | ||||||
| from django.utils import six, timezone |  | ||||||
| from django.utils.duration import duration_string |  | ||||||
| from django.utils.encoding import force_bytes, force_text |  | ||||||
| from django.utils.functional import cached_property |  | ||||||
|  |  | ||||||
|  |  | ||||||
| DatabaseError = Database.DatabaseError | DatabaseError = Database.DatabaseError | ||||||
| IntegrityError = Database.IntegrityError | IntegrityError = Database.IntegrityError | ||||||
|  |  | ||||||
| # Check whether cx_Oracle was compiled with the WITH_UNICODE option if cx_Oracle is pre-5.1. This will |  | ||||||
| # also be True for cx_Oracle 5.1 and in Python 3.0. See #19606 |  | ||||||
| if int(Database.version.split('.', 1)[0]) >= 5 and \ |  | ||||||
|         (int(Database.version.split('.', 2)[1]) >= 1 or |  | ||||||
|          not hasattr(Database, 'UNICODE')): |  | ||||||
|     convert_unicode = force_text |  | ||||||
| else: |  | ||||||
|     convert_unicode = force_bytes |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Oracle_datetime(datetime.datetime): |  | ||||||
|     """ |  | ||||||
|     A datetime object, with an additional class attribute |  | ||||||
|     to tell cx_Oracle to save the microseconds too. |  | ||||||
|     """ |  | ||||||
|     input_size = Database.TIMESTAMP |  | ||||||
|  |  | ||||||
|     @classmethod |  | ||||||
|     def from_datetime(cls, dt): |  | ||||||
|         return Oracle_datetime(dt.year, dt.month, dt.day, |  | ||||||
|                                dt.hour, dt.minute, dt.second, dt.microsecond) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class DatabaseFeatures(BaseDatabaseFeatures): |  | ||||||
|     empty_fetchmany_value = () |  | ||||||
|     needs_datetime_string_cast = False |  | ||||||
|     interprets_empty_strings_as_nulls = True |  | ||||||
|     uses_savepoints = True |  | ||||||
|     has_select_for_update = True |  | ||||||
|     has_select_for_update_nowait = True |  | ||||||
|     can_return_id_from_insert = True |  | ||||||
|     allow_sliced_subqueries = False |  | ||||||
|     supports_subqueries_in_group_by = False |  | ||||||
|     supports_transactions = True |  | ||||||
|     supports_timezones = False |  | ||||||
|     has_zoneinfo_database = pytz is not None |  | ||||||
|     supports_bitwise_or = False |  | ||||||
|     has_native_duration_field = True |  | ||||||
|     can_defer_constraint_checks = True |  | ||||||
|     supports_partially_nullable_unique_constraints = False |  | ||||||
|     truncates_names = True |  | ||||||
|     has_bulk_insert = True |  | ||||||
|     supports_tablespaces = True |  | ||||||
|     supports_sequence_reset = False |  | ||||||
|     can_introspect_max_length = False |  | ||||||
|     can_introspect_time_field = False |  | ||||||
|     atomic_transactions = False |  | ||||||
|     supports_combined_alters = False |  | ||||||
|     nulls_order_largest = True |  | ||||||
|     requires_literal_defaults = True |  | ||||||
|     connection_persists_old_columns = True |  | ||||||
|     closed_cursor_error_class = InterfaceError |  | ||||||
|     bare_select_suffix = " FROM DUAL" |  | ||||||
|     uppercases_column_names = True |  | ||||||
|     # select for update with limit can be achieved on Oracle, but not with the current backend. |  | ||||||
|     supports_select_for_update_with_limit = False |  | ||||||
|  |  | ||||||
|     def introspected_boolean_field_type(self, field=None, created_separately=False): |  | ||||||
|         """ |  | ||||||
|         Some versions of Oracle -- we've seen this on 11.2.0.1 and suspect |  | ||||||
|         it goes back -- have a weird bug where, when an integer column is |  | ||||||
|         added to an existing table with a default, its precision is later |  | ||||||
|         reported on introspection as 0, regardless of the real precision. |  | ||||||
|         For Django introspection, this means that such columns are reported |  | ||||||
|         as IntegerField even if they are really BigIntegerField or BooleanField. |  | ||||||
|  |  | ||||||
|         The bug is solved in Oracle 11.2.0.2 and up. |  | ||||||
|         """ |  | ||||||
|         if self.connection.oracle_full_version < '11.2.0.2' and field and field.has_default() and created_separately: |  | ||||||
|             return 'IntegerField' |  | ||||||
|         return super(DatabaseFeatures, self).introspected_boolean_field_type(field, created_separately) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class DatabaseOperations(BaseDatabaseOperations): |  | ||||||
|     compiler_module = "django.db.backends.oracle.compiler" |  | ||||||
|  |  | ||||||
|     # Oracle uses NUMBER(11) and NUMBER(19) for integer fields. |  | ||||||
|     integer_field_ranges = { |  | ||||||
|         'SmallIntegerField': (-99999999999, 99999999999), |  | ||||||
|         'IntegerField': (-99999999999, 99999999999), |  | ||||||
|         'BigIntegerField': (-9999999999999999999, 9999999999999999999), |  | ||||||
|         'PositiveSmallIntegerField': (0, 99999999999), |  | ||||||
|         'PositiveIntegerField': (0, 99999999999), |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     def autoinc_sql(self, table, column): |  | ||||||
|         # To simulate auto-incrementing primary keys in Oracle, we have to |  | ||||||
|         # create a sequence and a trigger. |  | ||||||
|         sq_name = self._get_sequence_name(table) |  | ||||||
|         tr_name = self._get_trigger_name(table) |  | ||||||
|         tbl_name = self.quote_name(table) |  | ||||||
|         col_name = self.quote_name(column) |  | ||||||
|         sequence_sql = """ |  | ||||||
| DECLARE |  | ||||||
|     i INTEGER; |  | ||||||
| BEGIN |  | ||||||
|     SELECT COUNT(*) INTO i FROM USER_CATALOG |  | ||||||
|         WHERE TABLE_NAME = '%(sq_name)s' AND TABLE_TYPE = 'SEQUENCE'; |  | ||||||
|     IF i = 0 THEN |  | ||||||
|         EXECUTE IMMEDIATE 'CREATE SEQUENCE "%(sq_name)s"'; |  | ||||||
|     END IF; |  | ||||||
| END; |  | ||||||
| /""" % locals() |  | ||||||
|         trigger_sql = """ |  | ||||||
| CREATE OR REPLACE TRIGGER "%(tr_name)s" |  | ||||||
| BEFORE INSERT ON %(tbl_name)s |  | ||||||
| FOR EACH ROW |  | ||||||
| WHEN (new.%(col_name)s IS NULL) |  | ||||||
|     BEGIN |  | ||||||
|         SELECT "%(sq_name)s".nextval |  | ||||||
|         INTO :new.%(col_name)s FROM dual; |  | ||||||
|     END; |  | ||||||
| /""" % locals() |  | ||||||
|         return sequence_sql, trigger_sql |  | ||||||
|  |  | ||||||
|     def cache_key_culling_sql(self): |  | ||||||
|         return """ |  | ||||||
|             SELECT cache_key |  | ||||||
|               FROM (SELECT cache_key, rank() OVER (ORDER BY cache_key) AS rank FROM %s) |  | ||||||
|              WHERE rank = %%s + 1 |  | ||||||
|         """ |  | ||||||
|  |  | ||||||
|     def date_extract_sql(self, lookup_type, field_name): |  | ||||||
|         if lookup_type == 'week_day': |  | ||||||
|             # TO_CHAR(field, 'D') returns an integer from 1-7, where 1=Sunday. |  | ||||||
|             return "TO_CHAR(%s, 'D')" % field_name |  | ||||||
|         else: |  | ||||||
|             # http://docs.oracle.com/cd/B19306_01/server.102/b14200/functions050.htm |  | ||||||
|             return "EXTRACT(%s FROM %s)" % (lookup_type.upper(), field_name) |  | ||||||
|  |  | ||||||
|     def date_interval_sql(self, timedelta): |  | ||||||
|         """ |  | ||||||
|         Implements the interval functionality for expressions |  | ||||||
|         format for Oracle: |  | ||||||
|         INTERVAL '3 00:03:20.000000' DAY(1) TO SECOND(6) |  | ||||||
|         """ |  | ||||||
|         minutes, seconds = divmod(timedelta.seconds, 60) |  | ||||||
|         hours, minutes = divmod(minutes, 60) |  | ||||||
|         days = str(timedelta.days) |  | ||||||
|         day_precision = len(days) |  | ||||||
|         fmt = "INTERVAL '%s %02d:%02d:%02d.%06d' DAY(%d) TO SECOND(6)" |  | ||||||
|         return fmt % (days, hours, minutes, seconds, timedelta.microseconds, |  | ||||||
|                 day_precision), [] |  | ||||||
|  |  | ||||||
|     def date_trunc_sql(self, lookup_type, field_name): |  | ||||||
|         # http://docs.oracle.com/cd/B19306_01/server.102/b14200/functions230.htm#i1002084 |  | ||||||
|         if lookup_type in ('year', 'month'): |  | ||||||
|             return "TRUNC(%s, '%s')" % (field_name, lookup_type.upper()) |  | ||||||
|         else: |  | ||||||
|             return "TRUNC(%s)" % field_name |  | ||||||
|  |  | ||||||
|     # Oracle crashes with "ORA-03113: end-of-file on communication channel" |  | ||||||
|     # if the time zone name is passed in parameter. Use interpolation instead. |  | ||||||
|     # https://groups.google.com/forum/#!msg/django-developers/zwQju7hbG78/9l934yelwfsJ |  | ||||||
|     # This regexp matches all time zone names from the zoneinfo database. |  | ||||||
|     _tzname_re = re.compile(r'^[\w/:+-]+$') |  | ||||||
|  |  | ||||||
|     def _convert_field_to_tz(self, field_name, tzname): |  | ||||||
|         if not self._tzname_re.match(tzname): |  | ||||||
|             raise ValueError("Invalid time zone name: %s" % tzname) |  | ||||||
|         # Convert from UTC to local time, returning TIMESTAMP WITH TIME ZONE. |  | ||||||
|         result = "(FROM_TZ(%s, '0:00') AT TIME ZONE '%s')" % (field_name, tzname) |  | ||||||
|         # Extracting from a TIMESTAMP WITH TIME ZONE ignore the time zone. |  | ||||||
|         # Convert to a DATETIME, which is called DATE by Oracle. There's no |  | ||||||
|         # built-in function to do that; the easiest is to go through a string. |  | ||||||
|         result = "TO_CHAR(%s, 'YYYY-MM-DD HH24:MI:SS')" % result |  | ||||||
|         result = "TO_DATE(%s, 'YYYY-MM-DD HH24:MI:SS')" % result |  | ||||||
|         # Re-convert to a TIMESTAMP because EXTRACT only handles the date part |  | ||||||
|         # on DATE values, even though they actually store the time part. |  | ||||||
|         return "CAST(%s AS TIMESTAMP)" % result |  | ||||||
|  |  | ||||||
|     def datetime_extract_sql(self, lookup_type, field_name, tzname): |  | ||||||
|         if settings.USE_TZ: |  | ||||||
|             field_name = self._convert_field_to_tz(field_name, tzname) |  | ||||||
|         if lookup_type == 'week_day': |  | ||||||
|             # TO_CHAR(field, 'D') returns an integer from 1-7, where 1=Sunday. |  | ||||||
|             sql = "TO_CHAR(%s, 'D')" % field_name |  | ||||||
|         else: |  | ||||||
|             # http://docs.oracle.com/cd/B19306_01/server.102/b14200/functions050.htm |  | ||||||
|             sql = "EXTRACT(%s FROM %s)" % (lookup_type.upper(), field_name) |  | ||||||
|         return sql, [] |  | ||||||
|  |  | ||||||
|     def datetime_trunc_sql(self, lookup_type, field_name, tzname): |  | ||||||
|         if settings.USE_TZ: |  | ||||||
|             field_name = self._convert_field_to_tz(field_name, tzname) |  | ||||||
|         # http://docs.oracle.com/cd/B19306_01/server.102/b14200/functions230.htm#i1002084 |  | ||||||
|         if lookup_type in ('year', 'month'): |  | ||||||
|             sql = "TRUNC(%s, '%s')" % (field_name, lookup_type.upper()) |  | ||||||
|         elif lookup_type == 'day': |  | ||||||
|             sql = "TRUNC(%s)" % field_name |  | ||||||
|         elif lookup_type == 'hour': |  | ||||||
|             sql = "TRUNC(%s, 'HH24')" % field_name |  | ||||||
|         elif lookup_type == 'minute': |  | ||||||
|             sql = "TRUNC(%s, 'MI')" % field_name |  | ||||||
|         else: |  | ||||||
|             sql = field_name    # Cast to DATE removes sub-second precision. |  | ||||||
|         return sql, [] |  | ||||||
|  |  | ||||||
|     def get_db_converters(self, expression): |  | ||||||
|         converters = super(DatabaseOperations, self).get_db_converters(expression) |  | ||||||
|         internal_type = expression.output_field.get_internal_type() |  | ||||||
|         if internal_type == 'TextField': |  | ||||||
|             converters.append(self.convert_textfield_value) |  | ||||||
|         elif internal_type == 'BinaryField': |  | ||||||
|             converters.append(self.convert_binaryfield_value) |  | ||||||
|         elif internal_type in ['BooleanField', 'NullBooleanField']: |  | ||||||
|             converters.append(self.convert_booleanfield_value) |  | ||||||
|         elif internal_type == 'DateField': |  | ||||||
|             converters.append(self.convert_datefield_value) |  | ||||||
|         elif internal_type == 'TimeField': |  | ||||||
|             converters.append(self.convert_timefield_value) |  | ||||||
|         elif internal_type == 'UUIDField': |  | ||||||
|             converters.append(self.convert_uuidfield_value) |  | ||||||
|         converters.append(self.convert_empty_values) |  | ||||||
|         return converters |  | ||||||
|  |  | ||||||
|     def convert_empty_values(self, value, expression, context): |  | ||||||
|         # Oracle stores empty strings as null. We need to undo this in |  | ||||||
|         # order to adhere to the Django convention of using the empty |  | ||||||
|         # string instead of null, but only if the field accepts the |  | ||||||
|         # empty string. |  | ||||||
|         field = expression.output_field |  | ||||||
|         if value is None and field.empty_strings_allowed: |  | ||||||
|             value = '' |  | ||||||
|             if field.get_internal_type() == 'BinaryField': |  | ||||||
|                 value = b'' |  | ||||||
|         return value |  | ||||||
|  |  | ||||||
|     def convert_textfield_value(self, value, expression, context): |  | ||||||
|         if isinstance(value, Database.LOB): |  | ||||||
|             value = force_text(value.read()) |  | ||||||
|         return value |  | ||||||
|  |  | ||||||
|     def convert_binaryfield_value(self, value, expression, context): |  | ||||||
|         if isinstance(value, Database.LOB): |  | ||||||
|             value = force_bytes(value.read()) |  | ||||||
|         return value |  | ||||||
|  |  | ||||||
|     def convert_booleanfield_value(self, value, expression, context): |  | ||||||
|         if value in (1, 0): |  | ||||||
|             value = bool(value) |  | ||||||
|         return value |  | ||||||
|  |  | ||||||
|     # cx_Oracle always returns datetime.datetime objects for |  | ||||||
|     # DATE and TIMESTAMP columns, but Django wants to see a |  | ||||||
|     # python datetime.date, .time, or .datetime. |  | ||||||
|     def convert_datefield_value(self, value, expression, context): |  | ||||||
|         if isinstance(value, Database.Timestamp): |  | ||||||
|             return value.date() |  | ||||||
|  |  | ||||||
|     def convert_timefield_value(self, value, expression, context): |  | ||||||
|         if isinstance(value, Database.Timestamp): |  | ||||||
|             value = value.time() |  | ||||||
|         return value |  | ||||||
|  |  | ||||||
|     def convert_uuidfield_value(self, value, expression, context): |  | ||||||
|         if value is not None: |  | ||||||
|             value = uuid.UUID(value) |  | ||||||
|         return value |  | ||||||
|  |  | ||||||
|     def deferrable_sql(self): |  | ||||||
|         return " DEFERRABLE INITIALLY DEFERRED" |  | ||||||
|  |  | ||||||
|     def drop_sequence_sql(self, table): |  | ||||||
|         return "DROP SEQUENCE %s;" % self.quote_name(self._get_sequence_name(table)) |  | ||||||
|  |  | ||||||
|     def fetch_returned_insert_id(self, cursor): |  | ||||||
|         return int(cursor._insert_id_var.getvalue()) |  | ||||||
|  |  | ||||||
|     def field_cast_sql(self, db_type, internal_type): |  | ||||||
|         if db_type and db_type.endswith('LOB'): |  | ||||||
|             return "DBMS_LOB.SUBSTR(%s)" |  | ||||||
|         else: |  | ||||||
|             return "%s" |  | ||||||
|  |  | ||||||
|     def last_executed_query(self, cursor, sql, params): |  | ||||||
|         # http://cx-oracle.sourceforge.net/html/cursor.html#Cursor.statement |  | ||||||
|         # The DB API definition does not define this attribute. |  | ||||||
|         statement = cursor.statement |  | ||||||
|         if statement and six.PY2 and not isinstance(statement, unicode): |  | ||||||
|             statement = statement.decode('utf-8') |  | ||||||
|         # Unlike Psycopg's `query` and MySQLdb`'s `_last_executed`, CxOracle's |  | ||||||
|         # `statement` doesn't contain the query parameters. refs #20010. |  | ||||||
|         return super(DatabaseOperations, self).last_executed_query(cursor, statement, params) |  | ||||||
|  |  | ||||||
|     def last_insert_id(self, cursor, table_name, pk_name): |  | ||||||
|         sq_name = self._get_sequence_name(table_name) |  | ||||||
|         cursor.execute('SELECT "%s".currval FROM dual' % sq_name) |  | ||||||
|         return cursor.fetchone()[0] |  | ||||||
|  |  | ||||||
|     def lookup_cast(self, lookup_type): |  | ||||||
|         if lookup_type in ('iexact', 'icontains', 'istartswith', 'iendswith'): |  | ||||||
|             return "UPPER(%s)" |  | ||||||
|         return "%s" |  | ||||||
|  |  | ||||||
|     def max_in_list_size(self): |  | ||||||
|         return 1000 |  | ||||||
|  |  | ||||||
|     def max_name_length(self): |  | ||||||
|         return 30 |  | ||||||
|  |  | ||||||
|     def prep_for_iexact_query(self, x): |  | ||||||
|         return x |  | ||||||
|  |  | ||||||
|     def process_clob(self, value): |  | ||||||
|         if value is None: |  | ||||||
|             return '' |  | ||||||
|         return force_text(value.read()) |  | ||||||
|  |  | ||||||
|     def quote_name(self, name): |  | ||||||
|         # SQL92 requires delimited (quoted) names to be case-sensitive.  When |  | ||||||
|         # not quoted, Oracle has case-insensitive behavior for identifiers, but |  | ||||||
|         # always defaults to uppercase. |  | ||||||
|         # We simplify things by making Oracle identifiers always uppercase. |  | ||||||
|         if not name.startswith('"') and not name.endswith('"'): |  | ||||||
|             name = '"%s"' % backend_utils.truncate_name(name.upper(), |  | ||||||
|                                                self.max_name_length()) |  | ||||||
|         # Oracle puts the query text into a (query % args) construct, so % signs |  | ||||||
|         # in names need to be escaped. The '%%' will be collapsed back to '%' at |  | ||||||
|         # that stage so we aren't really making the name longer here. |  | ||||||
|         name = name.replace('%', '%%') |  | ||||||
|         return name.upper() |  | ||||||
|  |  | ||||||
|     def random_function_sql(self): |  | ||||||
|         return "DBMS_RANDOM.RANDOM" |  | ||||||
|  |  | ||||||
|     def regex_lookup(self, lookup_type): |  | ||||||
|         if lookup_type == 'regex': |  | ||||||
|             match_option = "'c'" |  | ||||||
|         else: |  | ||||||
|             match_option = "'i'" |  | ||||||
|         return 'REGEXP_LIKE(%%s, %%s, %s)' % match_option |  | ||||||
|  |  | ||||||
|     def return_insert_id(self): |  | ||||||
|         return "RETURNING %s INTO %%s", (InsertIdVar(),) |  | ||||||
|  |  | ||||||
|     def savepoint_create_sql(self, sid): |  | ||||||
|         return convert_unicode("SAVEPOINT " + self.quote_name(sid)) |  | ||||||
|  |  | ||||||
|     def savepoint_rollback_sql(self, sid): |  | ||||||
|         return convert_unicode("ROLLBACK TO SAVEPOINT " + self.quote_name(sid)) |  | ||||||
|  |  | ||||||
|     def sql_flush(self, style, tables, sequences, allow_cascade=False): |  | ||||||
|         # Return a list of 'TRUNCATE x;', 'TRUNCATE y;', |  | ||||||
|         # 'TRUNCATE z;'... style SQL statements |  | ||||||
|         if tables: |  | ||||||
|             # Oracle does support TRUNCATE, but it seems to get us into |  | ||||||
|             # FK referential trouble, whereas DELETE FROM table works. |  | ||||||
|             sql = ['%s %s %s;' % ( |  | ||||||
|                 style.SQL_KEYWORD('DELETE'), |  | ||||||
|                 style.SQL_KEYWORD('FROM'), |  | ||||||
|                 style.SQL_FIELD(self.quote_name(table)) |  | ||||||
|             ) for table in tables] |  | ||||||
|             # Since we've just deleted all the rows, running our sequence |  | ||||||
|             # ALTER code will reset the sequence to 0. |  | ||||||
|             sql.extend(self.sequence_reset_by_name_sql(style, sequences)) |  | ||||||
|             return sql |  | ||||||
|         else: |  | ||||||
|             return [] |  | ||||||
|  |  | ||||||
|     def sequence_reset_by_name_sql(self, style, sequences): |  | ||||||
|         sql = [] |  | ||||||
|         for sequence_info in sequences: |  | ||||||
|             sequence_name = self._get_sequence_name(sequence_info['table']) |  | ||||||
|             table_name = self.quote_name(sequence_info['table']) |  | ||||||
|             column_name = self.quote_name(sequence_info['column'] or 'id') |  | ||||||
|             query = _get_sequence_reset_sql() % { |  | ||||||
|                 'sequence': sequence_name, |  | ||||||
|                 'table': table_name, |  | ||||||
|                 'column': column_name, |  | ||||||
|             } |  | ||||||
|             sql.append(query) |  | ||||||
|         return sql |  | ||||||
|  |  | ||||||
|     def sequence_reset_sql(self, style, model_list): |  | ||||||
|         from django.db import models |  | ||||||
|         output = [] |  | ||||||
|         query = _get_sequence_reset_sql() |  | ||||||
|         for model in model_list: |  | ||||||
|             for f in model._meta.local_fields: |  | ||||||
|                 if isinstance(f, models.AutoField): |  | ||||||
|                     table_name = self.quote_name(model._meta.db_table) |  | ||||||
|                     sequence_name = self._get_sequence_name(model._meta.db_table) |  | ||||||
|                     column_name = self.quote_name(f.column) |  | ||||||
|                     output.append(query % {'sequence': sequence_name, |  | ||||||
|                                            'table': table_name, |  | ||||||
|                                            'column': column_name}) |  | ||||||
|                     # Only one AutoField is allowed per model, so don't |  | ||||||
|                     # continue to loop |  | ||||||
|                     break |  | ||||||
|             for f in model._meta.many_to_many: |  | ||||||
|                 if not f.rel.through: |  | ||||||
|                     table_name = self.quote_name(f.m2m_db_table()) |  | ||||||
|                     sequence_name = self._get_sequence_name(f.m2m_db_table()) |  | ||||||
|                     column_name = self.quote_name('id') |  | ||||||
|                     output.append(query % {'sequence': sequence_name, |  | ||||||
|                                            'table': table_name, |  | ||||||
|                                            'column': column_name}) |  | ||||||
|         return output |  | ||||||
|  |  | ||||||
|     def start_transaction_sql(self): |  | ||||||
|         return '' |  | ||||||
|  |  | ||||||
|     def tablespace_sql(self, tablespace, inline=False): |  | ||||||
|         if inline: |  | ||||||
|             return "USING INDEX TABLESPACE %s" % self.quote_name(tablespace) |  | ||||||
|         else: |  | ||||||
|             return "TABLESPACE %s" % self.quote_name(tablespace) |  | ||||||
|  |  | ||||||
|     def value_to_db_date(self, value): |  | ||||||
|         """ |  | ||||||
|         Transform a date value to an object compatible with what is expected |  | ||||||
|         by the backend driver for date columns. |  | ||||||
|         The default implementation transforms the date to text, but that is not |  | ||||||
|         necessary for Oracle. |  | ||||||
|         """ |  | ||||||
|         return value |  | ||||||
|  |  | ||||||
|     def value_to_db_datetime(self, value): |  | ||||||
|         """ |  | ||||||
|         Transform a datetime value to an object compatible with what is expected |  | ||||||
|         by the backend driver for datetime columns. |  | ||||||
|  |  | ||||||
|         If naive datetime is passed assumes that is in UTC. Normally Django |  | ||||||
|         models.DateTimeField makes sure that if USE_TZ is True passed datetime |  | ||||||
|         is timezone aware. |  | ||||||
|         """ |  | ||||||
|  |  | ||||||
|         if value is None: |  | ||||||
|             return None |  | ||||||
|  |  | ||||||
|         # cx_Oracle doesn't support tz-aware datetimes |  | ||||||
|         if timezone.is_aware(value): |  | ||||||
|             if settings.USE_TZ: |  | ||||||
|                 value = value.astimezone(timezone.utc).replace(tzinfo=None) |  | ||||||
|             else: |  | ||||||
|                 raise ValueError("Oracle backend does not support timezone-aware datetimes when USE_TZ is False.") |  | ||||||
|  |  | ||||||
|         return Oracle_datetime.from_datetime(value) |  | ||||||
|  |  | ||||||
|     def value_to_db_time(self, value): |  | ||||||
|         if value is None: |  | ||||||
|             return None |  | ||||||
|  |  | ||||||
|         if isinstance(value, six.string_types): |  | ||||||
|             return datetime.datetime.strptime(value, '%H:%M:%S') |  | ||||||
|  |  | ||||||
|         # Oracle doesn't support tz-aware times |  | ||||||
|         if timezone.is_aware(value): |  | ||||||
|             raise ValueError("Oracle backend does not support timezone-aware times.") |  | ||||||
|  |  | ||||||
|         return Oracle_datetime(1900, 1, 1, value.hour, value.minute, |  | ||||||
|                                value.second, value.microsecond) |  | ||||||
|  |  | ||||||
|     def year_lookup_bounds_for_date_field(self, value): |  | ||||||
|         # Create bounds as real date values |  | ||||||
|         first = datetime.date(value, 1, 1) |  | ||||||
|         last = datetime.date(value, 12, 31) |  | ||||||
|         return [first, last] |  | ||||||
|  |  | ||||||
|     def year_lookup_bounds_for_datetime_field(self, value): |  | ||||||
|         # cx_Oracle doesn't support tz-aware datetimes |  | ||||||
|         bounds = super(DatabaseOperations, self).year_lookup_bounds_for_datetime_field(value) |  | ||||||
|         if settings.USE_TZ: |  | ||||||
|             bounds = [b.astimezone(timezone.utc) for b in bounds] |  | ||||||
|         return [Oracle_datetime.from_datetime(b) for b in bounds] |  | ||||||
|  |  | ||||||
|     def combine_expression(self, connector, sub_expressions): |  | ||||||
|         "Oracle requires special cases for %% and & operators in query expressions" |  | ||||||
|         if connector == '%%': |  | ||||||
|             return 'MOD(%s)' % ','.join(sub_expressions) |  | ||||||
|         elif connector == '&': |  | ||||||
|             return 'BITAND(%s)' % ','.join(sub_expressions) |  | ||||||
|         elif connector == '|': |  | ||||||
|             raise NotImplementedError("Bit-wise or is not supported in Oracle.") |  | ||||||
|         elif connector == '^': |  | ||||||
|             return 'POWER(%s)' % ','.join(sub_expressions) |  | ||||||
|         return super(DatabaseOperations, self).combine_expression(connector, sub_expressions) |  | ||||||
|  |  | ||||||
|     def _get_sequence_name(self, table): |  | ||||||
|         name_length = self.max_name_length() - 3 |  | ||||||
|         return '%s_SQ' % backend_utils.truncate_name(table, name_length).upper() |  | ||||||
|  |  | ||||||
|     def _get_trigger_name(self, table): |  | ||||||
|         name_length = self.max_name_length() - 3 |  | ||||||
|         return '%s_TR' % backend_utils.truncate_name(table, name_length).upper() |  | ||||||
|  |  | ||||||
|     def bulk_insert_sql(self, fields, num_values): |  | ||||||
|         items_sql = "SELECT %s FROM DUAL" % ", ".join(["%s"] * len(fields)) |  | ||||||
|         return " UNION ALL ".join([items_sql] * num_values) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class _UninitializedOperatorsDescriptor(object): | class _UninitializedOperatorsDescriptor(object): | ||||||
|  |  | ||||||
| @@ -897,19 +402,6 @@ class VariableWrapper(object): | |||||||
|             setattr(self.var, key, value) |             setattr(self.var, key, value) | ||||||
|  |  | ||||||
|  |  | ||||||
| class InsertIdVar(object): |  | ||||||
|     """ |  | ||||||
|     A late-binding cursor variable that can be passed to Cursor.execute |  | ||||||
|     as a parameter, in order to receive the id of the row created by an |  | ||||||
|     insert statement. |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     def bind_parameter(self, cursor): |  | ||||||
|         param = cursor.cursor.var(Database.NUMBER) |  | ||||||
|         cursor._insert_id_var = param |  | ||||||
|         return param |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class FormatStylePlaceholderCursor(object): | class FormatStylePlaceholderCursor(object): | ||||||
|     """ |     """ | ||||||
|     Django uses "format" (e.g. '%s') style placeholders, but Oracle uses ":var" |     Django uses "format" (e.g. '%s') style placeholders, but Oracle uses ":var" | ||||||
| @@ -1117,20 +609,3 @@ def to_unicode(s): | |||||||
|     if isinstance(s, six.string_types): |     if isinstance(s, six.string_types): | ||||||
|         return force_text(s) |         return force_text(s) | ||||||
|     return s |     return s | ||||||
|  |  | ||||||
|  |  | ||||||
| def _get_sequence_reset_sql(): |  | ||||||
|     # TODO: colorize this SQL code with style.SQL_KEYWORD(), etc. |  | ||||||
|     return """ |  | ||||||
| DECLARE |  | ||||||
|     table_value integer; |  | ||||||
|     seq_value integer; |  | ||||||
| BEGIN |  | ||||||
|     SELECT NVL(MAX(%(column)s), 0) INTO table_value FROM %(table)s; |  | ||||||
|     SELECT NVL(last_number - cache_size, 0) INTO seq_value FROM user_sequences |  | ||||||
|            WHERE sequence_name = '%(sequence)s'; |  | ||||||
|     WHILE table_value > seq_value LOOP |  | ||||||
|         SELECT "%(sequence)s".nextval INTO seq_value FROM dual; |  | ||||||
|     END LOOP; |  | ||||||
| END; |  | ||||||
| /""" |  | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| import subprocess | import subprocess | ||||||
|  |  | ||||||
| from django.db.backends import BaseDatabaseClient | from django.db.backends.base.client import BaseDatabaseClient | ||||||
|  |  | ||||||
|  |  | ||||||
| class DatabaseClient(BaseDatabaseClient): | class DatabaseClient(BaseDatabaseClient): | ||||||
|   | |||||||
| @@ -2,7 +2,7 @@ import sys | |||||||
| import time | import time | ||||||
|  |  | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| from django.db.backends.creation import BaseDatabaseCreation | from django.db.backends.base.creation import BaseDatabaseCreation | ||||||
| from django.db.utils import DatabaseError | from django.db.utils import DatabaseError | ||||||
| from django.utils.six.moves import input | from django.utils.six.moves import input | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										57
									
								
								django/db/backends/oracle/features.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										57
									
								
								django/db/backends/oracle/features.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,57 @@ | |||||||
|  | from django.db.backends.base.features import BaseDatabaseFeatures | ||||||
|  | from django.db.utils import InterfaceError | ||||||
|  |  | ||||||
|  | try: | ||||||
|  |     import pytz | ||||||
|  | except ImportError: | ||||||
|  |     pytz = None | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class DatabaseFeatures(BaseDatabaseFeatures): | ||||||
|  |     empty_fetchmany_value = () | ||||||
|  |     needs_datetime_string_cast = False | ||||||
|  |     interprets_empty_strings_as_nulls = True | ||||||
|  |     uses_savepoints = True | ||||||
|  |     has_select_for_update = True | ||||||
|  |     has_select_for_update_nowait = True | ||||||
|  |     can_return_id_from_insert = True | ||||||
|  |     allow_sliced_subqueries = False | ||||||
|  |     supports_subqueries_in_group_by = False | ||||||
|  |     supports_transactions = True | ||||||
|  |     supports_timezones = False | ||||||
|  |     has_zoneinfo_database = pytz is not None | ||||||
|  |     supports_bitwise_or = False | ||||||
|  |     has_native_duration_field = True | ||||||
|  |     can_defer_constraint_checks = True | ||||||
|  |     supports_partially_nullable_unique_constraints = False | ||||||
|  |     truncates_names = True | ||||||
|  |     has_bulk_insert = True | ||||||
|  |     supports_tablespaces = True | ||||||
|  |     supports_sequence_reset = False | ||||||
|  |     can_introspect_max_length = False | ||||||
|  |     can_introspect_time_field = False | ||||||
|  |     atomic_transactions = False | ||||||
|  |     supports_combined_alters = False | ||||||
|  |     nulls_order_largest = True | ||||||
|  |     requires_literal_defaults = True | ||||||
|  |     connection_persists_old_columns = True | ||||||
|  |     closed_cursor_error_class = InterfaceError | ||||||
|  |     bare_select_suffix = " FROM DUAL" | ||||||
|  |     uppercases_column_names = True | ||||||
|  |     # select for update with limit can be achieved on Oracle, but not with the current backend. | ||||||
|  |     supports_select_for_update_with_limit = False | ||||||
|  |  | ||||||
|  |     def introspected_boolean_field_type(self, field=None, created_separately=False): | ||||||
|  |         """ | ||||||
|  |         Some versions of Oracle -- we've seen this on 11.2.0.1 and suspect | ||||||
|  |         it goes back -- have a weird bug where, when an integer column is | ||||||
|  |         added to an existing table with a default, its precision is later | ||||||
|  |         reported on introspection as 0, regardless of the real precision. | ||||||
|  |         For Django introspection, this means that such columns are reported | ||||||
|  |         as IntegerField even if they are really BigIntegerField or BooleanField. | ||||||
|  |  | ||||||
|  |         The bug is solved in Oracle 11.2.0.2 and up. | ||||||
|  |         """ | ||||||
|  |         if self.connection.oracle_full_version < '11.2.0.2' and field and field.has_default() and created_separately: | ||||||
|  |             return 'IntegerField' | ||||||
|  |         return super(DatabaseFeatures, self).introspected_boolean_field_type(field, created_separately) | ||||||
| @@ -2,7 +2,9 @@ import re | |||||||
|  |  | ||||||
| import cx_Oracle | import cx_Oracle | ||||||
|  |  | ||||||
| from django.db.backends import BaseDatabaseIntrospection, FieldInfo, TableInfo | from django.db.backends.base.introspection import ( | ||||||
|  |     BaseDatabaseIntrospection, FieldInfo, TableInfo, | ||||||
|  | ) | ||||||
| from django.utils.encoding import force_text | from django.utils.encoding import force_text | ||||||
|  |  | ||||||
| foreign_key_re = re.compile(r"\sCONSTRAINT `[^`]*` FOREIGN KEY \(`([^`]*)`\) REFERENCES `([^`]*)` \(`([^`]*)`\)") | foreign_key_re = re.compile(r"\sCONSTRAINT `[^`]*` FOREIGN KEY \(`([^`]*)`\) REFERENCES `([^`]*)` \(`([^`]*)`\)") | ||||||
|   | |||||||
							
								
								
									
										447
									
								
								django/db/backends/oracle/operations.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										447
									
								
								django/db/backends/oracle/operations.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,447 @@ | |||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | import datetime | ||||||
|  | import re | ||||||
|  | import uuid | ||||||
|  |  | ||||||
|  | from django.conf import settings | ||||||
|  | from django.db.backends.base.operations import BaseDatabaseOperations | ||||||
|  | from django.db.backends.utils import truncate_name | ||||||
|  | from django.utils import six, timezone | ||||||
|  | from django.utils.encoding import force_bytes, force_text | ||||||
|  |  | ||||||
|  | from .base import Database | ||||||
|  | from .utils import convert_unicode, InsertIdVar, Oracle_datetime | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class DatabaseOperations(BaseDatabaseOperations): | ||||||
|  |     compiler_module = "django.db.backends.oracle.compiler" | ||||||
|  |  | ||||||
|  |     # Oracle uses NUMBER(11) and NUMBER(19) for integer fields. | ||||||
|  |     integer_field_ranges = { | ||||||
|  |         'SmallIntegerField': (-99999999999, 99999999999), | ||||||
|  |         'IntegerField': (-99999999999, 99999999999), | ||||||
|  |         'BigIntegerField': (-9999999999999999999, 9999999999999999999), | ||||||
|  |         'PositiveSmallIntegerField': (0, 99999999999), | ||||||
|  |         'PositiveIntegerField': (0, 99999999999), | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     # TODO: colorize this SQL code with style.SQL_KEYWORD(), etc. | ||||||
|  |     _sequence_reset_sql = """ | ||||||
|  | DECLARE | ||||||
|  |     table_value integer; | ||||||
|  |     seq_value integer; | ||||||
|  | BEGIN | ||||||
|  |     SELECT NVL(MAX(%(column)s), 0) INTO table_value FROM %(table)s; | ||||||
|  |     SELECT NVL(last_number - cache_size, 0) INTO seq_value FROM user_sequences | ||||||
|  |            WHERE sequence_name = '%(sequence)s'; | ||||||
|  |     WHILE table_value > seq_value LOOP | ||||||
|  |         SELECT "%(sequence)s".nextval INTO seq_value FROM dual; | ||||||
|  |     END LOOP; | ||||||
|  | END; | ||||||
|  | /""" | ||||||
|  |  | ||||||
|  |     def autoinc_sql(self, table, column): | ||||||
|  |         # To simulate auto-incrementing primary keys in Oracle, we have to | ||||||
|  |         # create a sequence and a trigger. | ||||||
|  |         sq_name = self._get_sequence_name(table) | ||||||
|  |         tr_name = self._get_trigger_name(table) | ||||||
|  |         tbl_name = self.quote_name(table) | ||||||
|  |         col_name = self.quote_name(column) | ||||||
|  |         sequence_sql = """ | ||||||
|  | DECLARE | ||||||
|  |     i INTEGER; | ||||||
|  | BEGIN | ||||||
|  |     SELECT COUNT(*) INTO i FROM USER_CATALOG | ||||||
|  |         WHERE TABLE_NAME = '%(sq_name)s' AND TABLE_TYPE = 'SEQUENCE'; | ||||||
|  |     IF i = 0 THEN | ||||||
|  |         EXECUTE IMMEDIATE 'CREATE SEQUENCE "%(sq_name)s"'; | ||||||
|  |     END IF; | ||||||
|  | END; | ||||||
|  | /""" % locals() | ||||||
|  |         trigger_sql = """ | ||||||
|  | CREATE OR REPLACE TRIGGER "%(tr_name)s" | ||||||
|  | BEFORE INSERT ON %(tbl_name)s | ||||||
|  | FOR EACH ROW | ||||||
|  | WHEN (new.%(col_name)s IS NULL) | ||||||
|  |     BEGIN | ||||||
|  |         SELECT "%(sq_name)s".nextval | ||||||
|  |         INTO :new.%(col_name)s FROM dual; | ||||||
|  |     END; | ||||||
|  | /""" % locals() | ||||||
|  |         return sequence_sql, trigger_sql | ||||||
|  |  | ||||||
|  |     def cache_key_culling_sql(self): | ||||||
|  |         return """ | ||||||
|  |             SELECT cache_key | ||||||
|  |               FROM (SELECT cache_key, rank() OVER (ORDER BY cache_key) AS rank FROM %s) | ||||||
|  |              WHERE rank = %%s + 1 | ||||||
|  |         """ | ||||||
|  |  | ||||||
|  |     def date_extract_sql(self, lookup_type, field_name): | ||||||
|  |         if lookup_type == 'week_day': | ||||||
|  |             # TO_CHAR(field, 'D') returns an integer from 1-7, where 1=Sunday. | ||||||
|  |             return "TO_CHAR(%s, 'D')" % field_name | ||||||
|  |         else: | ||||||
|  |             # http://docs.oracle.com/cd/B19306_01/server.102/b14200/functions050.htm | ||||||
|  |             return "EXTRACT(%s FROM %s)" % (lookup_type.upper(), field_name) | ||||||
|  |  | ||||||
|  |     def date_interval_sql(self, timedelta): | ||||||
|  |         """ | ||||||
|  |         Implements the interval functionality for expressions | ||||||
|  |         format for Oracle: | ||||||
|  |         INTERVAL '3 00:03:20.000000' DAY(1) TO SECOND(6) | ||||||
|  |         """ | ||||||
|  |         minutes, seconds = divmod(timedelta.seconds, 60) | ||||||
|  |         hours, minutes = divmod(minutes, 60) | ||||||
|  |         days = str(timedelta.days) | ||||||
|  |         day_precision = len(days) | ||||||
|  |         fmt = "INTERVAL '%s %02d:%02d:%02d.%06d' DAY(%d) TO SECOND(6)" | ||||||
|  |         return fmt % (days, hours, minutes, seconds, timedelta.microseconds, | ||||||
|  |                 day_precision), [] | ||||||
|  |  | ||||||
|  |     def date_trunc_sql(self, lookup_type, field_name): | ||||||
|  |         # http://docs.oracle.com/cd/B19306_01/server.102/b14200/functions230.htm#i1002084 | ||||||
|  |         if lookup_type in ('year', 'month'): | ||||||
|  |             return "TRUNC(%s, '%s')" % (field_name, lookup_type.upper()) | ||||||
|  |         else: | ||||||
|  |             return "TRUNC(%s)" % field_name | ||||||
|  |  | ||||||
|  |     # Oracle crashes with "ORA-03113: end-of-file on communication channel" | ||||||
|  |     # if the time zone name is passed in parameter. Use interpolation instead. | ||||||
|  |     # https://groups.google.com/forum/#!msg/django-developers/zwQju7hbG78/9l934yelwfsJ | ||||||
|  |     # This regexp matches all time zone names from the zoneinfo database. | ||||||
|  |     _tzname_re = re.compile(r'^[\w/:+-]+$') | ||||||
|  |  | ||||||
|  |     def _convert_field_to_tz(self, field_name, tzname): | ||||||
|  |         if not self._tzname_re.match(tzname): | ||||||
|  |             raise ValueError("Invalid time zone name: %s" % tzname) | ||||||
|  |         # Convert from UTC to local time, returning TIMESTAMP WITH TIME ZONE. | ||||||
|  |         result = "(FROM_TZ(%s, '0:00') AT TIME ZONE '%s')" % (field_name, tzname) | ||||||
|  |         # Extracting from a TIMESTAMP WITH TIME ZONE ignore the time zone. | ||||||
|  |         # Convert to a DATETIME, which is called DATE by Oracle. There's no | ||||||
|  |         # built-in function to do that; the easiest is to go through a string. | ||||||
|  |         result = "TO_CHAR(%s, 'YYYY-MM-DD HH24:MI:SS')" % result | ||||||
|  |         result = "TO_DATE(%s, 'YYYY-MM-DD HH24:MI:SS')" % result | ||||||
|  |         # Re-convert to a TIMESTAMP because EXTRACT only handles the date part | ||||||
|  |         # on DATE values, even though they actually store the time part. | ||||||
|  |         return "CAST(%s AS TIMESTAMP)" % result | ||||||
|  |  | ||||||
|  |     def datetime_extract_sql(self, lookup_type, field_name, tzname): | ||||||
|  |         if settings.USE_TZ: | ||||||
|  |             field_name = self._convert_field_to_tz(field_name, tzname) | ||||||
|  |         if lookup_type == 'week_day': | ||||||
|  |             # TO_CHAR(field, 'D') returns an integer from 1-7, where 1=Sunday. | ||||||
|  |             sql = "TO_CHAR(%s, 'D')" % field_name | ||||||
|  |         else: | ||||||
|  |             # http://docs.oracle.com/cd/B19306_01/server.102/b14200/functions050.htm | ||||||
|  |             sql = "EXTRACT(%s FROM %s)" % (lookup_type.upper(), field_name) | ||||||
|  |         return sql, [] | ||||||
|  |  | ||||||
|  |     def datetime_trunc_sql(self, lookup_type, field_name, tzname): | ||||||
|  |         if settings.USE_TZ: | ||||||
|  |             field_name = self._convert_field_to_tz(field_name, tzname) | ||||||
|  |         # http://docs.oracle.com/cd/B19306_01/server.102/b14200/functions230.htm#i1002084 | ||||||
|  |         if lookup_type in ('year', 'month'): | ||||||
|  |             sql = "TRUNC(%s, '%s')" % (field_name, lookup_type.upper()) | ||||||
|  |         elif lookup_type == 'day': | ||||||
|  |             sql = "TRUNC(%s)" % field_name | ||||||
|  |         elif lookup_type == 'hour': | ||||||
|  |             sql = "TRUNC(%s, 'HH24')" % field_name | ||||||
|  |         elif lookup_type == 'minute': | ||||||
|  |             sql = "TRUNC(%s, 'MI')" % field_name | ||||||
|  |         else: | ||||||
|  |             sql = field_name    # Cast to DATE removes sub-second precision. | ||||||
|  |         return sql, [] | ||||||
|  |  | ||||||
|  |     def get_db_converters(self, expression): | ||||||
|  |         converters = super(DatabaseOperations, self).get_db_converters(expression) | ||||||
|  |         internal_type = expression.output_field.get_internal_type() | ||||||
|  |         if internal_type == 'TextField': | ||||||
|  |             converters.append(self.convert_textfield_value) | ||||||
|  |         elif internal_type == 'BinaryField': | ||||||
|  |             converters.append(self.convert_binaryfield_value) | ||||||
|  |         elif internal_type in ['BooleanField', 'NullBooleanField']: | ||||||
|  |             converters.append(self.convert_booleanfield_value) | ||||||
|  |         elif internal_type == 'DateField': | ||||||
|  |             converters.append(self.convert_datefield_value) | ||||||
|  |         elif internal_type == 'TimeField': | ||||||
|  |             converters.append(self.convert_timefield_value) | ||||||
|  |         elif internal_type == 'UUIDField': | ||||||
|  |             converters.append(self.convert_uuidfield_value) | ||||||
|  |         converters.append(self.convert_empty_values) | ||||||
|  |         return converters | ||||||
|  |  | ||||||
|  |     def convert_empty_values(self, value, expression, context): | ||||||
|  |         # Oracle stores empty strings as null. We need to undo this in | ||||||
|  |         # order to adhere to the Django convention of using the empty | ||||||
|  |         # string instead of null, but only if the field accepts the | ||||||
|  |         # empty string. | ||||||
|  |         field = expression.output_field | ||||||
|  |         if value is None and field.empty_strings_allowed: | ||||||
|  |             value = '' | ||||||
|  |             if field.get_internal_type() == 'BinaryField': | ||||||
|  |                 value = b'' | ||||||
|  |         return value | ||||||
|  |  | ||||||
|  |     def convert_textfield_value(self, value, expression, context): | ||||||
|  |         if isinstance(value, Database.LOB): | ||||||
|  |             value = force_text(value.read()) | ||||||
|  |         return value | ||||||
|  |  | ||||||
|  |     def convert_binaryfield_value(self, value, expression, context): | ||||||
|  |         if isinstance(value, Database.LOB): | ||||||
|  |             value = force_bytes(value.read()) | ||||||
|  |         return value | ||||||
|  |  | ||||||
|  |     def convert_booleanfield_value(self, value, expression, context): | ||||||
|  |         if value in (1, 0): | ||||||
|  |             value = bool(value) | ||||||
|  |         return value | ||||||
|  |  | ||||||
|  |     # cx_Oracle always returns datetime.datetime objects for | ||||||
|  |     # DATE and TIMESTAMP columns, but Django wants to see a | ||||||
|  |     # python datetime.date, .time, or .datetime. | ||||||
|  |     def convert_datefield_value(self, value, expression, context): | ||||||
|  |         if isinstance(value, Database.Timestamp): | ||||||
|  |             return value.date() | ||||||
|  |  | ||||||
|  |     def convert_timefield_value(self, value, expression, context): | ||||||
|  |         if isinstance(value, Database.Timestamp): | ||||||
|  |             value = value.time() | ||||||
|  |         return value | ||||||
|  |  | ||||||
|  |     def convert_uuidfield_value(self, value, expression, context): | ||||||
|  |         if value is not None: | ||||||
|  |             value = uuid.UUID(value) | ||||||
|  |         return value | ||||||
|  |  | ||||||
|  |     def deferrable_sql(self): | ||||||
|  |         return " DEFERRABLE INITIALLY DEFERRED" | ||||||
|  |  | ||||||
|  |     def drop_sequence_sql(self, table): | ||||||
|  |         return "DROP SEQUENCE %s;" % self.quote_name(self._get_sequence_name(table)) | ||||||
|  |  | ||||||
|  |     def fetch_returned_insert_id(self, cursor): | ||||||
|  |         return int(cursor._insert_id_var.getvalue()) | ||||||
|  |  | ||||||
|  |     def field_cast_sql(self, db_type, internal_type): | ||||||
|  |         if db_type and db_type.endswith('LOB'): | ||||||
|  |             return "DBMS_LOB.SUBSTR(%s)" | ||||||
|  |         else: | ||||||
|  |             return "%s" | ||||||
|  |  | ||||||
|  |     def last_executed_query(self, cursor, sql, params): | ||||||
|  |         # http://cx-oracle.sourceforge.net/html/cursor.html#Cursor.statement | ||||||
|  |         # The DB API definition does not define this attribute. | ||||||
|  |         statement = cursor.statement | ||||||
|  |         if statement and six.PY2 and not isinstance(statement, unicode): | ||||||
|  |             statement = statement.decode('utf-8') | ||||||
|  |         # Unlike Psycopg's `query` and MySQLdb`'s `_last_executed`, CxOracle's | ||||||
|  |         # `statement` doesn't contain the query parameters. refs #20010. | ||||||
|  |         return super(DatabaseOperations, self).last_executed_query(cursor, statement, params) | ||||||
|  |  | ||||||
|  |     def last_insert_id(self, cursor, table_name, pk_name): | ||||||
|  |         sq_name = self._get_sequence_name(table_name) | ||||||
|  |         cursor.execute('SELECT "%s".currval FROM dual' % sq_name) | ||||||
|  |         return cursor.fetchone()[0] | ||||||
|  |  | ||||||
|  |     def lookup_cast(self, lookup_type): | ||||||
|  |         if lookup_type in ('iexact', 'icontains', 'istartswith', 'iendswith'): | ||||||
|  |             return "UPPER(%s)" | ||||||
|  |         return "%s" | ||||||
|  |  | ||||||
|  |     def max_in_list_size(self): | ||||||
|  |         return 1000 | ||||||
|  |  | ||||||
|  |     def max_name_length(self): | ||||||
|  |         return 30 | ||||||
|  |  | ||||||
|  |     def prep_for_iexact_query(self, x): | ||||||
|  |         return x | ||||||
|  |  | ||||||
|  |     def process_clob(self, value): | ||||||
|  |         if value is None: | ||||||
|  |             return '' | ||||||
|  |         return force_text(value.read()) | ||||||
|  |  | ||||||
|  |     def quote_name(self, name): | ||||||
|  |         # SQL92 requires delimited (quoted) names to be case-sensitive.  When | ||||||
|  |         # not quoted, Oracle has case-insensitive behavior for identifiers, but | ||||||
|  |         # always defaults to uppercase. | ||||||
|  |         # We simplify things by making Oracle identifiers always uppercase. | ||||||
|  |         if not name.startswith('"') and not name.endswith('"'): | ||||||
|  |             name = '"%s"' % truncate_name(name.upper(), self.max_name_length()) | ||||||
|  |         # Oracle puts the query text into a (query % args) construct, so % signs | ||||||
|  |         # in names need to be escaped. The '%%' will be collapsed back to '%' at | ||||||
|  |         # that stage so we aren't really making the name longer here. | ||||||
|  |         name = name.replace('%', '%%') | ||||||
|  |         return name.upper() | ||||||
|  |  | ||||||
|  |     def random_function_sql(self): | ||||||
|  |         return "DBMS_RANDOM.RANDOM" | ||||||
|  |  | ||||||
|  |     def regex_lookup(self, lookup_type): | ||||||
|  |         if lookup_type == 'regex': | ||||||
|  |             match_option = "'c'" | ||||||
|  |         else: | ||||||
|  |             match_option = "'i'" | ||||||
|  |         return 'REGEXP_LIKE(%%s, %%s, %s)' % match_option | ||||||
|  |  | ||||||
|  |     def return_insert_id(self): | ||||||
|  |         return "RETURNING %s INTO %%s", (InsertIdVar(),) | ||||||
|  |  | ||||||
|  |     def savepoint_create_sql(self, sid): | ||||||
|  |         return convert_unicode("SAVEPOINT " + self.quote_name(sid)) | ||||||
|  |  | ||||||
|  |     def savepoint_rollback_sql(self, sid): | ||||||
|  |         return convert_unicode("ROLLBACK TO SAVEPOINT " + self.quote_name(sid)) | ||||||
|  |  | ||||||
|  |     def sql_flush(self, style, tables, sequences, allow_cascade=False): | ||||||
|  |         # Return a list of 'TRUNCATE x;', 'TRUNCATE y;', | ||||||
|  |         # 'TRUNCATE z;'... style SQL statements | ||||||
|  |         if tables: | ||||||
|  |             # Oracle does support TRUNCATE, but it seems to get us into | ||||||
|  |             # FK referential trouble, whereas DELETE FROM table works. | ||||||
|  |             sql = ['%s %s %s;' % ( | ||||||
|  |                 style.SQL_KEYWORD('DELETE'), | ||||||
|  |                 style.SQL_KEYWORD('FROM'), | ||||||
|  |                 style.SQL_FIELD(self.quote_name(table)) | ||||||
|  |             ) for table in tables] | ||||||
|  |             # Since we've just deleted all the rows, running our sequence | ||||||
|  |             # ALTER code will reset the sequence to 0. | ||||||
|  |             sql.extend(self.sequence_reset_by_name_sql(style, sequences)) | ||||||
|  |             return sql | ||||||
|  |         else: | ||||||
|  |             return [] | ||||||
|  |  | ||||||
|  |     def sequence_reset_by_name_sql(self, style, sequences): | ||||||
|  |         sql = [] | ||||||
|  |         for sequence_info in sequences: | ||||||
|  |             sequence_name = self._get_sequence_name(sequence_info['table']) | ||||||
|  |             table_name = self.quote_name(sequence_info['table']) | ||||||
|  |             column_name = self.quote_name(sequence_info['column'] or 'id') | ||||||
|  |             query = self._sequence_reset_sql % { | ||||||
|  |                 'sequence': sequence_name, | ||||||
|  |                 'table': table_name, | ||||||
|  |                 'column': column_name, | ||||||
|  |             } | ||||||
|  |             sql.append(query) | ||||||
|  |         return sql | ||||||
|  |  | ||||||
|  |     def sequence_reset_sql(self, style, model_list): | ||||||
|  |         from django.db import models | ||||||
|  |         output = [] | ||||||
|  |         query = self._sequence_reset_sql | ||||||
|  |         for model in model_list: | ||||||
|  |             for f in model._meta.local_fields: | ||||||
|  |                 if isinstance(f, models.AutoField): | ||||||
|  |                     table_name = self.quote_name(model._meta.db_table) | ||||||
|  |                     sequence_name = self._get_sequence_name(model._meta.db_table) | ||||||
|  |                     column_name = self.quote_name(f.column) | ||||||
|  |                     output.append(query % {'sequence': sequence_name, | ||||||
|  |                                            'table': table_name, | ||||||
|  |                                            'column': column_name}) | ||||||
|  |                     # Only one AutoField is allowed per model, so don't | ||||||
|  |                     # continue to loop | ||||||
|  |                     break | ||||||
|  |             for f in model._meta.many_to_many: | ||||||
|  |                 if not f.rel.through: | ||||||
|  |                     table_name = self.quote_name(f.m2m_db_table()) | ||||||
|  |                     sequence_name = self._get_sequence_name(f.m2m_db_table()) | ||||||
|  |                     column_name = self.quote_name('id') | ||||||
|  |                     output.append(query % {'sequence': sequence_name, | ||||||
|  |                                            'table': table_name, | ||||||
|  |                                            'column': column_name}) | ||||||
|  |         return output | ||||||
|  |  | ||||||
|  |     def start_transaction_sql(self): | ||||||
|  |         return '' | ||||||
|  |  | ||||||
|  |     def tablespace_sql(self, tablespace, inline=False): | ||||||
|  |         if inline: | ||||||
|  |             return "USING INDEX TABLESPACE %s" % self.quote_name(tablespace) | ||||||
|  |         else: | ||||||
|  |             return "TABLESPACE %s" % self.quote_name(tablespace) | ||||||
|  |  | ||||||
|  |     def value_to_db_date(self, value): | ||||||
|  |         """ | ||||||
|  |         Transform a date value to an object compatible with what is expected | ||||||
|  |         by the backend driver for date columns. | ||||||
|  |         The default implementation transforms the date to text, but that is not | ||||||
|  |         necessary for Oracle. | ||||||
|  |         """ | ||||||
|  |         return value | ||||||
|  |  | ||||||
|  |     def value_to_db_datetime(self, value): | ||||||
|  |         """ | ||||||
|  |         Transform a datetime value to an object compatible with what is expected | ||||||
|  |         by the backend driver for datetime columns. | ||||||
|  |  | ||||||
|  |         If naive datetime is passed assumes that is in UTC. Normally Django | ||||||
|  |         models.DateTimeField makes sure that if USE_TZ is True passed datetime | ||||||
|  |         is timezone aware. | ||||||
|  |         """ | ||||||
|  |  | ||||||
|  |         if value is None: | ||||||
|  |             return None | ||||||
|  |  | ||||||
|  |         # cx_Oracle doesn't support tz-aware datetimes | ||||||
|  |         if timezone.is_aware(value): | ||||||
|  |             if settings.USE_TZ: | ||||||
|  |                 value = value.astimezone(timezone.utc).replace(tzinfo=None) | ||||||
|  |             else: | ||||||
|  |                 raise ValueError("Oracle backend does not support timezone-aware datetimes when USE_TZ is False.") | ||||||
|  |  | ||||||
|  |         return Oracle_datetime.from_datetime(value) | ||||||
|  |  | ||||||
|  |     def value_to_db_time(self, value): | ||||||
|  |         if value is None: | ||||||
|  |             return None | ||||||
|  |  | ||||||
|  |         if isinstance(value, six.string_types): | ||||||
|  |             return datetime.datetime.strptime(value, '%H:%M:%S') | ||||||
|  |  | ||||||
|  |         # Oracle doesn't support tz-aware times | ||||||
|  |         if timezone.is_aware(value): | ||||||
|  |             raise ValueError("Oracle backend does not support timezone-aware times.") | ||||||
|  |  | ||||||
|  |         return Oracle_datetime(1900, 1, 1, value.hour, value.minute, | ||||||
|  |                                value.second, value.microsecond) | ||||||
|  |  | ||||||
|  |     def year_lookup_bounds_for_date_field(self, value): | ||||||
|  |         # Create bounds as real date values | ||||||
|  |         first = datetime.date(value, 1, 1) | ||||||
|  |         last = datetime.date(value, 12, 31) | ||||||
|  |         return [first, last] | ||||||
|  |  | ||||||
|  |     def year_lookup_bounds_for_datetime_field(self, value): | ||||||
|  |         # cx_Oracle doesn't support tz-aware datetimes | ||||||
|  |         bounds = super(DatabaseOperations, self).year_lookup_bounds_for_datetime_field(value) | ||||||
|  |         if settings.USE_TZ: | ||||||
|  |             bounds = [b.astimezone(timezone.utc) for b in bounds] | ||||||
|  |         return [Oracle_datetime.from_datetime(b) for b in bounds] | ||||||
|  |  | ||||||
|  |     def combine_expression(self, connector, sub_expressions): | ||||||
|  |         "Oracle requires special cases for %% and & operators in query expressions" | ||||||
|  |         if connector == '%%': | ||||||
|  |             return 'MOD(%s)' % ','.join(sub_expressions) | ||||||
|  |         elif connector == '&': | ||||||
|  |             return 'BITAND(%s)' % ','.join(sub_expressions) | ||||||
|  |         elif connector == '|': | ||||||
|  |             raise NotImplementedError("Bit-wise or is not supported in Oracle.") | ||||||
|  |         elif connector == '^': | ||||||
|  |             return 'POWER(%s)' % ','.join(sub_expressions) | ||||||
|  |         return super(DatabaseOperations, self).combine_expression(connector, sub_expressions) | ||||||
|  |  | ||||||
|  |     def _get_sequence_name(self, table): | ||||||
|  |         name_length = self.max_name_length() - 3 | ||||||
|  |         return '%s_SQ' % truncate_name(table, name_length).upper() | ||||||
|  |  | ||||||
|  |     def _get_trigger_name(self, table): | ||||||
|  |         name_length = self.max_name_length() - 3 | ||||||
|  |         return '%s_TR' % truncate_name(table, name_length).upper() | ||||||
|  |  | ||||||
|  |     def bulk_insert_sql(self, fields, num_values): | ||||||
|  |         items_sql = "SELECT %s FROM DUAL" % ", ".join(["%s"] * len(fields)) | ||||||
|  |         return " UNION ALL ".join([items_sql] * num_values) | ||||||
| @@ -4,7 +4,7 @@ import binascii | |||||||
|  |  | ||||||
| from django.utils import six | from django.utils import six | ||||||
| from django.utils.text import force_text | from django.utils.text import force_text | ||||||
| from django.db.backends.schema import BaseDatabaseSchemaEditor | from django.db.backends.base.schema import BaseDatabaseSchemaEditor | ||||||
| from django.db.utils import DatabaseError | from django.db.utils import DatabaseError | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										42
									
								
								django/db/backends/oracle/utils.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								django/db/backends/oracle/utils.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,42 @@ | |||||||
|  | import datetime | ||||||
|  |  | ||||||
|  | from django.utils.encoding import force_bytes, force_text | ||||||
|  |  | ||||||
|  | from .base import Database | ||||||
|  |  | ||||||
|  | # Check whether cx_Oracle was compiled with the WITH_UNICODE option if cx_Oracle is pre-5.1. This will | ||||||
|  | # also be True for cx_Oracle 5.1 and in Python 3.0. See #19606 | ||||||
|  | if int(Database.version.split('.', 1)[0]) >= 5 and \ | ||||||
|  |         (int(Database.version.split('.', 2)[1]) >= 1 or | ||||||
|  |          not hasattr(Database, 'UNICODE')): | ||||||
|  |     convert_unicode = force_text | ||||||
|  | else: | ||||||
|  |     convert_unicode = force_bytes | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class InsertIdVar(object): | ||||||
|  |     """ | ||||||
|  |     A late-binding cursor variable that can be passed to Cursor.execute | ||||||
|  |     as a parameter, in order to receive the id of the row created by an | ||||||
|  |     insert statement. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     def bind_parameter(self, cursor): | ||||||
|  |         param = cursor.cursor.var(Database.NUMBER) | ||||||
|  |         cursor._insert_id_var = param | ||||||
|  |         return param | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Oracle_datetime(datetime.datetime): | ||||||
|  |     """ | ||||||
|  |     A datetime object, with an additional class attribute | ||||||
|  |     to tell cx_Oracle to save the microseconds too. | ||||||
|  |     """ | ||||||
|  |     input_size = Database.TIMESTAMP | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def from_datetime(cls, dt): | ||||||
|  |         return Oracle_datetime( | ||||||
|  |             dt.year, dt.month, dt.day, | ||||||
|  |             dt.hour, dt.minute, dt.second, dt.microsecond, | ||||||
|  |         ) | ||||||
| @@ -5,19 +5,11 @@ Requires psycopg 2: http://initd.org/projects/psycopg2 | |||||||
| """ | """ | ||||||
|  |  | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| from django.db.backends import (BaseDatabaseFeatures, BaseDatabaseWrapper, | from django.db.backends.base.base import BaseDatabaseWrapper | ||||||
|     BaseDatabaseValidation) | from django.db.backends.base.validation import BaseDatabaseValidation | ||||||
| from django.db.backends.postgresql_psycopg2.operations import DatabaseOperations |  | ||||||
| from django.db.backends.postgresql_psycopg2.client import DatabaseClient |  | ||||||
| from django.db.backends.postgresql_psycopg2.creation import DatabaseCreation |  | ||||||
| from django.db.backends.postgresql_psycopg2.version import get_version |  | ||||||
| from django.db.backends.postgresql_psycopg2.introspection import DatabaseIntrospection |  | ||||||
| from django.db.backends.postgresql_psycopg2.schema import DatabaseSchemaEditor |  | ||||||
| from django.db.utils import InterfaceError |  | ||||||
| from django.utils.encoding import force_str | from django.utils.encoding import force_str | ||||||
| from django.utils.functional import cached_property | from django.utils.functional import cached_property | ||||||
| from django.utils.safestring import SafeText, SafeBytes | from django.utils.safestring import SafeText, SafeBytes | ||||||
| from django.utils.timezone import utc |  | ||||||
|  |  | ||||||
| try: | try: | ||||||
|     import psycopg2 as Database |     import psycopg2 as Database | ||||||
| @@ -27,6 +19,16 @@ except ImportError as e: | |||||||
|     from django.core.exceptions import ImproperlyConfigured |     from django.core.exceptions import ImproperlyConfigured | ||||||
|     raise ImproperlyConfigured("Error loading psycopg2 module: %s" % e) |     raise ImproperlyConfigured("Error loading psycopg2 module: %s" % e) | ||||||
|  |  | ||||||
|  | # Some of these import psycopg2, so import them after checking if it's installed. | ||||||
|  | from .client import DatabaseClient | ||||||
|  | from .creation import DatabaseCreation | ||||||
|  | from .features import DatabaseFeatures | ||||||
|  | from .introspection import DatabaseIntrospection | ||||||
|  | from .operations import DatabaseOperations | ||||||
|  | from .schema import DatabaseSchemaEditor | ||||||
|  | from .utils import utc_tzinfo_factory | ||||||
|  | from .version import get_version | ||||||
|  |  | ||||||
| DatabaseError = Database.DatabaseError | DatabaseError = Database.DatabaseError | ||||||
| IntegrityError = Database.IntegrityError | IntegrityError = Database.IntegrityError | ||||||
|  |  | ||||||
| @@ -37,38 +39,6 @@ psycopg2.extensions.register_adapter(SafeText, psycopg2.extensions.QuotedString) | |||||||
| psycopg2.extras.register_uuid() | psycopg2.extras.register_uuid() | ||||||
|  |  | ||||||
|  |  | ||||||
| def utc_tzinfo_factory(offset): |  | ||||||
|     if offset != 0: |  | ||||||
|         raise AssertionError("database connection isn't set to UTC") |  | ||||||
|     return utc |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class DatabaseFeatures(BaseDatabaseFeatures): |  | ||||||
|     needs_datetime_string_cast = False |  | ||||||
|     can_return_id_from_insert = True |  | ||||||
|     has_real_datatype = True |  | ||||||
|     has_native_duration_field = True |  | ||||||
|     driver_supports_timedelta_args = True |  | ||||||
|     can_defer_constraint_checks = True |  | ||||||
|     has_select_for_update = True |  | ||||||
|     has_select_for_update_nowait = True |  | ||||||
|     has_bulk_insert = True |  | ||||||
|     uses_savepoints = True |  | ||||||
|     can_release_savepoints = True |  | ||||||
|     supports_tablespaces = True |  | ||||||
|     supports_transactions = True |  | ||||||
|     can_introspect_autofield = True |  | ||||||
|     can_introspect_ip_address_field = True |  | ||||||
|     can_introspect_small_integer_field = True |  | ||||||
|     can_distinct_on_fields = True |  | ||||||
|     can_rollback_ddl = True |  | ||||||
|     supports_combined_alters = True |  | ||||||
|     nulls_order_largest = True |  | ||||||
|     closed_cursor_error_class = InterfaceError |  | ||||||
|     has_case_insensitive_like = False |  | ||||||
|     requires_sqlparse_for_splitting = False |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class DatabaseWrapper(BaseDatabaseWrapper): | class DatabaseWrapper(BaseDatabaseWrapper): | ||||||
|     vendor = 'postgresql' |     vendor = 'postgresql' | ||||||
|     # This dictionary maps Field objects to their associated PostgreSQL column |     # This dictionary maps Field objects to their associated PostgreSQL column | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| import subprocess | import subprocess | ||||||
|  |  | ||||||
| from django.db.backends import BaseDatabaseClient | from django.db.backends.base.client import BaseDatabaseClient | ||||||
|  |  | ||||||
|  |  | ||||||
| class DatabaseClient(BaseDatabaseClient): | class DatabaseClient(BaseDatabaseClient): | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| from django.db.backends.creation import BaseDatabaseCreation | from django.db.backends.base.creation import BaseDatabaseCreation | ||||||
| from django.db.backends.utils import truncate_name | from django.db.backends.utils import truncate_name | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										28
									
								
								django/db/backends/postgresql_psycopg2/features.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								django/db/backends/postgresql_psycopg2/features.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,28 @@ | |||||||
|  | from django.db.backends.base.features import BaseDatabaseFeatures | ||||||
|  | from django.db.utils import InterfaceError | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class DatabaseFeatures(BaseDatabaseFeatures): | ||||||
|  |     needs_datetime_string_cast = False | ||||||
|  |     can_return_id_from_insert = True | ||||||
|  |     has_real_datatype = True | ||||||
|  |     has_native_duration_field = True | ||||||
|  |     driver_supports_timedelta_args = True | ||||||
|  |     can_defer_constraint_checks = True | ||||||
|  |     has_select_for_update = True | ||||||
|  |     has_select_for_update_nowait = True | ||||||
|  |     has_bulk_insert = True | ||||||
|  |     uses_savepoints = True | ||||||
|  |     can_release_savepoints = True | ||||||
|  |     supports_tablespaces = True | ||||||
|  |     supports_transactions = True | ||||||
|  |     can_introspect_autofield = True | ||||||
|  |     can_introspect_ip_address_field = True | ||||||
|  |     can_introspect_small_integer_field = True | ||||||
|  |     can_distinct_on_fields = True | ||||||
|  |     can_rollback_ddl = True | ||||||
|  |     supports_combined_alters = True | ||||||
|  |     nulls_order_largest = True | ||||||
|  |     closed_cursor_error_class = InterfaceError | ||||||
|  |     has_case_insensitive_like = False | ||||||
|  |     requires_sqlparse_for_splitting = False | ||||||
| @@ -1,7 +1,9 @@ | |||||||
| from __future__ import unicode_literals | from __future__ import unicode_literals | ||||||
| from collections import namedtuple | from collections import namedtuple | ||||||
|  |  | ||||||
| from django.db.backends import BaseDatabaseIntrospection, FieldInfo, TableInfo | from django.db.backends.base.introspection import ( | ||||||
|  |     BaseDatabaseIntrospection, FieldInfo, TableInfo, | ||||||
|  | ) | ||||||
| from django.utils.encoding import force_text | from django.utils.encoding import force_text | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| from __future__ import unicode_literals | from __future__ import unicode_literals | ||||||
|  |  | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| from django.db.backends import BaseDatabaseOperations | from django.db.backends.base.operations import BaseDatabaseOperations | ||||||
|  |  | ||||||
|  |  | ||||||
| class DatabaseOperations(BaseDatabaseOperations): | class DatabaseOperations(BaseDatabaseOperations): | ||||||
|   | |||||||
| @@ -1,4 +1,6 @@ | |||||||
| from django.db.backends.schema import BaseDatabaseSchemaEditor | from django.db.backends.base.schema import BaseDatabaseSchemaEditor | ||||||
|  |  | ||||||
|  | import psycopg2 | ||||||
|  |  | ||||||
|  |  | ||||||
| class DatabaseSchemaEditor(BaseDatabaseSchemaEditor): | class DatabaseSchemaEditor(BaseDatabaseSchemaEditor): | ||||||
| @@ -10,8 +12,6 @@ class DatabaseSchemaEditor(BaseDatabaseSchemaEditor): | |||||||
|     sql_create_text_index = "CREATE INDEX %(name)s ON %(table)s (%(columns)s text_pattern_ops)%(extra)s" |     sql_create_text_index = "CREATE INDEX %(name)s ON %(table)s (%(columns)s text_pattern_ops)%(extra)s" | ||||||
|  |  | ||||||
|     def quote_value(self, value): |     def quote_value(self, value): | ||||||
|         # Inner import so backend fails nicely if it's not present |  | ||||||
|         import psycopg2 |  | ||||||
|         return psycopg2.extensions.adapt(value) |         return psycopg2.extensions.adapt(value) | ||||||
|  |  | ||||||
|     def _model_indexes_sql(self, model): |     def _model_indexes_sql(self, model): | ||||||
|   | |||||||
							
								
								
									
										7
									
								
								django/db/backends/postgresql_psycopg2/utils.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								django/db/backends/postgresql_psycopg2/utils.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | |||||||
|  | from django.utils.timezone import utc | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def utc_tzinfo_factory(offset): | ||||||
|  |     if offset != 0: | ||||||
|  |         raise AssertionError("database connection isn't set to UTC") | ||||||
|  |     return utc | ||||||
| @@ -9,26 +9,22 @@ from __future__ import unicode_literals | |||||||
| import datetime | import datetime | ||||||
| import decimal | import decimal | ||||||
| import re | import re | ||||||
| import sys |  | ||||||
| import uuid |  | ||||||
| import warnings | import warnings | ||||||
|  |  | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| from django.db import utils | from django.db import utils | ||||||
| from django.db.backends import (utils as backend_utils, BaseDatabaseFeatures, | from django.db.backends import utils as backend_utils | ||||||
|     BaseDatabaseOperations, BaseDatabaseWrapper, BaseDatabaseValidation) | from django.db.backends.base.base import BaseDatabaseWrapper | ||||||
| from django.db.backends.sqlite3.client import DatabaseClient | from django.db.backends.base.validation import BaseDatabaseValidation | ||||||
| from django.db.backends.sqlite3.creation import DatabaseCreation | from django.utils import six, timezone | ||||||
| from django.db.backends.sqlite3.introspection import DatabaseIntrospection | from django.utils.dateparse import parse_date, parse_duration, parse_time | ||||||
| from django.db.backends.sqlite3.schema import DatabaseSchemaEditor |  | ||||||
| from django.db.models import fields, aggregates |  | ||||||
| from django.utils.dateparse import parse_date, parse_datetime, parse_time, parse_duration |  | ||||||
| from django.utils.duration import duration_string |  | ||||||
| from django.utils.encoding import force_text | from django.utils.encoding import force_text | ||||||
| from django.utils.functional import cached_property |  | ||||||
| from django.utils.safestring import SafeBytes | from django.utils.safestring import SafeBytes | ||||||
| from django.utils import six |  | ||||||
| from django.utils import timezone | try: | ||||||
|  |     import pytz | ||||||
|  | except ImportError: | ||||||
|  |     pytz = None | ||||||
|  |  | ||||||
| try: | try: | ||||||
|     try: |     try: | ||||||
| @@ -39,23 +35,19 @@ except ImportError as exc: | |||||||
|     from django.core.exceptions import ImproperlyConfigured |     from django.core.exceptions import ImproperlyConfigured | ||||||
|     raise ImproperlyConfigured("Error loading either pysqlite2 or sqlite3 modules (tried in that order): %s" % exc) |     raise ImproperlyConfigured("Error loading either pysqlite2 or sqlite3 modules (tried in that order): %s" % exc) | ||||||
|  |  | ||||||
| try: | # Some of these import sqlite3, so import them after checking if it's installed. | ||||||
|     import pytz | from .client import DatabaseClient | ||||||
| except ImportError: | from .creation import DatabaseCreation | ||||||
|     pytz = None | from .features import DatabaseFeatures | ||||||
|  | from .introspection import DatabaseIntrospection | ||||||
|  | from .operations import DatabaseOperations | ||||||
|  | from .schema import DatabaseSchemaEditor | ||||||
|  | from .utils import parse_datetime_with_timezone_support | ||||||
|  |  | ||||||
| DatabaseError = Database.DatabaseError | DatabaseError = Database.DatabaseError | ||||||
| IntegrityError = Database.IntegrityError | IntegrityError = Database.IntegrityError | ||||||
|  |  | ||||||
|  |  | ||||||
| def parse_datetime_with_timezone_support(value): |  | ||||||
|     dt = parse_datetime(value) |  | ||||||
|     # Confirm that dt is naive before overwriting its tzinfo. |  | ||||||
|     if dt is not None and settings.USE_TZ and timezone.is_naive(dt): |  | ||||||
|         dt = dt.replace(tzinfo=timezone.utc) |  | ||||||
|     return dt |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def adapt_datetime_with_timezone_support(value): | def adapt_datetime_with_timezone_support(value): | ||||||
|     # Equivalent to DateTimeField.get_db_prep_value. Used only by raw SQL. |     # Equivalent to DateTimeField.get_db_prep_value. Used only by raw SQL. | ||||||
|     if settings.USE_TZ: |     if settings.USE_TZ: | ||||||
| @@ -91,250 +83,6 @@ if six.PY2: | |||||||
|     Database.register_adapter(SafeBytes, lambda s: s.decode('utf-8')) |     Database.register_adapter(SafeBytes, lambda s: s.decode('utf-8')) | ||||||
|  |  | ||||||
|  |  | ||||||
| class DatabaseFeatures(BaseDatabaseFeatures): |  | ||||||
|     # SQLite cannot handle us only partially reading from a cursor's result set |  | ||||||
|     # and then writing the same rows to the database in another cursor. This |  | ||||||
|     # setting ensures we always read result sets fully into memory all in one |  | ||||||
|     # go. |  | ||||||
|     can_use_chunked_reads = False |  | ||||||
|     test_db_allows_multiple_connections = False |  | ||||||
|     supports_unspecified_pk = True |  | ||||||
|     supports_timezones = False |  | ||||||
|     supports_1000_query_parameters = False |  | ||||||
|     supports_mixed_date_datetime_comparisons = False |  | ||||||
|     has_bulk_insert = True |  | ||||||
|     can_combine_inserts_with_and_without_auto_increment_pk = False |  | ||||||
|     supports_foreign_keys = False |  | ||||||
|     supports_column_check_constraints = False |  | ||||||
|     autocommits_when_autocommit_is_off = True |  | ||||||
|     can_introspect_decimal_field = False |  | ||||||
|     can_introspect_positive_integer_field = True |  | ||||||
|     can_introspect_small_integer_field = True |  | ||||||
|     supports_transactions = True |  | ||||||
|     atomic_transactions = False |  | ||||||
|     can_rollback_ddl = True |  | ||||||
|     supports_paramstyle_pyformat = False |  | ||||||
|     supports_sequence_reset = False |  | ||||||
|  |  | ||||||
|     @cached_property |  | ||||||
|     def uses_savepoints(self): |  | ||||||
|         return Database.sqlite_version_info >= (3, 6, 8) |  | ||||||
|  |  | ||||||
|     @cached_property |  | ||||||
|     def can_release_savepoints(self): |  | ||||||
|         return self.uses_savepoints |  | ||||||
|  |  | ||||||
|     @cached_property |  | ||||||
|     def can_share_in_memory_db(self): |  | ||||||
|         return ( |  | ||||||
|             sys.version_info[:2] >= (3, 4) and |  | ||||||
|             Database.__name__ == 'sqlite3.dbapi2' and |  | ||||||
|             Database.sqlite_version_info >= (3, 7, 13) |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|     @cached_property |  | ||||||
|     def supports_stddev(self): |  | ||||||
|         """Confirm support for STDDEV and related stats functions |  | ||||||
|  |  | ||||||
|         SQLite supports STDDEV as an extension package; so |  | ||||||
|         connection.ops.check_aggregate_support() can't unilaterally |  | ||||||
|         rule out support for STDDEV. We need to manually check |  | ||||||
|         whether the call works. |  | ||||||
|         """ |  | ||||||
|         with self.connection.cursor() as cursor: |  | ||||||
|             cursor.execute('CREATE TABLE STDDEV_TEST (X INT)') |  | ||||||
|             try: |  | ||||||
|                 cursor.execute('SELECT STDDEV(*) FROM STDDEV_TEST') |  | ||||||
|                 has_support = True |  | ||||||
|             except utils.DatabaseError: |  | ||||||
|                 has_support = False |  | ||||||
|             cursor.execute('DROP TABLE STDDEV_TEST') |  | ||||||
|         return has_support |  | ||||||
|  |  | ||||||
|     @cached_property |  | ||||||
|     def has_zoneinfo_database(self): |  | ||||||
|         return pytz is not None |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class DatabaseOperations(BaseDatabaseOperations): |  | ||||||
|     def bulk_batch_size(self, fields, objs): |  | ||||||
|         """ |  | ||||||
|         SQLite has a compile-time default (SQLITE_LIMIT_VARIABLE_NUMBER) of |  | ||||||
|         999 variables per query. |  | ||||||
|  |  | ||||||
|         If there is just single field to insert, then we can hit another |  | ||||||
|         limit, SQLITE_MAX_COMPOUND_SELECT which defaults to 500. |  | ||||||
|         """ |  | ||||||
|         limit = 999 if len(fields) > 1 else 500 |  | ||||||
|         return (limit // len(fields)) if len(fields) > 0 else len(objs) |  | ||||||
|  |  | ||||||
|     def check_aggregate_support(self, aggregate): |  | ||||||
|         bad_fields = (fields.DateField, fields.DateTimeField, fields.TimeField) |  | ||||||
|         bad_aggregates = (aggregates.Sum, aggregates.Avg, |  | ||||||
|                           aggregates.Variance, aggregates.StdDev) |  | ||||||
|         if aggregate.refs_field(bad_aggregates, bad_fields): |  | ||||||
|             raise NotImplementedError( |  | ||||||
|                 'You cannot use Sum, Avg, StdDev and Variance aggregations ' |  | ||||||
|                 'on date/time fields in sqlite3 ' |  | ||||||
|                 'since date/time is saved as text.') |  | ||||||
|  |  | ||||||
|     def date_extract_sql(self, lookup_type, field_name): |  | ||||||
|         # sqlite doesn't support extract, so we fake it with the user-defined |  | ||||||
|         # function django_date_extract that's registered in connect(). Note that |  | ||||||
|         # single quotes are used because this is a string (and could otherwise |  | ||||||
|         # cause a collision with a field name). |  | ||||||
|         return "django_date_extract('%s', %s)" % (lookup_type.lower(), field_name) |  | ||||||
|  |  | ||||||
|     def date_interval_sql(self, timedelta): |  | ||||||
|         return "'%s'" % duration_string(timedelta), [] |  | ||||||
|  |  | ||||||
|     def format_for_duration_arithmetic(self, sql): |  | ||||||
|         """Do nothing here, we will handle it in the custom function.""" |  | ||||||
|         return sql |  | ||||||
|  |  | ||||||
|     def date_trunc_sql(self, lookup_type, field_name): |  | ||||||
|         # sqlite doesn't support DATE_TRUNC, so we fake it with a user-defined |  | ||||||
|         # function django_date_trunc that's registered in connect(). Note that |  | ||||||
|         # single quotes are used because this is a string (and could otherwise |  | ||||||
|         # cause a collision with a field name). |  | ||||||
|         return "django_date_trunc('%s', %s)" % (lookup_type.lower(), field_name) |  | ||||||
|  |  | ||||||
|     def datetime_extract_sql(self, lookup_type, field_name, tzname): |  | ||||||
|         # Same comment as in date_extract_sql. |  | ||||||
|         if settings.USE_TZ: |  | ||||||
|             if pytz is None: |  | ||||||
|                 from django.core.exceptions import ImproperlyConfigured |  | ||||||
|                 raise ImproperlyConfigured("This query requires pytz, " |  | ||||||
|                                            "but it isn't installed.") |  | ||||||
|         return "django_datetime_extract('%s', %s, %%s)" % ( |  | ||||||
|             lookup_type.lower(), field_name), [tzname] |  | ||||||
|  |  | ||||||
|     def datetime_trunc_sql(self, lookup_type, field_name, tzname): |  | ||||||
|         # Same comment as in date_trunc_sql. |  | ||||||
|         if settings.USE_TZ: |  | ||||||
|             if pytz is None: |  | ||||||
|                 from django.core.exceptions import ImproperlyConfigured |  | ||||||
|                 raise ImproperlyConfigured("This query requires pytz, " |  | ||||||
|                                            "but it isn't installed.") |  | ||||||
|         return "django_datetime_trunc('%s', %s, %%s)" % ( |  | ||||||
|             lookup_type.lower(), field_name), [tzname] |  | ||||||
|  |  | ||||||
|     def drop_foreignkey_sql(self): |  | ||||||
|         return "" |  | ||||||
|  |  | ||||||
|     def pk_default_value(self): |  | ||||||
|         return "NULL" |  | ||||||
|  |  | ||||||
|     def quote_name(self, name): |  | ||||||
|         if name.startswith('"') and name.endswith('"'): |  | ||||||
|             return name  # Quoting once is enough. |  | ||||||
|         return '"%s"' % name |  | ||||||
|  |  | ||||||
|     def no_limit_value(self): |  | ||||||
|         return -1 |  | ||||||
|  |  | ||||||
|     def sql_flush(self, style, tables, sequences, allow_cascade=False): |  | ||||||
|         # NB: The generated SQL below is specific to SQLite |  | ||||||
|         # Note: The DELETE FROM... SQL generated below works for SQLite databases |  | ||||||
|         # because constraints don't exist |  | ||||||
|         sql = ['%s %s %s;' % ( |  | ||||||
|             style.SQL_KEYWORD('DELETE'), |  | ||||||
|             style.SQL_KEYWORD('FROM'), |  | ||||||
|             style.SQL_FIELD(self.quote_name(table)) |  | ||||||
|         ) for table in tables] |  | ||||||
|         # Note: No requirement for reset of auto-incremented indices (cf. other |  | ||||||
|         # sql_flush() implementations). Just return SQL at this point |  | ||||||
|         return sql |  | ||||||
|  |  | ||||||
|     def value_to_db_datetime(self, value): |  | ||||||
|         if value is None: |  | ||||||
|             return None |  | ||||||
|  |  | ||||||
|         # SQLite doesn't support tz-aware datetimes |  | ||||||
|         if timezone.is_aware(value): |  | ||||||
|             if settings.USE_TZ: |  | ||||||
|                 value = value.astimezone(timezone.utc).replace(tzinfo=None) |  | ||||||
|             else: |  | ||||||
|                 raise ValueError("SQLite backend does not support timezone-aware datetimes when USE_TZ is False.") |  | ||||||
|  |  | ||||||
|         return six.text_type(value) |  | ||||||
|  |  | ||||||
|     def value_to_db_time(self, value): |  | ||||||
|         if value is None: |  | ||||||
|             return None |  | ||||||
|  |  | ||||||
|         # SQLite doesn't support tz-aware datetimes |  | ||||||
|         if timezone.is_aware(value): |  | ||||||
|             raise ValueError("SQLite backend does not support timezone-aware times.") |  | ||||||
|  |  | ||||||
|         return six.text_type(value) |  | ||||||
|  |  | ||||||
|     def get_db_converters(self, expression): |  | ||||||
|         converters = super(DatabaseOperations, self).get_db_converters(expression) |  | ||||||
|         internal_type = expression.output_field.get_internal_type() |  | ||||||
|         if internal_type == 'DateTimeField': |  | ||||||
|             converters.append(self.convert_datetimefield_value) |  | ||||||
|         elif internal_type == 'DateField': |  | ||||||
|             converters.append(self.convert_datefield_value) |  | ||||||
|         elif internal_type == 'TimeField': |  | ||||||
|             converters.append(self.convert_timefield_value) |  | ||||||
|         elif internal_type == 'DecimalField': |  | ||||||
|             converters.append(self.convert_decimalfield_value) |  | ||||||
|         elif internal_type == 'UUIDField': |  | ||||||
|             converters.append(self.convert_uuidfield_value) |  | ||||||
|         return converters |  | ||||||
|  |  | ||||||
|     def convert_decimalfield_value(self, value, expression, context): |  | ||||||
|         return backend_utils.typecast_decimal(expression.output_field.format_number(value)) |  | ||||||
|  |  | ||||||
|     def convert_datefield_value(self, value, expression, context): |  | ||||||
|         if value is not None and not isinstance(value, datetime.date): |  | ||||||
|             value = parse_date(value) |  | ||||||
|         return value |  | ||||||
|  |  | ||||||
|     def convert_datetimefield_value(self, value, expression, context): |  | ||||||
|         if value is not None and not isinstance(value, datetime.datetime): |  | ||||||
|             value = parse_datetime_with_timezone_support(value) |  | ||||||
|         return value |  | ||||||
|  |  | ||||||
|     def convert_timefield_value(self, value, expression, context): |  | ||||||
|         if value is not None and not isinstance(value, datetime.time): |  | ||||||
|             value = parse_time(value) |  | ||||||
|         return value |  | ||||||
|  |  | ||||||
|     def convert_uuidfield_value(self, value, expression, context): |  | ||||||
|         if value is not None: |  | ||||||
|             value = uuid.UUID(value) |  | ||||||
|         return value |  | ||||||
|  |  | ||||||
|     def bulk_insert_sql(self, fields, num_values): |  | ||||||
|         res = [] |  | ||||||
|         res.append("SELECT %s" % ", ".join( |  | ||||||
|             "%%s AS %s" % self.quote_name(f.column) for f in fields |  | ||||||
|         )) |  | ||||||
|         res.extend(["UNION ALL SELECT %s" % ", ".join(["%s"] * len(fields))] * (num_values - 1)) |  | ||||||
|         return " ".join(res) |  | ||||||
|  |  | ||||||
|     def combine_expression(self, connector, sub_expressions): |  | ||||||
|         # SQLite doesn't have a power function, so we fake it with a |  | ||||||
|         # user-defined function django_power that's registered in connect(). |  | ||||||
|         if connector == '^': |  | ||||||
|             return 'django_power(%s)' % ','.join(sub_expressions) |  | ||||||
|         return super(DatabaseOperations, self).combine_expression(connector, sub_expressions) |  | ||||||
|  |  | ||||||
|     def combine_duration_expression(self, connector, sub_expressions): |  | ||||||
|         if connector not in ['+', '-']: |  | ||||||
|             raise utils.DatabaseError('Invalid connector for timedelta: %s.' % connector) |  | ||||||
|         fn_params = ["'%s'" % connector] + sub_expressions |  | ||||||
|         if len(fn_params) > 3: |  | ||||||
|             raise ValueError('Too many params for timedelta operations.') |  | ||||||
|         return "django_format_dtdelta(%s)" % ', '.join(fn_params) |  | ||||||
|  |  | ||||||
|     def integer_field_range(self, internal_type): |  | ||||||
|         # SQLite doesn't enforce any integer constraints |  | ||||||
|         return (None, None) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class DatabaseWrapper(BaseDatabaseWrapper): | class DatabaseWrapper(BaseDatabaseWrapper): | ||||||
|     vendor = 'sqlite' |     vendor = 'sqlite' | ||||||
|     # SQLite doesn't actually support most of these types, but it "does the right |     # SQLite doesn't actually support most of these types, but it "does the right | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| import subprocess | import subprocess | ||||||
|  |  | ||||||
| from django.db.backends import BaseDatabaseClient | from django.db.backends.base.client import BaseDatabaseClient | ||||||
|  |  | ||||||
|  |  | ||||||
| class DatabaseClient(BaseDatabaseClient): | class DatabaseClient(BaseDatabaseClient): | ||||||
|   | |||||||
| @@ -2,7 +2,7 @@ import os | |||||||
| import sys | import sys | ||||||
|  |  | ||||||
| from django.core.exceptions import ImproperlyConfigured | from django.core.exceptions import ImproperlyConfigured | ||||||
| from django.db.backends.creation import BaseDatabaseCreation | from django.db.backends.base.creation import BaseDatabaseCreation | ||||||
| from django.utils.six.moves import input | from django.utils.six.moves import input | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										79
									
								
								django/db/backends/sqlite3/features.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										79
									
								
								django/db/backends/sqlite3/features.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,79 @@ | |||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | import sys | ||||||
|  |  | ||||||
|  | from django.db import utils | ||||||
|  | from django.db.backends.base.features import BaseDatabaseFeatures | ||||||
|  | from django.utils.functional import cached_property | ||||||
|  |  | ||||||
|  | from .base import Database | ||||||
|  |  | ||||||
|  | try: | ||||||
|  |     import pytz | ||||||
|  | except ImportError: | ||||||
|  |     pytz = None | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class DatabaseFeatures(BaseDatabaseFeatures): | ||||||
|  |     # SQLite cannot handle us only partially reading from a cursor's result set | ||||||
|  |     # and then writing the same rows to the database in another cursor. This | ||||||
|  |     # setting ensures we always read result sets fully into memory all in one | ||||||
|  |     # go. | ||||||
|  |     can_use_chunked_reads = False | ||||||
|  |     test_db_allows_multiple_connections = False | ||||||
|  |     supports_unspecified_pk = True | ||||||
|  |     supports_timezones = False | ||||||
|  |     supports_1000_query_parameters = False | ||||||
|  |     supports_mixed_date_datetime_comparisons = False | ||||||
|  |     has_bulk_insert = True | ||||||
|  |     can_combine_inserts_with_and_without_auto_increment_pk = False | ||||||
|  |     supports_foreign_keys = False | ||||||
|  |     supports_column_check_constraints = False | ||||||
|  |     autocommits_when_autocommit_is_off = True | ||||||
|  |     can_introspect_decimal_field = False | ||||||
|  |     can_introspect_positive_integer_field = True | ||||||
|  |     can_introspect_small_integer_field = True | ||||||
|  |     supports_transactions = True | ||||||
|  |     atomic_transactions = False | ||||||
|  |     can_rollback_ddl = True | ||||||
|  |     supports_paramstyle_pyformat = False | ||||||
|  |     supports_sequence_reset = False | ||||||
|  |  | ||||||
|  |     @cached_property | ||||||
|  |     def uses_savepoints(self): | ||||||
|  |         return Database.sqlite_version_info >= (3, 6, 8) | ||||||
|  |  | ||||||
|  |     @cached_property | ||||||
|  |     def can_release_savepoints(self): | ||||||
|  |         return self.uses_savepoints | ||||||
|  |  | ||||||
|  |     @cached_property | ||||||
|  |     def can_share_in_memory_db(self): | ||||||
|  |         return ( | ||||||
|  |             sys.version_info[:2] >= (3, 4) and | ||||||
|  |             Database.__name__ == 'sqlite3.dbapi2' and | ||||||
|  |             Database.sqlite_version_info >= (3, 7, 13) | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     @cached_property | ||||||
|  |     def supports_stddev(self): | ||||||
|  |         """Confirm support for STDDEV and related stats functions | ||||||
|  |  | ||||||
|  |         SQLite supports STDDEV as an extension package; so | ||||||
|  |         connection.ops.check_aggregate_support() can't unilaterally | ||||||
|  |         rule out support for STDDEV. We need to manually check | ||||||
|  |         whether the call works. | ||||||
|  |         """ | ||||||
|  |         with self.connection.cursor() as cursor: | ||||||
|  |             cursor.execute('CREATE TABLE STDDEV_TEST (X INT)') | ||||||
|  |             try: | ||||||
|  |                 cursor.execute('SELECT STDDEV(*) FROM STDDEV_TEST') | ||||||
|  |                 has_support = True | ||||||
|  |             except utils.DatabaseError: | ||||||
|  |                 has_support = False | ||||||
|  |             cursor.execute('DROP TABLE STDDEV_TEST') | ||||||
|  |         return has_support | ||||||
|  |  | ||||||
|  |     @cached_property | ||||||
|  |     def has_zoneinfo_database(self): | ||||||
|  |         return pytz is not None | ||||||
| @@ -1,6 +1,8 @@ | |||||||
| import re | import re | ||||||
|  |  | ||||||
| from django.db.backends import BaseDatabaseIntrospection, FieldInfo, TableInfo | from django.db.backends.base.introspection import ( | ||||||
|  |     BaseDatabaseIntrospection, FieldInfo, TableInfo, | ||||||
|  | ) | ||||||
|  |  | ||||||
|  |  | ||||||
| field_size_re = re.compile(r'^\s*(?:var)?char\s*\(\s*(\d+)\s*\)\s*$') | field_size_re = re.compile(r'^\s*(?:var)?char\s*\(\s*(\d+)\s*\)\s*$') | ||||||
|   | |||||||
							
								
								
									
										198
									
								
								django/db/backends/sqlite3/operations.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										198
									
								
								django/db/backends/sqlite3/operations.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,198 @@ | |||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
|  | import datetime | ||||||
|  | import uuid | ||||||
|  |  | ||||||
|  | from django.conf import settings | ||||||
|  | from django.core.exceptions import ImproperlyConfigured | ||||||
|  | from django.db import utils | ||||||
|  | from django.db.backends import utils as backend_utils | ||||||
|  | from django.db.backends.base.operations import BaseDatabaseOperations | ||||||
|  | from django.db.models import fields, aggregates | ||||||
|  | from django.utils.dateparse import parse_date, parse_time | ||||||
|  | from django.utils.duration import duration_string | ||||||
|  | from django.utils import six, timezone | ||||||
|  |  | ||||||
|  | from .utils import parse_datetime_with_timezone_support | ||||||
|  |  | ||||||
|  | try: | ||||||
|  |     import pytz | ||||||
|  | except ImportError: | ||||||
|  |     pytz = None | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class DatabaseOperations(BaseDatabaseOperations): | ||||||
|  |     def bulk_batch_size(self, fields, objs): | ||||||
|  |         """ | ||||||
|  |         SQLite has a compile-time default (SQLITE_LIMIT_VARIABLE_NUMBER) of | ||||||
|  |         999 variables per query. | ||||||
|  |  | ||||||
|  |         If there is just single field to insert, then we can hit another | ||||||
|  |         limit, SQLITE_MAX_COMPOUND_SELECT which defaults to 500. | ||||||
|  |         """ | ||||||
|  |         limit = 999 if len(fields) > 1 else 500 | ||||||
|  |         return (limit // len(fields)) if len(fields) > 0 else len(objs) | ||||||
|  |  | ||||||
|  |     def check_aggregate_support(self, aggregate): | ||||||
|  |         bad_fields = (fields.DateField, fields.DateTimeField, fields.TimeField) | ||||||
|  |         bad_aggregates = (aggregates.Sum, aggregates.Avg, | ||||||
|  |                           aggregates.Variance, aggregates.StdDev) | ||||||
|  |         if aggregate.refs_field(bad_aggregates, bad_fields): | ||||||
|  |             raise NotImplementedError( | ||||||
|  |                 'You cannot use Sum, Avg, StdDev and Variance aggregations ' | ||||||
|  |                 'on date/time fields in sqlite3 ' | ||||||
|  |                 'since date/time is saved as text.') | ||||||
|  |  | ||||||
|  |     def date_extract_sql(self, lookup_type, field_name): | ||||||
|  |         # sqlite doesn't support extract, so we fake it with the user-defined | ||||||
|  |         # function django_date_extract that's registered in connect(). Note that | ||||||
|  |         # single quotes are used because this is a string (and could otherwise | ||||||
|  |         # cause a collision with a field name). | ||||||
|  |         return "django_date_extract('%s', %s)" % (lookup_type.lower(), field_name) | ||||||
|  |  | ||||||
|  |     def date_interval_sql(self, timedelta): | ||||||
|  |         return "'%s'" % duration_string(timedelta), [] | ||||||
|  |  | ||||||
|  |     def format_for_duration_arithmetic(self, sql): | ||||||
|  |         """Do nothing here, we will handle it in the custom function.""" | ||||||
|  |         return sql | ||||||
|  |  | ||||||
|  |     def date_trunc_sql(self, lookup_type, field_name): | ||||||
|  |         # sqlite doesn't support DATE_TRUNC, so we fake it with a user-defined | ||||||
|  |         # function django_date_trunc that's registered in connect(). Note that | ||||||
|  |         # single quotes are used because this is a string (and could otherwise | ||||||
|  |         # cause a collision with a field name). | ||||||
|  |         return "django_date_trunc('%s', %s)" % (lookup_type.lower(), field_name) | ||||||
|  |  | ||||||
|  |     def datetime_extract_sql(self, lookup_type, field_name, tzname): | ||||||
|  |         # Same comment as in date_extract_sql. | ||||||
|  |         if settings.USE_TZ: | ||||||
|  |             if pytz is None: | ||||||
|  |                 raise ImproperlyConfigured("This query requires pytz, " | ||||||
|  |                                            "but it isn't installed.") | ||||||
|  |         return "django_datetime_extract('%s', %s, %%s)" % ( | ||||||
|  |             lookup_type.lower(), field_name), [tzname] | ||||||
|  |  | ||||||
|  |     def datetime_trunc_sql(self, lookup_type, field_name, tzname): | ||||||
|  |         # Same comment as in date_trunc_sql. | ||||||
|  |         if settings.USE_TZ: | ||||||
|  |             if pytz is None: | ||||||
|  |                 raise ImproperlyConfigured("This query requires pytz, " | ||||||
|  |                                            "but it isn't installed.") | ||||||
|  |         return "django_datetime_trunc('%s', %s, %%s)" % ( | ||||||
|  |             lookup_type.lower(), field_name), [tzname] | ||||||
|  |  | ||||||
|  |     def drop_foreignkey_sql(self): | ||||||
|  |         return "" | ||||||
|  |  | ||||||
|  |     def pk_default_value(self): | ||||||
|  |         return "NULL" | ||||||
|  |  | ||||||
|  |     def quote_name(self, name): | ||||||
|  |         if name.startswith('"') and name.endswith('"'): | ||||||
|  |             return name  # Quoting once is enough. | ||||||
|  |         return '"%s"' % name | ||||||
|  |  | ||||||
|  |     def no_limit_value(self): | ||||||
|  |         return -1 | ||||||
|  |  | ||||||
|  |     def sql_flush(self, style, tables, sequences, allow_cascade=False): | ||||||
|  |         # NB: The generated SQL below is specific to SQLite | ||||||
|  |         # Note: The DELETE FROM... SQL generated below works for SQLite databases | ||||||
|  |         # because constraints don't exist | ||||||
|  |         sql = ['%s %s %s;' % ( | ||||||
|  |             style.SQL_KEYWORD('DELETE'), | ||||||
|  |             style.SQL_KEYWORD('FROM'), | ||||||
|  |             style.SQL_FIELD(self.quote_name(table)) | ||||||
|  |         ) for table in tables] | ||||||
|  |         # Note: No requirement for reset of auto-incremented indices (cf. other | ||||||
|  |         # sql_flush() implementations). Just return SQL at this point | ||||||
|  |         return sql | ||||||
|  |  | ||||||
|  |     def value_to_db_datetime(self, value): | ||||||
|  |         if value is None: | ||||||
|  |             return None | ||||||
|  |  | ||||||
|  |         # SQLite doesn't support tz-aware datetimes | ||||||
|  |         if timezone.is_aware(value): | ||||||
|  |             if settings.USE_TZ: | ||||||
|  |                 value = value.astimezone(timezone.utc).replace(tzinfo=None) | ||||||
|  |             else: | ||||||
|  |                 raise ValueError("SQLite backend does not support timezone-aware datetimes when USE_TZ is False.") | ||||||
|  |  | ||||||
|  |         return six.text_type(value) | ||||||
|  |  | ||||||
|  |     def value_to_db_time(self, value): | ||||||
|  |         if value is None: | ||||||
|  |             return None | ||||||
|  |  | ||||||
|  |         # SQLite doesn't support tz-aware datetimes | ||||||
|  |         if timezone.is_aware(value): | ||||||
|  |             raise ValueError("SQLite backend does not support timezone-aware times.") | ||||||
|  |  | ||||||
|  |         return six.text_type(value) | ||||||
|  |  | ||||||
|  |     def get_db_converters(self, expression): | ||||||
|  |         converters = super(DatabaseOperations, self).get_db_converters(expression) | ||||||
|  |         internal_type = expression.output_field.get_internal_type() | ||||||
|  |         if internal_type == 'DateTimeField': | ||||||
|  |             converters.append(self.convert_datetimefield_value) | ||||||
|  |         elif internal_type == 'DateField': | ||||||
|  |             converters.append(self.convert_datefield_value) | ||||||
|  |         elif internal_type == 'TimeField': | ||||||
|  |             converters.append(self.convert_timefield_value) | ||||||
|  |         elif internal_type == 'DecimalField': | ||||||
|  |             converters.append(self.convert_decimalfield_value) | ||||||
|  |         elif internal_type == 'UUIDField': | ||||||
|  |             converters.append(self.convert_uuidfield_value) | ||||||
|  |         return converters | ||||||
|  |  | ||||||
|  |     def convert_decimalfield_value(self, value, expression, context): | ||||||
|  |         return backend_utils.typecast_decimal(expression.output_field.format_number(value)) | ||||||
|  |  | ||||||
|  |     def convert_datefield_value(self, value, expression, context): | ||||||
|  |         if value is not None and not isinstance(value, datetime.date): | ||||||
|  |             value = parse_date(value) | ||||||
|  |         return value | ||||||
|  |  | ||||||
|  |     def convert_datetimefield_value(self, value, expression, context): | ||||||
|  |         if value is not None and not isinstance(value, datetime.datetime): | ||||||
|  |             value = parse_datetime_with_timezone_support(value) | ||||||
|  |         return value | ||||||
|  |  | ||||||
|  |     def convert_timefield_value(self, value, expression, context): | ||||||
|  |         if value is not None and not isinstance(value, datetime.time): | ||||||
|  |             value = parse_time(value) | ||||||
|  |         return value | ||||||
|  |  | ||||||
|  |     def convert_uuidfield_value(self, value, expression, context): | ||||||
|  |         if value is not None: | ||||||
|  |             value = uuid.UUID(value) | ||||||
|  |         return value | ||||||
|  |  | ||||||
|  |     def bulk_insert_sql(self, fields, num_values): | ||||||
|  |         res = [] | ||||||
|  |         res.append("SELECT %s" % ", ".join( | ||||||
|  |             "%%s AS %s" % self.quote_name(f.column) for f in fields | ||||||
|  |         )) | ||||||
|  |         res.extend(["UNION ALL SELECT %s" % ", ".join(["%s"] * len(fields))] * (num_values - 1)) | ||||||
|  |         return " ".join(res) | ||||||
|  |  | ||||||
|  |     def combine_expression(self, connector, sub_expressions): | ||||||
|  |         # SQLite doesn't have a power function, so we fake it with a | ||||||
|  |         # user-defined function django_power that's registered in connect(). | ||||||
|  |         if connector == '^': | ||||||
|  |             return 'django_power(%s)' % ','.join(sub_expressions) | ||||||
|  |         return super(DatabaseOperations, self).combine_expression(connector, sub_expressions) | ||||||
|  |  | ||||||
|  |     def combine_duration_expression(self, connector, sub_expressions): | ||||||
|  |         if connector not in ['+', '-']: | ||||||
|  |             raise utils.DatabaseError('Invalid connector for timedelta: %s.' % connector) | ||||||
|  |         fn_params = ["'%s'" % connector] + sub_expressions | ||||||
|  |         if len(fn_params) > 3: | ||||||
|  |             raise ValueError('Too many params for timedelta operations.') | ||||||
|  |         return "django_format_dtdelta(%s)" % ', '.join(fn_params) | ||||||
|  |  | ||||||
|  |     def integer_field_range(self, internal_type): | ||||||
|  |         # SQLite doesn't enforce any integer constraints | ||||||
|  |         return (None, None) | ||||||
| @@ -1,10 +1,13 @@ | |||||||
| import codecs | import codecs | ||||||
| import copy | import copy | ||||||
| from decimal import Decimal | from decimal import Decimal | ||||||
| from django.utils import six |  | ||||||
| from django.apps.registry import Apps | from django.apps.registry import Apps | ||||||
| from django.db.backends.schema import BaseDatabaseSchemaEditor | from django.db.backends.base.schema import BaseDatabaseSchemaEditor | ||||||
| from django.db.models.fields.related import ManyToManyField | from django.db.models.fields.related import ManyToManyField | ||||||
|  | from django.utils import six | ||||||
|  |  | ||||||
|  | import _sqlite3 | ||||||
|  |  | ||||||
|  |  | ||||||
| class DatabaseSchemaEditor(BaseDatabaseSchemaEditor): | class DatabaseSchemaEditor(BaseDatabaseSchemaEditor): | ||||||
| @@ -13,8 +16,6 @@ class DatabaseSchemaEditor(BaseDatabaseSchemaEditor): | |||||||
|     sql_create_inline_fk = "REFERENCES %(to_table)s (%(to_column)s)" |     sql_create_inline_fk = "REFERENCES %(to_table)s (%(to_column)s)" | ||||||
|  |  | ||||||
|     def quote_value(self, value): |     def quote_value(self, value): | ||||||
|         # Inner import to allow nice failure for backend if not present |  | ||||||
|         import _sqlite3 |  | ||||||
|         try: |         try: | ||||||
|             value = _sqlite3.adapt(value) |             value = _sqlite3.adapt(value) | ||||||
|         except _sqlite3.ProgrammingError: |         except _sqlite3.ProgrammingError: | ||||||
|   | |||||||
							
								
								
									
										11
									
								
								django/db/backends/sqlite3/utils.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								django/db/backends/sqlite3/utils.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | |||||||
|  | from django.conf import settings | ||||||
|  | from django.utils import timezone | ||||||
|  | from django.utils.dateparse import parse_datetime | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def parse_datetime_with_timezone_support(value): | ||||||
|  |     dt = parse_datetime(value) | ||||||
|  |     # Confirm that dt is naive before overwriting its tzinfo. | ||||||
|  |     if dt is not None and settings.USE_TZ and timezone.is_naive(dt): | ||||||
|  |         dt = dt.replace(tzinfo=timezone.utc) | ||||||
|  |     return dt | ||||||
| @@ -272,7 +272,7 @@ if supplied) should be callable objects that accept two arguments; the first is | |||||||
| an instance of ``django.apps.registry.Apps`` containing historical models that | an instance of ``django.apps.registry.Apps`` containing historical models that | ||||||
| match the operation's place in the project history, and the second is an | match the operation's place in the project history, and the second is an | ||||||
| instance of :class:`SchemaEditor | instance of :class:`SchemaEditor | ||||||
| <django.db.backends.schema.BaseDatabaseSchemaEditor>`. | <django.db.backends.base.schema.BaseDatabaseSchemaEditor>`. | ||||||
|  |  | ||||||
| The optional ``hints`` argument will be passed as ``**hints`` to the | The optional ``hints`` argument will be passed as ``**hints`` to the | ||||||
| :meth:`allow_migrate` method of database routers to assist them in making a | :meth:`allow_migrate` method of database routers to assist them in making a | ||||||
|   | |||||||
| @@ -2,7 +2,7 @@ | |||||||
| ``SchemaEditor`` | ``SchemaEditor`` | ||||||
| ================ | ================ | ||||||
|  |  | ||||||
| .. module:: django.db.backends.schema | .. module:: django.db.backends.base.schema | ||||||
|  |  | ||||||
| .. class:: BaseDatabaseSchemaEditor | .. class:: BaseDatabaseSchemaEditor | ||||||
|  |  | ||||||
|   | |||||||
| @@ -942,6 +942,19 @@ Database backend API | |||||||
| The following changes to the database backend API are documented to assist | The following changes to the database backend API are documented to assist | ||||||
| those writing third-party backends in updating their code: | those writing third-party backends in updating their code: | ||||||
|  |  | ||||||
|  | * ``BaseDatabaseXXX`` classes have been moved to ``django.db.backends.base``. | ||||||
|  |   Please import them from the new locations:: | ||||||
|  |  | ||||||
|  |     from django.db.backends.base.base import BaseDatabaseWrapper | ||||||
|  |     from django.db.backends.base.client import BaseDatabaseClient | ||||||
|  |     from django.db.backends.base.creation import BaseDatabaseCreation | ||||||
|  |     from django.db.backends.base.features import BaseDatabaseFeatures | ||||||
|  |     from django.db.backends.base.introspection import BaseDatabaseIntrospection | ||||||
|  |     from django.db.backends.base.introspection import FieldInfo, TableInfo | ||||||
|  |     from django.db.backends.base.operations import BaseDatabaseOperations | ||||||
|  |     from django.db.backends.base.schema import BaseDatabaseSchemaEditor | ||||||
|  |     from django.db.backends.base.validation import BaseDatabaseValidation | ||||||
|  |  | ||||||
| * The ``data_types``, ``data_types_suffix``, and | * The ``data_types``, ``data_types_suffix``, and | ||||||
|   ``data_type_check_constraints`` attributes have moved from the |   ``data_type_check_constraints`` attributes have moved from the | ||||||
|   ``DatabaseCreation`` class to ``DatabaseWrapper``. |   ``DatabaseCreation`` class to ``DatabaseWrapper``. | ||||||
|   | |||||||
| @@ -15,7 +15,7 @@ from django.core.exceptions import ImproperlyConfigured | |||||||
| from django.core.management.color import no_style | from django.core.management.color import no_style | ||||||
| from django.db import (connection, connections, DEFAULT_DB_ALIAS, | from django.db import (connection, connections, DEFAULT_DB_ALIAS, | ||||||
|     DatabaseError, IntegrityError, reset_queries, transaction) |     DatabaseError, IntegrityError, reset_queries, transaction) | ||||||
| from django.db.backends import BaseDatabaseWrapper | from django.db.backends.base.base import BaseDatabaseWrapper | ||||||
| from django.db.backends.signals import connection_created | from django.db.backends.signals import connection_created | ||||||
| from django.db.backends.postgresql_psycopg2 import version as pg_version | from django.db.backends.postgresql_psycopg2 import version as pg_version | ||||||
| from django.db.backends.utils import format_number, CursorWrapper | from django.db.backends.utils import format_number, CursorWrapper | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user