There is a commonly encountered problem with Django and character sets. Windows applications such as Word/Outlook add characters that are not valid ISO-8859-1 and this results in problems saving a Django model to a database with a Latin 1 encoding. These characters should also be converted to avoid any display issues for the end users even if you are using a UTF-8 database encoding. The topic is well covered at Effbot and contains a list of appropriate conversions for each of hte problem characters.
Correcting this for all of your Django models is another issue. Do you handle the re-encoding during the form validation? The save for each model? Create a base class that all your models need to inherit from?
The simplest solution I have created leverages Signals
Combining the re-encoding method suggested at Effbot and the pre_save signal gives you the ability to convert all the problem characters right before the save occurs for any model.
kill_gremlins method replaced with Gabor's suggestion
1 2 3 4 5 6 7 8 9 10 11 12 | from django.db.models import signals
from django.dispatch import dispatcher
from django.db import models
def kill_gremlins(text):
return unicode(text).encode('iso-8859-1').decode('cp1252')
def charstrip(sender, instance):
for i_attr in instance._meta.fields:
if type(i_attr) == models.TextField or type(i_attr) == models.CharField:
if getattr(instance, i_attr.name):
setattr(instance, i_attr.name, kill_gremlins(getattr(instance, i_attr.name)))
dispatcher.connect(charstrip, signal=signals.pre_save)
|
More like this
- Template tag - list punctuation for a list of items by shapiromatron 1 year ago
- JSONRequestMiddleware adds a .json() method to your HttpRequests by cdcarter 1 year ago
- Serializer factory with Django Rest Framework by julio 1 year, 7 months ago
- Image compression before saving the new model / work with JPG, PNG by Schleidens 1 year, 8 months ago
- Help text hyperlinks by sa2812 1 year, 8 months ago
Comments
very nice/clean approach with the signals,
but the kill_gremlins function seems to be a little over-complex to me.
i mean, cannot we achieve the same with:
?
(assuming that we are dealing with mishandled unicode-strings.
#
Yes, that does appear to work correctly. I thought that route would drop the non iso compatible characters, but it appears to be correctly making the conversion. Very nice, I will update the method.
#
Note that encode("iso-8859-1") does not handle non-latin-1 characters in a Unicode string (obviously):
Maybe the usecase for this snippet is more limited, but it's not a full replacement for my (rather dated) code.
#
Please login first before commenting.