您在django中使用哪些方法设置数据库级别的默认值?
我理解这是一个django wontfix
决定,但如果为特定的后端(postgres)编写,必须有一些关于实现此方法的想法。
我想通过python启用它,所以我不想使用/sql/.sql
文件。
我在想我可能会在现有字段上设置属性,例如:
foo = models.CharField(max_length=256, blank=True)
foo.db_default = ''
然后过滤字段并构建ALTER COLUMN SET DEFAULT
查询。
虽然我会把它放在哪里?文档说不要改变post_syncdb
钩子中的数据库。
任何想法将这个动作放在哪里?我也在想我可以动态生成sql/.sql
个文件。管理命令总比没有好。自动会很棒。嗯,我想知道post_syncdb
是否可以及时生成sql文件来执行?将测试。
答案 0 :(得分:0)
我尝试过post_syncdb
但是使用.sql文件为时已晚。挂钩class_prepared
及时解决syncdb
,但我意识到我的事情过于复杂。我不妨在模型实例化时运行一些代码。
我最终得到了一个直接的解决方案,在我需要再次摆弄这个问题的6个月内可以轻松理解。
def db_default(field, db_default):
"""
Set attribute __db_default for signal to set default values with.
"""
field.__db_default = db_default
return field
class OrderShipLog(models.Model):
date = db_default(models.DateTimeField(), 'now()')
def generate_default_sql(model):
"""
Generate SQL for postgresql_psycopg2 default values because the shipping
database post doesn't guarantee posting certain fields but id like them all to be strings.
"""
db_fields = filter(lambda x: hasattr(x, '__db_default'), model._meta.fields)
db_table = model._meta.db_table
modelname = model._meta.object_name
sql_dir = os.path.join(os.path.dirname(__file__), 'sql/')
sql_filename = os.path.join(sql_dir, '{modelname}.postgresql_psycopg2.sql'.format(
modelname=modelname.lower()))
from django.db import connection
if not db_table in connection.introspection.table_names():
print >> sys.stderr, "{0} not in introspected table list; creating db default sql file : {1}".format(db_table, sql_filename)
try:
if not os.path.exists(sql_dir):
os.makedirs(sql_dir)
with open(sql_filename, 'w+') as f:
for field in db_fields:
attname, db_column = field.get_attname_column()
if not field.__db_default:
raise Exception("Must enter value for field {modelname}.{field}.__db_default".format(
modelname=modelname,
field=attname))
f.write('ALTER TABLE {db_table} ALTER COLUMN {db_column} SET DEFAULT {default};\n'.format(
db_table=db_table,
db_column=db_column,
default=field.__db_default,
))
except Exception, e:
print >> sys.stderr, "Could not generate SQL file. Manually set the defaults! {0}\n".format(e) * 10
generate_default_sql(OrderShipLog)