2005-07-20 23:00:34 +08:00
# Django management-related functions, including "CREATE TABLE" generation and
# development-server initialization.
import django
2005-12-06 13:17:52 +08:00
import os , re , sys , textwrap
from optparse import OptionParser
2005-07-20 23:00:34 +08:00
2006-02-05 04:08:30 +08:00
# For Python 2.3
if not hasattr ( __builtins__ , ' set ' ) :
from sets import Set as set
2005-07-20 23:00:34 +08:00
MODULE_TEMPLATE = ''' { %% if perms. %(app)s . %(addperm)s or perms. %(app)s . %(changeperm)s %% }
< tr >
2005-08-07 09:02:48 +08:00
< th > { % % if perms . % ( app ) s . % ( changeperm ) s % % } < a href = " %(app)s / %(mod)s / " > { % % endif % % } % ( name ) s { % % if perms . % ( app ) s . % ( changeperm ) s % % } < / a > { % % endif % % } < / th >
< td class = " x50 " > { % % if perms . % ( app ) s . % ( addperm ) s % % } < a href = " %(app)s / %(mod)s /add/ " class = " addlink " > { % % endif % % } Add { % % if perms . % ( app ) s . % ( addperm ) s % % } < / a > { % % endif % % } < / td >
< td class = " x75 " > { % % if perms . % ( app ) s . % ( changeperm ) s % % } < a href = " %(app)s / %(mod)s / " class = " changelink " > { % % endif % % } Change { % % if perms . % ( app ) s . % ( changeperm ) s % % } < / a > { % % endif % % } < / td >
2005-07-20 23:00:34 +08:00
< / tr >
{ % % endif % % } '''
2005-09-03 02:26:16 +08:00
APP_ARGS = ' [modelmodule ...] '
2005-07-20 23:00:34 +08:00
# Use django.__path__[0] because we don't know which directory django into
# which has been installed.
2005-10-06 22:30:35 +08:00
PROJECT_TEMPLATE_DIR = os . path . join ( django . __path__ [ 0 ] , ' conf ' , ' %s _template ' )
2005-07-20 23:00:34 +08:00
2005-12-06 13:17:52 +08:00
INVALID_PROJECT_NAMES = ( ' django ' , ' test ' )
2005-07-20 23:00:34 +08:00
def _get_packages_insert ( app_label ) :
2005-11-14 09:44:35 +08:00
from django . core . db import db
return " INSERT INTO %s ( %s , %s ) VALUES ( ' %s ' , ' %s ' ); " % \
( db . quote_name ( ' packages ' ) , db . quote_name ( ' label ' ) , db . quote_name ( ' name ' ) ,
app_label , app_label )
2005-07-20 23:00:34 +08:00
def _get_permission_codename ( action , opts ) :
return ' %s _ %s ' % ( action , opts . object_name . lower ( ) )
def _get_all_permissions ( opts ) :
" Returns (codename, name) for all permissions in the given opts. "
perms = [ ]
if opts . admin :
for action in ( ' add ' , ' change ' , ' delete ' ) :
perms . append ( ( _get_permission_codename ( action , opts ) , ' Can %s %s ' % ( action , opts . verbose_name ) ) )
return perms + list ( opts . permissions )
def _get_permission_insert ( name , codename , opts ) :
2005-11-14 09:44:35 +08:00
from django . core . db import db
return " INSERT INTO %s ( %s , %s , %s ) VALUES ( ' %s ' , ' %s ' , ' %s ' ); " % \
( db . quote_name ( ' auth_permissions ' ) , db . quote_name ( ' name ' ) , db . quote_name ( ' package ' ) ,
db . quote_name ( ' codename ' ) , name . replace ( " ' " , " ' ' " ) , opts . app_label , codename )
2005-07-20 23:00:34 +08:00
def _get_contenttype_insert ( opts ) :
2005-11-14 09:44:35 +08:00
from django . core . db import db
return " INSERT INTO %s ( %s , %s , %s ) VALUES ( ' %s ' , ' %s ' , ' %s ' ); " % \
( db . quote_name ( ' content_types ' ) , db . quote_name ( ' name ' ) , db . quote_name ( ' package ' ) ,
db . quote_name ( ' python_module_name ' ) , opts . verbose_name , opts . app_label , opts . module_name )
2005-07-20 23:00:34 +08:00
def _is_valid_dir_name ( s ) :
return bool ( re . search ( r ' ^ \ w+$ ' , s ) )
2006-01-15 13:48:13 +08:00
# If the foreign key points to an AutoField, a PositiveIntegerField or a
# PositiveSmallIntegerField, the foreign key should be an IntegerField, not the
# referred field type. Otherwise, the foreign key should be the same type of
# field as the field to which it points.
get_rel_data_type = lambda f : ( f . get_internal_type ( ) in ( ' AutoField ' , ' PositiveIntegerField ' , ' PositiveSmallIntegerField ' ) ) and ' IntegerField ' or f . get_internal_type ( )
2005-09-26 02:29:59 +08:00
2005-07-20 23:00:34 +08:00
def get_sql_create ( mod ) :
" Returns a list of the CREATE TABLE SQL statements for the given module. "
from django . core import db , meta
final_output = [ ]
for klass in mod . _MODELS :
opts = klass . _meta
table_output = [ ]
for f in opts . fields :
if isinstance ( f , meta . ForeignKey ) :
rel_field = f . rel . get_related_field ( )
2005-09-26 02:29:59 +08:00
data_type = get_rel_data_type ( rel_field )
2005-07-20 23:00:34 +08:00
else :
rel_field = f
2005-09-29 07:08:47 +08:00
data_type = f . get_internal_type ( )
2005-07-20 23:00:34 +08:00
col_type = db . DATA_TYPES [ data_type ]
if col_type is not None :
2005-11-14 09:44:35 +08:00
field_output = [ db . db . quote_name ( f . column ) , col_type % rel_field . __dict__ ]
2005-07-20 23:00:34 +08:00
field_output . append ( ' %s NULL ' % ( not f . null and ' NOT ' or ' ' ) )
if f . unique :
field_output . append ( ' UNIQUE ' )
if f . primary_key :
field_output . append ( ' PRIMARY KEY ' )
if f . rel :
field_output . append ( ' REFERENCES %s ( %s ) ' % \
2005-11-14 09:44:35 +08:00
( db . db . quote_name ( f . rel . to . db_table ) ,
db . db . quote_name ( f . rel . to . get_field ( f . rel . field_name ) . column ) ) )
2005-07-20 23:00:34 +08:00
table_output . append ( ' ' . join ( field_output ) )
if opts . order_with_respect_to :
2005-11-14 09:44:35 +08:00
table_output . append ( ' %s %s NULL ' % ( db . db . quote_name ( ' _order ' ) , db . DATA_TYPES [ ' IntegerField ' ] ) )
2005-07-20 23:00:34 +08:00
for field_constraints in opts . unique_together :
2005-11-14 09:44:35 +08:00
table_output . append ( ' UNIQUE ( %s ) ' % \
" , " . join ( [ db . db . quote_name ( opts . get_field ( f ) . column ) for f in field_constraints ] ) )
2005-07-20 23:00:34 +08:00
2005-11-14 09:44:35 +08:00
full_statement = [ ' CREATE TABLE %s ( ' % db . db . quote_name ( opts . db_table ) ]
2005-07-20 23:00:34 +08:00
for i , line in enumerate ( table_output ) : # Combine and add commas.
full_statement . append ( ' %s %s ' % ( line , i < len ( table_output ) - 1 and ' , ' or ' ' ) )
full_statement . append ( ' ); ' )
final_output . append ( ' \n ' . join ( full_statement ) )
for klass in mod . _MODELS :
opts = klass . _meta
for f in opts . many_to_many :
2005-11-14 09:44:35 +08:00
table_output = [ ' CREATE TABLE %s ( ' % db . db . quote_name ( f . get_m2m_db_table ( opts ) ) ]
table_output . append ( ' %s %s NOT NULL PRIMARY KEY, ' % ( db . db . quote_name ( ' id ' ) , db . DATA_TYPES [ ' AutoField ' ] ) )
table_output . append ( ' %s %s NOT NULL REFERENCES %s ( %s ), ' % \
( db . db . quote_name ( opts . object_name . lower ( ) + ' _id ' ) ,
db . DATA_TYPES [ get_rel_data_type ( opts . pk ) ] % opts . pk . __dict__ ,
db . db . quote_name ( opts . db_table ) ,
db . db . quote_name ( opts . pk . column ) ) )
table_output . append ( ' %s %s NOT NULL REFERENCES %s ( %s ), ' % \
( db . db . quote_name ( f . rel . to . object_name . lower ( ) + ' _id ' ) ,
db . DATA_TYPES [ get_rel_data_type ( f . rel . to . pk ) ] % f . rel . to . pk . __dict__ ,
db . db . quote_name ( f . rel . to . db_table ) ,
db . db . quote_name ( f . rel . to . pk . column ) ) )
table_output . append ( ' UNIQUE ( %s , %s ) ' % \
( db . db . quote_name ( opts . object_name . lower ( ) + ' _id ' ) ,
db . db . quote_name ( f . rel . to . object_name . lower ( ) + ' _id ' ) ) )
2005-07-20 23:00:34 +08:00
table_output . append ( ' ); ' )
final_output . append ( ' \n ' . join ( table_output ) )
return final_output
2005-09-03 02:26:16 +08:00
get_sql_create . help_doc = " Prints the CREATE TABLE SQL statements for the given model module name(s). "
2005-07-20 23:00:34 +08:00
get_sql_create . args = APP_ARGS
def get_sql_delete ( mod ) :
" Returns a list of the DROP TABLE SQL statements for the given module. "
from django . core import db
try :
cursor = db . db . cursor ( )
except :
cursor = None
2005-11-10 07:50:32 +08:00
# Determine whether the admin log table exists. It only exists if the
# person has installed the admin app.
try :
if cursor is not None :
# Check whether the table exists.
2005-11-14 09:44:35 +08:00
cursor . execute ( " SELECT 1 FROM %s LIMIT 1 " % db . db . quote_name ( ' django_admin_log ' ) )
2005-11-10 07:50:32 +08:00
except :
# The table doesn't exist, so it doesn't need to be dropped.
db . db . rollback ( )
admin_log_exists = False
else :
admin_log_exists = True
2005-07-20 23:00:34 +08:00
output = [ ]
2005-11-10 07:50:32 +08:00
# Output DROP TABLE statements for standard application tables.
2005-07-20 23:00:34 +08:00
for klass in mod . _MODELS :
try :
if cursor is not None :
# Check whether the table exists.
2005-11-14 09:44:35 +08:00
cursor . execute ( " SELECT 1 FROM %s LIMIT 1 " % db . db . quote_name ( klass . _meta . db_table ) )
2005-07-20 23:00:34 +08:00
except :
# The table doesn't exist, so it doesn't need to be dropped.
2005-07-27 07:18:08 +08:00
db . db . rollback ( )
2005-07-20 23:00:34 +08:00
else :
2005-11-14 09:44:35 +08:00
output . append ( " DROP TABLE %s ; " % db . db . quote_name ( klass . _meta . db_table ) )
2005-11-10 07:50:32 +08:00
# Output DROP TABLE statements for many-to-many tables.
2005-07-20 23:00:34 +08:00
for klass in mod . _MODELS :
opts = klass . _meta
for f in opts . many_to_many :
try :
if cursor is not None :
2005-11-14 09:44:35 +08:00
cursor . execute ( " SELECT 1 FROM %s LIMIT 1 " % db . db . quote_name ( f . get_m2m_db_table ( opts ) ) )
2005-07-20 23:00:34 +08:00
except :
2005-07-27 07:18:08 +08:00
db . db . rollback ( )
2005-07-20 23:00:34 +08:00
else :
2005-11-14 09:44:35 +08:00
output . append ( " DROP TABLE %s ; " % db . db . quote_name ( f . get_m2m_db_table ( opts ) ) )
2005-07-27 06:38:54 +08:00
app_label = mod . _MODELS [ 0 ] . _meta . app_label
# Delete from packages, auth_permissions, content_types.
2005-11-14 09:44:35 +08:00
output . append ( " DELETE FROM %s WHERE %s = ' %s ' ; " % \
( db . db . quote_name ( ' packages ' ) , db . db . quote_name ( ' label ' ) , app_label ) )
output . append ( " DELETE FROM %s WHERE %s = ' %s ' ; " % \
( db . db . quote_name ( ' auth_permissions ' ) , db . db . quote_name ( ' package ' ) , app_label ) )
output . append ( " DELETE FROM %s WHERE %s = ' %s ' ; " % \
( db . db . quote_name ( ' content_types ' ) , db . db . quote_name ( ' package ' ) , app_label ) )
2005-07-27 06:38:54 +08:00
# Delete from the admin log.
if cursor is not None :
2005-11-14 09:44:35 +08:00
cursor . execute ( " SELECT %s FROM %s WHERE %s = %% s " % \
( db . db . quote_name ( ' id ' ) , db . db . quote_name ( ' content_types ' ) ,
db . db . quote_name ( ' package ' ) ) , [ app_label ] )
2005-11-10 07:50:32 +08:00
if admin_log_exists :
for row in cursor . fetchall ( ) :
2005-11-14 09:44:35 +08:00
output . append ( " DELETE FROM %s WHERE %s = %s ; " % \
( db . db . quote_name ( ' django_admin_log ' ) , db . db . quote_name ( ' content_type_id ' ) , row [ 0 ] ) )
2005-07-27 06:38:54 +08:00
2005-10-11 04:14:56 +08:00
# Close database connection explicitly, in case this output is being piped
# directly into a database client, to avoid locking issues.
2006-02-02 12:50:16 +08:00
if cursor is not None :
cursor . close ( )
db . db . close ( )
2005-10-11 04:14:56 +08:00
2005-07-20 23:00:34 +08:00
return output [ : : - 1 ] # Reverse it, to deal with table dependencies.
2005-09-03 02:26:16 +08:00
get_sql_delete . help_doc = " Prints the DROP TABLE SQL statements for the given model module name(s). "
2005-07-20 23:00:34 +08:00
get_sql_delete . args = APP_ARGS
def get_sql_reset ( mod ) :
" Returns a list of the DROP TABLE SQL, then the CREATE TABLE SQL, for the given module. "
return get_sql_delete ( mod ) + get_sql_all ( mod )
2005-09-03 02:26:16 +08:00
get_sql_reset . help_doc = " Prints the DROP TABLE SQL, then the CREATE TABLE SQL, for the given model module name(s). "
2005-07-20 23:00:34 +08:00
get_sql_reset . args = APP_ARGS
def get_sql_initial_data ( mod ) :
" Returns a list of the initial INSERT SQL statements for the given module. "
2005-09-30 07:43:03 +08:00
from django . core import db
2005-07-20 23:00:34 +08:00
output = [ ]
app_label = mod . _MODELS [ 0 ] . _meta . app_label
output . append ( _get_packages_insert ( app_label ) )
2005-10-06 22:30:35 +08:00
app_dir = os . path . normpath ( os . path . join ( os . path . dirname ( mod . __file__ ) , ' .. ' , ' sql ' ) )
2005-07-20 23:00:34 +08:00
for klass in mod . _MODELS :
opts = klass . _meta
2005-09-30 07:43:03 +08:00
2005-07-20 23:00:34 +08:00
# Add custom SQL, if it's available.
2005-09-30 07:43:03 +08:00
sql_files = [ os . path . join ( app_dir , opts . module_name + ' . ' + db . DATABASE_ENGINE + ' .sql ' ) ,
os . path . join ( app_dir , opts . module_name + ' .sql ' ) ]
for sql_file in sql_files :
if os . path . exists ( sql_file ) :
fp = open ( sql_file )
output . append ( fp . read ( ) )
fp . close ( )
2005-09-30 07:23:11 +08:00
2005-07-20 23:00:34 +08:00
# Content types.
output . append ( _get_contenttype_insert ( opts ) )
# Permissions.
for codename , name in _get_all_permissions ( opts ) :
output . append ( _get_permission_insert ( name , codename , opts ) )
return output
2005-09-03 02:26:16 +08:00
get_sql_initial_data . help_doc = " Prints the initial INSERT SQL statements for the given model module name(s). "
2005-07-20 23:00:34 +08:00
get_sql_initial_data . args = APP_ARGS
def get_sql_sequence_reset ( mod ) :
" Returns a list of the SQL statements to reset PostgreSQL sequences for the given module. "
2005-11-14 09:44:35 +08:00
from django . core import db , meta
2005-07-20 23:00:34 +08:00
output = [ ]
for klass in mod . _MODELS :
for f in klass . _meta . fields :
if isinstance ( f , meta . AutoField ) :
2005-11-14 09:44:35 +08:00
output . append ( " SELECT setval( ' %s _ %s _seq ' , (SELECT max( %s ) FROM %s )); " % \
( klass . _meta . db_table , f . column , db . db . quote_name ( f . column ) ,
db . db . quote_name ( klass . _meta . db_table ) ) )
2005-12-01 14:09:36 +08:00
for f in klass . _meta . many_to_many :
output . append ( " SELECT setval( ' %s _id_seq ' , (SELECT max( %s ) FROM %s )); " % \
( f . get_m2m_db_table ( klass . _meta ) , db . db . quote_name ( ' id ' ) , f . get_m2m_db_table ( klass . _meta ) ) )
2005-07-20 23:00:34 +08:00
return output
2005-09-03 02:26:16 +08:00
get_sql_sequence_reset . help_doc = " Prints the SQL statements for resetting PostgreSQL sequences for the given model module name(s). "
2005-07-20 23:00:34 +08:00
get_sql_sequence_reset . args = APP_ARGS
def get_sql_indexes ( mod ) :
" Returns a list of the CREATE INDEX SQL statements for the given module. "
2005-11-14 09:44:35 +08:00
from django . core . db import db
2005-07-20 23:00:34 +08:00
output = [ ]
for klass in mod . _MODELS :
for f in klass . _meta . fields :
if f . db_index :
unique = f . unique and " UNIQUE " or " "
output . append ( " CREATE %s INDEX %s _ %s ON %s ( %s ); " % \
2005-11-14 09:44:35 +08:00
( unique , klass . _meta . db_table , f . column ,
db . quote_name ( klass . _meta . db_table ) , db . quote_name ( f . column ) ) )
2005-07-20 23:00:34 +08:00
return output
2005-09-03 02:26:16 +08:00
get_sql_indexes . help_doc = " Prints the CREATE INDEX SQL statements for the given model module name(s). "
2005-07-20 23:00:34 +08:00
get_sql_indexes . args = APP_ARGS
def get_sql_all ( mod ) :
" Returns a list of CREATE TABLE SQL and initial-data insert for the given module. "
return get_sql_create ( mod ) + get_sql_initial_data ( mod )
2005-09-03 02:26:16 +08:00
get_sql_all . help_doc = " Prints the CREATE TABLE and initial-data SQL statements for the given model module name(s). "
2005-07-20 23:00:34 +08:00
get_sql_all . args = APP_ARGS
2005-11-29 09:04:28 +08:00
def has_no_records ( cursor ) :
" Returns True if the cursor, having executed a query, returned no records. "
# This is necessary due to an inconsistency in the DB-API spec.
# cursor.rowcount can be -1 (undetermined), according to
# http://www.python.org/peps/pep-0249.html
if cursor . rowcount < 0 :
return cursor . fetchone ( ) is None
return cursor . rowcount < 1
2005-07-20 23:00:34 +08:00
def database_check ( mod ) :
" Checks that everything is properly installed in the database for the given module. "
from django . core import db
cursor = db . db . cursor ( )
app_label = mod . _MODELS [ 0 ] . _meta . app_label
# Check that the package exists in the database.
2005-11-14 09:44:35 +08:00
cursor . execute ( " SELECT 1 FROM %s WHERE %s = %% s " % \
( db . db . quote_name ( ' packages ' ) , db . db . quote_name ( ' label ' ) ) , [ app_label ] )
2005-11-29 09:04:28 +08:00
if has_no_records ( cursor ) :
2005-07-20 23:00:34 +08:00
# sys.stderr.write("The '%s' package isn't installed.\n" % app_label)
print _get_packages_insert ( app_label )
# Check that the permissions and content types are in the database.
perms_seen = { }
contenttypes_seen = { }
for klass in mod . _MODELS :
opts = klass . _meta
perms = _get_all_permissions ( opts )
perms_seen . update ( dict ( perms ) )
contenttypes_seen [ opts . module_name ] = 1
for codename , name in perms :
2005-11-14 09:44:35 +08:00
cursor . execute ( " SELECT 1 FROM %s WHERE %s = %% s AND %s = %% s " % \
( db . db . quote_name ( ' auth_permissions ' ) , db . db . quote_name ( ' package ' ) ,
db . db . quote_name ( ' codename ' ) ) , ( app_label , codename ) )
2005-11-29 09:04:28 +08:00
if has_no_records ( cursor ) :
2005-07-20 23:00:34 +08:00
# sys.stderr.write("The '%s.%s' permission doesn't exist.\n" % (app_label, codename))
print _get_permission_insert ( name , codename , opts )
2005-11-14 09:44:35 +08:00
cursor . execute ( " SELECT 1 FROM %s WHERE %s = %% s AND %s = %% s " % \
( db . db . quote_name ( ' content_types ' ) , db . db . quote_name ( ' package ' ) ,
db . db . quote_name ( ' python_module_name ' ) ) , ( app_label , opts . module_name ) )
2005-11-29 09:04:28 +08:00
if has_no_records ( cursor ) :
2005-07-20 23:00:34 +08:00
# sys.stderr.write("The '%s.%s' content type doesn't exist.\n" % (app_label, opts.module_name))
print _get_contenttype_insert ( opts )
# Check that there aren't any *extra* permissions in the DB that the model
# doesn't know about.
2005-11-14 09:44:35 +08:00
cursor . execute ( " SELECT %s FROM %s WHERE %s = %% s " % \
( db . db . quote_name ( ' codename ' ) , db . db . quote_name ( ' auth_permissions ' ) ,
db . db . quote_name ( ' package ' ) ) , ( app_label , ) )
2005-07-20 23:00:34 +08:00
for row in cursor . fetchall ( ) :
try :
perms_seen [ row [ 0 ] ]
except KeyError :
# sys.stderr.write("A permission called '%s.%s' was found in the database but not in the model.\n" % (app_label, row[0]))
2005-11-14 09:44:35 +08:00
print " DELETE FROM %s WHERE %s = ' %s ' AND %s = ' %s ' ; " % \
( db . db . quote_name ( ' auth_permissions ' ) , db . db . quote_name ( ' package ' ) ,
app_label , db . db . quote_name ( ' codename ' ) , row [ 0 ] )
2005-07-20 23:00:34 +08:00
# Check that there aren't any *extra* content types in the DB that the
# model doesn't know about.
2005-11-14 09:44:35 +08:00
cursor . execute ( " SELECT %s FROM %s WHERE %s = %% s " % \
( db . db . quote_name ( ' python_module_name ' ) , db . db . quote_name ( ' content_types ' ) ,
db . db . quote_name ( ' package ' ) ) , ( app_label , ) )
2005-07-20 23:00:34 +08:00
for row in cursor . fetchall ( ) :
try :
contenttypes_seen [ row [ 0 ] ]
except KeyError :
# sys.stderr.write("A content type called '%s.%s' was found in the database but not in the model.\n" % (app_label, row[0]))
2005-11-14 09:44:35 +08:00
print " DELETE FROM %s WHERE %s = ' %s ' AND %s = ' %s ' ; " % \
( db . db . quote_name ( ' content_types ' ) , db . db . quote_name ( ' package ' ) ,
app_label , db . db . quote_name ( ' python_module_name ' ) , row [ 0 ] )
2005-09-03 02:26:16 +08:00
database_check . help_doc = " Checks that everything is installed in the database for the given model module name(s) and prints SQL statements if needed. "
2005-07-20 23:00:34 +08:00
database_check . args = APP_ARGS
def get_admin_index ( mod ) :
" Returns admin-index template snippet (in list form) for the given module. "
2005-08-02 05:29:52 +08:00
from django . utils . text import capfirst
2005-07-20 23:00:34 +08:00
output = [ ]
app_label = mod . _MODELS [ 0 ] . _meta . app_label
output . append ( ' { %% if perms. %s %% } ' % app_label )
output . append ( ' <div class= " module " ><h2> %s </h2><table> ' % app_label . title ( ) )
for klass in mod . _MODELS :
if klass . _meta . admin :
output . append ( MODULE_TEMPLATE % {
' app ' : app_label ,
' mod ' : klass . _meta . module_name ,
2005-08-02 05:29:52 +08:00
' name ' : capfirst ( klass . _meta . verbose_name_plural ) ,
2005-07-20 23:00:34 +08:00
' addperm ' : klass . _meta . get_add_permission ( ) ,
' changeperm ' : klass . _meta . get_change_permission ( ) ,
} )
output . append ( ' </table></div> ' )
output . append ( ' { % e ndif % } ' )
return output
2005-09-03 02:26:16 +08:00
get_admin_index . help_doc = " Prints the admin-index template snippet for the given model module name(s). "
2005-07-20 23:00:34 +08:00
get_admin_index . args = APP_ARGS
def init ( ) :
" Initializes the database with auth and core. "
try :
2005-09-07 06:14:40 +08:00
from django . core import db , meta
auth = meta . get_app ( ' auth ' )
core = meta . get_app ( ' core ' )
2005-07-20 23:00:34 +08:00
cursor = db . db . cursor ( )
for sql in get_sql_create ( core ) + get_sql_create ( auth ) + get_sql_initial_data ( core ) + get_sql_initial_data ( auth ) :
cursor . execute ( sql )
2005-11-14 09:44:35 +08:00
cursor . execute ( " INSERT INTO %s ( %s , %s ) VALUES ( ' example.com ' , ' Example site ' ) " % \
( db . db . quote_name ( core . Site . _meta . db_table ) , db . db . quote_name ( ' domain ' ) ,
db . db . quote_name ( ' name ' ) ) )
2005-07-20 23:00:34 +08:00
except Exception , e :
2005-09-07 06:14:40 +08:00
sys . stderr . write ( " Error: The database couldn ' t be initialized. \n %s \n " % e )
try :
db . db . rollback ( )
except UnboundLocalError :
pass
2005-07-20 23:00:34 +08:00
sys . exit ( 1 )
2005-09-07 06:14:40 +08:00
else :
db . db . commit ( )
2005-07-20 23:00:34 +08:00
init . args = ' '
def install ( mod ) :
" Executes the equivalent of ' get_sql_all ' in the current database. "
from django . core import db
2005-09-03 02:38:23 +08:00
from cStringIO import StringIO
mod_name = mod . __name__ [ mod . __name__ . rindex ( ' . ' ) + 1 : ]
# First, try validating the models.
s = StringIO ( )
num_errors = get_validation_errors ( s )
if num_errors :
sys . stderr . write ( " Error: %s couldn ' t be installed, because there were errors in your model: \n " % mod_name )
s . seek ( 0 )
sys . stderr . write ( s . read ( ) )
sys . exit ( 1 )
2005-07-20 23:00:34 +08:00
sql_list = get_sql_all ( mod )
2005-09-03 02:38:23 +08:00
2005-07-20 23:00:34 +08:00
try :
cursor = db . db . cursor ( )
for sql in sql_list :
cursor . execute ( sql )
except Exception , e :
sys . stderr . write ( """ Error: %s couldn ' t be installed. Possible reasons:
* The database isn ' t running or isn ' t configured correctly .
* At least one of the database tables already exists .
* The SQL was invalid .
Hint : Look at the output of ' django-admin.py sqlall %s ' . That ' s the SQL this command wasn ' t able to run .
The full error : % s \n """ % \
( mod_name , mod_name , e ) )
db . db . rollback ( )
sys . exit ( 1 )
db . db . commit ( )
2005-09-03 02:26:16 +08:00
install . help_doc = " Executes ``sqlall`` for the given model module name(s) in the current database. "
2005-07-20 23:00:34 +08:00
install . args = APP_ARGS
2005-10-24 06:43:24 +08:00
def installperms ( mod ) :
" Installs any permissions for the given model, if needed. "
from django . models . auth import permissions
from django . models . core import packages
num_added = 0
package = packages . get_object ( pk = mod . _MODELS [ 0 ] . _meta . app_label )
for klass in mod . _MODELS :
opts = klass . _meta
for codename , name in _get_all_permissions ( opts ) :
try :
permissions . get_object ( name__exact = name , codename__exact = codename , package__label__exact = package . label )
except permissions . PermissionDoesNotExist :
p = permissions . Permission ( name = name , package = package , codename = codename )
p . save ( )
print " Added permission ' %r ' . " % p
num_added + = 1
if not num_added :
print " No permissions were added, because all necessary permissions were already installed. "
installperms . help_doc = " Installs any permissions for the given model module name(s), if needed. "
installperms . args = APP_ARGS
2005-07-20 23:00:34 +08:00
def _start_helper ( app_or_project , name , directory , other_name = ' ' ) :
other = { ' project ' : ' app ' , ' app ' : ' project ' } [ app_or_project ]
if not _is_valid_dir_name ( name ) :
sys . stderr . write ( " Error: %r is not a valid %s name. Please use only numbers, letters and underscores. \n " % ( name , app_or_project ) )
sys . exit ( 1 )
top_dir = os . path . join ( directory , name )
try :
os . mkdir ( top_dir )
except OSError , e :
sys . stderr . write ( " Error: %s \n " % e )
sys . exit ( 1 )
template_dir = PROJECT_TEMPLATE_DIR % app_or_project
for d , subdirs , files in os . walk ( template_dir ) :
relative_dir = d [ len ( template_dir ) + 1 : ] . replace ( ' %s _name ' % app_or_project , name )
if relative_dir :
os . mkdir ( os . path . join ( top_dir , relative_dir ) )
for i , subdir in enumerate ( subdirs ) :
if subdir . startswith ( ' . ' ) :
del subdirs [ i ]
for f in files :
if f . endswith ( ' .pyc ' ) :
continue
fp_old = open ( os . path . join ( d , f ) , ' r ' )
fp_new = open ( os . path . join ( top_dir , relative_dir , f . replace ( ' %s _name ' % app_or_project , name ) ) , ' w ' )
fp_new . write ( fp_old . read ( ) . replace ( ' {{ %s _name }} ' % app_or_project , name ) . replace ( ' {{ %s _name }} ' % other , other_name ) )
fp_old . close ( )
fp_new . close ( )
def startproject ( project_name , directory ) :
" Creates a Django project for the given project_name in the given directory. "
2005-07-21 10:20:40 +08:00
from random import choice
2005-12-06 13:17:52 +08:00
if project_name in INVALID_PROJECT_NAMES :
sys . stderr . write ( " Error: %r isn ' t a valid project name. Please try another. \n " % project_name )
sys . exit ( 1 )
2005-07-20 23:00:34 +08:00
_start_helper ( ' project ' , project_name , directory )
# Create a random SECRET_KEY hash, and put it in the main settings.
2005-10-19 09:09:05 +08:00
main_settings_file = os . path . join ( directory , project_name , ' settings.py ' )
2005-07-20 23:00:34 +08:00
settings_contents = open ( main_settings_file , ' r ' ) . read ( )
fp = open ( main_settings_file , ' w ' )
secret_key = ' ' . join ( [ choice ( ' abcdefghijklmnopqrstuvwxyz0123456789!@#$ % ^&*(-_=+) ' ) for i in range ( 50 ) ] )
settings_contents = re . sub ( r " (?<=SECRET_KEY = ' ) ' " , secret_key + " ' " , settings_contents )
fp . write ( settings_contents )
fp . close ( )
startproject . help_doc = " Creates a Django project directory structure for the given project name in the current directory. "
startproject . args = " [projectname] "
def startapp ( app_name , directory ) :
2005-09-03 02:26:16 +08:00
" Creates a Django app for the given app_name in the given directory. "
2005-07-20 23:00:34 +08:00
# Determine the project_name a bit naively -- by looking at the name of
# the parent directory.
2005-10-06 22:30:35 +08:00
project_dir = os . path . normpath ( os . path . join ( directory , ' .. ' ) )
2005-07-20 23:00:34 +08:00
project_name = os . path . basename ( project_dir )
_start_helper ( ' app ' , app_name , directory , project_name )
startapp . help_doc = " Creates a Django app directory structure for the given app name in the current directory. "
startapp . args = " [appname] "
2005-11-28 10:57:38 +08:00
def createsuperuser ( username = None , email = None , password = None ) :
2005-07-21 10:17:45 +08:00
" Creates a superuser account. "
from django . core import validators
from django . models . auth import users
import getpass
try :
while 1 :
2005-11-28 10:57:38 +08:00
if not username :
username = raw_input ( ' Username (only letters, digits and underscores): ' )
2005-07-21 10:17:45 +08:00
if not username . isalnum ( ) :
sys . stderr . write ( " Error: That username is invalid. \n " )
2005-11-28 10:57:38 +08:00
username = None
2005-07-21 10:17:45 +08:00
try :
users . get_object ( username__exact = username )
except users . UserDoesNotExist :
break
else :
sys . stderr . write ( " Error: That username is already taken. \n " )
2005-11-28 10:57:38 +08:00
username = None
2005-07-21 10:17:45 +08:00
while 1 :
2005-11-28 10:57:38 +08:00
if not email :
email = raw_input ( ' E-mail address: ' )
2005-07-21 10:17:45 +08:00
try :
validators . isValidEmail ( email , None )
except validators . ValidationError :
sys . stderr . write ( " Error: That e-mail address is invalid. \n " )
2005-11-28 10:57:38 +08:00
email = None
2005-07-21 10:17:45 +08:00
else :
break
while 1 :
2005-11-28 10:57:38 +08:00
if not password :
password = getpass . getpass ( )
password2 = getpass . getpass ( ' Password (again): ' )
if password != password2 :
sys . stderr . write ( " Error: Your passwords didn ' t match. \n " )
password = None
continue
2005-07-26 22:37:00 +08:00
if password . strip ( ) == ' ' :
sys . stderr . write ( " Error: Blank passwords aren ' t allowed. \n " )
2005-11-28 10:57:38 +08:00
password = None
2005-07-26 22:37:00 +08:00
continue
break
2005-07-21 10:17:45 +08:00
except KeyboardInterrupt :
sys . stderr . write ( " \n Operation cancelled. \n " )
sys . exit ( 1 )
u = users . create_user ( username , email , password )
u . is_staff = True
u . is_active = True
u . is_superuser = True
u . save ( )
print " User created successfully. "
2005-11-28 10:57:38 +08:00
createsuperuser . args = ' [username] [email] [password] (Either all or none) '
2005-07-21 10:17:45 +08:00
2005-08-03 01:08:24 +08:00
def inspectdb ( db_name ) :
" Generator that introspects the tables in the given database name and returns a Django model, one line at a time. "
from django . core import db
from django . conf import settings
2006-02-05 04:18:18 +08:00
import keyword
2005-08-03 06:33:39 +08:00
def table2model ( table_name ) :
object_name = table_name . title ( ) . replace ( ' _ ' , ' ' )
return object_name . endswith ( ' s ' ) and object_name [ : - 1 ] or object_name
2005-08-03 01:08:24 +08:00
settings . DATABASE_NAME = db_name
cursor = db . db . cursor ( )
2005-08-03 06:33:39 +08:00
yield " # This is an auto-generated Django model module. "
yield " # You ' ll have to do the following manually to clean this up: "
yield " # * Rearrange models ' order "
yield " # * Add primary_key=True to one field in each model. "
yield " # Feel free to rename the models, but don ' t rename db_table values or field names. "
2005-08-04 22:51:46 +08:00
yield " # "
yield " # Also note: You ' ll have to insert the output of ' django-admin.py sqlinitialdata [appname] ' "
yield " # into your database. "
2005-08-03 06:33:39 +08:00
yield ' '
2005-08-03 01:08:24 +08:00
yield ' from django.core import meta '
yield ' '
for table_name in db . get_table_list ( cursor ) :
2005-08-03 06:33:39 +08:00
yield ' class %s (meta.Model): ' % table2model ( table_name )
try :
relations = db . get_relations ( cursor , table_name )
except NotImplementedError :
relations = { }
2005-11-29 10:05:32 +08:00
for i , row in enumerate ( db . get_table_description ( cursor , table_name ) ) :
2005-08-26 12:06:18 +08:00
column_name = row [ 0 ]
2006-02-05 04:08:30 +08:00
comment_notes = [ ] # Holds Field notes, to be displayed in a Python comment.
extra_params = { } # Holds Field parameters such as 'db_column'.
2006-02-05 04:18:18 +08:00
if keyword . iskeyword ( column_name ) :
2006-02-05 04:08:30 +08:00
extra_params [ ' db_column ' ] = column_name
column_name + = ' _field '
comment_notes . append ( ' Field renamed because it was a Python reserved word. ' )
2005-08-03 06:33:39 +08:00
if relations . has_key ( i ) :
2006-02-05 04:08:30 +08:00
rel_to = relations [ i ] [ 1 ] == table_name and " ' self ' " or table2model ( relations [ i ] [ 1 ] )
field_type = ' ForeignKey( %s ' % rel_to
2005-08-26 12:06:18 +08:00
if column_name . endswith ( ' _id ' ) :
2006-02-05 04:08:30 +08:00
column_name = column_name [ : - 3 ]
2005-08-26 12:06:18 +08:00
else :
2006-02-05 04:08:30 +08:00
extra_params [ ' db_column ' ] = column_name
2005-08-03 06:33:39 +08:00
else :
2005-08-04 11:49:24 +08:00
try :
field_type = db . DATA_TYPES_REVERSE [ row [ 1 ] ]
except KeyError :
field_type = ' TextField '
2006-02-05 04:08:30 +08:00
comment_notes . append ( ' This field type is a guess. ' )
2005-12-01 14:32:25 +08:00
# This is a hook for DATA_TYPES_REVERSE to return a tuple of
# (field_type, extra_params_dict).
if type ( field_type ) is tuple :
2006-02-05 04:08:30 +08:00
field_type , new_params = field_type
extra_params . update ( new_params )
2005-12-01 14:32:25 +08:00
if field_type == ' CharField ' and row [ 3 ] :
extra_params [ ' maxlength ' ] = row [ 3 ]
2006-02-05 04:08:30 +08:00
field_type + = ' ( '
field_desc = ' %s = meta. %s ' % ( column_name , field_type )
if extra_params :
if not field_desc . endswith ( ' ( ' ) :
field_desc + = ' , '
field_desc + = ' , ' . join ( [ ' %s = %r ' % ( k , v ) for k , v in extra_params . items ( ) ] )
field_desc + = ' ) '
if comment_notes :
field_desc + = ' # ' + ' ' . join ( comment_notes )
2005-11-29 10:05:32 +08:00
yield ' %s ' % field_desc
2005-08-26 12:06:18 +08:00
yield ' class META: '
yield ' db_table = %r ' % table_name
2005-08-03 01:08:24 +08:00
yield ' '
inspectdb . help_doc = " Introspects the database tables in the given database and outputs a Django model module. "
inspectdb . args = " [dbname] "
2005-08-15 13:05:52 +08:00
class ModelErrorCollection :
def __init__ ( self , outfile = sys . stdout ) :
self . errors = [ ]
self . outfile = outfile
def add ( self , opts , error ) :
self . errors . append ( ( opts , error ) )
2005-08-26 06:51:30 +08:00
self . outfile . write ( " %s . %s : %s \n " % ( opts . app_label , opts . module_name , error ) )
2005-08-15 13:05:52 +08:00
2005-09-03 02:38:23 +08:00
def get_validation_errors ( outfile ) :
" Validates all installed models. Writes errors, if any, to outfile. Returns number of errors. "
2005-08-15 23:24:56 +08:00
import django . models
2005-08-15 13:05:52 +08:00
from django . core import meta
2005-09-03 02:28:26 +08:00
e = ModelErrorCollection ( outfile )
2005-08-15 13:05:52 +08:00
module_list = meta . get_installed_model_modules ( )
for module in module_list :
for mod in module . _MODELS :
opts = mod . _meta
# Do field-specific validation.
for f in opts . fields :
if isinstance ( f , meta . CharField ) and f . maxlength in ( None , 0 ) :
e . add ( opts , ' " %s " field: CharFields require a " maxlength " attribute. ' % f . name )
2005-11-21 06:28:57 +08:00
if isinstance ( f , meta . FloatField ) :
if f . decimal_places is None :
e . add ( opts , ' " %s " field: FloatFields require a " decimal_places " attribute. ' % f . name )
if f . max_digits is None :
e . add ( opts , ' " %s " field: FloatFields require a " max_digits " attribute. ' % f . name )
2005-09-05 23:04:20 +08:00
if isinstance ( f , meta . FileField ) and not f . upload_to :
e . add ( opts , ' " %s " field: FileFields require an " upload_to " attribute. ' % f . name )
2005-09-26 04:01:30 +08:00
if isinstance ( f , meta . ImageField ) :
try :
from PIL import Image
except ImportError :
2006-01-08 14:31:24 +08:00
e . add ( opts , ' " %s " field: To use ImageFields, you need to install the Python Imaging Library. Get it at http://www.pythonware.com/products/pil/ . ' % f . name )
2005-09-03 06:50:53 +08:00
if f . prepopulate_from is not None and type ( f . prepopulate_from ) not in ( list , tuple ) :
e . add ( opts , ' " %s " field: prepopulate_from should be a list or tuple. ' % f . name )
2005-08-16 03:00:50 +08:00
if f . choices :
if not type ( f . choices ) in ( tuple , list ) :
e . add ( opts , ' " %s " field: " choices " should be either a tuple or list. ' % f . name )
else :
for c in f . choices :
if not type ( c ) in ( tuple , list ) or len ( c ) != 2 :
e . add ( opts , ' " %s " field: " choices " should be a sequence of two-tuples. ' % f . name )
2006-01-15 09:15:44 +08:00
if f . db_index not in ( None , True , False ) :
e . add ( opts , ' " %s " field: " db_index " should be either None, True or False. ' % f . name )
2005-08-15 13:05:52 +08:00
2005-12-01 13:46:18 +08:00
# Check for multiple ManyToManyFields to the same object, and
# verify "singular" is set in that case.
for i , f in enumerate ( opts . many_to_many ) :
for previous_f in opts . many_to_many [ : i ] :
if f . rel . to == previous_f . rel . to and f . rel . singular == previous_f . rel . singular :
e . add ( opts , ' The " %s " field requires a " singular " parameter, because the %s model has more than one ManyToManyField to the same model ( %s ). ' % ( f . name , opts . object_name , previous_f . rel . to . object_name ) )
2005-08-15 13:05:52 +08:00
# Check admin attribute.
2005-09-20 10:48:25 +08:00
if opts . admin is not None :
if not isinstance ( opts . admin , meta . Admin ) :
e . add ( opts , ' " admin " attribute, if given, must be set to a meta.Admin() instance. ' )
else :
2005-10-06 09:41:54 +08:00
# list_display
if not isinstance ( opts . admin . list_display , ( list , tuple ) ) :
e . add ( opts , ' " admin.list_display " , if given, must be set to a list or tuple. ' )
else :
for fn in opts . admin . list_display :
try :
f = opts . get_field ( fn )
except meta . FieldDoesNotExist :
klass = opts . get_model_module ( ) . Klass
if not hasattr ( klass , fn ) or not callable ( getattr ( klass , fn ) ) :
e . add ( opts , ' " admin.list_display " refers to %r , which isn \' t a field or method. ' % fn )
else :
if isinstance ( f , meta . ManyToManyField ) :
e . add ( opts , ' " admin.list_display " doesn \' t support ManyToManyFields ( %r ). ' % fn )
# list_filter
if not isinstance ( opts . admin . list_filter , ( list , tuple ) ) :
e . add ( opts , ' " admin.list_filter " , if given, must be set to a list or tuple. ' )
else :
for fn in opts . admin . list_filter :
try :
f = opts . get_field ( fn )
except meta . FieldDoesNotExist :
e . add ( opts , ' " admin.list_filter " refers to %r , which isn \' t a field. ' % fn )
2005-08-15 13:05:52 +08:00
# Check ordering attribute.
if opts . ordering :
for field_name in opts . ordering :
if field_name == ' ? ' : continue
if field_name . startswith ( ' - ' ) :
field_name = field_name [ 1 : ]
2005-08-26 06:51:30 +08:00
if opts . order_with_respect_to and field_name == ' _order ' :
continue
2005-08-15 13:05:52 +08:00
try :
opts . get_field ( field_name , many_to_many = False )
except meta . FieldDoesNotExist :
e . add ( opts , ' " ordering " refers to " %s " , a field that doesn \' t exist. ' % field_name )
# Check core=True, if needed.
2005-11-26 05:20:09 +08:00
for related in opts . get_followed_related_objects ( ) :
2005-08-15 13:05:52 +08:00
try :
2005-11-26 05:20:09 +08:00
for f in related . opts . fields :
2005-08-15 13:05:52 +08:00
if f . core :
raise StopIteration
2005-11-26 05:20:09 +08:00
e . add ( related . opts , " At least one field in %s should have core=True, because it ' s being edited inline by %s . %s . " % ( related . opts . object_name , opts . module_name , opts . object_name ) )
2005-08-15 13:05:52 +08:00
except StopIteration :
pass
2005-11-21 09:45:15 +08:00
# Check unique_together.
for ut in opts . unique_together :
for field_name in ut :
try :
f = opts . get_field ( field_name , many_to_many = True )
except meta . FieldDoesNotExist :
e . add ( opts , ' " unique_together " refers to %s , a field that doesn \' t exist. Check your syntax. ' % field_name )
else :
if isinstance ( f . rel , meta . ManyToMany ) :
e . add ( opts , ' " unique_together " refers to %s . ManyToManyFields are not supported in unique_together. ' % f . name )
2005-09-03 02:38:23 +08:00
return len ( e . errors )
2005-08-15 13:05:52 +08:00
2005-09-03 02:38:23 +08:00
def validate ( outfile = sys . stdout ) :
" Validates all installed models. "
num_errors = get_validation_errors ( outfile )
2005-09-03 02:28:26 +08:00
outfile . write ( ' %s error %s found. \n ' % ( num_errors , num_errors != 1 and ' s ' or ' ' ) )
2005-08-15 13:05:52 +08:00
validate . args = ' '
2005-08-20 05:23:56 +08:00
def runserver ( addr , port ) :
2005-07-20 23:00:34 +08:00
" Starts a lightweight Web server for development. "
2005-08-12 12:16:29 +08:00
from django . core . servers . basehttp import run , AdminMediaHandler , WSGIServerException
from django . core . handlers . wsgi import WSGIHandler
2005-08-20 05:23:56 +08:00
if not addr :
addr = ' 127.0.0.1 '
2005-07-20 23:00:34 +08:00
if not port . isdigit ( ) :
sys . stderr . write ( " Error: %r is not a valid port number. \n " % port )
sys . exit ( 1 )
2005-07-21 12:08:54 +08:00
def inner_run ( ) :
from django . conf . settings import SETTINGS_MODULE
2005-08-15 23:24:56 +08:00
print " Validating models... "
validate ( )
2005-08-15 13:05:52 +08:00
print " \n Starting server on port %s with settings module %r . " % ( port , SETTINGS_MODULE )
2005-08-20 05:23:56 +08:00
print " Go to http:// %s : %s / for Django. " % ( addr , port )
2005-07-21 12:08:54 +08:00
print " Quit the server with CONTROL-C (Unix) or CTRL-BREAK (Windows). "
2005-07-20 23:00:34 +08:00
try :
2005-08-20 05:23:56 +08:00
run ( addr , int ( port ) , AdminMediaHandler ( WSGIHandler ( ) ) )
2005-07-21 12:08:54 +08:00
except WSGIServerException , e :
# Use helpful error messages instead of ugly tracebacks.
ERRORS = {
13 : " You don ' t have permission to access that port. " ,
98 : " That port is already in use. " ,
2005-08-20 05:23:56 +08:00
99 : " That IP address can ' t be assigned-to. " ,
2005-07-21 12:08:54 +08:00
}
try :
error_text = ERRORS [ e . args [ 0 ] . args [ 0 ] ]
except ( AttributeError , KeyError ) :
error_text = str ( e )
sys . stderr . write ( " Error: %s \n " % error_text )
sys . exit ( 1 )
except KeyboardInterrupt :
sys . exit ( 0 )
from django . utils import autoreload
autoreload . main ( inner_run )
2005-08-20 05:23:56 +08:00
runserver . args = ' [optional port number, or ipaddr:port] '
2005-09-26 06:03:30 +08:00
def createcachetable ( tablename ) :
" Creates the table needed to use the SQL cache backend "
from django . core import db , meta
fields = (
2005-09-26 11:14:37 +08:00
# "key" is a reserved word in MySQL, so use "cache_key" instead.
meta . CharField ( name = ' cache_key ' , maxlength = 255 , unique = True , primary_key = True ) ,
2005-09-26 06:03:30 +08:00
meta . TextField ( name = ' value ' ) ,
meta . DateTimeField ( name = ' expires ' , db_index = True ) ,
)
table_output = [ ]
index_output = [ ]
for f in fields :
2005-11-14 09:44:35 +08:00
field_output = [ db . db . quote_name ( f . column ) , db . DATA_TYPES [ f . get_internal_type ( ) ] % f . __dict__ ]
2005-09-26 06:03:30 +08:00
field_output . append ( " %s NULL " % ( not f . null and " NOT " or " " ) )
if f . unique :
field_output . append ( " UNIQUE " )
if f . primary_key :
field_output . append ( " PRIMARY KEY " )
if f . db_index :
unique = f . unique and " UNIQUE " or " "
2005-11-14 09:44:35 +08:00
index_output . append ( " CREATE %s INDEX %s _ %s ON %s ( %s ); " % \
( unique , tablename , f . column , db . db . quote_name ( tablename ) ,
db . db . quote_name ( f . column ) ) )
2005-09-26 06:03:30 +08:00
table_output . append ( " " . join ( field_output ) )
2005-11-14 09:44:35 +08:00
full_statement = [ " CREATE TABLE %s ( " % db . db . quote_name ( tablename ) ]
2005-09-26 06:03:30 +08:00
for i , line in enumerate ( table_output ) :
full_statement . append ( ' %s %s ' % ( line , i < len ( table_output ) - 1 and ' , ' or ' ' ) )
full_statement . append ( ' ); ' )
curs = db . db . cursor ( )
curs . execute ( " \n " . join ( full_statement ) )
for statement in index_output :
curs . execute ( statement )
db . db . commit ( )
2005-09-30 07:43:03 +08:00
createcachetable . args = " [tablename] "
2005-12-06 13:17:52 +08:00
2006-01-13 04:56:10 +08:00
def run_shell ( use_plain = False ) :
" Runs a Python interactive interpreter. Tries to use IPython, if it ' s available. "
try :
if use_plain :
# Don't bother loading IPython, because the user wants plain Python.
raise ImportError
import IPython
shell = IPython . Shell . IPShell ( )
shell . mainloop ( )
except ImportError :
import code
code . interact ( )
run_shell . args = ' [--plain] '
2006-01-11 09:57:22 +08:00
2005-12-06 13:17:52 +08:00
# Utilities for command-line script
2005-12-06 13:35:07 +08:00
DEFAULT_ACTION_MAPPING = {
2005-12-06 13:17:52 +08:00
' adminindex ' : get_admin_index ,
' createsuperuser ' : createsuperuser ,
' createcachetable ' : createcachetable ,
# 'dbcheck': database_check,
' init ' : init ,
' inspectdb ' : inspectdb ,
' install ' : install ,
' installperms ' : installperms ,
' runserver ' : runserver ,
2006-01-11 09:57:22 +08:00
' shell ' : run_shell ,
2005-12-06 13:17:52 +08:00
' sql ' : get_sql_create ,
' sqlall ' : get_sql_all ,
' sqlclear ' : get_sql_delete ,
' sqlindexes ' : get_sql_indexes ,
' sqlinitialdata ' : get_sql_initial_data ,
' sqlreset ' : get_sql_reset ,
' sqlsequencereset ' : get_sql_sequence_reset ,
' startapp ' : startapp ,
' startproject ' : startproject ,
' validate ' : validate ,
}
NO_SQL_TRANSACTION = ( ' adminindex ' , ' createcachetable ' , ' dbcheck ' , ' install ' , ' installperms ' , ' sqlindexes ' )
class DjangoOptionParser ( OptionParser ) :
def print_usage_and_exit ( self ) :
self . print_help ( sys . stderr )
sys . exit ( 1 )
2005-12-06 13:35:07 +08:00
def get_usage ( action_mapping ) :
2005-12-06 13:17:52 +08:00
"""
Returns a usage string . Doesn ' t do the options stuff, because optparse
takes care of that .
"""
usage = [ " usage: % prog action [options] \n actions: " ]
2005-12-06 13:35:07 +08:00
available_actions = action_mapping . keys ( )
2005-12-06 13:17:52 +08:00
available_actions . sort ( )
for a in available_actions :
2005-12-06 13:35:07 +08:00
func = action_mapping [ a ]
2005-12-06 13:17:52 +08:00
usage . append ( " %s %s " % ( a , func . args ) )
usage . extend ( textwrap . wrap ( getattr ( func , ' help_doc ' , func . __doc__ ) , initial_indent = ' ' , subsequent_indent = ' ' ) )
usage . append ( " " )
return ' \n ' . join ( usage [ : - 1 ] ) # Cut off last list element, an empty space.
def print_error ( msg , cmd ) :
sys . stderr . write ( ' Error: %s \n Run " %s --help " for help. \n ' % ( msg , cmd ) )
sys . exit ( 1 )
2005-12-06 13:35:07 +08:00
def execute_from_command_line ( action_mapping = DEFAULT_ACTION_MAPPING ) :
2005-12-06 13:17:52 +08:00
# Parse the command-line arguments. optparse handles the dirty work.
2005-12-06 13:35:07 +08:00
parser = DjangoOptionParser ( get_usage ( action_mapping ) )
2005-12-06 13:17:52 +08:00
parser . add_option ( ' --settings ' ,
help = ' Python path to settings module, e.g. " myproject.settings.main " . If this isn \' t provided, the DJANGO_SETTINGS_MODULE environment variable will be used. ' )
parser . add_option ( ' --pythonpath ' ,
help = ' Lets you manually add a directory the Python path, e.g. " /home/djangoprojects/myproject " . ' )
2006-01-13 04:56:10 +08:00
parser . add_option ( ' --plain ' , action = ' store_true ' , dest = ' plain ' ,
help = ' Tells Django to use plain Python, not IPython, for " shell " command. ' )
2005-12-06 13:17:52 +08:00
options , args = parser . parse_args ( )
# Take care of options.
if options . settings :
os . environ [ ' DJANGO_SETTINGS_MODULE ' ] = options . settings
if options . pythonpath :
sys . path . insert ( 0 , options . pythonpath )
# Run the appropriate action. Unfortunately, optparse can't handle
# positional arguments, so this has to parse/validate them.
try :
action = args [ 0 ]
except IndexError :
parser . print_usage_and_exit ( )
2005-12-06 13:35:07 +08:00
if not action_mapping . has_key ( action ) :
2005-12-06 13:17:52 +08:00
print_error ( " Your action, %r , was invalid. " % action , sys . argv [ 0 ] )
2005-12-06 13:38:56 +08:00
# Switch to English, because django-admin.py creates database content
2005-12-06 13:17:52 +08:00
# like permissions, and those shouldn't contain any translations.
# But only do this if we should have a working settings file.
if action not in ( ' startproject ' , ' startapp ' ) :
from django . utils import translation
translation . activate ( ' en-us ' )
if action == ' createsuperuser ' :
try :
username , email , password = args [ 1 ] , args [ 2 ] , args [ 3 ]
except IndexError :
if len ( args ) == 1 : # We got no arguments, just the action.
2005-12-06 13:35:07 +08:00
action_mapping [ action ] ( )
2005-12-06 13:17:52 +08:00
else :
sys . stderr . write ( " Error: %r requires arguments of ' username email password ' or no argument at all. \n " )
sys . exit ( 1 )
else :
2005-12-06 13:35:07 +08:00
action_mapping [ action ] ( username , email , password )
2006-01-13 04:56:10 +08:00
elif action == ' shell ' :
action_mapping [ action ] ( options . plain is True )
elif action in ( ' init ' , ' validate ' ) :
2005-12-06 13:35:07 +08:00
action_mapping [ action ] ( )
2005-12-06 13:17:52 +08:00
elif action == ' inspectdb ' :
try :
param = args [ 1 ]
except IndexError :
parser . print_usage_and_exit ( )
try :
2005-12-06 13:35:07 +08:00
for line in action_mapping [ action ] ( param ) :
2005-12-06 13:17:52 +08:00
print line
except NotImplementedError :
sys . stderr . write ( " Error: %r isn ' t supported for the currently selected database backend. \n " % action )
sys . exit ( 1 )
elif action == ' createcachetable ' :
try :
2005-12-06 13:35:07 +08:00
action_mapping [ action ] ( args [ 1 ] )
2005-12-06 13:17:52 +08:00
except IndexError :
parser . print_usage_and_exit ( )
elif action in ( ' startapp ' , ' startproject ' ) :
try :
name = args [ 1 ]
except IndexError :
parser . print_usage_and_exit ( )
2005-12-06 13:35:07 +08:00
action_mapping [ action ] ( name , os . getcwd ( ) )
2005-12-06 13:17:52 +08:00
elif action == ' runserver ' :
if len ( args ) < 2 :
addr = ' '
port = ' 8000 '
else :
try :
addr , port = args [ 1 ] . split ( ' : ' )
except ValueError :
addr , port = ' ' , args [ 1 ]
2005-12-06 13:35:07 +08:00
action_mapping [ action ] ( addr , port )
2005-12-06 13:17:52 +08:00
else :
from django . core import meta
if action == ' dbcheck ' :
mod_list = meta . get_all_installed_modules ( )
else :
try :
mod_list = [ meta . get_app ( app_label ) for app_label in args [ 1 : ] ]
except ImportError , e :
sys . stderr . write ( " Error: %s . Are you sure your INSTALLED_APPS setting is correct? \n " % e )
sys . exit ( 1 )
if not mod_list :
parser . print_usage_and_exit ( )
if action not in NO_SQL_TRANSACTION :
print " BEGIN; "
for mod in mod_list :
2005-12-06 13:35:07 +08:00
output = action_mapping [ action ] ( mod )
2005-12-06 13:17:52 +08:00
if output :
print ' \n ' . join ( output )
if action not in NO_SQL_TRANSACTION :
print " COMMIT; "
2005-12-06 13:53:31 +08:00
def execute_manager ( settings_mod ) :
# Add this project to sys.path so that it's importable in the conventional
# way. For example, if this file (manage.py) lives in a directory
# "myproject", this code would add "/path/to/myproject" to sys.path.
project_directory = os . path . dirname ( settings_mod . __file__ )
project_name = os . path . basename ( project_directory )
sys . path . append ( os . path . join ( project_directory , ' .. ' ) )
project_module = __import__ ( project_name , ' ' , ' ' , [ ' ' ] )
sys . path . pop ( )
# Set DJANGO_SETTINGS_MODULE appropriately.
os . environ [ ' DJANGO_SETTINGS_MODULE ' ] = ' %s .settings ' % project_name
# Remove the "startproject" command from the action_mapping, because that's
# a django-admin.py command, not a manage.py command.
action_mapping = DEFAULT_ACTION_MAPPING . copy ( )
del action_mapping [ ' startproject ' ]
# Run the django-admin.py command.
execute_from_command_line ( action_mapping )