mirror of
				https://github.com/django/django.git
				synced 2025-10-25 06:36:07 +00:00 
			
		
		
		
	newforms-admin: Merged from trunk up to [5625]. This includes the Unicode
merge, however, not all of admin/ (and none of admindocs/) has been ported and checked yet. git-svn-id: http://code.djangoproject.com/svn/django/branches/newforms-admin@5627 bcc190cf-cafb-0310-a4f2-bffc1f526a37
This commit is contained in:
		
							
								
								
									
										10
									
								
								AUTHORS
									
									
									
									
									
								
							
							
						
						
									
										10
									
								
								AUTHORS
									
									
									
									
									
								
							| @@ -44,6 +44,7 @@ answer newbie questions, and generally made Django that much better: | |||||||
|     alang@bright-green.com |     alang@bright-green.com | ||||||
|     Marty Alchin <gulopine@gamemusic.org> |     Marty Alchin <gulopine@gamemusic.org> | ||||||
|     Daniel Alves Barbosa de Oliveira Vaz <danielvaz@gmail.com> |     Daniel Alves Barbosa de Oliveira Vaz <danielvaz@gmail.com> | ||||||
|  |     AgarFu <heaven@croasanaso.sytes.net> | ||||||
|     Andreas |     Andreas | ||||||
|     andy@jadedplanet.net |     andy@jadedplanet.net | ||||||
|     Fabrice Aneche <akh@nobugware.com> |     Fabrice Aneche <akh@nobugware.com> | ||||||
| @@ -60,9 +61,11 @@ answer newbie questions, and generally made Django that much better: | |||||||
|     Ben <afternoon@uk2.net> |     Ben <afternoon@uk2.net> | ||||||
|     Paul Bissex <http://e-scribe.com/> |     Paul Bissex <http://e-scribe.com/> | ||||||
|     Simon Blanchard |     Simon Blanchard | ||||||
|  |     boobsd@gmail.com | ||||||
|     Andrew Brehaut <http://brehaut.net/blog> |     Andrew Brehaut <http://brehaut.net/blog> | ||||||
|     brut.alll@gmail.com |     brut.alll@gmail.com | ||||||
|     Jonathan Buchanan <jonathan.buchanan@gmail.com> |     Jonathan Buchanan <jonathan.buchanan@gmail.com> | ||||||
|  |     Ricardo Javier Cárdenes Medina <ricardo.cardenes@gmail.com> | ||||||
|     Antonio Cavedoni <http://cavedoni.com/> |     Antonio Cavedoni <http://cavedoni.com/> | ||||||
|     C8E |     C8E | ||||||
|     Chris Chamberlin <dja@cdc.msbx.net> |     Chris Chamberlin <dja@cdc.msbx.net> | ||||||
| @@ -106,13 +109,16 @@ answer newbie questions, and generally made Django that much better: | |||||||
|     Eric Floehr <eric@intellovations.com> |     Eric Floehr <eric@intellovations.com> | ||||||
|     Jorge Gajon <gajon@gajon.org> |     Jorge Gajon <gajon@gajon.org> | ||||||
|     gandalf@owca.info |     gandalf@owca.info | ||||||
|  |     Marc Garcia <marc.garcia@accopensys.com> | ||||||
|     Baishampayan Ghose |     Baishampayan Ghose | ||||||
|     glin@seznam.cz |     glin@seznam.cz | ||||||
|     martin.glueck@gmail.com |     martin.glueck@gmail.com | ||||||
|     GomoX <gomo@datafull.com> |     GomoX <gomo@datafull.com> | ||||||
|  |     Mario Gonzalez <gonzalemario @t gmail.com> | ||||||
|     Simon Greenhill <dev@simon.net.nz> |     Simon Greenhill <dev@simon.net.nz> | ||||||
|     Owen Griffiths |     Owen Griffiths | ||||||
|     Espen Grindhaug <http://grindhaug.org/> |     Espen Grindhaug <http://grindhaug.org/> | ||||||
|  |     Thomas Güttler <hv@tbz-pariv.de> | ||||||
|     Brian Harring <ferringb@gmail.com> |     Brian Harring <ferringb@gmail.com> | ||||||
|     Brant Harris |     Brant Harris | ||||||
|     Hawkeye |     Hawkeye | ||||||
| @@ -141,12 +147,15 @@ answer newbie questions, and generally made Django that much better: | |||||||
|     Sune Kirkeby <http://ibofobi.dk/> |     Sune Kirkeby <http://ibofobi.dk/> | ||||||
|     Bastian Kleineidam <calvin@debian.org> |     Bastian Kleineidam <calvin@debian.org> | ||||||
|     Cameron Knight (ckknight) |     Cameron Knight (ckknight) | ||||||
|  |     Nena Kojadin <nena@kiberpipa.org> | ||||||
|  |     Igor Kolar <ike@email.si> | ||||||
|     Gasper Koren |     Gasper Koren | ||||||
|     Martin Kosír <martin@martinkosir.net> |     Martin Kosír <martin@martinkosir.net> | ||||||
|     Meir Kriheli <http://mksoft.co.il/> |     Meir Kriheli <http://mksoft.co.il/> | ||||||
|     Bruce Kroeze <http://coderseye.com/> |     Bruce Kroeze <http://coderseye.com/> | ||||||
|     Joseph Kocherhans |     Joseph Kocherhans | ||||||
|     konrad@gwu.edu |     konrad@gwu.edu | ||||||
|  |     kurtiss@meetro.com | ||||||
|     lakin.wecker@gmail.com |     lakin.wecker@gmail.com | ||||||
|     Nick Lane <nick.lane.au@gmail.com> |     Nick Lane <nick.lane.au@gmail.com> | ||||||
|     Stuart Langridge <http://www.kryogenix.org/> |     Stuart Langridge <http://www.kryogenix.org/> | ||||||
| @@ -214,6 +223,7 @@ answer newbie questions, and generally made Django that much better: | |||||||
|     scott@staplefish.com |     scott@staplefish.com | ||||||
|     serbaut@gmail.com |     serbaut@gmail.com | ||||||
|     Pete Shinners <pete@shinners.org> |     Pete Shinners <pete@shinners.org> | ||||||
|  |     Jozko Skrablin <jozko.skrablin@gmail.com> | ||||||
|     SmileyChris <smileychris@gmail.com> |     SmileyChris <smileychris@gmail.com> | ||||||
|     smurf@smurf.noris.de |     smurf@smurf.noris.de | ||||||
|     sopel |     sopel | ||||||
|   | |||||||
| @@ -103,7 +103,7 @@ def make_messages(): | |||||||
|                         open(os.path.join(dirpath, '%s.py' % file), "wb").write(templatize(src)) |                         open(os.path.join(dirpath, '%s.py' % file), "wb").write(templatize(src)) | ||||||
|                         thefile = '%s.py' % file |                         thefile = '%s.py' % file | ||||||
|                     if verbose: sys.stdout.write('processing file %s in %s\n' % (file, dirpath)) |                     if verbose: sys.stdout.write('processing file %s in %s\n' % (file, dirpath)) | ||||||
|                     cmd = 'xgettext %s -d %s -L Python --keyword=gettext_noop --keyword=gettext_lazy --keyword=ngettext_lazy:1,2 --from-code UTF-8 -o - "%s"' % ( |                     cmd = 'xgettext %s -d %s -L Python --keyword=gettext_noop --keyword=gettext_lazy --keyword=ngettext_lazy:1,2 --keyword=ugettext_noop --keyword=ugettext_lazy --keyword=ungettext_lazy:1,2 --from-code UTF-8 -o - "%s"' % ( | ||||||
|                         os.path.exists(potfile) and '--omit-header' or '', domain, os.path.join(dirpath, thefile)) |                         os.path.exists(potfile) and '--omit-header' or '', domain, os.path.join(dirpath, thefile)) | ||||||
|                     (stdin, stdout, stderr) = os.popen3(cmd, 'b') |                     (stdin, stdout, stderr) = os.popen3(cmd, 'b') | ||||||
|                     msgs = stdout.read() |                     msgs = stdout.read() | ||||||
|   | |||||||
| @@ -97,6 +97,9 @@ MANAGERS = ADMINS | |||||||
| DEFAULT_CONTENT_TYPE = 'text/html' | DEFAULT_CONTENT_TYPE = 'text/html' | ||||||
| DEFAULT_CHARSET = 'utf-8' | DEFAULT_CHARSET = 'utf-8' | ||||||
|  |  | ||||||
|  | # Encoding of files read from disk (template and initial SQL files). | ||||||
|  | FILE_CHARSET = 'utf-8' | ||||||
|  |  | ||||||
| # E-mail address that error messages come from. | # E-mail address that error messages come from. | ||||||
| SERVER_EMAIL = 'root@localhost' | SERVER_EMAIL = 'root@localhost' | ||||||
|  |  | ||||||
| @@ -280,6 +283,7 @@ SESSION_EXPIRE_AT_BROWSER_CLOSE = False   # Whether sessions expire when a user | |||||||
| # possible values. | # possible values. | ||||||
| CACHE_BACKEND = 'simple://' | CACHE_BACKEND = 'simple://' | ||||||
| CACHE_MIDDLEWARE_KEY_PREFIX = '' | CACHE_MIDDLEWARE_KEY_PREFIX = '' | ||||||
|  | CACHE_MIDDLEWARE_SECONDS = 600 | ||||||
|  |  | ||||||
| #################### | #################### | ||||||
| # COMMENTS         # | # COMMENTS         # | ||||||
|   | |||||||
										
											Binary file not shown.
										
									
								
							| @@ -237,7 +237,7 @@ msgstr "Coreà" | |||||||
|  |  | ||||||
| #: conf/global_settings.py:61 | #: conf/global_settings.py:61 | ||||||
| msgid "Kannada" | msgid "Kannada" | ||||||
| msgstr "" | msgstr "Canès" | ||||||
|  |  | ||||||
| #: conf/global_settings.py:62 | #: conf/global_settings.py:62 | ||||||
| msgid "Latvian" | msgid "Latvian" | ||||||
| @@ -1230,7 +1230,7 @@ msgstr "Pot editar-lo de nou abaix." | |||||||
|  |  | ||||||
| #: contrib/admin/views/auth.py:30 | #: contrib/admin/views/auth.py:30 | ||||||
| msgid "Add user" | msgid "Add user" | ||||||
| msgstr "Agregar usuari" | msgstr "Afegir usuari" | ||||||
|  |  | ||||||
| #: contrib/admin/views/auth.py:57 | #: contrib/admin/views/auth.py:57 | ||||||
| msgid "Password changed successfully." | msgid "Password changed successfully." | ||||||
| @@ -1248,12 +1248,12 @@ msgstr "Lloc administratiu" | |||||||
| #: contrib/admin/views/main.py:276 contrib/admin/views/main.py:361 | #: contrib/admin/views/main.py:276 contrib/admin/views/main.py:361 | ||||||
| #, python-format | #, python-format | ||||||
| msgid "You may add another %s below." | msgid "You may add another %s below." | ||||||
| msgstr "Pot agregar un altre %s abaix." | msgstr "Pot afegir un altre %s a baix." | ||||||
|  |  | ||||||
| #: contrib/admin/views/main.py:294 | #: contrib/admin/views/main.py:294 | ||||||
| #, python-format | #, python-format | ||||||
| msgid "Add %s" | msgid "Add %s" | ||||||
| msgstr "Agregar %s" | msgstr "Afegir %s" | ||||||
|  |  | ||||||
| #: contrib/admin/views/main.py:340 | #: contrib/admin/views/main.py:340 | ||||||
| #, python-format | #, python-format | ||||||
|   | |||||||
										
											Binary file not shown.
										
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
										
											Binary file not shown.
										
									
								
							| @@ -5,8 +5,8 @@ msgid "" | |||||||
| msgstr "" | msgstr "" | ||||||
| "Project-Id-Version: DJANGO-JS\n" | "Project-Id-Version: DJANGO-JS\n" | ||||||
| "Report-Msgid-Bugs-To: \n" | "Report-Msgid-Bugs-To: \n" | ||||||
| "POT-Creation-Date: 2005-12-09 11:51+0100\n" | "POT-Creation-Date: 2007-07-02 13:55+0200\n" | ||||||
| "PO-Revision-Date: 2007-03-31 21:29+0100\n" | "PO-Revision-Date: 2007-07-02 13:56+0100\n" | ||||||
| "Last-Translator: Gasper Koren <skrat@owca.info>\n" | "Last-Translator: Gasper Koren <skrat@owca.info>\n" | ||||||
| "Language-Team: SLOVENIAN <lugos-slo@lugos.si>\n" | "Language-Team: SLOVENIAN <lugos-slo@lugos.si>\n" | ||||||
| "MIME-Version: 1.0\n" | "MIME-Version: 1.0\n" | ||||||
| @@ -15,6 +15,19 @@ msgstr "" | |||||||
| "X-Poedit-Language: Slovenian\n" | "X-Poedit-Language: Slovenian\n" | ||||||
| "X-Poedit-Country: SLOVENIA\n" | "X-Poedit-Country: SLOVENIA\n" | ||||||
|  |  | ||||||
|  | #: contrib/admin/media/js/calendar.js:24 | ||||||
|  | #: contrib/admin/media/js/dateparse.js:32 | ||||||
|  | msgid "January February March April May June July August September October November December" | ||||||
|  | msgstr "Januar Februar Marec April Maj Junij Julij Avgust September Oktober November December" | ||||||
|  |  | ||||||
|  | #: contrib/admin/media/js/calendar.js:25 | ||||||
|  | msgid "S M T W T F S" | ||||||
|  | msgstr "N P T S Č P S" | ||||||
|  |  | ||||||
|  | #: contrib/admin/media/js/dateparse.js:33 | ||||||
|  | msgid "Sunday Monday Tuesday Wednesday Thursday Friday Saturday" | ||||||
|  | msgstr "Nedelja Ponedeljek Torek Sreda Četrtek Petek Sobota" | ||||||
|  |  | ||||||
| #: contrib/admin/media/js/SelectFilter2.js:33 | #: contrib/admin/media/js/SelectFilter2.js:33 | ||||||
| #, perl-format | #, perl-format | ||||||
| msgid "Available %s" | msgid "Available %s" | ||||||
| @@ -45,63 +58,59 @@ msgstr "Izberite in kliknite" | |||||||
| msgid "Clear all" | msgid "Clear all" | ||||||
| msgstr "Izbriši vse" | msgstr "Izbriši vse" | ||||||
|  |  | ||||||
| #: contrib/admin/media/js/dateparse.js:26 | #: contrib/admin/media/js/admin/DateTimeShortcuts.js:47 | ||||||
| #: contrib/admin/media/js/calendar.js:24 | #: contrib/admin/media/js/admin/DateTimeShortcuts.js:81 | ||||||
| msgid "January February March April May June July August September October November December" |  | ||||||
| msgstr "Januar Februar Marec April Maj Junij Julij Avgust September Oktober November December" |  | ||||||
|  |  | ||||||
| #: contrib/admin/media/js/dateparse.js:27 |  | ||||||
| msgid "Sunday Monday Tuesday Wednesday Thursday Friday Saturday" |  | ||||||
| msgstr "Nedelja Ponedeljek Torek Sreda Četrtek Petek Sobota" |  | ||||||
|  |  | ||||||
| #: contrib/admin/media/js/calendar.js:25 |  | ||||||
| msgid "S M T W T F S" |  | ||||||
| msgstr "N P T S Č P S" |  | ||||||
|  |  | ||||||
| #: contrib/admin/media/js/admin/DateTimeShortcuts.js:45 |  | ||||||
| #: contrib/admin/media/js/admin/DateTimeShortcuts.js:80 |  | ||||||
| msgid "Now" | msgid "Now" | ||||||
| msgstr "Sedaj" | msgstr "Sedaj" | ||||||
|  |  | ||||||
| #: contrib/admin/media/js/admin/DateTimeShortcuts.js:48 | #: contrib/admin/media/js/admin/DateTimeShortcuts.js:51 | ||||||
| msgid "Clock" | msgid "Clock" | ||||||
| msgstr "URA" | msgstr "URA" | ||||||
|  |  | ||||||
| #: contrib/admin/media/js/admin/DateTimeShortcuts.js:77 | #: contrib/admin/media/js/admin/DateTimeShortcuts.js:78 | ||||||
| msgid "Choose a time" | msgid "Choose a time" | ||||||
| msgstr "Izberite čas" | msgstr "Izberite čas" | ||||||
|  |  | ||||||
| #: contrib/admin/media/js/admin/DateTimeShortcuts.js:81 | #: contrib/admin/media/js/admin/DateTimeShortcuts.js:82 | ||||||
| msgid "Midnight" | msgid "Midnight" | ||||||
| msgstr "Polnoč" | msgstr "Polnoč" | ||||||
|  |  | ||||||
| #: contrib/admin/media/js/admin/DateTimeShortcuts.js:82 | #: contrib/admin/media/js/admin/DateTimeShortcuts.js:83 | ||||||
| msgid "6 a.m." | msgid "6 a.m." | ||||||
| msgstr "Ob 6h" | msgstr "Ob 6h" | ||||||
|  |  | ||||||
| #: contrib/admin/media/js/admin/DateTimeShortcuts.js:83 | #: contrib/admin/media/js/admin/DateTimeShortcuts.js:84 | ||||||
| msgid "Noon" | msgid "Noon" | ||||||
| msgstr "Opoldne" | msgstr "Opoldne" | ||||||
|  |  | ||||||
| #: contrib/admin/media/js/admin/DateTimeShortcuts.js:87 | #: contrib/admin/media/js/admin/DateTimeShortcuts.js:88 | ||||||
| #: contrib/admin/media/js/admin/DateTimeShortcuts.js:168 | #: contrib/admin/media/js/admin/DateTimeShortcuts.js:183 | ||||||
| msgid "Cancel" | msgid "Cancel" | ||||||
| msgstr "Prekliči" | msgstr "Prekliči" | ||||||
|  |  | ||||||
| #: contrib/admin/media/js/admin/DateTimeShortcuts.js:111 | #: contrib/admin/media/js/admin/DateTimeShortcuts.js:128 | ||||||
| #: contrib/admin/media/js/admin/DateTimeShortcuts.js:162 | #: contrib/admin/media/js/admin/DateTimeShortcuts.js:177 | ||||||
| msgid "Today" | msgid "Today" | ||||||
| msgstr "Danes" | msgstr "Danes" | ||||||
|  |  | ||||||
| #: contrib/admin/media/js/admin/DateTimeShortcuts.js:114 | #: contrib/admin/media/js/admin/DateTimeShortcuts.js:132 | ||||||
| msgid "Calendar" | msgid "Calendar" | ||||||
| msgstr "Koledar" | msgstr "Koledar" | ||||||
|  |  | ||||||
| #: contrib/admin/media/js/admin/DateTimeShortcuts.js:160 | #: contrib/admin/media/js/admin/DateTimeShortcuts.js:175 | ||||||
| msgid "Yesterday" | msgid "Yesterday" | ||||||
| msgstr "Včeraj" | msgstr "Včeraj" | ||||||
|  |  | ||||||
| #: contrib/admin/media/js/admin/DateTimeShortcuts.js:164 | #: contrib/admin/media/js/admin/DateTimeShortcuts.js:179 | ||||||
| msgid "Tomorrow" | msgid "Tomorrow" | ||||||
| msgstr "Jutri" | msgstr "Jutri" | ||||||
|  |  | ||||||
|  | #: contrib/admin/media/js/admin/CollapsedFieldsets.js:34 | ||||||
|  | #: contrib/admin/media/js/admin/CollapsedFieldsets.js:72 | ||||||
|  | msgid "Show" | ||||||
|  | msgstr "Prikaži" | ||||||
|  |  | ||||||
|  | #: contrib/admin/media/js/admin/CollapsedFieldsets.js:63 | ||||||
|  | msgid "Hide" | ||||||
|  | msgstr "Skrij" | ||||||
|  |  | ||||||
|   | |||||||
| @@ -7,6 +7,8 @@ certain test -- e.g. being a DateField or ForeignKey. | |||||||
| """ | """ | ||||||
|  |  | ||||||
| from django.db import models | from django.db import models | ||||||
|  | from django.utils.encoding import smart_unicode, iri_to_uri | ||||||
|  | from django.utils.translation import ugettext as _ | ||||||
| import datetime | import datetime | ||||||
|  |  | ||||||
| class FilterSpec(object): | class FilterSpec(object): | ||||||
| @@ -37,12 +39,12 @@ class FilterSpec(object): | |||||||
|     def output(self, cl): |     def output(self, cl): | ||||||
|         t = [] |         t = [] | ||||||
|         if self.has_output(): |         if self.has_output(): | ||||||
|             t.append(_('<h3>By %s:</h3>\n<ul>\n') % self.title()) |             t.append(_(u'<h3>By %s:</h3>\n<ul>\n') % self.title()) | ||||||
|  |  | ||||||
|             for choice in self.choices(cl): |             for choice in self.choices(cl): | ||||||
|                 t.append('<li%s><a href="%s">%s</a></li>\n' % \ |                 t.append(u'<li%s><a href="%s">%s</a></li>\n' % \ | ||||||
|                     ((choice['selected'] and ' class="selected"' or ''), |                     ((choice['selected'] and ' class="selected"' or ''), | ||||||
|                      choice['query_string'] , |                      iri_to_uri(choice['query_string']), | ||||||
|                      choice['display'])) |                      choice['display'])) | ||||||
|             t.append('</ul>\n\n') |             t.append('</ul>\n\n') | ||||||
|         return "".join(t) |         return "".join(t) | ||||||
| @@ -70,7 +72,7 @@ class RelatedFilterSpec(FilterSpec): | |||||||
|                'display': _('All')} |                'display': _('All')} | ||||||
|         for val in self.lookup_choices: |         for val in self.lookup_choices: | ||||||
|             pk_val = getattr(val, self.field.rel.to._meta.pk.attname) |             pk_val = getattr(val, self.field.rel.to._meta.pk.attname) | ||||||
|             yield {'selected': self.lookup_val == str(pk_val), |             yield {'selected': self.lookup_val == smart_unicode(pk_val), | ||||||
|                    'query_string': cl.get_query_string({self.lookup_kwarg: pk_val}), |                    'query_string': cl.get_query_string({self.lookup_kwarg: pk_val}), | ||||||
|                    'display': val} |                    'display': val} | ||||||
|  |  | ||||||
| @@ -87,7 +89,7 @@ class ChoicesFilterSpec(FilterSpec): | |||||||
|                'query_string': cl.get_query_string({}, [self.lookup_kwarg]), |                'query_string': cl.get_query_string({}, [self.lookup_kwarg]), | ||||||
|                'display': _('All')} |                'display': _('All')} | ||||||
|         for k, v in self.field.choices: |         for k, v in self.field.choices: | ||||||
|             yield {'selected': str(k) == self.lookup_val, |             yield {'selected': smart_unicode(k) == self.lookup_val, | ||||||
|                     'query_string': cl.get_query_string({self.lookup_kwarg: k}), |                     'query_string': cl.get_query_string({self.lookup_kwarg: k}), | ||||||
|                     'display': v} |                     'display': v} | ||||||
|  |  | ||||||
| @@ -168,7 +170,7 @@ class AllValuesFilterSpec(FilterSpec): | |||||||
|                'query_string': cl.get_query_string({}, [self.field.name]), |                'query_string': cl.get_query_string({}, [self.field.name]), | ||||||
|                'display': _('All')} |                'display': _('All')} | ||||||
|         for val in self.lookup_choices: |         for val in self.lookup_choices: | ||||||
|             val = str(val[self.field.name]) |             val = smart_unicode(val[self.field.name]) | ||||||
|             yield {'selected': self.lookup_val == val, |             yield {'selected': self.lookup_val == val, | ||||||
|                    'query_string': cl.get_query_string({self.field.name: val}), |                    'query_string': cl.get_query_string({self.field.name: val}), | ||||||
|                    'display': val} |                    'display': val} | ||||||
|   | |||||||
| @@ -1,15 +1,116 @@ | |||||||
|  | var LATIN_MAP = { | ||||||
|  |     'À': 'A', 'Á': 'A', 'Â': 'A', 'Ã': 'A', 'Ä': 'A', 'Å': 'A', 'Æ': 'AE', 'Ç': | ||||||
|  |     'C', 'È': 'E', 'É': 'E', 'Ê': 'E', 'Ë': 'E', 'Ì': 'I', 'Í': 'I', 'Î': 'I', | ||||||
|  |     'Ï': 'I', 'Ð': 'D', 'Ñ': 'N', 'Ò': 'O', 'Ó': 'O', 'Ô': 'O', 'Õ': 'O', 'Ö': | ||||||
|  |     'O', 'Ø': 'O', 'Ù': 'U', 'Ú': 'U', 'Û': 'U', 'Ü': 'U', 'Ý': 'Y', 'Þ': 'TH', | ||||||
|  |     'ß': 'ss', 'à':'a', 'á':'a', 'â': 'a', 'ã': 'a', 'ä': 'a', 'å': 'a', 'æ': | ||||||
|  |     'ae', 'ç': 'c', 'è': 'e', 'é': 'e', 'ê': 'e', 'ë': 'e', 'ì': 'i', 'í': 'i', | ||||||
|  |     'î': 'i', 'ï': 'i', 'ð': 'o', 'ñ': 'n', 'ò': 'o', 'ó': 'o', 'ô': 'o', 'õ': | ||||||
|  |     'o', 'ö': 'o', 'ø': 'o', 'ù': 'u', 'ú': 'u', 'û': 'u', 'ü': 'u', 'ý': 'y', | ||||||
|  |     'þ': 'th', 'ÿ': 'y' | ||||||
|  | } | ||||||
|  | var LATIN_SYMBOLS_MAP = { | ||||||
|  |     '©':'(c)' | ||||||
|  | } | ||||||
|  | var GREEK_MAP = { | ||||||
|  |     'α':'a', 'β':'b', 'γ':'g', 'δ':'d', 'ε':'e', 'ζ':'z', 'η':'h', 'θ':'8', | ||||||
|  |     'ι':'i', 'κ':'k', 'λ':'l', 'μ':'m', 'ν':'n', 'ξ':'3', 'ο':'o', 'π':'p', | ||||||
|  |     'ρ':'r', 'σ':'s', 'τ':'t', 'υ':'y', 'φ':'f', 'χ':'x', 'ψ':'ps', 'ω':'w', | ||||||
|  |     'ά':'a', 'έ':'e', 'ί':'i', 'ό':'o', 'ύ':'y', 'ή':'h', 'ώ':'w', 'ς':'s', | ||||||
|  |     'ϊ':'i', 'ΰ':'y', 'ϋ':'y', 'ΐ':'i', | ||||||
|  |     'Α':'A', 'Β':'B', 'Γ':'G', 'Δ':'D', 'Ε':'E', 'Ζ':'Z', 'Η':'H', 'Θ':'8', | ||||||
|  |     'Ι':'I', 'Κ':'K', 'Λ':'L', 'Μ':'M', 'Ν':'N', 'Ξ':'3', 'Ο':'O', 'Π':'P', | ||||||
|  |     'Ρ':'R', 'Σ':'S', 'Τ':'T', 'Υ':'Y', 'Φ':'F', 'Χ':'X', 'Ψ':'PS', 'Ω':'W', | ||||||
|  |     'Ά':'A', 'Έ':'E', 'Ί':'I', 'Ό':'O', 'Ύ':'Y', 'Ή':'H', 'Ώ':'W', 'Ϊ':'I', | ||||||
|  |     'Ϋ':'Y' | ||||||
|  | } | ||||||
|  | var TURKISH_MAP = { | ||||||
|  |     'ş':'s', 'Ş':'S', 'ı':'i', 'İ':'I', 'ç':'c', 'Ç':'C', 'ü':'u', 'Ü':'U', | ||||||
|  |     'ö':'o', 'Ö':'O', 'ğ':'g', 'Ğ':'G' | ||||||
|  | } | ||||||
|  | var RUSSIAN_MAP = { | ||||||
|  |     'а':'a', 'б':'b', 'в':'v', 'г':'g', 'д':'d', 'е':'e', 'ё':'yo', 'ж':'zh', | ||||||
|  |     'з':'z', 'и':'i', 'й':'j', 'к':'k', 'л':'l', 'м':'m', 'н':'n', 'о':'o', | ||||||
|  |     'п':'p', 'р':'r', 'с':'s', 'т':'t', 'у':'u', 'ф':'f', 'х':'h', 'ц':'c', | ||||||
|  |     'ч':'ch', 'ш':'sh', 'щ':'sh', 'ъ':'', 'ы':'y', 'ь':'', 'э':'e', 'ю':'yu', | ||||||
|  |     'я':'ya', | ||||||
|  |     'А':'A', 'Б':'B', 'В':'V', 'Г':'G', 'Д':'D', 'Е':'E', 'Ё':'Yo', 'Ж':'Zh', | ||||||
|  |     'З':'Z', 'И':'I', 'Й':'J', 'К':'K', 'Л':'L', 'М':'M', 'Н':'N', 'О':'O', | ||||||
|  |     'П':'P', 'Р':'R', 'С':'S', 'Т':'T', 'У':'U', 'Ф':'F', 'Х':'H', 'Ц':'C', | ||||||
|  |     'Ч':'Ch', 'Ш':'Sh', 'Щ':'Sh', 'Ъ':'', 'Ы':'Y', 'Ь':'', 'Э':'E', 'Ю':'Yu', | ||||||
|  |     'Я':'Ya' | ||||||
|  | } | ||||||
|  |  | ||||||
|  | var ALL_DOWNCODE_MAPS=new Array() | ||||||
|  | ALL_DOWNCODE_MAPS[0]=LATIN_MAP | ||||||
|  | ALL_DOWNCODE_MAPS[1]=LATIN_SYMBOLS_MAP | ||||||
|  | ALL_DOWNCODE_MAPS[2]=GREEK_MAP | ||||||
|  | ALL_DOWNCODE_MAPS[3]=TURKISH_MAP | ||||||
|  | ALL_DOWNCODE_MAPS[4]=RUSSIAN_MAP | ||||||
|  |  | ||||||
|  | var Downcoder = new Object(); | ||||||
|  | Downcoder.Initialize = function() | ||||||
|  | { | ||||||
|  |     if (Downcoder.map) // already made | ||||||
|  |         return ; | ||||||
|  |     Downcoder.map ={} | ||||||
|  |     Downcoder.chars = '' ; | ||||||
|  |     for(var i in ALL_DOWNCODE_MAPS) | ||||||
|  |     { | ||||||
|  |         var lookup = ALL_DOWNCODE_MAPS[i] | ||||||
|  |         for (var c in lookup) | ||||||
|  |         { | ||||||
|  |             Downcoder.map[c] = lookup[c] ; | ||||||
|  |             Downcoder.chars += c ; | ||||||
|  |         } | ||||||
|  |      } | ||||||
|  |     Downcoder.regex = new RegExp('[' + Downcoder.chars + ']|[^' + Downcoder.chars + ']+','g') ; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | downcode= function( slug ) | ||||||
|  | { | ||||||
|  |     Downcoder.Initialize() ; | ||||||
|  |     var downcoded ="" | ||||||
|  |     var pieces = slug.match(Downcoder.regex); | ||||||
|  |     if(pieces) | ||||||
|  |     { | ||||||
|  |         for (var i = 0 ; i < pieces.length ; i++) | ||||||
|  |         { | ||||||
|  |             if (pieces[i].length == 1) | ||||||
|  |             { | ||||||
|  |                 var mapped = Downcoder.map[pieces[i]] ; | ||||||
|  |                 if (mapped != null) | ||||||
|  |                 { | ||||||
|  |                     downcoded+=mapped; | ||||||
|  |                     continue ; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             downcoded+=pieces[i]; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     else | ||||||
|  |     { | ||||||
|  |         downcoded = slug; | ||||||
|  |     } | ||||||
|  |     return downcoded; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
| function URLify(s, num_chars) { | function URLify(s, num_chars) { | ||||||
|     // changes, e.g., "Petty theft" to "petty_theft" |     // changes, e.g., "Petty theft" to "petty_theft" | ||||||
|     // remove all these words from the string before urlifying |     // remove all these words from the string before urlifying | ||||||
|  |     s = downcode(s); | ||||||
|     removelist = ["a", "an", "as", "at", "before", "but", "by", "for", "from", |     removelist = ["a", "an", "as", "at", "before", "but", "by", "for", "from", | ||||||
|                   "is", "in", "into", "like", "of", "off", "on", "onto", "per", |                   "is", "in", "into", "like", "of", "off", "on", "onto", "per", | ||||||
|                   "since", "than", "the", "this", "that", "to", "up", "via", |                   "since", "than", "the", "this", "that", "to", "up", "via", | ||||||
|                   "with"]; |                   "with"]; | ||||||
|     r = new RegExp('\\b(' + removelist.join('|') + ')\\b', 'gi'); |     r = new RegExp('\\b(' + removelist.join('|') + ')\\b', 'gi'); | ||||||
|     s = s.replace(r, ''); |     s = s.replace(r, ''); | ||||||
|  |     // if downcode doesn't hit, the char will be stripped here | ||||||
|     s = s.replace(/[^-\w\s]/g, '');  // remove unneeded chars |     s = s.replace(/[^-\w\s]/g, '');  // remove unneeded chars | ||||||
|     s = s.replace(/^\s+|\s+$/g, ''); // trim leading/trailing spaces |     s = s.replace(/^\s+|\s+$/g, ''); // trim leading/trailing spaces | ||||||
|     s = s.replace(/[-\s]+/g, '-');   // convert spaces to hyphens |     s = s.replace(/[-\s]+/g, '-');   // convert spaces to hyphens | ||||||
|     s = s.toLowerCase();             // convert to lowercase |     s = s.toLowerCase();             // convert to lowercase | ||||||
|     return s.substring(0, num_chars);// trim to first num_chars chars |     return s.substring(0, num_chars);// trim to first num_chars chars | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,7 +1,8 @@ | |||||||
| from django.db import models | from django.db import models | ||||||
| from django.contrib.contenttypes.models import ContentType | from django.contrib.contenttypes.models import ContentType | ||||||
| from django.contrib.auth.models import User | from django.contrib.auth.models import User | ||||||
| from django.utils.translation import gettext_lazy as _ | from django.utils.translation import ugettext_lazy as _ | ||||||
|  | from django.utils.encoding import smart_unicode | ||||||
|  |  | ||||||
| ADDITION = 1 | ADDITION = 1 | ||||||
| CHANGE = 2 | CHANGE = 2 | ||||||
| @@ -9,7 +10,7 @@ DELETION = 3 | |||||||
|  |  | ||||||
| class LogEntryManager(models.Manager): | class LogEntryManager(models.Manager): | ||||||
|     def log_action(self, user_id, content_type_id, object_id, object_repr, action_flag, change_message=''): |     def log_action(self, user_id, content_type_id, object_id, object_repr, action_flag, change_message=''): | ||||||
|         e = self.model(None, None, user_id, content_type_id, str(object_id), object_repr[:200], action_flag, change_message) |         e = self.model(None, None, user_id, content_type_id, smart_unicode(object_id), object_repr[:200], action_flag, change_message) | ||||||
|         e.save() |         e.save() | ||||||
|  |  | ||||||
| class LogEntry(models.Model): | class LogEntry(models.Model): | ||||||
| @@ -28,7 +29,7 @@ class LogEntry(models.Model): | |||||||
|         ordering = ('-action_time',) |         ordering = ('-action_time',) | ||||||
|  |  | ||||||
|     def __repr__(self): |     def __repr__(self): | ||||||
|         return str(self.action_time) |         return smart_unicode(self.action_time) | ||||||
|  |  | ||||||
|     def is_addition(self): |     def is_addition(self): | ||||||
|         return self.action_flag == ADDITION |         return self.action_flag == ADDITION | ||||||
| @@ -48,4 +49,4 @@ class LogEntry(models.Model): | |||||||
|         Returns the admin URL to edit the object represented by this log entry. |         Returns the admin URL to edit the object represented by this log entry. | ||||||
|         This is relative to the Django admin index page. |         This is relative to the Django admin index page. | ||||||
|         """ |         """ | ||||||
|         return "%s/%s/%s/" % (self.content_type.app_label, self.content_type.model, self.object_id) |         return u"%s/%s/%s/" % (self.content_type.app_label, self.content_type.model, self.object_id) | ||||||
|   | |||||||
| @@ -9,6 +9,8 @@ from django.http import Http404, HttpResponse, HttpResponseRedirect | |||||||
| from django.shortcuts import get_object_or_404, render_to_response | from django.shortcuts import get_object_or_404, render_to_response | ||||||
| from django.utils.html import escape | from django.utils.html import escape | ||||||
| from django.utils.text import capfirst, get_text_list | from django.utils.text import capfirst, get_text_list | ||||||
|  | from django.utils.translation import ugettext as _ | ||||||
|  | from django.utils.encoding import force_unicode | ||||||
| import sets | import sets | ||||||
|  |  | ||||||
| class IncorrectLookupParameters(Exception): | class IncorrectLookupParameters(Exception): | ||||||
| @@ -50,7 +52,7 @@ class AdminForm(object): | |||||||
| class Fieldset(object): | class Fieldset(object): | ||||||
|     def __init__(self, name=None, fields=(), classes=(), description=None): |     def __init__(self, name=None, fields=(), classes=(), description=None): | ||||||
|         self.name, self.fields = name, fields |         self.name, self.fields = name, fields | ||||||
|         self.classes = ' '.join(classes) |         self.classes = u' '.join(classes) | ||||||
|         self.description = description |         self.description = description | ||||||
|  |  | ||||||
| class BoundFieldset(object): | class BoundFieldset(object): | ||||||
| @@ -85,15 +87,15 @@ class BoundField(object): | |||||||
|     def label_tag(self): |     def label_tag(self): | ||||||
|         classes = [] |         classes = [] | ||||||
|         if self.is_checkbox: |         if self.is_checkbox: | ||||||
|             classes.append('vCheckboxLabel') |             classes.append(u'vCheckboxLabel') | ||||||
|             contents = escape(self.field.label) |             contents = escape(self.field.label) | ||||||
|         else: |         else: | ||||||
|             contents = escape(self.field.label) + ':' |             contents = escape(self.field.label) + u':' | ||||||
|         if self.field.field.required: |         if self.field.field.required: | ||||||
|             classes.append('required') |             classes.append(u'required') | ||||||
|         if not self.is_first: |         if not self.is_first: | ||||||
|             classes.append('inline') |             classes.append(u'inline') | ||||||
|         attrs = classes and {'class': ' '.join(classes)} or {} |         attrs = classes and {'class': u' '.join(classes)} or {} | ||||||
|         return self.field.label_tag(contents=contents, attrs=attrs) |         return self.field.label_tag(contents=contents, attrs=attrs) | ||||||
|  |  | ||||||
| class BaseModelAdmin(object): | class BaseModelAdmin(object): | ||||||
| @@ -561,7 +563,7 @@ class ModelAdmin(BaseModelAdmin): | |||||||
|  |  | ||||||
|         # Populate deleted_objects, a data structure of all related objects that |         # Populate deleted_objects, a data structure of all related objects that | ||||||
|         # will also be deleted. |         # will also be deleted. | ||||||
|         deleted_objects = ['%s: <a href="../../%s/">%s</a>' % (capfirst(opts.verbose_name), object_id, escape(str(obj))), []] |         deleted_objects = [u'%s: <a href="../../%s/">%s</a>' % (force_unicode(capfirst(opts.verbose_name)), object_id, escape(str(obj))), []] | ||||||
|         perms_needed = sets.Set() |         perms_needed = sets.Set() | ||||||
|         _get_deleted_objects(deleted_objects, perms_needed, request.user, obj, opts, 1) |         _get_deleted_objects(deleted_objects, perms_needed, request.user, obj, opts, 1) | ||||||
|  |  | ||||||
| @@ -571,7 +573,7 @@ class ModelAdmin(BaseModelAdmin): | |||||||
|             obj_display = str(obj) |             obj_display = str(obj) | ||||||
|             obj.delete() |             obj.delete() | ||||||
|             LogEntry.objects.log_action(request.user.id, ContentType.objects.get_for_model(self.model).id, object_id, obj_display, DELETION) |             LogEntry.objects.log_action(request.user.id, ContentType.objects.get_for_model(self.model).id, object_id, obj_display, DELETION) | ||||||
|             request.user.message_set.create(message=_('The %(name)s "%(obj)s" was deleted successfully.') % {'name': opts.verbose_name, 'obj': obj_display}) |             request.user.message_set.create(message=_('The %(name)s "%(obj)s" was deleted successfully.') % {'name': force_unicode(opts.verbose_name), 'obj': force_unicode(obj_display)}) | ||||||
|             return HttpResponseRedirect("../../") |             return HttpResponseRedirect("../../") | ||||||
|         extra_context = { |         extra_context = { | ||||||
|             "title": _("Are you sure?"), |             "title": _("Are you sure?"), | ||||||
| @@ -596,7 +598,7 @@ class ModelAdmin(BaseModelAdmin): | |||||||
|         # If no history was found, see whether this object even exists. |         # If no history was found, see whether this object even exists. | ||||||
|         obj = get_object_or_404(model, pk=object_id) |         obj = get_object_or_404(model, pk=object_id) | ||||||
|         extra_context = { |         extra_context = { | ||||||
|             'title': _('Change history: %s') % obj, |             'title': _('Change history: %s') % force_unicode(obj), | ||||||
|             'action_list': action_list, |             'action_list': action_list, | ||||||
|             'module_name': capfirst(opts.verbose_name_plural), |             'module_name': capfirst(opts.verbose_name_plural), | ||||||
|             'object': obj, |             'object': obj, | ||||||
|   | |||||||
| @@ -4,13 +4,13 @@ from django.contrib.auth import authenticate, login | |||||||
| from django.db.models import Model | from django.db.models import Model | ||||||
| from django.shortcuts import render_to_response | from django.shortcuts import render_to_response | ||||||
| from django.utils.text import capfirst | from django.utils.text import capfirst | ||||||
| from django.utils.translation import gettext_lazy | from django.utils.translation import ugettext_lazy, ugettext as _ | ||||||
| import base64 | import base64 | ||||||
| import cPickle as pickle | import cPickle as pickle | ||||||
| import datetime | import datetime | ||||||
| import md5 | import md5 | ||||||
|  |  | ||||||
| ERROR_MESSAGE = gettext_lazy("Please enter a correct username and password. Note that both fields are case-sensitive.") | ERROR_MESSAGE = ugettext_lazy("Please enter a correct username and password. Note that both fields are case-sensitive.") | ||||||
| LOGIN_FORM_KEY = 'this_is_the_login_form' | LOGIN_FORM_KEY = 'this_is_the_login_form' | ||||||
|  |  | ||||||
| class AlreadyRegistered(Exception): | class AlreadyRegistered(Exception): | ||||||
| @@ -199,7 +199,7 @@ class AdminSite(object): | |||||||
|         user = authenticate(username=username, password=password) |         user = authenticate(username=username, password=password) | ||||||
|         if user is None: |         if user is None: | ||||||
|             message = ERROR_MESSAGE |             message = ERROR_MESSAGE | ||||||
|             if '@' in username: |             if u'@' in username: | ||||||
|                 # Mistakenly entered e-mail address instead of username? Look it up. |                 # Mistakenly entered e-mail address instead of username? Look it up. | ||||||
|                 try: |                 try: | ||||||
|                     user = User.objects.get(email=username) |                     user = User.objects.get(email=username) | ||||||
|   | |||||||
| @@ -3,6 +3,6 @@ | |||||||
| <ul> | <ul> | ||||||
| {% for choice in choices %} | {% for choice in choices %} | ||||||
|     <li{% if choice.selected %} class="selected"{% endif %}> |     <li{% if choice.selected %} class="selected"{% endif %}> | ||||||
|     <a href="{{ choice.query_string }}">{{ choice.display|escape }}</a></li> |     <a href="{{ choice.query_string|iriencode }}">{{ choice.display|escape }}</a></li> | ||||||
| {% endfor %} | {% endfor %} | ||||||
| </ul> | </ul> | ||||||
|   | |||||||
| @@ -6,7 +6,8 @@ from django.db import models | |||||||
| from django.utils import dateformat | from django.utils import dateformat | ||||||
| from django.utils.html import escape | from django.utils.html import escape | ||||||
| from django.utils.text import capfirst | from django.utils.text import capfirst | ||||||
| from django.utils.translation import get_date_formats, get_partial_date_formats | from django.utils.translation import get_date_formats, get_partial_date_formats, ugettext as _ | ||||||
|  | from django.utils.encoding import smart_unicode, smart_str, force_unicode | ||||||
| from django.template import Library | from django.template import Library | ||||||
| import datetime | import datetime | ||||||
|  |  | ||||||
| @@ -16,11 +17,11 @@ DOT = '.' | |||||||
|  |  | ||||||
| def paginator_number(cl,i): | def paginator_number(cl,i): | ||||||
|     if i == DOT: |     if i == DOT: | ||||||
|         return '... ' |         return u'... ' | ||||||
|     elif i == cl.page_num: |     elif i == cl.page_num: | ||||||
|         return '<span class="this-page">%d</span> ' % (i+1) |         return u'<span class="this-page">%d</span> ' % (i+1) | ||||||
|     else: |     else: | ||||||
|         return '<a href="%s"%s>%d</a> ' % (cl.get_query_string({PAGE_VAR: i}), (i == cl.paginator.pages-1 and ' class="end"' or ''), i+1) |         return u'<a href="%s"%s>%d</a> ' % (cl.get_query_string({PAGE_VAR: i}), (i == cl.paginator.pages-1 and ' class="end"' or ''), i+1) | ||||||
| paginator_number = register.simple_tag(paginator_number) | paginator_number = register.simple_tag(paginator_number) | ||||||
|  |  | ||||||
| def pagination(cl): | def pagination(cl): | ||||||
| @@ -75,10 +76,12 @@ def result_headers(cl): | |||||||
|             admin_order_field = None |             admin_order_field = None | ||||||
|         except models.FieldDoesNotExist: |         except models.FieldDoesNotExist: | ||||||
|             # For non-field list_display values, check for the function |             # For non-field list_display values, check for the function | ||||||
|             # attribute "short_description". If that doesn't exist, fall |             # attribute "short_description". If that doesn't exist, fall back | ||||||
|             # back to the method name. And __str__ is a special-case. |             # to the method name. And __str__ and __unicode__ are special-cases. | ||||||
|             if field_name == '__str__': |             if field_name == '__unicode__': | ||||||
|                 header = lookup_opts.verbose_name |                 header = force_unicode(lookup_opts.verbose_name) | ||||||
|  |             elif field_name == '__str__': | ||||||
|  |                 header = smart_str(lookup_opts.verbose_name) | ||||||
|             else: |             else: | ||||||
|                 attr = getattr(cl.model, field_name) # Let AttributeErrors propagate. |                 attr = getattr(cl.model, field_name) # Let AttributeErrors propagate. | ||||||
|                 try: |                 try: | ||||||
| @@ -114,7 +117,7 @@ def result_headers(cl): | |||||||
|  |  | ||||||
| def _boolean_icon(field_val): | def _boolean_icon(field_val): | ||||||
|     BOOLEAN_MAPPING = {True: 'yes', False: 'no', None: 'unknown'} |     BOOLEAN_MAPPING = {True: 'yes', False: 'no', None: 'unknown'} | ||||||
|     return '<img src="%simg/admin/icon-%s.gif" alt="%s" />' % (settings.ADMIN_MEDIA_PREFIX, BOOLEAN_MAPPING[field_val], field_val) |     return u'<img src="%simg/admin/icon-%s.gif" alt="%s" />' % (settings.ADMIN_MEDIA_PREFIX, BOOLEAN_MAPPING[field_val], field_val) | ||||||
|  |  | ||||||
| def items_for_result(cl, result): | def items_for_result(cl, result): | ||||||
|     first = True |     first = True | ||||||
| @@ -136,7 +139,7 @@ def items_for_result(cl, result): | |||||||
|                     allow_tags = True |                     allow_tags = True | ||||||
|                     result_repr = _boolean_icon(attr) |                     result_repr = _boolean_icon(attr) | ||||||
|                 else: |                 else: | ||||||
|                     result_repr = str(attr) |                     result_repr = smart_unicode(attr) | ||||||
|             except (AttributeError, ObjectDoesNotExist): |             except (AttributeError, ObjectDoesNotExist): | ||||||
|                 result_repr = EMPTY_CHANGELIST_VALUE |                 result_repr = EMPTY_CHANGELIST_VALUE | ||||||
|             else: |             else: | ||||||
| @@ -179,19 +182,19 @@ def items_for_result(cl, result): | |||||||
|             elif f.choices: |             elif f.choices: | ||||||
|                 result_repr = dict(f.choices).get(field_val, EMPTY_CHANGELIST_VALUE) |                 result_repr = dict(f.choices).get(field_val, EMPTY_CHANGELIST_VALUE) | ||||||
|             else: |             else: | ||||||
|                 result_repr = escape(str(field_val)) |                 result_repr = escape(field_val) | ||||||
|         if result_repr == '': |         if force_unicode(result_repr) == '': | ||||||
|             result_repr = ' ' |             result_repr = ' ' | ||||||
|         # If list_display_links not defined, add the link tag to the first field |         # If list_display_links not defined, add the link tag to the first field | ||||||
|         if (first and not cl.list_display_links) or field_name in cl.list_display_links: |         if (first and not cl.list_display_links) or field_name in cl.list_display_links: | ||||||
|             table_tag = {True:'th', False:'td'}[first] |             table_tag = {True:'th', False:'td'}[first] | ||||||
|             first = False |             first = False | ||||||
|             url = cl.url_for_result(result) |             url = cl.url_for_result(result) | ||||||
|             result_id = str(getattr(result, pk)) # str() is needed in case of 23L (long ints) |             result_id = smart_unicode(getattr(result, pk)) # conversion to string is needed in case of 23L (long ints) | ||||||
|             yield ('<%s%s><a href="%s"%s>%s</a></%s>' % \ |             yield (u'<%s%s><a href="%s"%s>%s</a></%s>' % \ | ||||||
|                 (table_tag, row_class, url, (cl.is_popup and ' onclick="opener.dismissRelatedLookupPopup(window, %r); return false;"' % result_id or ''), result_repr, table_tag)) |                 (table_tag, row_class, url, (cl.is_popup and ' onclick="opener.dismissRelatedLookupPopup(window, %r); return false;"' % result_id or ''), result_repr, table_tag)) | ||||||
|         else: |         else: | ||||||
|             yield ('<td%s>%s</td>' % (row_class, result_repr)) |             yield (u'<td%s>%s</td>' % (row_class, result_repr)) | ||||||
|  |  | ||||||
| def results(cl): | def results(cl): | ||||||
|     for res in cl.result_list: |     for res in cl.result_list: | ||||||
|   | |||||||
| @@ -2,6 +2,7 @@ from django import template | |||||||
| from django.contrib.admin.views.main import AdminBoundField | from django.contrib.admin.views.main import AdminBoundField | ||||||
| from django.template import loader | from django.template import loader | ||||||
| from django.utils.text import capfirst | from django.utils.text import capfirst | ||||||
|  | from django.utils.encoding import force_unicode | ||||||
| from django.db import models | from django.db import models | ||||||
| from django.db.models.fields import Field | from django.db.models.fields import Field | ||||||
| from django.db.models.related import BoundRelatedObject | from django.db.models.related import BoundRelatedObject | ||||||
| @@ -13,7 +14,7 @@ register = template.Library() | |||||||
| word_re = re.compile('[A-Z][a-z]+') | word_re = re.compile('[A-Z][a-z]+') | ||||||
|  |  | ||||||
| def class_name_to_underscored(name): | def class_name_to_underscored(name): | ||||||
|     return '_'.join([s.lower() for s in word_re.findall(name)[:-1]]) |     return u'_'.join([s.lower() for s in word_re.findall(name)[:-1]]) | ||||||
|  |  | ||||||
| def submit_row(context): | def submit_row(context): | ||||||
|     opts = context['opts'] |     opts = context['opts'] | ||||||
| @@ -45,7 +46,7 @@ class FieldWidgetNode(template.Node): | |||||||
|         if klass not in cls.nodelists: |         if klass not in cls.nodelists: | ||||||
|             try: |             try: | ||||||
|                 field_class_name = klass.__name__ |                 field_class_name = klass.__name__ | ||||||
|                 template_name = "widget/%s.html" % class_name_to_underscored(field_class_name) |                 template_name = u"widget/%s.html" % class_name_to_underscored(field_class_name) | ||||||
|                 nodelist = loader.get_template(template_name).nodelist |                 nodelist = loader.get_template(template_name).nodelist | ||||||
|             except template.TemplateDoesNotExist: |             except template.TemplateDoesNotExist: | ||||||
|                 super_klass = bool(klass.__bases__) and klass.__bases__[0] or None |                 super_klass = bool(klass.__bases__) and klass.__bases__[0] or None | ||||||
| @@ -95,7 +96,7 @@ class FormFieldCollectionWrapper(object): | |||||||
|         self.index = index |         self.index = index | ||||||
|  |  | ||||||
| def output_all(form_fields): | def output_all(form_fields): | ||||||
|     return ''.join([str(f) for f in form_fields]) |     return u''.join([force_unicode(f) for f in form_fields]) | ||||||
| output_all = register.simple_tag(output_all) | output_all = register.simple_tag(output_all) | ||||||
|  |  | ||||||
| def field_widget(parser, token): | def field_widget(parser, token): | ||||||
|   | |||||||
| @@ -6,6 +6,7 @@ from django import oldforms, template | |||||||
| from django.shortcuts import render_to_response, get_object_or_404 | from django.shortcuts import render_to_response, get_object_or_404 | ||||||
| from django.http import HttpResponseRedirect | from django.http import HttpResponseRedirect | ||||||
| from django.utils.html import escape | from django.utils.html import escape | ||||||
|  | from django.utils.translation import ugettext as _ | ||||||
|  |  | ||||||
| def user_add_stage(request): | def user_add_stage(request): | ||||||
|     if not request.user.has_perm('auth.change_user'): |     if not request.user.has_perm('auth.change_user'): | ||||||
|   | |||||||
| @@ -3,11 +3,11 @@ from django.conf import settings | |||||||
| from django.contrib.auth.models import User | from django.contrib.auth.models import User | ||||||
| from django.contrib.auth import authenticate, login | from django.contrib.auth import authenticate, login | ||||||
| from django.shortcuts import render_to_response | from django.shortcuts import render_to_response | ||||||
| from django.utils.translation import gettext_lazy | from django.utils.translation import ugettext_lazy, ugettext as _ | ||||||
| import base64, datetime, md5 | import base64, datetime, md5 | ||||||
| import cPickle as pickle | import cPickle as pickle | ||||||
|  |  | ||||||
| ERROR_MESSAGE = gettext_lazy("Please enter a correct username and password. Note that both fields are case-sensitive.") | ERROR_MESSAGE = ugettext_lazy("Please enter a correct username and password. Note that both fields are case-sensitive.") | ||||||
| LOGIN_FORM_KEY = 'this_is_the_login_form' | LOGIN_FORM_KEY = 'this_is_the_login_form' | ||||||
|  |  | ||||||
| def _display_login_form(request, error_message=''): | def _display_login_form(request, error_message=''): | ||||||
|   | |||||||
| @@ -12,6 +12,8 @@ from django.db.models.query import handle_legacy_orderlist, QuerySet | |||||||
| from django.http import Http404 | from django.http import Http404 | ||||||
| from django.utils.html import escape | from django.utils.html import escape | ||||||
| from django.utils.text import capfirst | from django.utils.text import capfirst | ||||||
|  | from django.utils.encoding import force_unicode, smart_str | ||||||
|  | from django.utils.translation import ugettext as _ | ||||||
| import operator | import operator | ||||||
|  |  | ||||||
| try: | try: | ||||||
| @@ -77,11 +79,11 @@ class AdminBoundField(object): | |||||||
|  |  | ||||||
|         classes = [] |         classes = [] | ||||||
|         if self.raw_id_admin: |         if self.raw_id_admin: | ||||||
|             classes.append('nowrap') |             classes.append(u'nowrap') | ||||||
|         if max([bool(f.errors()) for f in self.form_fields]): |         if max([bool(f.errors()) for f in self.form_fields]): | ||||||
|             classes.append('error') |             classes.append(u'error') | ||||||
|         if classes: |         if classes: | ||||||
|             self.cell_class_attribute = ' class="%s" ' % ' '.join(classes) |             self.cell_class_attribute = u' class="%s" ' % ' '.join(classes) | ||||||
|         self._repr_filled = False |         self._repr_filled = False | ||||||
|  |  | ||||||
|     def original_value(self): |     def original_value(self): | ||||||
| @@ -93,9 +95,9 @@ class AdminBoundField(object): | |||||||
|             return self._display |             return self._display | ||||||
|         except AttributeError: |         except AttributeError: | ||||||
|             if isinstance(self.field.rel, models.ManyToOneRel): |             if isinstance(self.field.rel, models.ManyToOneRel): | ||||||
|                 self._display = getattr(self.original, self.field.name) |                 self._display = force_unicode(getattr(self.original, self.field.name), strings_only=True) | ||||||
|             elif isinstance(self.field.rel, models.ManyToManyRel): |             elif isinstance(self.field.rel, models.ManyToManyRel): | ||||||
|                 self._display = ", ".join([str(obj) for obj in getattr(self.original, self.field.name).all()]) |                 self._display = u", ".join([force_unicode(obj) for obj in getattr(self.original, self.field.name).all()]) | ||||||
|             return self._display |             return self._display | ||||||
|  |  | ||||||
|     def __repr__(self): |     def __repr__(self): | ||||||
| @@ -173,11 +175,11 @@ def _get_deleted_objects(deleted_objects, perms_needed, user, obj, opts, current | |||||||
|                 if related.field.rel.edit_inline or not related.opts.admin: |                 if related.field.rel.edit_inline or not related.opts.admin: | ||||||
|                     # Don't display link to edit, because it either has no |                     # Don't display link to edit, because it either has no | ||||||
|                     # admin or is edited inline. |                     # admin or is edited inline. | ||||||
|                     nh(deleted_objects, current_depth, ['%s: %s' % (capfirst(related.opts.verbose_name), sub_obj), []]) |                     nh(deleted_objects, current_depth, [u'%s: %s' % (force_unicode(capfirst(related.opts.verbose_name)), sub_obj), []]) | ||||||
|                 else: |                 else: | ||||||
|                     # Display a link to the admin page. |                     # Display a link to the admin page. | ||||||
|                     nh(deleted_objects, current_depth, ['%s: <a href="../../../../%s/%s/%s/">%s</a>' % \ |                     nh(deleted_objects, current_depth, [u'%s: <a href="../../../../%s/%s/%s/">%s</a>' % \ | ||||||
|                         (capfirst(related.opts.verbose_name), related.opts.app_label, related.opts.object_name.lower(), |                         (force_unicode(capfirst(related.opts.verbose_name)), related.opts.app_label, related.opts.object_name.lower(), | ||||||
|                         sub_obj._get_pk_val(), sub_obj), []]) |                         sub_obj._get_pk_val(), sub_obj), []]) | ||||||
|                 _get_deleted_objects(deleted_objects, perms_needed, user, sub_obj, related.opts, current_depth+2) |                 _get_deleted_objects(deleted_objects, perms_needed, user, sub_obj, related.opts, current_depth+2) | ||||||
|         else: |         else: | ||||||
| @@ -187,11 +189,11 @@ def _get_deleted_objects(deleted_objects, perms_needed, user, obj, opts, current | |||||||
|                 if related.field.rel.edit_inline or not related.opts.admin: |                 if related.field.rel.edit_inline or not related.opts.admin: | ||||||
|                     # Don't display link to edit, because it either has no |                     # Don't display link to edit, because it either has no | ||||||
|                     # admin or is edited inline. |                     # admin or is edited inline. | ||||||
|                     nh(deleted_objects, current_depth, ['%s: %s' % (capfirst(related.opts.verbose_name), escape(str(sub_obj))), []]) |                     nh(deleted_objects, current_depth, [u'%s: %s' % (force_unicode(capfirst(related.opts.verbose_name)), escape(sub_obj)), []]) | ||||||
|                 else: |                 else: | ||||||
|                     # Display a link to the admin page. |                     # Display a link to the admin page. | ||||||
|                     nh(deleted_objects, current_depth, ['%s: <a href="../../../../%s/%s/%s/">%s</a>' % \ |                     nh(deleted_objects, current_depth, [u'%s: <a href="../../../../%s/%s/%s/">%s</a>' % \ | ||||||
|                         (capfirst(related.opts.verbose_name), related.opts.app_label, related.opts.object_name.lower(), sub_obj._get_pk_val(), escape(str(sub_obj))), []]) |                         (force_unicode(capfirst(related.opts.verbose_name)), related.opts.app_label, related.opts.object_name.lower(), sub_obj._get_pk_val(), escape(sub_obj)), []]) | ||||||
|                 _get_deleted_objects(deleted_objects, perms_needed, user, sub_obj, related.opts, current_depth+2) |                 _get_deleted_objects(deleted_objects, perms_needed, user, sub_obj, related.opts, current_depth+2) | ||||||
|             # If there were related objects, and the user doesn't have |             # If there were related objects, and the user doesn't have | ||||||
|             # permission to delete them, add the missing perm to perms_needed. |             # permission to delete them, add the missing perm to perms_needed. | ||||||
| @@ -218,17 +220,17 @@ def _get_deleted_objects(deleted_objects, perms_needed, user, obj, opts, current | |||||||
|                     # Don't display link to edit, because it either has no |                     # Don't display link to edit, because it either has no | ||||||
|                     # admin or is edited inline. |                     # admin or is edited inline. | ||||||
|                     nh(deleted_objects, current_depth, [_('One or more %(fieldname)s in %(name)s: %(obj)s') % \ |                     nh(deleted_objects, current_depth, [_('One or more %(fieldname)s in %(name)s: %(obj)s') % \ | ||||||
|                         {'fieldname': related.field.verbose_name, 'name': related.opts.verbose_name, 'obj': escape(str(sub_obj))}, []]) |                         {'fieldname': force_unicode(related.field.verbose_name), 'name': force_unicode(related.opts.verbose_name), 'obj': escape(sub_obj)}, []]) | ||||||
|                 else: |                 else: | ||||||
|                     # Display a link to the admin page. |                     # Display a link to the admin page. | ||||||
|                     nh(deleted_objects, current_depth, [ |                     nh(deleted_objects, current_depth, [ | ||||||
|                         (_('One or more %(fieldname)s in %(name)s:') % {'fieldname': related.field.verbose_name, 'name':related.opts.verbose_name}) + \ |                         (_('One or more %(fieldname)s in %(name)s:') % {'fieldname': force_unicode(related.field.verbose_name), 'name': force_unicode(related.opts.verbose_name)}) + \ | ||||||
|                         (' <a href="../../../../%s/%s/%s/">%s</a>' % \ |                         (u' <a href="../../../../%s/%s/%s/">%s</a>' % \ | ||||||
|                             (related.opts.app_label, related.opts.module_name, sub_obj._get_pk_val(), escape(str(sub_obj)))), []]) |                             (related.opts.app_label, related.opts.module_name, sub_obj._get_pk_val(), escape(sub_obj))), []]) | ||||||
|         # If there were related objects, and the user doesn't have |         # If there were related objects, and the user doesn't have | ||||||
|         # permission to change them, add the missing perm to perms_needed. |         # permission to change them, add the missing perm to perms_needed. | ||||||
|         if related.opts.admin and has_related_objs: |         if related.opts.admin and has_related_objs: | ||||||
|             p = '%s.%s' % (related.opts.app_label, related.opts.get_change_permission()) |             p = u'%s.%s' % (related.opts.app_label, related.opts.get_change_permission()) | ||||||
|             if not user.has_perm(p): |             if not user.has_perm(p): | ||||||
|                 perms_needed.add(related.opts.verbose_name) |                 perms_needed.add(related.opts.verbose_name) | ||||||
|  |  | ||||||
| @@ -264,7 +266,7 @@ class ChangeList(object): | |||||||
|         self.query = request.GET.get(SEARCH_VAR, '') |         self.query = request.GET.get(SEARCH_VAR, '') | ||||||
|         self.query_set = self.get_query_set() |         self.query_set = self.get_query_set() | ||||||
|         self.get_results(request) |         self.get_results(request) | ||||||
|         self.title = (self.is_popup and _('Select %s') % self.opts.verbose_name or _('Select %s to change') % self.opts.verbose_name) |         self.title = (self.is_popup and _('Select %s') % force_unicode(self.opts.verbose_name) or _('Select %s to change') % force_unicode(self.opts.verbose_name)) | ||||||
|         self.filter_specs, self.has_filters = self.get_filters(request) |         self.filter_specs, self.has_filters = self.get_filters(request) | ||||||
|         self.pk_attname = self.lookup_opts.pk.attname |         self.pk_attname = self.lookup_opts.pk.attname | ||||||
|  |  | ||||||
| @@ -291,7 +293,7 @@ class ChangeList(object): | |||||||
|                 del p[k] |                 del p[k] | ||||||
|             elif v is not None: |             elif v is not None: | ||||||
|                 p[k] = v |                 p[k] = v | ||||||
|         return '?' + '&'.join(['%s=%s' % (k, v) for k, v in p.items()]).replace(' ', '%20') |         return '?' + '&'.join([u'%s=%s' % (k, v) for k, v in p.items()]).replace(' ', '%20') | ||||||
|  |  | ||||||
|     def get_results(self, request): |     def get_results(self, request): | ||||||
|         paginator = ObjectPaginator(self.query_set, self.list_per_page) |         paginator = ObjectPaginator(self.query_set, self.list_per_page) | ||||||
| @@ -377,6 +379,12 @@ class ChangeList(object): | |||||||
|         for i in (ALL_VAR, ORDER_VAR, ORDER_TYPE_VAR, SEARCH_VAR, IS_POPUP_VAR): |         for i in (ALL_VAR, ORDER_VAR, ORDER_TYPE_VAR, SEARCH_VAR, IS_POPUP_VAR): | ||||||
|             if i in lookup_params: |             if i in lookup_params: | ||||||
|                 del lookup_params[i] |                 del lookup_params[i] | ||||||
|  |         for key, value in lookup_params.items(): | ||||||
|  |             if not isinstance(key, str): | ||||||
|  |                 # 'key' will be used as a keyword argument later, so Python | ||||||
|  |                 # requires it to be a string. | ||||||
|  |                 del lookup_params[key] | ||||||
|  |                 lookup_params[smart_str(key)] = value | ||||||
|  |  | ||||||
|         # Apply lookup parameters from the query string. |         # Apply lookup parameters from the query string. | ||||||
|         qs = qs.filter(**lookup_params) |         qs = qs.filter(**lookup_params) | ||||||
|   | |||||||
| @@ -4,6 +4,7 @@ Form Widget classes specific to the Django admin site. | |||||||
|  |  | ||||||
| from django import newforms as forms | from django import newforms as forms | ||||||
| from django.utils.text import capfirst | from django.utils.text import capfirst | ||||||
|  | from django.utils.translation import ugettext as _ | ||||||
|  |  | ||||||
| class FilteredSelectMultiple(forms.SelectMultiple): | class FilteredSelectMultiple(forms.SelectMultiple): | ||||||
|     """ |     """ | ||||||
|   | |||||||
| @@ -4,7 +4,7 @@ from django.contrib.sites.models import Site | |||||||
| from django.template import Context, loader | from django.template import Context, loader | ||||||
| from django.core import validators | from django.core import validators | ||||||
| from django import oldforms | from django import oldforms | ||||||
| from django.utils.translation import gettext as _ | from django.utils.translation import ugettext as _ | ||||||
|  |  | ||||||
| class UserCreationForm(oldforms.Manipulator): | class UserCreationForm(oldforms.Manipulator): | ||||||
|     "A form that creates a user, with no privileges, from the given username and password." |     "A form that creates a user, with no privileges, from the given username and password." | ||||||
|   | |||||||
| @@ -7,13 +7,13 @@ from django.db.models import get_models, signals | |||||||
| from django.contrib.auth import models as auth_app | from django.contrib.auth import models as auth_app | ||||||
|  |  | ||||||
| def _get_permission_codename(action, opts): | def _get_permission_codename(action, opts): | ||||||
|     return '%s_%s' % (action, opts.object_name.lower()) |     return u'%s_%s' % (action, opts.object_name.lower()) | ||||||
|  |  | ||||||
| def _get_all_permissions(opts): | def _get_all_permissions(opts): | ||||||
|     "Returns (codename, name) for all permissions in the given opts." |     "Returns (codename, name) for all permissions in the given opts." | ||||||
|     perms = [] |     perms = [] | ||||||
|     for action in ('add', 'change', 'delete'): |     for action in ('add', 'change', 'delete'): | ||||||
|         perms.append((_get_permission_codename(action, opts), 'Can %s %s' % (action, opts.verbose_name))) |         perms.append((_get_permission_codename(action, opts), u'Can %s %s' % (action, opts.verbose_name_raw))) | ||||||
|     return perms + list(opts.permissions) |     return perms + list(opts.permissions) | ||||||
|  |  | ||||||
| def create_permissions(app, created_models, verbosity): | def create_permissions(app, created_models, verbosity): | ||||||
|   | |||||||
| @@ -2,8 +2,10 @@ from django.core import validators | |||||||
| from django.core.exceptions import ImproperlyConfigured | from django.core.exceptions import ImproperlyConfigured | ||||||
| from django.db import backend, connection, models | from django.db import backend, connection, models | ||||||
| from django.contrib.contenttypes.models import ContentType | from django.contrib.contenttypes.models import ContentType | ||||||
| from django.utils.translation import gettext_lazy as _ | from django.utils.encoding import smart_str | ||||||
|  | from django.utils.translation import ugettext_lazy as _ | ||||||
| import datetime | import datetime | ||||||
|  | import urllib | ||||||
|  |  | ||||||
| try: | try: | ||||||
|     set |     set | ||||||
| @@ -18,16 +20,16 @@ def check_password(raw_password, enc_password): | |||||||
|     algo, salt, hsh = enc_password.split('$') |     algo, salt, hsh = enc_password.split('$') | ||||||
|     if algo == 'md5': |     if algo == 'md5': | ||||||
|         import md5 |         import md5 | ||||||
|         return hsh == md5.new(salt+raw_password).hexdigest() |         return hsh == md5.new(smart_str(salt + raw_password)).hexdigest() | ||||||
|     elif algo == 'sha1': |     elif algo == 'sha1': | ||||||
|         import sha |         import sha | ||||||
|         return hsh == sha.new(salt+raw_password).hexdigest() |         return hsh == sha.new(smart_str(salt + raw_password)).hexdigest() | ||||||
|     elif algo == 'crypt': |     elif algo == 'crypt': | ||||||
|         try: |         try: | ||||||
|             import crypt |             import crypt | ||||||
|         except ImportError: |         except ImportError: | ||||||
|             raise ValueError, "Crypt password algorithm not supported in this environment." |             raise ValueError, "Crypt password algorithm not supported in this environment." | ||||||
|         return hsh == crypt.crypt(raw_password, salt) |         return hsh == crypt.crypt(smart_str(raw_password), smart_str(salt)) | ||||||
|     raise ValueError, "Got unknown password algorithm type in password." |     raise ValueError, "Got unknown password algorithm type in password." | ||||||
|  |  | ||||||
| class SiteProfileNotAvailable(Exception): | class SiteProfileNotAvailable(Exception): | ||||||
| @@ -56,8 +58,8 @@ class Permission(models.Model): | |||||||
|         unique_together = (('content_type', 'codename'),) |         unique_together = (('content_type', 'codename'),) | ||||||
|         ordering = ('content_type', 'codename') |         ordering = ('content_type', 'codename') | ||||||
|  |  | ||||||
|     def __str__(self): |     def __unicode__(self): | ||||||
|         return "%s | %s | %s" % (self.content_type.app_label, self.content_type, self.name) |         return u"%s | %s | %s" % (self.content_type.app_label, self.content_type, self.name) | ||||||
|  |  | ||||||
| class Group(models.Model): | class Group(models.Model): | ||||||
|     """Groups are a generic way of categorizing users to apply permissions, or some other label, to those users. A user can belong to any number of groups. |     """Groups are a generic way of categorizing users to apply permissions, or some other label, to those users. A user can belong to any number of groups. | ||||||
| @@ -74,7 +76,7 @@ class Group(models.Model): | |||||||
|         verbose_name_plural = _('groups') |         verbose_name_plural = _('groups') | ||||||
|         ordering = ('name',) |         ordering = ('name',) | ||||||
|  |  | ||||||
|     def __str__(self): |     def __unicode__(self): | ||||||
|         return self.name |         return self.name | ||||||
|  |  | ||||||
| class UserManager(models.Manager): | class UserManager(models.Manager): | ||||||
| @@ -118,11 +120,11 @@ class User(models.Model): | |||||||
|         verbose_name_plural = _('users') |         verbose_name_plural = _('users') | ||||||
|         ordering = ('username',) |         ordering = ('username',) | ||||||
|  |  | ||||||
|     def __str__(self): |     def __unicode__(self): | ||||||
|         return self.username |         return self.username | ||||||
|  |  | ||||||
|     def get_absolute_url(self): |     def get_absolute_url(self): | ||||||
|         return "/users/%s/" % self.username |         return "/users/%s/" % urllib.quote(smart_str(self.username)) | ||||||
|  |  | ||||||
|     def is_anonymous(self): |     def is_anonymous(self): | ||||||
|         "Always returns False. This is a way of comparing User objects to anonymous users." |         "Always returns False. This is a way of comparing User objects to anonymous users." | ||||||
| @@ -135,14 +137,14 @@ class User(models.Model): | |||||||
|  |  | ||||||
|     def get_full_name(self): |     def get_full_name(self): | ||||||
|         "Returns the first_name plus the last_name, with a space in between." |         "Returns the first_name plus the last_name, with a space in between." | ||||||
|         full_name = '%s %s' % (self.first_name, self.last_name) |         full_name = u'%s %s' % (self.first_name, self.last_name) | ||||||
|         return full_name.strip() |         return full_name.strip() | ||||||
|  |  | ||||||
|     def set_password(self, raw_password): |     def set_password(self, raw_password): | ||||||
|         import sha, random |         import sha, random | ||||||
|         algo = 'sha1' |         algo = 'sha1' | ||||||
|         salt = sha.new(str(random.random())).hexdigest()[:5] |         salt = sha.new(str(random.random())).hexdigest()[:5] | ||||||
|         hsh = sha.new(salt+raw_password).hexdigest() |         hsh = sha.new(salt + smart_str(raw_password)).hexdigest() | ||||||
|         self.password = '%s$%s$%s' % (algo, salt, hsh) |         self.password = '%s$%s$%s' % (algo, salt, hsh) | ||||||
|  |  | ||||||
|     def check_password(self, raw_password): |     def check_password(self, raw_password): | ||||||
| @@ -154,7 +156,7 @@ class User(models.Model): | |||||||
|         # algorithm or salt. |         # algorithm or salt. | ||||||
|         if '$' not in self.password: |         if '$' not in self.password: | ||||||
|             import md5 |             import md5 | ||||||
|             is_correct = (self.password == md5.new(raw_password).hexdigest()) |             is_correct = (self.password == md5.new(smart_str(raw_password)).hexdigest()) | ||||||
|             if is_correct: |             if is_correct: | ||||||
|                 # Convert the password to the new, more secure format. |                 # Convert the password to the new, more secure format. | ||||||
|                 self.set_password(raw_password) |                 self.set_password(raw_password) | ||||||
| @@ -194,7 +196,7 @@ class User(models.Model): | |||||||
|  |  | ||||||
|     def get_all_permissions(self): |     def get_all_permissions(self): | ||||||
|         if not hasattr(self, '_perm_cache'): |         if not hasattr(self, '_perm_cache'): | ||||||
|             self._perm_cache = set(["%s.%s" % (p.content_type.app_label, p.codename) for p in self.user_permissions.select_related()]) |             self._perm_cache = set([u"%s.%s" % (p.content_type.app_label, p.codename) for p in self.user_permissions.select_related()]) | ||||||
|             self._perm_cache.update(self.get_group_permissions()) |             self._perm_cache.update(self.get_group_permissions()) | ||||||
|         return self._perm_cache |         return self._perm_cache | ||||||
|  |  | ||||||
| @@ -256,7 +258,7 @@ class Message(models.Model): | |||||||
|     user = models.ForeignKey(User) |     user = models.ForeignKey(User) | ||||||
|     message = models.TextField(_('message')) |     message = models.TextField(_('message')) | ||||||
|  |  | ||||||
|     def __str__(self): |     def __unicode__(self): | ||||||
|         return self.message |         return self.message | ||||||
|  |  | ||||||
| class AnonymousUser(object): | class AnonymousUser(object): | ||||||
| @@ -266,9 +268,12 @@ class AnonymousUser(object): | |||||||
|     def __init__(self): |     def __init__(self): | ||||||
|         pass |         pass | ||||||
|  |  | ||||||
|     def __str__(self): |     def __unicode__(self): | ||||||
|         return 'AnonymousUser' |         return 'AnonymousUser' | ||||||
|  |  | ||||||
|  |     def __str__(self): | ||||||
|  |         return unicode(self).encode('utf-8') | ||||||
|  |  | ||||||
|     def __eq__(self, other): |     def __eq__(self, other): | ||||||
|         return isinstance(other, self.__class__) |         return isinstance(other, self.__class__) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -7,6 +7,7 @@ from django.contrib.sites.models import Site | |||||||
| from django.http import HttpResponseRedirect | from django.http import HttpResponseRedirect | ||||||
| from django.contrib.auth.decorators import login_required | from django.contrib.auth.decorators import login_required | ||||||
| from django.contrib.auth import REDIRECT_FIELD_NAME | from django.contrib.auth import REDIRECT_FIELD_NAME | ||||||
|  | from django.utils.translation import ugettext as _ | ||||||
|  |  | ||||||
| def login(request, template_name='registration/login.html'): | def login(request, template_name='registration/login.html'): | ||||||
|     "Displays the login form and handles the login action." |     "Displays the login form and handles the login action." | ||||||
|   | |||||||
| @@ -11,7 +11,7 @@ class LatestFreeCommentsFeed(Feed): | |||||||
|     def title(self): |     def title(self): | ||||||
|         if not hasattr(self, '_site'): |         if not hasattr(self, '_site'): | ||||||
|             self._site = Site.objects.get_current() |             self._site = Site.objects.get_current() | ||||||
|         return "%s comments" % self._site.name |         return u"%s comments" % self._site.name | ||||||
|  |  | ||||||
|     def link(self): |     def link(self): | ||||||
|         if not hasattr(self, '_site'): |         if not hasattr(self, '_site'): | ||||||
| @@ -21,7 +21,7 @@ class LatestFreeCommentsFeed(Feed): | |||||||
|     def description(self): |     def description(self): | ||||||
|         if not hasattr(self, '_site'): |         if not hasattr(self, '_site'): | ||||||
|             self._site = Site.objects.get_current() |             self._site = Site.objects.get_current() | ||||||
|         return "Latest comments on %s" % self._site.name |         return u"Latest comments on %s" % self._site.name | ||||||
|  |  | ||||||
|     def get_query_set(self): |     def get_query_set(self): | ||||||
|         return self.comments_class.objects.filter(site__pk=settings.SITE_ID, is_public=True) |         return self.comments_class.objects.filter(site__pk=settings.SITE_ID, is_public=True) | ||||||
|   | |||||||
| @@ -2,7 +2,7 @@ from django.db import models | |||||||
| from django.contrib.contenttypes.models import ContentType | from django.contrib.contenttypes.models import ContentType | ||||||
| from django.contrib.sites.models import Site | from django.contrib.sites.models import Site | ||||||
| from django.contrib.auth.models import User | from django.contrib.auth.models import User | ||||||
| from django.utils.translation import gettext_lazy as _ | from django.utils.translation import ugettext_lazy as _ | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| import datetime | import datetime | ||||||
|  |  | ||||||
|   | |||||||
| @@ -11,7 +11,8 @@ from django.contrib.auth.forms import AuthenticationForm | |||||||
| from django.http import HttpResponseRedirect | from django.http import HttpResponseRedirect | ||||||
| from django.utils.text import normalize_newlines | from django.utils.text import normalize_newlines | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| from django.utils.translation import ngettext | from django.utils.translation import ungettext, ugettext as _ | ||||||
|  | from django.utils.encoding import smart_unicode | ||||||
| import base64, datetime | import base64, datetime | ||||||
|  |  | ||||||
| COMMENTS_PER_PAGE = 20 | COMMENTS_PER_PAGE = 20 | ||||||
| @@ -108,7 +109,7 @@ class PublicCommentManipulator(AuthenticationForm): | |||||||
|         # If the commentor has posted fewer than COMMENTS_FIRST_FEW comments, |         # If the commentor has posted fewer than COMMENTS_FIRST_FEW comments, | ||||||
|         # send the comment to the managers. |         # send the comment to the managers. | ||||||
|         if self.user_cache.comment_set.count() <= settings.COMMENTS_FIRST_FEW: |         if self.user_cache.comment_set.count() <= settings.COMMENTS_FIRST_FEW: | ||||||
|             message = ngettext('This comment was posted by a user who has posted fewer than %(count)s comment:\n\n%(text)s', |             message = ungettext('This comment was posted by a user who has posted fewer than %(count)s comment:\n\n%(text)s', | ||||||
|                 'This comment was posted by a user who has posted fewer than %(count)s comments:\n\n%(text)s', settings.COMMENTS_FIRST_FEW) % \ |                 'This comment was posted by a user who has posted fewer than %(count)s comments:\n\n%(text)s', settings.COMMENTS_FIRST_FEW) % \ | ||||||
|                 {'count': settings.COMMENTS_FIRST_FEW, 'text': c.get_as_text()} |                 {'count': settings.COMMENTS_FIRST_FEW, 'text': c.get_as_text()} | ||||||
|             mail_managers("Comment posted by rookie user", message) |             mail_managers("Comment posted by rookie user", message) | ||||||
| @@ -248,7 +249,7 @@ def post_comment(request): | |||||||
|         # If the IP is banned, mail the admins, do NOT save the comment, and |         # If the IP is banned, mail the admins, do NOT save the comment, and | ||||||
|         # serve up the "Thanks for posting" page as if the comment WAS posted. |         # serve up the "Thanks for posting" page as if the comment WAS posted. | ||||||
|         if request.META['REMOTE_ADDR'] in settings.BANNED_IPS: |         if request.META['REMOTE_ADDR'] in settings.BANNED_IPS: | ||||||
|             mail_admins("Banned IP attempted to post comment", str(request.POST) + "\n\n" + str(request.META)) |             mail_admins("Banned IP attempted to post comment", smart_unicode(request.POST) + "\n\n" + str(request.META)) | ||||||
|         else: |         else: | ||||||
|             manipulator.do_html2python(new_data) |             manipulator.do_html2python(new_data) | ||||||
|             comment = manipulator.save(new_data) |             comment = manipulator.save(new_data) | ||||||
| @@ -312,7 +313,7 @@ def post_free_comment(request): | |||||||
|         # serve up the "Thanks for posting" page as if the comment WAS posted. |         # serve up the "Thanks for posting" page as if the comment WAS posted. | ||||||
|         if request.META['REMOTE_ADDR'] in settings.BANNED_IPS: |         if request.META['REMOTE_ADDR'] in settings.BANNED_IPS: | ||||||
|             from django.core.mail import mail_admins |             from django.core.mail import mail_admins | ||||||
|             mail_admins("Practical joker", str(request.POST) + "\n\n" + str(request.META)) |             mail_admins("Practical joker", smart_unicode(request.POST) + "\n\n" + str(request.META)) | ||||||
|         else: |         else: | ||||||
|             manipulator.do_html2python(new_data) |             manipulator.do_html2python(new_data) | ||||||
|             comment = manipulator.save(new_data) |             comment = manipulator.save(new_data) | ||||||
|   | |||||||
| @@ -2,6 +2,7 @@ from django.http import Http404 | |||||||
| from django.shortcuts import render_to_response | from django.shortcuts import render_to_response | ||||||
| from django.template import RequestContext | from django.template import RequestContext | ||||||
| from django.contrib.comments.models import Comment, KarmaScore | from django.contrib.comments.models import Comment, KarmaScore | ||||||
|  | from django.utils.translation import ugettext as _ | ||||||
|  |  | ||||||
| def vote(request, comment_id, vote): | def vote(request, comment_id, vote): | ||||||
|     """ |     """ | ||||||
|   | |||||||
| @@ -49,7 +49,7 @@ class GenericForeignKey(object): | |||||||
|          |          | ||||||
|     def __get__(self, instance, instance_type=None): |     def __get__(self, instance, instance_type=None): | ||||||
|         if instance is None: |         if instance is None: | ||||||
|             raise AttributeError, "%s must be accessed via instance" % self.name |             raise AttributeError, u"%s must be accessed via instance" % self.name | ||||||
|  |  | ||||||
|         try: |         try: | ||||||
|             return getattr(instance, self.cache_attr) |             return getattr(instance, self.cache_attr) | ||||||
| @@ -66,7 +66,7 @@ class GenericForeignKey(object): | |||||||
|  |  | ||||||
|     def __set__(self, instance, value): |     def __set__(self, instance, value): | ||||||
|         if instance is None: |         if instance is None: | ||||||
|             raise AttributeError, "%s must be accessed via instance" % self.related.opts.object_name |             raise AttributeError, u"%s must be accessed via instance" % self.related.opts.object_name | ||||||
|  |  | ||||||
|         ct = None |         ct = None | ||||||
|         fk = None |         fk = None | ||||||
|   | |||||||
| @@ -4,6 +4,7 @@ Creates content types for all installed models. | |||||||
|  |  | ||||||
| from django.dispatch import dispatcher | from django.dispatch import dispatcher | ||||||
| from django.db.models import get_apps, get_models, signals | from django.db.models import get_apps, get_models, signals | ||||||
|  | from django.utils.encoding import smart_unicode | ||||||
|  |  | ||||||
| def create_contenttypes(app, created_models, verbosity=2): | def create_contenttypes(app, created_models, verbosity=2): | ||||||
|     from django.contrib.contenttypes.models import ContentType |     from django.contrib.contenttypes.models import ContentType | ||||||
| @@ -17,7 +18,7 @@ def create_contenttypes(app, created_models, verbosity=2): | |||||||
|             ContentType.objects.get(app_label=opts.app_label, |             ContentType.objects.get(app_label=opts.app_label, | ||||||
|                 model=opts.object_name.lower()) |                 model=opts.object_name.lower()) | ||||||
|         except ContentType.DoesNotExist: |         except ContentType.DoesNotExist: | ||||||
|             ct = ContentType(name=str(opts.verbose_name), |             ct = ContentType(name=smart_unicode(opts.verbose_name_raw), | ||||||
|                 app_label=opts.app_label, model=opts.object_name.lower()) |                 app_label=opts.app_label, model=opts.object_name.lower()) | ||||||
|             ct.save() |             ct.save() | ||||||
|             if verbosity >= 2: |             if verbosity >= 2: | ||||||
|   | |||||||
| @@ -1,5 +1,6 @@ | |||||||
| from django.db import models | from django.db import models | ||||||
| from django.utils.translation import gettext_lazy as _ | from django.utils.translation import ugettext_lazy as _ | ||||||
|  | from django.utils.encoding import smart_unicode | ||||||
|  |  | ||||||
| CONTENT_TYPE_CACHE = {} | CONTENT_TYPE_CACHE = {} | ||||||
| class ContentTypeManager(models.Manager): | class ContentTypeManager(models.Manager): | ||||||
| @@ -13,10 +14,10 @@ class ContentTypeManager(models.Manager): | |||||||
|         try: |         try: | ||||||
|             ct = CONTENT_TYPE_CACHE[key] |             ct = CONTENT_TYPE_CACHE[key] | ||||||
|         except KeyError: |         except KeyError: | ||||||
|             # The str() is needed around opts.verbose_name because it's a |             # The smart_unicode() is needed around opts.verbose_name_raw because it might | ||||||
|             # django.utils.functional.__proxy__ object. |             # be a django.utils.functional.__proxy__ object. | ||||||
|             ct, created = self.model._default_manager.get_or_create(app_label=key[0], |             ct, created = self.model._default_manager.get_or_create(app_label=key[0], | ||||||
|                 model=key[1], defaults={'name': str(opts.verbose_name)}) |                 model=key[1], defaults={'name': smart_unicode(opts.verbose_name_raw)}) | ||||||
|             CONTENT_TYPE_CACHE[key] = ct |             CONTENT_TYPE_CACHE[key] = ct | ||||||
|         return ct |         return ct | ||||||
|  |  | ||||||
| @@ -42,7 +43,7 @@ class ContentType(models.Model): | |||||||
|         ordering = ('name',) |         ordering = ('name',) | ||||||
|         unique_together = (('app_label', 'model'),) |         unique_together = (('app_label', 'model'),) | ||||||
|  |  | ||||||
|     def __str__(self): |     def __unicode__(self): | ||||||
|         return self.name |         return self.name | ||||||
|  |  | ||||||
|     def model_class(self): |     def model_class(self): | ||||||
|   | |||||||
| @@ -7,6 +7,7 @@ from django.db import models | |||||||
| from django.utils import dateformat | from django.utils import dateformat | ||||||
| from django.utils.text import capfirst | from django.utils.text import capfirst | ||||||
| from django.utils.translation import get_date_formats | from django.utils.translation import get_date_formats | ||||||
|  | from django.utils.encoding import smart_unicode, smart_str, iri_to_uri | ||||||
|  |  | ||||||
| EMPTY_VALUE = '(None)' | EMPTY_VALUE = '(None)' | ||||||
|  |  | ||||||
| @@ -19,7 +20,7 @@ class EasyModel(object): | |||||||
|         self.verbose_name_plural = model._meta.verbose_name_plural |         self.verbose_name_plural = model._meta.verbose_name_plural | ||||||
|  |  | ||||||
|     def __repr__(self): |     def __repr__(self): | ||||||
|         return '<EasyModel for %s>' % self.model._meta.object_name |         return '<EasyModel for %s>' % smart_str(self.model._meta.object_name) | ||||||
|  |  | ||||||
|     def model_databrowse(self): |     def model_databrowse(self): | ||||||
|         "Returns the ModelDatabrowse class for this model." |         "Returns the ModelDatabrowse class for this model." | ||||||
| @@ -54,7 +55,7 @@ class EasyField(object): | |||||||
|         self.model, self.field = easy_model, field |         self.model, self.field = easy_model, field | ||||||
|  |  | ||||||
|     def __repr__(self): |     def __repr__(self): | ||||||
|         return '<EasyField for %s.%s>' % (self.model.model._meta.object_name, self.field.name) |         return smart_str(u'<EasyField for %s.%s>' % (self.model.model._meta.object_name, self.field.name)) | ||||||
|  |  | ||||||
|     def choices(self): |     def choices(self): | ||||||
|         for value, label in self.field.choices: |         for value, label in self.field.choices: | ||||||
| @@ -72,29 +73,32 @@ class EasyChoice(object): | |||||||
|         self.value, self.label = value, label |         self.value, self.label = value, label | ||||||
|  |  | ||||||
|     def __repr__(self): |     def __repr__(self): | ||||||
|         return '<EasyChoice for %s.%s>' % (self.model.model._meta.object_name, self.field.name) |         return smart_str(u'<EasyChoice for %s.%s>' % (self.model.model._meta.object_name, self.field.name)) | ||||||
|  |  | ||||||
|     def url(self): |     def url(self): | ||||||
|         return '%s%s/%s/%s/%s/' % (self.model.site.root_url, self.model.model._meta.app_label, self.model.model._meta.module_name, self.field.field.name, self.value) |         return '%s%s/%s/%s/%s/' % (self.model.site.root_url, self.model.model._meta.app_label, self.model.model._meta.module_name, self.field.field.name, iri_to_uri(self.value)) | ||||||
|  |  | ||||||
| class EasyInstance(object): | class EasyInstance(object): | ||||||
|     def __init__(self, easy_model, instance): |     def __init__(self, easy_model, instance): | ||||||
|         self.model, self.instance = easy_model, instance |         self.model, self.instance = easy_model, instance | ||||||
|  |  | ||||||
|     def __repr__(self): |     def __repr__(self): | ||||||
|         return '<EasyInstance for %s (%s)>' % (self.model.model._meta.object_name, self.instance._get_pk_val()) |         return smart_str(u'<EasyInstance for %s (%s)>' % (self.model.model._meta.object_name, self.instance._get_pk_val())) | ||||||
|  |  | ||||||
|  |     def __unicode__(self): | ||||||
|  |         val = smart_unicode(self.instance) | ||||||
|  |         if len(val) > 30: | ||||||
|  |             return val[:30] + u'...' | ||||||
|  |         return val | ||||||
|  |  | ||||||
|     def __str__(self): |     def __str__(self): | ||||||
|         val = str(self.instance) |         return self.__unicode__().encode('utf-8') | ||||||
|         if len(val) > 30: |  | ||||||
|             return val[:30] + '...' |  | ||||||
|         return val |  | ||||||
|  |  | ||||||
|     def pk(self): |     def pk(self): | ||||||
|         return self.instance._get_pk_val() |         return self.instance._get_pk_val() | ||||||
|  |  | ||||||
|     def url(self): |     def url(self): | ||||||
|         return '%s%s/%s/objects/%s/' % (self.model.site.root_url, self.model.model._meta.app_label, self.model.model._meta.module_name, self.pk()) |         return '%s%s/%s/objects/%s/' % (self.model.site.root_url, self.model.model._meta.app_label, self.model.model._meta.module_name, iri_to_uri(self.pk())) | ||||||
|  |  | ||||||
|     def fields(self): |     def fields(self): | ||||||
|         """ |         """ | ||||||
| @@ -126,7 +130,7 @@ class EasyInstanceField(object): | |||||||
|         self.raw_value = getattr(instance.instance, field.name) |         self.raw_value = getattr(instance.instance, field.name) | ||||||
|  |  | ||||||
|     def __repr__(self): |     def __repr__(self): | ||||||
|         return '<EasyInstanceField for %s.%s>' % (self.model.model._meta.object_name, self.field.name) |         return smart_str(u'<EasyInstanceField for %s.%s>' % (self.model.model._meta.object_name, self.field.name)) | ||||||
|  |  | ||||||
|     def values(self): |     def values(self): | ||||||
|         """ |         """ | ||||||
| @@ -175,18 +179,18 @@ class EasyInstanceField(object): | |||||||
|             if self.field.rel.to in self.model.model_list: |             if self.field.rel.to in self.model.model_list: | ||||||
|                 lst = [] |                 lst = [] | ||||||
|                 for value in self.values(): |                 for value in self.values(): | ||||||
|                     url = '%s%s/%s/objects/%s/' % (self.model.site.root_url, m.model._meta.app_label, m.model._meta.module_name, value._get_pk_val()) |                     url = '%s%s/%s/objects/%s/' % (self.model.site.root_url, m.model._meta.app_label, m.model._meta.module_name, iri_to_uri(value._get_pk_val())) | ||||||
|                     lst.append((str(value), url)) |                     lst.append((smart_unicode(value), url)) | ||||||
|             else: |             else: | ||||||
|                 lst = [(value, None) for value in self.values()] |                 lst = [(value, None) for value in self.values()] | ||||||
|         elif self.field.choices: |         elif self.field.choices: | ||||||
|             lst = [] |             lst = [] | ||||||
|             for value in self.values(): |             for value in self.values(): | ||||||
|                 url = '%s%s/%s/fields/%s/%s/' % (self.model.site.root_url, self.model.model._meta.app_label, self.model.model._meta.module_name, self.field.name, self.raw_value) |                 url = '%s%s/%s/fields/%s/%s/' % (self.model.site.root_url, self.model.model._meta.app_label, self.model.model._meta.module_name, self.field.name, iri_to_uri(self.raw_value)) | ||||||
|                 lst.append((value, url)) |                 lst.append((value, url)) | ||||||
|         elif isinstance(self.field, models.URLField): |         elif isinstance(self.field, models.URLField): | ||||||
|             val = self.values()[0] |             val = self.values()[0] | ||||||
|             lst = [(val, val)] |             lst = [(val, iri_to_uri(val))] | ||||||
|         else: |         else: | ||||||
|             lst = [(self.values()[0], None)] |             lst = [(self.values()[0], None)] | ||||||
|         return lst |         return lst | ||||||
|   | |||||||
| @@ -6,6 +6,7 @@ from django.shortcuts import render_to_response | |||||||
| from django.utils.text import capfirst | from django.utils.text import capfirst | ||||||
| from django.utils.translation import get_date_formats | from django.utils.translation import get_date_formats | ||||||
| from django.views.generic import date_based | from django.views.generic import date_based | ||||||
|  | from django.utils.encoding import force_unicode | ||||||
| import datetime | import datetime | ||||||
| import time | import time | ||||||
|  |  | ||||||
| @@ -27,13 +28,13 @@ class CalendarPlugin(DatabrowsePlugin): | |||||||
|     def model_index_html(self, request, model, site): |     def model_index_html(self, request, model, site): | ||||||
|         fields = self.field_dict(model) |         fields = self.field_dict(model) | ||||||
|         if not fields: |         if not fields: | ||||||
|             return '' |             return u'' | ||||||
|         return '<p class="filter"><strong>View calendar by:</strong> %s</p>' % \ |         return u'<p class="filter"><strong>View calendar by:</strong> %s</p>' % \ | ||||||
|             ', '.join(['<a href="calendars/%s/">%s</a>' % (f.name, capfirst(f.verbose_name)) for f in fields.values()]) |             u', '.join(['<a href="calendars/%s/">%s</a>' % (f.name, force_unicode(capfirst(f.verbose_name))) for f in fields.values()]) | ||||||
|  |  | ||||||
|     def urls(self, plugin_name, easy_instance_field): |     def urls(self, plugin_name, easy_instance_field): | ||||||
|         if isinstance(easy_instance_field.field, models.DateField): |         if isinstance(easy_instance_field.field, models.DateField): | ||||||
|             return ['%s%s/%s/%s/%s/%s/' % (easy_instance_field.model.url(), |             return [u'%s%s/%s/%s/%s/%s/' % (easy_instance_field.model.url(), | ||||||
|                 plugin_name, easy_instance_field.field.name, |                 plugin_name, easy_instance_field.field.name, | ||||||
|                 easy_instance_field.raw_value.year, |                 easy_instance_field.raw_value.year, | ||||||
|                 easy_instance_field.raw_value.strftime('%b').lower(), |                 easy_instance_field.raw_value.strftime('%b').lower(), | ||||||
|   | |||||||
| @@ -4,9 +4,11 @@ from django.contrib.databrowse.datastructures import EasyModel | |||||||
| from django.contrib.databrowse.sites import DatabrowsePlugin | from django.contrib.databrowse.sites import DatabrowsePlugin | ||||||
| from django.shortcuts import render_to_response | from django.shortcuts import render_to_response | ||||||
| from django.utils.text import capfirst | from django.utils.text import capfirst | ||||||
|  | from django.utils.encoding import smart_str, force_unicode | ||||||
| from django.views.generic import date_based | from django.views.generic import date_based | ||||||
| import datetime | import datetime | ||||||
| import time | import time | ||||||
|  | import urllib | ||||||
|  |  | ||||||
| class FieldChoicePlugin(DatabrowsePlugin): | class FieldChoicePlugin(DatabrowsePlugin): | ||||||
|     def __init__(self, field_filter=None): |     def __init__(self, field_filter=None): | ||||||
| @@ -29,15 +31,15 @@ class FieldChoicePlugin(DatabrowsePlugin): | |||||||
|     def model_index_html(self, request, model, site): |     def model_index_html(self, request, model, site): | ||||||
|         fields = self.field_dict(model) |         fields = self.field_dict(model) | ||||||
|         if not fields: |         if not fields: | ||||||
|             return '' |             return u'' | ||||||
|         return '<p class="filter"><strong>View by:</strong> %s</p>' % \ |         return u'<p class="filter"><strong>View by:</strong> %s</p>' % \ | ||||||
|             ', '.join(['<a href="fields/%s/">%s</a>' % (f.name, capfirst(f.verbose_name)) for f in fields.values()]) |             u', '.join(['<a href="fields/%s/">%s</a>' % (f.name, force_unicode(capfirst(f.verbose_name))) for f in fields.values()]) | ||||||
|  |  | ||||||
|     def urls(self, plugin_name, easy_instance_field): |     def urls(self, plugin_name, easy_instance_field): | ||||||
|         if easy_instance_field.field in self.field_dict(easy_instance_field.model.model).values(): |         if easy_instance_field.field in self.field_dict(easy_instance_field.model.model).values(): | ||||||
|             return ['%s%s/%s/%s/' % (easy_instance_field.model.url(), |             return [u'%s%s/%s/%s/' % (easy_instance_field.model.url(), | ||||||
|                 plugin_name, easy_instance_field.field.name, |                 plugin_name, easy_instance_field.field.name, | ||||||
|                 easy_instance_field.raw_value)] |                 urllib.quote(smart_str(easy_instance_field.raw_value)))] | ||||||
|  |  | ||||||
|     def model_view(self, request, model_databrowse, url): |     def model_view(self, request, model_databrowse, url): | ||||||
|         self.model, self.site = model_databrowse.model, model_databrowse.site |         self.model, self.site = model_databrowse.model, model_databrowse.site | ||||||
|   | |||||||
| @@ -60,7 +60,7 @@ class ModelDatabrowse(object): | |||||||
|  |  | ||||||
|     def main_view(self, request): |     def main_view(self, request): | ||||||
|         easy_model = EasyModel(self.site, self.model) |         easy_model = EasyModel(self.site, self.model) | ||||||
|         html_snippets = '\n'.join([p.model_index_html(request, self.model, self.site) for p in self.plugins.values()]) |         html_snippets = u'\n'.join([p.model_index_html(request, self.model, self.site) for p in self.plugins.values()]) | ||||||
|         return render_to_response('databrowse/model_detail.html', { |         return render_to_response('databrowse/model_detail.html', { | ||||||
|             'model': easy_model, |             'model': easy_model, | ||||||
|             'root_url': self.site.root_url, |             'root_url': self.site.root_url, | ||||||
|   | |||||||
| @@ -10,7 +10,7 @@ | |||||||
|  |  | ||||||
| <ul class="objectlist"> | <ul class="objectlist"> | ||||||
| {% for object in object_list %} | {% for object in object_list %} | ||||||
| <li class="{% cycle odd,even %}"><a href="{{ object }}/">{{ object|escape }}</a></li> | <li class="{% cycle odd,even %}"><a href="{{ object|iriencode }}/">{{ object|escape }}</a></li> | ||||||
| {% endfor %} | {% endfor %} | ||||||
| </ul> | </ul> | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| from django.core import validators | from django.core import validators | ||||||
| from django.db import models | from django.db import models | ||||||
| from django.contrib.sites.models import Site | from django.contrib.sites.models import Site | ||||||
| from django.utils.translation import gettext_lazy as _ | from django.utils.translation import ugettext_lazy as _ | ||||||
|  |  | ||||||
| class FlatPage(models.Model): | class FlatPage(models.Model): | ||||||
|     url = models.CharField(_('URL'), maxlength=100, validator_list=[validators.isAlphaNumericURL], db_index=True, |     url = models.CharField(_('URL'), maxlength=100, validator_list=[validators.isAlphaNumericURL], db_index=True, | ||||||
| @@ -20,8 +20,8 @@ class FlatPage(models.Model): | |||||||
|         verbose_name_plural = _('flat pages') |         verbose_name_plural = _('flat pages') | ||||||
|         ordering = ('url',) |         ordering = ('url',) | ||||||
|  |  | ||||||
|     def __str__(self): |     def __unicode__(self): | ||||||
|         return "%s -- %s" % (self.url, self.title) |         return u"%s -- %s" % (self.url, self.title) | ||||||
|  |  | ||||||
|     def get_absolute_url(self): |     def get_absolute_url(self): | ||||||
|         return self.url |         return self.url | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| from django.utils.translation import ngettext | from django.utils.translation import ungettext, ugettext as _ | ||||||
| from django.utils.translation import gettext_lazy as _ | from django.utils.encoding import force_unicode | ||||||
| from django import template | from django import template | ||||||
| import re | import re | ||||||
|  |  | ||||||
| @@ -16,8 +16,8 @@ def ordinal(value): | |||||||
|         return value |         return value | ||||||
|     t = (_('th'), _('st'), _('nd'), _('rd'), _('th'), _('th'), _('th'), _('th'), _('th'), _('th')) |     t = (_('th'), _('st'), _('nd'), _('rd'), _('th'), _('th'), _('th'), _('th'), _('th'), _('th')) | ||||||
|     if value % 100 in (11, 12, 13): # special case |     if value % 100 in (11, 12, 13): # special case | ||||||
|         return "%d%s" % (value, t[0]) |         return u"%d%s" % (value, t[0]) | ||||||
|     return '%d%s' % (value, t[value % 10]) |     return u'%d%s' % (value, t[value % 10]) | ||||||
| register.filter(ordinal) | register.filter(ordinal) | ||||||
|  |  | ||||||
| def intcomma(value): | def intcomma(value): | ||||||
| @@ -25,8 +25,8 @@ def intcomma(value): | |||||||
|     Converts an integer to a string containing commas every three digits. |     Converts an integer to a string containing commas every three digits. | ||||||
|     For example, 3000 becomes '3,000' and 45000 becomes '45,000'. |     For example, 3000 becomes '3,000' and 45000 becomes '45,000'. | ||||||
|     """ |     """ | ||||||
|     orig = str(value) |     orig = force_unicode(value) | ||||||
|     new = re.sub("^(-?\d+)(\d{3})", '\g<1>,\g<2>', str(value)) |     new = re.sub("^(-?\d+)(\d{3})", '\g<1>,\g<2>', orig) | ||||||
|     if orig == new: |     if orig == new: | ||||||
|         return new |         return new | ||||||
|     else: |     else: | ||||||
| @@ -44,13 +44,13 @@ def intword(value): | |||||||
|         return value |         return value | ||||||
|     if value < 1000000000: |     if value < 1000000000: | ||||||
|         new_value = value / 1000000.0 |         new_value = value / 1000000.0 | ||||||
|         return ngettext('%(value).1f million', '%(value).1f million', new_value) % {'value': new_value} |         return ungettext('%(value).1f million', '%(value).1f million', new_value) % {'value': new_value} | ||||||
|     if value < 1000000000000: |     if value < 1000000000000: | ||||||
|         new_value = value / 1000000000.0 |         new_value = value / 1000000000.0 | ||||||
|         return ngettext('%(value).1f billion', '%(value).1f billion', new_value) % {'value': new_value} |         return ungettext('%(value).1f billion', '%(value).1f billion', new_value) % {'value': new_value} | ||||||
|     if value < 1000000000000000: |     if value < 1000000000000000: | ||||||
|         new_value = value / 1000000000000.0 |         new_value = value / 1000000000000.0 | ||||||
|         return ngettext('%(value).1f trillion', '%(value).1f trillion', new_value) % {'value': new_value} |         return ungettext('%(value).1f trillion', '%(value).1f trillion', new_value) % {'value': new_value} | ||||||
|     return value |     return value | ||||||
| register.filter(intword) | register.filter(intword) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ Australian-specific Form helpers | |||||||
| from django.newforms import ValidationError | from django.newforms import ValidationError | ||||||
| from django.newforms.fields import Field, RegexField, Select, EMPTY_VALUES | from django.newforms.fields import Field, RegexField, Select, EMPTY_VALUES | ||||||
| from django.newforms.util import smart_unicode | from django.newforms.util import smart_unicode | ||||||
| from django.utils.translation import gettext | from django.utils.translation import ugettext | ||||||
| import re | import re | ||||||
|  |  | ||||||
| PHONE_DIGITS_RE = re.compile(r'^(\d{10})$') | PHONE_DIGITS_RE = re.compile(r'^(\d{10})$') | ||||||
| @@ -15,14 +15,14 @@ class AUPostCodeField(RegexField): | |||||||
|     def __init__(self, *args, **kwargs): |     def __init__(self, *args, **kwargs): | ||||||
|         super(AUPostCodeField, self).__init__(r'^\d{4}$', |         super(AUPostCodeField, self).__init__(r'^\d{4}$', | ||||||
|             max_length=None, min_length=None, |             max_length=None, min_length=None, | ||||||
|             error_message=gettext(u'Enter a 4 digit post code.'), |             error_message=ugettext('Enter a 4 digit post code.'), | ||||||
|             *args, **kwargs) |                     *args, **kwargs) | ||||||
|  |  | ||||||
| class AUPhoneNumberField(Field): | class AUPhoneNumberField(Field): | ||||||
|     """Australian phone number field.""" |     """Australian phone number field.""" | ||||||
|     def clean(self, value): |     def clean(self, value): | ||||||
|         """Validate a phone number. Strips parentheses, whitespace and |         """ | ||||||
|         hyphens. |         Validate a phone number. Strips parentheses, whitespace and hyphens. | ||||||
|         """ |         """ | ||||||
|         super(AUPhoneNumberField, self).clean(value) |         super(AUPhoneNumberField, self).clean(value) | ||||||
|         if value in EMPTY_VALUES: |         if value in EMPTY_VALUES: | ||||||
| @@ -39,5 +39,5 @@ class AUStateSelect(Select): | |||||||
|     choices. |     choices. | ||||||
|     """ |     """ | ||||||
|     def __init__(self, attrs=None): |     def __init__(self, attrs=None): | ||||||
|         from au_states import STATE_CHOICES # relative import |         from au_states import STATE_CHOICES | ||||||
|         super(AUStateSelect, self).__init__(attrs, choices=STATE_CHOICES) |         super(AUStateSelect, self).__init__(attrs, choices=STATE_CHOICES) | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ BR-specific Form helpers | |||||||
| from django.newforms import ValidationError | from django.newforms import ValidationError | ||||||
| from django.newforms.fields import Field, RegexField, CharField, Select, EMPTY_VALUES | from django.newforms.fields import Field, RegexField, CharField, Select, EMPTY_VALUES | ||||||
| from django.utils.encoding import smart_unicode | from django.utils.encoding import smart_unicode | ||||||
| from django.utils.translation import gettext | from django.utils.translation import ugettext | ||||||
| import re | import re | ||||||
|  |  | ||||||
| phone_digits_re = re.compile(r'^(\d{2})[-\.]?(\d{4})[-\.]?(\d{4})$') | phone_digits_re = re.compile(r'^(\d{2})[-\.]?(\d{4})[-\.]?(\d{4})$') | ||||||
| @@ -15,8 +15,8 @@ class BRZipCodeField(RegexField): | |||||||
|     def __init__(self, *args, **kwargs): |     def __init__(self, *args, **kwargs): | ||||||
|         super(BRZipCodeField, self).__init__(r'^\d{5}-\d{3}$', |         super(BRZipCodeField, self).__init__(r'^\d{5}-\d{3}$', | ||||||
|             max_length=None, min_length=None, |             max_length=None, min_length=None, | ||||||
|             error_message=gettext('Enter a zip code in the format XXXXX-XXX.'), |             error_message=ugettext('Enter a zip code in the format XXXXX-XXX.'), | ||||||
|             *args, **kwargs) |                     *args, **kwargs) | ||||||
|  |  | ||||||
| class BRPhoneNumberField(Field): | class BRPhoneNumberField(Field): | ||||||
|     def clean(self, value): |     def clean(self, value): | ||||||
| @@ -27,7 +27,7 @@ class BRPhoneNumberField(Field): | |||||||
|         m = phone_digits_re.search(value) |         m = phone_digits_re.search(value) | ||||||
|         if m: |         if m: | ||||||
|             return u'%s-%s-%s' % (m.group(1), m.group(2), m.group(3)) |             return u'%s-%s-%s' % (m.group(1), m.group(2), m.group(3)) | ||||||
|         raise ValidationError(gettext(u'Phone numbers must be in XX-XXXX-XXXX format.')) |         raise ValidationError(ugettext('Phone numbers must be in XX-XXXX-XXXX format.')) | ||||||
|  |  | ||||||
| class BRStateSelect(Select): | class BRStateSelect(Select): | ||||||
|     """ |     """ | ||||||
| @@ -35,7 +35,7 @@ class BRStateSelect(Select): | |||||||
|     as its choices. |     as its choices. | ||||||
|     """ |     """ | ||||||
|     def __init__(self, attrs=None): |     def __init__(self, attrs=None): | ||||||
|         from br_states import STATE_CHOICES # relative import |         from br_states import STATE_CHOICES | ||||||
|         super(BRStateSelect, self).__init__(attrs, choices=STATE_CHOICES) |         super(BRStateSelect, self).__init__(attrs, choices=STATE_CHOICES) | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -69,9 +69,9 @@ class BRCPFField(CharField): | |||||||
|         try: |         try: | ||||||
|             int(value) |             int(value) | ||||||
|         except ValueError: |         except ValueError: | ||||||
|             raise ValidationError(gettext("This field requires only numbers.")) |             raise ValidationError(ugettext("This field requires only numbers.")) | ||||||
|         if len(value) != 11: |         if len(value) != 11: | ||||||
|             raise ValidationError(gettext("This field requires at most 11 digits or 14 characters.")) |             raise ValidationError(ugettext("This field requires at most 11 digits or 14 characters.")) | ||||||
|         orig_dv = value[-2:] |         orig_dv = value[-2:] | ||||||
|  |  | ||||||
|         new_1dv = sum([i * int(value[idx]) for idx, i in enumerate(range(10, 1, -1))]) |         new_1dv = sum([i * int(value[idx]) for idx, i in enumerate(range(10, 1, -1))]) | ||||||
| @@ -81,7 +81,7 @@ class BRCPFField(CharField): | |||||||
|         new_2dv = DV_maker(new_2dv % 11) |         new_2dv = DV_maker(new_2dv % 11) | ||||||
|         value = value[:-1] + str(new_2dv) |         value = value[:-1] + str(new_2dv) | ||||||
|         if value[-2:] != orig_dv: |         if value[-2:] != orig_dv: | ||||||
|             raise ValidationError(gettext("Invalid CPF number.")) |             raise ValidationError(ugettext("Invalid CPF number.")) | ||||||
|  |  | ||||||
|         return orig_value |         return orig_value | ||||||
|  |  | ||||||
| @@ -103,7 +103,7 @@ class BRCNPJField(Field): | |||||||
|             raise ValidationError("This field requires only numbers.") |             raise ValidationError("This field requires only numbers.") | ||||||
|         if len(value) != 14: |         if len(value) != 14: | ||||||
|             raise ValidationError( |             raise ValidationError( | ||||||
|                 gettext("This field requires at least 14 digits")) |                 ugettext("This field requires at least 14 digits")) | ||||||
|         orig_dv = value[-2:] |         orig_dv = value[-2:] | ||||||
|  |  | ||||||
|         new_1dv = sum([i * int(value[idx]) for idx, i in enumerate(range(5, 1, -1) + range(9, 1, -1))]) |         new_1dv = sum([i * int(value[idx]) for idx, i in enumerate(range(5, 1, -1) + range(9, 1, -1))]) | ||||||
| @@ -113,7 +113,7 @@ class BRCNPJField(Field): | |||||||
|         new_2dv = DV_maker(new_2dv % 11) |         new_2dv = DV_maker(new_2dv % 11) | ||||||
|         value = value[:-1] + str(new_2dv) |         value = value[:-1] + str(new_2dv) | ||||||
|         if value[-2:] != orig_dv: |         if value[-2:] != orig_dv: | ||||||
|             raise ValidationError(gettext("Invalid CNPJ number.")) |             raise ValidationError(ugettext("Invalid CNPJ number.")) | ||||||
|  |  | ||||||
|         return orig_value |         return orig_value | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| # -*- coding: utf-8 -* | # -*- coding: utf-8 -* | ||||||
| from django.utils.translation import gettext_lazy as _ | from django.utils.translation import ugettext_lazy as _ | ||||||
|  |  | ||||||
| STATE_CHOICES = ( | STATE_CHOICES = ( | ||||||
|     ('AG', _('Aargau')), |     ('AG', _('Aargau')), | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ Swiss-specific Form helpers | |||||||
| from django.newforms import ValidationError | from django.newforms import ValidationError | ||||||
| from django.newforms.fields import Field, RegexField, Select, EMPTY_VALUES | from django.newforms.fields import Field, RegexField, Select, EMPTY_VALUES | ||||||
| from django.utils.encoding import smart_unicode | from django.utils.encoding import smart_unicode | ||||||
| from django.utils.translation import gettext | from django.utils.translation import ugettext | ||||||
| import re | import re | ||||||
|  |  | ||||||
| id_re = re.compile(r"^(?P<idnumber>\w{8})(?P<pos9>(\d{1}|<))(?P<checksum>\d{1})$") | id_re = re.compile(r"^(?P<idnumber>\w{8})(?P<pos9>(\d{1}|<))(?P<checksum>\d{1})$") | ||||||
| @@ -15,7 +15,7 @@ class CHZipCodeField(RegexField): | |||||||
|     def __init__(self, *args, **kwargs): |     def __init__(self, *args, **kwargs): | ||||||
|         super(CHZipCodeField, self).__init__(r'^\d{4}$', |         super(CHZipCodeField, self).__init__(r'^\d{4}$', | ||||||
|         max_length=None, min_length=None, |         max_length=None, min_length=None, | ||||||
|         error_message=gettext('Enter a zip code in the format XXXX.'), |         error_message=ugettext('Enter a zip code in the format XXXX.'), | ||||||
|         *args, **kwargs) |         *args, **kwargs) | ||||||
|  |  | ||||||
| class CHPhoneNumberField(Field): | class CHPhoneNumberField(Field): | ||||||
| @@ -87,7 +87,7 @@ class CHIdentityCardNumberField(Field): | |||||||
|  |  | ||||||
|     def clean(self, value): |     def clean(self, value): | ||||||
|         super(CHIdentityCardNumberField, self).clean(value) |         super(CHIdentityCardNumberField, self).clean(value) | ||||||
|         error_msg = gettext('Enter a valid Swiss identity or passport card number in X1234567<0 or 1234567890 format.') |         error_msg = ugettext('Enter a valid Swiss identity or passport card number in X1234567<0 or 1234567890 format.') | ||||||
|         if value in EMPTY_VALUES: |         if value in EMPTY_VALUES: | ||||||
|             return u'' |             return u'' | ||||||
|  |  | ||||||
|   | |||||||
| @@ -4,7 +4,8 @@ Chile specific form helpers. | |||||||
|  |  | ||||||
| from django.newforms import ValidationError | from django.newforms import ValidationError | ||||||
| from django.newforms.fields import RegexField, EMPTY_VALUES | from django.newforms.fields import RegexField, EMPTY_VALUES | ||||||
| from django.utils.translation import gettext | from django.utils.translation import ugettext | ||||||
|  | from django.utils.encoding import smart_unicode | ||||||
|  |  | ||||||
| class CLRutField(RegexField): | class CLRutField(RegexField): | ||||||
|     """ |     """ | ||||||
| @@ -18,12 +19,12 @@ class CLRutField(RegexField): | |||||||
|         if 'strict' in kwargs: |         if 'strict' in kwargs: | ||||||
|             del kwargs['strict'] |             del kwargs['strict'] | ||||||
|             super(CLRutField, self).__init__(r'^(\d{1,2}\.)?\d{3}\.\d{3}-[\dkK]$', |             super(CLRutField, self).__init__(r'^(\d{1,2}\.)?\d{3}\.\d{3}-[\dkK]$', | ||||||
|                 error_message=gettext('Enter valid a Chilean RUT. The format is XX.XXX.XXX-X.'), |                 error_message=ugettext('Enter valid a Chilean RUT. The format is XX.XXX.XXX-X.'), | ||||||
|                 *args, **kwargs) |                         *args, **kwargs) | ||||||
|         else: |         else: | ||||||
|             # In non-strict mode, accept RUTs that validate but do not exist in |             # In non-strict mode, accept RUTs that validate but do not exist in | ||||||
|             # the real world. |             # the real world. | ||||||
|             super(CLRutField, self).__init__(r'^[\d\.]{1,11}-?[\dkK]$', error_message=gettext(u'Enter valid a Chilean RUT'), *args, **kwargs) |             super(CLRutField, self).__init__(r'^[\d\.]{1,11}-?[\dkK]$', error_message=ugettext('Enter valid a Chilean RUT'), *args, **kwargs) | ||||||
|  |  | ||||||
|     def clean(self, value): |     def clean(self, value): | ||||||
|         """ |         """ | ||||||
| @@ -49,14 +50,14 @@ class CLRutField(RegexField): | |||||||
|             multi += 1 |             multi += 1 | ||||||
|             if multi == 8: |             if multi == 8: | ||||||
|                 multi = 2 |                 multi = 2 | ||||||
|         return '0123456789K0'[11 - suma % 11] |         return u'0123456789K0'[11 - suma % 11] | ||||||
|  |  | ||||||
|     def _canonify(self, rut): |     def _canonify(self, rut): | ||||||
|         """ |         """ | ||||||
|         Turns the RUT into one normalized format. Returns a (rut, verifier) |         Turns the RUT into one normalized format. Returns a (rut, verifier) | ||||||
|         tuple. |         tuple. | ||||||
|         """ |         """ | ||||||
|         rut = str(rut).replace(' ', '').replace('.', '').replace('-', '') |         rut = smart_unicode(rut).replace(' ', '').replace('.', '').replace('-', '') | ||||||
|         return rut[:-1], rut[-1] |         return rut[:-1], rut[-1] | ||||||
|  |  | ||||||
|     def _format(self, code, verifier=None): |     def _format(self, code, verifier=None): | ||||||
| @@ -74,5 +75,5 @@ class CLRutField(RegexField): | |||||||
|             else: |             else: | ||||||
|                 new_dot = pos - 3 |                 new_dot = pos - 3 | ||||||
|             code = code[:new_dot] + '.' + code[new_dot:] |             code = code[:new_dot] + '.' + code[new_dot:] | ||||||
|         return '%s-%s' % (code, verifier) |         return u'%s-%s' % (code, verifier) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| # -*- coding: utf-8 -* | # -*- coding: utf-8 -* | ||||||
| from django.utils.translation import gettext_lazy as _ | from django.utils.translation import ugettext_lazy as _ | ||||||
|  |  | ||||||
| STATE_CHOICES = ( | STATE_CHOICES = ( | ||||||
|     ('BW', _('Baden-Wuerttemberg')), |     ('BW', _('Baden-Wuerttemberg')), | ||||||
|   | |||||||
| @@ -4,7 +4,7 @@ DE-specific Form helpers | |||||||
|  |  | ||||||
| from django.newforms import ValidationError | from django.newforms import ValidationError | ||||||
| from django.newforms.fields import Field, RegexField, Select, EMPTY_VALUES | from django.newforms.fields import Field, RegexField, Select, EMPTY_VALUES | ||||||
| from django.utils.translation import gettext | from django.utils.translation import ugettext | ||||||
| import re | import re | ||||||
|  |  | ||||||
| id_re = re.compile(r"^(?P<residence>\d{10})(?P<origin>\w{1,3})[-\ ]?(?P<birthday>\d{7})[-\ ]?(?P<validity>\d{7})[-\ ]?(?P<checksum>\d{1})$") | id_re = re.compile(r"^(?P<residence>\d{10})(?P<origin>\w{1,3})[-\ ]?(?P<birthday>\d{7})[-\ ]?(?P<validity>\d{7})[-\ ]?(?P<checksum>\d{1})$") | ||||||
| @@ -13,15 +13,15 @@ class DEZipCodeField(RegexField): | |||||||
|     def __init__(self, *args, **kwargs): |     def __init__(self, *args, **kwargs): | ||||||
|         super(DEZipCodeField, self).__init__(r'^\d{5}$', |         super(DEZipCodeField, self).__init__(r'^\d{5}$', | ||||||
|             max_length=None, min_length=None, |             max_length=None, min_length=None, | ||||||
|             error_message=gettext(u'Enter a zip code in the format XXXXX.'), |             error_message=ugettext('Enter a zip code in the format XXXXX.'), | ||||||
|             *args, **kwargs) |                     *args, **kwargs) | ||||||
|  |  | ||||||
| class DEStateSelect(Select): | class DEStateSelect(Select): | ||||||
|     """ |     """ | ||||||
|     A Select widget that uses a list of DE states as its choices. |     A Select widget that uses a list of DE states as its choices. | ||||||
|     """ |     """ | ||||||
|     def __init__(self, attrs=None): |     def __init__(self, attrs=None): | ||||||
|         from de_states import STATE_CHOICES # relative import |         from de_states import STATE_CHOICES | ||||||
|         super(DEStateSelect, self).__init__(attrs, choices=STATE_CHOICES) |         super(DEStateSelect, self).__init__(attrs, choices=STATE_CHOICES) | ||||||
|  |  | ||||||
| class DEIdentityCardNumberField(Field): | class DEIdentityCardNumberField(Field): | ||||||
| @@ -57,7 +57,7 @@ class DEIdentityCardNumberField(Field): | |||||||
|  |  | ||||||
|     def clean(self, value): |     def clean(self, value): | ||||||
|         super(DEIdentityCardNumberField, self).clean(value) |         super(DEIdentityCardNumberField, self).clean(value) | ||||||
|         error_msg = gettext(u'Enter a valid German identity card number in XXXXXXXXXXX-XXXXXXX-XXXXXXX-X format.') |         error_msg = ugettext('Enter a valid German identity card number in XXXXXXXXXXX-XXXXXXX-XXXXXXX-X format.') | ||||||
|         if value in EMPTY_VALUES: |         if value in EMPTY_VALUES: | ||||||
|             return u'' |             return u'' | ||||||
|         match = re.match(id_re, value) |         match = re.match(id_re, value) | ||||||
| @@ -71,7 +71,7 @@ class DEIdentityCardNumberField(Field): | |||||||
|         if residence == '0000000000' or birthday == '0000000' or validity == '0000000': |         if residence == '0000000000' or birthday == '0000000' or validity == '0000000': | ||||||
|             raise ValidationError(error_msg) |             raise ValidationError(error_msg) | ||||||
|  |  | ||||||
|         all_digits = "%s%s%s%s" % (residence, birthday, validity, checksum) |         all_digits = u"%s%s%s%s" % (residence, birthday, validity, checksum) | ||||||
|         if not self.has_valid_checksum(residence) or not self.has_valid_checksum(birthday) or \ |         if not self.has_valid_checksum(residence) or not self.has_valid_checksum(birthday) or \ | ||||||
|             not self.has_valid_checksum(validity) or not self.has_valid_checksum(all_digits): |             not self.has_valid_checksum(validity) or not self.has_valid_checksum(all_digits): | ||||||
|                 raise ValidationError(error_msg) |                 raise ValidationError(error_msg) | ||||||
|   | |||||||
| @@ -5,21 +5,21 @@ FI-specific Form helpers | |||||||
| import re | import re | ||||||
| from django.newforms import ValidationError | from django.newforms import ValidationError | ||||||
| from django.newforms.fields import Field, RegexField, Select, EMPTY_VALUES | from django.newforms.fields import Field, RegexField, Select, EMPTY_VALUES | ||||||
| from django.utils.translation import gettext | from django.utils.translation import ugettext | ||||||
|  |  | ||||||
| class FIZipCodeField(RegexField): | class FIZipCodeField(RegexField): | ||||||
|     def __init__(self, *args, **kwargs): |     def __init__(self, *args, **kwargs): | ||||||
|         super(FIZipCodeField, self).__init__(r'^\d{5}$', |         super(FIZipCodeField, self).__init__(r'^\d{5}$', | ||||||
|             max_length=None, min_length=None, |             max_length=None, min_length=None, | ||||||
|             error_message=gettext(u'Enter a zip code in the format XXXXX.'), |             error_message=ugettext('Enter a zip code in the format XXXXX.'), | ||||||
|             *args, **kwargs) |                     *args, **kwargs) | ||||||
|  |  | ||||||
| class FIMunicipalitySelect(Select): | class FIMunicipalitySelect(Select): | ||||||
|     """ |     """ | ||||||
|     A Select widget that uses a list of Finnish municipalities as its choices. |     A Select widget that uses a list of Finnish municipalities as its choices. | ||||||
|     """ |     """ | ||||||
|     def __init__(self, attrs=None): |     def __init__(self, attrs=None): | ||||||
|         from fi_municipalities import MUNICIPALITY_CHOICES # relative import |         from fi_municipalities import MUNICIPALITY_CHOICES | ||||||
|         super(FIMunicipalitySelect, self).__init__(attrs, choices=MUNICIPALITY_CHOICES) |         super(FIMunicipalitySelect, self).__init__(attrs, choices=MUNICIPALITY_CHOICES) | ||||||
|  |  | ||||||
| class FISocialSecurityNumber(Field): | class FISocialSecurityNumber(Field): | ||||||
| @@ -37,9 +37,9 @@ class FISocialSecurityNumber(Field): | |||||||
|             (?P<serial>(\d{3})) |             (?P<serial>(\d{3})) | ||||||
|             (?P<checksum>[%s])$""" % checkmarks, value, re.VERBOSE | re.IGNORECASE) |             (?P<checksum>[%s])$""" % checkmarks, value, re.VERBOSE | re.IGNORECASE) | ||||||
|         if not result: |         if not result: | ||||||
|             raise ValidationError(gettext(u'Enter a valid Finnish social security number.')) |             raise ValidationError(ugettext('Enter a valid Finnish social security number.')) | ||||||
|         gd = result.groupdict() |         gd = result.groupdict() | ||||||
|         checksum = int(gd['date'] + gd['serial']) |         checksum = int(gd['date'] + gd['serial']) | ||||||
|         if checkmarks[checksum % len(checkmarks)] == gd['checksum'].upper(): |         if checkmarks[checksum % len(checkmarks)] == gd['checksum'].upper(): | ||||||
|             return u'%s' % value.upper() |             return u'%s' % value.upper() | ||||||
|         raise ValidationError(gettext(u'Enter a valid Finnish social security number.')) |         raise ValidationError(ugettext('Enter a valid Finnish social security number.')) | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ FR-specific Form helpers | |||||||
| from django.newforms import ValidationError | from django.newforms import ValidationError | ||||||
| from django.newforms.fields import Field, RegexField, Select, EMPTY_VALUES | from django.newforms.fields import Field, RegexField, Select, EMPTY_VALUES | ||||||
| from django.utils.encoding import smart_unicode | from django.utils.encoding import smart_unicode | ||||||
| from django.utils.translation import gettext | from django.utils.translation import ugettext | ||||||
| import re | import re | ||||||
|  |  | ||||||
| phone_digits_re = re.compile(r'^0\d(\s|\.)?(\d{2}(\s|\.)?){3}\d{2}$') | phone_digits_re = re.compile(r'^0\d(\s|\.)?(\d{2}(\s|\.)?){3}\d{2}$') | ||||||
| @@ -14,8 +14,8 @@ class FRZipCodeField(RegexField): | |||||||
|     def __init__(self, *args, **kwargs): |     def __init__(self, *args, **kwargs): | ||||||
|         super(FRZipCodeField, self).__init__(r'^\d{5}$', |         super(FRZipCodeField, self).__init__(r'^\d{5}$', | ||||||
|             max_length=None, min_length=None, |             max_length=None, min_length=None, | ||||||
|             error_message=gettext(u'Enter a zip code in the format XXXXX.'), |             error_message=ugettext('Enter a zip code in the format XXXXX.'), | ||||||
|             *args, **kwargs) |                     *args, **kwargs) | ||||||
|  |  | ||||||
| class FRPhoneNumberField(Field): | class FRPhoneNumberField(Field): | ||||||
|     """ |     """ | ||||||
| @@ -39,6 +39,6 @@ class FRDepartmentSelect(Select): | |||||||
|     A Select widget that uses a list of FR departments as its choices. |     A Select widget that uses a list of FR departments as its choices. | ||||||
|     """ |     """ | ||||||
|     def __init__(self, attrs=None): |     def __init__(self, attrs=None): | ||||||
|         from fr_department import DEPARTMENT_ASCII_CHOICES # relative import |         from fr_department import DEPARTMENT_ASCII_CHOICES | ||||||
|         super(FRDepartmentSelect, self).__init__(attrs, choices=DEPARTMENT_ASCII_CHOICES) |         super(FRDepartmentSelect, self).__init__(attrs, choices=DEPARTMENT_ASCII_CHOICES) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -5,7 +5,8 @@ Iceland specific form helpers. | |||||||
| from django.newforms import ValidationError | from django.newforms import ValidationError | ||||||
| from django.newforms.fields import RegexField, EMPTY_VALUES | from django.newforms.fields import RegexField, EMPTY_VALUES | ||||||
| from django.newforms.widgets import Select | from django.newforms.widgets import Select | ||||||
| from django.utils.translation import gettext | from django.utils.translation import ugettext | ||||||
|  | from django.utils.encoding import smart_unicode | ||||||
|  |  | ||||||
| class ISIdNumberField(RegexField): | class ISIdNumberField(RegexField): | ||||||
|     """ |     """ | ||||||
| @@ -13,7 +14,7 @@ class ISIdNumberField(RegexField): | |||||||
|     of Iceland has. |     of Iceland has. | ||||||
|     """ |     """ | ||||||
|     def __init__(self, *args, **kwargs): |     def __init__(self, *args, **kwargs): | ||||||
|         error_msg = gettext(u'Enter a valid Icelandic identification number. The format is XXXXXX-XXXX.') |         error_msg = ugettext('Enter a valid Icelandic identification number. The format is XXXXXX-XXXX.') | ||||||
|         kwargs['min_length'],kwargs['max_length'] = 10,11 |         kwargs['min_length'],kwargs['max_length'] = 10,11 | ||||||
|         super(ISIdNumberField, self).__init__(r'^\d{6}(-| )?\d{4}$', error_message=error_msg, *args, **kwargs) |         super(ISIdNumberField, self).__init__(r'^\d{6}(-| )?\d{4}$', error_message=error_msg, *args, **kwargs) | ||||||
|  |  | ||||||
| @@ -27,7 +28,7 @@ class ISIdNumberField(RegexField): | |||||||
|         if self._validate(value): |         if self._validate(value): | ||||||
|             return self._format(value) |             return self._format(value) | ||||||
|         else: |         else: | ||||||
|             raise ValidationError(gettext(u'The Icelandic identification number is not valid.')) |             raise ValidationError(ugettext(u'The Icelandic identification number is not valid.')) | ||||||
|  |  | ||||||
|     def _canonify(self, value): |     def _canonify(self, value): | ||||||
|         """ |         """ | ||||||
| @@ -48,7 +49,7 @@ class ISIdNumberField(RegexField): | |||||||
|         Takes in the value in canonical form and returns it in the common |         Takes in the value in canonical form and returns it in the common | ||||||
|         display format. |         display format. | ||||||
|         """ |         """ | ||||||
|         return value[:6]+'-'+value[6:] |         return smart_unicode(value[:6]+'-'+value[6:]) | ||||||
|  |  | ||||||
| class ISPhoneNumberField(RegexField): | class ISPhoneNumberField(RegexField): | ||||||
|     """ |     """ | ||||||
|   | |||||||
| @@ -4,7 +4,7 @@ IT-specific Form helpers | |||||||
|  |  | ||||||
| from django.newforms import ValidationError | from django.newforms import ValidationError | ||||||
| from django.newforms.fields import Field, RegexField, Select, EMPTY_VALUES | from django.newforms.fields import Field, RegexField, Select, EMPTY_VALUES | ||||||
| from django.utils.translation import gettext | from django.utils.translation import ugettext | ||||||
| from django.utils.encoding import smart_unicode | from django.utils.encoding import smart_unicode | ||||||
| from django.contrib.localflavor.it.util import ssn_check_digit, vat_number_check_digit | from django.contrib.localflavor.it.util import ssn_check_digit, vat_number_check_digit | ||||||
| import re | import re | ||||||
| @@ -13,15 +13,15 @@ class ITZipCodeField(RegexField): | |||||||
|     def __init__(self, *args, **kwargs): |     def __init__(self, *args, **kwargs): | ||||||
|         super(ITZipCodeField, self).__init__(r'^\d{5}$', |         super(ITZipCodeField, self).__init__(r'^\d{5}$', | ||||||
|         max_length=None, min_length=None, |         max_length=None, min_length=None, | ||||||
|         error_message=gettext(u'Enter a valid zip code.'), |         error_message=ugettext('Enter a valid zip code.'), | ||||||
|         *args, **kwargs) |                 *args, **kwargs) | ||||||
|  |  | ||||||
| class ITRegionSelect(Select): | class ITRegionSelect(Select): | ||||||
|     """ |     """ | ||||||
|     A Select widget that uses a list of IT regions as its choices. |     A Select widget that uses a list of IT regions as its choices. | ||||||
|     """ |     """ | ||||||
|     def __init__(self, attrs=None): |     def __init__(self, attrs=None): | ||||||
|         from it_region import REGION_CHOICES # relative import |         from it_region import REGION_CHOICES | ||||||
|         super(ITRegionSelect, self).__init__(attrs, choices=REGION_CHOICES) |         super(ITRegionSelect, self).__init__(attrs, choices=REGION_CHOICES) | ||||||
|  |  | ||||||
| class ITProvinceSelect(Select): | class ITProvinceSelect(Select): | ||||||
| @@ -29,7 +29,7 @@ class ITProvinceSelect(Select): | |||||||
|     A Select widget that uses a list of IT regions as its choices. |     A Select widget that uses a list of IT regions as its choices. | ||||||
|     """ |     """ | ||||||
|     def __init__(self, attrs=None): |     def __init__(self, attrs=None): | ||||||
|         from it_province import PROVINCE_CHOICES # relative import |         from it_province import PROVINCE_CHOICES | ||||||
|         super(ITProvinceSelect, self).__init__(attrs, choices=PROVINCE_CHOICES) |         super(ITProvinceSelect, self).__init__(attrs, choices=PROVINCE_CHOICES) | ||||||
|  |  | ||||||
| class ITSocialSecurityNumberField(RegexField): | class ITSocialSecurityNumberField(RegexField): | ||||||
| @@ -38,7 +38,7 @@ class ITSocialSecurityNumberField(RegexField): | |||||||
|     For reference see http://www.agenziaentrate.it/ and search for |     For reference see http://www.agenziaentrate.it/ and search for | ||||||
|     'Informazioni sulla codificazione delle persone fisiche'. |     'Informazioni sulla codificazione delle persone fisiche'. | ||||||
|     """ |     """ | ||||||
|     err_msg = gettext(u'Enter a valid Social Security number.') |     err_msg = ugettext(u'Enter a valid Social Security number.') | ||||||
|     def __init__(self, *args, **kwargs): |     def __init__(self, *args, **kwargs): | ||||||
|         super(ITSocialSecurityNumberField, self).__init__(r'^\w{3}\s*\w{3}\s*\w{5}\s*\w{5}$', |         super(ITSocialSecurityNumberField, self).__init__(r'^\w{3}\s*\w{3}\s*\w{5}\s*\w{5}$', | ||||||
|         max_length=None, min_length=None, error_message=self.err_msg, |         max_length=None, min_length=None, error_message=self.err_msg, | ||||||
| @@ -65,7 +65,7 @@ class ITVatNumberField(Field): | |||||||
|         value = super(ITVatNumberField, self).clean(value) |         value = super(ITVatNumberField, self).clean(value) | ||||||
|         if value == u'': |         if value == u'': | ||||||
|             return value |             return value | ||||||
|         err_msg = gettext(u'Enter a valid VAT number.') |         err_msg = ugettext(u'Enter a valid VAT number.') | ||||||
|         try: |         try: | ||||||
|             vat_number = int(value) |             vat_number = int(value) | ||||||
|         except ValueError: |         except ValueError: | ||||||
|   | |||||||
| @@ -1,23 +1,27 @@ | |||||||
|  | from django.utils.encoding import smart_str, smart_unicode | ||||||
|  |  | ||||||
| def ssn_check_digit(value): | def ssn_check_digit(value): | ||||||
|     "Calculate Italian social security number check digit." |     "Calculate Italian social security number check digit." | ||||||
|     ssn_even_chars = { |     ssn_even_chars = { | ||||||
|         '0': 0, '1': 1, '2': 2, '3': 3, '4': 4, '5': 5, '6': 6, '7': 7, '8': 8, '9': 9, |         '0': 0, '1': 1, '2': 2, '3': 3, '4': 4, '5': 5, '6': 6, '7': 7, '8': 8, | ||||||
|         'A': 0, 'B': 1, 'C': 2, 'D': 3, 'E': 4, 'F': 5, 'G': 6, 'H': 7, 'I': 8, 'J': 9, |         '9': 9, 'A': 0, 'B': 1, 'C': 2, 'D': 3, 'E': 4, 'F': 5, 'G': 6, 'H': 7, | ||||||
|         'K': 10, 'L': 11, 'M': 12, 'N': 13, 'O': 14, 'P': 15, 'Q': 16, 'R': 17, 'S': 18, |         'I': 8, 'J': 9, 'K': 10, 'L': 11, 'M': 12, 'N': 13, 'O': 14, 'P': 15, | ||||||
|         'T': 19, 'U': 20, 'V': 21, 'W': 22, 'X': 23, 'Y': 24, 'Z': 25 |         'Q': 16, 'R': 17, 'S': 18, 'T': 19, 'U': 20, 'V': 21, 'W': 22, 'X': 23, | ||||||
|  |         'Y': 24, 'Z': 25 | ||||||
|     } |     } | ||||||
|     ssn_odd_chars = { |     ssn_odd_chars = { | ||||||
|         '0': 1, '1': 0, '2': 5, '3': 7, '4': 9, '5': 13, '6': 15, '7': 17, '8': 19, '9': 21, |         '0': 1, '1': 0, '2': 5, '3': 7, '4': 9, '5': 13, '6': 15, '7': 17, '8': | ||||||
|         'A': 1, 'B': 0, 'C': 5, 'D': 7, 'E': 9, 'F': 13, 'G': 15, 'H': 17, 'I': 19, 'J': 21, |         19, '9': 21, 'A': 1, 'B': 0, 'C': 5, 'D': 7, 'E': 9, 'F': 13, 'G': 15, | ||||||
|         'K': 2, 'L': 4, 'M': 18, 'N': 20, 'O': 11, 'P': 3, 'Q': 6, 'R': 8, 'S': 12, |         'H': 17, 'I': 19, 'J': 21, 'K': 2, 'L': 4, 'M': 18, 'N': 20, 'O': 11, | ||||||
|         'T': 14, 'U': 16, 'V': 10, 'W': 22, 'X': 25, 'Y': 24, 'Z': 23 |         'P': 3, 'Q': 6, 'R': 8, 'S': 12, 'T': 14, 'U': 16, 'V': 10, 'W': 22, | ||||||
|  |         'X': 25, 'Y': 24, 'Z': 23 | ||||||
|     } |     } | ||||||
|     # Chars from 'A' to 'Z' |     # Chars from 'A' to 'Z' | ||||||
|     ssn_check_digits = [chr(x) for x in range(65, 91)] |     ssn_check_digits = [chr(x) for x in range(65, 91)] | ||||||
|  |  | ||||||
|     ssn = value.upper() |     ssn = value.upper() | ||||||
|     total = 0 |     total = 0 | ||||||
|     for i in range(0,15): |     for i in range(0, 15): | ||||||
|         try: |         try: | ||||||
|             if i % 2 == 0: |             if i % 2 == 0: | ||||||
|                 total += ssn_odd_chars[ssn[i]] |                 total += ssn_odd_chars[ssn[i]] | ||||||
| @@ -30,11 +34,11 @@ def ssn_check_digit(value): | |||||||
|  |  | ||||||
| def vat_number_check_digit(vat_number): | def vat_number_check_digit(vat_number): | ||||||
|     "Calculate Italian VAT number check digit." |     "Calculate Italian VAT number check digit." | ||||||
|     normalized_vat_number = str(vat_number).zfill(10) |     normalized_vat_number = smart_str(vat_number).zfill(10) | ||||||
|     total = 0 |     total = 0 | ||||||
|     for i in range(0, 10, 2): |     for i in range(0, 10, 2): | ||||||
|         total += int(normalized_vat_number[i]) |         total += int(normalized_vat_number[i]) | ||||||
|     for i in range(1, 11, 2): |     for i in range(1, 11, 2): | ||||||
|         quotient , remainder = divmod(int(normalized_vat_number[i]) * 2, 10) |         quotient , remainder = divmod(int(normalized_vat_number[i]) * 2, 10) | ||||||
|         total += quotient + remainder |         total += quotient + remainder | ||||||
|     return str((10 - total % 10) % 10) |     return smart_unicode((10 - total % 10) % 10) | ||||||
|   | |||||||
| @@ -4,7 +4,7 @@ JP-specific Form helpers | |||||||
|  |  | ||||||
| from django.core import validators | from django.core import validators | ||||||
| from django.newforms import ValidationError | from django.newforms import ValidationError | ||||||
| from django.utils.translation import gettext | from django.utils.translation import ugettext | ||||||
| from django.newforms.fields import RegexField, Select | from django.newforms.fields import RegexField, Select | ||||||
|  |  | ||||||
| import re | import re | ||||||
| @@ -18,8 +18,8 @@ class JPPostalCodeField(RegexField): | |||||||
|     def __init__(self, *args, **kwargs): |     def __init__(self, *args, **kwargs): | ||||||
|         super(JPPostalCodeField, self).__init__(r'^\d{3}-\d{4}$|^\d{7}$', |         super(JPPostalCodeField, self).__init__(r'^\d{3}-\d{4}$|^\d{7}$', | ||||||
|             max_length=None, min_length=None, |             max_length=None, min_length=None, | ||||||
|             error_message=gettext(u'Enter a postal code in the format XXXXXXX or XXX-XXXX.'), |             error_message=ugettext('Enter a postal code in the format XXXXXXX or XXX-XXXX.'), | ||||||
|             *args, **kwargs) |                     *args, **kwargs) | ||||||
|  |  | ||||||
|     def clean(self, value): |     def clean(self, value): | ||||||
|         """ |         """ | ||||||
|   | |||||||
| @@ -1,51 +1,51 @@ | |||||||
| from django.utils.translation import gettext_lazy as gettext_lazy | from django.utils.translation import ugettext_lazy | ||||||
|  |  | ||||||
| JP_PREFECTURES = ( | JP_PREFECTURES = ( | ||||||
|     ('hokkaido', gettext_lazy('Hokkaido'),), |     ('hokkaido', ugettext_lazy('Hokkaido'),), | ||||||
|     ('aomori', gettext_lazy('Aomori'),), |     ('aomori', ugettext_lazy('Aomori'),), | ||||||
|     ('iwate', gettext_lazy('Iwate'),), |     ('iwate', ugettext_lazy('Iwate'),), | ||||||
|     ('miyagi', gettext_lazy('Miyagi'),), |     ('miyagi', ugettext_lazy('Miyagi'),), | ||||||
|     ('akita', gettext_lazy('Akita'),), |     ('akita', ugettext_lazy('Akita'),), | ||||||
|     ('yamagata', gettext_lazy('Yamagata'),), |     ('yamagata', ugettext_lazy('Yamagata'),), | ||||||
|     ('fukushima', gettext_lazy('Fukushima'),), |     ('fukushima', ugettext_lazy('Fukushima'),), | ||||||
|     ('ibaraki', gettext_lazy('Ibaraki'),), |     ('ibaraki', ugettext_lazy('Ibaraki'),), | ||||||
|     ('tochigi', gettext_lazy('Tochigi'),), |     ('tochigi', ugettext_lazy('Tochigi'),), | ||||||
|     ('gunma', gettext_lazy('Gunma'),), |     ('gunma', ugettext_lazy('Gunma'),), | ||||||
|     ('saitama', gettext_lazy('Saitama'),), |     ('saitama', ugettext_lazy('Saitama'),), | ||||||
|     ('chiba', gettext_lazy('Chiba'),), |     ('chiba', ugettext_lazy('Chiba'),), | ||||||
|     ('tokyo', gettext_lazy('Tokyo'),), |     ('tokyo', ugettext_lazy('Tokyo'),), | ||||||
|     ('kanagawa', gettext_lazy('Kanagawa'),), |     ('kanagawa', ugettext_lazy('Kanagawa'),), | ||||||
|     ('yamanashi', gettext_lazy('Yamanashi'),), |     ('yamanashi', ugettext_lazy('Yamanashi'),), | ||||||
|     ('nagano', gettext_lazy('Nagano'),), |     ('nagano', ugettext_lazy('Nagano'),), | ||||||
|     ('niigata', gettext_lazy('Niigata'),), |     ('niigata', ugettext_lazy('Niigata'),), | ||||||
|     ('toyama', gettext_lazy('Toyama'),), |     ('toyama', ugettext_lazy('Toyama'),), | ||||||
|     ('ishikawa', gettext_lazy('Ishikawa'),), |     ('ishikawa', ugettext_lazy('Ishikawa'),), | ||||||
|     ('fukui', gettext_lazy('Fukui'),), |     ('fukui', ugettext_lazy('Fukui'),), | ||||||
|     ('gifu', gettext_lazy('Gifu'),), |     ('gifu', ugettext_lazy('Gifu'),), | ||||||
|     ('shizuoka', gettext_lazy('Shizuoka'),), |     ('shizuoka', ugettext_lazy('Shizuoka'),), | ||||||
|     ('aichi', gettext_lazy('Aichi'),), |     ('aichi', ugettext_lazy('Aichi'),), | ||||||
|     ('mie', gettext_lazy('Mie'),), |     ('mie', ugettext_lazy('Mie'),), | ||||||
|     ('shiga', gettext_lazy('Shiga'),), |     ('shiga', ugettext_lazy('Shiga'),), | ||||||
|     ('kyoto', gettext_lazy('Kyoto'),), |     ('kyoto', ugettext_lazy('Kyoto'),), | ||||||
|     ('osaka', gettext_lazy('Osaka'),), |     ('osaka', ugettext_lazy('Osaka'),), | ||||||
|     ('hyogo', gettext_lazy('Hyogo'),), |     ('hyogo', ugettext_lazy('Hyogo'),), | ||||||
|     ('nara', gettext_lazy('Nara'),), |     ('nara', ugettext_lazy('Nara'),), | ||||||
|     ('wakayama', gettext_lazy('Wakayama'),), |     ('wakayama', ugettext_lazy('Wakayama'),), | ||||||
|     ('tottori', gettext_lazy('Tottori'),), |     ('tottori', ugettext_lazy('Tottori'),), | ||||||
|     ('shimane', gettext_lazy('Shimane'),), |     ('shimane', ugettext_lazy('Shimane'),), | ||||||
|     ('okayama', gettext_lazy('Okayama'),), |     ('okayama', ugettext_lazy('Okayama'),), | ||||||
|     ('hiroshima', gettext_lazy('Hiroshima'),), |     ('hiroshima', ugettext_lazy('Hiroshima'),), | ||||||
|     ('yamaguchi', gettext_lazy('Yamaguchi'),), |     ('yamaguchi', ugettext_lazy('Yamaguchi'),), | ||||||
|     ('tokushima', gettext_lazy('Tokushima'),), |     ('tokushima', ugettext_lazy('Tokushima'),), | ||||||
|     ('kagawa', gettext_lazy('Kagawa'),), |     ('kagawa', ugettext_lazy('Kagawa'),), | ||||||
|     ('ehime', gettext_lazy('Ehime'),), |     ('ehime', ugettext_lazy('Ehime'),), | ||||||
|     ('kochi', gettext_lazy('Kochi'),), |     ('kochi', ugettext_lazy('Kochi'),), | ||||||
|     ('fukuoka', gettext_lazy('Fukuoka'),), |     ('fukuoka', ugettext_lazy('Fukuoka'),), | ||||||
|     ('saga', gettext_lazy('Saga'),), |     ('saga', ugettext_lazy('Saga'),), | ||||||
|     ('nagasaki', gettext_lazy('Nagasaki'),), |     ('nagasaki', ugettext_lazy('Nagasaki'),), | ||||||
|     ('kumamoto', gettext_lazy('Kumamoto'),), |     ('kumamoto', ugettext_lazy('Kumamoto'),), | ||||||
|     ('oita', gettext_lazy('Oita'),), |     ('oita', ugettext_lazy('Oita'),), | ||||||
|     ('miyazaki', gettext_lazy('Miyazaki'),), |     ('miyazaki', ugettext_lazy('Miyazaki'),), | ||||||
|     ('kagoshima', gettext_lazy('Kagoshima'),), |     ('kagoshima', ugettext_lazy('Kagoshima'),), | ||||||
|     ('okinawa', gettext_lazy('Okinawa'),), |     ('okinawa', ugettext_lazy('Okinawa'),), | ||||||
| ) | ) | ||||||
|   | |||||||
| @@ -6,14 +6,14 @@ Norwegian-specific Form helpers | |||||||
| import re, datetime | import re, datetime | ||||||
| from django.newforms import ValidationError | from django.newforms import ValidationError | ||||||
| from django.newforms.fields import Field, RegexField, Select, EMPTY_VALUES | from django.newforms.fields import Field, RegexField, Select, EMPTY_VALUES | ||||||
| from django.utils.translation import gettext | from django.utils.translation import ugettext | ||||||
|  |  | ||||||
| class NOZipCodeField(RegexField): | class NOZipCodeField(RegexField): | ||||||
|     def __init__(self, *args, **kwargs): |     def __init__(self, *args, **kwargs): | ||||||
|         super(NOZipCodeField, self).__init__(r'^\d{4}$', |         super(NOZipCodeField, self).__init__(r'^\d{4}$', | ||||||
|             max_length=None, min_length=None, |             max_length=None, min_length=None, | ||||||
|             error_message=gettext(u'Enter a zip code in the format XXXX.'), |             error_message=ugettext('Enter a zip code in the format XXXX.'), | ||||||
|             *args, **kwargs) |                     *args, **kwargs) | ||||||
|  |  | ||||||
| class NOMunicipalitySelect(Select): | class NOMunicipalitySelect(Select): | ||||||
|     """ |     """ | ||||||
| @@ -33,7 +33,7 @@ class NOSocialSecurityNumber(Field): | |||||||
|         if value in EMPTY_VALUES: |         if value in EMPTY_VALUES: | ||||||
|             return u'' |             return u'' | ||||||
|  |  | ||||||
|         msg = gettext(u'Enter a valid Norwegian social security number.') |         msg = ugettext(u'Enter a valid Norwegian social security number.') | ||||||
|         if not re.match(r'^\d{11}$', value): |         if not re.match(r'^\d{11}$', value): | ||||||
|             raise ValidationError(msg) |             raise ValidationError(msg) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # -*- coding: iso-8859-1 -*- | # -*- coding: utf-8 -*- | ||||||
| """ | """ | ||||||
| An alphabetical list of Norwegian municipalities (fylker) fro use as `choices` | An alphabetical list of Norwegian municipalities (fylker) fro use as `choices` | ||||||
| in a formfield. | in a formfield. | ||||||
| @@ -15,18 +15,18 @@ MUNICIPALITY_CHOICES = ( | |||||||
|     ('hedmark', u'Hedmark'), |     ('hedmark', u'Hedmark'), | ||||||
|     ('hordaland', u'Hordaland'), |     ('hordaland', u'Hordaland'), | ||||||
|     ('janmayen', u'Jan Mayen'), |     ('janmayen', u'Jan Mayen'), | ||||||
|     ('moreogromsdal', u'M<EFBFBD>re og Romsdal'), |     ('moreogromsdal', u'Møre og Romsdal'), | ||||||
|     ('nordtrondelag', u'Nord-Tr<EFBFBD>ndelag'), |     ('nordtrondelag', u'Nord-Trøndelag'), | ||||||
|     ('nordland', u'Nordland'), |     ('nordland', u'Nordland'), | ||||||
|     ('oppland', u'Oppland'), |     ('oppland', u'Oppland'), | ||||||
|     ('oslo', u'Oslo'), |     ('oslo', u'Oslo'), | ||||||
|     ('rogaland', u'Rogaland'), |     ('rogaland', u'Rogaland'), | ||||||
|     ('sognogfjordane', u'Sogn og Fjordane'), |     ('sognogfjordane', u'Sogn og Fjordane'), | ||||||
|     ('svalbard', u'Svalbard'), |     ('svalbard', u'Svalbard'), | ||||||
|     ('sortrondelag', u'S<EFBFBD>r-Tr<EFBFBD>ndelag'), |     ('sortrondelag', u'Sør-Trøndelag'), | ||||||
|     ('telemark', u'Telemark'), |     ('telemark', u'Telemark'), | ||||||
|     ('troms', u'Troms'), |     ('troms', u'Troms'), | ||||||
|     ('vestagder', u'Vest-Agder'), |     ('vestagder', u'Vest-Agder'), | ||||||
|     ('vestfold', u'Vestfold'), |     ('vestfold', u'Vestfold'), | ||||||
|     ('ostfold', u'<EFBFBD>stfold') |     ('ostfold', u'Østfold') | ||||||
| ) | ) | ||||||
|   | |||||||
| @@ -3,7 +3,7 @@ UK-specific Form helpers | |||||||
| """ | """ | ||||||
|  |  | ||||||
| from django.newforms.fields import RegexField | from django.newforms.fields import RegexField | ||||||
| from django.utils.translation import gettext | from django.utils.translation import ugettext | ||||||
|  |  | ||||||
| class UKPostcodeField(RegexField): | class UKPostcodeField(RegexField): | ||||||
|     """ |     """ | ||||||
| @@ -15,5 +15,5 @@ class UKPostcodeField(RegexField): | |||||||
|     def __init__(self, *args, **kwargs): |     def __init__(self, *args, **kwargs): | ||||||
|         super(UKPostcodeField, self).__init__(r'^(GIR 0AA|[A-PR-UWYZ]([0-9]{1,2}|([A-HIK-Y][0-9](|[0-9]|[ABEHMNPRVWXY]))|[0-9][A-HJKSTUW]) [0-9][ABD-HJLNP-UW-Z]{2})$', |         super(UKPostcodeField, self).__init__(r'^(GIR 0AA|[A-PR-UWYZ]([0-9]{1,2}|([A-HIK-Y][0-9](|[0-9]|[ABEHMNPRVWXY]))|[0-9][A-HJKSTUW]) [0-9][ABD-HJLNP-UW-Z]{2})$', | ||||||
|             max_length=None, min_length=None, |             max_length=None, min_length=None, | ||||||
|             error_message=gettext(u'Enter a postcode. A space is required between the two postcode parts.'), |             error_message=ugettext(u'Enter a postcode. A space is required between the two postcode parts.'), | ||||||
|             *args, **kwargs) |             *args, **kwargs) | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ USA-specific Form helpers | |||||||
| from django.newforms import ValidationError | from django.newforms import ValidationError | ||||||
| from django.newforms.fields import Field, RegexField, Select, EMPTY_VALUES | from django.newforms.fields import Field, RegexField, Select, EMPTY_VALUES | ||||||
| from django.utils.encoding import smart_unicode | from django.utils.encoding import smart_unicode | ||||||
| from django.utils.translation import gettext | from django.utils.translation import ugettext | ||||||
| import re | import re | ||||||
|  |  | ||||||
| phone_digits_re = re.compile(r'^(?:1-?)?(\d{3})[-\.]?(\d{3})[-\.]?(\d{4})$') | phone_digits_re = re.compile(r'^(?:1-?)?(\d{3})[-\.]?(\d{3})[-\.]?(\d{4})$') | ||||||
| @@ -15,8 +15,8 @@ class USZipCodeField(RegexField): | |||||||
|     def __init__(self, *args, **kwargs): |     def __init__(self, *args, **kwargs): | ||||||
|         super(USZipCodeField, self).__init__(r'^\d{5}(?:-\d{4})?$', |         super(USZipCodeField, self).__init__(r'^\d{5}(?:-\d{4})?$', | ||||||
|             max_length=None, min_length=None, |             max_length=None, min_length=None, | ||||||
|             error_message=gettext(u'Enter a zip code in the format XXXXX or XXXXX-XXXX.'), |             error_message=ugettext('Enter a zip code in the format XXXXX or XXXXX-XXXX.'), | ||||||
|             *args, **kwargs) |                     *args, **kwargs) | ||||||
|  |  | ||||||
| class USPhoneNumberField(Field): | class USPhoneNumberField(Field): | ||||||
|     def clean(self, value): |     def clean(self, value): | ||||||
| @@ -38,17 +38,17 @@ class USSocialSecurityNumberField(Field): | |||||||
|         * Conforms to the XXX-XX-XXXX format. |         * Conforms to the XXX-XX-XXXX format. | ||||||
|         * No group consists entirely of zeroes. |         * No group consists entirely of zeroes. | ||||||
|         * The leading group is not "666" (block "666" will never be allocated). |         * The leading group is not "666" (block "666" will never be allocated). | ||||||
|         * The number is not in the promotional block 987-65-4320 through 987-65-4329, |         * The number is not in the promotional block 987-65-4320 through | ||||||
|           which are permanently invalid. |           987-65-4329, which are permanently invalid. | ||||||
|         * The number is not one known to be invalid due to otherwise widespread |         * The number is not one known to be invalid due to otherwise widespread | ||||||
|           promotional use or distribution (e.g., the Woolworth's number or the 1962 |           promotional use or distribution (e.g., the Woolworth's number or the | ||||||
|           promotional number). |           1962 promotional number). | ||||||
|     """ |     """ | ||||||
|     def clean(self, value): |     def clean(self, value): | ||||||
|         super(USSocialSecurityNumberField, self).clean(value) |         super(USSocialSecurityNumberField, self).clean(value) | ||||||
|         if value in EMPTY_VALUES: |         if value in EMPTY_VALUES: | ||||||
|             return u'' |             return u'' | ||||||
|         msg = gettext(u'Enter a valid U.S. Social Security number in XXX-XX-XXXX format.') |         msg = ugettext('Enter a valid U.S. Social Security number in XXX-XX-XXXX format.') | ||||||
|         match = re.match(ssn_re, value) |         match = re.match(ssn_re, value) | ||||||
|         if not match: |         if not match: | ||||||
|             raise ValidationError(msg) |             raise ValidationError(msg) | ||||||
| @@ -75,7 +75,7 @@ class USStateField(Field): | |||||||
|     abbreviation for the given state. |     abbreviation for the given state. | ||||||
|     """ |     """ | ||||||
|     def clean(self, value): |     def clean(self, value): | ||||||
|         from us_states import STATES_NORMALIZED # relative import |         from us_states import STATES_NORMALIZED | ||||||
|         super(USStateField, self).clean(value) |         super(USStateField, self).clean(value) | ||||||
|         if value in EMPTY_VALUES: |         if value in EMPTY_VALUES: | ||||||
|             return u'' |             return u'' | ||||||
| @@ -95,5 +95,5 @@ class USStateSelect(Select): | |||||||
|     A Select widget that uses a list of U.S. states/territories as its choices. |     A Select widget that uses a list of U.S. states/territories as its choices. | ||||||
|     """ |     """ | ||||||
|     def __init__(self, attrs=None): |     def __init__(self, attrs=None): | ||||||
|         from us_states import STATE_CHOICES # relative import |         from us_states import STATE_CHOICES | ||||||
|         super(USStateSelect, self).__init__(attrs, choices=STATE_CHOICES) |         super(USStateSelect, self).__init__(attrs, choices=STATE_CHOICES) | ||||||
|   | |||||||
| @@ -16,6 +16,7 @@ silently fail and return the un-marked-up text. | |||||||
|  |  | ||||||
| from django import template | from django import template | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
|  | from django.utils.encoding import smart_str, force_unicode | ||||||
|  |  | ||||||
| register = template.Library() | register = template.Library() | ||||||
|  |  | ||||||
| @@ -25,9 +26,9 @@ def textile(value): | |||||||
|     except ImportError: |     except ImportError: | ||||||
|         if settings.DEBUG: |         if settings.DEBUG: | ||||||
|             raise template.TemplateSyntaxError, "Error in {% textile %} filter: The Python textile library isn't installed." |             raise template.TemplateSyntaxError, "Error in {% textile %} filter: The Python textile library isn't installed." | ||||||
|         return value |         return force_unicode(value) | ||||||
|     else: |     else: | ||||||
|         return textile.textile(value, encoding=settings.DEFAULT_CHARSET, output=settings.DEFAULT_CHARSET) |         return force_unicode(textile.textile(smart_str(value), encoding='utf-8', output='utf-8')) | ||||||
|  |  | ||||||
| def markdown(value): | def markdown(value): | ||||||
|     try: |     try: | ||||||
| @@ -35,9 +36,9 @@ def markdown(value): | |||||||
|     except ImportError: |     except ImportError: | ||||||
|         if settings.DEBUG: |         if settings.DEBUG: | ||||||
|             raise template.TemplateSyntaxError, "Error in {% markdown %} filter: The Python markdown library isn't installed." |             raise template.TemplateSyntaxError, "Error in {% markdown %} filter: The Python markdown library isn't installed." | ||||||
|         return value |         return force_unicode(value) | ||||||
|     else: |     else: | ||||||
|         return markdown.markdown(value) |         return force_unicode(markdown.markdown(smart_str(value))) | ||||||
|  |  | ||||||
| def restructuredtext(value): | def restructuredtext(value): | ||||||
|     try: |     try: | ||||||
| @@ -45,11 +46,11 @@ def restructuredtext(value): | |||||||
|     except ImportError: |     except ImportError: | ||||||
|         if settings.DEBUG: |         if settings.DEBUG: | ||||||
|             raise template.TemplateSyntaxError, "Error in {% restructuredtext %} filter: The Python docutils library isn't installed." |             raise template.TemplateSyntaxError, "Error in {% restructuredtext %} filter: The Python docutils library isn't installed." | ||||||
|         return value |         return force_unicode(value) | ||||||
|     else: |     else: | ||||||
|         docutils_settings = getattr(settings, "RESTRUCTUREDTEXT_FILTER_SETTINGS", {}) |         docutils_settings = getattr(settings, "RESTRUCTUREDTEXT_FILTER_SETTINGS", {}) | ||||||
|         parts = publish_parts(source=value, writer_name="html4css1", settings_overrides=docutils_settings) |         parts = publish_parts(source=smart_str(value), writer_name="html4css1", settings_overrides=docutils_settings) | ||||||
|         return parts["fragment"] |         return force_unicode(parts["fragment"]) | ||||||
|  |  | ||||||
| register.filter(textile) | register.filter(textile) | ||||||
| register.filter(markdown) | register.filter(markdown) | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| from django.db import models | from django.db import models | ||||||
| from django.contrib.sites.models import Site | from django.contrib.sites.models import Site | ||||||
| from django.utils.translation import gettext_lazy as _ | from django.utils.translation import ugettext_lazy as _ | ||||||
|  |  | ||||||
| class Redirect(models.Model): | class Redirect(models.Model): | ||||||
|     site = models.ForeignKey(Site, radio_admin=models.VERTICAL) |     site = models.ForeignKey(Site, radio_admin=models.VERTICAL) | ||||||
| @@ -16,7 +16,7 @@ class Redirect(models.Model): | |||||||
|         unique_together=(('site', 'old_path'),) |         unique_together=(('site', 'old_path'),) | ||||||
|         ordering = ('old_path',) |         ordering = ('old_path',) | ||||||
|  |  | ||||||
|     def __str__(self): |     def __unicode__(self): | ||||||
|         return "%s ---> %s" % (self.old_path, self.new_path) |         return "%s ---> %s" % (self.old_path, self.new_path) | ||||||
|  |  | ||||||
| # Register the admin options for these models. | # Register the admin options for these models. | ||||||
| @@ -30,3 +30,4 @@ class RedirectAdmin(admin.ModelAdmin): | |||||||
|     search_fields = ('old_path', 'new_path') |     search_fields = ('old_path', 'new_path') | ||||||
|  |  | ||||||
| admin.site.register(Redirect, RedirectAdmin) | admin.site.register(Redirect, RedirectAdmin) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| import base64, md5, random, sys, datetime, os, time | import base64, md5, random, sys, datetime, os, time | ||||||
| import cPickle as pickle | import cPickle as pickle | ||||||
| from django.db import models | from django.db import models | ||||||
| from django.utils.translation import gettext_lazy as _ | from django.utils.translation import ugettext_lazy as _ | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
|  |  | ||||||
| class SessionManager(models.Manager): | class SessionManager(models.Manager): | ||||||
|   | |||||||
| @@ -2,6 +2,7 @@ from django.http import HttpResponse, Http404 | |||||||
| from django.template import loader | from django.template import loader | ||||||
| from django.contrib.sites.models import Site | from django.contrib.sites.models import Site | ||||||
| from django.core import urlresolvers | from django.core import urlresolvers | ||||||
|  | from django.utils.encoding import smart_str | ||||||
|  |  | ||||||
| def index(request, sitemaps): | def index(request, sitemaps): | ||||||
|     current_site = Site.objects.get_current() |     current_site = Site.objects.get_current() | ||||||
| @@ -26,5 +27,5 @@ def sitemap(request, sitemaps, section=None): | |||||||
|             urls.extend(site().get_urls()) |             urls.extend(site().get_urls()) | ||||||
|         else: |         else: | ||||||
|             urls.extend(site.get_urls()) |             urls.extend(site.get_urls()) | ||||||
|     xml = loader.render_to_string('sitemap.xml', {'urlset': urls}) |     xml = smart_str(loader.render_to_string('sitemap.xml', {'urlset': urls})) | ||||||
|     return HttpResponse(xml, mimetype='application/xml') |     return HttpResponse(xml, mimetype='application/xml') | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| from django.db import models | from django.db import models | ||||||
| from django.utils.translation import gettext_lazy as _ | from django.utils.translation import ugettext_lazy as _ | ||||||
|  |  | ||||||
| class SiteManager(models.Manager): | class SiteManager(models.Manager): | ||||||
|     def get_current(self): |     def get_current(self): | ||||||
| @@ -17,7 +17,7 @@ class Site(models.Model): | |||||||
|         verbose_name_plural = _('sites') |         verbose_name_plural = _('sites') | ||||||
|         ordering = ('domain',) |         ordering = ('domain',) | ||||||
|  |  | ||||||
|     def __str__(self): |     def __unicode__(self): | ||||||
|         return self.domain |         return self.domain | ||||||
|  |  | ||||||
| # Register the admin options for these models. | # Register the admin options for these models. | ||||||
|   | |||||||
| @@ -2,10 +2,13 @@ from django.core.exceptions import ImproperlyConfigured, ObjectDoesNotExist | |||||||
| from django.template import Context, loader, Template, TemplateDoesNotExist | from django.template import Context, loader, Template, TemplateDoesNotExist | ||||||
| from django.contrib.sites.models import Site | from django.contrib.sites.models import Site | ||||||
| from django.utils import feedgenerator | from django.utils import feedgenerator | ||||||
|  | from django.utils.encoding import smart_unicode | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
|  |  | ||||||
| def add_domain(domain, url): | def add_domain(domain, url): | ||||||
|     if not url.startswith('http://'): |     if not url.startswith('http://'): | ||||||
|  |         # 'url' must already be ASCII and URL-quoted, so no need for encoding | ||||||
|  |         # conversions here. | ||||||
|         url = u'http://%s%s' % (domain, url) |         url = u'http://%s%s' % (domain, url) | ||||||
|     return url |     return url | ||||||
|  |  | ||||||
| @@ -97,9 +100,9 @@ class Feed(object): | |||||||
|             enc_url = self.__get_dynamic_attr('item_enclosure_url', item) |             enc_url = self.__get_dynamic_attr('item_enclosure_url', item) | ||||||
|             if enc_url: |             if enc_url: | ||||||
|                 enc = feedgenerator.Enclosure( |                 enc = feedgenerator.Enclosure( | ||||||
|                     url = enc_url.decode('utf-8'), |                     url = smart_unicode(enc_url), | ||||||
|                     length = str(self.__get_dynamic_attr('item_enclosure_length', item)).decode('utf-8'), |                     length = smart_unicode(self.__get_dynamic_attr('item_enclosure_length', item)), | ||||||
|                     mime_type = self.__get_dynamic_attr('item_enclosure_mime_type', item).decode('utf-8'), |                     mime_type = smart_unicode(self.__get_dynamic_attr('item_enclosure_mime_type', item)) | ||||||
|                 ) |                 ) | ||||||
|             author_name = self.__get_dynamic_attr('item_author_name', item) |             author_name = self.__get_dynamic_attr('item_author_name', item) | ||||||
|             if author_name is not None: |             if author_name is not None: | ||||||
| @@ -108,9 +111,9 @@ class Feed(object): | |||||||
|             else: |             else: | ||||||
|                 author_email = author_link = None |                 author_email = author_link = None | ||||||
|             feed.add_item( |             feed.add_item( | ||||||
|                 title = title_tmp.render(Context({'obj': item, 'site': current_site})).decode('utf-8'), |                 title = title_tmp.render(Context({'obj': item, 'site': current_site})), | ||||||
|                 link = link, |                 link = link, | ||||||
|                 description = description_tmp.render(Context({'obj': item, 'site': current_site})).decode('utf-8'), |                 description = description_tmp.render(Context({'obj': item, 'site': current_site})), | ||||||
|                 unique_id = link, |                 unique_id = link, | ||||||
|                 enclosure = enc, |                 enclosure = enc, | ||||||
|                 pubdate = self.__get_dynamic_attr('item_pubdate', item), |                 pubdate = self.__get_dynamic_attr('item_pubdate', item), | ||||||
|   | |||||||
| @@ -5,8 +5,40 @@ Utility functions for generating "lorem ipsum" Latin text. | |||||||
| import random | import random | ||||||
|  |  | ||||||
| COMMON_P = 'Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum.' | COMMON_P = 'Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum.' | ||||||
| WORDS = ('exercitationem', 'perferendis', 'perspiciatis', 'laborum', 'eveniet', 'sunt', 'iure', 'nam', 'nobis', 'eum', 'cum', 'officiis', 'excepturi', 'odio', 'consectetur', 'quasi', 'aut', 'quisquam', 'vel', 'eligendi', 'itaque', 'non', 'odit', 'tempore', 'quaerat', 'dignissimos', 'facilis', 'neque', 'nihil', 'expedita', 'vitae', 'vero', 'ipsum', 'nisi', 'animi', 'cumque', 'pariatur', 'velit', 'modi', 'natus', 'iusto', 'eaque', 'sequi', 'illo', 'sed', 'ex', 'et', 'voluptatibus', 'tempora', 'veritatis', 'ratione', 'assumenda', 'incidunt', 'nostrum', 'placeat', 'aliquid', 'fuga', 'provident', 'praesentium', 'rem', 'necessitatibus', 'suscipit', 'adipisci', 'quidem', 'possimus', 'voluptas', 'debitis', 'sint', 'accusantium', 'unde', 'sapiente', 'voluptate', 'qui', 'aspernatur', 'laudantium', 'soluta', 'amet', 'quo', 'aliquam', 'saepe', 'culpa', 'libero', 'ipsa', 'dicta', 'reiciendis', 'nesciunt', 'doloribus', 'autem', 'impedit', 'minima', 'maiores', 'repudiandae', 'ipsam', 'obcaecati', 'ullam', 'enim', 'totam', 'delectus', 'ducimus', 'quis', 'voluptates', 'dolores', 'molestiae', 'harum', 'dolorem', 'quia', 'voluptatem', 'molestias', 'magni', 'distinctio', 'omnis', 'illum', 'dolorum', 'voluptatum', 'ea', 'quas', 'quam', 'corporis', 'quae', 'blanditiis', 'atque', 'deserunt', 'laboriosam', 'earum', 'consequuntur', 'hic', 'cupiditate', 'quibusdam', 'accusamus', 'ut', 'rerum', 'error', 'minus', 'eius', 'ab', 'ad', 'nemo', 'fugit', 'officia', 'at', 'in', 'id', 'quos', 'reprehenderit', 'numquam', 'iste', 'fugiat', 'sit', 'inventore', 'beatae', 'repellendus', 'magnam', 'recusandae', 'quod', 'explicabo', 'doloremque', 'aperiam', 'consequatur', 'asperiores', 'commodi', 'optio', 'dolor', 'labore', 'temporibus', 'repellat', 'veniam', 'architecto', 'est', 'esse', 'mollitia', 'nulla', 'a', 'similique', 'eos', 'alias', 'dolore', 'tenetur', 'deleniti', 'porro', 'facere', 'maxime', 'corrupti') |  | ||||||
| COMMON_WORDS = ('lorem', 'ipsum', 'dolor', 'sit', 'amet', 'consectetur', 'adipisicing', 'elit', 'sed', 'do', 'eiusmod', 'tempor', 'incididunt', 'ut', 'labore', 'et', 'dolore', 'magna', 'aliqua') | WORDS = ('exercitationem', 'perferendis', 'perspiciatis', 'laborum', 'eveniet', | ||||||
|  |         'sunt', 'iure', 'nam', 'nobis', 'eum', 'cum', 'officiis', 'excepturi', | ||||||
|  |         'odio', 'consectetur', 'quasi', 'aut', 'quisquam', 'vel', 'eligendi', | ||||||
|  |         'itaque', 'non', 'odit', 'tempore', 'quaerat', 'dignissimos', | ||||||
|  |         'facilis', 'neque', 'nihil', 'expedita', 'vitae', 'vero', 'ipsum', | ||||||
|  |         'nisi', 'animi', 'cumque', 'pariatur', 'velit', 'modi', 'natus', | ||||||
|  |         'iusto', 'eaque', 'sequi', 'illo', 'sed', 'ex', 'et', 'voluptatibus', | ||||||
|  |         'tempora', 'veritatis', 'ratione', 'assumenda', 'incidunt', 'nostrum', | ||||||
|  |         'placeat', 'aliquid', 'fuga', 'provident', 'praesentium', 'rem', | ||||||
|  |         'necessitatibus', 'suscipit', 'adipisci', 'quidem', 'possimus', | ||||||
|  |         'voluptas', 'debitis', 'sint', 'accusantium', 'unde', 'sapiente', | ||||||
|  |         'voluptate', 'qui', 'aspernatur', 'laudantium', 'soluta', 'amet', | ||||||
|  |         'quo', 'aliquam', 'saepe', 'culpa', 'libero', 'ipsa', 'dicta', | ||||||
|  |         'reiciendis', 'nesciunt', 'doloribus', 'autem', 'impedit', 'minima', | ||||||
|  |         'maiores', 'repudiandae', 'ipsam', 'obcaecati', 'ullam', 'enim', | ||||||
|  |         'totam', 'delectus', 'ducimus', 'quis', 'voluptates', 'dolores', | ||||||
|  |         'molestiae', 'harum', 'dolorem', 'quia', 'voluptatem', 'molestias', | ||||||
|  |         'magni', 'distinctio', 'omnis', 'illum', 'dolorum', 'voluptatum', 'ea', | ||||||
|  |         'quas', 'quam', 'corporis', 'quae', 'blanditiis', 'atque', 'deserunt', | ||||||
|  |         'laboriosam', 'earum', 'consequuntur', 'hic', 'cupiditate', | ||||||
|  |         'quibusdam', 'accusamus', 'ut', 'rerum', 'error', 'minus', 'eius', | ||||||
|  |         'ab', 'ad', 'nemo', 'fugit', 'officia', 'at', 'in', 'id', 'quos', | ||||||
|  |         'reprehenderit', 'numquam', 'iste', 'fugiat', 'sit', 'inventore', | ||||||
|  |         'beatae', 'repellendus', 'magnam', 'recusandae', 'quod', 'explicabo', | ||||||
|  |         'doloremque', 'aperiam', 'consequatur', 'asperiores', 'commodi', | ||||||
|  |         'optio', 'dolor', 'labore', 'temporibus', 'repellat', 'veniam', | ||||||
|  |         'architecto', 'est', 'esse', 'mollitia', 'nulla', 'a', 'similique', | ||||||
|  |         'eos', 'alias', 'dolore', 'tenetur', 'deleniti', 'porro', 'facere', | ||||||
|  |         'maxime', 'corrupti') | ||||||
|  |  | ||||||
|  | COMMON_WORDS = ('lorem', 'ipsum', 'dolor', 'sit', 'amet', 'consectetur', | ||||||
|  |         'adipisicing', 'elit', 'sed', 'do', 'eiusmod', 'tempor', 'incididunt', | ||||||
|  |         'ut', 'labore', 'et', 'dolore', 'magna', 'aliqua') | ||||||
|  |  | ||||||
| def sentence(): | def sentence(): | ||||||
|     """ |     """ | ||||||
| @@ -17,10 +49,10 @@ def sentence(): | |||||||
|     """ |     """ | ||||||
|     # Determine the number of comma-separated sections and number of words in |     # Determine the number of comma-separated sections and number of words in | ||||||
|     # each section for this sentence. |     # each section for this sentence. | ||||||
|     sections = [' '.join(random.sample(WORDS, random.randint(3, 12))) for i in range(random.randint(1, 5))] |     sections = [u' '.join(random.sample(WORDS, random.randint(3, 12))) for i in range(random.randint(1, 5))] | ||||||
|     s = ', '.join(sections) |     s = u', '.join(sections) | ||||||
|     # Convert to sentence case and add end punctuation. |     # Convert to sentence case and add end punctuation. | ||||||
|     return '%s%s%s' % (s[0].upper(), s[1:], random.choice('?.')) |     return u'%s%s%s' % (s[0].upper(), s[1:], random.choice('?.')) | ||||||
|  |  | ||||||
| def paragraph(): | def paragraph(): | ||||||
|     """ |     """ | ||||||
| @@ -28,7 +60,7 @@ def paragraph(): | |||||||
|  |  | ||||||
|     The paragraph consists of between 1 and 4 sentences, inclusive. |     The paragraph consists of between 1 and 4 sentences, inclusive. | ||||||
|     """ |     """ | ||||||
|     return ' '.join([sentence() for i in range(random.randint(1, 4))]) |     return u' '.join([sentence() for i in range(random.randint(1, 4))]) | ||||||
|  |  | ||||||
| def paragraphs(count, common=True): | def paragraphs(count, common=True): | ||||||
|     """ |     """ | ||||||
| @@ -66,4 +98,4 @@ def words(count, common=True): | |||||||
|             word_list += random.sample(WORDS, c) |             word_list += random.sample(WORDS, c) | ||||||
|     else: |     else: | ||||||
|         word_list = word_list[:count] |         word_list = word_list[:count] | ||||||
|     return ' '.join(word_list) |     return u' '.join(word_list) | ||||||
|   | |||||||
| @@ -18,7 +18,7 @@ class LoremNode(template.Node): | |||||||
|             paras = paragraphs(count, common=self.common) |             paras = paragraphs(count, common=self.common) | ||||||
|         if self.method == 'p': |         if self.method == 'p': | ||||||
|             paras = ['<p>%s</p>' % p for p in paras] |             paras = ['<p>%s</p>' % p for p in paras] | ||||||
|         return '\n\n'.join(paras) |         return u'\n\n'.join(paras) | ||||||
|  |  | ||||||
| #@register.tag | #@register.tag | ||||||
| def lorem(parser, token): | def lorem(parser, token): | ||||||
|   | |||||||
| @@ -2,7 +2,7 @@ | |||||||
|  |  | ||||||
| r""" | r""" | ||||||
| >>> words(7) | >>> words(7) | ||||||
| 'lorem ipsum dolor sit amet consectetur adipisicing' | u'lorem ipsum dolor sit amet consectetur adipisicing' | ||||||
|  |  | ||||||
| >>> paragraphs(1) | >>> paragraphs(1) | ||||||
| ['Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum.'] | ['Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum.'] | ||||||
|   | |||||||
| @@ -49,7 +49,7 @@ class ModPythonRequest(http.HttpRequest): | |||||||
|         if 'content-type' in self._req.headers_in and self._req.headers_in['content-type'].startswith('multipart'): |         if 'content-type' in self._req.headers_in and self._req.headers_in['content-type'].startswith('multipart'): | ||||||
|             self._post, self._files = http.parse_file_upload(self._req.headers_in, self.raw_post_data) |             self._post, self._files = http.parse_file_upload(self._req.headers_in, self.raw_post_data) | ||||||
|         else: |         else: | ||||||
|             self._post, self._files = http.QueryDict(self.raw_post_data), datastructures.MultiValueDict() |             self._post, self._files = http.QueryDict(self.raw_post_data, encoding=self._encoding), datastructures.MultiValueDict() | ||||||
|  |  | ||||||
|     def _get_request(self): |     def _get_request(self): | ||||||
|         if not hasattr(self, '_request'): |         if not hasattr(self, '_request'): | ||||||
| @@ -58,7 +58,7 @@ class ModPythonRequest(http.HttpRequest): | |||||||
|  |  | ||||||
|     def _get_get(self): |     def _get_get(self): | ||||||
|         if not hasattr(self, '_get'): |         if not hasattr(self, '_get'): | ||||||
|             self._get = http.QueryDict(self._req.args) |             self._get = http.QueryDict(self._req.args, encoding=self._encoding) | ||||||
|         return self._get |         return self._get | ||||||
|  |  | ||||||
|     def _set_get(self, get): |     def _set_get(self, get): | ||||||
| @@ -160,7 +160,7 @@ class ModPythonHandler(BaseHandler): | |||||||
|         req.content_type = response['Content-Type'] |         req.content_type = response['Content-Type'] | ||||||
|         for key, value in response.headers.items(): |         for key, value in response.headers.items(): | ||||||
|             if key != 'Content-Type': |             if key != 'Content-Type': | ||||||
|                 req.headers_out[key] = value |                 req.headers_out[str(key)] = str(value) | ||||||
|         for c in response.cookies.values(): |         for c in response.cookies.values(): | ||||||
|             req.headers_out.add('Set-Cookie', c.output(header='')) |             req.headers_out.add('Set-Cookie', c.output(header='')) | ||||||
|         req.status = response.status_code |         req.status = response.status_code | ||||||
|   | |||||||
| @@ -113,9 +113,9 @@ class WSGIRequest(http.HttpRequest): | |||||||
|                 header_dict['Content-Type'] = self.environ.get('CONTENT_TYPE', '') |                 header_dict['Content-Type'] = self.environ.get('CONTENT_TYPE', '') | ||||||
|                 self._post, self._files = http.parse_file_upload(header_dict, self.raw_post_data) |                 self._post, self._files = http.parse_file_upload(header_dict, self.raw_post_data) | ||||||
|             else: |             else: | ||||||
|                 self._post, self._files = http.QueryDict(self.raw_post_data), datastructures.MultiValueDict() |                 self._post, self._files = http.QueryDict(self.raw_post_data, encoding=self._encoding), datastructures.MultiValueDict() | ||||||
|         else: |         else: | ||||||
|             self._post, self._files = http.QueryDict(''), datastructures.MultiValueDict() |             self._post, self._files = http.QueryDict('', encoding=self._encoding), datastructures.MultiValueDict() | ||||||
|  |  | ||||||
|     def _get_request(self): |     def _get_request(self): | ||||||
|         if not hasattr(self, '_request'): |         if not hasattr(self, '_request'): | ||||||
| @@ -125,7 +125,7 @@ class WSGIRequest(http.HttpRequest): | |||||||
|     def _get_get(self): |     def _get_get(self): | ||||||
|         if not hasattr(self, '_get'): |         if not hasattr(self, '_get'): | ||||||
|             # The WSGI spec says 'QUERY_STRING' may be absent. |             # The WSGI spec says 'QUERY_STRING' may be absent. | ||||||
|             self._get = http.QueryDict(self.environ.get('QUERY_STRING', '')) |             self._get = http.QueryDict(self.environ.get('QUERY_STRING', ''), encoding=self._encoding) | ||||||
|         return self._get |         return self._get | ||||||
|  |  | ||||||
|     def _set_get(self, get): |     def _set_get(self, get): | ||||||
| @@ -200,8 +200,8 @@ class WSGIHandler(BaseHandler): | |||||||
|         except KeyError: |         except KeyError: | ||||||
|             status_text = 'UNKNOWN STATUS CODE' |             status_text = 'UNKNOWN STATUS CODE' | ||||||
|         status = '%s %s' % (response.status_code, status_text) |         status = '%s %s' % (response.status_code, status_text) | ||||||
|         response_headers = response.headers.items() |         response_headers = [(str(k), str(v)) for k, v in response.headers.items()] | ||||||
|         for c in response.cookies.values(): |         for c in response.cookies.values(): | ||||||
|             response_headers.append(('Set-Cookie', c.output(header=''))) |             response_headers.append(('Set-Cookie', str(c.output(header='')))) | ||||||
|         start_response(status, response_headers) |         start_response(status, response_headers) | ||||||
|         return response |         return response | ||||||
|   | |||||||
| @@ -3,12 +3,13 @@ Tools for sending email. | |||||||
| """ | """ | ||||||
|  |  | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
|  | from django.utils.encoding import smart_str, force_unicode | ||||||
| from email import Charset, Encoders | from email import Charset, Encoders | ||||||
| from email.MIMEText import MIMEText | from email.MIMEText import MIMEText | ||||||
| from email.MIMEMultipart import MIMEMultipart | from email.MIMEMultipart import MIMEMultipart | ||||||
| from email.MIMEBase import MIMEBase | from email.MIMEBase import MIMEBase | ||||||
| from email.Header import Header | from email.Header import Header | ||||||
| from email.Utils import formatdate | from email.Utils import formatdate, parseaddr, formataddr | ||||||
| import mimetypes | import mimetypes | ||||||
| import os | import os | ||||||
| import smtplib | import smtplib | ||||||
| @@ -67,8 +68,18 @@ class SafeMIMEText(MIMEText): | |||||||
|         "Forbids multi-line headers, to prevent header injection." |         "Forbids multi-line headers, to prevent header injection." | ||||||
|         if '\n' in val or '\r' in val: |         if '\n' in val or '\r' in val: | ||||||
|             raise BadHeaderError, "Header values can't contain newlines (got %r for header %r)" % (val, name) |             raise BadHeaderError, "Header values can't contain newlines (got %r for header %r)" % (val, name) | ||||||
|         if name == "Subject": |         try: | ||||||
|             val = Header(val, settings.DEFAULT_CHARSET) |             val = str(force_unicode(val)) | ||||||
|  |         except UnicodeEncodeError: | ||||||
|  |             if name.lower() in ('to', 'from', 'cc'): | ||||||
|  |                 result = [] | ||||||
|  |                 for item in val.split(', '): | ||||||
|  |                     nm, addr = parseaddr(item) | ||||||
|  |                     nm = str(Header(nm, settings.DEFAULT_CHARSET)) | ||||||
|  |                     result.append(formataddr((nm, str(addr)))) | ||||||
|  |                 val = ', '.join(result) | ||||||
|  |             else: | ||||||
|  |                 val = Header(force_unicode(val), settings.DEFAULT_CHARSET) | ||||||
|         MIMEText.__setitem__(self, name, val) |         MIMEText.__setitem__(self, name, val) | ||||||
|  |  | ||||||
| class SafeMIMEMultipart(MIMEMultipart): | class SafeMIMEMultipart(MIMEMultipart): | ||||||
| @@ -76,8 +87,18 @@ class SafeMIMEMultipart(MIMEMultipart): | |||||||
|         "Forbids multi-line headers, to prevent header injection." |         "Forbids multi-line headers, to prevent header injection." | ||||||
|         if '\n' in val or '\r' in val: |         if '\n' in val or '\r' in val: | ||||||
|             raise BadHeaderError, "Header values can't contain newlines (got %r for header %r)" % (val, name) |             raise BadHeaderError, "Header values can't contain newlines (got %r for header %r)" % (val, name) | ||||||
|         if name == "Subject": |         try: | ||||||
|             val = Header(val, settings.DEFAULT_CHARSET) |             val = str(force_unicode(val)) | ||||||
|  |         except UnicodeEncodeError: | ||||||
|  |             if name.lower() in ('to', 'from', 'cc'): | ||||||
|  |                 result = [] | ||||||
|  |                 for item in val.split(', '): | ||||||
|  |                     nm, addr = parseaddr(item) | ||||||
|  |                     nm = str(Header(nm, settings.DEFAULT_CHARSET)) | ||||||
|  |                     result.append(formataddr((nm, str(addr)))) | ||||||
|  |                 val = ', '.join(result) | ||||||
|  |             else: | ||||||
|  |                 val = Header(force_unicode(val), settings.DEFAULT_CHARSET) | ||||||
|         MIMEMultipart.__setitem__(self, name, val) |         MIMEMultipart.__setitem__(self, name, val) | ||||||
|  |  | ||||||
| class SMTPConnection(object): | class SMTPConnection(object): | ||||||
| @@ -176,6 +197,14 @@ class EmailMessage(object): | |||||||
|  |  | ||||||
|     def __init__(self, subject='', body='', from_email=None, to=None, bcc=None, |     def __init__(self, subject='', body='', from_email=None, to=None, bcc=None, | ||||||
|             connection=None, attachments=None, headers=None): |             connection=None, attachments=None, headers=None): | ||||||
|  |         """ | ||||||
|  |         Initialise a single email message (which can be sent to multiple | ||||||
|  |         recipients). | ||||||
|  |  | ||||||
|  |         All strings used to create the message can be unicode strings (or UTF-8 | ||||||
|  |         bytestrings). The SafeMIMEText class will handle any necessary encoding | ||||||
|  |         conversions. | ||||||
|  |         """ | ||||||
|         self.to = to or [] |         self.to = to or [] | ||||||
|         self.bcc = bcc or [] |         self.bcc = bcc or [] | ||||||
|         self.from_email = from_email or settings.DEFAULT_FROM_EMAIL |         self.from_email = from_email or settings.DEFAULT_FROM_EMAIL | ||||||
| @@ -192,7 +221,7 @@ class EmailMessage(object): | |||||||
|  |  | ||||||
|     def message(self): |     def message(self): | ||||||
|         encoding = self.encoding or settings.DEFAULT_CHARSET |         encoding = self.encoding or settings.DEFAULT_CHARSET | ||||||
|         msg = SafeMIMEText(self.body, self.content_subtype, encoding) |         msg = SafeMIMEText(smart_str(self.body, settings.DEFAULT_CHARSET), self.content_subtype, encoding) | ||||||
|         if self.attachments: |         if self.attachments: | ||||||
|             body_msg = msg |             body_msg = msg | ||||||
|             msg = SafeMIMEMultipart(_subtype=self.multipart_subtype) |             msg = SafeMIMEMultipart(_subtype=self.multipart_subtype) | ||||||
| @@ -236,7 +265,7 @@ class EmailMessage(object): | |||||||
|         """ |         """ | ||||||
|         if isinstance(filename, MIMEBase): |         if isinstance(filename, MIMEBase): | ||||||
|             assert content == mimetype == None |             assert content == mimetype == None | ||||||
|             self.attachements.append(filename) |             self.attachments.append(filename) | ||||||
|         else: |         else: | ||||||
|             assert content is not None |             assert content is not None | ||||||
|             self.attachments.append((filename, content, mimetype)) |             self.attachments.append((filename, content, mimetype)) | ||||||
|   | |||||||
| @@ -427,11 +427,11 @@ def get_custom_sql_for_model(model): | |||||||
|     for sql_file in sql_files: |     for sql_file in sql_files: | ||||||
|         if os.path.exists(sql_file): |         if os.path.exists(sql_file): | ||||||
|             fp = open(sql_file, 'U') |             fp = open(sql_file, 'U') | ||||||
|             for statement in statements.split(fp.read()): |             for statement in statements.split(fp.read().decode(settings.FILE_CHARSET)): | ||||||
|                 # Remove any comments from the file |                 # Remove any comments from the file | ||||||
|                 statement = re.sub(r"--.*[\n\Z]", "", statement) |                 statement = re.sub(ur"--.*[\n\Z]", "", statement) | ||||||
|                 if statement.strip(): |                 if statement.strip(): | ||||||
|                     output.append(statement + ";") |                     output.append(statement + u";") | ||||||
|             fp.close() |             fp.close() | ||||||
|  |  | ||||||
|     return output |     return output | ||||||
|   | |||||||
| @@ -7,6 +7,7 @@ try: | |||||||
| except ImportError: | except ImportError: | ||||||
|     from StringIO import StringIO |     from StringIO import StringIO | ||||||
| from django.db import models | from django.db import models | ||||||
|  | from django.utils.encoding import smart_str, smart_unicode | ||||||
|  |  | ||||||
| class SerializationError(Exception): | class SerializationError(Exception): | ||||||
|     """Something bad happened during serialization.""" |     """Something bad happened during serialization.""" | ||||||
| @@ -59,7 +60,7 @@ class Serializer(object): | |||||||
|             value = getattr(obj, "get_%s_url" % field.name, lambda: None)() |             value = getattr(obj, "get_%s_url" % field.name, lambda: None)() | ||||||
|         else: |         else: | ||||||
|             value = field.flatten_data(follow=None, obj=obj).get(field.name, "") |             value = field.flatten_data(follow=None, obj=obj).get(field.name, "") | ||||||
|         return str(value) |         return smart_unicode(value) | ||||||
|  |  | ||||||
|     def start_serialization(self): |     def start_serialization(self): | ||||||
|         """ |         """ | ||||||
| @@ -154,7 +155,7 @@ class DeserializedObject(object): | |||||||
|         self.m2m_data = m2m_data |         self.m2m_data = m2m_data | ||||||
|  |  | ||||||
|     def __repr__(self): |     def __repr__(self): | ||||||
|         return "<DeserializedObject: %s>" % str(self.object) |         return "<DeserializedObject: %s>" % smart_str(self.object) | ||||||
|  |  | ||||||
|     def save(self, save_m2m=True): |     def save(self, save_m2m=True): | ||||||
|         self.object.save() |         self.object.save() | ||||||
|   | |||||||
| @@ -7,6 +7,7 @@ other serializers. | |||||||
| from django.conf import settings | from django.conf import settings | ||||||
| from django.core.serializers import base | from django.core.serializers import base | ||||||
| from django.db import models | from django.db import models | ||||||
|  | from django.utils.encoding import smart_unicode | ||||||
|  |  | ||||||
| class Serializer(base.Serializer): | class Serializer(base.Serializer): | ||||||
|     """ |     """ | ||||||
| @@ -25,8 +26,8 @@ class Serializer(base.Serializer): | |||||||
|  |  | ||||||
|     def end_object(self, obj): |     def end_object(self, obj): | ||||||
|         self.objects.append({ |         self.objects.append({ | ||||||
|             "model"  : str(obj._meta), |             "model"  : smart_unicode(obj._meta), | ||||||
|             "pk"     : str(obj._get_pk_val()), |             "pk"     : smart_unicode(obj._get_pk_val()), | ||||||
|             "fields" : self._current |             "fields" : self._current | ||||||
|         }) |         }) | ||||||
|         self._current = None |         self._current = None | ||||||
| @@ -67,21 +68,15 @@ def Deserializer(object_list, **options): | |||||||
|  |  | ||||||
|         # Handle each field |         # Handle each field | ||||||
|         for (field_name, field_value) in d["fields"].iteritems(): |         for (field_name, field_value) in d["fields"].iteritems(): | ||||||
|             if isinstance(field_value, unicode): |             if isinstance(field_value, str): | ||||||
|                 field_value = field_value.encode(options.get("encoding", settings.DEFAULT_CHARSET)) |                 field_value = smart_unicode(field_value, options.get("encoding", settings.DEFAULT_CHARSET), strings_only=True) | ||||||
|  |  | ||||||
|             field = Model._meta.get_field(field_name) |             field = Model._meta.get_field(field_name) | ||||||
|  |  | ||||||
|             # Handle M2M relations |             # Handle M2M relations | ||||||
|             if field.rel and isinstance(field.rel, models.ManyToManyRel): |             if field.rel and isinstance(field.rel, models.ManyToManyRel): | ||||||
|                 pks = [] |  | ||||||
|                 m2m_convert = field.rel.to._meta.pk.to_python |                 m2m_convert = field.rel.to._meta.pk.to_python | ||||||
|                 for pk in field_value: |                 m2m_data[field.name] = [m2m_convert(smart_unicode(pk)) for pk in field_value] | ||||||
|                     if isinstance(pk, unicode): |  | ||||||
|                         pks.append(m2m_convert(pk.encode(options.get("encoding", settings.DEFAULT_CHARSET)))) |  | ||||||
|                     else: |  | ||||||
|                         pks.append(m2m_convert(pk)) |  | ||||||
|                 m2m_data[field.name] = pks |  | ||||||
|  |  | ||||||
|             # Handle FK fields |             # Handle FK fields | ||||||
|             elif field.rel and isinstance(field.rel, models.ManyToOneRel): |             elif field.rel and isinstance(field.rel, models.ManyToOneRel): | ||||||
| @@ -105,5 +100,5 @@ def _get_model(model_identifier): | |||||||
|     except TypeError: |     except TypeError: | ||||||
|         Model = None |         Model = None | ||||||
|     if Model is None: |     if Model is None: | ||||||
|         raise base.DeserializationError("Invalid model identifier: '%s'" % model_identifier) |         raise base.DeserializationError(u"Invalid model identifier: '%s'" % model_identifier) | ||||||
|     return Model |     return Model | ||||||
|   | |||||||
| @@ -6,6 +6,7 @@ from django.conf import settings | |||||||
| from django.core.serializers import base | from django.core.serializers import base | ||||||
| from django.db import models | from django.db import models | ||||||
| from django.utils.xmlutils import SimplerXMLGenerator | from django.utils.xmlutils import SimplerXMLGenerator | ||||||
|  | from django.utils.encoding import smart_unicode | ||||||
| from xml.dom import pulldom | from xml.dom import pulldom | ||||||
|  |  | ||||||
| class Serializer(base.Serializer): | class Serializer(base.Serializer): | ||||||
| @@ -42,8 +43,8 @@ class Serializer(base.Serializer): | |||||||
|  |  | ||||||
|         self.indent(1) |         self.indent(1) | ||||||
|         self.xml.startElement("object", { |         self.xml.startElement("object", { | ||||||
|             "pk"    : str(obj._get_pk_val()), |             "pk"    : smart_unicode(obj._get_pk_val()), | ||||||
|             "model" : str(obj._meta), |             "model" : smart_unicode(obj._meta), | ||||||
|         }) |         }) | ||||||
|  |  | ||||||
|     def end_object(self, obj): |     def end_object(self, obj): | ||||||
| @@ -68,7 +69,7 @@ class Serializer(base.Serializer): | |||||||
|         # serializer base class). |         # serializer base class). | ||||||
|         if getattr(obj, field.name) is not None: |         if getattr(obj, field.name) is not None: | ||||||
|             value = self.get_string_value(obj, field) |             value = self.get_string_value(obj, field) | ||||||
|             self.xml.characters(str(value)) |             self.xml.characters(smart_unicode(value)) | ||||||
|         else: |         else: | ||||||
|             self.xml.addQuickElement("None") |             self.xml.addQuickElement("None") | ||||||
|  |  | ||||||
| @@ -88,7 +89,7 @@ class Serializer(base.Serializer): | |||||||
|             else: |             else: | ||||||
|                 # Related to remote object via other field |                 # Related to remote object via other field | ||||||
|                 related = getattr(related, field.rel.field_name) |                 related = getattr(related, field.rel.field_name) | ||||||
|             self.xml.characters(str(related)) |             self.xml.characters(smart_unicode(related)) | ||||||
|         else: |         else: | ||||||
|             self.xml.addQuickElement("None") |             self.xml.addQuickElement("None") | ||||||
|         self.xml.endElement("field") |         self.xml.endElement("field") | ||||||
| @@ -101,7 +102,7 @@ class Serializer(base.Serializer): | |||||||
|         """ |         """ | ||||||
|         self._start_relational_field(field) |         self._start_relational_field(field) | ||||||
|         for relobj in getattr(obj, field.name).iterator(): |         for relobj in getattr(obj, field.name).iterator(): | ||||||
|             self.xml.addQuickElement("object", attrs={"pk" : str(relobj._get_pk_val())}) |             self.xml.addQuickElement("object", attrs={"pk" : smart_unicode(relobj._get_pk_val())}) | ||||||
|         self.xml.endElement("field") |         self.xml.endElement("field") | ||||||
|  |  | ||||||
|     def _start_relational_field(self, field): |     def _start_relational_field(self, field): | ||||||
| @@ -112,7 +113,7 @@ class Serializer(base.Serializer): | |||||||
|         self.xml.startElement("field", { |         self.xml.startElement("field", { | ||||||
|             "name" : field.name, |             "name" : field.name, | ||||||
|             "rel"  : field.rel.__class__.__name__, |             "rel"  : field.rel.__class__.__name__, | ||||||
|             "to"   : str(field.rel.to._meta), |             "to"   : smart_unicode(field.rel.to._meta), | ||||||
|         }) |         }) | ||||||
|  |  | ||||||
| class Deserializer(base.Deserializer): | class Deserializer(base.Deserializer): | ||||||
| @@ -122,7 +123,6 @@ class Deserializer(base.Deserializer): | |||||||
|  |  | ||||||
|     def __init__(self, stream_or_string, **options): |     def __init__(self, stream_or_string, **options): | ||||||
|         super(Deserializer, self).__init__(stream_or_string, **options) |         super(Deserializer, self).__init__(stream_or_string, **options) | ||||||
|         self.encoding = self.options.get("encoding", settings.DEFAULT_CHARSET) |  | ||||||
|         self.event_stream = pulldom.parse(self.stream) |         self.event_stream = pulldom.parse(self.stream) | ||||||
|  |  | ||||||
|     def next(self): |     def next(self): | ||||||
| @@ -136,7 +136,8 @@ class Deserializer(base.Deserializer): | |||||||
|         """ |         """ | ||||||
|         Convert an <object> node to a DeserializedObject. |         Convert an <object> node to a DeserializedObject. | ||||||
|         """ |         """ | ||||||
|         # Look up the model using the model loading mechanism. If this fails, bail. |         # Look up the model using the model loading mechanism. If this fails, | ||||||
|  |         # bail. | ||||||
|         Model = self._get_model_from_node(node, "model") |         Model = self._get_model_from_node(node, "model") | ||||||
|  |  | ||||||
|         # Start building a data dictionary from the object.  If the node is |         # Start building a data dictionary from the object.  If the node is | ||||||
| @@ -173,7 +174,7 @@ class Deserializer(base.Deserializer): | |||||||
|                 if len(field_node.childNodes) == 1 and field_node.childNodes[0].nodeName == 'None': |                 if len(field_node.childNodes) == 1 and field_node.childNodes[0].nodeName == 'None': | ||||||
|                     value = None |                     value = None | ||||||
|                 else: |                 else: | ||||||
|                     value = field.to_python(getInnerText(field_node).strip().encode(self.encoding)) |                     value = field.to_python(getInnerText(field_node).strip()) | ||||||
|                 data[field.name] = value |                 data[field.name] = value | ||||||
|  |  | ||||||
|         # Return a DeserializedObject so that the m2m data has a place to live. |         # Return a DeserializedObject so that the m2m data has a place to live. | ||||||
| @@ -188,14 +189,14 @@ class Deserializer(base.Deserializer): | |||||||
|             return None |             return None | ||||||
|         else: |         else: | ||||||
|             return field.rel.to._meta.get_field(field.rel.field_name).to_python( |             return field.rel.to._meta.get_field(field.rel.field_name).to_python( | ||||||
|                        getInnerText(node).strip().encode(self.encoding)) |                        getInnerText(node).strip()) | ||||||
|  |  | ||||||
|     def _handle_m2m_field_node(self, node, field): |     def _handle_m2m_field_node(self, node, field): | ||||||
|         """ |         """ | ||||||
|         Handle a <field> node for a ManyToManyField |         Handle a <field> node for a ManyToManyField. | ||||||
|         """ |         """ | ||||||
|         return [field.rel.to._meta.pk.to_python( |         return [field.rel.to._meta.pk.to_python( | ||||||
|                     c.getAttribute("pk").encode(self.encoding))  |                     c.getAttribute("pk")) | ||||||
|                     for c in node.getElementsByTagName("object")] |                     for c in node.getElementsByTagName("object")] | ||||||
|  |  | ||||||
|     def _get_model_from_node(self, node, attr): |     def _get_model_from_node(self, node, attr): | ||||||
| @@ -232,4 +233,5 @@ def getInnerText(node): | |||||||
|             inner_text.extend(getInnerText(child)) |             inner_text.extend(getInnerText(child)) | ||||||
|         else: |         else: | ||||||
|            pass |            pass | ||||||
|     return "".join(inner_text) |     return u"".join(inner_text) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -9,6 +9,7 @@ a string) and returns a tuple in this format: | |||||||
|  |  | ||||||
| from django.http import Http404 | from django.http import Http404 | ||||||
| from django.core.exceptions import ImproperlyConfigured, ViewDoesNotExist | from django.core.exceptions import ImproperlyConfigured, ViewDoesNotExist | ||||||
|  | from django.utils.encoding import iri_to_uri | ||||||
| from django.utils.functional import memoize | from django.utils.functional import memoize | ||||||
| import re | import re | ||||||
|  |  | ||||||
| @@ -37,14 +38,20 @@ def get_callable(lookup_view, can_fail=False): | |||||||
|     If can_fail is True, lookup_view might be a URL pattern label, so errors |     If can_fail is True, lookup_view might be a URL pattern label, so errors | ||||||
|     during the import fail and the string is returned. |     during the import fail and the string is returned. | ||||||
|     """ |     """ | ||||||
|     if not callable(lookup_view): |     try: | ||||||
|         mod_name, func_name = get_mod_func(lookup_view) |         # Bail out early if lookup_view is not ASCII. This can't be a function. | ||||||
|         try: |         lookup_view = lookup_view.encode('ascii') | ||||||
|             if func_name != '': |  | ||||||
|                 lookup_view = getattr(__import__(mod_name, {}, {}, ['']), func_name) |         if not callable(lookup_view): | ||||||
|         except (ImportError, AttributeError): |             mod_name, func_name = get_mod_func(lookup_view) | ||||||
|             if not can_fail: |             try: | ||||||
|                 raise |                 if func_name != '': | ||||||
|  |                     lookup_view = getattr(__import__(mod_name, {}, {}, ['']), func_name) | ||||||
|  |             except (ImportError, AttributeError): | ||||||
|  |                 if not can_fail: | ||||||
|  |                     raise | ||||||
|  |     except UnicodeEncodeError: | ||||||
|  |         pass | ||||||
|     return lookup_view |     return lookup_view | ||||||
| get_callable = memoize(get_callable, _callable_cache) | get_callable = memoize(get_callable, _callable_cache) | ||||||
|  |  | ||||||
| @@ -265,7 +272,7 @@ class RegexURLResolver(object): | |||||||
|         except (ImportError, AttributeError): |         except (ImportError, AttributeError): | ||||||
|             raise NoReverseMatch |             raise NoReverseMatch | ||||||
|         if lookup_view in self.reverse_dict: |         if lookup_view in self.reverse_dict: | ||||||
|             return ''.join([reverse_helper(part.regex, *args, **kwargs) for part in self.reverse_dict[lookup_view]]) |             return u''.join([reverse_helper(part.regex, *args, **kwargs) for part in self.reverse_dict[lookup_view]]) | ||||||
|         raise NoReverseMatch |         raise NoReverseMatch | ||||||
|  |  | ||||||
|     def reverse_helper(self, lookup_view, *args, **kwargs): |     def reverse_helper(self, lookup_view, *args, **kwargs): | ||||||
| @@ -279,5 +286,5 @@ def resolve(path, urlconf=None): | |||||||
| def reverse(viewname, urlconf=None, args=None, kwargs=None): | def reverse(viewname, urlconf=None, args=None, kwargs=None): | ||||||
|     args = args or [] |     args = args or [] | ||||||
|     kwargs = kwargs or {} |     kwargs = kwargs or {} | ||||||
|     return '/' + get_resolver(urlconf).reverse(viewname, *args, **kwargs) |     return iri_to_uri(u'/' + get_resolver(urlconf).reverse(viewname, *args, **kwargs)) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -10,8 +10,9 @@ form field is required. | |||||||
|  |  | ||||||
| import urllib2 | import urllib2 | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| from django.utils.translation import gettext, gettext_lazy, ngettext | from django.utils.translation import ugettext as _, ugettext_lazy, ungettext | ||||||
| from django.utils.functional import Promise, lazy | from django.utils.functional import Promise, lazy | ||||||
|  | from django.utils.encoding import force_unicode | ||||||
| import re | import re | ||||||
|  |  | ||||||
| _datere = r'\d{4}-\d{1,2}-\d{1,2}' | _datere = r'\d{4}-\d{1,2}-\d{1,2}' | ||||||
| @@ -32,16 +33,17 @@ phone_re = re.compile(r'^[A-PR-Y0-9]{3}-[A-PR-Y0-9]{3}-[A-PR-Y0-9]{4}$', re.IGNO | |||||||
| slug_re = re.compile(r'^[-\w]+$') | slug_re = re.compile(r'^[-\w]+$') | ||||||
| url_re = re.compile(r'^https?://\S+$') | url_re = re.compile(r'^https?://\S+$') | ||||||
|  |  | ||||||
| lazy_inter = lazy(lambda a,b: str(a) % b, str) | lazy_inter = lazy(lambda a,b: force_unicode(a) % b, unicode) | ||||||
|  |  | ||||||
| class ValidationError(Exception): | class ValidationError(Exception): | ||||||
|     def __init__(self, message): |     def __init__(self, message): | ||||||
|         "ValidationError can be passed a string or a list." |         "ValidationError can be passed a string or a list." | ||||||
|         if isinstance(message, list): |         if isinstance(message, list): | ||||||
|             self.messages = message |             self.messages = [force_unicode(msg) for msg in message] | ||||||
|         else: |         else: | ||||||
|             assert isinstance(message, (basestring, Promise)), ("%s should be a string" % repr(message)) |             assert isinstance(message, (basestring, Promise)), ("%s should be a string" % repr(message)) | ||||||
|             self.messages = [message] |             self.messages = [force_unicode(message)] | ||||||
|  |  | ||||||
|     def __str__(self): |     def __str__(self): | ||||||
|         # This is needed because, without a __str__(), printing an exception |         # This is needed because, without a __str__(), printing an exception | ||||||
|         # instance would result in this: |         # instance would result in this: | ||||||
| @@ -53,39 +55,40 @@ class CriticalValidationError(Exception): | |||||||
|     def __init__(self, message): |     def __init__(self, message): | ||||||
|         "ValidationError can be passed a string or a list." |         "ValidationError can be passed a string or a list." | ||||||
|         if isinstance(message, list): |         if isinstance(message, list): | ||||||
|             self.messages = message |             self.messages = [force_unicode(msg) for msg in message] | ||||||
|         else: |         else: | ||||||
|             assert isinstance(message, (basestring, Promise)), ("'%s' should be a string" % message) |             assert isinstance(message, (basestring, Promise)), ("'%s' should be a string" % message) | ||||||
|             self.messages = [message] |             self.messages = [force_unicode(message)] | ||||||
|  |  | ||||||
|     def __str__(self): |     def __str__(self): | ||||||
|         return str(self.messages) |         return str(self.messages) | ||||||
|  |  | ||||||
| def isAlphaNumeric(field_data, all_data): | def isAlphaNumeric(field_data, all_data): | ||||||
|     if not alnum_re.search(field_data): |     if not alnum_re.search(field_data): | ||||||
|         raise ValidationError, gettext("This value must contain only letters, numbers and underscores.") |         raise ValidationError, _("This value must contain only letters, numbers and underscores.") | ||||||
|  |  | ||||||
| def isAlphaNumericURL(field_data, all_data): | def isAlphaNumericURL(field_data, all_data): | ||||||
|     if not alnumurl_re.search(field_data): |     if not alnumurl_re.search(field_data): | ||||||
|         raise ValidationError, gettext("This value must contain only letters, numbers, underscores, dashes or slashes.") |         raise ValidationError, _("This value must contain only letters, numbers, underscores, dashes or slashes.") | ||||||
|  |  | ||||||
| def isSlug(field_data, all_data): | def isSlug(field_data, all_data): | ||||||
|     if not slug_re.search(field_data): |     if not slug_re.search(field_data): | ||||||
|         raise ValidationError, gettext("This value must contain only letters, numbers, underscores or hyphens.") |         raise ValidationError, _("This value must contain only letters, numbers, underscores or hyphens.") | ||||||
|  |  | ||||||
| def isLowerCase(field_data, all_data): | def isLowerCase(field_data, all_data): | ||||||
|     if field_data.lower() != field_data: |     if field_data.lower() != field_data: | ||||||
|         raise ValidationError, gettext("Uppercase letters are not allowed here.") |         raise ValidationError, _("Uppercase letters are not allowed here.") | ||||||
|  |  | ||||||
| def isUpperCase(field_data, all_data): | def isUpperCase(field_data, all_data): | ||||||
|     if field_data.upper() != field_data: |     if field_data.upper() != field_data: | ||||||
|         raise ValidationError, gettext("Lowercase letters are not allowed here.") |         raise ValidationError, _("Lowercase letters are not allowed here.") | ||||||
|  |  | ||||||
| def isCommaSeparatedIntegerList(field_data, all_data): | def isCommaSeparatedIntegerList(field_data, all_data): | ||||||
|     for supposed_int in field_data.split(','): |     for supposed_int in field_data.split(','): | ||||||
|         try: |         try: | ||||||
|             int(supposed_int) |             int(supposed_int) | ||||||
|         except ValueError: |         except ValueError: | ||||||
|             raise ValidationError, gettext("Enter only digits separated by commas.") |             raise ValidationError, _("Enter only digits separated by commas.") | ||||||
|  |  | ||||||
| def isCommaSeparatedEmailList(field_data, all_data): | def isCommaSeparatedEmailList(field_data, all_data): | ||||||
|     """ |     """ | ||||||
| @@ -97,32 +100,32 @@ def isCommaSeparatedEmailList(field_data, all_data): | |||||||
|         try: |         try: | ||||||
|             isValidEmail(supposed_email.strip(), '') |             isValidEmail(supposed_email.strip(), '') | ||||||
|         except ValidationError: |         except ValidationError: | ||||||
|             raise ValidationError, gettext("Enter valid e-mail addresses separated by commas.") |             raise ValidationError, _("Enter valid e-mail addresses separated by commas.") | ||||||
|  |  | ||||||
| def isValidIPAddress4(field_data, all_data): | def isValidIPAddress4(field_data, all_data): | ||||||
|     if not ip4_re.search(field_data): |     if not ip4_re.search(field_data): | ||||||
|         raise ValidationError, gettext("Please enter a valid IP address.") |         raise ValidationError, _("Please enter a valid IP address.") | ||||||
|  |  | ||||||
| def isNotEmpty(field_data, all_data): | def isNotEmpty(field_data, all_data): | ||||||
|     if field_data.strip() == '': |     if field_data.strip() == '': | ||||||
|         raise ValidationError, gettext("Empty values are not allowed here.") |         raise ValidationError, _("Empty values are not allowed here.") | ||||||
|  |  | ||||||
| def isOnlyDigits(field_data, all_data): | def isOnlyDigits(field_data, all_data): | ||||||
|     if not field_data.isdigit(): |     if not field_data.isdigit(): | ||||||
|         raise ValidationError, gettext("Non-numeric characters aren't allowed here.") |         raise ValidationError, _("Non-numeric characters aren't allowed here.") | ||||||
|  |  | ||||||
| def isNotOnlyDigits(field_data, all_data): | def isNotOnlyDigits(field_data, all_data): | ||||||
|     if field_data.isdigit(): |     if field_data.isdigit(): | ||||||
|         raise ValidationError, gettext("This value can't be comprised solely of digits.") |         raise ValidationError, _("This value can't be comprised solely of digits.") | ||||||
|  |  | ||||||
| def isInteger(field_data, all_data): | def isInteger(field_data, all_data): | ||||||
|     # This differs from isOnlyDigits because this accepts the negative sign |     # This differs from isOnlyDigits because this accepts the negative sign | ||||||
|     if not integer_re.search(field_data): |     if not integer_re.search(field_data): | ||||||
|         raise ValidationError, gettext("Enter a whole number.") |         raise ValidationError, _("Enter a whole number.") | ||||||
|  |  | ||||||
| def isOnlyLetters(field_data, all_data): | def isOnlyLetters(field_data, all_data): | ||||||
|     if not field_data.isalpha(): |     if not field_data.isalpha(): | ||||||
|         raise ValidationError, gettext("Only alphabetical characters are allowed here.") |         raise ValidationError, _("Only alphabetical characters are allowed here.") | ||||||
|  |  | ||||||
| def _isValidDate(date_string): | def _isValidDate(date_string): | ||||||
|     """ |     """ | ||||||
| @@ -137,30 +140,30 @@ def _isValidDate(date_string): | |||||||
|     # This check is needed because strftime is used when saving the date |     # This check is needed because strftime is used when saving the date | ||||||
|     # value to the database, and strftime requires that the year be >=1900. |     # value to the database, and strftime requires that the year be >=1900. | ||||||
|     if year < 1900: |     if year < 1900: | ||||||
|         raise ValidationError, gettext('Year must be 1900 or later.') |         raise ValidationError, _('Year must be 1900 or later.') | ||||||
|     try: |     try: | ||||||
|         date(year, month, day) |         date(year, month, day) | ||||||
|     except ValueError, e: |     except ValueError, e: | ||||||
|         msg = gettext('Invalid date: %s') % gettext(str(e)) |         msg = _('Invalid date: %s') % _(str(e)) | ||||||
|         raise ValidationError, msg     |         raise ValidationError, msg     | ||||||
|  |  | ||||||
| def isValidANSIDate(field_data, all_data): | def isValidANSIDate(field_data, all_data): | ||||||
|     if not ansi_date_re.search(field_data): |     if not ansi_date_re.search(field_data): | ||||||
|         raise ValidationError, gettext('Enter a valid date in YYYY-MM-DD format.') |         raise ValidationError, _('Enter a valid date in YYYY-MM-DD format.') | ||||||
|     _isValidDate(field_data) |     _isValidDate(field_data) | ||||||
|  |  | ||||||
| def isValidANSITime(field_data, all_data): | def isValidANSITime(field_data, all_data): | ||||||
|     if not ansi_time_re.search(field_data): |     if not ansi_time_re.search(field_data): | ||||||
|         raise ValidationError, gettext('Enter a valid time in HH:MM format.') |         raise ValidationError, _('Enter a valid time in HH:MM format.') | ||||||
|  |  | ||||||
| def isValidANSIDatetime(field_data, all_data): | def isValidANSIDatetime(field_data, all_data): | ||||||
|     if not ansi_datetime_re.search(field_data): |     if not ansi_datetime_re.search(field_data): | ||||||
|         raise ValidationError, gettext('Enter a valid date/time in YYYY-MM-DD HH:MM format.') |         raise ValidationError, _('Enter a valid date/time in YYYY-MM-DD HH:MM format.') | ||||||
|     _isValidDate(field_data.split()[0]) |     _isValidDate(field_data.split()[0]) | ||||||
|  |  | ||||||
| def isValidEmail(field_data, all_data): | def isValidEmail(field_data, all_data): | ||||||
|     if not email_re.search(field_data): |     if not email_re.search(field_data): | ||||||
|         raise ValidationError, gettext('Enter a valid e-mail address.') |         raise ValidationError, _('Enter a valid e-mail address.') | ||||||
|  |  | ||||||
| def isValidImage(field_data, all_data): | def isValidImage(field_data, all_data): | ||||||
|     """ |     """ | ||||||
| @@ -172,22 +175,22 @@ def isValidImage(field_data, all_data): | |||||||
|     try: |     try: | ||||||
|         content = field_data['content'] |         content = field_data['content'] | ||||||
|     except TypeError: |     except TypeError: | ||||||
|         raise ValidationError, gettext("No file was submitted. Check the encoding type on the form.") |         raise ValidationError, _("No file was submitted. Check the encoding type on the form.") | ||||||
|     try: |     try: | ||||||
|         Image.open(StringIO(content)) |         Image.open(StringIO(content)) | ||||||
|     except IOError: # Python Imaging Library doesn't recognize it as an image |     except IOError: # Python Imaging Library doesn't recognize it as an image | ||||||
|         raise ValidationError, gettext("Upload a valid image. The file you uploaded was either not an image or a corrupted image.") |         raise ValidationError, _("Upload a valid image. The file you uploaded was either not an image or a corrupted image.") | ||||||
|  |  | ||||||
| def isValidImageURL(field_data, all_data): | def isValidImageURL(field_data, all_data): | ||||||
|     uc = URLMimeTypeCheck(('image/jpeg', 'image/gif', 'image/png')) |     uc = URLMimeTypeCheck(('image/jpeg', 'image/gif', 'image/png')) | ||||||
|     try: |     try: | ||||||
|         uc(field_data, all_data) |         uc(field_data, all_data) | ||||||
|     except URLMimeTypeCheck.InvalidContentType: |     except URLMimeTypeCheck.InvalidContentType: | ||||||
|         raise ValidationError, gettext("The URL %s does not point to a valid image.") % field_data |         raise ValidationError, _("The URL %s does not point to a valid image.") % field_data | ||||||
|  |  | ||||||
| def isValidPhone(field_data, all_data): | def isValidPhone(field_data, all_data): | ||||||
|     if not phone_re.search(field_data): |     if not phone_re.search(field_data): | ||||||
|         raise ValidationError, gettext('Phone numbers must be in XXX-XXX-XXXX format. "%s" is invalid.') % field_data |         raise ValidationError, _('Phone numbers must be in XXX-XXX-XXXX format. "%s" is invalid.') % field_data | ||||||
|  |  | ||||||
| def isValidQuicktimeVideoURL(field_data, all_data): | def isValidQuicktimeVideoURL(field_data, all_data): | ||||||
|     "Checks that the given URL is a video that can be played by QuickTime (qt, mpeg)" |     "Checks that the given URL is a video that can be played by QuickTime (qt, mpeg)" | ||||||
| @@ -195,11 +198,11 @@ def isValidQuicktimeVideoURL(field_data, all_data): | |||||||
|     try: |     try: | ||||||
|         uc(field_data, all_data) |         uc(field_data, all_data) | ||||||
|     except URLMimeTypeCheck.InvalidContentType: |     except URLMimeTypeCheck.InvalidContentType: | ||||||
|         raise ValidationError, gettext("The URL %s does not point to a valid QuickTime video.") % field_data |         raise ValidationError, _("The URL %s does not point to a valid QuickTime video.") % field_data | ||||||
|  |  | ||||||
| def isValidURL(field_data, all_data): | def isValidURL(field_data, all_data): | ||||||
|     if not url_re.search(field_data): |     if not url_re.search(field_data): | ||||||
|         raise ValidationError, gettext("A valid URL is required.") |         raise ValidationError, _("A valid URL is required.") | ||||||
|  |  | ||||||
| def isValidHTML(field_data, all_data): | def isValidHTML(field_data, all_data): | ||||||
|     import urllib, urllib2 |     import urllib, urllib2 | ||||||
| @@ -213,14 +216,14 @@ def isValidHTML(field_data, all_data): | |||||||
|         return |         return | ||||||
|     from xml.dom.minidom import parseString |     from xml.dom.minidom import parseString | ||||||
|     error_messages = [e.firstChild.wholeText for e in parseString(u.read()).getElementsByTagName('messages')[0].getElementsByTagName('msg')] |     error_messages = [e.firstChild.wholeText for e in parseString(u.read()).getElementsByTagName('messages')[0].getElementsByTagName('msg')] | ||||||
|     raise ValidationError, gettext("Valid HTML is required. Specific errors are:\n%s") % "\n".join(error_messages) |     raise ValidationError, _("Valid HTML is required. Specific errors are:\n%s") % "\n".join(error_messages) | ||||||
|  |  | ||||||
| def isWellFormedXml(field_data, all_data): | def isWellFormedXml(field_data, all_data): | ||||||
|     from xml.dom.minidom import parseString |     from xml.dom.minidom import parseString | ||||||
|     try: |     try: | ||||||
|         parseString(field_data) |         parseString(field_data) | ||||||
|     except Exception, e: # Naked except because we're not sure what will be thrown |     except Exception, e: # Naked except because we're not sure what will be thrown | ||||||
|         raise ValidationError, gettext("Badly formed XML: %s") % str(e) |         raise ValidationError, _("Badly formed XML: %s") % str(e) | ||||||
|  |  | ||||||
| def isWellFormedXmlFragment(field_data, all_data): | def isWellFormedXmlFragment(field_data, all_data): | ||||||
|     isWellFormedXml('<root>%s</root>' % field_data, all_data) |     isWellFormedXml('<root>%s</root>' % field_data, all_data) | ||||||
| @@ -250,7 +253,7 @@ def isValidUSState(field_data, all_data): | |||||||
|     "Checks that the given string is a valid two-letter U.S. state abbreviation" |     "Checks that the given string is a valid two-letter U.S. state abbreviation" | ||||||
|     states = ['AA', 'AE', 'AK', 'AL', 'AP', 'AR', 'AS', 'AZ', 'CA', 'CO', 'CT', 'DC', 'DE', 'FL', 'FM', 'GA', 'GU', 'HI', 'IA', 'ID', 'IL', 'IN', 'KS', 'KY', 'LA', 'MA', 'MD', 'ME', 'MH', 'MI', 'MN', 'MO', 'MP', 'MS', 'MT', 'NC', 'ND', 'NE', 'NH', 'NJ', 'NM', 'NV', 'NY', 'OH', 'OK', 'OR', 'PA', 'PR', 'PW', 'RI', 'SC', 'SD', 'TN', 'TX', 'UT', 'VA', 'VI', 'VT', 'WA', 'WI', 'WV', 'WY'] |     states = ['AA', 'AE', 'AK', 'AL', 'AP', 'AR', 'AS', 'AZ', 'CA', 'CO', 'CT', 'DC', 'DE', 'FL', 'FM', 'GA', 'GU', 'HI', 'IA', 'ID', 'IL', 'IN', 'KS', 'KY', 'LA', 'MA', 'MD', 'ME', 'MH', 'MI', 'MN', 'MO', 'MP', 'MS', 'MT', 'NC', 'ND', 'NE', 'NH', 'NJ', 'NM', 'NV', 'NY', 'OH', 'OK', 'OR', 'PA', 'PR', 'PW', 'RI', 'SC', 'SD', 'TN', 'TX', 'UT', 'VA', 'VI', 'VT', 'WA', 'WI', 'WV', 'WY'] | ||||||
|     if field_data.upper() not in states: |     if field_data.upper() not in states: | ||||||
|         raise ValidationError, gettext("Enter a valid U.S. state abbreviation.") |         raise ValidationError, _("Enter a valid U.S. state abbreviation.") | ||||||
|  |  | ||||||
| def hasNoProfanities(field_data, all_data): | def hasNoProfanities(field_data, all_data): | ||||||
|     """ |     """ | ||||||
| @@ -264,14 +267,14 @@ def hasNoProfanities(field_data, all_data): | |||||||
|     if words_seen: |     if words_seen: | ||||||
|         from django.utils.text import get_text_list |         from django.utils.text import get_text_list | ||||||
|         plural = len(words_seen) > 1 |         plural = len(words_seen) > 1 | ||||||
|         raise ValidationError, ngettext("Watch your mouth! The word %s is not allowed here.", |         raise ValidationError, ungettext("Watch your mouth! The word %s is not allowed here.", | ||||||
|             "Watch your mouth! The words %s are not allowed here.", plural) % \ |             "Watch your mouth! The words %s are not allowed here.", plural) % \ | ||||||
|             get_text_list(['"%s%s%s"' % (i[0], '-'*(len(i)-2), i[-1]) for i in words_seen], 'and') |             get_text_list(['"%s%s%s"' % (i[0], '-'*(len(i)-2), i[-1]) for i in words_seen], 'and') | ||||||
|  |  | ||||||
| class AlwaysMatchesOtherField(object): | class AlwaysMatchesOtherField(object): | ||||||
|     def __init__(self, other_field_name, error_message=None): |     def __init__(self, other_field_name, error_message=None): | ||||||
|         self.other = other_field_name |         self.other = other_field_name | ||||||
|         self.error_message = error_message or lazy_inter(gettext_lazy("This field must match the '%s' field."), self.other) |         self.error_message = error_message or lazy_inter(ugettext_lazy("This field must match the '%s' field."), self.other) | ||||||
|         self.always_test = True |         self.always_test = True | ||||||
|  |  | ||||||
|     def __call__(self, field_data, all_data): |     def __call__(self, field_data, all_data): | ||||||
| @@ -290,7 +293,7 @@ class ValidateIfOtherFieldEquals(object): | |||||||
|                 v(field_data, all_data) |                 v(field_data, all_data) | ||||||
|  |  | ||||||
| class RequiredIfOtherFieldNotGiven(object): | class RequiredIfOtherFieldNotGiven(object): | ||||||
|     def __init__(self, other_field_name, error_message=gettext_lazy("Please enter something for at least one field.")): |     def __init__(self, other_field_name, error_message=ugettext_lazy("Please enter something for at least one field.")): | ||||||
|         self.other, self.error_message = other_field_name, error_message |         self.other, self.error_message = other_field_name, error_message | ||||||
|         self.always_test = True |         self.always_test = True | ||||||
|  |  | ||||||
| @@ -299,7 +302,7 @@ class RequiredIfOtherFieldNotGiven(object): | |||||||
|             raise ValidationError, self.error_message |             raise ValidationError, self.error_message | ||||||
|  |  | ||||||
| class RequiredIfOtherFieldsGiven(object): | class RequiredIfOtherFieldsGiven(object): | ||||||
|     def __init__(self, other_field_names, error_message=gettext_lazy("Please enter both fields or leave them both empty.")): |     def __init__(self, other_field_names, error_message=ugettext_lazy("Please enter both fields or leave them both empty.")): | ||||||
|         self.other, self.error_message = other_field_names, error_message |         self.other, self.error_message = other_field_names, error_message | ||||||
|         self.always_test = True |         self.always_test = True | ||||||
|  |  | ||||||
| @@ -310,7 +313,7 @@ class RequiredIfOtherFieldsGiven(object): | |||||||
|  |  | ||||||
| class RequiredIfOtherFieldGiven(RequiredIfOtherFieldsGiven): | class RequiredIfOtherFieldGiven(RequiredIfOtherFieldsGiven): | ||||||
|     "Like RequiredIfOtherFieldsGiven, but takes a single field name instead of a list." |     "Like RequiredIfOtherFieldsGiven, but takes a single field name instead of a list." | ||||||
|     def __init__(self, other_field_name, error_message=gettext_lazy("Please enter both fields or leave them both empty.")): |     def __init__(self, other_field_name, error_message=ugettext_lazy("Please enter both fields or leave them both empty.")): | ||||||
|         RequiredIfOtherFieldsGiven.__init__(self, [other_field_name], error_message) |         RequiredIfOtherFieldsGiven.__init__(self, [other_field_name], error_message) | ||||||
|  |  | ||||||
| class RequiredIfOtherFieldEquals(object): | class RequiredIfOtherFieldEquals(object): | ||||||
| @@ -318,7 +321,7 @@ class RequiredIfOtherFieldEquals(object): | |||||||
|         self.other_field = other_field |         self.other_field = other_field | ||||||
|         self.other_value = other_value |         self.other_value = other_value | ||||||
|         other_label = other_label or other_value |         other_label = other_label or other_value | ||||||
|         self.error_message = error_message or lazy_inter(gettext_lazy("This field must be given if %(field)s is %(value)s"), { |         self.error_message = error_message or lazy_inter(ugettext_lazy("This field must be given if %(field)s is %(value)s"), { | ||||||
|             'field': other_field, 'value': other_label}) |             'field': other_field, 'value': other_label}) | ||||||
|         self.always_test = True |         self.always_test = True | ||||||
|  |  | ||||||
| @@ -331,7 +334,7 @@ class RequiredIfOtherFieldDoesNotEqual(object): | |||||||
|         self.other_field = other_field |         self.other_field = other_field | ||||||
|         self.other_value = other_value |         self.other_value = other_value | ||||||
|         other_label = other_label or other_value |         other_label = other_label or other_value | ||||||
|         self.error_message = error_message or lazy_inter(gettext_lazy("This field must be given if %(field)s is not %(value)s"), { |         self.error_message = error_message or lazy_inter(ugettext_lazy("This field must be given if %(field)s is not %(value)s"), { | ||||||
|             'field': other_field, 'value': other_label}) |             'field': other_field, 'value': other_label}) | ||||||
|         self.always_test = True |         self.always_test = True | ||||||
|  |  | ||||||
| @@ -350,7 +353,7 @@ class IsLessThanOtherField(object): | |||||||
| class UniqueAmongstFieldsWithPrefix(object): | class UniqueAmongstFieldsWithPrefix(object): | ||||||
|     def __init__(self, field_name, prefix, error_message): |     def __init__(self, field_name, prefix, error_message): | ||||||
|         self.field_name, self.prefix = field_name, prefix |         self.field_name, self.prefix = field_name, prefix | ||||||
|         self.error_message = error_message or gettext_lazy("Duplicate values are not allowed.") |         self.error_message = error_message or ugettext_lazy("Duplicate values are not allowed.") | ||||||
|  |  | ||||||
|     def __call__(self, field_data, all_data): |     def __call__(self, field_data, all_data): | ||||||
|         for field_name, value in all_data.items(): |         for field_name, value in all_data.items(): | ||||||
| @@ -365,11 +368,11 @@ class NumberIsInRange(object): | |||||||
|         self.lower, self.upper = lower, upper |         self.lower, self.upper = lower, upper | ||||||
|         if not error_message: |         if not error_message: | ||||||
|             if lower and upper: |             if lower and upper: | ||||||
|                  self.error_message = gettext("This value must be between %(lower)s and %(upper)s.") % {'lower': lower, 'upper': upper} |                  self.error_message = _("This value must be between %(lower)s and %(upper)s.") % {'lower': lower, 'upper': upper} | ||||||
|             elif lower: |             elif lower: | ||||||
|                 self.error_message = gettext("This value must be at least %s.") % lower |                 self.error_message = _("This value must be at least %s.") % lower | ||||||
|             elif upper: |             elif upper: | ||||||
|                 self.error_message = gettext("This value must be no more than %s.") % upper |                 self.error_message = _("This value must be no more than %s.") % upper | ||||||
|         else: |         else: | ||||||
|             self.error_message = error_message |             self.error_message = error_message | ||||||
|  |  | ||||||
| @@ -405,7 +408,7 @@ class IsAPowerOf(object): | |||||||
|         from math import log |         from math import log | ||||||
|         val = log(int(field_data)) / log(self.power_of) |         val = log(int(field_data)) / log(self.power_of) | ||||||
|         if val != int(val): |         if val != int(val): | ||||||
|             raise ValidationError, gettext("This value must be a power of %s.") % self.power_of |             raise ValidationError, _("This value must be a power of %s.") % self.power_of | ||||||
|  |  | ||||||
| class IsValidDecimal(object): | class IsValidDecimal(object): | ||||||
|     def __init__(self, max_digits, decimal_places): |     def __init__(self, max_digits, decimal_places): | ||||||
| @@ -414,19 +417,19 @@ class IsValidDecimal(object): | |||||||
|     def __call__(self, field_data, all_data): |     def __call__(self, field_data, all_data): | ||||||
|         match = decimal_re.search(str(field_data)) |         match = decimal_re.search(str(field_data)) | ||||||
|         if not match: |         if not match: | ||||||
|             raise ValidationError, gettext("Please enter a valid decimal number.") |             raise ValidationError, _("Please enter a valid decimal number.") | ||||||
|          |          | ||||||
|         digits = len(match.group('digits') or '') |         digits = len(match.group('digits') or '') | ||||||
|         decimals = len(match.group('decimals') or '') |         decimals = len(match.group('decimals') or '') | ||||||
|          |          | ||||||
|         if digits + decimals > self.max_digits: |         if digits + decimals > self.max_digits: | ||||||
|             raise ValidationError, ngettext("Please enter a valid decimal number with at most %s total digit.", |             raise ValidationError, ungettext("Please enter a valid decimal number with at most %s total digit.", | ||||||
|                 "Please enter a valid decimal number with at most %s total digits.", self.max_digits) % self.max_digits |                 "Please enter a valid decimal number with at most %s total digits.", self.max_digits) % self.max_digits | ||||||
|         if digits > (self.max_digits - self.decimal_places): |         if digits > (self.max_digits - self.decimal_places): | ||||||
|             raise ValidationError, ngettext( "Please enter a valid decimal number with a whole part of at most %s digit.", |             raise ValidationError, ungettext( "Please enter a valid decimal number with a whole part of at most %s digit.", | ||||||
|                 "Please enter a valid decimal number with a whole part of at most %s digits.", str(self.max_digits-self.decimal_places)) % str(self.max_digits-self.decimal_places) |                 "Please enter a valid decimal number with a whole part of at most %s digits.", str(self.max_digits-self.decimal_places)) % str(self.max_digits-self.decimal_places) | ||||||
|         if decimals > self.decimal_places: |         if decimals > self.decimal_places: | ||||||
|             raise ValidationError, ngettext("Please enter a valid decimal number with at most %s decimal place.", |             raise ValidationError, ungettext("Please enter a valid decimal number with at most %s decimal place.", | ||||||
|                 "Please enter a valid decimal number with at most %s decimal places.", self.decimal_places) % self.decimal_places |                 "Please enter a valid decimal number with at most %s decimal places.", self.decimal_places) % self.decimal_places | ||||||
|  |  | ||||||
| def isValidFloat(field_data, all_data): | def isValidFloat(field_data, all_data): | ||||||
| @@ -434,7 +437,7 @@ def isValidFloat(field_data, all_data): | |||||||
|     try: |     try: | ||||||
|         float(data) |         float(data) | ||||||
|     except ValueError: |     except ValueError: | ||||||
|         raise ValidationError, gettext("Please enter a valid floating point number.") |         raise ValidationError, ugettext("Please enter a valid floating point number.") | ||||||
|  |  | ||||||
| class HasAllowableSize(object): | class HasAllowableSize(object): | ||||||
|     """ |     """ | ||||||
| @@ -443,14 +446,14 @@ class HasAllowableSize(object): | |||||||
|     """ |     """ | ||||||
|     def __init__(self, min_size=None, max_size=None, min_error_message=None, max_error_message=None): |     def __init__(self, min_size=None, max_size=None, min_error_message=None, max_error_message=None): | ||||||
|         self.min_size, self.max_size = min_size, max_size |         self.min_size, self.max_size = min_size, max_size | ||||||
|         self.min_error_message = min_error_message or lazy_inter(gettext_lazy("Make sure your uploaded file is at least %s bytes big."), min_size) |         self.min_error_message = min_error_message or lazy_inter(ugettext_lazy("Make sure your uploaded file is at least %s bytes big."), min_size) | ||||||
|         self.max_error_message = max_error_message or lazy_inter(gettext_lazy("Make sure your uploaded file is at most %s bytes big."), max_size) |         self.max_error_message = max_error_message or lazy_inter(ugettext_lazy("Make sure your uploaded file is at most %s bytes big."), max_size) | ||||||
|  |  | ||||||
|     def __call__(self, field_data, all_data): |     def __call__(self, field_data, all_data): | ||||||
|         try: |         try: | ||||||
|             content = field_data['content'] |             content = field_data['content'] | ||||||
|         except TypeError: |         except TypeError: | ||||||
|             raise ValidationError, gettext_lazy("No file was submitted. Check the encoding type on the form.") |             raise ValidationError, ugettext_lazy("No file was submitted. Check the encoding type on the form.") | ||||||
|         if self.min_size is not None and len(content) < self.min_size: |         if self.min_size is not None and len(content) < self.min_size: | ||||||
|             raise ValidationError, self.min_error_message |             raise ValidationError, self.min_error_message | ||||||
|         if self.max_size is not None and len(content) > self.max_size: |         if self.max_size is not None and len(content) > self.max_size: | ||||||
| @@ -461,7 +464,7 @@ class MatchesRegularExpression(object): | |||||||
|     Checks that the field matches the given regular-expression. The regex |     Checks that the field matches the given regular-expression. The regex | ||||||
|     should be in string format, not already compiled. |     should be in string format, not already compiled. | ||||||
|     """ |     """ | ||||||
|     def __init__(self, regexp, error_message=gettext_lazy("The format for this field is wrong.")): |     def __init__(self, regexp, error_message=ugettext_lazy("The format for this field is wrong.")): | ||||||
|         self.regexp = re.compile(regexp) |         self.regexp = re.compile(regexp) | ||||||
|         self.error_message = error_message |         self.error_message = error_message | ||||||
|  |  | ||||||
| @@ -476,7 +479,7 @@ class AnyValidator(object): | |||||||
|     as a validation error. The message is rather unspecific, so it's best to |     as a validation error. The message is rather unspecific, so it's best to | ||||||
|     specify one on instantiation. |     specify one on instantiation. | ||||||
|     """ |     """ | ||||||
|     def __init__(self, validator_list=None, error_message=gettext_lazy("This field is invalid.")): |     def __init__(self, validator_list=None, error_message=ugettext_lazy("This field is invalid.")): | ||||||
|         if validator_list is None: validator_list = [] |         if validator_list is None: validator_list = [] | ||||||
|         self.validator_list = validator_list |         self.validator_list = validator_list | ||||||
|         self.error_message = error_message |         self.error_message = error_message | ||||||
| @@ -512,10 +515,10 @@ class URLMimeTypeCheck(object): | |||||||
|         try: |         try: | ||||||
|             info = urllib2.urlopen(field_data).info() |             info = urllib2.urlopen(field_data).info() | ||||||
|         except (urllib2.HTTPError, urllib2.URLError): |         except (urllib2.HTTPError, urllib2.URLError): | ||||||
|             raise URLMimeTypeCheck.CouldNotRetrieve, gettext("Could not retrieve anything from %s.") % field_data |             raise URLMimeTypeCheck.CouldNotRetrieve, _("Could not retrieve anything from %s.") % field_data | ||||||
|         content_type = info['content-type'] |         content_type = info['content-type'] | ||||||
|         if content_type not in self.mime_type_list: |         if content_type not in self.mime_type_list: | ||||||
|             raise URLMimeTypeCheck.InvalidContentType, gettext("The URL %(url)s returned the invalid Content-Type header '%(contenttype)s'.") % { |             raise URLMimeTypeCheck.InvalidContentType, _("The URL %(url)s returned the invalid Content-Type header '%(contenttype)s'.") % { | ||||||
|                 'url': field_data, 'contenttype': content_type} |                 'url': field_data, 'contenttype': content_type} | ||||||
|  |  | ||||||
| class RelaxNGCompact(object): | class RelaxNGCompact(object): | ||||||
|   | |||||||
| @@ -84,7 +84,7 @@ class DatabaseWrapper(local): | |||||||
|             kwargs = { |             kwargs = { | ||||||
|                 'conv': django_conversions, |                 'conv': django_conversions, | ||||||
|                 'charset': 'utf8', |                 'charset': 'utf8', | ||||||
|                 'use_unicode': False, |                 'use_unicode': True, | ||||||
|             } |             } | ||||||
|             if settings.DATABASE_USER: |             if settings.DATABASE_USER: | ||||||
|                 kwargs['user'] = settings.DATABASE_USER |                 kwargs['user'] = settings.DATABASE_USER | ||||||
|   | |||||||
| @@ -5,6 +5,7 @@ Requires MySQLdb: http://sourceforge.net/projects/mysql-python | |||||||
| """ | """ | ||||||
|  |  | ||||||
| from django.db.backends import util | from django.db.backends import util | ||||||
|  | from django.utils.encoding import force_unicode | ||||||
| try: | try: | ||||||
|     import MySQLdb as Database |     import MySQLdb as Database | ||||||
| except ImportError, e: | except ImportError, e: | ||||||
| @@ -25,6 +26,9 @@ django_conversions.update({ | |||||||
|     FIELD_TYPE.DATE: util.typecast_date, |     FIELD_TYPE.DATE: util.typecast_date, | ||||||
|     FIELD_TYPE.TIME: util.typecast_time, |     FIELD_TYPE.TIME: util.typecast_time, | ||||||
|     FIELD_TYPE.DECIMAL: util.typecast_decimal, |     FIELD_TYPE.DECIMAL: util.typecast_decimal, | ||||||
|  |     FIELD_TYPE.STRING: force_unicode, | ||||||
|  |     FIELD_TYPE.VAR_STRING: force_unicode, | ||||||
|  |     # Note: We don't add a convertor for BLOB here. Doesn't seem to be required. | ||||||
| }) | }) | ||||||
|  |  | ||||||
| # This should match the numerical portion of the version numbers (we can treat | # This should match the numerical portion of the version numbers (we can treat | ||||||
| @@ -87,6 +91,8 @@ class DatabaseWrapper(local): | |||||||
|         from django.conf import settings |         from django.conf import settings | ||||||
|         if not self._valid_connection(): |         if not self._valid_connection(): | ||||||
|             kwargs = { |             kwargs = { | ||||||
|  |                 # Note: use_unicode intentonally not set to work around some | ||||||
|  |                 # backwards-compat issues. We do it manually. | ||||||
|                 'user': settings.DATABASE_USER, |                 'user': settings.DATABASE_USER, | ||||||
|                 'db': settings.DATABASE_NAME, |                 'db': settings.DATABASE_NAME, | ||||||
|                 'passwd': settings.DATABASE_PASSWORD, |                 'passwd': settings.DATABASE_PASSWORD, | ||||||
| @@ -101,8 +107,16 @@ class DatabaseWrapper(local): | |||||||
|             kwargs.update(self.options) |             kwargs.update(self.options) | ||||||
|             self.connection = Database.connect(**kwargs) |             self.connection = Database.connect(**kwargs) | ||||||
|             cursor = self.connection.cursor() |             cursor = self.connection.cursor() | ||||||
|             if self.connection.get_server_info() >= '4.1': |             if self.connection.get_server_info() >= '4.1' and not self.connection.character_set_name().startswith('utf8'): | ||||||
|                 cursor.execute("SET NAMES 'utf8'") |                 if hasattr(self.connection, 'charset'): | ||||||
|  |                     # MySQLdb < 1.2.1 backwards-compat hacks. | ||||||
|  |                     conn = self.connection | ||||||
|  |                     cursor.execute("SET NAMES 'utf8'") | ||||||
|  |                     cursor.execute("SET CHARACTER SET 'utf8'") | ||||||
|  |                     to_str = lambda u, dummy=None, c=conn: c.literal(u.encode('utf-8')) | ||||||
|  |                     conn.converter[unicode] = to_str | ||||||
|  |                 else: | ||||||
|  |                     self.connection.set_character_set('utf8') | ||||||
|         else: |         else: | ||||||
|             cursor = self.connection.cursor() |             cursor = self.connection.cursor() | ||||||
|         if settings.DEBUG: |         if settings.DEBUG: | ||||||
|   | |||||||
| @@ -6,13 +6,18 @@ Requires cx_Oracle: http://www.python.net/crew/atuining/cx_Oracle/ | |||||||
|  |  | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| from django.db.backends import util | from django.db.backends import util | ||||||
|  | from django.utils.datastructures import SortedDict | ||||||
|  | from django.utils.encoding import smart_str, force_unicode | ||||||
|  | import datetime | ||||||
|  | import os | ||||||
|  |  | ||||||
|  | # Oracle takes client-side character set encoding from the environment. | ||||||
|  | os.environ['NLS_LANG'] = '.UTF8' | ||||||
| try: | try: | ||||||
|     import cx_Oracle as Database |     import cx_Oracle as Database | ||||||
| except ImportError, e: | except ImportError, e: | ||||||
|     from django.core.exceptions import ImproperlyConfigured |     from django.core.exceptions import ImproperlyConfigured | ||||||
|     raise ImproperlyConfigured, "Error loading cx_Oracle module: %s" % e |     raise ImproperlyConfigured, "Error loading cx_Oracle module: %s" % e | ||||||
| import datetime |  | ||||||
| from django.utils.datastructures import SortedDict |  | ||||||
|  |  | ||||||
|  |  | ||||||
| DatabaseError = Database.Error | DatabaseError = Database.Error | ||||||
| @@ -45,9 +50,9 @@ class DatabaseWrapper(local): | |||||||
|                 conn_string = "%s/%s@%s" % (settings.DATABASE_USER, settings.DATABASE_PASSWORD, settings.DATABASE_NAME) |                 conn_string = "%s/%s@%s" % (settings.DATABASE_USER, settings.DATABASE_PASSWORD, settings.DATABASE_NAME) | ||||||
|                 self.connection = Database.connect(conn_string, **self.options) |                 self.connection = Database.connect(conn_string, **self.options) | ||||||
|         cursor = FormatStylePlaceholderCursor(self.connection) |         cursor = FormatStylePlaceholderCursor(self.connection) | ||||||
|         # default arraysize of 1 is highly sub-optimal |         # Default arraysize of 1 is highly sub-optimal. | ||||||
|         cursor.arraysize = 100 |         cursor.arraysize = 100 | ||||||
|         # set oracle date to ansi date format |         # Set oracle date to ansi date format. | ||||||
|         cursor.execute("ALTER SESSION SET NLS_DATE_FORMAT = 'YYYY-MM-DD'") |         cursor.execute("ALTER SESSION SET NLS_DATE_FORMAT = 'YYYY-MM-DD'") | ||||||
|         cursor.execute("ALTER SESSION SET NLS_TIMESTAMP_FORMAT = 'YYYY-MM-DD HH24:MI:SS.FF'") |         cursor.execute("ALTER SESSION SET NLS_TIMESTAMP_FORMAT = 'YYYY-MM-DD HH24:MI:SS.FF'") | ||||||
|         if settings.DEBUG: |         if settings.DEBUG: | ||||||
| @@ -78,23 +83,22 @@ uses_case_insensitive_names = True | |||||||
|  |  | ||||||
| class FormatStylePlaceholderCursor(Database.Cursor): | class FormatStylePlaceholderCursor(Database.Cursor): | ||||||
|     """ |     """ | ||||||
|     Django uses "format" (e.g. '%s') style placeholders, but Oracle uses ":var" style. |     Django uses "format" (e.g. '%s') style placeholders, but Oracle uses ":var" | ||||||
|     This fixes it -- but note that if you want to use a literal "%s" in a query, |     style. This fixes it -- but note that if you want to use a literal "%s" in | ||||||
|     you'll need to use "%%s". |     a query, you'll need to use "%%s". | ||||||
|  |  | ||||||
|  |     We also do automatic conversion between Unicode on the Python side and | ||||||
|  |     UTF-8 -- for talking to Oracle -- in here. | ||||||
|     """ |     """ | ||||||
|  |     charset = 'utf-8' | ||||||
|  |  | ||||||
|     def _rewrite_args(self, query, params=None): |     def _rewrite_args(self, query, params=None): | ||||||
|         if params is None: |         if params is None: | ||||||
|             params = [] |             params = [] | ||||||
|         else: |         else: | ||||||
|             # cx_Oracle can't handle unicode parameters, so cast to str for now |             params = self._format_params(params) | ||||||
|             for i, param in enumerate(params): |  | ||||||
|                 if type(param) == unicode: |  | ||||||
|                     try: |  | ||||||
|                         params[i] = param.encode('utf-8') |  | ||||||
|                     except UnicodeError: |  | ||||||
|                         params[i] = str(param) |  | ||||||
|         args = [(':arg%d' % i) for i in range(len(params))] |         args = [(':arg%d' % i) for i in range(len(params))] | ||||||
|         query = query % tuple(args) |         query = smart_str(query, self.charset) % tuple(args) | ||||||
|         # cx_Oracle wants no trailing ';' for SQL statements.  For PL/SQL, it |         # cx_Oracle wants no trailing ';' for SQL statements.  For PL/SQL, it | ||||||
|         # it does want a trailing ';' but not a trailing '/'.  However, these |         # it does want a trailing ';' but not a trailing '/'.  However, these | ||||||
|         # characters must be included in the original query in case the query |         # characters must be included in the original query in case the query | ||||||
| @@ -103,6 +107,16 @@ class FormatStylePlaceholderCursor(Database.Cursor): | |||||||
|             query = query[:-1] |             query = query[:-1] | ||||||
|         return query, params |         return query, params | ||||||
|  |  | ||||||
|  |     def _format_params(self, params): | ||||||
|  |         if isinstance(params, dict): | ||||||
|  |             result = {} | ||||||
|  |             charset = self.charset | ||||||
|  |             for key, value in params.items(): | ||||||
|  |                 result[smart_str(key, charset)] = smart_str(value, charset) | ||||||
|  |             return result | ||||||
|  |         else: | ||||||
|  |             return tuple([smart_str(p, self.charset, True) for p in params]) | ||||||
|  |  | ||||||
|     def execute(self, query, params=None): |     def execute(self, query, params=None): | ||||||
|         query, params = self._rewrite_args(query, params) |         query, params = self._rewrite_args(query, params) | ||||||
|         return Database.Cursor.execute(self, query, params) |         return Database.Cursor.execute(self, query, params) | ||||||
| @@ -111,6 +125,26 @@ class FormatStylePlaceholderCursor(Database.Cursor): | |||||||
|         query, params = self._rewrite_args(query, params) |         query, params = self._rewrite_args(query, params) | ||||||
|         return Database.Cursor.executemany(self, query, params) |         return Database.Cursor.executemany(self, query, params) | ||||||
|  |  | ||||||
|  |     def fetchone(self): | ||||||
|  |         return to_unicode(Database.Cursor.fetchone(self)) | ||||||
|  |  | ||||||
|  |     def fetchmany(self, size=None): | ||||||
|  |         if size is None: | ||||||
|  |             size = self.arraysize | ||||||
|  |         return tuple([tuple([to_unicode(e) for e in r]) for r in Database.Cursor.fetchmany(self, size)]) | ||||||
|  |  | ||||||
|  |     def fetchall(self): | ||||||
|  |         return tuple([tuple([to_unicode(e) for e in r]) for r in Database.Cursor.fetchall(self)]) | ||||||
|  |  | ||||||
|  | def to_unicode(s): | ||||||
|  |     """ | ||||||
|  |     Convert strings to Unicode objects (and return all other data types | ||||||
|  |     unchanged). | ||||||
|  |     """ | ||||||
|  |     if isinstance(s, basestring): | ||||||
|  |         return force_unicode(s) | ||||||
|  |     return s | ||||||
|  |  | ||||||
| def quote_name(name): | def quote_name(name): | ||||||
|     # SQL92 requires delimited (quoted) names to be case-sensitive.  When |     # SQL92 requires delimited (quoted) names to be case-sensitive.  When | ||||||
|     # not quoted, Oracle has case-insensitive behavior for identifiers, but |     # not quoted, Oracle has case-insensitive behavior for identifiers, but | ||||||
|   | |||||||
| @@ -8,15 +8,15 @@ from django.core import management | |||||||
| DATA_TYPES = { | DATA_TYPES = { | ||||||
|     'AutoField':                    'NUMBER(11)', |     'AutoField':                    'NUMBER(11)', | ||||||
|     'BooleanField':                 'NUMBER(1) CHECK (%(column)s IN (0,1))', |     'BooleanField':                 'NUMBER(1) CHECK (%(column)s IN (0,1))', | ||||||
|     'CharField':                    'VARCHAR2(%(maxlength)s)', |     'CharField':                    'NVARCHAR2(%(maxlength)s)', | ||||||
|     'CommaSeparatedIntegerField':   'VARCHAR2(%(maxlength)s)', |     'CommaSeparatedIntegerField':   'VARCHAR2(%(maxlength)s)', | ||||||
|     'DateField':                    'DATE', |     'DateField':                    'DATE', | ||||||
|     'DateTimeField':                'TIMESTAMP', |     'DateTimeField':                'TIMESTAMP', | ||||||
|     'DecimalField':                 'NUMBER(%(max_digits)s, %(decimal_places)s)', |     'DecimalField':                 'NUMBER(%(max_digits)s, %(decimal_places)s)', | ||||||
|     'FileField':                    'VARCHAR2(100)', |     'FileField':                    'NVARCHAR2(100)', | ||||||
|     'FilePathField':                'VARCHAR2(100)', |     'FilePathField':                'NVARCHAR2(100)', | ||||||
|     'FloatField':                   'DOUBLE PRECISION', |     'FloatField':                   'DOUBLE PRECISION', | ||||||
|     'ImageField':                   'VARCHAR2(100)', |     'ImageField':                   'NVARCHAR2(100)', | ||||||
|     'IntegerField':                 'NUMBER(11)', |     'IntegerField':                 'NUMBER(11)', | ||||||
|     'IPAddressField':               'VARCHAR2(15)', |     'IPAddressField':               'VARCHAR2(15)', | ||||||
|     'ManyToManyField':              None, |     'ManyToManyField':              None, | ||||||
| @@ -25,7 +25,7 @@ DATA_TYPES = { | |||||||
|     'PhoneNumberField':             'VARCHAR2(20)', |     'PhoneNumberField':             'VARCHAR2(20)', | ||||||
|     'PositiveIntegerField':         'NUMBER(11) CHECK (%(column)s >= 0)', |     'PositiveIntegerField':         'NUMBER(11) CHECK (%(column)s >= 0)', | ||||||
|     'PositiveSmallIntegerField':    'NUMBER(11) CHECK (%(column)s >= 0)', |     'PositiveSmallIntegerField':    'NUMBER(11) CHECK (%(column)s >= 0)', | ||||||
|     'SlugField':                    'VARCHAR2(50)', |     'SlugField':                    'NVARCHAR2(50)', | ||||||
|     'SmallIntegerField':            'NUMBER(11)', |     'SmallIntegerField':            'NUMBER(11)', | ||||||
|     'TextField':                    'NCLOB', |     'TextField':                    'NCLOB', | ||||||
|     'TimeField':                    'TIMESTAMP', |     'TimeField':                    'TIMESTAMP', | ||||||
|   | |||||||
| @@ -4,6 +4,7 @@ PostgreSQL database backend for Django. | |||||||
| Requires psycopg 1: http://initd.org/projects/psycopg1 | Requires psycopg 1: http://initd.org/projects/psycopg1 | ||||||
| """ | """ | ||||||
|  |  | ||||||
|  | from django.utils.encoding import smart_str, smart_unicode | ||||||
| from django.db.backends import util | from django.db.backends import util | ||||||
| try: | try: | ||||||
|     import psycopg as Database |     import psycopg as Database | ||||||
| @@ -21,11 +22,6 @@ except ImportError: | |||||||
|     # Import copy of _thread_local.py from Python 2.4 |     # Import copy of _thread_local.py from Python 2.4 | ||||||
|     from django.utils._threading_local import local |     from django.utils._threading_local import local | ||||||
|  |  | ||||||
| def smart_basestring(s, charset): |  | ||||||
|     if isinstance(s, unicode): |  | ||||||
|         return s.encode(charset) |  | ||||||
|     return s |  | ||||||
|  |  | ||||||
| class UnicodeCursorWrapper(object): | class UnicodeCursorWrapper(object): | ||||||
|     """ |     """ | ||||||
|     A thin wrapper around psycopg cursors that allows them to accept Unicode |     A thin wrapper around psycopg cursors that allows them to accept Unicode | ||||||
| @@ -33,18 +29,31 @@ class UnicodeCursorWrapper(object): | |||||||
|  |  | ||||||
|     This is necessary because psycopg doesn't apply any DB quoting to |     This is necessary because psycopg doesn't apply any DB quoting to | ||||||
|     parameters that are Unicode strings. If a param is Unicode, this will |     parameters that are Unicode strings. If a param is Unicode, this will | ||||||
|     convert it to a bytestring using DEFAULT_CHARSET before passing it to |     convert it to a bytestring using database client's encoding before passing | ||||||
|     psycopg. |     it to psycopg. | ||||||
|  |  | ||||||
|  |     All results retrieved from the database are converted into Unicode strings | ||||||
|  |     before being returned to the caller. | ||||||
|     """ |     """ | ||||||
|     def __init__(self, cursor, charset): |     def __init__(self, cursor, charset): | ||||||
|         self.cursor = cursor |         self.cursor = cursor | ||||||
|         self.charset = charset |         self.charset = charset | ||||||
|  |  | ||||||
|  |     def format_params(self, params): | ||||||
|  |         if isinstance(params, dict): | ||||||
|  |             result = {} | ||||||
|  |             charset = self.charset | ||||||
|  |             for key, value in params.items(): | ||||||
|  |                 result[smart_str(key, charset)] = smart_str(value, charset) | ||||||
|  |             return result | ||||||
|  |         else: | ||||||
|  |             return tuple([smart_str(p, self.charset, True) for p in params]) | ||||||
|  |  | ||||||
|     def execute(self, sql, params=()): |     def execute(self, sql, params=()): | ||||||
|         return self.cursor.execute(sql, [smart_basestring(p, self.charset) for p in params]) |         return self.cursor.execute(smart_str(sql, self.charset), self.format_params(params)) | ||||||
|  |  | ||||||
|     def executemany(self, sql, param_list): |     def executemany(self, sql, param_list): | ||||||
|         new_param_list = [tuple([smart_basestring(p, self.charset) for p in params]) for params in param_list] |         new_param_list = [self.format_params(params) for params in param_list] | ||||||
|         return self.cursor.executemany(sql, new_param_list) |         return self.cursor.executemany(sql, new_param_list) | ||||||
|  |  | ||||||
|     def __getattr__(self, attr): |     def __getattr__(self, attr): | ||||||
| @@ -83,7 +92,8 @@ class DatabaseWrapper(local): | |||||||
|         cursor = self.connection.cursor() |         cursor = self.connection.cursor() | ||||||
|         if set_tz: |         if set_tz: | ||||||
|             cursor.execute("SET TIME ZONE %s", [settings.TIME_ZONE]) |             cursor.execute("SET TIME ZONE %s", [settings.TIME_ZONE]) | ||||||
|         cursor = UnicodeCursorWrapper(cursor, settings.DEFAULT_CHARSET) |         cursor.execute("SET client_encoding to 'UNICODE'") | ||||||
|  |         cursor = UnicodeCursorWrapper(cursor, 'utf-8') | ||||||
|         global postgres_version |         global postgres_version | ||||||
|         if not postgres_version: |         if not postgres_version: | ||||||
|             cursor.execute("SELECT version()") |             cursor.execute("SELECT version()") | ||||||
| @@ -186,16 +196,17 @@ def get_sql_flush(style, tables, sequences): | |||||||
|     """ |     """ | ||||||
|     if tables: |     if tables: | ||||||
|         if postgres_version[0] >= 8 and postgres_version[1] >= 1: |         if postgres_version[0] >= 8 and postgres_version[1] >= 1: | ||||||
|             # Postgres 8.1+ can do 'TRUNCATE x, y, z...;'. In fact, it *has to* in order to be able to |             # Postgres 8.1+ can do 'TRUNCATE x, y, z...;'. In fact, it *has to* | ||||||
|             # truncate tables referenced by a foreign key in any other table. The result is a |             # in order to be able to truncate tables referenced by a foreign | ||||||
|             # single SQL TRUNCATE statement. |             # key in any other table. The result is a single SQL TRUNCATE | ||||||
|  |             # statement. | ||||||
|             sql = ['%s %s;' % \ |             sql = ['%s %s;' % \ | ||||||
|                 (style.SQL_KEYWORD('TRUNCATE'), |                 (style.SQL_KEYWORD('TRUNCATE'), | ||||||
|                  style.SQL_FIELD(', '.join([quote_name(table) for table in tables])) |                  style.SQL_FIELD(', '.join([quote_name(table) for table in tables])) | ||||||
|             )] |             )] | ||||||
|         else: |         else: | ||||||
|             # Older versions of Postgres can't do TRUNCATE in a single call, so they must use |             # Older versions of Postgres can't do TRUNCATE in a single call, so | ||||||
|             # a simple delete. |             # they must use a simple delete. | ||||||
|             sql = ['%s %s %s;' % \ |             sql = ['%s %s %s;' % \ | ||||||
|                     (style.SQL_KEYWORD('DELETE'), |                     (style.SQL_KEYWORD('DELETE'), | ||||||
|                      style.SQL_KEYWORD('FROM'), |                      style.SQL_KEYWORD('FROM'), | ||||||
| @@ -263,6 +274,14 @@ def get_sql_sequence_reset(style, model_list): | |||||||
|                 style.SQL_TABLE(f.m2m_db_table()))) |                 style.SQL_TABLE(f.m2m_db_table()))) | ||||||
|     return output |     return output | ||||||
|  |  | ||||||
|  | def typecast_string(s): | ||||||
|  |     """ | ||||||
|  |     Cast all returned strings to unicode strings. | ||||||
|  |     """ | ||||||
|  |     if not s: | ||||||
|  |         return s | ||||||
|  |     return smart_unicode(s) | ||||||
|  |  | ||||||
| # Register these custom typecasts, because Django expects dates/times to be | # Register these custom typecasts, because Django expects dates/times to be | ||||||
| # in Python's native (standard-library) datetime/time format, whereas psycopg | # in Python's native (standard-library) datetime/time format, whereas psycopg | ||||||
| # use mx.DateTime by default. | # use mx.DateTime by default. | ||||||
| @@ -274,6 +293,7 @@ Database.register_type(Database.new_type((1083,1266), "TIME", util.typecast_time | |||||||
| Database.register_type(Database.new_type((1114,1184), "TIMESTAMP", util.typecast_timestamp)) | Database.register_type(Database.new_type((1114,1184), "TIMESTAMP", util.typecast_timestamp)) | ||||||
| Database.register_type(Database.new_type((16,), "BOOLEAN", util.typecast_boolean)) | Database.register_type(Database.new_type((16,), "BOOLEAN", util.typecast_boolean)) | ||||||
| Database.register_type(Database.new_type((1700,), "NUMERIC", util.typecast_decimal)) | Database.register_type(Database.new_type((1700,), "NUMERIC", util.typecast_decimal)) | ||||||
|  | Database.register_type(Database.new_type(Database.types[1043].values, 'STRING', typecast_string)) | ||||||
|  |  | ||||||
| OPERATOR_MAPPING = { | OPERATOR_MAPPING = { | ||||||
|     'exact': '= %s', |     'exact': '= %s', | ||||||
|   | |||||||
| @@ -7,6 +7,7 @@ Requires psycopg 2: http://initd.org/projects/psycopg2 | |||||||
| from django.db.backends import util | from django.db.backends import util | ||||||
| try: | try: | ||||||
|     import psycopg2 as Database |     import psycopg2 as Database | ||||||
|  |     import psycopg2.extensions | ||||||
| except ImportError, e: | except ImportError, e: | ||||||
|     from django.core.exceptions import ImproperlyConfigured |     from django.core.exceptions import ImproperlyConfigured | ||||||
|     raise ImproperlyConfigured, "Error loading psycopg2 module: %s" % e |     raise ImproperlyConfigured, "Error loading psycopg2 module: %s" % e | ||||||
| @@ -21,6 +22,8 @@ except ImportError: | |||||||
|     # Import copy of _thread_local.py from Python 2.4 |     # Import copy of _thread_local.py from Python 2.4 | ||||||
|     from django.utils._threading_local import local |     from django.utils._threading_local import local | ||||||
|  |  | ||||||
|  | psycopg2.extensions.register_type(psycopg2.extensions.UNICODE) | ||||||
|  |  | ||||||
| postgres_version = None | postgres_version = None | ||||||
|  |  | ||||||
| class DatabaseWrapper(local): | class DatabaseWrapper(local): | ||||||
| @@ -48,6 +51,7 @@ class DatabaseWrapper(local): | |||||||
|                 conn_string += " port=%s" % settings.DATABASE_PORT |                 conn_string += " port=%s" % settings.DATABASE_PORT | ||||||
|             self.connection = Database.connect(conn_string, **self.options) |             self.connection = Database.connect(conn_string, **self.options) | ||||||
|             self.connection.set_isolation_level(1) # make transactions transparent to all cursors |             self.connection.set_isolation_level(1) # make transactions transparent to all cursors | ||||||
|  |             self.connection.set_client_encoding('UTF8') | ||||||
|         cursor = self.connection.cursor() |         cursor = self.connection.cursor() | ||||||
|         cursor.tzinfo_factory = None |         cursor.tzinfo_factory = None | ||||||
|         if set_tz: |         if set_tz: | ||||||
|   | |||||||
| @@ -34,14 +34,6 @@ Database.register_converter("TIMESTAMP", util.typecast_timestamp) | |||||||
| Database.register_converter("decimal", util.typecast_decimal) | Database.register_converter("decimal", util.typecast_decimal) | ||||||
| Database.register_adapter(decimal.Decimal, util.rev_typecast_decimal) | Database.register_adapter(decimal.Decimal, util.rev_typecast_decimal) | ||||||
|  |  | ||||||
| def utf8rowFactory(cursor, row): |  | ||||||
|     def utf8(s): |  | ||||||
|         if type(s) == unicode: |  | ||||||
|             return s.encode("utf-8") |  | ||||||
|         else: |  | ||||||
|             return s |  | ||||||
|     return [utf8(r) for r in row] |  | ||||||
|  |  | ||||||
| try: | try: | ||||||
|     # Only exists in Python 2.4+ |     # Only exists in Python 2.4+ | ||||||
|     from threading import local |     from threading import local | ||||||
| @@ -69,7 +61,6 @@ class DatabaseWrapper(local): | |||||||
|             self.connection.create_function("django_date_trunc", 2, _sqlite_date_trunc) |             self.connection.create_function("django_date_trunc", 2, _sqlite_date_trunc) | ||||||
|             self.connection.create_function("regexp", 2, _sqlite_regexp) |             self.connection.create_function("regexp", 2, _sqlite_regexp) | ||||||
|         cursor = self.connection.cursor(factory=SQLiteCursorWrapper) |         cursor = self.connection.cursor(factory=SQLiteCursorWrapper) | ||||||
|         cursor.row_factory = utf8rowFactory |  | ||||||
|         if settings.DEBUG: |         if settings.DEBUG: | ||||||
|             return util.CursorDebugWrapper(cursor, self) |             return util.CursorDebugWrapper(cursor, self) | ||||||
|         else: |         else: | ||||||
| @@ -85,8 +76,9 @@ class DatabaseWrapper(local): | |||||||
|  |  | ||||||
|     def close(self): |     def close(self): | ||||||
|         from django.conf import settings |         from django.conf import settings | ||||||
|         # If database is in memory, closing the connection destroys the database. |         # If database is in memory, closing the connection destroys the | ||||||
|         # To prevent accidental data loss, ignore close requests on an in-memory db. |         # database.  To prevent accidental data loss, ignore close requests on | ||||||
|  |         # an in-memory db. | ||||||
|         if self.connection is not None and settings.DATABASE_NAME != ":memory:": |         if self.connection is not None and settings.DATABASE_NAME != ":memory:": | ||||||
|             self.connection.close() |             self.connection.close() | ||||||
|             self.connection = None |             self.connection = None | ||||||
| @@ -181,10 +173,10 @@ def get_autoinc_sql(table): | |||||||
|     return None |     return None | ||||||
|  |  | ||||||
| def get_sql_flush(style, tables, sequences): | def get_sql_flush(style, tables, sequences): | ||||||
|     """Return a list of SQL statements required to remove all data from |     """ | ||||||
|  |     Return a list of SQL statements required to remove all data from | ||||||
|     all tables in the database (without actually removing the tables |     all tables in the database (without actually removing the tables | ||||||
|     themselves) and put the database in an empty 'initial' state |     themselves) and put the database in an empty 'initial' state | ||||||
|  |  | ||||||
|     """ |     """ | ||||||
|     # NB: The generated SQL below is specific to SQLite |     # NB: The generated SQL below is specific to SQLite | ||||||
|     # Note: The DELETE FROM... SQL generated below works for SQLite databases |     # Note: The DELETE FROM... SQL generated below works for SQLite databases | ||||||
| @@ -241,3 +233,4 @@ OPERATOR_MAPPING = { | |||||||
|     'istartswith': "LIKE %s ESCAPE '\\'", |     'istartswith': "LIKE %s ESCAPE '\\'", | ||||||
|     'iendswith': "LIKE %s ESCAPE '\\'", |     'iendswith': "LIKE %s ESCAPE '\\'", | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,6 +1,7 @@ | |||||||
| import datetime | import datetime | ||||||
| import md5 | import md5 | ||||||
| from time import time | from time import time | ||||||
|  | from django.utils.encoding import smart_unicode, force_unicode | ||||||
|  |  | ||||||
| try: | try: | ||||||
|     import decimal |     import decimal | ||||||
| @@ -18,12 +19,8 @@ class CursorDebugWrapper(object): | |||||||
|             return self.cursor.execute(sql, params) |             return self.cursor.execute(sql, params) | ||||||
|         finally: |         finally: | ||||||
|             stop = time() |             stop = time() | ||||||
|             # If params was a list, convert it to a tuple, because string |  | ||||||
|             # formatting with '%' only works with tuples or dicts. |  | ||||||
|             if not isinstance(params, (tuple, dict)): |  | ||||||
|                 params = tuple(params) |  | ||||||
|             self.db.queries.append({ |             self.db.queries.append({ | ||||||
|                 'sql': sql % params, |                 'sql': smart_unicode(sql) % convert_args(params), | ||||||
|                 'time': "%.3f" % (stop - start), |                 'time': "%.3f" % (stop - start), | ||||||
|             }) |             }) | ||||||
|  |  | ||||||
| @@ -34,7 +31,7 @@ class CursorDebugWrapper(object): | |||||||
|         finally: |         finally: | ||||||
|             stop = time() |             stop = time() | ||||||
|             self.db.queries.append({ |             self.db.queries.append({ | ||||||
|                 'sql': 'MANY: ' + sql + ' ' + str(tuple(param_list)), |                 'sql': 'MANY: ' + sql + ' ' + smart_unicode(tuple(param_list)), | ||||||
|                 'time': "%.3f" % (stop - start), |                 'time': "%.3f" % (stop - start), | ||||||
|             }) |             }) | ||||||
|  |  | ||||||
| @@ -44,6 +41,16 @@ class CursorDebugWrapper(object): | |||||||
|         else: |         else: | ||||||
|             return getattr(self.cursor, attr) |             return getattr(self.cursor, attr) | ||||||
|  |  | ||||||
|  | def convert_args(args): | ||||||
|  |     """ | ||||||
|  |     Convert sequence or dictionary to contain unicode values. | ||||||
|  |     """ | ||||||
|  |     to_unicode = lambda s: force_unicode(s, strings_only=True) | ||||||
|  |     if isinstance(args, (list, tuple)): | ||||||
|  |         return tuple([to_unicode(val) for val in args]) | ||||||
|  |     else: | ||||||
|  |         return dict([(to_unicode(k), to_unicode(v)) for k, v in args.items()]) | ||||||
|  |  | ||||||
| ############################################### | ############################################### | ||||||
| # Converters from database (string) to Python # | # Converters from database (string) to Python # | ||||||
| ############################################### | ############################################### | ||||||
|   | |||||||
| @@ -12,6 +12,7 @@ from django.db.models.loading import register_models, get_model | |||||||
| from django.dispatch import dispatcher | from django.dispatch import dispatcher | ||||||
| from django.utils.datastructures import SortedDict | from django.utils.datastructures import SortedDict | ||||||
| from django.utils.functional import curry | from django.utils.functional import curry | ||||||
|  | from django.utils.encoding import smart_str, force_unicode | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| from itertools import izip | from itertools import izip | ||||||
| import types | import types | ||||||
| @@ -83,9 +84,11 @@ class Model(object): | |||||||
|         return getattr(self, self._meta.pk.attname) |         return getattr(self, self._meta.pk.attname) | ||||||
|  |  | ||||||
|     def __repr__(self): |     def __repr__(self): | ||||||
|         return '<%s: %s>' % (self.__class__.__name__, self) |         return smart_str(u'<%s: %s>' % (self.__class__.__name__, unicode(self))) | ||||||
|  |  | ||||||
|     def __str__(self): |     def __str__(self): | ||||||
|  |         if hasattr(self, '__unicode__'): | ||||||
|  |             return force_unicode(self).encode('utf-8') | ||||||
|         return '%s object' % self.__class__.__name__ |         return '%s object' % self.__class__.__name__ | ||||||
|  |  | ||||||
|     def __eq__(self, other): |     def __eq__(self, other): | ||||||
| @@ -323,14 +326,14 @@ class Model(object): | |||||||
|  |  | ||||||
|     def _get_FIELD_display(self, field): |     def _get_FIELD_display(self, field): | ||||||
|         value = getattr(self, field.attname) |         value = getattr(self, field.attname) | ||||||
|         return dict(field.choices).get(value, value) |         return force_unicode(dict(field.choices).get(value, value), strings_only=True) | ||||||
|  |  | ||||||
|     def _get_next_or_previous_by_FIELD(self, field, is_next, **kwargs): |     def _get_next_or_previous_by_FIELD(self, field, is_next, **kwargs): | ||||||
|         op = is_next and '>' or '<' |         op = is_next and '>' or '<' | ||||||
|         where = '(%s %s %%s OR (%s = %%s AND %s.%s %s %%s))' % \ |         where = '(%s %s %%s OR (%s = %%s AND %s.%s %s %%s))' % \ | ||||||
|             (backend.quote_name(field.column), op, backend.quote_name(field.column), |             (backend.quote_name(field.column), op, backend.quote_name(field.column), | ||||||
|             backend.quote_name(self._meta.db_table), backend.quote_name(self._meta.pk.column), op) |             backend.quote_name(self._meta.db_table), backend.quote_name(self._meta.pk.column), op) | ||||||
|         param = str(getattr(self, field.attname)) |         param = smart_str(getattr(self, field.attname)) | ||||||
|         q = self.__class__._default_manager.filter(**kwargs).order_by((not is_next and '-' or '') + field.name, (not is_next and '-' or '') + self._meta.pk.name) |         q = self.__class__._default_manager.filter(**kwargs).order_by((not is_next and '-' or '') + field.name, (not is_next and '-' or '') + self._meta.pk.name) | ||||||
|         q._where.append(where) |         q._where.append(where) | ||||||
|         q._params.extend([param, param, getattr(self, self._meta.pk.attname)]) |         q._params.extend([param, param, getattr(self, self._meta.pk.attname)]) | ||||||
|   | |||||||
| @@ -8,7 +8,8 @@ from django.core.exceptions import ObjectDoesNotExist | |||||||
| from django.utils.functional import curry | from django.utils.functional import curry | ||||||
| from django.utils.itercompat import tee | from django.utils.itercompat import tee | ||||||
| from django.utils.text import capfirst | from django.utils.text import capfirst | ||||||
| from django.utils.translation import gettext, gettext_lazy | from django.utils.translation import ugettext_lazy, ugettext as _ | ||||||
|  | from django.utils.encoding import smart_unicode, force_unicode, smart_str | ||||||
| import datetime, os, time | import datetime, os, time | ||||||
| try: | try: | ||||||
|     import decimal |     import decimal | ||||||
| @@ -25,7 +26,7 @@ BLANK_CHOICE_DASH = [("", "---------")] | |||||||
| BLANK_CHOICE_NONE = [("", "None")] | BLANK_CHOICE_NONE = [("", "None")] | ||||||
|  |  | ||||||
| # prepares a value for use in a LIKE query | # prepares a value for use in a LIKE query | ||||||
| prep_for_like_query = lambda x: str(x).replace("\\", "\\\\").replace("%", "\%").replace("_", "\_") | prep_for_like_query = lambda x: smart_unicode(x).replace("\\", "\\\\").replace("%", "\%").replace("_", "\_") | ||||||
|  |  | ||||||
| # returns the <ul> class for a given radio_admin value | # returns the <ul> class for a given radio_admin value | ||||||
| get_ul_class = lambda x: 'radiolist%s' % ((x == HORIZONTAL) and ' inline' or '') | get_ul_class = lambda x: 'radiolist%s' % ((x == HORIZONTAL) and ' inline' or '') | ||||||
| @@ -42,7 +43,7 @@ def manipulator_validator_unique(f, opts, self, field_data, all_data): | |||||||
|         return |         return | ||||||
|     if getattr(self, 'original_object', None) and self.original_object._get_pk_val() == old_obj._get_pk_val(): |     if getattr(self, 'original_object', None) and self.original_object._get_pk_val() == old_obj._get_pk_val(): | ||||||
|         return |         return | ||||||
|     raise validators.ValidationError, gettext("%(optname)s with this %(fieldname)s already exists.") % {'optname': capfirst(opts.verbose_name), 'fieldname': f.verbose_name} |     raise validators.ValidationError, _("%(optname)s with this %(fieldname)s already exists.") % {'optname': capfirst(opts.verbose_name), 'fieldname': f.verbose_name} | ||||||
|  |  | ||||||
| # A guide to Field parameters: | # A guide to Field parameters: | ||||||
| # | # | ||||||
| @@ -121,7 +122,7 @@ class Field(object): | |||||||
|         Subclasses should implement validate(), not validate_full(). |         Subclasses should implement validate(), not validate_full(). | ||||||
|         """ |         """ | ||||||
|         if not self.blank and not field_data: |         if not self.blank and not field_data: | ||||||
|             return [gettext_lazy('This field is required.')] |             return [_('This field is required.')] | ||||||
|         try: |         try: | ||||||
|             self.validate(field_data, all_data) |             self.validate(field_data, all_data) | ||||||
|         except validators.ValidationError, e: |         except validators.ValidationError, e: | ||||||
| @@ -278,7 +279,7 @@ class Field(object): | |||||||
|                     core_field_names.extend(f.get_manipulator_field_names(name_prefix)) |                     core_field_names.extend(f.get_manipulator_field_names(name_prefix)) | ||||||
|             # Now, if there are any, add the validator to this FormField. |             # Now, if there are any, add the validator to this FormField. | ||||||
|             if core_field_names: |             if core_field_names: | ||||||
|                 params['validator_list'].append(validators.RequiredIfOtherFieldsGiven(core_field_names, gettext_lazy("This field is required."))) |                 params['validator_list'].append(validators.RequiredIfOtherFieldsGiven(core_field_names, ugettext_lazy("This field is required."))) | ||||||
|  |  | ||||||
|         # Finally, add the field_names. |         # Finally, add the field_names. | ||||||
|         field_names = self.get_manipulator_field_names(name_prefix) |         field_names = self.get_manipulator_field_names(name_prefix) | ||||||
| @@ -306,9 +307,9 @@ class Field(object): | |||||||
|             return first_choice + list(self.choices) |             return first_choice + list(self.choices) | ||||||
|         rel_model = self.rel.to |         rel_model = self.rel.to | ||||||
|         if hasattr(self.rel, 'get_related_field'): |         if hasattr(self.rel, 'get_related_field'): | ||||||
|             lst = [(getattr(x, self.rel.get_related_field().attname), str(x)) for x in rel_model._default_manager.complex_filter(self.rel.limit_choices_to)] |             lst = [(getattr(x, self.rel.get_related_field().attname), smart_unicode(x)) for x in rel_model._default_manager.complex_filter(self.rel.limit_choices_to)] | ||||||
|         else: |         else: | ||||||
|             lst = [(x._get_pk_val(), str(x)) for x in rel_model._default_manager.complex_filter(self.rel.limit_choices_to)] |             lst = [(x._get_pk_val(), smart_unicode(x)) for x in rel_model._default_manager.complex_filter(self.rel.limit_choices_to)] | ||||||
|         return first_choice + lst |         return first_choice + lst | ||||||
|  |  | ||||||
|     def get_choices_default(self): |     def get_choices_default(self): | ||||||
| @@ -373,7 +374,7 @@ class AutoField(Field): | |||||||
|         try: |         try: | ||||||
|             return int(value) |             return int(value) | ||||||
|         except (TypeError, ValueError): |         except (TypeError, ValueError): | ||||||
|             raise validators.ValidationError, gettext("This value must be an integer.") |             raise validators.ValidationError, _("This value must be an integer.") | ||||||
|  |  | ||||||
|     def get_manipulator_fields(self, opts, manipulator, change, name_prefix='', rel=False, follow=True): |     def get_manipulator_fields(self, opts, manipulator, change, name_prefix='', rel=False, follow=True): | ||||||
|         if not rel: |         if not rel: | ||||||
| @@ -408,7 +409,7 @@ class BooleanField(Field): | |||||||
|         if value in (True, False): return value |         if value in (True, False): return value | ||||||
|         if value in ('t', 'True', '1'): return True |         if value in ('t', 'True', '1'): return True | ||||||
|         if value in ('f', 'False', '0'): return False |         if value in ('f', 'False', '0'): return False | ||||||
|         raise validators.ValidationError, gettext("This value must be either True or False.") |         raise validators.ValidationError, _("This value must be either True or False.") | ||||||
|  |  | ||||||
|     def get_manipulator_field_objs(self): |     def get_manipulator_field_objs(self): | ||||||
|         return [oldforms.CheckboxField] |         return [oldforms.CheckboxField] | ||||||
| @@ -429,8 +430,8 @@ class CharField(Field): | |||||||
|             if self.null: |             if self.null: | ||||||
|                 return value |                 return value | ||||||
|             else: |             else: | ||||||
|                 raise validators.ValidationError, gettext_lazy("This field cannot be null.") |                 raise validators.ValidationError, ugettext_lazy("This field cannot be null.") | ||||||
|         return str(value) |         return smart_unicode(value) | ||||||
|  |  | ||||||
|     def formfield(self, **kwargs): |     def formfield(self, **kwargs): | ||||||
|         defaults = {'max_length': self.maxlength} |         defaults = {'max_length': self.maxlength} | ||||||
| @@ -463,15 +464,15 @@ class DateField(Field): | |||||||
|         try: |         try: | ||||||
|             return datetime.date(*time.strptime(value, '%Y-%m-%d')[:3]) |             return datetime.date(*time.strptime(value, '%Y-%m-%d')[:3]) | ||||||
|         except ValueError: |         except ValueError: | ||||||
|             raise validators.ValidationError, gettext('Enter a valid date in YYYY-MM-DD format.') |             raise validators.ValidationError, _('Enter a valid date in YYYY-MM-DD format.') | ||||||
|  |  | ||||||
|     def get_db_prep_lookup(self, lookup_type, value): |     def get_db_prep_lookup(self, lookup_type, value): | ||||||
|         if lookup_type == 'range': |         if lookup_type == 'range': | ||||||
|             value = [str(v) for v in value] |             value = [smart_unicode(v) for v in value] | ||||||
|         elif lookup_type in ('exact', 'gt', 'gte', 'lt', 'lte') and hasattr(value, 'strftime'): |         elif lookup_type in ('exact', 'gt', 'gte', 'lt', 'lte') and hasattr(value, 'strftime'): | ||||||
|             value = value.strftime('%Y-%m-%d') |             value = value.strftime('%Y-%m-%d') | ||||||
|         else: |         else: | ||||||
|             value = str(value) |             value = smart_unicode(value) | ||||||
|         return Field.get_db_prep_lookup(self, lookup_type, value) |         return Field.get_db_prep_lookup(self, lookup_type, value) | ||||||
|  |  | ||||||
|     def pre_save(self, model_instance, add): |     def pre_save(self, model_instance, add): | ||||||
| @@ -532,7 +533,7 @@ class DateTimeField(DateField): | |||||||
|                 try: |                 try: | ||||||
|                     return datetime.datetime(*time.strptime(value, '%Y-%m-%d')[:3]) |                     return datetime.datetime(*time.strptime(value, '%Y-%m-%d')[:3]) | ||||||
|                 except ValueError: |                 except ValueError: | ||||||
|                     raise validators.ValidationError, gettext('Enter a valid date/time in YYYY-MM-DD HH:MM format.') |                     raise validators.ValidationError, _('Enter a valid date/time in YYYY-MM-DD HH:MM format.') | ||||||
|  |  | ||||||
|     def get_db_prep_save(self, value): |     def get_db_prep_save(self, value): | ||||||
|         # Casts dates into string format for entry into database. |         # Casts dates into string format for entry into database. | ||||||
| @@ -541,14 +542,14 @@ class DateTimeField(DateField): | |||||||
|             # doesn't support microseconds. |             # doesn't support microseconds. | ||||||
|             if settings.DATABASE_ENGINE == 'mysql' and hasattr(value, 'microsecond'): |             if settings.DATABASE_ENGINE == 'mysql' and hasattr(value, 'microsecond'): | ||||||
|                 value = value.replace(microsecond=0) |                 value = value.replace(microsecond=0) | ||||||
|             value = str(value) |             value = smart_unicode(value) | ||||||
|         return Field.get_db_prep_save(self, value) |         return Field.get_db_prep_save(self, value) | ||||||
|  |  | ||||||
|     def get_db_prep_lookup(self, lookup_type, value): |     def get_db_prep_lookup(self, lookup_type, value): | ||||||
|         if lookup_type == 'range': |         if lookup_type == 'range': | ||||||
|             value = [str(v) for v in value] |             value = [smart_unicode(v) for v in value] | ||||||
|         else: |         else: | ||||||
|             value = str(value) |             value = smart_unicode(value) | ||||||
|         return Field.get_db_prep_lookup(self, lookup_type, value) |         return Field.get_db_prep_lookup(self, lookup_type, value) | ||||||
|  |  | ||||||
|     def get_manipulator_field_objs(self): |     def get_manipulator_field_objs(self): | ||||||
| @@ -592,7 +593,7 @@ class DecimalField(Field): | |||||||
|         try: |         try: | ||||||
|             return decimal.Decimal(value) |             return decimal.Decimal(value) | ||||||
|         except decimal.InvalidOperation: |         except decimal.InvalidOperation: | ||||||
|             raise validators.ValidationError, gettext("This value must be a decimal number.") |             raise validators.ValidationError, ugettext("This value must be a decimal number.") | ||||||
|  |  | ||||||
|     def _format(self, value): |     def _format(self, value): | ||||||
|         if isinstance(value, basestring): |         if isinstance(value, basestring): | ||||||
| @@ -613,7 +614,7 @@ class DecimalField(Field): | |||||||
|         if value < 0: |         if value < 0: | ||||||
|             num_chars += 1 |             num_chars += 1 | ||||||
|  |  | ||||||
|         return "%.*f" % (self.decimal_places, value) |         return u"%.*f" % (self.decimal_places, value) | ||||||
|  |  | ||||||
|     def get_db_prep_save(self, value): |     def get_db_prep_save(self, value): | ||||||
|         if value is not None: |         if value is not None: | ||||||
| @@ -675,7 +676,7 @@ class FileField(Field): | |||||||
|                         self.always_test = True |                         self.always_test = True | ||||||
|                     def __call__(self, field_data, all_data): |                     def __call__(self, field_data, all_data): | ||||||
|                         if not all_data.get(self.other_file_field_name, False): |                         if not all_data.get(self.other_file_field_name, False): | ||||||
|                             c = validators.RequiredIfOtherFieldsGiven(self.other_field_names, gettext_lazy("This field is required.")) |                             c = validators.RequiredIfOtherFieldsGiven(self.other_field_names, ugettext_lazy("This field is required.")) | ||||||
|                             c(field_data, all_data) |                             c(field_data, all_data) | ||||||
|                 # First, get the core fields, if any. |                 # First, get the core fields, if any. | ||||||
|                 core_field_names = [] |                 core_field_names = [] | ||||||
| @@ -686,7 +687,7 @@ class FileField(Field): | |||||||
|                 if core_field_names: |                 if core_field_names: | ||||||
|                     field_list[0].validator_list.append(RequiredFileField(core_field_names, field_list[1].field_name)) |                     field_list[0].validator_list.append(RequiredFileField(core_field_names, field_list[1].field_name)) | ||||||
|             else: |             else: | ||||||
|                 v = validators.RequiredIfOtherFieldNotGiven(field_list[1].field_name, gettext_lazy("This field is required.")) |                 v = validators.RequiredIfOtherFieldNotGiven(field_list[1].field_name, ugettext_lazy("This field is required.")) | ||||||
|                 v.always_test = True |                 v.always_test = True | ||||||
|                 field_list[0].validator_list.append(v) |                 field_list[0].validator_list.append(v) | ||||||
|                 field_list[0].is_required = field_list[1].is_required = False |                 field_list[0].is_required = field_list[1].is_required = False | ||||||
| @@ -732,7 +733,7 @@ class FileField(Field): | |||||||
|                 func(new_data[upload_field_name]["filename"], new_data[upload_field_name]["content"], save) |                 func(new_data[upload_field_name]["filename"], new_data[upload_field_name]["content"], save) | ||||||
|  |  | ||||||
|     def get_directory_name(self): |     def get_directory_name(self): | ||||||
|         return os.path.normpath(datetime.datetime.now().strftime(self.upload_to)) |         return os.path.normpath(force_unicode(datetime.datetime.now().strftime(smart_str(self.upload_to)))) | ||||||
|  |  | ||||||
|     def get_filename(self, filename): |     def get_filename(self, filename): | ||||||
|         from django.utils.text import get_valid_filename |         from django.utils.text import get_valid_filename | ||||||
| @@ -820,7 +821,7 @@ class NullBooleanField(Field): | |||||||
|         if value in ('None'): return None |         if value in ('None'): return None | ||||||
|         if value in ('t', 'True', '1'): return True |         if value in ('t', 'True', '1'): return True | ||||||
|         if value in ('f', 'False', '0'): return False |         if value in ('f', 'False', '0'): return False | ||||||
|         raise validators.ValidationError, gettext("This value must be either None, True or False.") |         raise validators.ValidationError, _("This value must be either None, True or False.") | ||||||
|  |  | ||||||
|     def get_manipulator_field_objs(self): |     def get_manipulator_field_objs(self): | ||||||
|         return [oldforms.NullBooleanField] |         return [oldforms.NullBooleanField] | ||||||
| @@ -890,9 +891,9 @@ class TimeField(Field): | |||||||
|             def prep(value): |             def prep(value): | ||||||
|                 if isinstance(value, datetime.time): |                 if isinstance(value, datetime.time): | ||||||
|                     value = datetime.datetime.combine(datetime.date(1900, 1, 1), value) |                     value = datetime.datetime.combine(datetime.date(1900, 1, 1), value) | ||||||
|                 return str(value) |                 return smart_unicode(value) | ||||||
|         else: |         else: | ||||||
|             prep = str |             prep = smart_unicode | ||||||
|         if lookup_type == 'range': |         if lookup_type == 'range': | ||||||
|             value = [prep(v) for v in value] |             value = [prep(v) for v in value] | ||||||
|         else: |         else: | ||||||
| @@ -922,7 +923,7 @@ class TimeField(Field): | |||||||
|                 elif isinstance(value, basestring): |                 elif isinstance(value, basestring): | ||||||
|                     value = datetime.datetime(*(time.strptime(value, '%H:%M:%S')[:6])) |                     value = datetime.datetime(*(time.strptime(value, '%H:%M:%S')[:6])) | ||||||
|             else: |             else: | ||||||
|                 value = str(value) |                 value = smart_unicode(value) | ||||||
|         return Field.get_db_prep_save(self, value) |         return Field.get_db_prep_save(self, value) | ||||||
|  |  | ||||||
|     def get_manipulator_field_objs(self): |     def get_manipulator_field_objs(self): | ||||||
|   | |||||||
| @@ -3,8 +3,9 @@ from django.db.models import signals, get_model | |||||||
| from django.db.models.fields import AutoField, Field, IntegerField, get_ul_class | from django.db.models.fields import AutoField, Field, IntegerField, get_ul_class | ||||||
| from django.db.models.related import RelatedObject | from django.db.models.related import RelatedObject | ||||||
| from django.utils.text import capfirst | from django.utils.text import capfirst | ||||||
| from django.utils.translation import gettext_lazy, string_concat, ngettext | from django.utils.translation import ugettext_lazy, string_concat, ungettext, ugettext as _ | ||||||
| from django.utils.functional import curry | from django.utils.functional import curry | ||||||
|  | from django.utils.encoding import smart_unicode | ||||||
| from django.core import validators | from django.core import validators | ||||||
| from django import oldforms | from django import oldforms | ||||||
| from django import newforms as forms | from django import newforms as forms | ||||||
| @@ -619,7 +620,7 @@ class ManyToManyField(RelatedField, Field): | |||||||
|         self.db_table = kwargs.pop('db_table', None) |         self.db_table = kwargs.pop('db_table', None) | ||||||
|         Field.__init__(self, **kwargs) |         Field.__init__(self, **kwargs) | ||||||
|  |  | ||||||
|         msg = gettext_lazy('Hold down "Control", or "Command" on a Mac, to select more than one.') |         msg = ugettext_lazy('Hold down "Control", or "Command" on a Mac, to select more than one.') | ||||||
|         self.help_text = string_concat(self.help_text, ' ', msg) |         self.help_text = string_concat(self.help_text, ' ', msg) | ||||||
|  |  | ||||||
|     def get_manipulator_field_objs(self): |     def get_manipulator_field_objs(self): | ||||||
| @@ -663,7 +664,7 @@ class ManyToManyField(RelatedField, Field): | |||||||
|         objects = mod._default_manager.in_bulk(pks) |         objects = mod._default_manager.in_bulk(pks) | ||||||
|         if len(objects) != len(pks): |         if len(objects) != len(pks): | ||||||
|             badkeys = [k for k in pks if k not in objects] |             badkeys = [k for k in pks if k not in objects] | ||||||
|             raise validators.ValidationError, ngettext("Please enter valid %(self)s IDs. The value %(value)r is invalid.", |             raise validators.ValidationError, ungettext("Please enter valid %(self)s IDs. The value %(value)r is invalid.", | ||||||
|                     "Please enter valid %(self)s IDs. The values %(value)r are invalid.", len(badkeys)) % { |                     "Please enter valid %(self)s IDs. The values %(value)r are invalid.", len(badkeys)) % { | ||||||
|                 'self': self.verbose_name, |                 'self': self.verbose_name, | ||||||
|                 'value': len(badkeys) == 1 and badkeys[0] or tuple(badkeys), |                 'value': len(badkeys) == 1 and badkeys[0] or tuple(badkeys), | ||||||
|   | |||||||
| @@ -7,6 +7,8 @@ from django.db.models import signals | |||||||
| from django.utils.functional import curry | from django.utils.functional import curry | ||||||
| from django.utils.datastructures import DotExpandedDict | from django.utils.datastructures import DotExpandedDict | ||||||
| from django.utils.text import capfirst | from django.utils.text import capfirst | ||||||
|  | from django.utils.encoding import smart_str | ||||||
|  | from django.utils.translation import ugettext as _ | ||||||
| import types | import types | ||||||
|  |  | ||||||
| def add_manipulators(sender): | def add_manipulators(sender): | ||||||
| @@ -111,7 +113,7 @@ class AutomaticManipulator(oldforms.Manipulator): | |||||||
|         if self.change: |         if self.change: | ||||||
|             self.fields_added, self.fields_changed, self.fields_deleted = [], [], [] |             self.fields_added, self.fields_changed, self.fields_deleted = [], [], [] | ||||||
|             for f in self.opts.fields: |             for f in self.opts.fields: | ||||||
|                 if not f.primary_key and str(getattr(self.original_object, f.attname)) != str(getattr(new_object, f.attname)): |                 if not f.primary_key and smart_str(getattr(self.original_object, f.attname)) != smart_str(getattr(new_object, f.attname)): | ||||||
|                     self.fields_changed.append(f.verbose_name) |                     self.fields_changed.append(f.verbose_name) | ||||||
|  |  | ||||||
|         # Save many-to-many objects. Example: Set sites for a poll. |         # Save many-to-many objects. Example: Set sites for a poll. | ||||||
| @@ -208,7 +210,7 @@ class AutomaticManipulator(oldforms.Manipulator): | |||||||
|                                 self.fields_added.append('%s "%s"' % (related.opts.verbose_name, new_rel_obj)) |                                 self.fields_added.append('%s "%s"' % (related.opts.verbose_name, new_rel_obj)) | ||||||
|                             else: |                             else: | ||||||
|                                 for f in related.opts.fields: |                                 for f in related.opts.fields: | ||||||
|                                     if not f.primary_key and f != related.field and str(getattr(old_rel_obj, f.attname)) != str(getattr(new_rel_obj, f.attname)): |                                     if not f.primary_key and f != related.field and smart_str(getattr(old_rel_obj, f.attname)) != smart_str(getattr(new_rel_obj, f.attname)): | ||||||
|                                         self.fields_changed.append('%s for %s "%s"' % (f.verbose_name, related.opts.verbose_name, new_rel_obj)) |                                         self.fields_changed.append('%s for %s "%s"' % (f.verbose_name, related.opts.verbose_name, new_rel_obj)) | ||||||
|  |  | ||||||
|                         # Save many-to-many objects. |                         # Save many-to-many objects. | ||||||
|   | |||||||
| @@ -4,6 +4,8 @@ from django.db.models.fields.related import ManyToManyRel | |||||||
| from django.db.models.fields import AutoField, FieldDoesNotExist | from django.db.models.fields import AutoField, FieldDoesNotExist | ||||||
| from django.db.models.loading import get_models | from django.db.models.loading import get_models | ||||||
| from django.db.models.query import orderlist2sql | from django.db.models.query import orderlist2sql | ||||||
|  | from django.utils.translation import activate, deactivate_all, get_language, string_concat | ||||||
|  | from django.utils.encoding import force_unicode, smart_str | ||||||
| from bisect import bisect | from bisect import bisect | ||||||
| import re | import re | ||||||
|  |  | ||||||
| @@ -41,6 +43,7 @@ class Options(object): | |||||||
|         self.object_name = cls.__name__ |         self.object_name = cls.__name__ | ||||||
|         self.module_name = self.object_name.lower() |         self.module_name = self.object_name.lower() | ||||||
|         self.verbose_name = get_verbose_name(self.object_name) |         self.verbose_name = get_verbose_name(self.object_name) | ||||||
|  |  | ||||||
|         # Next, apply any overridden values from 'class Meta'. |         # Next, apply any overridden values from 'class Meta'. | ||||||
|         if self.meta: |         if self.meta: | ||||||
|             meta_attrs = self.meta.__dict__ |             meta_attrs = self.meta.__dict__ | ||||||
| @@ -50,12 +53,12 @@ class Options(object): | |||||||
|                 setattr(self, attr_name, meta_attrs.pop(attr_name, getattr(self, attr_name))) |                 setattr(self, attr_name, meta_attrs.pop(attr_name, getattr(self, attr_name))) | ||||||
|             # verbose_name_plural is a special case because it uses a 's' |             # verbose_name_plural is a special case because it uses a 's' | ||||||
|             # by default. |             # by default. | ||||||
|             setattr(self, 'verbose_name_plural', meta_attrs.pop('verbose_name_plural', self.verbose_name + 's')) |             setattr(self, 'verbose_name_plural', meta_attrs.pop('verbose_name_plural', string_concat(self.verbose_name, 's'))) | ||||||
|             # Any leftover attributes must be invalid. |             # Any leftover attributes must be invalid. | ||||||
|             if meta_attrs != {}: |             if meta_attrs != {}: | ||||||
|                 raise TypeError, "'class Meta' got invalid attribute(s): %s" % ','.join(meta_attrs.keys()) |                 raise TypeError, "'class Meta' got invalid attribute(s): %s" % ','.join(meta_attrs.keys()) | ||||||
|         else: |         else: | ||||||
|             self.verbose_name_plural = self.verbose_name + 's' |             self.verbose_name_plural = string_concat(self.verbose_name, 's') | ||||||
|         del self.meta |         del self.meta | ||||||
|  |  | ||||||
|     def _prepare(self, model): |     def _prepare(self, model): | ||||||
| @@ -94,7 +97,20 @@ class Options(object): | |||||||
|         return '<Options for %s>' % self.object_name |         return '<Options for %s>' % self.object_name | ||||||
|  |  | ||||||
|     def __str__(self): |     def __str__(self): | ||||||
|         return "%s.%s" % (self.app_label, self.module_name) |         return "%s.%s" % (smart_str(self.app_label), smart_str(self.module_name)) | ||||||
|  |  | ||||||
|  |     def verbose_name_raw(self): | ||||||
|  |         """ | ||||||
|  |         There are a few places where the untranslated verbose name is needed | ||||||
|  |         (so that we get the same value regardless of currently active | ||||||
|  |         locale). | ||||||
|  |         """ | ||||||
|  |         lang = get_language() | ||||||
|  |         deactivate_all() | ||||||
|  |         raw = force_unicode(self.verbose_name) | ||||||
|  |         activate(lang) | ||||||
|  |         return raw | ||||||
|  |     verbose_name_raw = property(verbose_name_raw) | ||||||
|  |  | ||||||
|     def get_field(self, name, many_to_many=True): |     def get_field(self, name, many_to_many=True): | ||||||
|         "Returns the requested field by name. Raises FieldDoesNotExist on error." |         "Returns the requested field by name. Raises FieldDoesNotExist on error." | ||||||
|   | |||||||
| @@ -4,6 +4,7 @@ from django.db.models.fields import DateField, FieldDoesNotExist | |||||||
| from django.db.models import signals, loading | from django.db.models import signals, loading | ||||||
| from django.dispatch import dispatcher | from django.dispatch import dispatcher | ||||||
| from django.utils.datastructures import SortedDict | from django.utils.datastructures import SortedDict | ||||||
|  | from django.utils.encoding import smart_unicode | ||||||
| from django.contrib.contenttypes import generic | from django.contrib.contenttypes import generic | ||||||
| import datetime | import datetime | ||||||
| import operator | import operator | ||||||
| @@ -52,7 +53,7 @@ def handle_legacy_orderlist(order_list): | |||||||
|         return order_list |         return order_list | ||||||
|     else: |     else: | ||||||
|         import warnings |         import warnings | ||||||
|         new_order_list = [LEGACY_ORDERING_MAPPING[j.upper()].replace('_', str(i)) for i, j in order_list] |         new_order_list = [LEGACY_ORDERING_MAPPING[j.upper()].replace('_', smart_unicode(i)) for i, j in order_list] | ||||||
|         warnings.warn("%r ordering syntax is deprecated. Use %r instead." % (order_list, new_order_list), DeprecationWarning) |         warnings.warn("%r ordering syntax is deprecated. Use %r instead." % (order_list, new_order_list), DeprecationWarning) | ||||||
|         return new_order_list |         return new_order_list | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,8 +1,9 @@ | |||||||
| import os | import os | ||||||
| from Cookie import SimpleCookie | from Cookie import SimpleCookie | ||||||
| from pprint import pformat | from pprint import pformat | ||||||
| from urllib import urlencode, quote | from urllib import urlencode | ||||||
| from django.utils.datastructures import MultiValueDict | from django.utils.datastructures import MultiValueDict | ||||||
|  | from django.utils.encoding import smart_str, iri_to_uri, force_unicode | ||||||
|  |  | ||||||
| RESERVED_CHARS="!*'();:@&=+$,/?%#[]" | RESERVED_CHARS="!*'();:@&=+$,/?%#[]" | ||||||
|  |  | ||||||
| @@ -17,6 +18,10 @@ class Http404(Exception): | |||||||
|  |  | ||||||
| class HttpRequest(object): | class HttpRequest(object): | ||||||
|     "A basic HTTP request" |     "A basic HTTP request" | ||||||
|  |  | ||||||
|  |     # The encoding used in GET/POST dicts. None means use default setting. | ||||||
|  |     _encoding = None | ||||||
|  |  | ||||||
|     def __init__(self): |     def __init__(self): | ||||||
|         self.GET, self.POST, self.COOKIES, self.META, self.FILES = {}, {}, {}, {}, {} |         self.GET, self.POST, self.COOKIES, self.META, self.FILES = {}, {}, {}, {}, {} | ||||||
|         self.path = '' |         self.path = '' | ||||||
| @@ -42,14 +47,31 @@ class HttpRequest(object): | |||||||
|     def is_secure(self): |     def is_secure(self): | ||||||
|         return os.environ.get("HTTPS") == "on" |         return os.environ.get("HTTPS") == "on" | ||||||
|  |  | ||||||
|  |     def _set_encoding(self, val): | ||||||
|  |         """ | ||||||
|  |         Sets the encoding used for GET/POST accesses. If the GET or POST | ||||||
|  |         dictionary has already been created, it is removed and recreated on the | ||||||
|  |         next access (so that it is decoded correctly). | ||||||
|  |         """ | ||||||
|  |         self._encoding = val | ||||||
|  |         if hasattr(self, '_get'): | ||||||
|  |             del self._get | ||||||
|  |         if hasattr(self, '_post'): | ||||||
|  |             del self._post | ||||||
|  |  | ||||||
|  |     def _get_encoding(self): | ||||||
|  |         return self._encoding | ||||||
|  |  | ||||||
|  |     encoding = property(_get_encoding, _set_encoding) | ||||||
|  |  | ||||||
| def parse_file_upload(header_dict, post_data): | def parse_file_upload(header_dict, post_data): | ||||||
|     "Returns a tuple of (POST MultiValueDict, FILES MultiValueDict)" |     "Returns a tuple of (POST QueryDict, FILES MultiValueDict)" | ||||||
|     import email, email.Message |     import email, email.Message | ||||||
|     from cgi import parse_header |     from cgi import parse_header | ||||||
|     raw_message = '\r\n'.join(['%s:%s' % pair for pair in header_dict.items()]) |     raw_message = '\r\n'.join(['%s:%s' % pair for pair in header_dict.items()]) | ||||||
|     raw_message += '\r\n\r\n' + post_data |     raw_message += '\r\n\r\n' + post_data | ||||||
|     msg = email.message_from_string(raw_message) |     msg = email.message_from_string(raw_message) | ||||||
|     POST = MultiValueDict() |     POST = QueryDict('', mutable=True) | ||||||
|     FILES = MultiValueDict() |     FILES = MultiValueDict() | ||||||
|     for submessage in msg.get_payload(): |     for submessage in msg.get_payload(): | ||||||
|         if submessage and isinstance(submessage, email.Message.Message): |         if submessage and isinstance(submessage, email.Message.Message): | ||||||
| @@ -74,13 +96,25 @@ def parse_file_upload(header_dict, post_data): | |||||||
|     return POST, FILES |     return POST, FILES | ||||||
|  |  | ||||||
| class QueryDict(MultiValueDict): | class QueryDict(MultiValueDict): | ||||||
|     """A specialized MultiValueDict that takes a query string when initialized. |     """ | ||||||
|     This is immutable unless you create a copy of it.""" |     A specialized MultiValueDict that takes a query string when initialized. | ||||||
|     def __init__(self, query_string, mutable=False): |     This is immutable unless you create a copy of it. | ||||||
|  |  | ||||||
|  |     Values retrieved from this class are converted from the given encoding | ||||||
|  |     (DEFAULT_CHARSET by default) to unicode. | ||||||
|  |     """ | ||||||
|  |     def __init__(self, query_string, mutable=False, encoding=None): | ||||||
|         MultiValueDict.__init__(self) |         MultiValueDict.__init__(self) | ||||||
|  |         if not encoding: | ||||||
|  |             # *Important*: do not import settings any earlier because of note | ||||||
|  |             # in core.handlers.modpython. | ||||||
|  |             from django.conf import settings | ||||||
|  |             self.encoding = settings.DEFAULT_CHARSET | ||||||
|  |         else: | ||||||
|  |             self.encoding = encoding | ||||||
|         self._mutable = True |         self._mutable = True | ||||||
|         for key, value in parse_qsl((query_string or ''), True): # keep_blank_values=True |         for key, value in parse_qsl((query_string or ''), True): # keep_blank_values=True | ||||||
|             self.appendlist(key, value) |             self.appendlist(force_unicode(key, errors='replace'), force_unicode(value, errors='replace')) | ||||||
|         self._mutable = mutable |         self._mutable = mutable | ||||||
|  |  | ||||||
|     def _assert_mutable(self): |     def _assert_mutable(self): | ||||||
| @@ -89,6 +123,8 @@ class QueryDict(MultiValueDict): | |||||||
|  |  | ||||||
|     def __setitem__(self, key, value): |     def __setitem__(self, key, value): | ||||||
|         self._assert_mutable() |         self._assert_mutable() | ||||||
|  |         key = str_to_unicode(key, self.encoding) | ||||||
|  |         value = str_to_unicode(value, self.encoding) | ||||||
|         MultiValueDict.__setitem__(self, key, value) |         MultiValueDict.__setitem__(self, key, value) | ||||||
|  |  | ||||||
|     def __delitem__(self, key): |     def __delitem__(self, key): | ||||||
| @@ -111,15 +147,27 @@ class QueryDict(MultiValueDict): | |||||||
|  |  | ||||||
|     def setlist(self, key, list_): |     def setlist(self, key, list_): | ||||||
|         self._assert_mutable() |         self._assert_mutable() | ||||||
|  |         key = str_to_unicode(key, self.encoding) | ||||||
|  |         list_ = [str_to_unicode(elt, self.encoding) for elt in list_] | ||||||
|         MultiValueDict.setlist(self, key, list_) |         MultiValueDict.setlist(self, key, list_) | ||||||
|  |  | ||||||
|  |     def setlistdefault(self, key, default_list=()): | ||||||
|  |         self._assert_mutable() | ||||||
|  |         if key not in self: | ||||||
|  |             self.setlist(key, default_list) | ||||||
|  |         return MultiValueDict.getlist(self, key) | ||||||
|  |  | ||||||
|     def appendlist(self, key, value): |     def appendlist(self, key, value): | ||||||
|         self._assert_mutable() |         self._assert_mutable() | ||||||
|  |         key = str_to_unicode(key, self.encoding) | ||||||
|  |         value = str_to_unicode(value, self.encoding) | ||||||
|         MultiValueDict.appendlist(self, key, value) |         MultiValueDict.appendlist(self, key, value) | ||||||
|  |  | ||||||
|     def update(self, other_dict): |     def update(self, other_dict): | ||||||
|         self._assert_mutable() |         self._assert_mutable() | ||||||
|         MultiValueDict.update(self, other_dict) |         f = lambda s: str_to_unicode(s, self.encoding) | ||||||
|  |         d = dict([(f(k), f(v)) for k, v in other_dict.items()]) | ||||||
|  |         MultiValueDict.update(self, d) | ||||||
|  |  | ||||||
|     def pop(self, key, *args): |     def pop(self, key, *args): | ||||||
|         self._assert_mutable() |         self._assert_mutable() | ||||||
| @@ -133,9 +181,11 @@ class QueryDict(MultiValueDict): | |||||||
|         self._assert_mutable() |         self._assert_mutable() | ||||||
|         MultiValueDict.clear(self) |         MultiValueDict.clear(self) | ||||||
|  |  | ||||||
|     def setdefault(self, *args): |     def setdefault(self, key, default=None): | ||||||
|         self._assert_mutable() |         self._assert_mutable() | ||||||
|         return MultiValueDict.setdefault(self, *args) |         key = str_to_unicode(key, self.encoding) | ||||||
|  |         default = str_to_unicode(default, self.encoding) | ||||||
|  |         return MultiValueDict.setdefault(self, key, default) | ||||||
|  |  | ||||||
|     def copy(self): |     def copy(self): | ||||||
|         "Returns a mutable copy of this object." |         "Returns a mutable copy of this object." | ||||||
| @@ -144,7 +194,8 @@ class QueryDict(MultiValueDict): | |||||||
|     def urlencode(self): |     def urlencode(self): | ||||||
|         output = [] |         output = [] | ||||||
|         for k, list_ in self.lists(): |         for k, list_ in self.lists(): | ||||||
|             output.extend([urlencode({k: v}) for v in list_]) |             k = smart_str(k, self.encoding) | ||||||
|  |             output.extend([urlencode({k: smart_str(v, self.encoding)}) for v in list_]) | ||||||
|         return '&'.join(output) |         return '&'.join(output) | ||||||
|  |  | ||||||
| def parse_cookie(cookie): | def parse_cookie(cookie): | ||||||
| @@ -221,9 +272,7 @@ class HttpResponse(object): | |||||||
|         self.cookies[key]['max-age'] = 0 |         self.cookies[key]['max-age'] = 0 | ||||||
|  |  | ||||||
|     def _get_content(self): |     def _get_content(self): | ||||||
|         content = ''.join(self._container) |         content = smart_str(''.join(self._container), self._charset) | ||||||
|         if isinstance(content, unicode): |  | ||||||
|             content = content.encode(self._charset) |  | ||||||
|         return content |         return content | ||||||
|  |  | ||||||
|     def _set_content(self, value): |     def _set_content(self, value): | ||||||
| @@ -266,14 +315,14 @@ class HttpResponseRedirect(HttpResponse): | |||||||
|  |  | ||||||
|     def __init__(self, redirect_to): |     def __init__(self, redirect_to): | ||||||
|         HttpResponse.__init__(self) |         HttpResponse.__init__(self) | ||||||
|         self['Location'] = quote(redirect_to, safe=RESERVED_CHARS) |         self['Location'] = iri_to_uri(redirect_to) | ||||||
|  |  | ||||||
| class HttpResponsePermanentRedirect(HttpResponse): | class HttpResponsePermanentRedirect(HttpResponse): | ||||||
|     status_code = 301 |     status_code = 301 | ||||||
|  |  | ||||||
|     def __init__(self, redirect_to): |     def __init__(self, redirect_to): | ||||||
|         HttpResponse.__init__(self) |         HttpResponse.__init__(self) | ||||||
|         self['Location'] = quote(redirect_to, safe=RESERVED_CHARS) |         self['Location'] = iri_to_uri(redirect_to) | ||||||
|  |  | ||||||
| class HttpResponseNotModified(HttpResponse): | class HttpResponseNotModified(HttpResponse): | ||||||
|     status_code = 304 |     status_code = 304 | ||||||
| @@ -312,3 +361,20 @@ def get_host(request): | |||||||
|     if not host: |     if not host: | ||||||
|         host = request.META.get('HTTP_HOST', '') |         host = request.META.get('HTTP_HOST', '') | ||||||
|     return host |     return host | ||||||
|  |  | ||||||
|  | # It's neither necessary nor appropriate to use | ||||||
|  | # django.utils.encoding.smart_unicode for parsing URLs and form inputs. Thus, | ||||||
|  | # this slightly more restricted function. | ||||||
|  | def str_to_unicode(s, encoding): | ||||||
|  |     """ | ||||||
|  |     Convert basestring objects to unicode, using the given encoding. Illegaly | ||||||
|  |     encoded input characters are replaced with Unicode "unknown" codepoint | ||||||
|  |     (\ufffd). | ||||||
|  |  | ||||||
|  |     Returns any non-basestring objects without change. | ||||||
|  |     """ | ||||||
|  |     if isinstance(s, str): | ||||||
|  |         return unicode(s, encoding, 'replace') | ||||||
|  |     else: | ||||||
|  |         return s | ||||||
|  |  | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ import datetime | |||||||
| import re | import re | ||||||
| import time | import time | ||||||
|  |  | ||||||
| from django.utils.translation import gettext | from django.utils.translation import ugettext | ||||||
| from django.utils.encoding import smart_unicode | from django.utils.encoding import smart_unicode | ||||||
|  |  | ||||||
| from util import ErrorList, ValidationError | from util import ErrorList, ValidationError | ||||||
| @@ -84,7 +84,7 @@ class Field(object): | |||||||
|         Raises ValidationError for any errors. |         Raises ValidationError for any errors. | ||||||
|         """ |         """ | ||||||
|         if self.required and value in EMPTY_VALUES: |         if self.required and value in EMPTY_VALUES: | ||||||
|             raise ValidationError(gettext(u'This field is required.')) |             raise ValidationError(ugettext(u'This field is required.')) | ||||||
|         return value |         return value | ||||||
|  |  | ||||||
|     def widget_attrs(self, widget): |     def widget_attrs(self, widget): | ||||||
| @@ -107,9 +107,9 @@ class CharField(Field): | |||||||
|             return u'' |             return u'' | ||||||
|         value = smart_unicode(value) |         value = smart_unicode(value) | ||||||
|         if self.max_length is not None and len(value) > self.max_length: |         if self.max_length is not None and len(value) > self.max_length: | ||||||
|             raise ValidationError(gettext(u'Ensure this value has at most %d characters.') % self.max_length) |             raise ValidationError(ugettext(u'Ensure this value has at most %d characters.') % self.max_length) | ||||||
|         if self.min_length is not None and len(value) < self.min_length: |         if self.min_length is not None and len(value) < self.min_length: | ||||||
|             raise ValidationError(gettext(u'Ensure this value has at least %d characters.') % self.min_length) |             raise ValidationError(ugettext(u'Ensure this value has at least %d characters.') % self.min_length) | ||||||
|         return value |         return value | ||||||
|  |  | ||||||
|     def widget_attrs(self, widget): |     def widget_attrs(self, widget): | ||||||
| @@ -132,11 +132,11 @@ class IntegerField(Field): | |||||||
|         try: |         try: | ||||||
|             value = int(value) |             value = int(value) | ||||||
|         except (ValueError, TypeError): |         except (ValueError, TypeError): | ||||||
|             raise ValidationError(gettext(u'Enter a whole number.')) |             raise ValidationError(ugettext(u'Enter a whole number.')) | ||||||
|         if self.max_value is not None and value > self.max_value: |         if self.max_value is not None and value > self.max_value: | ||||||
|             raise ValidationError(gettext(u'Ensure this value is less than or equal to %s.') % self.max_value) |             raise ValidationError(ugettext(u'Ensure this value is less than or equal to %s.') % self.max_value) | ||||||
|         if self.min_value is not None and value < self.min_value: |         if self.min_value is not None and value < self.min_value: | ||||||
|             raise ValidationError(gettext(u'Ensure this value is greater than or equal to %s.') % self.min_value) |             raise ValidationError(ugettext(u'Ensure this value is greater than or equal to %s.') % self.min_value) | ||||||
|         return value |         return value | ||||||
|  |  | ||||||
| class FloatField(Field): | class FloatField(Field): | ||||||
| @@ -155,11 +155,11 @@ class FloatField(Field): | |||||||
|         try: |         try: | ||||||
|             value = float(value) |             value = float(value) | ||||||
|         except (ValueError, TypeError): |         except (ValueError, TypeError): | ||||||
|             raise ValidationError(gettext('Enter a number.')) |             raise ValidationError(ugettext('Enter a number.')) | ||||||
|         if self.max_value is not None and value > self.max_value: |         if self.max_value is not None and value > self.max_value: | ||||||
|             raise ValidationError(gettext('Ensure this value is less than or equal to %s.') % self.max_value) |             raise ValidationError(ugettext('Ensure this value is less than or equal to %s.') % self.max_value) | ||||||
|         if self.min_value is not None and value < self.min_value: |         if self.min_value is not None and value < self.min_value: | ||||||
|             raise ValidationError(gettext('Ensure this value is greater than or equal to %s.') % self.min_value) |             raise ValidationError(ugettext('Ensure this value is greater than or equal to %s.') % self.min_value) | ||||||
|         return value |         return value | ||||||
|  |  | ||||||
| decimal_re = re.compile(r'^-?(?P<digits>\d+)(\.(?P<decimals>\d+))?$') | decimal_re = re.compile(r'^-?(?P<digits>\d+)(\.(?P<decimals>\d+))?$') | ||||||
| @@ -183,21 +183,21 @@ class DecimalField(Field): | |||||||
|         value = value.strip() |         value = value.strip() | ||||||
|         match = decimal_re.search(value) |         match = decimal_re.search(value) | ||||||
|         if not match: |         if not match: | ||||||
|             raise ValidationError(gettext('Enter a number.')) |             raise ValidationError(ugettext('Enter a number.')) | ||||||
|         else: |         else: | ||||||
|             value = Decimal(value) |             value = Decimal(value) | ||||||
|         digits = len(match.group('digits') or '') |         digits = len(match.group('digits') or '') | ||||||
|         decimals = len(match.group('decimals') or '') |         decimals = len(match.group('decimals') or '') | ||||||
|         if self.max_value is not None and value > self.max_value: |         if self.max_value is not None and value > self.max_value: | ||||||
|             raise ValidationError(gettext('Ensure this value is less than or equal to %s.') % self.max_value) |             raise ValidationError(ugettext('Ensure this value is less than or equal to %s.') % self.max_value) | ||||||
|         if self.min_value is not None and value < self.min_value: |         if self.min_value is not None and value < self.min_value: | ||||||
|             raise ValidationError(gettext('Ensure this value is greater than or equal to %s.') % self.min_value) |             raise ValidationError(ugettext('Ensure this value is greater than or equal to %s.') % self.min_value) | ||||||
|         if self.max_digits is not None and (digits + decimals) > self.max_digits: |         if self.max_digits is not None and (digits + decimals) > self.max_digits: | ||||||
|             raise ValidationError(gettext('Ensure that there are no more than %s digits in total.') % self.max_digits) |             raise ValidationError(ugettext('Ensure that there are no more than %s digits in total.') % self.max_digits) | ||||||
|         if self.decimal_places is not None and decimals > self.decimal_places: |         if self.decimal_places is not None and decimals > self.decimal_places: | ||||||
|             raise ValidationError(gettext('Ensure that there are no more than %s decimal places.') % self.decimal_places) |             raise ValidationError(ugettext('Ensure that there are no more than %s decimal places.') % self.decimal_places) | ||||||
|         if self.max_digits is not None and self.decimal_places is not None and digits > (self.max_digits - self.decimal_places): |         if self.max_digits is not None and self.decimal_places is not None and digits > (self.max_digits - self.decimal_places): | ||||||
|             raise ValidationError(gettext('Ensure that there are no more than %s digits before the decimal point.') % (self.max_digits - self.decimal_places)) |             raise ValidationError(ugettext('Ensure that there are no more than %s digits before the decimal point.') % (self.max_digits - self.decimal_places)) | ||||||
|         return value |         return value | ||||||
|  |  | ||||||
| DEFAULT_DATE_INPUT_FORMATS = ( | DEFAULT_DATE_INPUT_FORMATS = ( | ||||||
| @@ -230,7 +230,7 @@ class DateField(Field): | |||||||
|                 return datetime.date(*time.strptime(value, format)[:3]) |                 return datetime.date(*time.strptime(value, format)[:3]) | ||||||
|             except ValueError: |             except ValueError: | ||||||
|                 continue |                 continue | ||||||
|         raise ValidationError(gettext(u'Enter a valid date.')) |         raise ValidationError(ugettext(u'Enter a valid date.')) | ||||||
|  |  | ||||||
| DEFAULT_TIME_INPUT_FORMATS = ( | DEFAULT_TIME_INPUT_FORMATS = ( | ||||||
|     '%H:%M:%S',     # '14:30:59' |     '%H:%M:%S',     # '14:30:59' | ||||||
| @@ -257,7 +257,7 @@ class TimeField(Field): | |||||||
|                 return datetime.time(*time.strptime(value, format)[3:6]) |                 return datetime.time(*time.strptime(value, format)[3:6]) | ||||||
|             except ValueError: |             except ValueError: | ||||||
|                 continue |                 continue | ||||||
|         raise ValidationError(gettext(u'Enter a valid time.')) |         raise ValidationError(ugettext(u'Enter a valid time.')) | ||||||
|  |  | ||||||
| DEFAULT_DATETIME_INPUT_FORMATS = ( | DEFAULT_DATETIME_INPUT_FORMATS = ( | ||||||
|     '%Y-%m-%d %H:%M:%S',     # '2006-10-25 14:30:59' |     '%Y-%m-%d %H:%M:%S',     # '2006-10-25 14:30:59' | ||||||
| @@ -293,7 +293,7 @@ class DateTimeField(Field): | |||||||
|                 return datetime.datetime(*time.strptime(value, format)[:6]) |                 return datetime.datetime(*time.strptime(value, format)[:6]) | ||||||
|             except ValueError: |             except ValueError: | ||||||
|                 continue |                 continue | ||||||
|         raise ValidationError(gettext(u'Enter a valid date/time.')) |         raise ValidationError(ugettext(u'Enter a valid date/time.')) | ||||||
|  |  | ||||||
| class RegexField(Field): | class RegexField(Field): | ||||||
|     def __init__(self, regex, max_length=None, min_length=None, error_message=None, *args, **kwargs): |     def __init__(self, regex, max_length=None, min_length=None, error_message=None, *args, **kwargs): | ||||||
| @@ -307,7 +307,7 @@ class RegexField(Field): | |||||||
|             regex = re.compile(regex) |             regex = re.compile(regex) | ||||||
|         self.regex = regex |         self.regex = regex | ||||||
|         self.max_length, self.min_length = max_length, min_length |         self.max_length, self.min_length = max_length, min_length | ||||||
|         self.error_message = error_message or gettext(u'Enter a valid value.') |         self.error_message = error_message or ugettext(u'Enter a valid value.') | ||||||
|  |  | ||||||
|     def clean(self, value): |     def clean(self, value): | ||||||
|         """ |         """ | ||||||
| @@ -321,9 +321,9 @@ class RegexField(Field): | |||||||
|         if value == u'': |         if value == u'': | ||||||
|             return value |             return value | ||||||
|         if self.max_length is not None and len(value) > self.max_length: |         if self.max_length is not None and len(value) > self.max_length: | ||||||
|             raise ValidationError(gettext(u'Ensure this value has at most %d characters.') % self.max_length) |             raise ValidationError(ugettext(u'Ensure this value has at most %d characters.') % self.max_length) | ||||||
|         if self.min_length is not None and len(value) < self.min_length: |         if self.min_length is not None and len(value) < self.min_length: | ||||||
|             raise ValidationError(gettext(u'Ensure this value has at least %d characters.') % self.min_length) |             raise ValidationError(ugettext(u'Ensure this value has at least %d characters.') % self.min_length) | ||||||
|         if not self.regex.search(value): |         if not self.regex.search(value): | ||||||
|             raise ValidationError(self.error_message) |             raise ValidationError(self.error_message) | ||||||
|         return value |         return value | ||||||
| @@ -336,7 +336,7 @@ email_re = re.compile( | |||||||
| class EmailField(RegexField): | class EmailField(RegexField): | ||||||
|     def __init__(self, max_length=None, min_length=None, *args, **kwargs): |     def __init__(self, max_length=None, min_length=None, *args, **kwargs): | ||||||
|         RegexField.__init__(self, email_re, max_length, min_length, |         RegexField.__init__(self, email_re, max_length, min_length, | ||||||
|             gettext(u'Enter a valid e-mail address.'), *args, **kwargs) |             ugettext(u'Enter a valid e-mail address.'), *args, **kwargs) | ||||||
|  |  | ||||||
| url_re = re.compile( | url_re = re.compile( | ||||||
|     r'^https?://' # http:// or https:// |     r'^https?://' # http:// or https:// | ||||||
| @@ -354,7 +354,7 @@ except ImportError: | |||||||
| class URLField(RegexField): | class URLField(RegexField): | ||||||
|     def __init__(self, max_length=None, min_length=None, verify_exists=False, |     def __init__(self, max_length=None, min_length=None, verify_exists=False, | ||||||
|             validator_user_agent=URL_VALIDATOR_USER_AGENT, *args, **kwargs): |             validator_user_agent=URL_VALIDATOR_USER_AGENT, *args, **kwargs): | ||||||
|         super(URLField, self).__init__(url_re, max_length, min_length, gettext(u'Enter a valid URL.'), *args, **kwargs) |         super(URLField, self).__init__(url_re, max_length, min_length, ugettext(u'Enter a valid URL.'), *args, **kwargs) | ||||||
|         self.verify_exists = verify_exists |         self.verify_exists = verify_exists | ||||||
|         self.user_agent = validator_user_agent |         self.user_agent = validator_user_agent | ||||||
|  |  | ||||||
| @@ -376,9 +376,9 @@ class URLField(RegexField): | |||||||
|                 req = urllib2.Request(value, None, headers) |                 req = urllib2.Request(value, None, headers) | ||||||
|                 u = urllib2.urlopen(req) |                 u = urllib2.urlopen(req) | ||||||
|             except ValueError: |             except ValueError: | ||||||
|                 raise ValidationError(gettext(u'Enter a valid URL.')) |                 raise ValidationError(ugettext(u'Enter a valid URL.')) | ||||||
|             except: # urllib2.URLError, httplib.InvalidURL, etc. |             except: # urllib2.URLError, httplib.InvalidURL, etc. | ||||||
|                 raise ValidationError(gettext(u'This URL appears to be a broken link.')) |                 raise ValidationError(ugettext(u'This URL appears to be a broken link.')) | ||||||
|         return value |         return value | ||||||
|  |  | ||||||
| class BooleanField(Field): | class BooleanField(Field): | ||||||
| @@ -427,9 +427,9 @@ class ChoiceField(Field): | |||||||
|         value = smart_unicode(value) |         value = smart_unicode(value) | ||||||
|         if value == u'': |         if value == u'': | ||||||
|             return value |             return value | ||||||
|         valid_values = set([str(k) for k, v in self.choices]) |         valid_values = set([smart_unicode(k) for k, v in self.choices]) | ||||||
|         if value not in valid_values: |         if value not in valid_values: | ||||||
|             raise ValidationError(gettext(u'Select a valid choice. That choice is not one of the available choices.')) |             raise ValidationError(ugettext(u'Select a valid choice. That choice is not one of the available choices.')) | ||||||
|         return value |         return value | ||||||
|  |  | ||||||
| class MultipleChoiceField(ChoiceField): | class MultipleChoiceField(ChoiceField): | ||||||
| @@ -441,11 +441,11 @@ class MultipleChoiceField(ChoiceField): | |||||||
|         Validates that the input is a list or tuple. |         Validates that the input is a list or tuple. | ||||||
|         """ |         """ | ||||||
|         if self.required and not value: |         if self.required and not value: | ||||||
|             raise ValidationError(gettext(u'This field is required.')) |             raise ValidationError(ugettext(u'This field is required.')) | ||||||
|         elif not self.required and not value: |         elif not self.required and not value: | ||||||
|             return [] |             return [] | ||||||
|         if not isinstance(value, (list, tuple)): |         if not isinstance(value, (list, tuple)): | ||||||
|             raise ValidationError(gettext(u'Enter a list of values.')) |             raise ValidationError(ugettext(u'Enter a list of values.')) | ||||||
|         new_value = [] |         new_value = [] | ||||||
|         for val in value: |         for val in value: | ||||||
|             val = smart_unicode(val) |             val = smart_unicode(val) | ||||||
| @@ -454,7 +454,7 @@ class MultipleChoiceField(ChoiceField): | |||||||
|         valid_values = set([smart_unicode(k) for k, v in self.choices]) |         valid_values = set([smart_unicode(k) for k, v in self.choices]) | ||||||
|         for val in new_value: |         for val in new_value: | ||||||
|             if val not in valid_values: |             if val not in valid_values: | ||||||
|                 raise ValidationError(gettext(u'Select a valid choice. %s is not one of the available choices.') % val) |                 raise ValidationError(ugettext(u'Select a valid choice. %s is not one of the available choices.') % val) | ||||||
|         return new_value |         return new_value | ||||||
|  |  | ||||||
| class ComboField(Field): | class ComboField(Field): | ||||||
| @@ -520,18 +520,18 @@ class MultiValueField(Field): | |||||||
|         if not value or isinstance(value, (list, tuple)): |         if not value or isinstance(value, (list, tuple)): | ||||||
|             if not value or not [v for v in value if v not in EMPTY_VALUES]: |             if not value or not [v for v in value if v not in EMPTY_VALUES]: | ||||||
|                 if self.required: |                 if self.required: | ||||||
|                     raise ValidationError(gettext(u'This field is required.')) |                     raise ValidationError(ugettext(u'This field is required.')) | ||||||
|                 else: |                 else: | ||||||
|                     return self.compress([]) |                     return self.compress([]) | ||||||
|         else: |         else: | ||||||
|             raise ValidationError(gettext(u'Enter a list of values.')) |             raise ValidationError(ugettext(u'Enter a list of values.')) | ||||||
|         for i, field in enumerate(self.fields): |         for i, field in enumerate(self.fields): | ||||||
|             try: |             try: | ||||||
|                 field_value = value[i] |                 field_value = value[i] | ||||||
|             except IndexError: |             except IndexError: | ||||||
|                 field_value = None |                 field_value = None | ||||||
|             if self.required and field_value in EMPTY_VALUES: |             if self.required and field_value in EMPTY_VALUES: | ||||||
|                 raise ValidationError(gettext(u'This field is required.')) |                 raise ValidationError(ugettext(u'This field is required.')) | ||||||
|             try: |             try: | ||||||
|                 clean_data.append(field.clean(field_value)) |                 clean_data.append(field.clean(field_value)) | ||||||
|             except ValidationError, e: |             except ValidationError, e: | ||||||
| @@ -564,8 +564,8 @@ class SplitDateTimeField(MultiValueField): | |||||||
|             # Raise a validation error if time or date is empty |             # Raise a validation error if time or date is empty | ||||||
|             # (possible if SplitDateTimeField has required=False). |             # (possible if SplitDateTimeField has required=False). | ||||||
|             if data_list[0] in EMPTY_VALUES: |             if data_list[0] in EMPTY_VALUES: | ||||||
|                 raise ValidationError(gettext(u'Enter a valid date.')) |                 raise ValidationError(ugettext(u'Enter a valid date.')) | ||||||
|             if data_list[1] in EMPTY_VALUES: |             if data_list[1] in EMPTY_VALUES: | ||||||
|                 raise ValidationError(gettext(u'Enter a valid time.')) |                 raise ValidationError(ugettext(u'Enter a valid time.')) | ||||||
|             return datetime.datetime.combine(*data_list) |             return datetime.datetime.combine(*data_list) | ||||||
|         return None |         return None | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ import copy | |||||||
|  |  | ||||||
| from django.utils.datastructures import SortedDict | from django.utils.datastructures import SortedDict | ||||||
| from django.utils.html import escape | from django.utils.html import escape | ||||||
| from django.utils.encoding import StrAndUnicode | from django.utils.encoding import StrAndUnicode, smart_unicode, force_unicode | ||||||
|  |  | ||||||
| from fields import Field | from fields import Field | ||||||
| from widgets import TextInput, Textarea | from widgets import TextInput, Textarea | ||||||
| @@ -119,13 +119,13 @@ class BaseForm(StrAndUnicode): | |||||||
|             bf_errors = ErrorList([escape(error) for error in bf.errors]) # Escape and cache in local variable. |             bf_errors = ErrorList([escape(error) for error in bf.errors]) # Escape and cache in local variable. | ||||||
|             if bf.is_hidden: |             if bf.is_hidden: | ||||||
|                 if bf_errors: |                 if bf_errors: | ||||||
|                     top_errors.extend(['(Hidden field %s) %s' % (name, e) for e in bf_errors]) |                     top_errors.extend(['(Hidden field %s) %s' % (name, force_unicode(e)) for e in bf_errors]) | ||||||
|                 hidden_fields.append(unicode(bf)) |                 hidden_fields.append(unicode(bf)) | ||||||
|             else: |             else: | ||||||
|                 if errors_on_separate_row and bf_errors: |                 if errors_on_separate_row and bf_errors: | ||||||
|                     output.append(error_row % bf_errors) |                     output.append(error_row % force_unicode(bf_errors)) | ||||||
|                 if bf.label: |                 if bf.label: | ||||||
|                     label = escape(bf.label) |                     label = escape(force_unicode(bf.label)) | ||||||
|                     # Only add a colon if the label does not end in punctuation. |                     # Only add a colon if the label does not end in punctuation. | ||||||
|                     if label[-1] not in ':?.!': |                     if label[-1] not in ':?.!': | ||||||
|                         label += ':' |                         label += ':' | ||||||
| @@ -133,10 +133,10 @@ class BaseForm(StrAndUnicode): | |||||||
|                 else: |                 else: | ||||||
|                     label = '' |                     label = '' | ||||||
|                 if field.help_text: |                 if field.help_text: | ||||||
|                     help_text = help_text_html % field.help_text |                     help_text = help_text_html % force_unicode(field.help_text) | ||||||
|                 else: |                 else: | ||||||
|                     help_text = u'' |                     help_text = u'' | ||||||
|                 output.append(normal_row % {'errors': bf_errors, 'label': label, 'field': unicode(bf), 'help_text': help_text}) |                 output.append(normal_row % {'errors': force_unicode(bf_errors), 'label': force_unicode(label), 'field': unicode(bf), 'help_text': help_text}) | ||||||
|         if top_errors: |         if top_errors: | ||||||
|             output.insert(0, error_row % top_errors) |             output.insert(0, error_row % top_errors) | ||||||
|         if hidden_fields: # Insert any hidden fields in the last row. |         if hidden_fields: # Insert any hidden fields in the last row. | ||||||
| @@ -339,8 +339,8 @@ class BoundField(StrAndUnicode): | |||||||
|         associated Form has specified auto_id. Returns an empty string otherwise. |         associated Form has specified auto_id. Returns an empty string otherwise. | ||||||
|         """ |         """ | ||||||
|         auto_id = self.form.auto_id |         auto_id = self.form.auto_id | ||||||
|         if auto_id and '%s' in str(auto_id): |         if auto_id and '%s' in smart_unicode(auto_id): | ||||||
|             return str(auto_id) % self.html_name |             return smart_unicode(auto_id) % self.html_name | ||||||
|         elif auto_id: |         elif auto_id: | ||||||
|             return self.html_name |             return self.html_name | ||||||
|         return '' |         return '' | ||||||
|   | |||||||
| @@ -3,7 +3,9 @@ Helper functions for creating Form classes from Django models | |||||||
| and database field objects. | and database field objects. | ||||||
| """ | """ | ||||||
|  |  | ||||||
| from django.utils.translation import gettext | from django.utils.translation import ugettext | ||||||
|  | from django.utils.encoding import smart_unicode | ||||||
|  |  | ||||||
|  |  | ||||||
| from util import ValidationError | from util import ValidationError | ||||||
| from forms import BaseForm, SortedDictFromList | from forms import BaseForm, SortedDictFromList | ||||||
| @@ -123,7 +125,7 @@ class QuerySetIterator(object): | |||||||
|         if self.empty_label is not None: |         if self.empty_label is not None: | ||||||
|             yield (u"", self.empty_label) |             yield (u"", self.empty_label) | ||||||
|         for obj in self.queryset: |         for obj in self.queryset: | ||||||
|             yield (obj._get_pk_val(), str(obj)) |             yield (obj._get_pk_val(), smart_unicode(obj)) | ||||||
|         # Clear the QuerySet cache if required. |         # Clear the QuerySet cache if required. | ||||||
|         if not self.cache_choices: |         if not self.cache_choices: | ||||||
|             self.queryset._result_cache = None |             self.queryset._result_cache = None | ||||||
| @@ -170,7 +172,7 @@ class ModelChoiceField(ChoiceField): | |||||||
|         try: |         try: | ||||||
|             value = self.queryset.model._default_manager.get(pk=value) |             value = self.queryset.model._default_manager.get(pk=value) | ||||||
|         except self.queryset.model.DoesNotExist: |         except self.queryset.model.DoesNotExist: | ||||||
|             raise ValidationError(gettext(u'Select a valid choice. That choice is not one of the available choices.')) |             raise ValidationError(ugettext(u'Select a valid choice. That choice is not one of the available choices.')) | ||||||
|         return value |         return value | ||||||
|  |  | ||||||
| class ModelMultipleChoiceField(ModelChoiceField): | class ModelMultipleChoiceField(ModelChoiceField): | ||||||
| @@ -183,17 +185,17 @@ class ModelMultipleChoiceField(ModelChoiceField): | |||||||
|  |  | ||||||
|     def clean(self, value): |     def clean(self, value): | ||||||
|         if self.required and not value: |         if self.required and not value: | ||||||
|             raise ValidationError(gettext(u'This field is required.')) |             raise ValidationError(ugettext(u'This field is required.')) | ||||||
|         elif not self.required and not value: |         elif not self.required and not value: | ||||||
|             return [] |             return [] | ||||||
|         if not isinstance(value, (list, tuple)): |         if not isinstance(value, (list, tuple)): | ||||||
|             raise ValidationError(gettext(u'Enter a list of values.')) |             raise ValidationError(ugettext(u'Enter a list of values.')) | ||||||
|         final_values = [] |         final_values = [] | ||||||
|         for val in value: |         for val in value: | ||||||
|             try: |             try: | ||||||
|                 obj = self.queryset.model._default_manager.get(pk=val) |                 obj = self.queryset.model._default_manager.get(pk=val) | ||||||
|             except self.queryset.model.DoesNotExist: |             except self.queryset.model.DoesNotExist: | ||||||
|                 raise ValidationError(gettext(u'Select a valid choice. %s is not one of the available choices.') % val) |                 raise ValidationError(ugettext(u'Select a valid choice. %s is not one of the available choices.') % val) | ||||||
|             else: |             else: | ||||||
|                 final_values.append(obj) |                 final_values.append(obj) | ||||||
|         return final_values |         return final_values | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| from django.utils.html import escape | from django.utils.html import escape | ||||||
| from django.utils.encoding import smart_unicode | from django.utils.encoding import smart_unicode, StrAndUnicode | ||||||
|  |  | ||||||
| def flatatt(attrs): | def flatatt(attrs): | ||||||
|     """ |     """ | ||||||
| @@ -10,36 +10,36 @@ def flatatt(attrs): | |||||||
|     """ |     """ | ||||||
|     return u''.join([u' %s="%s"' % (k, escape(v)) for k, v in attrs.items()]) |     return u''.join([u' %s="%s"' % (k, escape(v)) for k, v in attrs.items()]) | ||||||
|  |  | ||||||
| class ErrorDict(dict): | class ErrorDict(dict, StrAndUnicode): | ||||||
|     """ |     """ | ||||||
|     A collection of errors that knows how to display itself in various formats. |     A collection of errors that knows how to display itself in various formats. | ||||||
|  |  | ||||||
|     The dictionary keys are the field names, and the values are the errors. |     The dictionary keys are the field names, and the values are the errors. | ||||||
|     """ |     """ | ||||||
|     def __str__(self): |     def __unicode__(self): | ||||||
|         return self.as_ul() |         return self.as_ul() | ||||||
|  |  | ||||||
|     def as_ul(self): |     def as_ul(self): | ||||||
|         if not self: return u'' |         if not self: return u'' | ||||||
|         return u'<ul class="errorlist">%s</ul>' % ''.join([u'<li>%s%s</li>' % (k, v) for k, v in self.items()]) |         return u'<ul class="errorlist">%s</ul>' % ''.join([u'<li>%s%s</li>' % (k, smart_unicode(v)) for k, v in self.items()]) | ||||||
|  |  | ||||||
|     def as_text(self): |     def as_text(self): | ||||||
|         return u'\n'.join([u'* %s\n%s' % (k, u'\n'.join([u'  * %s' % i for i in v])) for k, v in self.items()]) |         return u'\n'.join([u'* %s\n%s' % (k, u'\n'.join([u'  * %s' % smart_unicode(i) for i in v])) for k, v in self.items()]) | ||||||
|  |  | ||||||
| class ErrorList(list): | class ErrorList(list, StrAndUnicode): | ||||||
|     """ |     """ | ||||||
|     A collection of errors that knows how to display itself in various formats. |     A collection of errors that knows how to display itself in various formats. | ||||||
|     """ |     """ | ||||||
|     def __str__(self): |     def __unicode__(self): | ||||||
|         return self.as_ul() |         return self.as_ul() | ||||||
|  |  | ||||||
|     def as_ul(self): |     def as_ul(self): | ||||||
|         if not self: return u'' |         if not self: return u'' | ||||||
|         return u'<ul class="errorlist">%s</ul>' % ''.join([u'<li>%s</li>' % e for e in self]) |         return u'<ul class="errorlist">%s</ul>' % ''.join([u'<li>%s</li>' % smart_unicode(e) for e in self]) | ||||||
|  |  | ||||||
|     def as_text(self): |     def as_text(self): | ||||||
|         if not self: return u'' |         if not self: return u'' | ||||||
|         return u'\n'.join([u'* %s' % e for e in self]) |         return u'\n'.join([u'* %s' % smart_unicode(e) for e in self]) | ||||||
|  |  | ||||||
| class ValidationError(Exception): | class ValidationError(Exception): | ||||||
|     def __init__(self, message): |     def __init__(self, message): | ||||||
|   | |||||||
| @@ -10,8 +10,8 @@ except NameError: | |||||||
| from itertools import chain | from itertools import chain | ||||||
| from django.utils.datastructures import MultiValueDict | from django.utils.datastructures import MultiValueDict | ||||||
| from django.utils.html import escape | from django.utils.html import escape | ||||||
| from django.utils.translation import gettext | from django.utils.translation import ugettext | ||||||
| from django.utils.encoding import StrAndUnicode, smart_unicode | from django.utils.encoding import StrAndUnicode, force_unicode | ||||||
| from util import flatatt | from util import flatatt | ||||||
|  |  | ||||||
| __all__ = ( | __all__ = ( | ||||||
| @@ -77,7 +77,7 @@ class Input(Widget): | |||||||
|     def render(self, name, value, attrs=None): |     def render(self, name, value, attrs=None): | ||||||
|         if value is None: value = '' |         if value is None: value = '' | ||||||
|         final_attrs = self.build_attrs(attrs, type=self.input_type, name=name) |         final_attrs = self.build_attrs(attrs, type=self.input_type, name=name) | ||||||
|         if value != '': final_attrs['value'] = smart_unicode(value) # Only add the 'value' attribute if a value is non-empty. |         if value != '': final_attrs['value'] = force_unicode(value) # Only add the 'value' attribute if a value is non-empty. | ||||||
|         return u'<input%s />' % flatatt(final_attrs) |         return u'<input%s />' % flatatt(final_attrs) | ||||||
|  |  | ||||||
| class TextInput(Input): | class TextInput(Input): | ||||||
| @@ -111,7 +111,7 @@ class MultipleHiddenInput(HiddenInput): | |||||||
|     def render(self, name, value, attrs=None, choices=()): |     def render(self, name, value, attrs=None, choices=()): | ||||||
|         if value is None: value = [] |         if value is None: value = [] | ||||||
|         final_attrs = self.build_attrs(attrs, type=self.input_type, name=name) |         final_attrs = self.build_attrs(attrs, type=self.input_type, name=name) | ||||||
|         return u'\n'.join([(u'<input%s />' % flatatt(dict(value=smart_unicode(v), **final_attrs))) for v in value]) |         return u'\n'.join([(u'<input%s />' % flatatt(dict(value=force_unicode(v), **final_attrs))) for v in value]) | ||||||
|  |  | ||||||
|     def value_from_datadict(self, data, name): |     def value_from_datadict(self, data, name): | ||||||
|         if isinstance(data, MultiValueDict): |         if isinstance(data, MultiValueDict): | ||||||
| @@ -130,7 +130,7 @@ class Textarea(Widget): | |||||||
|  |  | ||||||
|     def render(self, name, value, attrs=None): |     def render(self, name, value, attrs=None): | ||||||
|         if value is None: value = '' |         if value is None: value = '' | ||||||
|         value = smart_unicode(value) |         value = force_unicode(value) | ||||||
|         final_attrs = self.build_attrs(attrs, name=name) |         final_attrs = self.build_attrs(attrs, name=name) | ||||||
|         return u'<textarea%s>%s</textarea>' % (flatatt(final_attrs), escape(value)) |         return u'<textarea%s>%s</textarea>' % (flatatt(final_attrs), escape(value)) | ||||||
|  |  | ||||||
| @@ -150,7 +150,7 @@ class CheckboxInput(Widget): | |||||||
|         if result: |         if result: | ||||||
|             final_attrs['checked'] = 'checked' |             final_attrs['checked'] = 'checked' | ||||||
|         if value not in ('', True, False, None): |         if value not in ('', True, False, None): | ||||||
|             final_attrs['value'] = smart_unicode(value) # Only add the 'value' attribute if a value is non-empty. |             final_attrs['value'] = force_unicode(value) # Only add the 'value' attribute if a value is non-empty. | ||||||
|         return u'<input%s />' % flatatt(final_attrs) |         return u'<input%s />' % flatatt(final_attrs) | ||||||
|  |  | ||||||
| class Select(Widget): | class Select(Widget): | ||||||
| @@ -165,11 +165,11 @@ class Select(Widget): | |||||||
|         if value is None: value = '' |         if value is None: value = '' | ||||||
|         final_attrs = self.build_attrs(attrs, name=name) |         final_attrs = self.build_attrs(attrs, name=name) | ||||||
|         output = [u'<select%s>' % flatatt(final_attrs)] |         output = [u'<select%s>' % flatatt(final_attrs)] | ||||||
|         str_value = smart_unicode(value) # Normalize to string. |         str_value = force_unicode(value) # Normalize to string. | ||||||
|         for option_value, option_label in chain(self.choices, choices): |         for option_value, option_label in chain(self.choices, choices): | ||||||
|             option_value = smart_unicode(option_value) |             option_value = force_unicode(option_value) | ||||||
|             selected_html = (option_value == str_value) and u' selected="selected"' or '' |             selected_html = (option_value == str_value) and u' selected="selected"' or '' | ||||||
|             output.append(u'<option value="%s"%s>%s</option>' % (escape(option_value), selected_html, escape(smart_unicode(option_label)))) |             output.append(u'<option value="%s"%s>%s</option>' % (escape(option_value), selected_html, escape(force_unicode(option_label)))) | ||||||
|         output.append(u'</select>') |         output.append(u'</select>') | ||||||
|         return u'\n'.join(output) |         return u'\n'.join(output) | ||||||
|  |  | ||||||
| @@ -178,7 +178,7 @@ class NullBooleanSelect(Select): | |||||||
|     A Select Widget intended to be used with NullBooleanField. |     A Select Widget intended to be used with NullBooleanField. | ||||||
|     """ |     """ | ||||||
|     def __init__(self, attrs=None): |     def __init__(self, attrs=None): | ||||||
|         choices = ((u'1', gettext('Unknown')), (u'2', gettext('Yes')), (u'3', gettext('No'))) |         choices = ((u'1', ugettext('Unknown')), (u'2', ugettext('Yes')), (u'3', ugettext('No'))) | ||||||
|         super(NullBooleanSelect, self).__init__(attrs, choices) |         super(NullBooleanSelect, self).__init__(attrs, choices) | ||||||
|  |  | ||||||
|     def render(self, name, value, attrs=None, choices=()): |     def render(self, name, value, attrs=None, choices=()): | ||||||
| @@ -202,11 +202,11 @@ class SelectMultiple(Widget): | |||||||
|         if value is None: value = [] |         if value is None: value = [] | ||||||
|         final_attrs = self.build_attrs(attrs, name=name) |         final_attrs = self.build_attrs(attrs, name=name) | ||||||
|         output = [u'<select multiple="multiple"%s>' % flatatt(final_attrs)] |         output = [u'<select multiple="multiple"%s>' % flatatt(final_attrs)] | ||||||
|         str_values = set([smart_unicode(v) for v in value]) # Normalize to strings. |         str_values = set([force_unicode(v) for v in value]) # Normalize to strings. | ||||||
|         for option_value, option_label in chain(self.choices, choices): |         for option_value, option_label in chain(self.choices, choices): | ||||||
|             option_value = smart_unicode(option_value) |             option_value = force_unicode(option_value) | ||||||
|             selected_html = (option_value in str_values) and ' selected="selected"' or '' |             selected_html = (option_value in str_values) and ' selected="selected"' or '' | ||||||
|             output.append(u'<option value="%s"%s>%s</option>' % (escape(option_value), selected_html, escape(smart_unicode(option_label)))) |             output.append(u'<option value="%s"%s>%s</option>' % (escape(option_value), selected_html, escape(force_unicode(option_label)))) | ||||||
|         output.append(u'</select>') |         output.append(u'</select>') | ||||||
|         return u'\n'.join(output) |         return u'\n'.join(output) | ||||||
|  |  | ||||||
| @@ -220,8 +220,8 @@ class RadioInput(StrAndUnicode): | |||||||
|     def __init__(self, name, value, attrs, choice, index): |     def __init__(self, name, value, attrs, choice, index): | ||||||
|         self.name, self.value = name, value |         self.name, self.value = name, value | ||||||
|         self.attrs = attrs |         self.attrs = attrs | ||||||
|         self.choice_value = smart_unicode(choice[0]) |         self.choice_value = force_unicode(choice[0]) | ||||||
|         self.choice_label = smart_unicode(choice[1]) |         self.choice_label = force_unicode(choice[1]) | ||||||
|         self.index = index |         self.index = index | ||||||
|  |  | ||||||
|     def __unicode__(self): |     def __unicode__(self): | ||||||
| @@ -254,13 +254,13 @@ class RadioFieldRenderer(StrAndUnicode): | |||||||
|  |  | ||||||
|     def __unicode__(self): |     def __unicode__(self): | ||||||
|         "Outputs a <ul> for this set of radio fields." |         "Outputs a <ul> for this set of radio fields." | ||||||
|         return u'<ul>\n%s\n</ul>' % u'\n'.join([u'<li>%s</li>' % w for w in self]) |         return u'<ul>\n%s\n</ul>' % u'\n'.join([u'<li>%s</li>' % force_unicode(w) for w in self]) | ||||||
|  |  | ||||||
| class RadioSelect(Select): | class RadioSelect(Select): | ||||||
|     def render(self, name, value, attrs=None, choices=()): |     def render(self, name, value, attrs=None, choices=()): | ||||||
|         "Returns a RadioFieldRenderer instance rather than a Unicode string." |         "Returns a RadioFieldRenderer instance rather than a Unicode string." | ||||||
|         if value is None: value = '' |         if value is None: value = '' | ||||||
|         str_value = smart_unicode(value) # Normalize to string. |         str_value = force_unicode(value) # Normalize to string. | ||||||
|         final_attrs = self.build_attrs(attrs) |         final_attrs = self.build_attrs(attrs) | ||||||
|         return RadioFieldRenderer(name, str_value, final_attrs, list(chain(self.choices, choices))) |         return RadioFieldRenderer(name, str_value, final_attrs, list(chain(self.choices, choices))) | ||||||
|  |  | ||||||
| @@ -280,16 +280,16 @@ class CheckboxSelectMultiple(SelectMultiple): | |||||||
|         has_id = attrs and 'id' in attrs |         has_id = attrs and 'id' in attrs | ||||||
|         final_attrs = self.build_attrs(attrs, name=name) |         final_attrs = self.build_attrs(attrs, name=name) | ||||||
|         output = [u'<ul>'] |         output = [u'<ul>'] | ||||||
|         str_values = set([smart_unicode(v) for v in value]) # Normalize to strings. |         str_values = set([force_unicode(v) for v in value]) # Normalize to strings. | ||||||
|         for i, (option_value, option_label) in enumerate(chain(self.choices, choices)): |         for i, (option_value, option_label) in enumerate(chain(self.choices, choices)): | ||||||
|             # If an ID attribute was given, add a numeric index as a suffix, |             # If an ID attribute was given, add a numeric index as a suffix, | ||||||
|             # so that the checkboxes don't all have the same ID attribute. |             # so that the checkboxes don't all have the same ID attribute. | ||||||
|             if has_id: |             if has_id: | ||||||
|                 final_attrs = dict(final_attrs, id='%s_%s' % (attrs['id'], i)) |                 final_attrs = dict(final_attrs, id='%s_%s' % (attrs['id'], i)) | ||||||
|             cb = CheckboxInput(final_attrs, check_test=lambda value: value in str_values) |             cb = CheckboxInput(final_attrs, check_test=lambda value: value in str_values) | ||||||
|             option_value = smart_unicode(option_value) |             option_value = force_unicode(option_value) | ||||||
|             rendered_cb = cb.render(name, option_value) |             rendered_cb = cb.render(name, option_value) | ||||||
|             output.append(u'<li><label>%s %s</label></li>' % (rendered_cb, escape(smart_unicode(option_label)))) |             output.append(u'<li><label>%s %s</label></li>' % (rendered_cb, escape(force_unicode(option_label)))) | ||||||
|         output.append(u'</ul>') |         output.append(u'</ul>') | ||||||
|         return u'\n'.join(output) |         return u'\n'.join(output) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -2,7 +2,8 @@ from django.core import validators | |||||||
| from django.core.exceptions import PermissionDenied | from django.core.exceptions import PermissionDenied | ||||||
| from django.utils.html import escape | from django.utils.html import escape | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| from django.utils.translation import gettext, ngettext | from django.utils.translation import ugettext, ungettext | ||||||
|  | from django.utils.encoding import smart_unicode, force_unicode, smart_str | ||||||
|  |  | ||||||
| FORM_FIELD_ID_PREFIX = 'id_' | FORM_FIELD_ID_PREFIX = 'id_' | ||||||
|  |  | ||||||
| @@ -66,7 +67,7 @@ class Manipulator(object): | |||||||
|                     errors.setdefault(field.field_name, []).extend(e.messages) |                     errors.setdefault(field.field_name, []).extend(e.messages) | ||||||
|  |  | ||||||
| #            if field.is_required and not new_data.get(field.field_name, False): | #            if field.is_required and not new_data.get(field.field_name, False): | ||||||
| #                errors.setdefault(field.field_name, []).append(gettext_lazy('This field is required.')) | #                errors.setdefault(field.field_name, []).append(ugettext_lazy('This field is required.')) | ||||||
| #                continue | #                continue | ||||||
| #            try: | #            try: | ||||||
| #                validator_list = field.validator_list | #                validator_list = field.validator_list | ||||||
| @@ -166,7 +167,11 @@ class FormFieldWrapper(object): | |||||||
|  |  | ||||||
|     def __str__(self): |     def __str__(self): | ||||||
|         "Renders the field" |         "Renders the field" | ||||||
|         return str(self.formfield.render(self.data)) |         return unicode(self).encode('utf-8') | ||||||
|  |  | ||||||
|  |     def __unicode__(self): | ||||||
|  |         "Renders the field" | ||||||
|  |         return force_unicode(self.formfield.render(self.data)) | ||||||
|  |  | ||||||
|     def __repr__(self): |     def __repr__(self): | ||||||
|         return '<FormFieldWrapper for "%s">' % self.formfield.field_name |         return '<FormFieldWrapper for "%s">' % self.formfield.field_name | ||||||
| @@ -196,7 +201,10 @@ class FormFieldCollection(FormFieldWrapper): | |||||||
|         self.formfield_dict = formfield_dict |         self.formfield_dict = formfield_dict | ||||||
|  |  | ||||||
|     def __str__(self): |     def __str__(self): | ||||||
|         return str(self.formfield_dict) |         return unicode(self).encode('utf-8') | ||||||
|  |  | ||||||
|  |     def __str__(self): | ||||||
|  |         return unicode(self.formfield_dict) | ||||||
|  |  | ||||||
|     def __getitem__(self, template_key): |     def __getitem__(self, template_key): | ||||||
|         "Look up field by template key; raise KeyError on failure" |         "Look up field by template key; raise KeyError on failure" | ||||||
| @@ -294,8 +302,12 @@ class FormField(object): | |||||||
|     Subclasses should also implement a render(data) method, which is responsible |     Subclasses should also implement a render(data) method, which is responsible | ||||||
|     for rending the form field in XHTML. |     for rending the form field in XHTML. | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     def __str__(self): |     def __str__(self): | ||||||
|         return self.render('') |         return unicode(self).encode('utf-8') | ||||||
|  |  | ||||||
|  |     def __unicode__(self): | ||||||
|  |         return self.render(u'') | ||||||
|  |  | ||||||
|     def __repr__(self): |     def __repr__(self): | ||||||
|         return 'FormField "%s"' % self.field_name |         return 'FormField "%s"' % self.field_name | ||||||
| @@ -354,7 +366,7 @@ class FormField(object): | |||||||
|     def get_validation_errors(self, new_data): |     def get_validation_errors(self, new_data): | ||||||
|         errors = {} |         errors = {} | ||||||
|         if self.is_required and not new_data.get(self.field_name, False): |         if self.is_required and not new_data.get(self.field_name, False): | ||||||
|             errors.setdefault(self.field_name, []).append(gettext('This field is required.')) |             errors.setdefault(self.field_name, []).append(ugettext('This field is required.')) | ||||||
|             return errors |             return errors | ||||||
|         try: |         try: | ||||||
|             for validator in self.validator_list: |             for validator in self.validator_list: | ||||||
| @@ -388,24 +400,22 @@ class TextField(FormField): | |||||||
|             self.member_name = member_name |             self.member_name = member_name | ||||||
|  |  | ||||||
|     def isValidLength(self, data, form): |     def isValidLength(self, data, form): | ||||||
|         if data and self.maxlength and len(data.decode(settings.DEFAULT_CHARSET)) > self.maxlength: |         if data and self.maxlength and len(smart_unicode(data)) > self.maxlength: | ||||||
|             raise validators.ValidationError, ngettext("Ensure your text is less than %s character.", |             raise validators.ValidationError, ungettext("Ensure your text is less than %s character.", | ||||||
|                 "Ensure your text is less than %s characters.", self.maxlength) % self.maxlength |                 "Ensure your text is less than %s characters.", self.maxlength) % self.maxlength | ||||||
|  |  | ||||||
|     def hasNoNewlines(self, data, form): |     def hasNoNewlines(self, data, form): | ||||||
|         if data and '\n' in data: |         if data and '\n' in data: | ||||||
|             raise validators.ValidationError, gettext("Line breaks are not allowed here.") |             raise validators.ValidationError, ugettext("Line breaks are not allowed here.") | ||||||
|  |  | ||||||
|     def render(self, data): |     def render(self, data): | ||||||
|         if data is None: |         if data is None: | ||||||
|             data = '' |             data = u'' | ||||||
|         maxlength = '' |         maxlength = u'' | ||||||
|         if self.maxlength: |         if self.maxlength: | ||||||
|             maxlength = 'maxlength="%s" ' % self.maxlength |             maxlength = u'maxlength="%s" ' % self.maxlength | ||||||
|         if isinstance(data, unicode): |         return u'<input type="%s" id="%s" class="v%s%s" name="%s" size="%s" value="%s" %s/>' % \ | ||||||
|             data = data.encode(settings.DEFAULT_CHARSET) |             (self.input_type, self.get_id(), self.__class__.__name__, self.is_required and u' required' or '', | ||||||
|         return '<input type="%s" id="%s" class="v%s%s" name="%s" size="%s" value="%s" %s/>' % \ |  | ||||||
|             (self.input_type, self.get_id(), self.__class__.__name__, self.is_required and ' required' or '', |  | ||||||
|             self.field_name, self.length, escape(data), maxlength) |             self.field_name, self.length, escape(data), maxlength) | ||||||
|  |  | ||||||
|     def html2python(data): |     def html2python(data): | ||||||
| @@ -428,10 +438,8 @@ class LargeTextField(TextField): | |||||||
|     def render(self, data): |     def render(self, data): | ||||||
|         if data is None: |         if data is None: | ||||||
|             data = '' |             data = '' | ||||||
|         if isinstance(data, unicode): |         return u'<textarea id="%s" class="v%s%s" name="%s" rows="%s" cols="%s">%s</textarea>' % \ | ||||||
|             data = data.encode(settings.DEFAULT_CHARSET) |             (self.get_id(), self.__class__.__name__, self.is_required and u' required' or u'', | ||||||
|         return '<textarea id="%s" class="v%s%s" name="%s" rows="%s" cols="%s">%s</textarea>' % \ |  | ||||||
|             (self.get_id(), self.__class__.__name__, self.is_required and ' required' or '', |  | ||||||
|             self.field_name, self.rows, self.cols, escape(data)) |             self.field_name, self.rows, self.cols, escape(data)) | ||||||
|  |  | ||||||
| class HiddenField(FormField): | class HiddenField(FormField): | ||||||
| @@ -441,7 +449,7 @@ class HiddenField(FormField): | |||||||
|         self.validator_list = validator_list[:] |         self.validator_list = validator_list[:] | ||||||
|  |  | ||||||
|     def render(self, data): |     def render(self, data): | ||||||
|         return '<input type="hidden" id="%s" name="%s" value="%s" />' % \ |         return u'<input type="hidden" id="%s" name="%s" value="%s" />' % \ | ||||||
|             (self.get_id(), self.field_name, escape(data)) |             (self.get_id(), self.field_name, escape(data)) | ||||||
|  |  | ||||||
| class CheckboxField(FormField): | class CheckboxField(FormField): | ||||||
| @@ -456,7 +464,7 @@ class CheckboxField(FormField): | |||||||
|         checked_html = '' |         checked_html = '' | ||||||
|         if data or (data is '' and self.checked_by_default): |         if data or (data is '' and self.checked_by_default): | ||||||
|             checked_html = ' checked="checked"' |             checked_html = ' checked="checked"' | ||||||
|         return '<input type="checkbox" id="%s" class="v%s" name="%s"%s />' % \ |         return u'<input type="checkbox" id="%s" class="v%s" name="%s"%s />' % \ | ||||||
|             (self.get_id(), self.__class__.__name__, |             (self.get_id(), self.__class__.__name__, | ||||||
|             self.field_name, checked_html) |             self.field_name, checked_html) | ||||||
|  |  | ||||||
| @@ -471,6 +479,7 @@ class SelectField(FormField): | |||||||
|     def __init__(self, field_name, choices=None, size=1, is_required=False, validator_list=None, member_name=None): |     def __init__(self, field_name, choices=None, size=1, is_required=False, validator_list=None, member_name=None): | ||||||
|         if validator_list is None: validator_list = [] |         if validator_list is None: validator_list = [] | ||||||
|         if choices is None: choices = [] |         if choices is None: choices = [] | ||||||
|  |         choices = [(k, smart_unicode(v, strings_only=True)) for k, v in choices] | ||||||
|         self.field_name = field_name |         self.field_name = field_name | ||||||
|         # choices is a list of (value, human-readable key) tuples because order matters |         # choices is a list of (value, human-readable key) tuples because order matters | ||||||
|         self.choices, self.size, self.is_required = choices, size, is_required |         self.choices, self.size, self.is_required = choices, size, is_required | ||||||
| @@ -479,23 +488,23 @@ class SelectField(FormField): | |||||||
|             self.member_name = member_name |             self.member_name = member_name | ||||||
|  |  | ||||||
|     def render(self, data): |     def render(self, data): | ||||||
|         output = ['<select id="%s" class="v%s%s" name="%s" size="%s">' % \ |         output = [u'<select id="%s" class="v%s%s" name="%s" size="%s">' % \ | ||||||
|             (self.get_id(), self.__class__.__name__, |             (self.get_id(), self.__class__.__name__, | ||||||
|              self.is_required and ' required' or '', self.field_name, self.size)] |              self.is_required and u' required' or u'', self.field_name, self.size)] | ||||||
|         str_data = str(data) # normalize to string |         str_data = smart_unicode(data) # normalize to string | ||||||
|         for value, display_name in self.choices: |         for value, display_name in self.choices: | ||||||
|             selected_html = '' |             selected_html = u'' | ||||||
|             if str(value) == str_data: |             if smart_unicode(value) == str_data: | ||||||
|                 selected_html = ' selected="selected"' |                 selected_html = u' selected="selected"' | ||||||
|             output.append('    <option value="%s"%s>%s</option>' % (escape(value), selected_html, escape(display_name))) |             output.append(u'    <option value="%s"%s>%s</option>' % (escape(value), selected_html, escape(display_name))) | ||||||
|         output.append('  </select>') |         output.append(u'  </select>') | ||||||
|         return '\n'.join(output) |         return u'\n'.join(output) | ||||||
|  |  | ||||||
|     def isValidChoice(self, data, form): |     def isValidChoice(self, data, form): | ||||||
|         str_data = str(data) |         str_data = smart_unicode(data) | ||||||
|         str_choices = [str(item[0]) for item in self.choices] |         str_choices = [smart_str(item[0]) for item in self.choices] | ||||||
|         if str_data not in str_choices: |         if str_data not in str_choices: | ||||||
|             raise validators.ValidationError, gettext("Select a valid choice; '%(data)s' is not in %(choices)s.") % {'data': str_data, 'choices': str_choices} |             raise validators.ValidationError, ugettext("Select a valid choice; '%(data)s' is not in %(choices)s.") % {'data': str_data, 'choices': str_choices} | ||||||
|  |  | ||||||
| class NullSelectField(SelectField): | class NullSelectField(SelectField): | ||||||
|     "This SelectField converts blank fields to None" |     "This SelectField converts blank fields to None" | ||||||
| @@ -509,6 +518,7 @@ class RadioSelectField(FormField): | |||||||
|     def __init__(self, field_name, choices=None, ul_class='', is_required=False, validator_list=None, member_name=None): |     def __init__(self, field_name, choices=None, ul_class='', is_required=False, validator_list=None, member_name=None): | ||||||
|         if validator_list is None: validator_list = [] |         if validator_list is None: validator_list = [] | ||||||
|         if choices is None: choices = [] |         if choices is None: choices = [] | ||||||
|  |         choices = [(k, smart_unicode(v)) for k, v in choices] | ||||||
|         self.field_name = field_name |         self.field_name = field_name | ||||||
|         # choices is a list of (value, human-readable key) tuples because order matters |         # choices is a list of (value, human-readable key) tuples because order matters | ||||||
|         self.choices, self.is_required = choices, is_required |         self.choices, self.is_required = choices, is_required | ||||||
| @@ -520,7 +530,7 @@ class RadioSelectField(FormField): | |||||||
|     def render(self, data): |     def render(self, data): | ||||||
|         """ |         """ | ||||||
|         Returns a special object, RadioFieldRenderer, that is iterable *and* |         Returns a special object, RadioFieldRenderer, that is iterable *and* | ||||||
|         has a default str() rendered output. |         has a default unicode() rendered output. | ||||||
|  |  | ||||||
|         This allows for flexible use in templates. You can just use the default |         This allows for flexible use in templates. You can just use the default | ||||||
|         rendering: |         rendering: | ||||||
| @@ -537,44 +547,44 @@ class RadioSelectField(FormField): | |||||||
|         class RadioFieldRenderer: |         class RadioFieldRenderer: | ||||||
|             def __init__(self, datalist, ul_class): |             def __init__(self, datalist, ul_class): | ||||||
|                 self.datalist, self.ul_class = datalist, ul_class |                 self.datalist, self.ul_class = datalist, ul_class | ||||||
|             def __str__(self): |             def __unicode__(self): | ||||||
|                 "Default str() output for this radio field -- a <ul>" |                 "Default unicode() output for this radio field -- a <ul>" | ||||||
|                 output = ['<ul%s>' % (self.ul_class and ' class="%s"' % self.ul_class or '')] |                 output = [u'<ul%s>' % (self.ul_class and u' class="%s"' % self.ul_class or u'')] | ||||||
|                 output.extend(['<li>%s %s</li>' % (d['field'], d['label']) for d in self.datalist]) |                 output.extend([u'<li>%s %s</li>' % (d['field'], d['label']) for d in self.datalist]) | ||||||
|                 output.append('</ul>') |                 output.append(u'</ul>') | ||||||
|                 return ''.join(output) |                 return u''.join(output) | ||||||
|             def __iter__(self): |             def __iter__(self): | ||||||
|                 for d in self.datalist: |                 for d in self.datalist: | ||||||
|                     yield d |                     yield d | ||||||
|             def __len__(self): |             def __len__(self): | ||||||
|                 return len(self.datalist) |                 return len(self.datalist) | ||||||
|         datalist = [] |         datalist = [] | ||||||
|         str_data = str(data) # normalize to string |         str_data = smart_unicode(data) # normalize to string | ||||||
|         for i, (value, display_name) in enumerate(self.choices): |         for i, (value, display_name) in enumerate(self.choices): | ||||||
|             selected_html = '' |             selected_html = '' | ||||||
|             if str(value) == str_data: |             if smart_unicode(value) == str_data: | ||||||
|                 selected_html = ' checked="checked"' |                 selected_html = u' checked="checked"' | ||||||
|             datalist.append({ |             datalist.append({ | ||||||
|                 'value': value, |                 'value': value, | ||||||
|                 'name': display_name, |                 'name': display_name, | ||||||
|                 'field': '<input type="radio" id="%s" name="%s" value="%s"%s/>' % \ |                 'field': u'<input type="radio" id="%s" name="%s" value="%s"%s/>' % \ | ||||||
|                     (self.get_id() + '_' + str(i), self.field_name, value, selected_html), |                     (self.get_id() + u'_' + unicode(i), self.field_name, value, selected_html), | ||||||
|                 'label': '<label for="%s">%s</label>' % \ |                 'label': u'<label for="%s">%s</label>' % \ | ||||||
|                     (self.get_id() + '_' + str(i), display_name), |                     (self.get_id() + u'_' + unicode(i), display_name), | ||||||
|             }) |             }) | ||||||
|         return RadioFieldRenderer(datalist, self.ul_class) |         return RadioFieldRenderer(datalist, self.ul_class) | ||||||
|  |  | ||||||
|     def isValidChoice(self, data, form): |     def isValidChoice(self, data, form): | ||||||
|         str_data = str(data) |         str_data = smart_unicode(data) | ||||||
|         str_choices = [str(item[0]) for item in self.choices] |         str_choices = [smart_unicode(item[0]) for item in self.choices] | ||||||
|         if str_data not in str_choices: |         if str_data not in str_choices: | ||||||
|             raise validators.ValidationError, gettext("Select a valid choice; '%(data)s' is not in %(choices)s.") % {'data':str_data, 'choices':str_choices} |             raise validators.ValidationError, ugettext("Select a valid choice; '%(data)s' is not in %(choices)s.") % {'data':str_data, 'choices':str_choices} | ||||||
|  |  | ||||||
| class NullBooleanField(SelectField): | class NullBooleanField(SelectField): | ||||||
|     "This SelectField provides 'Yes', 'No' and 'Unknown', mapping results to True, False or None" |     "This SelectField provides 'Yes', 'No' and 'Unknown', mapping results to True, False or None" | ||||||
|     def __init__(self, field_name, is_required=False, validator_list=None): |     def __init__(self, field_name, is_required=False, validator_list=None): | ||||||
|         if validator_list is None: validator_list = [] |         if validator_list is None: validator_list = [] | ||||||
|         SelectField.__init__(self, field_name, choices=[('1', _('Unknown')), ('2', _('Yes')), ('3', _('No'))], |         SelectField.__init__(self, field_name, choices=[('1', ugettext('Unknown')), ('2', ugettext('Yes')), ('3', ugettext('No'))], | ||||||
|             is_required=is_required, validator_list=validator_list) |             is_required=is_required, validator_list=validator_list) | ||||||
|  |  | ||||||
|     def render(self, data): |     def render(self, data): | ||||||
| @@ -590,24 +600,24 @@ class NullBooleanField(SelectField): | |||||||
| class SelectMultipleField(SelectField): | class SelectMultipleField(SelectField): | ||||||
|     requires_data_list = True |     requires_data_list = True | ||||||
|     def render(self, data): |     def render(self, data): | ||||||
|         output = ['<select id="%s" class="v%s%s" name="%s" size="%s" multiple="multiple">' % \ |         output = [u'<select id="%s" class="v%s%s" name="%s" size="%s" multiple="multiple">' % \ | ||||||
|             (self.get_id(), self.__class__.__name__, self.is_required and ' required' or '', |             (self.get_id(), self.__class__.__name__, self.is_required and u' required' or u'', | ||||||
|             self.field_name, self.size)] |             self.field_name, self.size)] | ||||||
|         str_data_list = map(str, data) # normalize to strings |         str_data_list = map(smart_unicode, data) # normalize to strings | ||||||
|         for value, choice in self.choices: |         for value, choice in self.choices: | ||||||
|             selected_html = '' |             selected_html = u'' | ||||||
|             if str(value) in str_data_list: |             if smart_unicode(value) in str_data_list: | ||||||
|                 selected_html = ' selected="selected"' |                 selected_html = u' selected="selected"' | ||||||
|             output.append('    <option value="%s"%s>%s</option>' % (escape(value), selected_html, escape(choice))) |             output.append(u'    <option value="%s"%s>%s</option>' % (escape(value), selected_html, escape(choice))) | ||||||
|         output.append('  </select>') |         output.append(u'  </select>') | ||||||
|         return '\n'.join(output) |         return u'\n'.join(output) | ||||||
|  |  | ||||||
|     def isValidChoice(self, field_data, all_data): |     def isValidChoice(self, field_data, all_data): | ||||||
|         # data is something like ['1', '2', '3'] |         # data is something like ['1', '2', '3'] | ||||||
|         str_choices = [str(item[0]) for item in self.choices] |         str_choices = [smart_unicode(item[0]) for item in self.choices] | ||||||
|         for val in map(str, field_data): |         for val in map(smart_unicode, field_data): | ||||||
|             if val not in str_choices: |             if val not in str_choices: | ||||||
|                 raise validators.ValidationError, gettext("Select a valid choice; '%(data)s' is not in %(choices)s.") % {'data':val, 'choices':str_choices} |                 raise validators.ValidationError, ugettext("Select a valid choice; '%(data)s' is not in %(choices)s.") % {'data':val, 'choices':str_choices} | ||||||
|  |  | ||||||
|     def html2python(data): |     def html2python(data): | ||||||
|         if data is None: |         if data is None: | ||||||
| @@ -642,18 +652,18 @@ class CheckboxSelectMultipleField(SelectMultipleField): | |||||||
|         new_data.setlist(self.field_name, data_list) |         new_data.setlist(self.field_name, data_list) | ||||||
|  |  | ||||||
|     def render(self, data): |     def render(self, data): | ||||||
|         output = ['<ul%s>' % (self.ul_class and ' class="%s"' % self.ul_class or '')] |         output = [u'<ul%s>' % (self.ul_class and u' class="%s"' % self.ul_class or u'')] | ||||||
|         str_data_list = map(str, data) # normalize to strings |         str_data_list = map(smart_unicode, data) # normalize to strings | ||||||
|         for value, choice in self.choices: |         for value, choice in self.choices: | ||||||
|             checked_html = '' |             checked_html = u'' | ||||||
|             if str(value) in str_data_list: |             if smart_unicode(value) in str_data_list: | ||||||
|                 checked_html = ' checked="checked"' |                 checked_html = u' checked="checked"' | ||||||
|             field_name = '%s%s' % (self.field_name, value) |             field_name = u'%s%s' % (self.field_name, value) | ||||||
|             output.append('<li><input type="checkbox" id="%s" class="v%s" name="%s"%s value="on" /> <label for="%s">%s</label></li>' % \ |             output.append(u'<li><input type="checkbox" id="%s" class="v%s" name="%s"%s value="on" /> <label for="%s">%s</label></li>' % \ | ||||||
|                 (self.get_id() + escape(value), self.__class__.__name__, field_name, checked_html, |                 (self.get_id() + escape(value), self.__class__.__name__, field_name, checked_html, | ||||||
|                 self.get_id() + escape(value), choice)) |                 self.get_id() + escape(value), choice)) | ||||||
|         output.append('</ul>') |         output.append(u'</ul>') | ||||||
|         return '\n'.join(output) |         return u'\n'.join(output) | ||||||
|  |  | ||||||
| #################### | #################### | ||||||
| # FILE UPLOADS     # | # FILE UPLOADS     # | ||||||
| @@ -669,12 +679,12 @@ class FileUploadField(FormField): | |||||||
|         try: |         try: | ||||||
|             content = field_data['content'] |             content = field_data['content'] | ||||||
|         except TypeError: |         except TypeError: | ||||||
|             raise validators.CriticalValidationError, gettext("No file was submitted. Check the encoding type on the form.") |             raise validators.CriticalValidationError, ugettext("No file was submitted. Check the encoding type on the form.") | ||||||
|         if not content: |         if not content: | ||||||
|             raise validators.CriticalValidationError, gettext("The submitted file is empty.") |             raise validators.CriticalValidationError, ugettext("The submitted file is empty.") | ||||||
|  |  | ||||||
|     def render(self, data): |     def render(self, data): | ||||||
|         return '<input type="file" id="%s" class="v%s" name="%s" />' % \ |         return u'<input type="file" id="%s" class="v%s" name="%s" />' % \ | ||||||
|             (self.get_id(), self.__class__.__name__, self.field_name) |             (self.get_id(), self.__class__.__name__, self.field_name) | ||||||
|  |  | ||||||
|     def html2python(data): |     def html2python(data): | ||||||
| @@ -727,7 +737,7 @@ class SmallIntegerField(IntegerField): | |||||||
|  |  | ||||||
|     def isSmallInteger(self, field_data, all_data): |     def isSmallInteger(self, field_data, all_data): | ||||||
|         if not -32768 <= int(field_data) <= 32767: |         if not -32768 <= int(field_data) <= 32767: | ||||||
|             raise validators.CriticalValidationError, gettext("Enter a whole number between -32,768 and 32,767.") |             raise validators.CriticalValidationError, ugettext("Enter a whole number between -32,768 and 32,767.") | ||||||
|  |  | ||||||
| class PositiveIntegerField(IntegerField): | class PositiveIntegerField(IntegerField): | ||||||
|     def __init__(self, field_name, length=10, maxlength=None, is_required=False, validator_list=None): |     def __init__(self, field_name, length=10, maxlength=None, is_required=False, validator_list=None): | ||||||
| @@ -737,7 +747,7 @@ class PositiveIntegerField(IntegerField): | |||||||
|  |  | ||||||
|     def isPositive(self, field_data, all_data): |     def isPositive(self, field_data, all_data): | ||||||
|         if int(field_data) < 0: |         if int(field_data) < 0: | ||||||
|             raise validators.CriticalValidationError, gettext("Enter a positive number.") |             raise validators.CriticalValidationError, ugettext("Enter a positive number.") | ||||||
|  |  | ||||||
| class PositiveSmallIntegerField(IntegerField): | class PositiveSmallIntegerField(IntegerField): | ||||||
|     def __init__(self, field_name, length=5, maxlength=None, is_required=False, validator_list=None): |     def __init__(self, field_name, length=5, maxlength=None, is_required=False, validator_list=None): | ||||||
| @@ -747,7 +757,7 @@ class PositiveSmallIntegerField(IntegerField): | |||||||
|  |  | ||||||
|     def isPositiveSmall(self, field_data, all_data): |     def isPositiveSmall(self, field_data, all_data): | ||||||
|         if not 0 <= int(field_data) <= 32767: |         if not 0 <= int(field_data) <= 32767: | ||||||
|             raise validators.CriticalValidationError, gettext("Enter a whole number between 0 and 32,767.") |             raise validators.CriticalValidationError, ugettext("Enter a whole number between 0 and 32,767.") | ||||||
|  |  | ||||||
| class FloatField(TextField): | class FloatField(TextField): | ||||||
|     def __init__(self, field_name, is_required=False, validator_list=None):  |     def __init__(self, field_name, is_required=False, validator_list=None):  | ||||||
| @@ -1005,9 +1015,9 @@ class CommaSeparatedIntegerField(TextField): | |||||||
|  |  | ||||||
|     def render(self, data): |     def render(self, data): | ||||||
|         if data is None: |         if data is None: | ||||||
|             data = '' |             data = u'' | ||||||
|         elif isinstance(data, (list, tuple)): |         elif isinstance(data, (list, tuple)): | ||||||
|             data = ','.join(data) |             data = u','.join(data) | ||||||
|         return super(CommaSeparatedIntegerField, self).render(data) |         return super(CommaSeparatedIntegerField, self).render(data) | ||||||
|  |  | ||||||
| class RawIdAdminField(CommaSeparatedIntegerField): | class RawIdAdminField(CommaSeparatedIntegerField): | ||||||
|   | |||||||
| @@ -58,8 +58,10 @@ import re | |||||||
| from inspect import getargspec | from inspect import getargspec | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| from django.template.context import Context, RequestContext, ContextPopException | from django.template.context import Context, RequestContext, ContextPopException | ||||||
| from django.utils.functional import curry | from django.utils.functional import curry, Promise | ||||||
| from django.utils.text import smart_split | from django.utils.text import smart_split | ||||||
|  | from django.utils.encoding import smart_unicode, force_unicode | ||||||
|  | from django.utils.translation import ugettext as _ | ||||||
|  |  | ||||||
| __all__ = ('Template', 'Context', 'RequestContext', 'compile_string') | __all__ = ('Template', 'Context', 'RequestContext', 'compile_string') | ||||||
|  |  | ||||||
| @@ -122,6 +124,9 @@ class TemplateSyntaxError(Exception): | |||||||
| class TemplateDoesNotExist(Exception): | class TemplateDoesNotExist(Exception): | ||||||
|     pass |     pass | ||||||
|  |  | ||||||
|  | class TemplateEncodingError(Exception): | ||||||
|  |     pass | ||||||
|  |  | ||||||
| class VariableDoesNotExist(Exception): | class VariableDoesNotExist(Exception): | ||||||
|  |  | ||||||
|     def __init__(self, msg, params=()): |     def __init__(self, msg, params=()): | ||||||
| @@ -155,6 +160,10 @@ class StringOrigin(Origin): | |||||||
| class Template(object): | class Template(object): | ||||||
|     def __init__(self, template_string, origin=None, name='<Unknown Template>'): |     def __init__(self, template_string, origin=None, name='<Unknown Template>'): | ||||||
|         "Compilation stage" |         "Compilation stage" | ||||||
|  |         try: | ||||||
|  |             template_string = smart_unicode(template_string) | ||||||
|  |         except UnicodeDecodeError: | ||||||
|  |             raise TemplateEncodingError("Templates can only be constructed from unicode or UTF-8 strings.") | ||||||
|         if settings.TEMPLATE_DEBUG and origin == None: |         if settings.TEMPLATE_DEBUG and origin == None: | ||||||
|             origin = StringOrigin(template_string) |             origin = StringOrigin(template_string) | ||||||
|             # Could do some crazy stack-frame stuff to record where this string |             # Could do some crazy stack-frame stuff to record where this string | ||||||
| @@ -693,6 +702,13 @@ def resolve_variable(path, context): | |||||||
|                     else: |                     else: | ||||||
|                         raise |                         raise | ||||||
|             del bits[0] |             del bits[0] | ||||||
|  |     if isinstance(current, (basestring, Promise)): | ||||||
|  |         try: | ||||||
|  |             current = force_unicode(current) | ||||||
|  |         except UnicodeDecodeError: | ||||||
|  |             # Failing to convert to unicode can happen sometimes (e.g. debug | ||||||
|  |             # tracebacks). So we allow it in this particular instance. | ||||||
|  |             pass | ||||||
|     return current |     return current | ||||||
|  |  | ||||||
| class Node(object): | class Node(object): | ||||||
| @@ -720,7 +736,7 @@ class NodeList(list): | |||||||
|                 bits.append(self.render_node(node, context)) |                 bits.append(self.render_node(node, context)) | ||||||
|             else: |             else: | ||||||
|                 bits.append(node) |                 bits.append(node) | ||||||
|         return ''.join(bits) |         return ''.join([force_unicode(b) for b in bits]) | ||||||
|  |  | ||||||
|     def get_nodes_by_type(self, nodetype): |     def get_nodes_by_type(self, nodetype): | ||||||
|         "Return a list of all nodes of the given type" |         "Return a list of all nodes of the given type" | ||||||
| @@ -730,7 +746,7 @@ class NodeList(list): | |||||||
|         return nodes |         return nodes | ||||||
|  |  | ||||||
|     def render_node(self, node, context): |     def render_node(self, node, context): | ||||||
|         return(node.render(context)) |         return node.render(context) | ||||||
|  |  | ||||||
| class DebugNodeList(NodeList): | class DebugNodeList(NodeList): | ||||||
|     def render_node(self, node, context): |     def render_node(self, node, context): | ||||||
| @@ -765,32 +781,17 @@ class VariableNode(Node): | |||||||
|     def __repr__(self): |     def __repr__(self): | ||||||
|         return "<Variable Node: %s>" % self.filter_expression |         return "<Variable Node: %s>" % self.filter_expression | ||||||
|  |  | ||||||
|     def encode_output(self, output): |  | ||||||
|         # Check type so that we don't run str() on a Unicode object |  | ||||||
|         if not isinstance(output, basestring): |  | ||||||
|             try: |  | ||||||
|                 return str(output) |  | ||||||
|             except UnicodeEncodeError: |  | ||||||
|                 # If __str__() returns a Unicode object, convert it to bytestring. |  | ||||||
|                 return unicode(output).encode(settings.DEFAULT_CHARSET) |  | ||||||
|         elif isinstance(output, unicode): |  | ||||||
|             return output.encode(settings.DEFAULT_CHARSET) |  | ||||||
|         else: |  | ||||||
|             return output |  | ||||||
|  |  | ||||||
|     def render(self, context): |     def render(self, context): | ||||||
|         output = self.filter_expression.resolve(context) |         return self.filter_expression.resolve(context) | ||||||
|         return self.encode_output(output) |  | ||||||
|  |  | ||||||
| class DebugVariableNode(VariableNode): | class DebugVariableNode(VariableNode): | ||||||
|     def render(self, context): |     def render(self, context): | ||||||
|         try: |         try: | ||||||
|             output = self.filter_expression.resolve(context) |             return self.filter_expression.resolve(context) | ||||||
|         except TemplateSyntaxError, e: |         except TemplateSyntaxError, e: | ||||||
|             if not hasattr(e, 'source'): |             if not hasattr(e, 'source'): | ||||||
|                 e.source = self.source |                 e.source = self.source | ||||||
|             raise |             raise | ||||||
|         return self.encode_output(output) |  | ||||||
|  |  | ||||||
| def generic_tag_compiler(params, defaults, name, node_class, parser, token): | def generic_tag_compiler(params, defaults, name, node_class, parser, token): | ||||||
|     "Returns a template.Node subclass." |     "Returns a template.Node subclass." | ||||||
|   | |||||||
| @@ -2,7 +2,8 @@ | |||||||
|  |  | ||||||
| from django.template import resolve_variable, Library | from django.template import resolve_variable, Library | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| from django.utils.translation import gettext, ngettext | from django.utils.translation import ugettext, ungettext | ||||||
|  | from django.utils.encoding import force_unicode, smart_str, iri_to_uri | ||||||
| import re | import re | ||||||
| import random as random_module | import random as random_module | ||||||
|  |  | ||||||
| @@ -12,27 +13,16 @@ register = Library() | |||||||
| # STRING DECORATOR    # | # STRING DECORATOR    # | ||||||
| ####################### | ####################### | ||||||
|  |  | ||||||
| def smart_string(obj): |  | ||||||
|     # FUTURE: Unicode strings should probably be normalized to a specific |  | ||||||
|     # encoding and non-unicode strings should be converted to unicode too. |  | ||||||
| #    if isinstance(obj, unicode): |  | ||||||
| #        obj = obj.encode(settings.DEFAULT_CHARSET) |  | ||||||
| #    else: |  | ||||||
| #        obj = unicode(obj, settings.DEFAULT_CHARSET) |  | ||||||
|     # FUTURE: Replace dumb string logic below with cool unicode logic above. |  | ||||||
|     if not isinstance(obj, basestring): |  | ||||||
|         obj = str(obj) |  | ||||||
|     return obj |  | ||||||
|  |  | ||||||
| def stringfilter(func): | def stringfilter(func): | ||||||
|     """ |     """ | ||||||
|     Decorator for filters which should only receive strings. The object passed |     Decorator for filters which should only receive unicode objects. The object | ||||||
|     as the first positional argument will be converted to a string. |     passed as the first positional argument will be converted to a unicode | ||||||
|  |     object. | ||||||
|     """ |     """ | ||||||
|     def _dec(*args, **kwargs): |     def _dec(*args, **kwargs): | ||||||
|         if args: |         if args: | ||||||
|             args = list(args) |             args = list(args) | ||||||
|             args[0] = smart_string(args[0]) |             args[0] = force_unicode(args[0]) | ||||||
|         return func(*args, **kwargs) |         return func(*args, **kwargs) | ||||||
|  |  | ||||||
|     # Include a reference to the real function (used to check original |     # Include a reference to the real function (used to check original | ||||||
| @@ -83,27 +73,32 @@ def floatformat(text, arg=-1): | |||||||
|     try: |     try: | ||||||
|         f = float(text) |         f = float(text) | ||||||
|     except ValueError: |     except ValueError: | ||||||
|         return '' |         return u'' | ||||||
|     try: |     try: | ||||||
|         d = int(arg) |         d = int(arg) | ||||||
|     except ValueError: |     except ValueError: | ||||||
|         return smart_string(f) |         return force_unicode(f) | ||||||
|     m = f - int(f) |     m = f - int(f) | ||||||
|     if not m and d < 0: |     if not m and d < 0: | ||||||
|         return '%d' % int(f) |         return u'%d' % int(f) | ||||||
|     else: |     else: | ||||||
|         formatstr = '%%.%df' % abs(d) |         formatstr = u'%%.%df' % abs(d) | ||||||
|         return formatstr % f |         return formatstr % f | ||||||
|  |  | ||||||
|  | def iriencode(value): | ||||||
|  |     "Escapes an IRI value for use in a URL" | ||||||
|  |     return force_unicode(iri_to_uri(value)) | ||||||
|  | iriencode = stringfilter(iriencode) | ||||||
|  |  | ||||||
| def linenumbers(value): | def linenumbers(value): | ||||||
|     "Displays text with line numbers" |     "Displays text with line numbers" | ||||||
|     from django.utils.html import escape |     from django.utils.html import escape | ||||||
|     lines = value.split('\n') |     lines = value.split(u'\n') | ||||||
|     # Find the maximum width of the line count, for use with zero padding string format command |     # Find the maximum width of the line count, for use with zero padding string format command | ||||||
|     width = str(len(str(len(lines)))) |     width = unicode(len(unicode(len(lines)))) | ||||||
|     for i, line in enumerate(lines): |     for i, line in enumerate(lines): | ||||||
|         lines[i] = ("%0" + width  + "d. %s") % (i + 1, escape(line)) |         lines[i] = (u"%0" + width  + u"d. %s") % (i + 1, escape(line)) | ||||||
|     return '\n'.join(lines) |     return u'\n'.join(lines) | ||||||
| linenumbers = stringfilter(linenumbers) | linenumbers = stringfilter(linenumbers) | ||||||
|  |  | ||||||
| def lower(value): | def lower(value): | ||||||
| @@ -120,8 +115,13 @@ def make_list(value): | |||||||
| make_list = stringfilter(make_list) | make_list = stringfilter(make_list) | ||||||
|  |  | ||||||
| def slugify(value): | def slugify(value): | ||||||
|     "Converts to lowercase, removes non-alpha chars and converts spaces to hyphens" |     """ | ||||||
|     value = re.sub('[^\w\s-]', '', value).strip().lower() |     Normalizes string, converts to lowercase, removes non-alpha chars and | ||||||
|  |     converts spaces to hyphens. | ||||||
|  |     """ | ||||||
|  |     import unicodedata | ||||||
|  |     value = unicodedata.normalize('NFKD', value).encode('ascii', 'ignore') | ||||||
|  |     value = unicode(re.sub('[^\w\s-]', '', value).strip().lower()) | ||||||
|     return re.sub('[-\s]+', '-', value) |     return re.sub('[-\s]+', '-', value) | ||||||
| slugify = stringfilter(slugify) | slugify = stringfilter(slugify) | ||||||
|  |  | ||||||
| @@ -135,9 +135,9 @@ def stringformat(value, arg): | |||||||
|     of Python string formatting |     of Python string formatting | ||||||
|     """ |     """ | ||||||
|     try: |     try: | ||||||
|         return ("%" + str(arg)) % value |         return (u"%" + unicode(arg)) % value | ||||||
|     except (ValueError, TypeError): |     except (ValueError, TypeError): | ||||||
|         return "" |         return u"" | ||||||
|  |  | ||||||
| def title(value): | def title(value): | ||||||
|     "Converts a string into titlecase" |     "Converts a string into titlecase" | ||||||
| @@ -155,8 +155,6 @@ def truncatewords(value, arg): | |||||||
|         length = int(arg) |         length = int(arg) | ||||||
|     except ValueError: # invalid literal for int() |     except ValueError: # invalid literal for int() | ||||||
|         return value # Fail silently. |         return value # Fail silently. | ||||||
|     if not isinstance(value, basestring): |  | ||||||
|         value = str(value) |  | ||||||
|     return truncate_words(value, length) |     return truncate_words(value, length) | ||||||
| truncatewords = stringfilter(truncatewords) | truncatewords = stringfilter(truncatewords) | ||||||
|  |  | ||||||
| @@ -171,8 +169,6 @@ def truncatewords_html(value, arg): | |||||||
|         length = int(arg) |         length = int(arg) | ||||||
|     except ValueError: # invalid literal for int() |     except ValueError: # invalid literal for int() | ||||||
|         return value # Fail silently. |         return value # Fail silently. | ||||||
|     if not isinstance(value, basestring): |  | ||||||
|         value = str(value) |  | ||||||
|     return truncate_html_words(value, length) |     return truncate_html_words(value, length) | ||||||
| truncatewords_html = stringfilter(truncatewords_html) | truncatewords_html = stringfilter(truncatewords_html) | ||||||
|  |  | ||||||
| @@ -183,10 +179,8 @@ upper = stringfilter(upper) | |||||||
|  |  | ||||||
| def urlencode(value): | def urlencode(value): | ||||||
|     "Escapes a value for use in a URL" |     "Escapes a value for use in a URL" | ||||||
|     import urllib |     from django.utils.http import urlquote | ||||||
|     if not isinstance(value, basestring): |     return urlquote(value) | ||||||
|         value = str(value) |  | ||||||
|     return urllib.quote(value) |  | ||||||
| urlencode = stringfilter(urlencode) | urlencode = stringfilter(urlencode) | ||||||
|  |  | ||||||
| def urlize(value): | def urlize(value): | ||||||
| @@ -246,7 +240,7 @@ center = stringfilter(center) | |||||||
|  |  | ||||||
| def cut(value, arg): | def cut(value, arg): | ||||||
|     "Removes all values of arg from the given string" |     "Removes all values of arg from the given string" | ||||||
|     return value.replace(arg, '') |     return value.replace(arg, u'') | ||||||
| cut = stringfilter(cut) | cut = stringfilter(cut) | ||||||
|  |  | ||||||
| ################### | ################### | ||||||
| @@ -273,11 +267,11 @@ linebreaksbr = stringfilter(linebreaksbr) | |||||||
| def removetags(value, tags): | def removetags(value, tags): | ||||||
|     "Removes a space separated list of [X]HTML tags from the output" |     "Removes a space separated list of [X]HTML tags from the output" | ||||||
|     tags = [re.escape(tag) for tag in tags.split()] |     tags = [re.escape(tag) for tag in tags.split()] | ||||||
|     tags_re = '(%s)' % '|'.join(tags) |     tags_re = u'(%s)' % u'|'.join(tags) | ||||||
|     starttag_re = re.compile(r'<%s(/?>|(\s+[^>]*>))' % tags_re) |     starttag_re = re.compile(ur'<%s(/?>|(\s+[^>]*>))' % tags_re, re.U) | ||||||
|     endtag_re = re.compile('</%s>' % tags_re) |     endtag_re = re.compile(u'</%s>' % tags_re) | ||||||
|     value = starttag_re.sub('', value) |     value = starttag_re.sub(u'', value) | ||||||
|     value = endtag_re.sub('', value) |     value = endtag_re.sub(u'', value) | ||||||
|     return value |     return value | ||||||
| removetags = stringfilter(removetags) | removetags = stringfilter(removetags) | ||||||
|  |  | ||||||
| @@ -296,7 +290,7 @@ def dictsort(value, arg): | |||||||
|     Takes a list of dicts, returns that list sorted by the property given in |     Takes a list of dicts, returns that list sorted by the property given in | ||||||
|     the argument. |     the argument. | ||||||
|     """ |     """ | ||||||
|     decorated = [(resolve_variable('var.' + arg, {'var' : item}), item) for item in value] |     decorated = [(resolve_variable(u'var.' + arg, {u'var' : item}), item) for item in value] | ||||||
|     decorated.sort() |     decorated.sort() | ||||||
|     return [item[1] for item in decorated] |     return [item[1] for item in decorated] | ||||||
|  |  | ||||||
| @@ -305,7 +299,7 @@ def dictsortreversed(value, arg): | |||||||
|     Takes a list of dicts, returns that list sorted in reverse order by the |     Takes a list of dicts, returns that list sorted in reverse order by the | ||||||
|     property given in the argument. |     property given in the argument. | ||||||
|     """ |     """ | ||||||
|     decorated = [(resolve_variable('var.' + arg, {'var' : item}), item) for item in value] |     decorated = [(resolve_variable(u'var.' + arg, {u'var' : item}), item) for item in value] | ||||||
|     decorated.sort() |     decorated.sort() | ||||||
|     decorated.reverse() |     decorated.reverse() | ||||||
|     return [item[1] for item in decorated] |     return [item[1] for item in decorated] | ||||||
| @@ -315,12 +309,12 @@ def first(value): | |||||||
|     try: |     try: | ||||||
|         return value[0] |         return value[0] | ||||||
|     except IndexError: |     except IndexError: | ||||||
|         return '' |         return u'' | ||||||
|  |  | ||||||
| def join(value, arg): | def join(value, arg): | ||||||
|     "Joins a list with a string, like Python's ``str.join(list)``" |     "Joins a list with a string, like Python's ``str.join(list)``" | ||||||
|     try: |     try: | ||||||
|         return arg.join(map(smart_string, value)) |         return arg.join(map(force_unicode, value)) | ||||||
|     except AttributeError: # fail silently but nicely |     except AttributeError: # fail silently but nicely | ||||||
|         return value |         return value | ||||||
|  |  | ||||||
| @@ -346,7 +340,7 @@ def slice_(value, arg): | |||||||
|     """ |     """ | ||||||
|     try: |     try: | ||||||
|         bits = [] |         bits = [] | ||||||
|         for x in arg.split(':'): |         for x in arg.split(u':'): | ||||||
|             if len(x) == 0: |             if len(x) == 0: | ||||||
|                 bits.append(None) |                 bits.append(None) | ||||||
|             else: |             else: | ||||||
| @@ -378,12 +372,12 @@ def unordered_list(value): | |||||||
|         </li> |         </li> | ||||||
|     """ |     """ | ||||||
|     def _helper(value, tabs): |     def _helper(value, tabs): | ||||||
|         indent = '\t' * tabs |         indent = u'\t' * tabs | ||||||
|         if value[1]: |         if value[1]: | ||||||
|             return '%s<li>%s\n%s<ul>\n%s\n%s</ul>\n%s</li>' % (indent, value[0], indent, |             return u'%s<li>%s\n%s<ul>\n%s\n%s</ul>\n%s</li>' % (indent, force_unicode(value[0]), indent, | ||||||
|                 '\n'.join([_helper(v, tabs+1) for v in value[1]]), indent, indent) |                 u'\n'.join([_helper(v, tabs+1) for v in value[1]]), indent, indent) | ||||||
|         else: |         else: | ||||||
|             return '%s<li>%s</li>' % (indent, value[0]) |             return u'%s<li>%s</li>' % (indent, force_unicode(value[0])) | ||||||
|     return _helper(value, 1) |     return _helper(value, 1) | ||||||
|  |  | ||||||
| ################### | ################### | ||||||
| @@ -421,7 +415,7 @@ def date(value, arg=None): | |||||||
|     "Formats a date according to the given format" |     "Formats a date according to the given format" | ||||||
|     from django.utils.dateformat import format |     from django.utils.dateformat import format | ||||||
|     if not value: |     if not value: | ||||||
|         return '' |         return u'' | ||||||
|     if arg is None: |     if arg is None: | ||||||
|         arg = settings.DATE_FORMAT |         arg = settings.DATE_FORMAT | ||||||
|     return format(value, arg) |     return format(value, arg) | ||||||
| @@ -429,8 +423,8 @@ def date(value, arg=None): | |||||||
| def time(value, arg=None): | def time(value, arg=None): | ||||||
|     "Formats a time according to the given format" |     "Formats a time according to the given format" | ||||||
|     from django.utils.dateformat import time_format |     from django.utils.dateformat import time_format | ||||||
|     if value in (None, ''): |     if value in (None, u''): | ||||||
|         return '' |         return u'' | ||||||
|     if arg is None: |     if arg is None: | ||||||
|         arg = settings.TIME_FORMAT |         arg = settings.TIME_FORMAT | ||||||
|     return time_format(value, arg) |     return time_format(value, arg) | ||||||
| @@ -439,7 +433,7 @@ def timesince(value, arg=None): | |||||||
|     'Formats a date as the time since that date (i.e. "4 days, 6 hours")' |     'Formats a date as the time since that date (i.e. "4 days, 6 hours")' | ||||||
|     from django.utils.timesince import timesince |     from django.utils.timesince import timesince | ||||||
|     if not value: |     if not value: | ||||||
|         return '' |         return u'' | ||||||
|     if arg: |     if arg: | ||||||
|         return timesince(arg, value) |         return timesince(arg, value) | ||||||
|     return timesince(value) |     return timesince(value) | ||||||
| @@ -449,7 +443,7 @@ def timeuntil(value, arg=None): | |||||||
|     from django.utils.timesince import timesince |     from django.utils.timesince import timesince | ||||||
|     from datetime import datetime |     from datetime import datetime | ||||||
|     if not value: |     if not value: | ||||||
|         return '' |         return u'' | ||||||
|     if arg: |     if arg: | ||||||
|         return timesince(arg, value) |         return timesince(arg, value) | ||||||
|     return timesince(datetime.now(), value) |     return timesince(datetime.now(), value) | ||||||
| @@ -488,8 +482,8 @@ def yesno(value, arg=None): | |||||||
|     ==========  ======================  ================================== |     ==========  ======================  ================================== | ||||||
|     """ |     """ | ||||||
|     if arg is None: |     if arg is None: | ||||||
|         arg = gettext('yes,no,maybe') |         arg = ugettext('yes,no,maybe') | ||||||
|     bits = arg.split(',') |     bits = arg.split(u',') | ||||||
|     if len(bits) < 2: |     if len(bits) < 2: | ||||||
|         return value # Invalid arg. |         return value # Invalid arg. | ||||||
|     try: |     try: | ||||||
| @@ -514,28 +508,28 @@ def filesizeformat(bytes): | |||||||
|     try: |     try: | ||||||
|         bytes = float(bytes) |         bytes = float(bytes) | ||||||
|     except TypeError: |     except TypeError: | ||||||
|         return "0 bytes" |         return u"0 bytes" | ||||||
|  |  | ||||||
|     if bytes < 1024: |     if bytes < 1024: | ||||||
|         return ngettext("%(size)d byte", "%(size)d bytes", bytes) % {'size': bytes} |         return ungettext("%(size)d byte", "%(size)d bytes", bytes) % {'size': bytes} | ||||||
|     if bytes < 1024 * 1024: |     if bytes < 1024 * 1024: | ||||||
|         return gettext("%.1f KB") % (bytes / 1024) |         return ugettext("%.1f KB") % (bytes / 1024) | ||||||
|     if bytes < 1024 * 1024 * 1024: |     if bytes < 1024 * 1024 * 1024: | ||||||
|         return gettext("%.1f MB") % (bytes / (1024 * 1024)) |         return ugettext("%.1f MB") % (bytes / (1024 * 1024)) | ||||||
|     return gettext("%.1f GB") % (bytes / (1024 * 1024 * 1024)) |     return ugettext("%.1f GB") % (bytes / (1024 * 1024 * 1024)) | ||||||
|  |  | ||||||
| def pluralize(value, arg='s'): | def pluralize(value, arg=u's'): | ||||||
|     """ |     """ | ||||||
|     Returns a plural suffix if the value is not 1, for '1 vote' vs. '2 votes' |     Returns a plural suffix if the value is not 1, for '1 vote' vs. '2 votes' | ||||||
|     By default, 's' is used as a suffix; if an argument is provided, that string |     By default, 's' is used as a suffix; if an argument is provided, that string | ||||||
|     is used instead. If the provided argument contains a comma, the text before |     is used instead. If the provided argument contains a comma, the text before | ||||||
|     the comma is used for the singular case. |     the comma is used for the singular case. | ||||||
|     """ |     """ | ||||||
|     if not ',' in arg: |     if not u',' in arg: | ||||||
|         arg = ',' + arg |         arg = u',' + arg | ||||||
|     bits = arg.split(',') |     bits = arg.split(u',') | ||||||
|     if len(bits) > 2: |     if len(bits) > 2: | ||||||
|         return '' |         return u'' | ||||||
|     singular_suffix, plural_suffix = bits[:2] |     singular_suffix, plural_suffix = bits[:2] | ||||||
|  |  | ||||||
|     try: |     try: | ||||||
| @@ -562,7 +556,7 @@ def pprint(value): | |||||||
|     try: |     try: | ||||||
|         return pformat(value) |         return pformat(value) | ||||||
|     except Exception, e: |     except Exception, e: | ||||||
|         return "Error in formatting:%s" % e |         return u"Error in formatting:%s" % force_unicode(e) | ||||||
|  |  | ||||||
| # Syntax: register.filter(name of filter, callback) | # Syntax: register.filter(name of filter, callback) | ||||||
| register.filter(add) | register.filter(add) | ||||||
| @@ -582,6 +576,7 @@ register.filter(first) | |||||||
| register.filter(fix_ampersands) | register.filter(fix_ampersands) | ||||||
| register.filter(floatformat) | register.filter(floatformat) | ||||||
| register.filter(get_digit) | register.filter(get_digit) | ||||||
|  | register.filter(iriencode) | ||||||
| register.filter(join) | register.filter(join) | ||||||
| register.filter(length) | register.filter(length) | ||||||
| register.filter(length_is) | register.filter(length_is) | ||||||
|   | |||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user