Как удалить все таблицы из базы данных с помощью manage.py и командной строки? Есть ли способ сделать это, выполнив manage.py с соответствующими параметрами, чтобы я мог выполнить его из приложения .NET?
Ответы:
Насколько я знаю, нет команды управления для удаления всех таблиц. Если вы не против взлома Python, вы можете написать для этого свою собственную команду. Возможно, вам понравится этот sqlclear
вариант. В документации говорится, что ./manage.py sqlclear
выводит операторы SQL DROP TABLE для заданных имен приложений.
Обновление : бесстыдное присвоение комментария @Mike DeSimone под этим ответом, чтобы дать полный ответ.
./manage.py sqlclear | ./manage.py dbshell
Начиная с django 1.9 это сейчас ./manage.py sqlflush
./manage.py sqlclear myAppName | ./manage.py dbshell
Нет встроенной команды управления Django для удаления всех таблиц. Оба sqlclear
и reset
требуют имени приложения.
Однако вы можете установить Django Extensions, который дает вам то manage.py reset_db
, что делает именно то, что вы хотите (и дает вам доступ ко многим другим полезным командам управления).
manage.py reset_db
требуется флаг `-c, --close-sessions` для закрытия соединений с базой данных перед удалением базы данных (только для PostgreSQL)
Если вы используете пакет South для обработки миграции базы данных (настоятельно рекомендуется), вы можете просто использовать ./manage.py migrate appname zero
команду.
В противном случае я бы порекомендовал ./manage.py dbshell
команду, вводящую команды SQL в стандартный ввод.
python manage.py migrate <app> zero
sqlclear
был удален из 1.9.
В примечаниях к выпуску упоминается, что это связано с введением миграций: https://docs.djangoproject.com/en/1.9/releases/1.9/
К сожалению, я не смог найти ни метода, который работал бы со всеми приложениями одновременно, ни встроенного способа перечислить все установленные приложения от администратора: как вывести список всех установленных приложений с помощью manage.py в Django?
Связанный: Как сбросить миграции в Django 1.7?
простой (?) способ сделать это из python (на mysql):
from django.db import connection
cursor = connection.cursor()
cursor.execute('show tables;')
parts = ('DROP TABLE IF EXISTS %s;' % table for (table,) in cursor.fetchall())
sql = 'SET FOREIGN_KEY_CHECKS = 0;\n' + '\n'.join(parts) + 'SET FOREIGN_KEY_CHECKS = 1;\n'
connection.cursor().execute(sql)
Если вы хотите полностью стереть базу данных и повторно синхронизировать ее одновременно, вам понадобится что-то вроде следующего. Я также комбинирую добавление тестовых данных в эту команду:
#!/usr/bin/env python
import os
os.environ.setdefault("DJANGO_SETTINGS_MODULE", "main.settings") # Replace with your app name.
from django.db import connection
from django.core.management import call_command
from django.conf import settings
# If you're using postgres you can't use django's sql stuff for some reason that I
# can't remember. It has to do with that autocommit thing I think.
# import psychodb2 as db
def recreateDb():
print("Wiping database")
dbinfo = settings.DATABASES['default']
# Postgres version
#conn = db.connect(host=dbinfo['HOST'], user=dbinfo['USER'],
# password=dbinfo['PASSWORD'], port=int(dbinfo['PORT'] or 5432))
#conn.autocommit = True
#cursor = conn.cursor()
#cursor.execute("DROP DATABASE " + dbinfo['NAME'])
#cursor.execute("CREATE DATABASE " + dbinfo['NAME'] + " WITH ENCODING 'UTF8'") # Default is UTF8, but can be changed so lets be sure.
# Mysql version:
print("Dropping and creating database " + dbinfo['NAME'])
cursor = connection.cursor()
cursor.execute("DROP DATABASE " + dbinfo["NAME"] + "; CREATE DATABASE " + dbinfo["NAME"] + "; USE " + dbinfo["NAME"] + ";")
print("Done")
if __name__ == "__main__":
recreateDb();
print("Syncing DB")
call_command('syncdb', interactive=False)
print("Adding test data")
addTestData() # ...
Было бы хорошо , чтобы быть в состоянии сделать , cursor.execute(call_command('sqlclear', 'main'))
но call_command
печатает SQL на стандартный вывод , а не возвращать его в виде строки, и я не могу работать в sql_delete
код ...
USE DATABASE
я рекомендую вам создать пакет django-reconate-db с командой управления, которая будет автоматически переключаться в зависимости от настроек для переключения между SQLite3 и PostGresql.
Вот сценарий оболочки, который я собрал по кусочкам для решения этой проблемы. Надеюсь, это кому-то сэкономит время.
#!/bin/sh
drop() {
echo "Droping all tables prefixed with $1_."
echo
echo "show tables" | ./manage.py dbshell |
egrep "^$1_" | xargs -I "@@" echo "DROP TABLE @@;" |
./manage.py dbshell
echo "Tables dropped."
echo
}
cancel() {
echo "Cancelling Table Drop."
echo
}
if [ -z "$1" ]; then
echo "Please specify a table prefix to drop."
else
echo "Drop all tables with $1_ prefix?"
select choice in drop cancel;do
$choice $1
break
done
fi
Используя Python для создания команды flushproject, вы используете:
from django.db import connection
cursor = connection.cursor()
cursor.execute(“DROP DATABASE %s;”, [connection.settings_dict['NAME']])
cursor.execute(“CREATE DATABASE %s;”, [connection.settings_dict['NAME']])
flushdb
и после того, как я запустил другую команду. если вам это нужно в другом сценарии, вы можете использоватьcall_command
call_command
. Вы говорите, что я должен сделать call_command("flushdb")
раньше call_command("syncdb")
?
Команда ./manage.py sqlclear
или, ./manage.py sqlflush
кажется, очищает таблицу, а не удаляет их, однако, если вы хотите удалить всю базу данных, попробуйте следующее:manage.py flush
.
Предупреждение: это полностью удалит вашу базу данных, и вы потеряете все свои данные, поэтому, если это не важно, попробуйте.
Вот пример Makefile, чтобы делать приятные вещи с несколькими файлами настроек:
test:
python manage.py test --settings=my_project.test
db_drop:
echo 'DROP DATABASE my_project_development;' | ./manage.py dbshell
echo 'DROP DATABASE my_project_test;' | ./manage.py dbshell
db_create:
echo 'CREATE DATABASE my_project_development;' | ./manage.py dbshell
echo 'CREATE DATABASE my_project_test;' | ./manage.py dbshell
db_migrate:
python manage.py migrate --settings=my_project.base
python manage.py migrate --settings=my_project.test
db_reset: db_drop db_create db_migrate
.PHONY: test db_drop db_create db_migrate db_reset
Затем вы можете делать такие вещи, как:
$ make db_reset
Этот ответ предназначен для БД postgresql:
Выполнить: echo 'drop, владельцем которого является some_user ' | ./manage.py dbshell
ПРИМЕЧАНИЕ: some_user - это имя пользователя, которого вы используете для доступа к базе данных, см. Файл settings.py:
default_database = {
'ENGINE': 'django.db.backends.postgresql_psycopg2',
'NAME': 'somedbname',
'USER': 'some_user',
'PASSWORD': 'somepass',
'HOST': 'postgresql',
'PORT': '',
}
Если вы используете psql и у вас установлен django-more 2.0.0, вы можете сделать
manage.py reset_schema
Вот версия ответа @peter-g на юг. Я часто играю с необработанным sql, поэтому он пригодится как 0001_initial.py для любых сбитых с толку приложений. Он будет работать только с БД, которые поддерживают SHOW TABLES
(например, mysql). Замените что-то вроде, SELECT table_name FROM information_schema.tables WHERE table_schema = 'public';
если вы используете PostgreSQL. Кроме того, я часто делаю одно и то же как для миграции, так forwards
и для backwards
миграции.
from south.db import db
from south.v2 import SchemaMigration
from django.db.utils import DatabaseError
from os import path
from logging import getLogger
logger = getLogger(__name__)
class Migration(SchemaMigration):
def forwards(self, orm):
app_name = path.basename(path.split(path.split(path.abspath(__file__))[0])[0])
table_tuples = db.execute(r"SHOW TABLES;")
for tt in table_tuples:
table = tt[0]
if not table.startswith(app_name + '_'):
continue
try:
logger.warn('Deleting db table %s ...' % table)
db.delete_table(table)
except DatabaseError:
from traceback import format_exc
logger.error("Error running %s: \n %s" % (repr(self.forwards), format_exc()))
Однако коллеги / сокодеры убили бы меня, если бы узнали, что я это сделал.
Есть еще более простой ответ, если вы хотите удалить ВСЕ свои таблицы. Вам просто нужно перейти в свою папку, содержащую базу данных (которая может называться mydatabase.db), щелкнуть правой кнопкой мыши файл .db и нажать «удалить». Старомодный способ, надежный для работы.
Удаляет все таблицы и воссоздает их:
python manage.py sqlclear app1 app2 appN | sed -n "2,$p" | sed -n "$ !p" | sed "s/";/" CASCADE;/" | sed -e "1s/^/BEGIN;/" -e "$s/$/COMMIT;/" | python manage.py dbshell
python manage.py syncdb
Пояснение:
manage.py sqlclear
- «печатает операторы SQL DROP TABLE для заданных имен приложений»
sed -n "2,$p"
- захватывает все строки, кроме первой
sed -n "$ !p"
- захватывает все строки, кроме последней
sed "s/";/" CASCADE;/"
- заменяет все точки с запятой (;) на (CASCADE;)
sed -e "1s/^/BEGIN;/" -e "$s/$/COMMIT;/"
- вставляет (BEGIN;) как первый текст, вставляет (COMMIT;) как последний текст
manage.py dbshell
- "Запускает клиент командной строки для ядра базы данных, указанного в настройках ENGINE, с параметрами подключения, указанными в настройках ПОЛЬЗОВАТЕЛЯ, ПАРОЛЯ и т. Д."
manage.py syncdb
- «Создает таблицы базы данных для всех приложений в INSTALLED_APPS, таблицы которых еще не были созданы»
Зависимости:
Кредиты:
@Manoj Govindan и @Mike DeSimone для sqlclear подключены к dbshell
@jpic для 'sed "s /"; / "CASCADE; /"'
используйте команду "python manage.py sqlflush" в Windows 10, для других введите manage.py
Я бы порекомендовал вам установить django-extensions и использовать python manage.py reset_db
команду. Он делает именно то, что вы хотите.