diff --git a/django/contrib/gis/db/backends/spatialite/operations.py b/django/contrib/gis/db/backends/spatialite/operations.py index 8d8435f824..827c5217a5 100644 --- a/django/contrib/gis/db/backends/spatialite/operations.py +++ b/django/contrib/gis/db/backends/spatialite/operations.py @@ -53,6 +53,8 @@ class SpatiaLiteRelate(SpatiaLiteFunctionParam): # Valid distance types and substitutions dtypes = (Decimal, Distance, float) + six.integer_types + + def get_dist_ops(operator): "Returns operations for regular distances; spherical distances are not currently supported." return (SpatiaLiteDistance(operator),) diff --git a/django/contrib/gis/gdal/error.py b/django/contrib/gis/gdal/error.py index 5feef41873..528b9b5607 100644 --- a/django/contrib/gis/gdal/error.py +++ b/django/contrib/gis/gdal/error.py @@ -4,6 +4,7 @@ OGR methods. """ + #### OGR & SRS Exceptions #### class GDALException(Exception): pass diff --git a/django/contrib/gis/gdal/geometries.py b/django/contrib/gis/gdal/geometries.py index 98ae5a1108..a331053eeb 100644 --- a/django/contrib/gis/gdal/geometries.py +++ b/django/contrib/gis/gdal/geometries.py @@ -522,6 +522,7 @@ class OGRGeometry(GDALBase): """ return self._geomgen(capi.geom_union, other) + # The subclasses for OGR Geometry. class Point(OGRGeometry): @@ -550,6 +551,7 @@ class Point(OGRGeometry): return (self.x, self.y, self.z) coords = tuple + class LineString(OGRGeometry): def __getitem__(self, index): @@ -605,10 +607,12 @@ class LineString(OGRGeometry): if self.coord_dim == 3: return self._listarr(capi.getz) + # LinearRings are used in Polygons. class LinearRing(LineString): pass + class Polygon(OGRGeometry): def __len__(self): @@ -654,6 +658,7 @@ class Polygon(OGRGeometry): capi.get_centroid(self.ptr, p.ptr) return p + # Geometry Collection base class. class GeometryCollection(OGRGeometry): "The Geometry Collection class." @@ -700,13 +705,16 @@ class GeometryCollection(OGRGeometry): return tuple(self[i].tuple for i in xrange(self.geom_count)) coords = tuple + # Multiple Geometry types. class MultiPoint(GeometryCollection): pass + class MultiLineString(GeometryCollection): pass + class MultiPolygon(GeometryCollection): pass diff --git a/django/contrib/gis/gdal/geomtype.py b/django/contrib/gis/gdal/geomtype.py index e943a03ee8..026e427b43 100644 --- a/django/contrib/gis/gdal/geomtype.py +++ b/django/contrib/gis/gdal/geomtype.py @@ -2,7 +2,7 @@ from django.contrib.gis.gdal.error import OGRException from django.utils import six -#### OGRGeomType #### + class OGRGeomType(object): "Encapulates OGR Geometry Types." diff --git a/django/contrib/gis/gdal/libgdal.py b/django/contrib/gis/gdal/libgdal.py index 77487bbf0e..bdd79bf402 100644 --- a/django/contrib/gis/gdal/libgdal.py +++ b/django/contrib/gis/gdal/libgdal.py @@ -55,6 +55,7 @@ if os.name == 'nt': from ctypes import WinDLL lwingdal = WinDLL(lib_path) + def std_call(func): """ Returns the correct STDCALL function for certain OSR routines on Win32 @@ -72,15 +73,19 @@ _version_info = std_call('GDALVersionInfo') _version_info.argtypes = [c_char_p] _version_info.restype = c_char_p + def gdal_version(): "Returns only the GDAL version number information." return _version_info(b'RELEASE_NAME') + def gdal_full_version(): "Returns the full GDAL version information." return _version_info('') version_regex = re.compile(r'^(?P\d+)\.(?P\d+)(\.(?P\d+))?') + + def gdal_version_info(): ver = gdal_version().decode() m = version_regex.match(ver) @@ -97,10 +102,13 @@ del _verinfo # Set library error handling so as errors are logged CPLErrorHandler = CFUNCTYPE(None, c_int, c_int, c_char_p) + + def err_handler(error_class, error_number, message): logger.error('GDAL_ERROR %d: %s' % (error_number, message)) err_handler = CPLErrorHandler(err_handler) + def function(name, args, restype): func = std_call(name) func.argtypes = args diff --git a/django/contrib/gis/gdal/prototypes/errcheck.py b/django/contrib/gis/gdal/prototypes/errcheck.py index 1dbd924ab5..491641646e 100644 --- a/django/contrib/gis/gdal/prototypes/errcheck.py +++ b/django/contrib/gis/gdal/prototypes/errcheck.py @@ -15,10 +15,12 @@ def arg_byref(args, offset=-1): "Returns the pointer argument's by-refernece value." return args[offset]._obj.value + def ptr_byref(args, offset=-1): "Returns the pointer argument passed in by-reference." return args[offset]._obj + ### String checking Routines ### def check_const_string(result, func, cargs, offset=None): """ @@ -31,6 +33,7 @@ def check_const_string(result, func, cargs, offset=None): else: return result + def check_string(result, func, cargs, offset=-1, str_result=False): """ Checks the string output returned from the given function, and frees @@ -61,12 +64,14 @@ def check_string(result, func, cargs, offset=-1, str_result=False): ### DataSource, Layer error-checking ### + ### Envelope checking ### def check_envelope(result, func, cargs, offset=-1): "Checks a function that returns an OGR Envelope by reference." env = ptr_byref(cargs, offset) return env + ### Geometry error-checking routines ### def check_geom(result, func, cargs): "Checks a function that returns a geometry." @@ -78,12 +83,14 @@ def check_geom(result, func, cargs): raise OGRException('Invalid geometry pointer returned from "%s".' % func.__name__) return result + def check_geom_offset(result, func, cargs, offset=-1): "Chcks the geometry at the given offset in the C parameter list." check_err(result) geom = ptr_byref(cargs, offset=offset) return check_geom(geom, func, cargs) + ### Spatial Reference error-checking routines ### def check_srs(result, func, cargs): if isinstance(result, six.integer_types): @@ -92,6 +99,7 @@ def check_srs(result, func, cargs): raise SRSException('Invalid spatial reference pointer returned from "%s".' % func.__name__) return result + ### Other error-checking routines ### def check_arg_errcode(result, func, cargs): """ @@ -101,12 +109,14 @@ def check_arg_errcode(result, func, cargs): check_err(arg_byref(cargs)) return result + def check_errcode(result, func, cargs): """ Check the error code returned (c_int). """ check_err(result) + def check_pointer(result, func, cargs): "Makes sure the result pointer is valid." if isinstance(result, six.integer_types): @@ -116,6 +126,7 @@ def check_pointer(result, func, cargs): else: raise OGRException('Invalid pointer returned from "%s"' % func.__name__) + def check_str_arg(result, func, cargs): """ This is for the OSRGet[Angular|Linear]Units functions, which diff --git a/django/contrib/gis/gdal/prototypes/generation.py b/django/contrib/gis/gdal/prototypes/generation.py index bf516ce796..6b38dc593f 100644 --- a/django/contrib/gis/gdal/prototypes/generation.py +++ b/django/contrib/gis/gdal/prototypes/generation.py @@ -8,9 +8,11 @@ from django.contrib.gis.gdal.prototypes.errcheck import ( check_arg_errcode, check_errcode, check_geom, check_geom_offset, check_pointer, check_srs, check_str_arg, check_string, check_const_string) + class gdal_char_p(c_char_p): pass + def double_output(func, argtypes, errcheck=False, strarg=False): "Generates a ctypes function that returns a double value." func.argtypes = argtypes @@ -21,6 +23,7 @@ def double_output(func, argtypes, errcheck=False, strarg=False): func.errcheck = check_str_arg return func + def geom_output(func, argtypes, offset=None): """ Generates a function that returns a Geometry either by reference @@ -43,12 +46,14 @@ def geom_output(func, argtypes, offset=None): return func + def int_output(func, argtypes): "Generates a ctypes function that returns an integer value." func.argtypes = argtypes func.restype = c_int return func + def srs_output(func, argtypes): """ Generates a ctypes prototype for the given function with @@ -60,6 +65,7 @@ def srs_output(func, argtypes): func.errcheck = check_srs return func + def const_string_output(func, argtypes, offset=None, decoding=None): func.argtypes = argtypes if offset: @@ -76,6 +82,7 @@ def const_string_output(func, argtypes, offset=None, decoding=None): return func + def string_output(func, argtypes, offset=-1, str_result=False, decoding=None): """ Generates a ctypes prototype for the given function with the @@ -104,6 +111,7 @@ def string_output(func, argtypes, offset=-1, str_result=False, decoding=None): func.errcheck = _check_str return func + def void_output(func, argtypes, errcheck=True): """ For functions that don't only return an error code that needs to @@ -121,6 +129,7 @@ def void_output(func, argtypes, errcheck=True): return func + def voidptr_output(func, argtypes): "For functions that return c_void_p." func.argtypes = argtypes diff --git a/django/contrib/gis/gdal/prototypes/geom.py b/django/contrib/gis/gdal/prototypes/geom.py index f8016bc7bf..f5dc7e10e9 100644 --- a/django/contrib/gis/gdal/prototypes/geom.py +++ b/django/contrib/gis/gdal/prototypes/geom.py @@ -5,6 +5,7 @@ from django.contrib.gis.gdal.prototypes.errcheck import check_envelope from django.contrib.gis.gdal.prototypes.generation import (const_string_output, double_output, geom_output, int_output, srs_output, string_output, void_output) + ### Generation routines specific to this module ### def env_func(f, argtypes): "For getting OGREnvelopes." @@ -13,10 +14,12 @@ def env_func(f, argtypes): f.errcheck = check_envelope return f + def pnt_func(f): "For accessing point information." return double_output(f, [c_void_p, c_int]) + def topology_func(f): f.argtypes = [c_void_p, c_void_p] f.restype = c_int diff --git a/django/contrib/gis/gdal/prototypes/srs.py b/django/contrib/gis/gdal/prototypes/srs.py index 29a73ddcc7..f7491200de 100644 --- a/django/contrib/gis/gdal/prototypes/srs.py +++ b/django/contrib/gis/gdal/prototypes/srs.py @@ -3,6 +3,7 @@ from django.contrib.gis.gdal.libgdal import lgdal, std_call from django.contrib.gis.gdal.prototypes.generation import (const_string_output, double_output, int_output, srs_output, string_output, void_output) + ## Shortcut generation for routines with known parameters. def srs_double(f): """ @@ -11,6 +12,7 @@ def srs_double(f): """ return double_output(f, [c_void_p, POINTER(c_int)], errcheck=True) + def units_func(f): """ Creates a ctypes function prototype for OSR units functions, e.g., diff --git a/django/contrib/gis/geoip/prototypes.py b/django/contrib/gis/geoip/prototypes.py index 6d55a1041b..3eae261930 100644 --- a/django/contrib/gis/geoip/prototypes.py +++ b/django/contrib/gis/geoip/prototypes.py @@ -1,6 +1,7 @@ from ctypes import c_char_p, c_float, c_int, string_at, Structure, POINTER from django.contrib.gis.geoip.libgeoip import lgeoip, free + #### GeoIP C Structure definitions #### class GeoIPRecord(Structure): @@ -27,6 +28,7 @@ geoip_encodings = { 1: 'utf8', } + class GeoIPTag(Structure): pass @@ -48,6 +50,7 @@ GeoIPRecord_delete = lgeoip.GeoIPRecord_delete GeoIPRecord_delete.argtypes = [RECTYPE] GeoIPRecord_delete.restype = None + # For retrieving records by name or address. def check_record(result, func, cargs): if result: @@ -68,6 +71,7 @@ def check_record(result, func, cargs): else: return None + def record_output(func): func.argtypes = [DBTYPE, c_char_p] func.restype = RECTYPE @@ -84,10 +88,12 @@ GeoIP_delete = lgeoip.GeoIP_delete GeoIP_delete.argtypes = [DBTYPE] GeoIP_delete.restype = None + # This is so the string pointer can be freed within Python. class geoip_char_p(c_char_p): pass + def check_string(result, func, cargs): if result: s = string_at(result) @@ -100,6 +106,7 @@ GeoIP_database_info = lgeoip.GeoIP_database_info GeoIP_database_info.restype = geoip_char_p GeoIP_database_info.errcheck = check_string + # String output routines. def string_output(func): def _err_check(result, func, cargs): diff --git a/django/contrib/gis/geos/collections.py b/django/contrib/gis/geos/collections.py index 5a7dcf56ae..426cf3833d 100644 --- a/django/contrib/gis/geos/collections.py +++ b/django/contrib/gis/geos/collections.py @@ -12,6 +12,7 @@ from django.contrib.gis.geos.polygon import Polygon from django.contrib.gis.geos import prototypes as capi from django.utils.six.moves import xrange + class GeometryCollection(GEOSGeometry): _typeid = 7 @@ -91,11 +92,13 @@ class GeometryCollection(GEOSGeometry): return tuple(g.tuple for g in self) coords = tuple + # MultiPoint, MultiLineString, and MultiPolygon class definitions. class MultiPoint(GeometryCollection): _allowed = Point _typeid = 4 + class MultiLineString(GeometryCollection): _allowed = (LineString, LinearRing) _typeid = 5 @@ -108,6 +111,7 @@ class MultiLineString(GeometryCollection): """ return self._topology(capi.geos_linemerge(self.ptr)) + class MultiPolygon(GeometryCollection): _allowed = Polygon _typeid = 6 diff --git a/django/contrib/gis/geos/coordseq.py b/django/contrib/gis/geos/coordseq.py index 436d39d411..c0e455a5c6 100644 --- a/django/contrib/gis/geos/coordseq.py +++ b/django/contrib/gis/geos/coordseq.py @@ -10,6 +10,7 @@ from django.contrib.gis.geos.libgeos import CS_PTR from django.contrib.gis.geos import prototypes as capi from django.utils.six.moves import xrange + class GEOSCoordSeq(GEOSBase): "The internal representation of a list of coordinates inside a Geometry." diff --git a/django/contrib/gis/geos/error.py b/django/contrib/gis/geos/error.py index 0ebc54f7cc..e19bf9a47a 100644 --- a/django/contrib/gis/geos/error.py +++ b/django/contrib/gis/geos/error.py @@ -3,10 +3,12 @@ GEOSGeometryIndexError. """ + class GEOSException(Exception): "The base GEOS exception, indicates a GEOS-related error." pass + class GEOSIndexError(GEOSException, KeyError): """ This exception is raised when an invalid index is encountered, and has diff --git a/django/contrib/gis/geos/libgeos.py b/django/contrib/gis/geos/libgeos.py index d87f55853b..059eb5b602 100644 --- a/django/contrib/gis/geos/libgeos.py +++ b/django/contrib/gis/geos/libgeos.py @@ -62,6 +62,8 @@ lgeos = CDLL(lib_path) # Supposed to mimic the GEOS message handler (C below): # typedef void (*GEOSMessageHandler)(const char *fmt, ...); NOTICEFUNC = CFUNCTYPE(None, c_char_p, c_char_p) + + def notice_h(fmt, lst): fmt, lst = fmt.decode(), lst.decode() try: @@ -72,6 +74,8 @@ def notice_h(fmt, lst): notice_h = NOTICEFUNC(notice_h) ERRORFUNC = CFUNCTYPE(None, c_char_p, c_char_p) + + def error_h(fmt, lst): fmt, lst = fmt.decode(), lst.decode() try: @@ -83,16 +87,20 @@ error_h = ERRORFUNC(error_h) #### GEOS Geometry C data structures, and utility functions. #### + # Opaque GEOS geometry structures, used for GEOM_PTR and CS_PTR class GEOSGeom_t(Structure): pass + class GEOSPrepGeom_t(Structure): pass + class GEOSCoordSeq_t(Structure): pass + class GEOSContextHandle_t(Structure): pass @@ -102,6 +110,7 @@ PREPGEOM_PTR = POINTER(GEOSPrepGeom_t) CS_PTR = POINTER(GEOSCoordSeq_t) CONTEXT_PTR = POINTER(GEOSContextHandle_t) + # Used specifically by the GEOSGeom_createPolygon and GEOSGeom_createCollection # GEOS routines def get_pointer_arr(n): @@ -121,6 +130,8 @@ version_regex = re.compile( r'^(?P(?P\d+)\.(?P\d+)\.(?P\d+))' r'((rc(?P\d+))|dev)?-CAPI-(?P\d+\.\d+\.\d+)( r\d+)?$' ) + + def geos_version_info(): """ Returns a dictionary containing the various version metadata parsed from diff --git a/django/contrib/gis/geos/linestring.py b/django/contrib/gis/geos/linestring.py index be23ea485f..7a03712600 100644 --- a/django/contrib/gis/geos/linestring.py +++ b/django/contrib/gis/geos/linestring.py @@ -6,6 +6,7 @@ from django.contrib.gis.geos.point import Point from django.contrib.gis.geos import prototypes as capi from django.utils.six.moves import xrange + class LineString(GEOSGeometry): _init_func = capi.create_linestring _minlength = 2 @@ -161,6 +162,7 @@ class LineString(GEOSGeometry): else: return self._listarr(self._cs.getZ) + # LinearRings are LineStrings used within Polygons. class LinearRing(LineString): _minLength = 4 diff --git a/django/contrib/gis/geos/point.py b/django/contrib/gis/geos/point.py index c097ccef88..5fcf1bb407 100644 --- a/django/contrib/gis/geos/point.py +++ b/django/contrib/gis/geos/point.py @@ -5,6 +5,7 @@ from django.contrib.gis.geos import prototypes as capi from django.utils import six from django.utils.six.moves import xrange + class Point(GEOSGeometry): _minlength = 2 _maxlength = 3 diff --git a/django/contrib/gis/geos/prototypes/coordseq.py b/django/contrib/gis/geos/prototypes/coordseq.py index 68b9480ff8..c47f064f32 100644 --- a/django/contrib/gis/geos/prototypes/coordseq.py +++ b/django/contrib/gis/geos/prototypes/coordseq.py @@ -3,6 +3,7 @@ from django.contrib.gis.geos.libgeos import GEOM_PTR, CS_PTR from django.contrib.gis.geos.prototypes.errcheck import last_arg_byref, GEOSException from django.contrib.gis.geos.prototypes.threadsafe import GEOSFunc + ## Error-checking routines specific to coordinate sequences. ## def check_cs_ptr(result, func, cargs): "Error checking on routines that return Geometries." @@ -10,6 +11,7 @@ def check_cs_ptr(result, func, cargs): raise GEOSException('Error encountered checking Coordinate Sequence returned from GEOS C function "%s".' % func.__name__) return result + def check_cs_op(result, func, cargs): "Checks the status code of a coordinate sequence operation." if result == 0: @@ -17,12 +19,14 @@ def check_cs_op(result, func, cargs): else: return result + def check_cs_get(result, func, cargs): "Checking the coordinate sequence retrieval." check_cs_op(result, func, cargs) # Object in by reference, return its value. return last_arg_byref(cargs) + ## Coordinate sequence prototype generation functions. ## def cs_int(func): "For coordinate sequence routines that return an integer." @@ -31,6 +35,7 @@ def cs_int(func): func.errcheck = check_cs_get return func + def cs_operation(func, ordinate=False, get=False): "For coordinate sequence operations." if get: @@ -50,6 +55,7 @@ def cs_operation(func, ordinate=False, get=False): func.restype = c_int return func + def cs_output(func, argtypes): "For routines that return a coordinate sequence." func.argtypes = argtypes diff --git a/django/contrib/gis/geos/prototypes/errcheck.py b/django/contrib/gis/geos/prototypes/errcheck.py index 40ae37eb2f..bd99047604 100644 --- a/django/contrib/gis/geos/prototypes/errcheck.py +++ b/django/contrib/gis/geos/prototypes/errcheck.py @@ -25,11 +25,13 @@ else: libc = CDLL(None) free = libc.free + ### ctypes error checking routines ### def last_arg_byref(args): "Returns the last C argument's value by reference." return args[-1]._obj.value + def check_dbl(result, func, cargs): "Checks the status code and returns the double value passed in by reference." # Checking the status code @@ -38,12 +40,14 @@ def check_dbl(result, func, cargs): # Double passed in by reference, return its value. return last_arg_byref(cargs) + def check_geom(result, func, cargs): "Error checking on routines that return Geometries." if not result: raise GEOSException('Error encountered checking Geometry returned from GEOS C function "%s".' % func.__name__) return result + def check_minus_one(result, func, cargs): "Error checking on routines that should not return -1." if result == -1: @@ -51,6 +55,7 @@ def check_minus_one(result, func, cargs): else: return result + def check_predicate(result, func, cargs): "Error checking for unary/binary predicate functions." val = ord(result) # getting the ordinal from the character @@ -61,6 +66,7 @@ def check_predicate(result, func, cargs): else: raise GEOSException('Error encountered on GEOS C predicate function "%s".' % func.__name__) + def check_sized_string(result, func, cargs): """ Error checking for routines that return explicitly sized strings. @@ -77,6 +83,7 @@ def check_sized_string(result, func, cargs): free(result) return s + def check_string(result, func, cargs): """ Error checking for routines that return strings. @@ -91,6 +98,7 @@ def check_string(result, func, cargs): free(result) return s + def check_zero(result, func, cargs): "Error checking on routines that should not return 0." if result == 0: diff --git a/django/contrib/gis/geos/prototypes/geom.py b/django/contrib/gis/geos/prototypes/geom.py index 7ca2f89aed..021fe6e469 100644 --- a/django/contrib/gis/geos/prototypes/geom.py +++ b/django/contrib/gis/geos/prototypes/geom.py @@ -7,6 +7,7 @@ from django.contrib.gis.geos.prototypes.threadsafe import GEOSFunc # This is the return type used by binary output (WKB, HEX) routines. c_uchar_p = POINTER(c_ubyte) + # We create a simple subclass of c_char_p here because when the response # type is set to c_char_p, you get a _Python_ string and there's no way # to access the string's address inside the error checking function. @@ -17,6 +18,7 @@ c_uchar_p = POINTER(c_ubyte) class geos_char_p(c_char_p): pass + ### ctypes generation functions ### def bin_constructor(func): "Generates a prototype for binary construction (HEX, WKB) GEOS routines." @@ -25,6 +27,7 @@ def bin_constructor(func): func.errcheck = check_geom return func + # HEX & WKB output def bin_output(func): "Generates a prototype for the routines that return a sized string." @@ -33,6 +36,7 @@ def bin_output(func): func.restype = c_uchar_p return func + def geom_output(func, argtypes): "For GEOS routines that return a geometry." if argtypes: @@ -41,10 +45,12 @@ def geom_output(func, argtypes): func.errcheck = check_geom return func + def geom_index(func): "For GEOS routines that return geometries from an index." return geom_output(func, [GEOM_PTR, c_int]) + def int_from_geom(func, zero=False): "Argument is a geometry, return type is an integer." func.argtypes = [GEOM_PTR] @@ -55,6 +61,7 @@ def int_from_geom(func, zero=False): func.errcheck = check_minus_one return func + def string_from_geom(func): "Argument is a Geometry, return type is a string." func.argtypes = [GEOM_PTR] diff --git a/django/contrib/gis/geos/prototypes/io.py b/django/contrib/gis/geos/prototypes/io.py index acbd3ebc59..7f95146207 100644 --- a/django/contrib/gis/geos/prototypes/io.py +++ b/django/contrib/gis/geos/prototypes/io.py @@ -10,16 +10,20 @@ from django.contrib.gis.geos.prototypes.threadsafe import GEOSFunc from django.utils import six from django.utils.encoding import force_bytes + ### The WKB/WKT Reader/Writer structures and pointers ### class WKTReader_st(Structure): pass + class WKTWriter_st(Structure): pass + class WKBReader_st(Structure): pass + class WKBWriter_st(Structure): pass @@ -71,6 +75,7 @@ wkb_reader_create.restype = WKB_READ_PTR wkb_reader_destroy = GEOSFunc('GEOSWKBReader_destroy') wkb_reader_destroy.argtypes = [WKB_READ_PTR] + def wkb_read_func(func): # Although the function definitions take `const unsigned char *` # as their parameter, we use c_char_p here so the function may @@ -92,6 +97,7 @@ wkb_writer_create.restype = WKB_WRITE_PTR wkb_writer_destroy = GEOSFunc('GEOSWKBWriter_destroy') wkb_writer_destroy.argtypes = [WKB_WRITE_PTR] + # WKB Writing prototypes. def wkb_write_func(func): func.argtypes = [WKB_WRITE_PTR, GEOM_PTR, POINTER(c_size_t)] @@ -102,12 +108,14 @@ def wkb_write_func(func): wkb_writer_write = wkb_write_func(GEOSFunc('GEOSWKBWriter_write')) wkb_writer_write_hex = wkb_write_func(GEOSFunc('GEOSWKBWriter_writeHEX')) + # WKBWriter property getter/setter prototypes. def wkb_writer_get(func, restype=c_int): func.argtypes = [WKB_WRITE_PTR] func.restype = restype return func + def wkb_writer_set(func, argtype=c_int): func.argtypes = [WKB_WRITE_PTR, argtype] return func @@ -119,6 +127,7 @@ wkb_writer_set_outdim = wkb_writer_set(GEOSFunc('GEOSWKBWriter_setOutputDimensio wkb_writer_get_include_srid = wkb_writer_get(GEOSFunc('GEOSWKBWriter_getIncludeSRID'), restype=c_char) wkb_writer_set_include_srid = wkb_writer_set(GEOSFunc('GEOSWKBWriter_setIncludeSRID'), argtype=c_char) + ### Base I/O Class ### class IOBase(GEOSBase): "Base class for GEOS I/O objects." @@ -133,6 +142,7 @@ class IOBase(GEOSBase): ### Base WKB/WKT Reading and Writing objects ### + # Non-public WKB/WKT reader classes for internal use because # their `read` methods return _pointers_ instead of GEOSGeometry # objects. @@ -146,6 +156,7 @@ class _WKTReader(IOBase): raise TypeError return wkt_reader_read(self.ptr, force_bytes(wkt)) + class _WKBReader(IOBase): _constructor = wkb_reader_create _destructor = wkb_reader_destroy @@ -161,6 +172,7 @@ class _WKBReader(IOBase): else: raise TypeError + ### WKB/WKT Writer Classes ### class WKTWriter(IOBase): _constructor = wkt_writer_create @@ -232,6 +244,7 @@ class WKBWriter(IOBase): srid = property(_get_include_srid, _set_include_srid) + # `ThreadLocalIO` object holds instances of the WKT and WKB reader/writer # objects that are local to the thread. The `GEOSGeometry` internals # access these instances by calling the module-level functions, defined @@ -245,6 +258,7 @@ class ThreadLocalIO(threading.local): thread_context = ThreadLocalIO() + # These module-level routines return the I/O object that is local to the # thread. If the I/O object does not exist yet it will be initialized. def wkt_r(): @@ -252,23 +266,27 @@ def wkt_r(): thread_context.wkt_r = _WKTReader() return thread_context.wkt_r + def wkt_w(dim=2): if not thread_context.wkt_w: thread_context.wkt_w = WKTWriter() thread_context.wkt_w.outdim = dim return thread_context.wkt_w + def wkb_r(): if not thread_context.wkb_r: thread_context.wkb_r = _WKBReader() return thread_context.wkb_r + def wkb_w(dim=2): if not thread_context.wkb_w: thread_context.wkb_w = WKBWriter() thread_context.wkb_w.outdim = dim return thread_context.wkb_w + def ewkb_w(dim=2): if not thread_context.ewkb_w: thread_context.ewkb_w = WKBWriter() diff --git a/django/contrib/gis/geos/prototypes/misc.py b/django/contrib/gis/geos/prototypes/misc.py index 5a9b5555ad..55381bf7fd 100644 --- a/django/contrib/gis/geos/prototypes/misc.py +++ b/django/contrib/gis/geos/prototypes/misc.py @@ -11,6 +11,7 @@ from django.utils.six.moves import xrange __all__ = ['geos_area', 'geos_distance', 'geos_length'] + ### ctypes generator function ### def dbl_from_geom(func, num_geom=1): """ diff --git a/django/contrib/gis/geos/prototypes/predicates.py b/django/contrib/gis/geos/prototypes/predicates.py index 04739903f3..f3c30a5062 100644 --- a/django/contrib/gis/geos/prototypes/predicates.py +++ b/django/contrib/gis/geos/prototypes/predicates.py @@ -7,6 +7,7 @@ from django.contrib.gis.geos.libgeos import GEOM_PTR from django.contrib.gis.geos.prototypes.errcheck import check_predicate from django.contrib.gis.geos.prototypes.threadsafe import GEOSFunc + ## Binary & unary predicate functions ## def binary_predicate(func, *args): "For GEOS binary predicate functions." @@ -18,6 +19,7 @@ def binary_predicate(func, *args): func.errcheck = check_predicate return func + def unary_predicate(func): "For GEOS unary predicate functions." func.argtypes = [GEOM_PTR] diff --git a/django/contrib/gis/geos/prototypes/prepared.py b/django/contrib/gis/geos/prototypes/prepared.py index ff46979de3..eef141a2f7 100644 --- a/django/contrib/gis/geos/prototypes/prepared.py +++ b/django/contrib/gis/geos/prototypes/prepared.py @@ -12,6 +12,7 @@ prepared_destroy = GEOSFunc('GEOSPreparedGeom_destroy') prepared_destroy.argtpes = [PREPGEOM_PTR] prepared_destroy.restype = None + # Prepared geometry binary predicate support. def prepared_predicate(func): func.argtypes = [PREPGEOM_PTR, GEOM_PTR] diff --git a/django/contrib/gis/geos/prototypes/threadsafe.py b/django/contrib/gis/geos/prototypes/threadsafe.py index facb306ad4..3f979b3def 100644 --- a/django/contrib/gis/geos/prototypes/threadsafe.py +++ b/django/contrib/gis/geos/prototypes/threadsafe.py @@ -1,6 +1,7 @@ import threading from django.contrib.gis.geos.libgeos import lgeos, notice_h, error_h, CONTEXT_PTR + class GEOSContextHandle(object): """ Python object representing a GEOS context handle. @@ -14,6 +15,7 @@ class GEOSContextHandle(object): if self.ptr: lgeos.finishGEOS_r(self.ptr) + # Defining a thread-local object and creating an instance # to hold a reference to GEOSContextHandle for this thread. class GEOSContext(threading.local): @@ -21,6 +23,7 @@ class GEOSContext(threading.local): thread_context = GEOSContext() + class GEOSFunc(object): """ Class that serves as a wrapper for GEOS C Functions, and will diff --git a/django/contrib/gis/geos/prototypes/topology.py b/django/contrib/gis/geos/prototypes/topology.py index a7098ee6e7..5bc99e8409 100644 --- a/django/contrib/gis/geos/prototypes/topology.py +++ b/django/contrib/gis/geos/prototypes/topology.py @@ -13,6 +13,7 @@ from django.contrib.gis.geos.prototypes.errcheck import check_geom, check_minus_ from django.contrib.gis.geos.prototypes.geom import geos_char_p from django.contrib.gis.geos.prototypes.threadsafe import GEOSFunc + def topology(func, *args, **kwargs): "For GEOS unary topology functions." argtypes = [GEOM_PTR] diff --git a/django/contrib/gis/geos/tests/test_geos_mutation.py b/django/contrib/gis/geos/tests/test_geos_mutation.py index a5512cf123..ba95ff7582 100644 --- a/django/contrib/gis/geos/tests/test_geos_mutation.py +++ b/django/contrib/gis/geos/tests/test_geos_mutation.py @@ -16,42 +16,55 @@ if HAS_GEOS: def api_get_distance(x): return x.distance(Point(-200, -200)) + def api_get_buffer(x): return x.buffer(10) + def api_get_geom_typeid(x): return x.geom_typeid + def api_get_num_coords(x): return x.num_coords + def api_get_centroid(x): return x.centroid + def api_get_empty(x): return x.empty + def api_get_valid(x): return x.valid + def api_get_simple(x): return x.simple + def api_get_ring(x): return x.ring + def api_get_boundary(x): return x.boundary + def api_get_convex_hull(x): return x.convex_hull + def api_get_extent(x): return x.extent + def api_get_area(x): return x.area + def api_get_length(x): return x.length diff --git a/django/contrib/gis/geos/tests/test_mutable_list.py b/django/contrib/gis/geos/tests/test_mutable_list.py index c8a8cc7b30..fc3b247f1f 100644 --- a/django/contrib/gis/geos/tests/test_mutable_list.py +++ b/django/contrib/gis/geos/tests/test_mutable_list.py @@ -39,18 +39,21 @@ class UserListA(ListMixin): def _get_single_external(self, index): return self._list[index] + class UserListB(UserListA): _mytype = list def _set_single(self, index, value): self._list[index] = value + def nextRange(length): nextRange.start += 100 return range(nextRange.start, nextRange.start + length) nextRange.start = 0 + class ListMixinTest(unittest.TestCase): """ Tests base class ListMixin by comparing a list clone which is @@ -418,5 +421,6 @@ class ListMixinTest(unittest.TestCase): self.assertTrue(pl < ul, 'cmp for lt self') self.assertTrue(pl < ul, 'cmp for lt self') + class ListMixinTestSingle(ListMixinTest): listType = UserListB diff --git a/django/contrib/gis/management/commands/inspectdb.py b/django/contrib/gis/management/commands/inspectdb.py index bbf40e1baa..679fe84ca3 100644 --- a/django/contrib/gis/management/commands/inspectdb.py +++ b/django/contrib/gis/management/commands/inspectdb.py @@ -1,5 +1,6 @@ from django.core.management.commands.inspectdb import Command as InspectDBCommand + class Command(InspectDBCommand): db_module = 'django.contrib.gis.db' gis_tables = {} diff --git a/django/contrib/gis/management/commands/ogrinspect.py b/django/contrib/gis/management/commands/ogrinspect.py index 1c6482e3b5..c7b1389574 100644 --- a/django/contrib/gis/management/commands/ogrinspect.py +++ b/django/contrib/gis/management/commands/ogrinspect.py @@ -2,6 +2,7 @@ from optparse import make_option from django.contrib.gis import gdal from django.core.management.base import LabelCommand, CommandError + def layer_option(option, opt, value, parser): """ Callback for `make_option` for the `ogrinspect` `layer_key` @@ -13,6 +14,7 @@ def layer_option(option, opt, value, parser): dest = value setattr(parser.values, option.dest, dest) + def list_option(option, opt, value, parser): """ Callback for `make_option` for `ogrinspect` keywords that require @@ -25,6 +27,7 @@ def list_option(option, opt, value, parser): dest = [s for s in value.split(',')] setattr(parser.values, option.dest, dest) + class Command(LabelCommand): help = ('Inspects the given OGR-compatible data source (e.g., a shapefile) and outputs\n' 'a GeoDjango model with the given model name. For example:\n' diff --git a/django/contrib/gis/maps/google/gmap.py b/django/contrib/gis/maps/google/gmap.py index 590b9999af..ee820fc3c6 100644 --- a/django/contrib/gis/maps/google/gmap.py +++ b/django/contrib/gis/maps/google/gmap.py @@ -6,6 +6,7 @@ from django.utils.six.moves import xrange from django.contrib.gis.maps.google.overlays import GPolygon, GPolyline, GMarker + class GoogleMapException(Exception): pass @@ -154,6 +155,7 @@ class GoogleMap(object): "Returns a sequence of GIcon objects in this map." return set(marker.icon for marker in self.markers if marker.icon) + class GoogleMapSet(GoogleMap): def __init__(self, *args, **kwargs): diff --git a/django/contrib/gis/maps/google/overlays.py b/django/contrib/gis/maps/google/overlays.py index 6b201cd4e5..a2025b852e 100644 --- a/django/contrib/gis/maps/google/overlays.py +++ b/django/contrib/gis/maps/google/overlays.py @@ -54,6 +54,7 @@ class GEvent(object): "Returns the parameter part of a GEvent." return mark_safe('"%s", %s' % (self.event, self.action)) + @python_2_unicode_compatible class GOverlayBase(object): def __init__(self): @@ -71,6 +72,7 @@ class GOverlayBase(object): "The string representation is the JavaScript API call." return mark_safe('%s(%s)' % (self.__class__.__name__, self.js_params)) + class GPolygon(GOverlayBase): """ A Python wrapper for the Google GPolygon object. For more information @@ -130,6 +132,7 @@ class GPolygon(GOverlayBase): return '%s, "%s", %s, %s, "%s", %s' % (self.points, self.stroke_color, self.stroke_weight, self.stroke_opacity, self.fill_color, self.fill_opacity) + class GPolyline(GOverlayBase): """ A Python wrapper for the Google GPolyline object. For more information @@ -253,6 +256,7 @@ class GIcon(object): # equal hash(GIcon('varname')). return hash(self.__class__) ^ hash(self.varname) + class GMarker(GOverlayBase): """ A Python wrapper for the Google GMarker object. For more information diff --git a/django/contrib/gis/maps/google/zoom.py b/django/contrib/gis/maps/google/zoom.py index f4e8a8ec7c..6bbadcdf16 100644 --- a/django/contrib/gis/maps/google/zoom.py +++ b/django/contrib/gis/maps/google/zoom.py @@ -7,6 +7,7 @@ from math import pi, sin, log, exp, atan DTOR = pi / 180. RTOD = 180. / pi + class GoogleZoom(object): """ GoogleZoom is a utility for performing operations related to the zoom diff --git a/django/contrib/gis/sitemaps/views.py b/django/contrib/gis/sitemaps/views.py index 5dc3bc9273..8dad5c505c 100644 --- a/django/contrib/gis/sitemaps/views.py +++ b/django/contrib/gis/sitemaps/views.py @@ -37,6 +37,7 @@ def index(request, sitemaps): xml = loader.render_to_string('sitemap_index.xml', {'sitemaps': sites}) return HttpResponse(xml, content_type='application/xml') + def sitemap(request, sitemaps, section=None): """ This view generates a sitemap with additional geographic @@ -65,6 +66,7 @@ def sitemap(request, sitemaps, section=None): xml = loader.render_to_string('gis/sitemaps/geo_sitemap.xml', {'urlset': urls}) return HttpResponse(xml, content_type='application/xml') + def kml(request, label, model, field_name=None, compress=False, using=DEFAULT_DB_ALIAS): """ This view generates KML for the given app label, model, and field name. @@ -109,6 +111,7 @@ def kml(request, label, model, field_name=None, compress=False, using=DEFAULT_DB render = render_to_kml return render('gis/kml/placemarks.kml', {'places': placemarks}) + def kmz(request, label, model, field_name=None, using=DEFAULT_DB_ALIAS): """ This view returns KMZ for the given app label, model, and field name. diff --git a/django/contrib/gis/tests/distapp/models.py b/django/contrib/gis/tests/distapp/models.py index cc98d9b556..811c275faf 100644 --- a/django/contrib/gis/tests/distapp/models.py +++ b/django/contrib/gis/tests/distapp/models.py @@ -1,6 +1,7 @@ from django.contrib.gis.db import models from django.utils.encoding import python_2_unicode_compatible + @python_2_unicode_compatible class SouthTexasCity(models.Model): "City model on projected coordinate system for South Texas." @@ -11,6 +12,7 @@ class SouthTexasCity(models.Model): def __str__(self): return self.name + @python_2_unicode_compatible class SouthTexasCityFt(models.Model): "Same City model as above, but U.S. survey feet are the units." @@ -21,6 +23,7 @@ class SouthTexasCityFt(models.Model): def __str__(self): return self.name + @python_2_unicode_compatible class AustraliaCity(models.Model): "City model for Australia, using WGS84." @@ -31,6 +34,7 @@ class AustraliaCity(models.Model): def __str__(self): return self.name + @python_2_unicode_compatible class CensusZipcode(models.Model): "Model for a few South Texas ZIP codes (in original Census NAD83)." @@ -41,6 +45,7 @@ class CensusZipcode(models.Model): def __str__(self): return self.name + @python_2_unicode_compatible class SouthTexasZipcode(models.Model): "Model for a few South Texas ZIP codes." @@ -51,6 +56,7 @@ class SouthTexasZipcode(models.Model): def __str__(self): return self.name + @python_2_unicode_compatible class Interstate(models.Model): "Geodetic model for U.S. Interstates." @@ -61,6 +67,7 @@ class Interstate(models.Model): def __str__(self): return self.name + @python_2_unicode_compatible class SouthTexasInterstate(models.Model): "Projected model for South Texas Interstates." diff --git a/django/contrib/gis/tests/geo3d/models.py b/django/contrib/gis/tests/geo3d/models.py index 81e5f55f78..023c71ead1 100644 --- a/django/contrib/gis/tests/geo3d/models.py +++ b/django/contrib/gis/tests/geo3d/models.py @@ -1,6 +1,7 @@ from django.contrib.gis.db import models from django.utils.encoding import python_2_unicode_compatible + @python_2_unicode_compatible class City3D(models.Model): name = models.CharField(max_length=30) @@ -10,6 +11,7 @@ class City3D(models.Model): def __str__(self): return self.name + @python_2_unicode_compatible class Interstate2D(models.Model): name = models.CharField(max_length=30) @@ -19,6 +21,7 @@ class Interstate2D(models.Model): def __str__(self): return self.name + @python_2_unicode_compatible class Interstate3D(models.Model): name = models.CharField(max_length=30) @@ -28,6 +31,7 @@ class Interstate3D(models.Model): def __str__(self): return self.name + @python_2_unicode_compatible class InterstateProj2D(models.Model): name = models.CharField(max_length=30) @@ -37,6 +41,7 @@ class InterstateProj2D(models.Model): def __str__(self): return self.name + @python_2_unicode_compatible class InterstateProj3D(models.Model): name = models.CharField(max_length=30) @@ -46,6 +51,7 @@ class InterstateProj3D(models.Model): def __str__(self): return self.name + @python_2_unicode_compatible class Polygon2D(models.Model): name = models.CharField(max_length=30) @@ -55,6 +61,7 @@ class Polygon2D(models.Model): def __str__(self): return self.name + @python_2_unicode_compatible class Polygon3D(models.Model): name = models.CharField(max_length=30) @@ -64,14 +71,17 @@ class Polygon3D(models.Model): def __str__(self): return self.name + class Point2D(models.Model): point = models.PointField() objects = models.GeoManager() + class Point3D(models.Model): point = models.PointField(dim=3) objects = models.GeoManager() + class MultiPoint3D(models.Model): mpoint = models.MultiPointField(dim=3) objects = models.GeoManager() diff --git a/django/contrib/gis/tests/geoadmin/models.py b/django/contrib/gis/tests/geoadmin/models.py index f995bbf5c5..d1d671954a 100644 --- a/django/contrib/gis/tests/geoadmin/models.py +++ b/django/contrib/gis/tests/geoadmin/models.py @@ -2,6 +2,7 @@ from django.contrib.gis.db import models from django.contrib.gis import admin from django.utils.encoding import python_2_unicode_compatible + @python_2_unicode_compatible class City(models.Model): name = models.CharField(max_length=30) diff --git a/django/contrib/gis/tests/geoapp/feeds.py b/django/contrib/gis/tests/geoapp/feeds.py index a59593061f..f2ebc46109 100644 --- a/django/contrib/gis/tests/geoapp/feeds.py +++ b/django/contrib/gis/tests/geoapp/feeds.py @@ -18,6 +18,7 @@ class TestGeoRSS1(feeds.Feed): def item_geometry(self, item): return item.point + class TestGeoRSS2(TestGeoRSS1): def geometry(self, obj): # This should attach a element for the extent of @@ -30,9 +31,11 @@ class TestGeoRSS2(TestGeoRSS1): # Returning a simple tuple for the geometry. return item.point.x, item.point.y + class TestGeoAtom1(TestGeoRSS1): feed_type = feeds.GeoAtom1Feed + class TestGeoAtom2(TestGeoRSS2): feed_type = feeds.GeoAtom1Feed @@ -40,13 +43,16 @@ class TestGeoAtom2(TestGeoRSS2): # This time we'll use a 2-tuple of coordinates for the box. return ((-123.30, -41.32), (174.78, 48.46)) + class TestW3CGeo1(TestGeoRSS1): feed_type = feeds.W3CGeoFeed + # The following feeds are invalid, and will raise exceptions. class TestW3CGeo2(TestGeoRSS2): feed_type = feeds.W3CGeoFeed + class TestW3CGeo3(TestGeoRSS1): feed_type = feeds.W3CGeoFeed diff --git a/django/contrib/gis/tests/geoapp/models.py b/django/contrib/gis/tests/geoapp/models.py index 90ece4d96c..251173e432 100644 --- a/django/contrib/gis/tests/geoapp/models.py +++ b/django/contrib/gis/tests/geoapp/models.py @@ -5,6 +5,7 @@ from django.utils.encoding import python_2_unicode_compatible # MySQL spatial indices can't handle NULL geometries. null_flag = not mysql + @python_2_unicode_compatible class Country(models.Model): name = models.CharField(max_length=30) @@ -14,6 +15,7 @@ class Country(models.Model): def __str__(self): return self.name + @python_2_unicode_compatible class City(models.Model): name = models.CharField(max_length=30) @@ -23,12 +25,14 @@ class City(models.Model): def __str__(self): return self.name + # This is an inherited model from City class PennsylvaniaCity(City): county = models.CharField(max_length=30) founded = models.DateTimeField(null=True) objects = models.GeoManager() # TODO: This should be implicitly inherited. + @python_2_unicode_compatible class State(models.Model): name = models.CharField(max_length=30) @@ -38,6 +42,7 @@ class State(models.Model): def __str__(self): return self.name + @python_2_unicode_compatible class Track(models.Model): name = models.CharField(max_length=30) @@ -47,6 +52,7 @@ class Track(models.Model): def __str__(self): return self.name + class Truth(models.Model): val = models.BooleanField(default=False) objects = models.GeoManager() diff --git a/django/contrib/gis/tests/geogapp/models.py b/django/contrib/gis/tests/geogapp/models.py index 3b7efeebd3..1953bc5a29 100644 --- a/django/contrib/gis/tests/geogapp/models.py +++ b/django/contrib/gis/tests/geogapp/models.py @@ -1,6 +1,7 @@ from django.contrib.gis.db import models from django.utils.encoding import python_2_unicode_compatible + @python_2_unicode_compatible class City(models.Model): name = models.CharField(max_length=30) @@ -10,6 +11,7 @@ class City(models.Model): def __str__(self): return self.name + @python_2_unicode_compatible class Zipcode(models.Model): code = models.CharField(max_length=10) @@ -19,6 +21,7 @@ class Zipcode(models.Model): def __str__(self): return self.code + @python_2_unicode_compatible class County(models.Model): name = models.CharField(max_length=25) diff --git a/django/contrib/gis/tests/inspectapp/models.py b/django/contrib/gis/tests/inspectapp/models.py index e4c0f542a9..37c67a72fe 100644 --- a/django/contrib/gis/tests/inspectapp/models.py +++ b/django/contrib/gis/tests/inspectapp/models.py @@ -1,5 +1,6 @@ from django.contrib.gis.db import models + class AllOGRFields(models.Model): f_decimal = models.FloatField() f_float = models.FloatField() diff --git a/django/contrib/gis/tests/layermap/models.py b/django/contrib/gis/tests/layermap/models.py index 24048aa796..5e8b0879ed 100644 --- a/django/contrib/gis/tests/layermap/models.py +++ b/django/contrib/gis/tests/layermap/models.py @@ -1,20 +1,24 @@ from django.contrib.gis.db import models + class State(models.Model): name = models.CharField(max_length=20) objects = models.GeoManager() + class County(models.Model): name = models.CharField(max_length=25) state = models.ForeignKey(State) mpoly = models.MultiPolygonField(srid=4269) # Multipolygon in NAD83 objects = models.GeoManager() + class CountyFeat(models.Model): name = models.CharField(max_length=25) poly = models.PolygonField(srid=4269) objects = models.GeoManager() + class City(models.Model): name = models.CharField(max_length=25) name_txt = models.TextField(default='') @@ -24,12 +28,14 @@ class City(models.Model): point = models.PointField() objects = models.GeoManager() + class Interstate(models.Model): name = models.CharField(max_length=20) length = models.DecimalField(max_digits=6, decimal_places=2) path = models.LineStringField() objects = models.GeoManager() + # Same as `City` above, but for testing model inheritance. class CityBase(models.Model): name = models.CharField(max_length=25) @@ -38,12 +44,15 @@ class CityBase(models.Model): point = models.PointField() objects = models.GeoManager() + class ICity1(CityBase): dt = models.DateField() + class ICity2(ICity1): dt_time = models.DateTimeField(auto_now=True) + class Invalid(models.Model): point = models.PointField() diff --git a/django/contrib/gis/tests/layermap/tests.py b/django/contrib/gis/tests/layermap/tests.py index d352e96bd2..d42fe1d6ee 100644 --- a/django/contrib/gis/tests/layermap/tests.py +++ b/django/contrib/gis/tests/layermap/tests.py @@ -304,6 +304,7 @@ class LayerMapTest(TestCase): self.assertEqual(City.objects.count(), 1) self.assertEqual(City.objects.all()[0].name, "Zürich") + class OtherRouter(object): def db_for_read(self, model, **hints): return 'other' diff --git a/django/contrib/gis/tests/relatedapp/models.py b/django/contrib/gis/tests/relatedapp/models.py index 8c4112f38a..6d4b530c58 100644 --- a/django/contrib/gis/tests/relatedapp/models.py +++ b/django/contrib/gis/tests/relatedapp/models.py @@ -1,6 +1,7 @@ from django.contrib.gis.db import models from django.utils.encoding import python_2_unicode_compatible + @python_2_unicode_compatible class Location(models.Model): point = models.PointField() @@ -9,6 +10,7 @@ class Location(models.Model): def __str__(self): return self.point.wkt + @python_2_unicode_compatible class City(models.Model): name = models.CharField(max_length=50) @@ -19,15 +21,18 @@ class City(models.Model): def __str__(self): return self.name + class AugmentedLocation(Location): extra_text = models.TextField(blank=True) objects = models.GeoManager() + class DirectoryEntry(models.Model): listing_text = models.CharField(max_length=50) location = models.ForeignKey(AugmentedLocation) objects = models.GeoManager() + @python_2_unicode_compatible class Parcel(models.Model): name = models.CharField(max_length=30) @@ -42,17 +47,20 @@ class Parcel(models.Model): def __str__(self): return self.name + # These use the GeoManager but do not have any geographic fields. class Author(models.Model): name = models.CharField(max_length=100) dob = models.DateField() objects = models.GeoManager() + class Article(models.Model): title = models.CharField(max_length=100) author = models.ForeignKey(Author, unique=True) objects = models.GeoManager() + class Book(models.Model): title = models.CharField(max_length=100) author = models.ForeignKey(Author, related_name='books', null=True) diff --git a/django/contrib/gis/tests/test_measure.py b/django/contrib/gis/tests/test_measure.py index 58a1af0bf1..1c4fa448d2 100644 --- a/django/contrib/gis/tests/test_measure.py +++ b/django/contrib/gis/tests/test_measure.py @@ -264,12 +264,14 @@ class AreaTest(unittest.TestCase): self.assertEqual(repr(a1), 'Area(sq_m=100.0)') self.assertEqual(repr(a2), 'Area(sq_km=3.5)') + def suite(): s = unittest.TestSuite() s.addTest(unittest.makeSuite(DistanceTest)) s.addTest(unittest.makeSuite(AreaTest)) return s + def run(verbosity=2): unittest.TextTestRunner(verbosity=verbosity).run(suite()) diff --git a/django/contrib/gis/tests/test_spatialrefsys.py b/django/contrib/gis/tests/test_spatialrefsys.py index 281ec16672..8bc9177fb4 100644 --- a/django/contrib/gis/tests/test_spatialrefsys.py +++ b/django/contrib/gis/tests/test_spatialrefsys.py @@ -32,6 +32,7 @@ test_srs = ({'srid': 4326, }, ) + @unittest.skipUnless(HAS_GDAL and HAS_SPATIALREFSYS, "SpatialRefSysTest needs gdal support and a spatial database") class SpatialRefSysTest(unittest.TestCase): @@ -94,10 +95,12 @@ class SpatialRefSysTest(unittest.TestCase): for i in range(3): self.assertAlmostEqual(ellps1[i], ellps2[i], prec[i]) + def suite(): s = unittest.TestSuite() s.addTest(unittest.makeSuite(SpatialRefSysTest)) return s + def run(verbosity=2): unittest.TextTestRunner(verbosity=verbosity).run(suite()) diff --git a/django/contrib/gis/utils/layermapping.py b/django/contrib/gis/utils/layermapping.py index f46511c6c3..9d70026dbb 100644 --- a/django/contrib/gis/utils/layermapping.py +++ b/django/contrib/gis/utils/layermapping.py @@ -24,18 +24,23 @@ from django.utils.encoding import force_text class LayerMapError(Exception): pass + class InvalidString(LayerMapError): pass + class InvalidDecimal(LayerMapError): pass + class InvalidInteger(LayerMapError): pass + class MissingForeignKey(LayerMapError): pass + class LayerMapping(object): "A class that maps OGR Layers to GeoDjango Models." diff --git a/django/contrib/gis/utils/ogrinfo.py b/django/contrib/gis/utils/ogrinfo.py index 32f3371e42..a0804ecee3 100644 --- a/django/contrib/gis/utils/ogrinfo.py +++ b/django/contrib/gis/utils/ogrinfo.py @@ -7,6 +7,7 @@ produced by the `ogrinfo` utility. from django.contrib.gis.gdal import DataSource from django.contrib.gis.gdal.geometries import GEO_CLASSES + def ogrinfo(data_source, num_features=10): """ Walks the available layers in the supplied `data_source`, displaying diff --git a/django/contrib/gis/utils/ogrinspect.py b/django/contrib/gis/utils/ogrinspect.py index 565bb70645..c8b340d8ad 100644 --- a/django/contrib/gis/utils/ogrinspect.py +++ b/django/contrib/gis/utils/ogrinspect.py @@ -9,6 +9,7 @@ from django.contrib.gis.gdal import DataSource from django.contrib.gis.gdal.field import OFTDate, OFTDateTime, OFTInteger, OFTReal, OFTString, OFTTime from django.utils import six + def mapping(data_source, geom_name='geom', layer_key=0, multi_geom=False): """ Given a DataSource, generates a dictionary that may be used @@ -48,6 +49,7 @@ def mapping(data_source, geom_name='geom', layer_key=0, multi_geom=False): _mapping[geom_name] = prefix + str(gtype).upper() return _mapping + def ogrinspect(*args, **kwargs): """ Given a data source (either a string or a DataSource object) and a string @@ -118,6 +120,7 @@ def ogrinspect(*args, **kwargs): """ return '\n'.join(s for s in _ogrinspect(*args, **kwargs)) + def _ogrinspect(data_source, model_name, geom_name='geom', layer_key=0, srid=None, multi_geom=False, name_field=None, imports=True, decimal=False, blank=False, null=False): diff --git a/django/contrib/gis/utils/srs.py b/django/contrib/gis/utils/srs.py index 7b02219ac3..3e39bbdd36 100644 --- a/django/contrib/gis/utils/srs.py +++ b/django/contrib/gis/utils/srs.py @@ -1,5 +1,6 @@ from django.contrib.gis.gdal import SpatialReference + def add_srs_entry(srs, auth_name='EPSG', auth_srid=None, ref_sys_name=None, database=None): """ diff --git a/django/contrib/gis/utils/wkt.py b/django/contrib/gis/utils/wkt.py index 81df084297..65f2082d18 100644 --- a/django/contrib/gis/utils/wkt.py +++ b/django/contrib/gis/utils/wkt.py @@ -4,6 +4,7 @@ from django.utils import six + def precision_wkt(geom, prec): """ Returns WKT text of the geometry according to the given precision (an diff --git a/django/contrib/messages/storage/cookie.py b/django/contrib/messages/storage/cookie.py index 968f3f3605..2ec3fe484b 100644 --- a/django/contrib/messages/storage/cookie.py +++ b/django/contrib/messages/storage/cookie.py @@ -46,6 +46,7 @@ class MessageDecoder(json.JSONDecoder): decoded = super(MessageDecoder, self).decode(s, **kwargs) return self.process_messages(decoded) + class CookieStorage(BaseStorage): """ Stores messages in a cookie. diff --git a/django/contrib/messages/storage/fallback.py b/django/contrib/messages/storage/fallback.py index 6c35343eb7..24dd6561d0 100644 --- a/django/contrib/messages/storage/fallback.py +++ b/django/contrib/messages/storage/fallback.py @@ -2,6 +2,7 @@ from django.contrib.messages.storage.base import BaseStorage from django.contrib.messages.storage.cookie import CookieStorage from django.contrib.messages.storage.session import SessionStorage + class FallbackStorage(BaseStorage): """ Tries to store all messages in the first backend, storing any unstored diff --git a/django/contrib/messages/tests/test_mixins.py b/django/contrib/messages/tests/test_mixins.py index a24d580bd8..954bc8ec6a 100644 --- a/django/contrib/messages/tests/test_mixins.py +++ b/django/contrib/messages/tests/test_mixins.py @@ -2,6 +2,7 @@ from django.test import TestCase from django.contrib.messages.tests.urls import ContactFormViewWithMsg from django.core.urlresolvers import reverse + class SuccessMessageMixinTests(TestCase): urls = 'django.contrib.messages.tests.urls' diff --git a/django/contrib/messages/tests/urls.py b/django/contrib/messages/tests/urls.py index 0541b5a336..7f77636742 100644 --- a/django/contrib/messages/tests/urls.py +++ b/django/contrib/messages/tests/urls.py @@ -20,6 +20,7 @@ TEMPLATE = """{% if messages %} {% endif %} """ + @never_cache def add(request, message_type): # don't default to False here, because we want to test that it defaults @@ -35,6 +36,7 @@ def add(request, message_type): show_url = reverse('django.contrib.messages.tests.urls.show') return HttpResponseRedirect(show_url) + @never_cache def add_template_response(request, message_type): for msg in request.POST.getlist('messages'): @@ -43,11 +45,13 @@ def add_template_response(request, message_type): show_url = reverse('django.contrib.messages.tests.urls.show_template_response') return HttpResponseRedirect(show_url) + @never_cache def show(request): t = Template(TEMPLATE) return HttpResponse(t.render(RequestContext(request))) + @never_cache def show_template_response(request): return TemplateResponse(request, Template(TEMPLATE)) diff --git a/django/contrib/redirects/models.py b/django/contrib/redirects/models.py index b2257cf166..2051ac039a 100644 --- a/django/contrib/redirects/models.py +++ b/django/contrib/redirects/models.py @@ -3,6 +3,7 @@ from django.contrib.sites.models import Site from django.utils.translation import ugettext_lazy as _ from django.utils.encoding import python_2_unicode_compatible + @python_2_unicode_compatible class Redirect(models.Model): site = models.ForeignKey(Site) diff --git a/django/contrib/sessions/backends/base.py b/django/contrib/sessions/backends/base.py index 08e79530f4..2a497fe9a1 100644 --- a/django/contrib/sessions/backends/base.py +++ b/django/contrib/sessions/backends/base.py @@ -20,6 +20,7 @@ from django.contrib.sessions.exceptions import SuspiciousSession # on case insensitive file systems. VALID_KEY_CHARS = string.ascii_lowercase + string.digits + class CreateError(Exception): """ Used internally as a consistent exception type to catch from save (see the @@ -27,6 +28,7 @@ class CreateError(Exception): """ pass + class SessionBase(object): """ Base class for all Session classes. diff --git a/django/contrib/sessions/backends/db.py b/django/contrib/sessions/backends/db.py index 7be99c3e16..a087061c1f 100644 --- a/django/contrib/sessions/backends/db.py +++ b/django/contrib/sessions/backends/db.py @@ -6,6 +6,7 @@ from django.db import IntegrityError, transaction, router from django.utils import timezone from django.utils.encoding import force_text + class SessionStore(SessionBase): """ Implements database session store. diff --git a/django/contrib/sessions/backends/file.py b/django/contrib/sessions/backends/file.py index f47aa2d867..6569dafbe3 100644 --- a/django/contrib/sessions/backends/file.py +++ b/django/contrib/sessions/backends/file.py @@ -13,6 +13,7 @@ from django.utils.encoding import force_text from django.contrib.sessions.exceptions import InvalidSessionKey + class SessionStore(SessionBase): """ Implements a file based session store. diff --git a/django/contrib/sessions/middleware.py b/django/contrib/sessions/middleware.py index e8ebf3d5f9..211ef57d45 100644 --- a/django/contrib/sessions/middleware.py +++ b/django/contrib/sessions/middleware.py @@ -5,6 +5,7 @@ from django.conf import settings from django.utils.cache import patch_vary_headers from django.utils.http import cookie_date + class SessionMiddleware(object): def __init__(self): engine = import_module(settings.SESSION_ENGINE) diff --git a/django/contrib/sitemaps/tests/test_flatpages.py b/django/contrib/sitemaps/tests/test_flatpages.py index e034a457c2..75ebb10f96 100644 --- a/django/contrib/sitemaps/tests/test_flatpages.py +++ b/django/contrib/sitemaps/tests/test_flatpages.py @@ -6,6 +6,7 @@ from django.conf import settings from .base import SitemapTestsBase + class FlatpagesSitemapTests(SitemapTestsBase): @skipUnless("django.contrib.flatpages" in settings.INSTALLED_APPS, diff --git a/django/contrib/sitemaps/tests/test_https.py b/django/contrib/sitemaps/tests/test_https.py index baad02ac07..9594d0f7a4 100644 --- a/django/contrib/sitemaps/tests/test_https.py +++ b/django/contrib/sitemaps/tests/test_https.py @@ -6,6 +6,7 @@ from django.test.utils import override_settings from .base import SitemapTestsBase + class HTTPSSitemapTests(SitemapTestsBase): protocol = 'https' urls = 'django.contrib.sitemaps.tests.urls.https' diff --git a/django/contrib/sitemaps/tests/urls/https.py b/django/contrib/sitemaps/tests/urls/https.py index a1b4b939ae..db2ef28d82 100644 --- a/django/contrib/sitemaps/tests/urls/https.py +++ b/django/contrib/sitemaps/tests/urls/https.py @@ -2,6 +2,7 @@ from django.conf.urls import patterns from .http import SimpleSitemap + class HTTPSSitemap(SimpleSitemap): protocol = 'https' diff --git a/django/contrib/sitemaps/views.py b/django/contrib/sitemaps/views.py index 626154ad19..f310ee113f 100644 --- a/django/contrib/sitemaps/views.py +++ b/django/contrib/sitemaps/views.py @@ -9,6 +9,7 @@ from django.template.response import TemplateResponse from django.utils import six from django.utils.http import http_date + def x_robots_tag(func): @wraps(func) def inner(request, *args, **kwargs): @@ -17,6 +18,7 @@ def x_robots_tag(func): return response return inner + @x_robots_tag def index(request, sitemaps, template_name='sitemap_index.xml', content_type='application/xml', @@ -40,6 +42,7 @@ def index(request, sitemaps, return TemplateResponse(request, template_name, {'sitemaps': sites}, content_type=content_type) + @x_robots_tag def sitemap(request, sitemaps, section=None, template_name='sitemap.xml', content_type='application/xml'): diff --git a/django/contrib/sites/managers.py b/django/contrib/sites/managers.py index ceaccc7ae5..9cdc1ed413 100644 --- a/django/contrib/sites/managers.py +++ b/django/contrib/sites/managers.py @@ -2,6 +2,7 @@ from django.conf import settings from django.db import models from django.db.models.fields import FieldDoesNotExist + class CurrentSiteManager(models.Manager): "Use this to limit objects to those associated with the current site." def __init__(self, field_name=None): diff --git a/django/contrib/staticfiles/management/commands/findstatic.py b/django/contrib/staticfiles/management/commands/findstatic.py index eaf63f7179..6c12e48979 100644 --- a/django/contrib/staticfiles/management/commands/findstatic.py +++ b/django/contrib/staticfiles/management/commands/findstatic.py @@ -7,6 +7,7 @@ from django.utils.encoding import force_text from django.contrib.staticfiles import finders + class Command(LabelCommand): help = "Finds the absolute paths for the given static file(s)." args = "[file ...]" diff --git a/django/contrib/staticfiles/management/commands/runserver.py b/django/contrib/staticfiles/management/commands/runserver.py index 0f9e39f7c0..64c711ae54 100644 --- a/django/contrib/staticfiles/management/commands/runserver.py +++ b/django/contrib/staticfiles/management/commands/runserver.py @@ -5,6 +5,7 @@ from django.core.management.commands.runserver import Command as RunserverComman from django.contrib.staticfiles.handlers import StaticFilesHandler + class Command(RunserverCommand): option_list = RunserverCommand.option_list + ( make_option('--nostatic', action="store_false", dest='use_static_handler', default=True, diff --git a/django/contrib/staticfiles/utils.py b/django/contrib/staticfiles/utils.py index f500ed6d20..92bca71d57 100644 --- a/django/contrib/staticfiles/utils.py +++ b/django/contrib/staticfiles/utils.py @@ -3,6 +3,7 @@ import fnmatch from django.conf import settings from django.core.exceptions import ImproperlyConfigured + def matches_patterns(path, patterns=None): """ Return True or False depending on whether the ``path`` should be @@ -15,6 +16,7 @@ def matches_patterns(path, patterns=None): return True return False + def get_files(storage, ignore_patterns=None, location=''): """ Recursively walk the storage directories yielding the paths @@ -37,6 +39,7 @@ def get_files(storage, ignore_patterns=None, location=''): for fn in get_files(storage, ignore_patterns, dir): yield fn + def check_settings(base_url=None): """ Checks if the staticfiles settings have sane values. diff --git a/django/contrib/staticfiles/views.py b/django/contrib/staticfiles/views.py index 246a4b1aa4..5312de2931 100644 --- a/django/contrib/staticfiles/views.py +++ b/django/contrib/staticfiles/views.py @@ -13,6 +13,7 @@ from django.views import static from django.contrib.staticfiles import finders + def serve(request, path, insecure=False, **kwargs): """ Serve static files below a given point in the directory structure or diff --git a/django/contrib/webdesign/lorem_ipsum.py b/django/contrib/webdesign/lorem_ipsum.py index 36680c81a3..b91fecf92b 100644 --- a/django/contrib/webdesign/lorem_ipsum.py +++ b/django/contrib/webdesign/lorem_ipsum.py @@ -42,6 +42,7 @@ COMMON_WORDS = ('lorem', 'ipsum', 'dolor', 'sit', 'amet', 'consectetur', 'adipisicing', 'elit', 'sed', 'do', 'eiusmod', 'tempor', 'incididunt', 'ut', 'labore', 'et', 'dolore', 'magna', 'aliqua') + def sentence(): """ Returns a randomly generated sentence of lorem ipsum text. @@ -56,6 +57,7 @@ def sentence(): # Convert to sentence case and add end punctuation. return '%s%s%s' % (s[0].upper(), s[1:], random.choice('?.')) + def paragraph(): """ Returns a randomly generated paragraph of lorem ipsum text. @@ -64,6 +66,7 @@ def paragraph(): """ return ' '.join(sentence() for i in range(random.randint(1, 4))) + def paragraphs(count, common=True): """ Returns a list of paragraphs as returned by paragraph(). @@ -80,6 +83,7 @@ def paragraphs(count, common=True): paras.append(paragraph()) return paras + def words(count, common=True): """ Returns a string of `count` lorem ipsum words separated by a single space. diff --git a/django/contrib/webdesign/templatetags/webdesign.py b/django/contrib/webdesign/templatetags/webdesign.py index 055de8999d..2ff73e75d0 100644 --- a/django/contrib/webdesign/templatetags/webdesign.py +++ b/django/contrib/webdesign/templatetags/webdesign.py @@ -5,6 +5,7 @@ from django import template register = template.Library() + class LoremNode(template.Node): def __init__(self, count, method, common): self.count, self.method, self.common = count, method, common @@ -22,6 +23,7 @@ class LoremNode(template.Node): paras = ['

%s

' % p for p in paras] return '\n\n'.join(paras) + @register.tag def lorem(parser, token): """ diff --git a/django/core/cache/backends/db.py b/django/core/cache/backends/db.py index 84786f7cfc..a21777aaba 100644 --- a/django/core/cache/backends/db.py +++ b/django/core/cache/backends/db.py @@ -31,6 +31,7 @@ class Options(object): self.managed = True self.proxy = False + class BaseDatabaseCache(BaseCache): def __init__(self, table, params): BaseCache.__init__(self, params) @@ -40,6 +41,7 @@ class BaseDatabaseCache(BaseCache): _meta = Options(table) self.cache_model_class = CacheEntry + class DatabaseCache(BaseDatabaseCache): # This class uses cursors provided by the database connection. This means @@ -198,6 +200,7 @@ class DatabaseCache(BaseDatabaseCache): cursor = connections[db].cursor() cursor.execute('DELETE FROM %s' % table) + # For backwards compatibility class CacheClass(DatabaseCache): pass diff --git a/django/core/cache/backends/dummy.py b/django/core/cache/backends/dummy.py index 7ca6114ee4..5b6646da96 100644 --- a/django/core/cache/backends/dummy.py +++ b/django/core/cache/backends/dummy.py @@ -2,6 +2,7 @@ from django.core.cache.backends.base import BaseCache, DEFAULT_TIMEOUT + class DummyCache(BaseCache): def __init__(self, host, *args, **kwargs): BaseCache.__init__(self, *args, **kwargs) @@ -41,6 +42,7 @@ class DummyCache(BaseCache): def clear(self): pass + # For backwards compatibility class CacheClass(DummyCache): pass diff --git a/django/core/cache/backends/filebased.py b/django/core/cache/backends/filebased.py index acfd1c38ed..db9b242dd4 100644 --- a/django/core/cache/backends/filebased.py +++ b/django/core/cache/backends/filebased.py @@ -152,6 +152,7 @@ class FileBasedCache(BaseCache): except (IOError, OSError): pass + # For backwards compatibility class CacheClass(FileBasedCache): pass diff --git a/django/core/cache/backends/memcached.py b/django/core/cache/backends/memcached.py index edb756c365..9b2c5e8bd1 100644 --- a/django/core/cache/backends/memcached.py +++ b/django/core/cache/backends/memcached.py @@ -157,6 +157,7 @@ class BaseMemcachedCache(six.with_metaclass(BaseMemcachedCacheMethods, BaseCache def clear(self): self._cache.flush_all() + class MemcachedCache(BaseMemcachedCache): "An implementation of a cache binding using python-memcached" def __init__(self, server, params): @@ -171,6 +172,7 @@ class MemcachedCache(BaseMemcachedCache): self._client = self._lib.Client(self._servers, pickleProtocol=pickle.HIGHEST_PROTOCOL) return self._client + class PyLibMCCache(BaseMemcachedCache): "An implementation of a cache binding using pylibmc" def __init__(self, server, params): diff --git a/django/core/checks/compatibility/django_1_6_0.py b/django/core/checks/compatibility/django_1_6_0.py index 96b29d6b77..a00196f2c0 100644 --- a/django/core/checks/compatibility/django_1_6_0.py +++ b/django/core/checks/compatibility/django_1_6_0.py @@ -2,6 +2,7 @@ from __future__ import unicode_literals from django.db import models + def check_test_runner(): """ Checks if the user has *not* overridden the ``TEST_RUNNER`` setting & @@ -23,6 +24,7 @@ def check_test_runner(): ] return ' '.join(message) + def check_boolean_field_default_value(): """ Checks if there are any BooleanFields without a default value, & diff --git a/django/core/context_processors.py b/django/core/context_processors.py index 910ceb72d5..ae67ec7211 100644 --- a/django/core/context_processors.py +++ b/django/core/context_processors.py @@ -33,6 +33,7 @@ def csrf(request): return {'csrf_token': _get_val()} + def debug(request): "Returns context variables helpful for debugging." context_extras = {} @@ -42,6 +43,7 @@ def debug(request): context_extras['sql_queries'] = connection.queries return context_extras + def i18n(request): from django.utils import translation @@ -52,11 +54,13 @@ def i18n(request): return context_extras + def tz(request): from django.utils import timezone return {'TIME_ZONE': timezone.get_current_timezone_name()} + def static(request): """ Adds static-related context variables to the context. @@ -64,6 +68,7 @@ def static(request): """ return {'STATIC_URL': settings.STATIC_URL} + def media(request): """ Adds media-related context variables to the context. @@ -71,5 +76,6 @@ def media(request): """ return {'MEDIA_URL': settings.MEDIA_URL} + def request(request): return {'request': request} diff --git a/django/core/files/base.py b/django/core/files/base.py index 2d10100019..7abb94b1f2 100644 --- a/django/core/files/base.py +++ b/django/core/files/base.py @@ -8,6 +8,7 @@ from django.core.files.utils import FileProxyMixin from django.utils import six from django.utils.encoding import force_bytes, python_2_unicode_compatible + @python_2_unicode_compatible class File(FileProxyMixin): DEFAULT_CHUNK_SIZE = 64 * 2**10 @@ -128,6 +129,7 @@ class File(FileProxyMixin): def close(self): self.file.close() + @python_2_unicode_compatible class ContentFile(File): """ diff --git a/django/core/files/locks.py b/django/core/files/locks.py index 04ae67fd30..7a23e5c52d 100644 --- a/django/core/files/locks.py +++ b/django/core/files/locks.py @@ -39,6 +39,7 @@ try: except (ImportError, AttributeError): pass + def fd(f): """Get a filedescriptor from something which could be a file or an fd.""" return f.fileno() if hasattr(f, 'fileno') else f diff --git a/django/core/files/move.py b/django/core/files/move.py index 14b9b649d7..2462dbee2e 100644 --- a/django/core/files/move.py +++ b/django/core/files/move.py @@ -24,6 +24,7 @@ except ImportError: __all__ = ['file_move_safe'] + def _samefile(src, dst): # Macintosh, Unix. if hasattr(os.path, 'samefile'): @@ -36,6 +37,7 @@ def _samefile(src, dst): return (os.path.normcase(os.path.abspath(src)) == os.path.normcase(os.path.abspath(dst))) + def file_move_safe(old_file_name, new_file_name, chunk_size = 1024*64, allow_overwrite=False): """ Moves a file from one location to another in the safest way possible. diff --git a/django/core/files/storage.py b/django/core/files/storage.py index b05f3647c0..aae3d7e984 100644 --- a/django/core/files/storage.py +++ b/django/core/files/storage.py @@ -17,6 +17,7 @@ from django.utils._os import safe_join, abspathu __all__ = ('Storage', 'FileSystemStorage', 'DefaultStorage', 'default_storage') + class Storage(object): """ A base storage class, providing some default behaviors that all other @@ -142,6 +143,7 @@ class Storage(object): """ raise NotImplementedError('subclasses of Storage must provide a modified_time() method') + class FileSystemStorage(Storage): """ Standard filesystem storage @@ -292,9 +294,11 @@ class FileSystemStorage(Storage): def modified_time(self, name): return datetime.fromtimestamp(os.path.getmtime(self.path(name))) + def get_storage_class(import_path=None): return import_by_path(import_path or settings.DEFAULT_FILE_STORAGE) + class DefaultStorage(LazyObject): def _setup(self): self._wrapped = get_storage_class()() diff --git a/django/core/files/uploadhandler.py b/django/core/files/uploadhandler.py index d267871bbd..3bd6993771 100644 --- a/django/core/files/uploadhandler.py +++ b/django/core/files/uploadhandler.py @@ -17,12 +17,14 @@ __all__ = [ 'StopFutureHandlers' ] + class UploadFileException(Exception): """ Any error having to do with uploading files. """ pass + @python_2_unicode_compatible class StopUpload(UploadFileException): """ @@ -42,12 +44,14 @@ class StopUpload(UploadFileException): else: return 'StopUpload: Consume request data, then halt.' + class SkipFile(UploadFileException): """ This exception is raised by an upload handler that wants to skip a given file. """ pass + class StopFutureHandlers(UploadFileException): """ Upload handers that have handled a file and do not want future handlers to @@ -55,6 +59,7 @@ class StopFutureHandlers(UploadFileException): """ pass + class FileUploadHandler(object): """ Base class for streaming upload handlers. @@ -124,6 +129,7 @@ class FileUploadHandler(object): """ pass + class TemporaryFileUploadHandler(FileUploadHandler): """ Upload handler that streams data into a temporary file. @@ -146,6 +152,7 @@ class TemporaryFileUploadHandler(FileUploadHandler): self.file.size = file_size return self.file + class MemoryFileUploadHandler(FileUploadHandler): """ File upload handler to stream uploads into memory (used for small files). diff --git a/django/core/mail/backends/base.py b/django/core/mail/backends/base.py index 2be4510363..38f0523191 100644 --- a/django/core/mail/backends/base.py +++ b/django/core/mail/backends/base.py @@ -1,5 +1,6 @@ """Base email backend class.""" + class BaseEmailBackend(object): """ Base class for email backend implementations. diff --git a/django/core/mail/backends/console.py b/django/core/mail/backends/console.py index 1ec3d3afdc..d0ecac74ea 100644 --- a/django/core/mail/backends/console.py +++ b/django/core/mail/backends/console.py @@ -6,6 +6,7 @@ import threading from django.core.mail.backends.base import BaseEmailBackend + class EmailBackend(BaseEmailBackend): def __init__(self, *args, **kwargs): self.stream = kwargs.pop('stream', sys.stdout) diff --git a/django/core/mail/backends/dummy.py b/django/core/mail/backends/dummy.py index 7ae4878f13..7e47fe7564 100644 --- a/django/core/mail/backends/dummy.py +++ b/django/core/mail/backends/dummy.py @@ -4,6 +4,7 @@ Dummy email backend that does nothing. from django.core.mail.backends.base import BaseEmailBackend + class EmailBackend(BaseEmailBackend): def send_messages(self, email_messages): return len(list(email_messages)) diff --git a/django/core/mail/backends/filebased.py b/django/core/mail/backends/filebased.py index 9d1e6cb060..3f26551dab 100644 --- a/django/core/mail/backends/filebased.py +++ b/django/core/mail/backends/filebased.py @@ -8,6 +8,7 @@ from django.core.exceptions import ImproperlyConfigured from django.core.mail.backends.console import EmailBackend as ConsoleEmailBackend from django.utils import six + class EmailBackend(ConsoleEmailBackend): def __init__(self, *args, **kwargs): self._fname = None diff --git a/django/core/mail/backends/locmem.py b/django/core/mail/backends/locmem.py index 8e893af358..92f55f74c2 100644 --- a/django/core/mail/backends/locmem.py +++ b/django/core/mail/backends/locmem.py @@ -5,6 +5,7 @@ Backend for test environment. from django.core import mail from django.core.mail.backends.base import BaseEmailBackend + class EmailBackend(BaseEmailBackend): """A email backend for use during test sessions. diff --git a/django/core/management/__init__.py b/django/core/management/__init__.py index 373f9e927e..3516ac330b 100644 --- a/django/core/management/__init__.py +++ b/django/core/management/__init__.py @@ -17,6 +17,7 @@ from django import get_version # doesn't have to reload every time it's called. _commands = None + def find_commands(management_dir): """ Given a path to a management directory, returns a list of all the command @@ -31,6 +32,7 @@ def find_commands(management_dir): except OSError: return [] + def find_management_module(app_name): """ Determines the path to the management module for the given app_name, @@ -66,6 +68,7 @@ def find_management_module(app_name): f.close() return path + def load_command_class(app_name, name): """ Given a command name and an application name, returns the Command @@ -75,6 +78,7 @@ def load_command_class(app_name, name): module = import_module('%s.management.commands.%s' % (app_name, name)) return module.Command() + def get_commands(): """ Returns a dictionary mapping command names to their callback applications. @@ -121,6 +125,7 @@ def get_commands(): return _commands + def call_command(name, *args, **options): """ Calls the given command, with the given options and args/kwargs. @@ -158,6 +163,7 @@ def call_command(name, *args, **options): return klass.execute(*args, **defaults) + class LaxOptionParser(OptionParser): """ An option parser that doesn't raise any errors on unknown options. @@ -212,6 +218,7 @@ class LaxOptionParser(OptionParser): except: # Needed because we might need to catch a SystemExit largs.append(arg) + class ManagementUtility(object): """ Encapsulates the logic of the django-admin.py and manage.py utilities. @@ -400,6 +407,7 @@ class ManagementUtility(object): else: self.fetch_command(subcommand).run_from_argv(self.argv) + def execute_from_command_line(argv=None): """ A simple method that runs a ManagementUtility. diff --git a/django/core/management/color.py b/django/core/management/color.py index 46a1de2a9e..20e31fff93 100644 --- a/django/core/management/color.py +++ b/django/core/management/color.py @@ -7,6 +7,7 @@ import sys from django.utils import termcolors + def supports_color(): """ Returns True if the running system's terminal supports color, and False @@ -19,6 +20,7 @@ def supports_color(): return False return True + def color_style(): """Returns a Style object with the Django color scheme.""" if not supports_color(): @@ -43,6 +45,7 @@ def color_style(): style = no_style() return style + def no_style(): """Returns a Style object that has no colors.""" class dummy: diff --git a/django/core/management/commands/compilemessages.py b/django/core/management/commands/compilemessages.py index cb41492004..931d9948e1 100644 --- a/django/core/management/commands/compilemessages.py +++ b/django/core/management/commands/compilemessages.py @@ -8,6 +8,7 @@ from django.core.management.base import BaseCommand, CommandError from django.core.management.utils import find_command, popen_wrapper from django.utils._os import npath + def has_bom(fn): with open(fn, 'rb') as f: sample = f.read(4) @@ -15,6 +16,7 @@ def has_bom(fn): sample.startswith(codecs.BOM_UTF16_LE) or \ sample.startswith(codecs.BOM_UTF16_BE) + def compile_messages(stdout, locale=None): program = 'msgfmt' if find_command(program) is None: diff --git a/django/core/management/commands/dbshell.py b/django/core/management/commands/dbshell.py index cda1c8e990..866dab0ed3 100644 --- a/django/core/management/commands/dbshell.py +++ b/django/core/management/commands/dbshell.py @@ -3,6 +3,7 @@ from optparse import make_option from django.core.management.base import BaseCommand, CommandError from django.db import connections, DEFAULT_DB_ALIAS + class Command(BaseCommand): help = ("Runs the command-line client for specified database, or the " "default database if none is provided.") diff --git a/django/core/management/commands/diffsettings.py b/django/core/management/commands/diffsettings.py index 9e70e9ad8f..4d1f4cb5b3 100644 --- a/django/core/management/commands/diffsettings.py +++ b/django/core/management/commands/diffsettings.py @@ -2,10 +2,12 @@ from optparse import make_option from django.core.management.base import NoArgsCommand + def module_to_dict(module, omittable=lambda k: k.startswith('_')): """Converts a module namespace to a Python dictionary.""" return dict((k, repr(v)) for k, v in module.__dict__.items() if not omittable(k)) + class Command(NoArgsCommand): help = """Displays differences between the current settings.py and Django's default settings. Settings that don't appear in the defaults are diff --git a/django/core/management/commands/dumpdata.py b/django/core/management/commands/dumpdata.py index 03215ae41c..85817e9194 100644 --- a/django/core/management/commands/dumpdata.py +++ b/django/core/management/commands/dumpdata.py @@ -152,6 +152,7 @@ class Command(BaseCommand): raise raise CommandError("Unable to serialize database: %s" % e) + def sort_dependencies(app_list): """Sort a list of app,modellist pairs into a single list of models. diff --git a/django/core/management/commands/makemessages.py b/django/core/management/commands/makemessages.py index 6f837ae1e4..03ddface55 100644 --- a/django/core/management/commands/makemessages.py +++ b/django/core/management/commands/makemessages.py @@ -150,6 +150,7 @@ class TranslatableFile(object): if is_templatized: os.unlink(work_file) + def write_pot_file(potfile, msgs): """ Write the :param potfile: POT file with the :param msgs: contents, diff --git a/django/core/management/commands/runfcgi.py b/django/core/management/commands/runfcgi.py index 4e9331fc80..cf4a010bc0 100644 --- a/django/core/management/commands/runfcgi.py +++ b/django/core/management/commands/runfcgi.py @@ -2,6 +2,7 @@ import warnings from django.core.management.base import BaseCommand + class Command(BaseCommand): help = "Runs this project as a FastCGI application. Requires flup." args = '[various KEY=val options, use `runfcgi help` for help]' diff --git a/django/core/management/commands/sql.py b/django/core/management/commands/sql.py index 52b2058650..399458693e 100644 --- a/django/core/management/commands/sql.py +++ b/django/core/management/commands/sql.py @@ -6,6 +6,7 @@ from django.core.management.base import AppCommand from django.core.management.sql import sql_create from django.db import connections, DEFAULT_DB_ALIAS + class Command(AppCommand): help = "Prints the CREATE TABLE SQL statements for the given app name(s)." diff --git a/django/core/management/commands/sqlclear.py b/django/core/management/commands/sqlclear.py index ec2602d2a3..efcdbbca35 100644 --- a/django/core/management/commands/sqlclear.py +++ b/django/core/management/commands/sqlclear.py @@ -6,6 +6,7 @@ from django.core.management.base import AppCommand from django.core.management.sql import sql_delete from django.db import connections, DEFAULT_DB_ALIAS + class Command(AppCommand): help = "Prints the DROP TABLE SQL statements for the given app name(s)." diff --git a/django/core/management/commands/sqlcustom.py b/django/core/management/commands/sqlcustom.py index 0d46c4ec70..b71a3dd6d4 100644 --- a/django/core/management/commands/sqlcustom.py +++ b/django/core/management/commands/sqlcustom.py @@ -6,6 +6,7 @@ from django.core.management.base import AppCommand from django.core.management.sql import sql_custom from django.db import connections, DEFAULT_DB_ALIAS + class Command(AppCommand): help = "Prints the custom table modifying SQL statements for the given app name(s)." diff --git a/django/core/management/commands/sqldropindexes.py b/django/core/management/commands/sqldropindexes.py index 029378eaa6..d531808996 100644 --- a/django/core/management/commands/sqldropindexes.py +++ b/django/core/management/commands/sqldropindexes.py @@ -6,6 +6,7 @@ from django.core.management.base import AppCommand from django.core.management.sql import sql_destroy_indexes from django.db import connections, DEFAULT_DB_ALIAS + class Command(AppCommand): help = "Prints the DROP INDEX SQL statements for the given model module name(s)." diff --git a/django/core/management/commands/sqlflush.py b/django/core/management/commands/sqlflush.py index b98ecfd8ff..8ccc61e9a8 100644 --- a/django/core/management/commands/sqlflush.py +++ b/django/core/management/commands/sqlflush.py @@ -6,6 +6,7 @@ from django.core.management.base import NoArgsCommand from django.core.management.sql import sql_flush from django.db import connections, DEFAULT_DB_ALIAS + class Command(NoArgsCommand): help = "Returns a list of the SQL statements required to return all tables in the database to the state they were in just after they were installed." diff --git a/django/core/management/commands/sqlindexes.py b/django/core/management/commands/sqlindexes.py index f95d4f158c..2dc4f35dc6 100644 --- a/django/core/management/commands/sqlindexes.py +++ b/django/core/management/commands/sqlindexes.py @@ -6,6 +6,7 @@ from django.core.management.base import AppCommand from django.core.management.sql import sql_indexes from django.db import connections, DEFAULT_DB_ALIAS + class Command(AppCommand): help = "Prints the CREATE INDEX SQL statements for the given model module name(s)." diff --git a/django/core/management/commands/sqlinitialdata.py b/django/core/management/commands/sqlinitialdata.py index b9e2249c29..851baa86fe 100644 --- a/django/core/management/commands/sqlinitialdata.py +++ b/django/core/management/commands/sqlinitialdata.py @@ -1,5 +1,6 @@ from django.core.management.base import AppCommand, CommandError + class Command(AppCommand): help = "RENAMED: see 'sqlcustom'" diff --git a/django/core/management/commands/sqlsequencereset.py b/django/core/management/commands/sqlsequencereset.py index 7b9e85a9ee..8f4ea823c7 100644 --- a/django/core/management/commands/sqlsequencereset.py +++ b/django/core/management/commands/sqlsequencereset.py @@ -5,6 +5,7 @@ from optparse import make_option from django.core.management.base import AppCommand from django.db import connections, models, DEFAULT_DB_ALIAS + class Command(AppCommand): help = 'Prints the SQL statements for resetting sequences for the given app name(s).' diff --git a/django/core/management/commands/testserver.py b/django/core/management/commands/testserver.py index 97fc8ea8e1..182587b0f5 100644 --- a/django/core/management/commands/testserver.py +++ b/django/core/management/commands/testserver.py @@ -2,6 +2,7 @@ from django.core.management.base import BaseCommand from optparse import make_option + class Command(BaseCommand): option_list = BaseCommand.option_list + ( make_option('--noinput', action='store_false', dest='interactive', default=True, diff --git a/django/core/management/utils.py b/django/core/management/utils.py index bc007208f0..3a3badc93f 100644 --- a/django/core/management/utils.py +++ b/django/core/management/utils.py @@ -54,6 +54,7 @@ def handle_extensions(extensions=('html',), ignored=('py',)): ext_list[i] = '.%s' % ext_list[i] return set(x for x in ext_list if x.strip('.') not in ignored) + def find_command(cmd, path=None, pathext=None): if path is None: path = os.environ.get('PATH', []).split(os.pathsep) diff --git a/django/core/serializers/__init__.py b/django/core/serializers/__init__.py index d8e573a730..3720c770d5 100644 --- a/django/core/serializers/__init__.py +++ b/django/core/serializers/__init__.py @@ -90,6 +90,7 @@ def unregister_serializer(format): raise SerializerDoesNotExist(format) del _serializers[format] + def get_serializer(format): if not _serializers: _load_serializers() @@ -97,16 +98,19 @@ def get_serializer(format): raise SerializerDoesNotExist(format) return _serializers[format].Serializer + def get_serializer_formats(): if not _serializers: _load_serializers() return list(_serializers) + def get_public_serializer_formats(): if not _serializers: _load_serializers() return [k for k, v in six.iteritems(_serializers) if not v.Serializer.internal_use_only] + def get_deserializer(format): if not _serializers: _load_serializers() @@ -114,6 +118,7 @@ def get_deserializer(format): raise SerializerDoesNotExist(format) return _serializers[format].Deserializer + def serialize(format, queryset, **options): """ Serialize a queryset (or any iterator that returns database objects) using @@ -123,6 +128,7 @@ def serialize(format, queryset, **options): s.serialize(queryset, **options) return s.getvalue() + def deserialize(format, stream_or_string, **options): """ Deserialize a stream or a string. Returns an iterator that yields ``(obj, @@ -133,6 +139,7 @@ def deserialize(format, stream_or_string, **options): d = get_deserializer(format) return d(stream_or_string, **options) + def _load_serializers(): """ Register built-in and settings-defined serializers. This is done lazily so diff --git a/django/core/serializers/base.py b/django/core/serializers/base.py index 96ebd5ea53..b7d3e28a0d 100644 --- a/django/core/serializers/base.py +++ b/django/core/serializers/base.py @@ -6,18 +6,22 @@ import warnings from django.db import models from django.utils import six + class SerializerDoesNotExist(KeyError): """The requested serializer was not found.""" pass + class SerializationError(Exception): """Something bad happened during serialization.""" pass + class DeserializationError(Exception): """Something bad happened during deserialization.""" pass + class Serializer(object): """ Abstract serializer base class. @@ -117,6 +121,7 @@ class Serializer(object): if callable(getattr(self.stream, 'getvalue', None)): return self.stream.getvalue() + class Deserializer(six.Iterator): """ Abstract base deserializer class. @@ -143,6 +148,7 @@ class Deserializer(six.Iterator): """Iteration iterface -- return the next item in the stream""" raise NotImplementedError('subclasses of Deserializer must provide a __next__() method') + class DeserializedObject(object): """ A deserialized model. @@ -176,6 +182,7 @@ class DeserializedObject(object): # the m2m data twice. self.m2m_data = None + def build_instance(Model, data, db): """ Build a model instance. diff --git a/django/core/serializers/json.py b/django/core/serializers/json.py index fea5f3c087..72053e1d49 100644 --- a/django/core/serializers/json.py +++ b/django/core/serializers/json.py @@ -17,6 +17,7 @@ from django.core.serializers.python import Deserializer as PythonDeserializer from django.utils import six from django.utils.timezone import is_aware + class Serializer(PythonSerializer): """ Convert a queryset to JSON. diff --git a/django/core/serializers/python.py b/django/core/serializers/python.py index 89c0bdfa28..4ac7cc4cf1 100644 --- a/django/core/serializers/python.py +++ b/django/core/serializers/python.py @@ -147,6 +147,7 @@ def Deserializer(object_list, **options): obj = base.build_instance(Model, data, db) yield base.DeserializedObject(obj, m2m_data) + def _get_model(model_identifier): """ Helper to look up a model from an "app_label.model_name" string. diff --git a/django/core/serializers/pyyaml.py b/django/core/serializers/pyyaml.py index 478f14b53f..2d8bb187c1 100644 --- a/django/core/serializers/pyyaml.py +++ b/django/core/serializers/pyyaml.py @@ -29,6 +29,7 @@ class DjangoSafeDumper(SafeDumper): DjangoSafeDumper.add_representer(decimal.Decimal, DjangoSafeDumper.represent_decimal) + class Serializer(PythonSerializer): """ Convert a queryset to YAML. @@ -55,6 +56,7 @@ class Serializer(PythonSerializer): # Grand-parent super return super(PythonSerializer, self).getvalue() + def Deserializer(stream_or_string, **options): """ Deserialize a stream or string of YAML data. diff --git a/django/core/serializers/xml_serializer.py b/django/core/serializers/xml_serializer.py index b1b7357f6b..6870ac7d44 100644 --- a/django/core/serializers/xml_serializer.py +++ b/django/core/serializers/xml_serializer.py @@ -13,6 +13,7 @@ from xml.dom import pulldom from xml.sax import handler from xml.sax.expatreader import ExpatParser as _ExpatParser + class Serializer(base.Serializer): """ Serializes a QuerySet to XML. @@ -143,6 +144,7 @@ class Serializer(base.Serializer): "to": smart_text(field.rel.to._meta), }) + class Deserializer(base.Deserializer): """ Deserialize XML. diff --git a/django/core/servers/fastcgi.py b/django/core/servers/fastcgi.py index c42817c393..1dcd22723c 100644 --- a/django/core/servers/fastcgi.py +++ b/django/core/servers/fastcgi.py @@ -81,12 +81,14 @@ Examples: """ % FASTCGI_OPTIONS + def fastcgi_help(message=None): print(FASTCGI_HELP) if message: print(message) return False + def runfastcgi(argset=[], **kwargs): options = FASTCGI_OPTIONS.copy() options.update(kwargs) diff --git a/django/core/urlresolvers.py b/django/core/urlresolvers.py index d20f27e73f..723fe626be 100644 --- a/django/core/urlresolvers.py +++ b/django/core/urlresolvers.py @@ -71,12 +71,15 @@ class ResolverMatch(object): return "ResolverMatch(func=%s, args=%s, kwargs=%s, url_name='%s', app_name='%s', namespace='%s')" % ( self.func, self.args, self.kwargs, self.url_name, self.app_name, self.namespace) + class Resolver404(Http404): pass + class NoReverseMatch(Exception): pass + def get_callable(lookup_view, can_fail=False): """ Convert a string version of a function name to the callable object. @@ -118,6 +121,7 @@ def get_callable(lookup_view, can_fail=False): return lookup_view get_callable = memoize(get_callable, _callable_cache, 1) + def get_resolver(urlconf): if urlconf is None: from django.conf import settings @@ -125,6 +129,7 @@ def get_resolver(urlconf): return RegexURLResolver(r'^/', urlconf) get_resolver = memoize(get_resolver, _resolver_cache, 1) + def get_ns_resolver(ns_pattern, resolver): # Build a namespaced resolver for the given parent urlconf pattern. # This makes it possible to have captured parameters in the parent @@ -134,6 +139,7 @@ def get_ns_resolver(ns_pattern, resolver): return RegexURLResolver(r'^/', [ns_resolver]) get_ns_resolver = memoize(get_ns_resolver, _ns_resolver_cache, 2) + def get_mod_func(callback): # Converts 'django.views.news.stories.story_detail' to # ['django.views.news.stories', 'story_detail'] @@ -143,6 +149,7 @@ def get_mod_func(callback): return callback, '' return callback[:dot], callback[dot+1:] + class LocaleRegexProvider(object): """ A mixin to provide a default regex property which can vary by active @@ -228,6 +235,7 @@ class RegexURLPattern(LocaleRegexProvider): self._callback = get_callable(self._callback_str) return self._callback + class RegexURLResolver(LocaleRegexProvider): def __init__(self, regex, urlconf_name, default_kwargs=None, app_name=None, namespace=None): LocaleRegexProvider.__init__(self, regex) @@ -428,6 +436,7 @@ class RegexURLResolver(LocaleRegexProvider): "arguments '%s' not found. %d pattern(s) tried: %s" % (lookup_view_s, args, kwargs, len(patterns), patterns)) + class LocaleRegexURLResolver(RegexURLResolver): """ A URL resolver that always matches the active language code as URL prefix. @@ -447,11 +456,13 @@ class LocaleRegexURLResolver(RegexURLResolver): self._regex_dict[language_code] = regex_compiled return self._regex_dict[language_code] + def resolve(path, urlconf=None): if urlconf is None: urlconf = get_urlconf() return get_resolver(urlconf).resolve(path) + def reverse(viewname, urlconf=None, args=None, kwargs=None, prefix=None, current_app=None): if urlconf is None: urlconf = get_urlconf() @@ -510,6 +521,7 @@ def reverse(viewname, urlconf=None, args=None, kwargs=None, prefix=None, current reverse_lazy = lazy(reverse, str) + def clear_url_caches(): global _resolver_cache global _ns_resolver_cache @@ -518,6 +530,7 @@ def clear_url_caches(): _ns_resolver_cache.clear() _callable_cache.clear() + def set_script_prefix(prefix): """ Sets the script prefix for the current thread. @@ -526,6 +539,7 @@ def set_script_prefix(prefix): prefix += '/' _prefixes.value = prefix + def get_script_prefix(): """ Returns the currently active script prefix. Useful for client code that @@ -534,6 +548,7 @@ def get_script_prefix(): """ return getattr(_prefixes, "value", '/') + def clear_script_prefix(): """ Unsets the script prefix for the current thread. @@ -543,6 +558,7 @@ def clear_script_prefix(): except AttributeError: pass + def set_urlconf(urlconf_name): """ Sets the URLconf for the current thread (overriding the default one in @@ -554,6 +570,7 @@ def set_urlconf(urlconf_name): if hasattr(_urlconfs, "value"): del _urlconfs.value + def get_urlconf(default=None): """ Returns the root URLconf to use for the current thread if it has been @@ -561,6 +578,7 @@ def get_urlconf(default=None): """ return getattr(_urlconfs, "value", default) + def is_valid_path(path, urlconf=None): """ Returns True if the given path resolves against the default URL resolver, diff --git a/django/db/migrations/optimizer.py b/django/db/migrations/optimizer.py index 6b14292362..980eedddf7 100644 --- a/django/db/migrations/optimizer.py +++ b/django/db/migrations/optimizer.py @@ -1,5 +1,6 @@ from django.db import migrations + class MigrationOptimizer(object): """ Powers the optimization process, where you provide a list of Operations diff --git a/django/db/models/fields/related.py b/django/db/models/fields/related.py index f82fdc91b9..6299b20fba 100644 --- a/django/db/models/fields/related.py +++ b/django/db/models/fields/related.py @@ -18,6 +18,7 @@ from django import forms RECURSIVE_RELATIONSHIP_CONSTANT = 'self' + def add_lazy_relation(cls, field, relation, operation): """ Adds a lookup on ``cls`` when a related field is defined using a string, diff --git a/django/db/models/loading.py b/django/db/models/loading.py index 162f334846..125630bec8 100644 --- a/django/db/models/loading.py +++ b/django/db/models/loading.py @@ -18,6 +18,7 @@ __all__ = ('get_apps', 'get_app', 'get_models', 'get_model', 'register_models', MODELS_MODULE_NAME = 'models' + class ModelDict(OrderedDict): """ We need to special-case the deepcopy for this, as the keys are modules, @@ -27,6 +28,7 @@ class ModelDict(OrderedDict): return self.__class__([(key, copy.deepcopy(value, memo)) for key, value in self.items()]) + class UnavailableApp(Exception): pass diff --git a/django/db/models/options.py b/django/db/models/options.py index 523ca1a7bf..d4d72c81b5 100644 --- a/django/db/models/options.py +++ b/django/db/models/options.py @@ -25,6 +25,7 @@ DEFAULT_NAMES = ('verbose_name', 'verbose_name_plural', 'db_table', 'ordering', 'index_together', 'app_cache', 'default_permissions', 'select_on_save') + def normalize_unique_together(unique_together): """ unique_together can be either a tuple of tuples, or a single @@ -35,6 +36,7 @@ def normalize_unique_together(unique_together): unique_together = (unique_together,) return unique_together + @python_2_unicode_compatible class Options(object): def __init__(self, meta, app_label=None): diff --git a/django/dispatch/saferef.py b/django/dispatch/saferef.py index 5a5fd45c4b..c6fca2be7b 100644 --- a/django/dispatch/saferef.py +++ b/django/dispatch/saferef.py @@ -8,6 +8,7 @@ aren't handled by the core weakref module). import traceback import weakref + def safeRef(target, onDelete = None): """Return a *safe* weak reference to a callable target @@ -34,6 +35,7 @@ def safeRef(target, onDelete = None): else: return weakref.ref(target) + class BoundMethodWeakref(object): """'Safe' and reusable weak references to instance methods @@ -186,6 +188,7 @@ class BoundMethodWeakref(object): return function.__get__(target) return None + class BoundNonDescriptorMethodWeakref(BoundMethodWeakref): """A specialized BoundMethodWeakref, for platforms where instance methods are not descriptors. @@ -246,6 +249,7 @@ class BoundNonDescriptorMethodWeakref(BoundMethodWeakref): return getattr(target, function.__name__) return None + def get_bound_method_weakref(target, onDelete): """Instantiates the appropiate BoundMethodWeakRef, depending on the details of the underlying class method implementation""" diff --git a/django/forms/forms.py b/django/forms/forms.py index ddd51ffaae..51a3b928ae 100644 --- a/django/forms/forms.py +++ b/django/forms/forms.py @@ -23,6 +23,7 @@ __all__ = ('BaseForm', 'Form') NON_FIELD_ERRORS = '__all__' + def pretty_name(name): """Converts 'first_name' to 'First name'""" if not name: diff --git a/django/forms/formsets.py b/django/forms/formsets.py index 5b2efee82e..9c8b790d59 100644 --- a/django/forms/formsets.py +++ b/django/forms/formsets.py @@ -29,6 +29,7 @@ DEFAULT_MIN_NUM = 0 # default maximum number of forms in a formset, to prevent memory exhaustion DEFAULT_MAX_NUM = 1000 + class ManagementForm(Form): """ ``ManagementForm`` is used to keep track of how many form instances @@ -45,6 +46,7 @@ class ManagementForm(Form): self.base_fields[MAX_NUM_FORM_COUNT] = IntegerField(required=False, widget=HiddenInput) super(ManagementForm, self).__init__(*args, **kwargs) + @python_2_unicode_compatible class BaseFormSet(object): """ @@ -407,6 +409,7 @@ class BaseFormSet(object): forms = ' '.join(form.as_ul() for form in self) return mark_safe('\n'.join([six.text_type(self.management_form), forms])) + def formset_factory(form, formset=BaseFormSet, extra=1, can_order=False, can_delete=False, max_num=None, validate_max=False, min_num=None, validate_min=False): @@ -427,6 +430,7 @@ def formset_factory(form, formset=BaseFormSet, extra=1, can_order=False, 'validate_max': validate_max} return type(form.__name__ + str('FormSet'), (formset,), attrs) + def all_valid(formsets): """Returns true if every formset in formsets is valid.""" valid = True diff --git a/django/forms/utils.py b/django/forms/utils.py index 106074be5c..85297b1a1e 100644 --- a/django/forms/utils.py +++ b/django/forms/utils.py @@ -14,6 +14,7 @@ import sys # module to maintain backwards compatibility. from django.core.exceptions import ValidationError + def flatatt(attrs): """ Convert a dictionary of attributes to a single string. @@ -37,6 +38,7 @@ def flatatt(attrs): ) return format_html_join('', ' {0}="{1}"', sorted(attrs.items())) + @python_2_unicode_compatible class ErrorDict(dict): """ @@ -59,6 +61,7 @@ class ErrorDict(dict): def as_text(self): return '\n'.join('* %s\n%s' % (k, '\n'.join(' * %s' % force_text(i) for i in v)) for k, v in self.items()) + @python_2_unicode_compatible class ErrorList(list): """ @@ -84,6 +87,7 @@ class ErrorList(list): def __repr__(self): return repr([force_text(e) for e in self]) + # Utilities for time zone support in DateTimeField et al. def from_current_timezone(value): @@ -109,6 +113,7 @@ def from_current_timezone(value): ), sys.exc_info()[2]) return value + def to_current_timezone(value): """ When time zone support is enabled, convert aware datetimes diff --git a/django/forms/widgets.py b/django/forms/widgets.py index 55ccada4a1..88dc28d664 100644 --- a/django/forms/widgets.py +++ b/django/forms/widgets.py @@ -30,6 +30,7 @@ __all__ = ( MEDIA_TYPES = ('css', 'js') + @python_2_unicode_compatible class Media(object): def __init__(self, media=None, **kwargs): @@ -143,6 +144,7 @@ class MediaDefiningClass(type): return new_class + @python_2_unicode_compatible class SubWidget(object): """ diff --git a/django/http/multipartparser.py b/django/http/multipartparser.py index af0222b423..664a817f37 100644 --- a/django/http/multipartparser.py +++ b/django/http/multipartparser.py @@ -21,9 +21,11 @@ from django.core.files.uploadhandler import StopUpload, SkipFile, StopFutureHand __all__ = ('MultiPartParser', 'MultiPartParserError', 'InputStreamExhausted') + class MultiPartParserError(Exception): pass + class InputStreamExhausted(Exception): """ No more reads are allowed from this device. @@ -36,6 +38,7 @@ FIELD = "field" _BASE64_DECODE_ERROR = TypeError if six.PY2 else binascii.Error + class MultiPartParser(object): """ A rfc2388 multipart/form-data parser. diff --git a/django/middleware/cache.py b/django/middleware/cache.py index dbd5b80fb2..96c6b24034 100644 --- a/django/middleware/cache.py +++ b/django/middleware/cache.py @@ -112,6 +112,7 @@ class UpdateCacheMiddleware(object): self.cache.set(cache_key, response, timeout) return response + class FetchFromCacheMiddleware(object): """ Request-phase cache middleware that fetches a page from the cache. @@ -154,6 +155,7 @@ class FetchFromCacheMiddleware(object): request._cache_update_cache = False return response + class CacheMiddleware(UpdateCacheMiddleware, FetchFromCacheMiddleware): """ Cache middleware that provides basic behavior for many simple sites. diff --git a/django/middleware/clickjacking.py b/django/middleware/clickjacking.py index 81763ef41f..d9ee12d8a5 100644 --- a/django/middleware/clickjacking.py +++ b/django/middleware/clickjacking.py @@ -7,6 +7,7 @@ malicious site loading resources from your site in a hidden frame. from django.conf import settings + class XFrameOptionsMiddleware(object): """ Middleware that sets the X-Frame-Options HTTP header in HTTP responses. diff --git a/django/middleware/csrf.py b/django/middleware/csrf.py index 46b1891615..b28676ab18 100644 --- a/django/middleware/csrf.py +++ b/django/middleware/csrf.py @@ -26,6 +26,7 @@ REASON_BAD_TOKEN = "CSRF token missing or incorrect." CSRF_KEY_LENGTH = 32 + def _get_failure_view(): """ Returns the view to be used for CSRF rejections diff --git a/django/middleware/gzip.py b/django/middleware/gzip.py index fb54501a03..e9e955d704 100644 --- a/django/middleware/gzip.py +++ b/django/middleware/gzip.py @@ -5,6 +5,7 @@ from django.utils.cache import patch_vary_headers re_accepts_gzip = re.compile(r'\bgzip\b') + class GZipMiddleware(object): """ This middleware compresses content if the browser allows gzip compression. diff --git a/django/middleware/http.py b/django/middleware/http.py index 5a46e04946..9954267196 100644 --- a/django/middleware/http.py +++ b/django/middleware/http.py @@ -1,5 +1,6 @@ from django.utils.http import http_date, parse_http_date_safe + class ConditionalGetMiddleware(object): """ Handles conditional GET operations. If the response has a ETag or diff --git a/django/middleware/transaction.py b/django/middleware/transaction.py index 510176758a..677d1fc504 100644 --- a/django/middleware/transaction.py +++ b/django/middleware/transaction.py @@ -3,6 +3,7 @@ import warnings from django.core.exceptions import MiddlewareNotUsed from django.db import connection, transaction + class TransactionMiddleware(object): """ Transaction middleware. If this is enabled, each view function will be run diff --git a/django/shortcuts/__init__.py b/django/shortcuts/__init__.py index 36fe032cf2..17a860d87e 100644 --- a/django/shortcuts/__init__.py +++ b/django/shortcuts/__init__.py @@ -11,6 +11,7 @@ from django.db.models.manager import Manager from django.db.models.query import QuerySet from django.core import urlresolvers + def render_to_response(*args, **kwargs): """ Returns a HttpResponse whose content is filled with the result of calling @@ -20,6 +21,7 @@ def render_to_response(*args, **kwargs): return HttpResponse(loader.render_to_string(*args, **kwargs), **httpresponse_kwargs) + def render(request, *args, **kwargs): """ Returns a HttpResponse whose content is filled with the result of calling @@ -45,6 +47,7 @@ def render(request, *args, **kwargs): return HttpResponse(loader.render_to_string(*args, **kwargs), **httpresponse_kwargs) + def redirect(to, *args, **kwargs): """ Returns an HttpResponseRedirect to the appropriate URL for the arguments @@ -69,6 +72,7 @@ def redirect(to, *args, **kwargs): return redirect_class(resolve_url(to, *args, **kwargs)) + def _get_queryset(klass): """ Returns a QuerySet from a Model, Manager, or QuerySet. Created to make @@ -91,6 +95,7 @@ def _get_queryset(klass): "Manager, or QuerySet" % klass__name) return manager.all() + def get_object_or_404(klass, *args, **kwargs): """ Uses get() to return an object, or raises a Http404 exception if the object @@ -108,6 +113,7 @@ def get_object_or_404(klass, *args, **kwargs): except queryset.model.DoesNotExist: raise Http404('No %s matches the given query.' % queryset.model._meta.object_name) + def get_list_or_404(klass, *args, **kwargs): """ Uses filter() to return a list of objects, or raise a Http404 exception if @@ -122,6 +128,7 @@ def get_list_or_404(klass, *args, **kwargs): raise Http404('No %s matches the given query.' % queryset.model._meta.object_name) return obj_list + def resolve_url(to, *args, **kwargs): """ Return a URL appropriate for the arguments passed. diff --git a/django/template/base.py b/django/template/base.py index 5f6cac1c6f..ad13108e4e 100644 --- a/django/template/base.py +++ b/django/template/base.py @@ -71,15 +71,19 @@ builtins = [] # uninitialised. invalid_var_format_string = None + class TemplateSyntaxError(Exception): pass + class TemplateDoesNotExist(Exception): pass + class TemplateEncodingError(Exception): pass + @python_2_unicode_compatible class VariableDoesNotExist(Exception): diff --git a/django/template/context.py b/django/template/context.py index 70ed3f698b..d16a0a7533 100644 --- a/django/template/context.py +++ b/django/template/context.py @@ -8,6 +8,7 @@ _standard_context_processors = None # this to human error or failure to read migration instructions. _builtin_context_processors = ('django.core.context_processors.csrf',) + class ContextPopException(Exception): "pop() has been called more times than push()" pass diff --git a/django/template/debug.py b/django/template/debug.py index 9406551251..cf9c2a59fc 100644 --- a/django/template/debug.py +++ b/django/template/debug.py @@ -30,6 +30,7 @@ class DebugLexer(Lexer): token.source = self.origin, source return token + class DebugParser(Parser): def __init__(self, lexer): super(DebugParser, self).__init__(lexer) @@ -72,6 +73,7 @@ class DebugParser(Parser): if not hasattr(e, 'django_template_source'): e.django_template_source = token.source + class DebugNodeList(NodeList): def render_node(self, node, context): try: diff --git a/django/template/defaultfilters.py b/django/template/defaultfilters.py index bb3c9b320f..eb6f855cad 100644 --- a/django/template/defaultfilters.py +++ b/django/template/defaultfilters.py @@ -24,6 +24,7 @@ from django.utils.text import normalize_newlines register = Library() + ####################### # STRING DECORATOR # ####################### diff --git a/django/template/smartif.py b/django/template/smartif.py index 7cf1c6a023..4a9e2ef541 100644 --- a/django/template/smartif.py +++ b/django/template/smartif.py @@ -2,6 +2,7 @@ Parser and utilities for the smart 'if' tag """ + # Using a simple top down parser, as described here: # http://effbot.org/zone/simple-top-down-parsing.htm. # 'led' = left denotation diff --git a/django/templatetags/cache.py b/django/templatetags/cache.py index eb147b4210..0de00210fe 100644 --- a/django/templatetags/cache.py +++ b/django/templatetags/cache.py @@ -48,6 +48,7 @@ class CacheNode(Node): cache.set(cache_key, value, expire_time) return value + @register.tag('cache') def do_cache(parser, token): """ diff --git a/django/utils/_os.py b/django/utils/_os.py index 3e60c1082c..40a8e278fc 100644 --- a/django/utils/_os.py +++ b/django/utils/_os.py @@ -34,6 +34,7 @@ else: path = join(os.getcwdu(), path) return normpath(path) + def upath(path): """ Always return a unicode path. @@ -42,6 +43,7 @@ def upath(path): return path.decode(fs_encoding) return path + def npath(path): """ Always return a native path, that is unicode on Python 3 and bytestring on diff --git a/django/utils/autoreload.py b/django/utils/autoreload.py index fed2fd84b8..ca162d18d0 100644 --- a/django/utils/autoreload.py +++ b/django/utils/autoreload.py @@ -86,6 +86,7 @@ _win = (sys.platform == "win32") _error_files = [] + def gen_filenames(): """ Yields a generator over filenames referenced in sys.modules and translation diff --git a/django/utils/baseconv.py b/django/utils/baseconv.py index 053ce3e97e..f550c245d0 100644 --- a/django/utils/baseconv.py +++ b/django/utils/baseconv.py @@ -44,6 +44,7 @@ BASE36_ALPHABET = '0123456789abcdefghijklmnopqrstuvwxyz' BASE62_ALPHABET = '0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz' BASE64_ALPHABET = BASE62_ALPHABET + '-_' + class BaseConverter(object): decimal_digits = '0123456789' diff --git a/django/utils/checksums.py b/django/utils/checksums.py index 1113f3b796..01c1e68108 100644 --- a/django/utils/checksums.py +++ b/django/utils/checksums.py @@ -8,6 +8,7 @@ from django.utils import six LUHN_ODD_LOOKUP = (0, 2, 4, 6, 8, 1, 3, 5, 7, 9) # sum_of_digits(index * 2) + def luhn(candidate): """ Checks a candidate number for validity according to the Luhn diff --git a/django/utils/datetime_safe.py b/django/utils/datetime_safe.py index 286b5f127b..6aa32f30bf 100644 --- a/django/utils/datetime_safe.py +++ b/django/utils/datetime_safe.py @@ -11,10 +11,12 @@ from datetime import date as real_date, datetime as real_datetime import re import time + class date(real_date): def strftime(self, fmt): return strftime(self, fmt) + class datetime(real_datetime): def strftime(self, fmt): return strftime(self, fmt) diff --git a/django/utils/encoding.py b/django/utils/encoding.py index 6d1d575faf..ad4473df6a 100644 --- a/django/utils/encoding.py +++ b/django/utils/encoding.py @@ -9,6 +9,7 @@ from django.utils.functional import Promise from django.utils import six from django.utils.six.moves.urllib.parse import quote + class DjangoUnicodeDecodeError(UnicodeDecodeError): def __init__(self, obj, *args): self.obj = obj @@ -19,6 +20,7 @@ class DjangoUnicodeDecodeError(UnicodeDecodeError): return '%s. You passed in %r (%s)' % (original, self.obj, type(self.obj)) + def python_2_unicode_compatible(klass): """ A decorator that defines __unicode__ and __str__ methods under Python 2. diff --git a/django/utils/feedgenerator.py b/django/utils/feedgenerator.py index a5bdec22ee..416234521f 100644 --- a/django/utils/feedgenerator.py +++ b/django/utils/feedgenerator.py @@ -32,6 +32,7 @@ from django.utils.six import StringIO from django.utils.six.moves.urllib.parse import urlparse from django.utils.timezone import is_aware + def rfc2822_date(date): # We can't use strftime() because it produces locale-dependent results, so # we have to map english month and day names manually diff --git a/django/utils/html.py b/django/utils/html.py index 5b2416f03f..6d04ae5205 100644 --- a/django/utils/html.py +++ b/django/utils/html.py @@ -56,11 +56,13 @@ _js_escapes = { # Escape every ASCII character with a value less than 32. _js_escapes.update((ord('%c' % z), '\\u%04X' % z) for z in range(32)) + def escapejs(value): """Hex encodes characters for use in JavaScript strings.""" return mark_safe(force_text(value).translate(_js_escapes)) escapejs = allow_lazy(escapejs, six.text_type) + def conditional_escape(text): """ Similar to escape(), except that it doesn't operate on pre-escaped strings. diff --git a/django/utils/ipv6.py b/django/utils/ipv6.py index 4d5352272b..0aaa326c01 100644 --- a/django/utils/ipv6.py +++ b/django/utils/ipv6.py @@ -5,6 +5,7 @@ from django.core.exceptions import ValidationError from django.utils.translation import ugettext_lazy as _ from django.utils.six.moves import xrange + def clean_ipv6_address(ip_str, unpack_ipv4=False, error_message=_("This is not a valid IPv6 address.")): """ @@ -123,6 +124,7 @@ def _sanitize_ipv4_mapping(ip_str): return result + def _unpack_ipv4(ip_str): """ Unpack an IPv4 address that was mapped in a compressed IPv6 address. @@ -141,6 +143,7 @@ def _unpack_ipv4(ip_str): return ip_str.rsplit(':', 1)[1] + def is_valid_ipv6_address(ip_str): """ Ensure we have a valid IPv6 address. diff --git a/django/utils/regex_helper.py b/django/utils/regex_helper.py index ade1448f27..c8390cc7eb 100644 --- a/django/utils/regex_helper.py +++ b/django/utils/regex_helper.py @@ -26,6 +26,7 @@ ESCAPE_MAPPINGS = { "Z": None, } + class Choice(list): """ Used to represent multiple possibilities at this point in a pattern string. @@ -33,16 +34,19 @@ class Choice(list): code is clear. """ + class Group(list): """ Used to represent a capturing group in the pattern string. """ + class NonCapture(list): """ Used to represent a non-capturing group in the pattern string. """ + def normalize(pattern): """ Given a reg-exp pattern, normalizes it to an iterable of forms that diff --git a/django/utils/safestring.py b/django/utils/safestring.py index aee6427b3c..c44726090f 100644 --- a/django/utils/safestring.py +++ b/django/utils/safestring.py @@ -7,15 +7,18 @@ be interpreted by the HTML engine (e.g. '<') into the appropriate entities. from django.utils.functional import curry, Promise, allow_lazy from django.utils import six + class EscapeData(object): pass + class EscapeBytes(bytes, EscapeData): """ A byte string that should be HTML-escaped when output. """ __new__ = allow_lazy(bytes.__new__, bytes) + class EscapeText(six.text_type, EscapeData): """ A unicode string object that should be HTML-escaped when output. diff --git a/django/utils/tree.py b/django/utils/tree.py index 3f93738b4f..381dcdd06c 100644 --- a/django/utils/tree.py +++ b/django/utils/tree.py @@ -5,6 +5,7 @@ ORM. import copy + class Node(object): """ A single internal node in the tree graph. A Node should be viewed as a diff --git a/django/utils/tzinfo.py b/django/utils/tzinfo.py index 62c8e07a66..8d996567fe 100644 --- a/django/utils/tzinfo.py +++ b/django/utils/tzinfo.py @@ -50,6 +50,7 @@ class FixedOffset(tzinfo): def dst(self, dt): return timedelta(0) + # This implementation is used for display purposes. It uses an approximation # for DST computations on dates >= 2038. diff --git a/django/utils/version.py b/django/utils/version.py index e0a8286e48..6798749d43 100644 --- a/django/utils/version.py +++ b/django/utils/version.py @@ -4,6 +4,7 @@ import datetime import os import subprocess + def get_version(version=None): "Returns a PEP 386-compliant version number from VERSION." if version is None: @@ -32,6 +33,7 @@ def get_version(version=None): return str(main + sub) + def get_git_changeset(): """Returns a numeric identifier of the latest git changeset. diff --git a/django/utils/xmlutils.py b/django/utils/xmlutils.py index a09fcb0f6b..b3f7e4defb 100644 --- a/django/utils/xmlutils.py +++ b/django/utils/xmlutils.py @@ -4,6 +4,7 @@ Utilities for XML generation/parsing. from xml.sax.saxutils import XMLGenerator + class SimplerXMLGenerator(XMLGenerator): def addQuickElement(self, name, contents=None, attrs=None): "Convenience method for adding an element with no children"