# # This file is part of pretix (Community Edition). # # Copyright (C) 2014-2020 Raphael Michel and contributors # Copyright (C) 2020-today pretix GmbH and contributors # # This program is free software: you can redistribute it and/or modify it under the terms of the GNU Affero General # Public License as published by the Free Software Foundation in version 3 of the License. # # ADDITIONAL TERMS APPLY: Pursuant to Section 7 of the GNU Affero General Public License, additional terms are # applicable granting you additional permissions and placing additional restrictions on your usage of this software. # Please refer to the pretix LICENSE file to obtain the full terms applicable to this work. If you did not receive # this file, see . # # This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied # warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public License for more # details. # # You should have received a copy of the GNU Affero General Public License along with this program. If not, see # . # from babel.core import Locale from django.core.cache import cache from django.utils import translation from django.utils.translation import gettext_noop from django_countries import Countries, collator from django_countries.fields import CountryField from phonenumbers import ( COUNTRY_CODE_TO_REGION_CODE, REGION_CODE_FOR_NON_GEO_ENTITY, ) from pretix.base.i18n import get_babel_locale, get_language_without_region class CachedCountries(Countries): _cached_lists = {} cache_subkey = None def __iter__(self): """ Iterate through countries, sorted by name, but cache the results based on the locale. django-countries performs a unicode-aware sorting based on pyuca which is incredibly slow. """ # Starting in django-countries 8.1, django-countries implemented a similar caching, but only on object level. # We keep our caching for now for the added caching to the cache store. Unfortunately we can't really avoid # the double-caching on object level if we want the caches od be used in a sensible order. We could re-evaluate # and drop this in the future if it ever causes bugs or memory issues. cache_key = "countries:all:{}".format(get_language_without_region()) if self.cache_subkey: cache_key += ":" + self.cache_subkey if cache_key in self._cached_lists: yield from self._cached_lists[cache_key] return val = cache.get(cache_key) if val: self._cached_lists[cache_key] = val yield from val return val = list(super().__iter__()) self._cached_lists[cache_key] = val cache.set(cache_key, val, 3600 * 24 * 30) yield from val class FastCountryField(CountryField): def __init__(self, *args, **kwargs): kwargs.setdefault("countries", CachedCountries) if "max_length" not in kwargs: # Override logic from CountryField to include 20% buffer. We don't want to migrate our database # every time a new country is added to the system! if kwargs.get("multiple", False): kwargs["max_length"] = int(len(kwargs['countries']()) * 3 * 1.2) else: kwargs["max_length"] = 2 super().__init__(*args, **kwargs) def check(self, **kwargs): # Disable _check_choices since it would require sorting all country names at every import of this field, # which takes 1-2 seconds return [ *self._check_field_name(), # *self._check_choices(), *self._check_db_index(), *self._check_null_allowed_for_primary_keys(), *self._check_backend_specific_checks(**kwargs), *self._check_validators(), *self._check_deprecation_details(), ] _cached_phone_prefixes = {} def get_phone_prefixes_sorted_and_localized(): language = get_babel_locale() # changed from default implementation that used the django locale cache_key = "phoneprefixes:all:{}".format(language) if cache_key in _cached_phone_prefixes: return _cached_phone_prefixes[cache_key] val = cache.get(cache_key) if val: _cached_phone_prefixes[cache_key] = val return val val = [] locale = Locale(translation.to_locale(language)) for prefix, values in COUNTRY_CODE_TO_REGION_CODE.items(): prefix = "+%d" % prefix for country_code in values: if country_code == REGION_CODE_FOR_NON_GEO_ENTITY: continue country_name = locale.territories.get(country_code) if country_name: val.append((prefix, "{} {}".format(country_name, prefix))) val = sorted(val, key=lambda item: collator.sort_key(item[1])) _cached_phone_prefixes[cache_key] = val cache.set(cache_key, val, 3600 * 24 * 30) return val custom_translations = [ # Hotfix to allow pretix to provide custom translations until # https://github.com/SmileyChris/django-countries/pull/471 # is merged gettext_noop("Belarus"), gettext_noop("French Guiana"), gettext_noop("North Macedonia"), gettext_noop("Macao"), ] def pycountry_add(db, **kw): # Workaround for https://github.com/pycountry/pycountry/issues/281 db._load() obj = db.factory(**kw) db.objects.append(obj) for key, value in kw.items(): if key in db.no_index: continue value = value.lower() index = db.indices.setdefault(key, {}) if key in ["country_code"]: index.setdefault(value, set()).add(obj) else: index[value] = obj