Django 1.2 + Юг 0.7 + django-раздражает AutoOneToOneField приводит к TypeError: объект 'LegacyConnection' не повторяется - PullRequest
4 голосов
/ 06 мая 2010

Я использую транк Django 1.2 с South 0.7 и AutoOneToOneField, скопированный с django-раздражает. Саут жаловался, что в поле не определены правила, а в новой версии Саута больше нет автоматического анализатора типов полей. Поэтому я прочитал южную документацию и написал следующее определение (в основном, точную копию правил OneToOneField):

rules = [
  (
    (AutoOneToOneField),
    [],
    {
        "to": ["rel.to", {}],
        "to_field": ["rel.field_name", {"default_attr": "rel.to._meta.pk.name"}],
        "related_name": ["rel.related_name", {"default": None}],
        "db_index": ["db_index", {"default": True}],
    },
  )
]
from south.modelsinspector import add_introspection_rules
add_introspection_rules(rules, ["^myapp"]) 

Теперь Юг вызывает следующую ошибку, когда я делаю схему миграции.

Traceback (most recent call last):
  File "manage.py", line 11, in <module>
    execute_manager(settings)
  File "django/core/management/__init__.py", line 438, in execute_manager
    utility.execute()
  File "django/core/management/__init__.py", line 379, in execute
    self.fetch_command(subcommand).run_from_argv(self.argv)
  File "django/core/management/base.py", line 196, in run_from_argv
    self.execute(*args, **options.__dict__)
  File "django/core/management/base.py", line 223, in execute
    output = self.handle(*args, **options)
  File "South-0.7-py2.6.egg/south/management/commands/schemamigration.py", line 92, in handle
    (k, v) for k, v in freezer.freeze_apps([migrations.app_label()]).items()
  File "South-0.7-py2.6.egg/south/creator/freezer.py", line 33, in freeze_apps
    model_defs[model_key(model)] = prep_for_freeze(model)
  File "South-0.7-py2.6.egg/south/creator/freezer.py", line 65, in prep_for_freeze
    fields = modelsinspector.get_model_fields(model, m2m=True)
  File "South-0.7-py2.6.egg/south/modelsinspector.py", line 322, in get_model_fields
    args, kwargs = introspector(field)
  File "South-0.7-py2.6.egg/south/modelsinspector.py", line 271, in introspector
    arg_defs, kwarg_defs = matching_details(field)
  File "South-0.7-py2.6.egg/south/modelsinspector.py", line 187, in matching_details
    if any([isinstance(field, x) for x in classes]):
TypeError: 'LegacyConnection' object is not iterable

Связано ли это с недавним изменением в стволе Django 1.2? Как мне это исправить?

Я использую это поле следующим образом:

class Bar(models.Model):
    foo = AutoOneToOneField("foo.Foo", primary_key=True, related_name="bar")

Для справки код поля из django-tagging:

class AutoSingleRelatedObjectDescriptor(SingleRelatedObjectDescriptor):
    def __get__(self, instance, instance_type=None):
        try:
            return super(AutoSingleRelatedObjectDescriptor, self).__get__(instance, instance_type)
        except self.related.model.DoesNotExist:
            obj = self.related.model(**{self.related.field.name: instance})
            obj.save()
            return obj

class AutoOneToOneField(OneToOneField):
    def contribute_to_related_class(self, cls, related):
        setattr(cls, related.get_accessor_name(), AutoSingleRelatedObjectDescriptor(related))

Ответы [ 3 ]

5 голосов
/ 06 мая 2010

Попробуйте изменить эту строку

(AutoOneToOneField),

К этому:

(AutoOneToOneField,),

Кортеж, объявленный, как вы, не повторяется.

3 голосов
/ 06 мая 2010

Решил проблему, удалив правила и добавив следующий метод в AutoOneToOneField:

def south_field_triple(self):
    "Returns a suitable description of this field for South."
    from south.modelsinspector import introspector
    field_class = OneToOneField.__module__ + "." + OneToOneField.__name__
    args, kwargs = introspector(self)
    return (field_class, args, kwargs)
1 голос
/ 06 мая 2010

В вашем правиле есть простая проблема, связанная с Python. В кортеже вы должны добавить запятую, если внутри только один элемент.

Таким образом, измените (AutoOneToOneField) на (AutoOneToOneField,),

Но, честно говоря, я не знал, что могу использовать метод внутри поля вместо правил. Я приложу ваш патч и отправлю в раздражающий django репозиторий.

...