2

我有 geodjango 的自定义字段:

from django.contrib.gis.db import models as geomodels

class PointField(geomodels.PointField):
    def formfield(self, **kwargs):
        defaults = {
            'form_class': CustomFormField
        }
        defaults.update(kwargs)
        return super(PointField, self).formfield(**defaults)


try:
    from south.modelsinspector import add_introspection_rules
    from south.introspection_plugins.geodjango import rules

    add_introspection_rules(rules, ["^project\.apps\.appname\.fields\.PointField"])
except ImportError:
    pass

和 Django-South 迁移:

class Migration(SchemaMigration):

    def forwards(self, orm):
        # Adding field 'Address.geoposition'
        db.add_column('company_address', 'geoposition',
                      self.gf('project.apps.appname.fields.PointField')(srid=900913, geography=True),
                      keep_default=False)


    def backwards(self, orm):
        # Deleting field 'Address.geoposition'
        db.delete_column('company_address', 'geoposition')

当执行此迁移时,我收到错误:

AddGeometryColumn() error: "duplicate column name: geoposition"
CreateSpatialIndex() error: either "company_address"."geoposition" isn't a Geometry column or a SpatialIndex is already defined

我已经没有想法了,不知道如何解决这个问题。

软件:

  • 姜戈 1.4
  • 空间视界 3.0.1
  • gdal 1.9.2
  • 南 0.7.6

db 的地理元数据已启动:

subprocess.call(["spatialite", settings.DATABASES['default']['NAME'], "SELECT InitSpatialMetaData();"])

并添加了 srs 条目:

from django.contrib.gis.utils import add_srs_entry
add_srs_entry(900913)

当我禁用迁移时,我没有收到任何错误。禁用迁移的代码:

SOUTH_MIGRATION_MODULES = {
    'appname': 'ignore',
}

但是我需要一切都可以很好地处理迁移。任何想法是什么导致了问题?

4

2 回答 2

2

深入研究代码,传统的 alter table 语句不适用于空间字段,它会尝试创建“None”列,然后使用AddGeometryColumn命令正确创建它。但是,South 似乎没有得到提示,因此AddGeometryColumn失败了,因为该列已经存在。

仍在努力寻找答案,但我希望它只是升级到新版本。我会及时向大家发布!

Django 源码

于 2014-07-17T05:42:03.210 回答
0

您正在尝试覆盖现有的模型字段,imo 您将不得不修补基类以使其工作。

例如,这样的事情应该可以解决问题

from django.contrib.gis.db.models import Address

class CustomPointField(geomodels.PointField):
    def formfield(self, **kwargs):
        defaults = {
            'form_class': CustomFormField
        }
        defaults.update(kwargs)
        return super(PointField, self).formfield(**defaults)

Address.add_to_class('geoposition', CustomPointField())

我不确定南是否会接受这个,所以你可能需要一个新的同步数据库或手动更改。

于 2013-02-16T15:04:22.573 回答