mirror of
https://github.com/SectorLabs/django-localized-fields.git
synced 2025-10-30 02:28:57 +03:00
Merge branch 'master' of https://github.com/SectorLabs/django-localized-fields into extra-fields
This commit is contained in:
@@ -1,6 +1,6 @@
|
||||
from .localized_field import LocalizedField
|
||||
from .localized_autoslug_field import LocalizedAutoSlugField
|
||||
from .localized_uniqueslug_field import LocalizedUniqueSlugField
|
||||
from .field import LocalizedField
|
||||
from .autoslug_field import LocalizedAutoSlugField
|
||||
from .uniqueslug_field import LocalizedUniqueSlugField
|
||||
from .localized_char_field import LocalizedCharField
|
||||
from .localized_text_field import LocalizedTextField
|
||||
from .localized_file_field import LocalizedFileField
|
||||
@@ -16,7 +16,7 @@ __all__ = [
|
||||
]
|
||||
|
||||
try:
|
||||
from .localized_bleach_field import LocalizedBleachField
|
||||
from .bleach_field import LocalizedBleachField
|
||||
__all__ += [
|
||||
'LocalizedBleachField'
|
||||
]
|
||||
|
||||
@@ -1,12 +1,12 @@
|
||||
from typing import Callable
|
||||
from typing import Callable, Tuple
|
||||
from datetime import datetime
|
||||
|
||||
from django import forms
|
||||
from django.conf import settings
|
||||
from django.utils.text import slugify
|
||||
|
||||
from .localized_field import LocalizedField
|
||||
from ..localized_value import LocalizedValue
|
||||
from .field import LocalizedField
|
||||
from ..value import LocalizedValue
|
||||
|
||||
|
||||
class LocalizedAutoSlugField(LocalizedField):
|
||||
@@ -69,13 +69,7 @@ class LocalizedAutoSlugField(LocalizedField):
|
||||
|
||||
slugs = LocalizedValue()
|
||||
|
||||
for lang_code, _ in settings.LANGUAGES:
|
||||
value = self._get_populate_from_value(
|
||||
instance,
|
||||
self.populate_from,
|
||||
lang_code
|
||||
)
|
||||
|
||||
for lang_code, value in self._get_populate_values(instance):
|
||||
if not value:
|
||||
continue
|
||||
|
||||
@@ -128,6 +122,30 @@ class LocalizedAutoSlugField(LocalizedField):
|
||||
|
||||
return unique_slug
|
||||
|
||||
def _get_populate_values(self, instance) -> Tuple[str, str]:
|
||||
"""Gets all values (for each language) from the
|
||||
specified's instance's `populate_from` field.
|
||||
|
||||
Arguments:
|
||||
instance:
|
||||
The instance to get the values from.
|
||||
|
||||
Returns:
|
||||
A list of (lang_code, value) tuples.
|
||||
"""
|
||||
|
||||
return [
|
||||
(
|
||||
lang_code,
|
||||
self._get_populate_from_value(
|
||||
instance,
|
||||
self.populate_from,
|
||||
lang_code
|
||||
),
|
||||
)
|
||||
for lang_code, _ in settings.LANGUAGES
|
||||
]
|
||||
|
||||
@staticmethod
|
||||
def _get_populate_from_value(instance, field_name: str, language: str):
|
||||
"""Gets the value to create a slug from in the specified language.
|
||||
@@ -1,8 +1,9 @@
|
||||
import bleach
|
||||
|
||||
from django.conf import settings
|
||||
from django_bleach.utils import get_bleach_default_options
|
||||
|
||||
from .localized_field import LocalizedField
|
||||
from .field import LocalizedField
|
||||
|
||||
|
||||
class LocalizedBleachField(LocalizedField):
|
||||
@@ -1,70 +1,15 @@
|
||||
import json
|
||||
|
||||
from typing import Union
|
||||
|
||||
from django.conf import settings
|
||||
from django.db.utils import IntegrityError
|
||||
from django.utils import six, translation
|
||||
|
||||
from psqlextra.fields import HStoreField
|
||||
|
||||
from ..forms import LocalizedFieldForm
|
||||
from ..localized_value import LocalizedValue
|
||||
|
||||
|
||||
class LocalizedValueDescriptor(object):
|
||||
"""
|
||||
The descriptor for the localized value attribute on the model instance.
|
||||
Returns a :see:LocalizedValue when accessed so you can do stuff like::
|
||||
|
||||
>>> from myapp.models import MyModel
|
||||
>>> instance = MyModel()
|
||||
>>> instance.value.en = 'English value'
|
||||
|
||||
Assigns a strings to active language key in :see:LocalizedValue on
|
||||
assignment so you can do::
|
||||
|
||||
>>> from django.utils import translation
|
||||
>>> from myapp.models import MyModel
|
||||
|
||||
>>> translation.activate('nl')
|
||||
>>> instance = MyModel()
|
||||
>>> instance.title = 'dutch title'
|
||||
>>> print(instance.title.nl) # prints 'dutch title'
|
||||
"""
|
||||
def __init__(self, field):
|
||||
self.field = field
|
||||
|
||||
def __get__(self, instance, cls=None):
|
||||
if instance is None:
|
||||
return self
|
||||
|
||||
# This is slightly complicated, so worth an explanation.
|
||||
# `instance.localizedvalue` needs to ultimately return some instance of
|
||||
# `LocalizedValue`, probably a subclass.
|
||||
|
||||
# The instance dict contains whatever was originally assigned
|
||||
# in __set__.
|
||||
if self.field.name in instance.__dict__:
|
||||
value = instance.__dict__[self.field.name]
|
||||
elif instance.pk is not None:
|
||||
instance.refresh_from_db(fields=[self.field.name])
|
||||
value = getattr(instance, self.field.name)
|
||||
else:
|
||||
value = None
|
||||
|
||||
if value is None:
|
||||
attr = self.field.attr_class()
|
||||
instance.__dict__[self.field.name] = attr
|
||||
|
||||
if isinstance(value, dict):
|
||||
attr = self.field.attr_class(value)
|
||||
instance.__dict__[self.field.name] = attr
|
||||
|
||||
return instance.__dict__[self.field.name]
|
||||
|
||||
def __set__(self, instance, value):
|
||||
if isinstance(value, six.string_types):
|
||||
self.__get__(instance).set(translation.get_language() or
|
||||
settings.LANGUAGE_CODE, value)
|
||||
else:
|
||||
instance.__dict__[self.field.name] = value
|
||||
from ..value import LocalizedValue
|
||||
from ..descriptor import LocalizedValueDescriptor
|
||||
|
||||
|
||||
class LocalizedField(HStoreField):
|
||||
@@ -87,9 +32,18 @@ class LocalizedField(HStoreField):
|
||||
|
||||
super(LocalizedField, self).__init__(*args, **kwargs)
|
||||
|
||||
def contribute_to_class(self, cls, name, **kwargs):
|
||||
super(LocalizedField, self).contribute_to_class(cls, name, **kwargs)
|
||||
setattr(cls, self.name, self.descriptor_class(self))
|
||||
def contribute_to_class(self, model, name, **kwargs):
|
||||
"""Adds this field to the specifed model.
|
||||
|
||||
Arguments:
|
||||
cls:
|
||||
The model to add the field to.
|
||||
|
||||
name:
|
||||
The name of the field to add.
|
||||
"""
|
||||
super(LocalizedField, self).contribute_to_class(model, name, **kwargs)
|
||||
setattr(model, self.name, self.descriptor_class(self))
|
||||
|
||||
@classmethod
|
||||
def from_db_value(cls, value, *_):
|
||||
@@ -112,9 +66,31 @@ class LocalizedField(HStoreField):
|
||||
else:
|
||||
return cls.attr_class()
|
||||
|
||||
# we can get a list if an aggregation expression was used..
|
||||
# if we the expression was flattened when only one key was selected
|
||||
# then we don't wrap each value in a localized value, otherwise we do
|
||||
if isinstance(value, list):
|
||||
result = []
|
||||
for inner_val in value:
|
||||
if isinstance(inner_val, dict):
|
||||
if inner_val is None:
|
||||
result.append(None)
|
||||
else:
|
||||
result.append(cls.attr_class(inner_val))
|
||||
else:
|
||||
result.append(inner_val)
|
||||
|
||||
return result
|
||||
|
||||
# this is for when you select an individual key, it will be string,
|
||||
# not a dictionary, we'll give it to you as a flat value, not as a
|
||||
# localized value instance
|
||||
if not isinstance(value, dict):
|
||||
return value
|
||||
|
||||
return cls.attr_class(value)
|
||||
|
||||
def to_python(self, value: dict) -> LocalizedValue:
|
||||
def to_python(self, value: Union[dict, str, None]) -> LocalizedValue:
|
||||
"""Turns the specified database value into its Python
|
||||
equivalent.
|
||||
|
||||
@@ -128,10 +104,17 @@ class LocalizedField(HStoreField):
|
||||
data extracted from the database.
|
||||
"""
|
||||
|
||||
if not value or not isinstance(value, dict):
|
||||
# first let the base class handle the deserialization, this is in case we
|
||||
# get specified a json string representing a dict
|
||||
try:
|
||||
deserialized_value = super(LocalizedField, self).to_python(value)
|
||||
except json.JSONDecodeError:
|
||||
deserialized_value = value
|
||||
|
||||
if not deserialized_value:
|
||||
return self.attr_class()
|
||||
|
||||
return self.attr_class(value)
|
||||
return self.attr_class(deserialized_value)
|
||||
|
||||
def get_prep_value(self, value: LocalizedValue) -> dict:
|
||||
"""Turns the specified value into something the database
|
||||
@@ -1,6 +1,6 @@
|
||||
from ..forms import LocalizedCharFieldForm
|
||||
from .localized_field import LocalizedField
|
||||
from ..localized_value import LocalizedStringValue
|
||||
from .field import LocalizedField
|
||||
from ..value import LocalizedStringValue
|
||||
|
||||
|
||||
class LocalizedCharField(LocalizedField):
|
||||
|
||||
@@ -8,10 +8,10 @@ from django.core.files.storage import default_storage
|
||||
from django.utils.encoding import force_str, force_text
|
||||
|
||||
from localized_fields.fields import LocalizedField
|
||||
from localized_fields.fields.localized_field import LocalizedValueDescriptor
|
||||
from localized_fields.localized_value import LocalizedValue
|
||||
from localized_fields.fields.field import LocalizedValueDescriptor
|
||||
from localized_fields.value import LocalizedValue
|
||||
|
||||
from ..localized_value import LocalizedFileValue
|
||||
from ..value import LocalizedFileValue
|
||||
from ..forms import LocalizedFileFieldForm
|
||||
|
||||
|
||||
|
||||
@@ -1,156 +0,0 @@
|
||||
from datetime import datetime
|
||||
|
||||
from django.conf import settings
|
||||
from django import forms
|
||||
from django.utils.text import slugify
|
||||
from django.db import transaction
|
||||
from django.db.utils import IntegrityError
|
||||
|
||||
|
||||
from ..util import get_language_codes
|
||||
from ..localized_value import LocalizedValue
|
||||
from .localized_field import LocalizedField
|
||||
|
||||
|
||||
class LocalizedUniqueSlugField(LocalizedField):
|
||||
"""Automatically provides slugs for a localized field upon saving."
|
||||
|
||||
An improved version of :see:LocalizedAutoSlugField,
|
||||
which adds:
|
||||
|
||||
- Concurrency safety
|
||||
- Improved performance
|
||||
|
||||
When in doubt, use this over :see:LocalizedAutoSlugField.
|
||||
"""
|
||||
|
||||
def __init__(self, *args, **kwargs):
|
||||
"""Initializes a new instance of :see:LocalizedUniqueSlugField."""
|
||||
|
||||
kwargs['uniqueness'] = kwargs.pop('uniqueness', get_language_codes())
|
||||
|
||||
self.populate_from = kwargs.pop('populate_from')
|
||||
self.include_time = kwargs.pop('include_time', False)
|
||||
|
||||
super().__init__(*args, **kwargs)
|
||||
|
||||
def deconstruct(self):
|
||||
"""Deconstructs the field into something the database
|
||||
can store."""
|
||||
|
||||
name, path, args, kwargs = super(
|
||||
LocalizedUniqueSlugField, self).deconstruct()
|
||||
|
||||
kwargs['populate_from'] = self.populate_from
|
||||
kwargs['include_time'] = self.include_time
|
||||
return name, path, args, kwargs
|
||||
|
||||
def formfield(self, **kwargs):
|
||||
"""Gets the form field associated with this field.
|
||||
|
||||
Because this is a slug field which is automatically
|
||||
populated, it should be hidden from the form.
|
||||
"""
|
||||
|
||||
defaults = {
|
||||
'form_class': forms.CharField,
|
||||
'required': False
|
||||
}
|
||||
|
||||
defaults.update(kwargs)
|
||||
|
||||
form_field = super().formfield(**defaults)
|
||||
form_field.widget = forms.HiddenInput()
|
||||
|
||||
return form_field
|
||||
|
||||
def contribute_to_class(self, cls, name, *args, **kwargs):
|
||||
"""Hook that allow us to operate with model class. We overwrite save()
|
||||
method to run retry logic.
|
||||
|
||||
Arguments:
|
||||
cls:
|
||||
Model class.
|
||||
|
||||
name:
|
||||
Name of field in model.
|
||||
"""
|
||||
# apparently in inheritance cases, contribute_to_class is called more
|
||||
# than once, so we have to be careful not to overwrite the original
|
||||
# save method.
|
||||
if not hasattr(cls, '_orig_save'):
|
||||
cls._orig_save = cls.save
|
||||
max_retries = getattr(
|
||||
settings,
|
||||
'LOCALIZED_FIELDS_MAX_RETRIES',
|
||||
100
|
||||
)
|
||||
|
||||
def _new_save(instance, *args_, **kwargs_):
|
||||
retries = 0
|
||||
while True:
|
||||
with transaction.atomic():
|
||||
try:
|
||||
slugs = self.populate_slugs(instance, retries)
|
||||
setattr(instance, name, slugs)
|
||||
instance._orig_save(*args_, **kwargs_)
|
||||
break
|
||||
except IntegrityError as e:
|
||||
if retries >= max_retries:
|
||||
raise e
|
||||
# check to be sure a slug fight caused
|
||||
# the IntegrityError
|
||||
s_e = str(e)
|
||||
if name in s_e and 'unique' in s_e:
|
||||
retries += 1
|
||||
else:
|
||||
raise e
|
||||
|
||||
cls.save = _new_save
|
||||
super().contribute_to_class(cls, name, *args, **kwargs)
|
||||
|
||||
def populate_slugs(self, instance, retries=0):
|
||||
"""Built the slug from populate_from field.
|
||||
|
||||
Arguments:
|
||||
instance:
|
||||
The model that is being saved.
|
||||
|
||||
retries:
|
||||
The value of the current attempt.
|
||||
|
||||
Returns:
|
||||
The localized slug that was generated.
|
||||
"""
|
||||
slugs = LocalizedValue()
|
||||
populates_slugs = getattr(instance, self.populate_from, {})
|
||||
for lang_code, _ in settings.LANGUAGES:
|
||||
|
||||
value = populates_slugs.get(lang_code)
|
||||
|
||||
if not value:
|
||||
continue
|
||||
|
||||
slug = slugify(value, allow_unicode=True)
|
||||
|
||||
# verify whether it's needed to re-generate a slug,
|
||||
# if not, re-use the same slug
|
||||
if instance.pk is not None:
|
||||
current_slug = getattr(instance, self.name).get(lang_code)
|
||||
if current_slug is not None:
|
||||
stripped_slug = current_slug[0:current_slug.rfind('-')]
|
||||
if slug == stripped_slug:
|
||||
slugs.set(lang_code, current_slug)
|
||||
continue
|
||||
|
||||
if self.include_time:
|
||||
slug += '-%d' % datetime.now().microsecond
|
||||
|
||||
if retries > 0:
|
||||
# do not add another - if we already added time
|
||||
if not self.include_time:
|
||||
slug += '-'
|
||||
slug += '%d' % retries
|
||||
|
||||
slugs.set(lang_code, slug)
|
||||
return slugs
|
||||
104
localized_fields/fields/uniqueslug_field.py
Normal file
104
localized_fields/fields/uniqueslug_field.py
Normal file
@@ -0,0 +1,104 @@
|
||||
from datetime import datetime
|
||||
|
||||
from django.utils.text import slugify
|
||||
from django.core.exceptions import ImproperlyConfigured
|
||||
|
||||
from .autoslug_field import LocalizedAutoSlugField
|
||||
from ..util import get_language_codes
|
||||
from ..mixins import AtomicSlugRetryMixin
|
||||
from ..value import LocalizedValue
|
||||
|
||||
|
||||
class LocalizedUniqueSlugField(LocalizedAutoSlugField):
|
||||
"""Automatically provides slugs for a localized
|
||||
field upon saving."
|
||||
|
||||
An improved version of :see:LocalizedAutoSlugField,
|
||||
which adds:
|
||||
|
||||
- Concurrency safety
|
||||
- Improved performance
|
||||
|
||||
When in doubt, use this over :see:LocalizedAutoSlugField.
|
||||
Inherit from :see:AtomicSlugRetryMixin in your model to
|
||||
make this field work properly.
|
||||
"""
|
||||
|
||||
def __init__(self, *args, **kwargs):
|
||||
"""Initializes a new instance of :see:LocalizedUniqueSlugField."""
|
||||
|
||||
kwargs['uniqueness'] = kwargs.pop('uniqueness', get_language_codes())
|
||||
|
||||
super(LocalizedUniqueSlugField, self).__init__(
|
||||
*args,
|
||||
**kwargs
|
||||
)
|
||||
|
||||
self.populate_from = kwargs.pop('populate_from')
|
||||
self.include_time = kwargs.pop('include_time', False)
|
||||
|
||||
def deconstruct(self):
|
||||
"""Deconstructs the field into something the database
|
||||
can store."""
|
||||
|
||||
name, path, args, kwargs = super(
|
||||
LocalizedUniqueSlugField, self).deconstruct()
|
||||
|
||||
kwargs['populate_from'] = self.populate_from
|
||||
kwargs['include_time'] = self.include_time
|
||||
return name, path, args, kwargs
|
||||
|
||||
def pre_save(self, instance, add: bool):
|
||||
"""Ran just before the model is saved, allows us to built
|
||||
the slug.
|
||||
|
||||
Arguments:
|
||||
instance:
|
||||
The model that is being saved.
|
||||
|
||||
add:
|
||||
Indicates whether this is a new entry
|
||||
to the database or an update.
|
||||
|
||||
Returns:
|
||||
The localized slug that was generated.
|
||||
"""
|
||||
|
||||
if not isinstance(instance, AtomicSlugRetryMixin):
|
||||
raise ImproperlyConfigured((
|
||||
'Model \'%s\' does not inherit from AtomicSlugRetryMixin. '
|
||||
'Without this, the LocalizedUniqueSlugField will not work.'
|
||||
) % type(instance).__name__)
|
||||
|
||||
slugs = LocalizedValue()
|
||||
|
||||
for lang_code, value in self._get_populate_values(instance):
|
||||
if not value:
|
||||
continue
|
||||
|
||||
slug = slugify(value, allow_unicode=True)
|
||||
|
||||
# verify whether it's needed to re-generate a slug,
|
||||
# if not, re-use the same slug
|
||||
if instance.pk is not None:
|
||||
current_slug = getattr(instance, self.name).get(lang_code)
|
||||
if current_slug is not None:
|
||||
stripped_slug = current_slug[0:current_slug.rfind('-')]
|
||||
if slug == stripped_slug:
|
||||
slugs.set(lang_code, current_slug)
|
||||
continue
|
||||
|
||||
if self.include_time:
|
||||
slug += '-%d' % datetime.now().microsecond
|
||||
|
||||
retries = getattr(instance, 'retries', 0)
|
||||
if retries > 0:
|
||||
# do not add another - if we already added time
|
||||
if not self.include_time:
|
||||
slug += '-'
|
||||
slug += '%d' % retries
|
||||
|
||||
slugs.set(lang_code, slug)
|
||||
|
||||
setattr(instance, self.name, slugs)
|
||||
return slugs
|
||||
Reference in New Issue
Block a user