Les tests de Django se plaignent des tables manquantes
Quand je lance mon test de traiter avec mon Customer
modèle, j'obtiens l'erreur suivante:
DatabaseError: (1146, "Table 'test_mcif2.customer' doesn't exist")
Je ne suis pas totalement surpris parce que j'ai mon projet Django connecté à un "héritage" de la base de données. Depuis mes tables n'ont pas été créés "le Django façon," il n'est pas choquant que Django ne serait pas en mesure de parler sans finagling. Voici mon modèle:
from django.db import models
from django.db import connection, transaction
from mcif.models.mcif_model import McifModel
class Customer(McifModel):
class Meta:
db_table = u'customer'
app_name = 'mcif'
id = models.BigIntegerField(primary_key=True)
customer_number = models.CharField(unique=True, max_length=255)
social_security_number = models.CharField(unique=True, max_length=33)
name = models.CharField(unique=True, max_length=255)
phone = models.CharField(unique=True, max_length=255)
deceased = models.IntegerField(unique=True, null=True, blank=True)
do_not_mail = models.IntegerField(null=True, blank=True)
created_at = models.DateTimeField()
updated_at = models.DateTimeField()
def distinguishing_column_names(self):
return ['name', 'customer_number', 'social_security_number', 'phone']
Aucune idée pourquoi exactement ce qui ne fonctionne pas?
Edit: Voici McifModel
:
from django.db import models
from django.db import connection, transaction
class McifModel(models.Model):
class Meta:
abstract = True
def upsert(self):
cursor = connection.cursor()
cursor.execute(self.upsert_sql())
transaction.commit_unless_managed()
return self
def value_list(self):
return ','.join(map(lambda column_name: "'{c}'".format(c=getattr(self, column_name)), self.distinguishing_column_names()))
def upsert_sql(self):
column_names = ','.join(self.distinguishing_column_names())
return "INSERT IGNORE INTO {t} ({c}) VALUES ({v})".format(t=self._meta.db_table, c=column_names, v=self.value_list())
@classmethod
def save_from_row(cls, row):
object = cls()
map(lambda column_name: setattr(object, column_name, row.value(object._meta.db_table, column_name)), object.distinguishing_column_names())
return object.upsert()
Edit: j'ai pris tarequeh de conseils et de mettre le contenu de la Caktus fichier dans mcif/utils.py
. J'ai également réglé TEST_RUNNER = 'mcif.utils.ManagedModelTestRunner'
. Si je vais sur la console que j'ai pu vérifier que Customer
est pas géré:
>>> [m for m in get_models() if not m._meta.managed]
[<class 'mcif.models.customer.Customer'>]
Cependant, mon test se plaint encore que la table n'existe pas. Ce qui me manque?
Voici mon settings.py:
# Django settings for mcifdjango project.
DEBUG = True
TEMPLATE_DEBUG = DEBUG
ADMINS = (
('Jason Swett', '[email protected]'),
)
MANAGERS = ADMINS
DATABASES = {
'default': {
'ENGINE': 'django.db.backends.mysql', # Add 'postgresql_psycopg2', 'postgresql', 'mysql', 'sqlite3' or 'oracle'.
'NAME': 'xxxxx', # Or path to database file if using sqlite3.
'USER': 'xxxxx', # Not used with sqlite3.
'PASSWORD': 'xxxxx', # Not used with sqlite3.
'HOST': '', # Set to empty string for localhost. Not used with sqlite3.
'PORT': '', # Set to empty string for default. Not used with sqlite3.
}
}
# Local time zone for this installation. Choices can be found here:
# http://en.wikipedia.org/wiki/List_of_tz_zones_by_name
# although not all choices may be available on all operating systems.
# On Unix systems, a value of None will cause Django to use the same
# timezone as the operating system.
# If running in a Windows environment this must be set to the same as your
# system time zone.
TIME_ZONE = 'America/Chicago'
# Language code for this installation. All choices can be found here:
# http://www.i18nguy.com/unicode/language-identifiers.html
LANGUAGE_CODE = 'en-us'
SITE_ID = 1
# If you set this to False, Django will make some optimizations so as not
# to load the internationalization machinery.
USE_I18N = True
# If you set this to False, Django will not format dates, numbers and
# calendars according to the current locale
USE_L10N = True
# Absolute path to the directory that holds media.
# Example: "/home/media/media.lawrence.com/"
MEDIA_ROOT = ''
# URL that handles the media served from MEDIA_ROOT. Make sure to use a
# trailing slash if there is a path component (optional in other cases).
# Examples: "http://media.lawrence.com", "http://example.com/media/"
MEDIA_URL = ''
# URL prefix for admin media -- CSS, JavaScript and images. Make sure to use a
# trailing slash.
# Examples: "http://foo.com/media/", "/media/".
ADMIN_MEDIA_PREFIX = '/media/'
# Make this unique, and don't share it with anybody.
SECRET_KEY = '#7+qm%hqfe+z8ul5@x_i&sqmu!n=4sa0&i0_#)m99*w$fbk3%#'
# List of callables that know how to import templates from various sources.
TEMPLATE_LOADERS = (
'django.template.loaders.filesystem.Loader',
'django.template.loaders.app_directories.Loader',
# 'django.template.loaders.eggs.Loader',
)
MIDDLEWARE_CLASSES = (
'django.middleware.common.CommonMiddleware',
'django.contrib.sessions.middleware.SessionMiddleware',
'django.middleware.csrf.CsrfViewMiddleware',
'django.contrib.auth.middleware.AuthenticationMiddleware',
'django.contrib.messages.middleware.MessageMiddleware',
)
ROOT_URLCONF = 'mcifdjango.urls'
TEMPLATE_DIRS = (
# Put strings here, like "/home/html/django_templates" or "C:/www/django/templates".
# Always use forward slashes, even on Windows.
# Don't forget to use absolute paths, not relative paths.
)
INSTALLED_APPS = (
'django.contrib.auth',
'django.contrib.contenttypes',
'django.contrib.sessions',
'django.contrib.sites',
'django.contrib.messages',
'django.contrib.admin',
'django_extensions',
'mcif',
# Uncomment the next line to enable the admin:
# 'django.contrib.admin',
# Uncomment the next line to enable admin documentation:
# 'django.contrib.admindocs',
)
TEST_RUNNER = 'mcif.utils.ManagedModelTestRunner'
import os
ROOTDIR = os.path.abspath(os.path.dirname(__file__))
TEMPLATE_DIRS = (
# Put strings here, like "/home/html/django_templates" or "C:/www/django/templates".
# Always use forward slashes, even on Windows.
# Don't forget to use absolute paths, not relative paths.
ROOTDIR + '/mcif/templates',
)
Edit 2:
Voici mon Customer
classe maintenant:
from django.db import models
from django.db import connection, transaction
from mcif.models.mcif_model import McifModel
class Customer(McifModel):
class Meta:
db_table = u'customer'
managed = False
id = models.BigIntegerField(primary_key=True)
customer_number = models.CharField(unique=True, max_length=255)
social_security_number = models.CharField(unique=True, max_length=33)
name = models.CharField(unique=True, max_length=255)
phone = models.CharField(unique=True, max_length=255)
deceased = models.IntegerField(unique=True, null=True, blank=True)
do_not_mail = models.IntegerField(null=True, blank=True)
created_at = models.DateTimeField()
updated_at = models.DateTimeField()
def distinguishing_column_names(self):
return ['name', 'customer_number', 'social_security_number', 'phone']
Voici ce que j'obtiens quand je lance le test:
$ ./manage.py test mcif.CustomerUpsertTest
Creating test database 'default'...
Creating table auth_permission
Creating table auth_group_permissions
Creating table auth_group
Creating table auth_user_user_permissions
Creating table auth_user_groups
Creating table auth_user
Creating table auth_message
Creating table django_content_type
Creating table django_session
Creating table django_site
Creating table django_admin_log
Installing index for auth.Permission model
Installing index for auth.Group_permissions model
Installing index for auth.User_user_permissions model
Installing index for auth.User_groups model
Installing index for auth.Message model
Installing index for admin.LogEntry model
No fixtures found.
E
======================================================================
ERROR: test_upsert (mcif.tests.customer_upsert_test.CustomerUpsertTest)
----------------------------------------------------------------------
Traceback (most recent call last):
File "/home/jason/projects/mcifdjango/mcif/tests/customer_upsert_test.py", line 9, in test_upsert
customer.upsert()
File "/home/jason/projects/mcifdjango/mcif/models/mcif_model.py", line 11, in upsert
cursor.execute(self.upsert_sql())
File "/usr/lib/pymodules/python2.6/django/db/backends/mysql/base.py", line 86, in execute
return self.cursor.execute(query, args)
File "/usr/lib/pymodules/python2.6/MySQLdb/cursors.py", line 166, in execute
self.errorhandler(self, exc, value)
File "/usr/lib/pymodules/python2.6/MySQLdb/connections.py", line 35, in defaulterrorhandler
raise errorclass, errorvalue
DatabaseError: (1146, "Table 'test_mcif_django.customer' doesn't exist")
----------------------------------------------------------------------
Ran 1 test in 3.724s
FAILED (errors=1)
Destroying test database 'default'...
source d'informationauteur Jason Swett
Vous devez vous connecter pour publier un commentaire.
Depuis que vous utilisez une base de données existante, vous êtes probablement ne pas ajouter le nom de l'application à INSTALLED_APPS. Si une application n'est pas inclus dans INSTALLED_APPS, les tableaux, pour les applications les modèles ne seront pas créés sur syncdb. Cela fonctionne pour vous dans la production, car vous avez déjà une table, mais pas dans l'environnement de test.
Vous pouvez adopter l'une des opérations suivantes:
La supermonkeypatch façon: app_name de la Clientèle de la classe Meta, mettre le modèle dans un models.py fichier à l'intérieur d'un module python nom mcif, et ajouter mcif à INSTALLED_APPS - juste pour le plaisir de tester
De la plus belle manière: Étendre DjangoTestSuiteRunner et remplacer setup_test_environment appeler super et puis créer votre héritage table manuellement dans le test DB.
La plus belle façon: Mettre votre modèle en bien nommé app module. Supprimer app_name à partir du modèle de Méta, mais ajouter managed=False docs. Inclure nom de l'application dans INSTALLED_APPS. Maintenant, django va pas créer une table pour ce modèle. Ensuite, utilisez cette extrait de nice la Caktus groupe de gens ont compilé pour exécuter vos tests.
Cheers!
Edition - Comment utiliser le substituée DjangoTestSuiteRunner
Vous aurez besoin d'au moins Django 1.2 pour cela.
Copiez le code de la ici. Le mettre dans utils.py à l'intérieur de la mcif app.
Ajouter/modifier les éléments suivants dans settings.py:
Maintenant, lorsque vous exécutez des tests, tous les non géré tables seront traités comme des géré tableau uniquement pour la durée de l'essai. Donc, les tables seront créées avant l'exécution des tests.
Avis de cette partie du code, c'est là que la magie se produit.
2e Édition: les éventuels Pièges
Vérifiez les points suivants:
3e Edit: Débogage
À l'intérieur de mcif.utils.ManagedModelTestRunner remplacer setup_test_environment fonction avec le suivant et laissez-moi savoir si le résultat de votre test de changements:
Les solutions présentées par tarequeh a fonctionné pour moi après primordial DATABASE_ROUTERS.
Je suis à l'aide de routeurs afin d'empêcher l'écriture sur l'héritage de la base de données. Pour contourner cela, j'ai créé un test_settings fichier avec le contenu suivant:
Puis lors de l'exécution des tests:
Il n'y a pas assez d'info ci-dessus pour répondre à votre première question. Cependant, une fois cette question résolue, vous aurez probablement souhaitez installer django-extensions pour la raison suivante: Il est incroyablement utile
sqldiff
commande qui vous informera s'il existe une incompatibilité entre l'héritage de la base de données et votre modèle d'application.