Django y error No module named models

Esta mañana estaba refactorizando un poco el código de la aplicación de tiendas antes de subirlo al repositorio de un cliente y me he encontrado con el error que describe el título y que me ha traído de cabeza toda la mañana.

Uno de los cambios que estaba haciendo al código era sustituir líneas del tipo: from tienda.checkout.models import * por from checkout.models import * para mejorar la portabilidad de las aplicaciones, pero a raíz de eso el sitio dejó de funcionar y lo único que recibía era ese mensaje de error.

A la postre resulta que tenía una estructura de directorios y ficheros tal que así:
cart\
   __init__.py
   admin.py
   models.py
   cart.py
   views.py
checkout\
   __init__.py
   admin.py
   models.py
   checkout.py
   views.py

etc.

El problema ocurría cuando python interpretaba los ficheros cart.py y checkout.py (entre otros) e intentaba luego buscar modelos en ellos al tener el mismo nombre que el directorio en el que se encuentran.

La solución una vez sabido esto fué sencilla, cambiar los nombres por cart_functions.py y checkout_functions.py a los archivos y en todos aquellos lugares en los que se usaban.

Error en Django Admin al buscar: Related Field has invalid lookup: icontains

Mientras buscaba en un gran listado de productos desde la administración de django me he topado con el siguiente error:

Related Field has invalid lookup: icontains

A la postre resulta que el problema venía por cómo estaba definida la clase de administración del modelo:

class ProductAdmin(admin.ModelAdmin):
   # sets values for how the admin site lists your products
   list_display = ('name', 'price', 'brand', 'stock', 'is_pack', 'created_at', 'updated_at',)
   ...
   search_fields = ['name', 'categories', 'brand', 'internal_reference', 'description']
   ...

Y es que ‘categories’ es una clave externa (ForeignKey) y por eso hay que definir sobre qué campo asociado se buscará, se soluciona, por ejemplo, con:

   search_fields = ['name', 'categories__name', 'brand', 'internal_reference', 'description']

Django: Flatpages devuelven error 404

Los últimos proyectos en los que estoy trabajando van sobre una plataforma de comercio electrónico.
Como tal, a la hora de mostrar errores a los visitantes/clientes, es importante no desconcertarlos y hacer que pierdan la confianza en el sitio web, por lo cual se personalizan las páginas de error para que se integren en el resto del sitio y además se aprovecha para mostrarle productos que podrían interesarle.

Debido a estos listados de productos no podemos simplemente renderizar un template para los errores, sino que he implementado un manejador para los errores: file_not_found_404 y handler_500. Estos hacían bien su trabajo mientras realizaba el desarrollo en local, sin embargo, cuando se pasaba a pruebas en el entorno de producción todas las páginas estáticas definidas con el framework flatpages direccionaban a la página de error 404.

Después de muchas vueltas, pruebas, desactivar el manejador, revisar logs y volver a probar y a dar vueltas ví que en los logs de acceso las páginas de errores 404 devolvían un status 200 (Ok), lo cual finalmente me dió la pista que necesitaba: tenía que sobreescribir el status_code de la respuesta a 404, ya que por defecto el valor es 200.

El manejador quedó con algo parecido a esto:

def file_not_found_404(request):
page_title = _('Page Not Found')

featured = Product.featured.all()[0:FEATURED_PER_ROW]
bestseller = Product.bestseller.all()[0:BESTSELLER_PER_ROW]
recommended = Product.recommended.all()[0:RECOMMENDED_PER_ROW]

response = render_to_response('404.html', locals(), context_instance=RequestContext(request))
response.status_code = 404
return response

Django: acceder al value de los campos de un formulario

Recientemente he tenido que preparar una plantilla en django que mostrara una serie de imágenes del usuario y un formulario adyacente en el que podían sustituir esta, así que me encontraba con el problema de mostrar al usuario sus fotos actuales, si es que tenían.

Para ello se crea un formulario heredando del modelo:

forms.py:
class PhotosForm(forms.ModelForm):
   photo = forms.ImageField(label=_(u'Fotografía'), required=False)
   current = forms.CharField(label='', widget=forms.HiddenInput(), required=False)

   class Meta:
      model = ProfilePhotos
      exclude = ('profile','accepted',)

Luego, en la vista, puedes consultar el perfil del usuario en cuestión y ver si tiene fotografías:

views.py
@login_required
def edit_photos(request, template_name='members/photos.html'):
   page_title = _(u'Editar fotografías')
   PhotosFormSet = formset_factory(PhotosForm, extra=5, max_num=5)
   profile = request.user.profiles_set.filter(accepted=True).latest('date')

   if request.method == 'POST':
       ...
   else:
      profile_photos = profile.profilephotos_set.filter(accepted=True)
      if profile_photos:
         data = []
         for photo in profile_photos:
            data.append({'current': photo.photo.name})
         formset = PhotosFormSet(initial = data)
      else:
         formset = PhotosFormSet()

Y finalmente, en la plantilla se puede acceder a los valores de los campos mediante

photos.html:
   {{ formset.management_form }}
   {% for photos_form in formset.forms %}
   

   {% if photos_form.initial.current %}
      [img src="{{photos_form.initial.current }}" alt="" /]
   {% else %}
      [img src="default.jpg" alt="" /]
   {% endif %}
   

   ...

Y del mismo modo, se puede acceder mediante form.initial.nombre_clave a cualquier otro dato inicial.

Si en lugar de proporcionar datos iniciales lo que hacemos es asociar el formulario a una instancia ( form.is_bound == True ) entonces podremos acceder a los datos mediante form.data.nombre_clave

Bah, como salta a la vista, el formateo de código de Wordpress no es muy bueno…

Actualización de MotrilCenter

De cara a las fiestas Motrilcenter ha lanzado una serie de planes promocionales para facilitar el acceso a sus viviendas en Motril y he estado actualizando la página.

Actualización Diciembre 09 Motrilcenter

Actualización Diciembre 09 Motrilcenter

Con esto me he dado cuenta de una cosa ¡Aún no le había puesto favicon a la página!