2011-10-18 02:45:22 +08:00
|
|
|
from __future__ import absolute_import
|
|
|
|
|
2011-09-18 03:54:52 +08:00
|
|
|
import os
|
2011-10-18 02:45:22 +08:00
|
|
|
|
2011-09-18 03:54:52 +08:00
|
|
|
from django.db import connections
|
|
|
|
from django.test import TestCase
|
2012-10-08 11:08:31 +08:00
|
|
|
from django.contrib.gis.gdal import Driver, GDAL_VERSION
|
2011-09-18 03:54:52 +08:00
|
|
|
from django.contrib.gis.geometry.test_data import TEST_DATA
|
|
|
|
from django.contrib.gis.utils.ogrinspect import ogrinspect
|
2011-10-18 02:45:22 +08:00
|
|
|
|
|
|
|
from .models import AllOGRFields
|
2011-09-18 03:54:52 +08:00
|
|
|
|
|
|
|
|
|
|
|
class OGRInspectTest(TestCase):
|
|
|
|
def test_poly(self):
|
|
|
|
shp_file = os.path.join(TEST_DATA, 'test_poly', 'test_poly.shp')
|
|
|
|
model_def = ogrinspect(shp_file, 'MyModel')
|
|
|
|
|
|
|
|
expected = [
|
|
|
|
'# This is an auto-generated Django model module created by ogrinspect.',
|
|
|
|
'from django.contrib.gis.db import models',
|
|
|
|
'',
|
|
|
|
'class MyModel(models.Model):',
|
|
|
|
' float = models.FloatField()',
|
|
|
|
' int = models.FloatField()',
|
|
|
|
' str = models.CharField(max_length=80)',
|
|
|
|
' geom = models.PolygonField(srid=-1)',
|
|
|
|
' objects = models.GeoManager()',
|
|
|
|
]
|
|
|
|
|
|
|
|
self.assertEqual(model_def, '\n'.join(expected))
|
|
|
|
|
|
|
|
def test_date_field(self):
|
|
|
|
shp_file = os.path.join(TEST_DATA, 'cities', 'cities.shp')
|
|
|
|
model_def = ogrinspect(shp_file, 'City')
|
|
|
|
|
|
|
|
expected = [
|
|
|
|
'# This is an auto-generated Django model module created by ogrinspect.',
|
|
|
|
'from django.contrib.gis.db import models',
|
|
|
|
'',
|
|
|
|
'class City(models.Model):',
|
|
|
|
' name = models.CharField(max_length=80)',
|
|
|
|
' population = models.FloatField()',
|
|
|
|
' density = models.FloatField()',
|
|
|
|
' created = models.DateField()',
|
|
|
|
' geom = models.PointField(srid=-1)',
|
|
|
|
' objects = models.GeoManager()',
|
|
|
|
]
|
|
|
|
|
|
|
|
self.assertEqual(model_def, '\n'.join(expected))
|
|
|
|
|
|
|
|
def test_time_field(self):
|
|
|
|
# Only possible to test this on PostGIS at the momemnt. MySQL
|
|
|
|
# complains about permissions, and SpatiaLite/Oracle are
|
|
|
|
# insanely difficult to get support compiled in for in GDAL.
|
|
|
|
if not connections['default'].ops.postgis:
|
|
|
|
return
|
|
|
|
|
|
|
|
# Getting the database identifier used by OGR, if None returned
|
|
|
|
# GDAL does not have the support compiled in.
|
|
|
|
ogr_db = get_ogr_db_string()
|
|
|
|
if not ogr_db:
|
|
|
|
return
|
|
|
|
|
|
|
|
# writing shapefules via GDAL currently does not support writing OGRTime
|
|
|
|
# fields, so we need to actually use a database
|
|
|
|
model_def = ogrinspect(ogr_db, 'Measurement',
|
|
|
|
layer_key=AllOGRFields._meta.db_table,
|
|
|
|
decimal=['f_decimal'])
|
|
|
|
|
|
|
|
expected = [
|
|
|
|
'# This is an auto-generated Django model module created by ogrinspect.',
|
|
|
|
'from django.contrib.gis.db import models',
|
|
|
|
'',
|
|
|
|
'class Measurement(models.Model):',
|
|
|
|
' f_decimal = models.DecimalField(max_digits=0, decimal_places=0)',
|
2012-10-08 11:08:31 +08:00
|
|
|
]
|
|
|
|
|
|
|
|
if GDAL_VERSION < (1, 9, 0):
|
|
|
|
# Prior to GDAL 1.9, the order of the model fields was not
|
|
|
|
# the same as the columns in the database.
|
|
|
|
expected.extend([
|
2011-09-18 03:54:52 +08:00
|
|
|
' f_int = models.IntegerField()',
|
|
|
|
' f_datetime = models.DateTimeField()',
|
|
|
|
' f_time = models.TimeField()',
|
|
|
|
' f_float = models.FloatField()',
|
|
|
|
' f_char = models.CharField(max_length=10)',
|
|
|
|
' f_date = models.DateField()',
|
2012-10-08 11:08:31 +08:00
|
|
|
])
|
|
|
|
else:
|
|
|
|
expected.extend([
|
|
|
|
' f_float = models.FloatField()',
|
|
|
|
' f_int = models.IntegerField()',
|
|
|
|
' f_char = models.CharField(max_length=10)',
|
|
|
|
' f_date = models.DateField()',
|
|
|
|
' f_datetime = models.DateTimeField()',
|
|
|
|
' f_time = models.TimeField()',
|
|
|
|
])
|
|
|
|
|
|
|
|
expected.extend([
|
2011-09-18 03:54:52 +08:00
|
|
|
' geom = models.PolygonField()',
|
|
|
|
' objects = models.GeoManager()',
|
2012-10-08 11:08:31 +08:00
|
|
|
])
|
2011-09-18 03:54:52 +08:00
|
|
|
|
|
|
|
self.assertEqual(model_def, '\n'.join(expected))
|
|
|
|
|
|
|
|
def get_ogr_db_string():
|
|
|
|
# Construct the DB string that GDAL will use to inspect the database.
|
|
|
|
# GDAL will create its own connection to the database, so we re-use the
|
|
|
|
# connection settings from the Django test. This approach is a bit fragile
|
|
|
|
# and cannot work on any other database other than PostgreSQL at the moment.
|
|
|
|
db = connections.databases['default']
|
|
|
|
|
|
|
|
# Map from the django backend into the OGR driver name and database identifier
|
|
|
|
# http://www.gdal.org/ogr/ogr_formats.html
|
|
|
|
#
|
|
|
|
# TODO: Support Oracle (OCI), MySQL, and SpatiaLite.
|
|
|
|
drivers = {
|
|
|
|
'django.contrib.gis.db.backends.postgis': ('PostgreSQL', 'PG'),
|
|
|
|
}
|
|
|
|
|
|
|
|
drv_name, db_str = drivers[db['ENGINE']]
|
|
|
|
|
|
|
|
# Ensure that GDAL library has driver support for the database.
|
|
|
|
try:
|
|
|
|
Driver(drv_name)
|
|
|
|
except:
|
|
|
|
return None
|
|
|
|
|
|
|
|
# Build the params of the OGR database connection string
|
|
|
|
# TODO: connection strings are database-dependent, thus if
|
|
|
|
# we ever test other backends, this will need to change.
|
|
|
|
params = ["dbname='%s'" % db['NAME']]
|
|
|
|
def add(key, template):
|
|
|
|
value = db.get(key, None)
|
|
|
|
# Don't add the parameter if it is not in django's settings
|
|
|
|
if value:
|
|
|
|
params.append(template % value)
|
|
|
|
add('HOST', "host='%s'")
|
|
|
|
add('PORT', "port='%s'")
|
|
|
|
add('USER', "user='%s'")
|
|
|
|
add('PASSWORD', "password='%s'")
|
|
|
|
|
|
|
|
return '%s:%s' % (db_str, ' '.join(params))
|