アプリの起動時に、yaml 記述ファイルから動的に django モデルを生成しています。それらをdjango adminに登録しましたが、管理インターフェイスで無効に表示されます。このように見えます。
yaml 形式の記述ファイルからモデルを動的に作成するための manage.py コマンドを作成しました。
from django.core.management.base import BaseCommand
from django.core.management import call_command
import yaml
from generation.generate import generate_fields, generate_model
class Command(BaseCommand):
help = 'Generates or updates django models from specified file'
def handle(self, *args, **options):
try:
f = open(args[0], 'r')
models = yaml.safe_load(f)
f.close()
for name, definition in models.items():
fields = generate_fields(definition['fields'])
generate_model(name, definition['title'], fields, "generation", "generation.models")
call_command('syncdb', interactive=True)
except IOError:
self.stderr.write('Cannot open file: {0}'.format(args[0]))
except yaml.YAMLError as e:
self.stderr.write('Cannot parse file {0}: {1}'.format(args[0], e))
次のように、ルート urls.py からこのコマンドを起動します。
from django.conf.urls import patterns, include, url
from django.contrib import admin
from generation import views
from generation.management.commands.updatemodels import Command
admin.autodiscover()
urlpatterns = patterns('',
url(r'^$', views.main, name='main'),
# url(r'^modgen/', include('modgen.foo.urls')),
url(r'^admin/', include(admin.site.urls)),
)
Command().handle('models.yaml')
モデルを生成し、次の方法で管理者に登録します。
from sys import stderr
from django.contrib import admin
from django.db import models
class Types():
INT = 'int'
CHAR = 'char'
DATE = 'date'
def generate_model(model_name, model_title, fields, app_label, module):
class Meta:
pass
if app_label:
setattr(Meta, 'app_label', app_label)
if model_title:
setattr(Meta, 'verbose_name_plural', model_title)
attrs = {'__module__': module, 'Meta': Meta}
if fields is not None:
attrs = dict(attrs.items() + fields.items())
model = type(model_name, (models.Model,), attrs)
admin.site.register(model)
def generate_fields(definition):
fields = {}
for field in definition:
fields[field['id']] = generate_field(field['title'], field['type'])
return fields
def generate_field(title, field_type):
if field_type == Types.CHAR:
return models.CharField(title, max_length=255)
elif field_type == Types.INT:
return models.IntegerField(title)
elif field_type == Types.DATE:
return models.DateField(title)
else:
stderr.write('Unknown field type specified: {0}'.format(field_type))
モデルが管理インターフェイスで無効に表示されるのはなぜですか? Django はそれらを正常に登録します。