Вы не можете выбрать более 25 тем Темы должны начинаться с буквы или цифры, могут содержать дефисы(-) и должны содержать не более 35 символов.

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362
  1. import string
  2. import random
  3. from django.contrib.auth.forms import AuthenticationForm
  4. from django.utils.translation import ugettext_lazy as _
  5. from django.core.exceptions import ObjectDoesNotExist
  6. from django import forms
  7. from django.db.models import get_model
  8. from django.contrib.auth.models import User
  9. from django.contrib.auth import forms as auth_forms
  10. from django.conf import settings
  11. from django.core import validators
  12. from django.core.exceptions import ValidationError
  13. from django.utils.http import int_to_base36
  14. from django.contrib.sites.models import get_current_site
  15. from django.contrib.auth.tokens import default_token_generator
  16. from oscar.core.loading import get_profile_class, get_class
  17. Dispatcher = get_class('customer.utils', 'Dispatcher')
  18. CommunicationEventType = get_model('customer', 'communicationeventtype')
  19. ProductAlert = get_model('customer', 'ProductAlert')
  20. def generate_username():
  21. uname = ''.join([random.choice(string.letters + string.digits + '_') for i in range(30)])
  22. try:
  23. User.objects.get(username=uname)
  24. return generate_username()
  25. except User.DoesNotExist:
  26. return uname
  27. class PasswordResetForm(auth_forms.PasswordResetForm):
  28. communication_type_code = "PASSWORD_RESET"
  29. def save(self, subject_template_name='registration/password_reset_subject.txt',
  30. email_template_name='registration/password_reset_email.html',
  31. use_https=False, token_generator=default_token_generator,
  32. from_email=None, request=None, **kwargs):
  33. """
  34. Generates a one-use only link for resetting password and sends to the
  35. user.
  36. """
  37. for user in self.users_cache:
  38. current_site = get_current_site(request)
  39. ctx = {
  40. 'email': user.email,
  41. 'domain': current_site.domain,
  42. 'site_name': current_site.name,
  43. 'uid': int_to_base36(user.id),
  44. 'token': token_generator.make_token(user),
  45. 'protocol': use_https and 'https' or 'http',
  46. 'site': current_site,
  47. }
  48. self.send_reset_email(user, ctx)
  49. def send_reset_email(self, user, extra_context=None):
  50. code = self.communication_type_code
  51. ctx = {
  52. 'user': user,
  53. }
  54. if extra_context:
  55. ctx.update(extra_context)
  56. try:
  57. event_type = CommunicationEventType.objects.get(code=code)
  58. except CommunicationEventType.DoesNotExist:
  59. # No event in database, attempt to find templates for this type
  60. messages = CommunicationEventType.objects.get_and_render(code, ctx)
  61. else:
  62. # Create order event
  63. messages = event_type.get_messages(ctx)
  64. if messages and messages['body']:
  65. dispatcher = Dispatcher()
  66. dispatcher.dispatch_user_messages(user, messages)
  67. class EmailAuthenticationForm(AuthenticationForm):
  68. """
  69. Extends the standard django AuthenticationForm, to support 75 character
  70. usernames. 75 character usernames are needed to support the EmailOrUsername
  71. auth backend.
  72. """
  73. username = forms.EmailField(label=_('Email Address'))
  74. class CommonPasswordValidator(validators.BaseValidator):
  75. # See http://www.smartplanet.com/blog/business-brains/top-20-most-common-passwords-of-all-time-revealed-8216123456-8216princess-8216qwerty/4519
  76. forbidden_passwords = [
  77. 'password',
  78. '1234',
  79. '12345'
  80. '123456',
  81. '123456y',
  82. '123456789',
  83. 'iloveyou',
  84. 'princess',
  85. 'monkey',
  86. 'rockyou',
  87. 'babygirl',
  88. 'monkey',
  89. 'qwerty',
  90. '654321',
  91. 'dragon',
  92. 'pussy',
  93. 'baseball',
  94. 'football',
  95. 'letmein',
  96. 'monkey',
  97. '696969',
  98. 'abc123',
  99. 'qwe123',
  100. 'qweasd',
  101. 'mustang',
  102. 'michael',
  103. 'shadow',
  104. 'master',
  105. 'jennifer',
  106. '111111',
  107. '2000',
  108. 'jordan',
  109. 'superman'
  110. 'harley'
  111. ]
  112. message = _("Please choose a less common password")
  113. code = 'password'
  114. def __init__(self, password_file=None):
  115. self.limit_value = password_file
  116. def clean(self, value):
  117. return value.strip()
  118. def compare(self, value, limit):
  119. return value in self.forbidden_passwords
  120. def get_forbidden_passwords(self):
  121. if self.limit_value is None:
  122. return self.forbidden_passwords
  123. class EmailUserCreationForm(forms.ModelForm):
  124. email = forms.EmailField(label=_('Email Address'))
  125. password1 = forms.CharField(label=_('Password'), widget=forms.PasswordInput,
  126. validators=[validators.MinLengthValidator(6),
  127. CommonPasswordValidator()])
  128. password2 = forms.CharField(label=_('Confirm Password'), widget=forms.PasswordInput)
  129. class Meta:
  130. model = User
  131. fields = ('email',)
  132. def clean_email(self):
  133. email = self.cleaned_data['email'].lower()
  134. try:
  135. User.objects.get(email=email)
  136. except User.DoesNotExist:
  137. return email
  138. raise forms.ValidationError(_("A user with that email address already exists."))
  139. def clean_password2(self):
  140. password1 = self.cleaned_data.get('password1', '')
  141. password2 = self.cleaned_data.get('password2', '')
  142. if password1 != password2:
  143. raise forms.ValidationError(_("The two password fields didn't match."))
  144. return password2
  145. def save(self, commit=True):
  146. user = super(EmailUserCreationForm, self).save(commit=False)
  147. user.set_password(self.cleaned_data['password1'])
  148. user.username = generate_username()
  149. if commit:
  150. user.save()
  151. return user
  152. class SearchByDateRangeForm(forms.Form):
  153. date_from = forms.DateField(required=False, label="From")
  154. date_to = forms.DateField(required=False, label="To")
  155. def clean(self):
  156. if self.is_valid() and not self.cleaned_data['date_from'] and not self.cleaned_data['date_to']:
  157. raise forms.ValidationError(_("At least one date field is required."))
  158. return super(SearchByDateRangeForm, self).clean()
  159. def description(self):
  160. if not self.is_bound or not self.is_valid():
  161. return 'All orders'
  162. date_from = self.cleaned_data['date_from']
  163. date_to = self.cleaned_data['date_to']
  164. if date_from and date_to:
  165. return 'Orders placed between %s and %s' % (date_from, date_to)
  166. elif date_from and not date_to:
  167. return 'Orders placed since %s' % date_from
  168. elif not date_from and date_to:
  169. return 'Orders placed until %s' % date_to
  170. def get_filters(self):
  171. date_from = self.cleaned_data['date_from']
  172. date_to = self.cleaned_data['date_to']
  173. if date_from and date_to:
  174. return {'date_placed__range': [date_from, date_to]}
  175. elif date_from and not date_to:
  176. return {'date_placed__gt': date_from}
  177. elif not date_from and date_to:
  178. return {'date_placed__lt': date_to}
  179. return {}
  180. class CleanEmailMixin(object):
  181. def clean_email(self):
  182. """
  183. Make sure that the email address is aways unique as it is
  184. used instead of the username. This is necessary because the
  185. unique-ness of email addresses is *not* enforced on the model
  186. level in ``django.contrib.auth.models.User``.
  187. """
  188. email = self.cleaned_data['email']
  189. try:
  190. user = User.objects.get(email=email)
  191. except User.DoesNotExist:
  192. # this email address is unique so we don't have to worry
  193. # about it
  194. return email
  195. if self.instance and self.instance.id != user.id:
  196. raise ValidationError(
  197. _("A user with this email address already exists")
  198. )
  199. return email
  200. class UserForm(forms.ModelForm, CleanEmailMixin):
  201. def __init__(self, user, *args, **kwargs):
  202. self.user = user
  203. kwargs['instance'] = user
  204. super(UserForm, self).__init__(*args, **kwargs)
  205. class Meta:
  206. model = User
  207. exclude = ('username', 'password', 'is_staff', 'is_superuser',
  208. 'is_active', 'last_login', 'date_joined',
  209. 'user_permissions', 'groups')
  210. if hasattr(settings, 'AUTH_PROFILE_MODULE'):
  211. Profile = get_profile_class()
  212. class UserAndProfileForm(forms.ModelForm, CleanEmailMixin):
  213. first_name = forms.CharField(label=_('First name'), max_length=128)
  214. last_name = forms.CharField(label=_('Last name'), max_length=128)
  215. email = forms.EmailField(label=_('Email address'))
  216. # Fields from user model
  217. user_fields = ('first_name', 'last_name', 'email')
  218. def __init__(self, user, *args, **kwargs):
  219. self.user = user
  220. try:
  221. instance = user.get_profile()
  222. except ObjectDoesNotExist:
  223. # User has no profile, try a blank one
  224. instance = Profile(user=user)
  225. kwargs['instance'] = instance
  226. super(UserAndProfileForm, self).__init__(*args, **kwargs)
  227. # Add user fields
  228. self.fields['first_name'].initial = self.instance.user.first_name
  229. self.fields['last_name'].initial = self.instance.user.last_name
  230. self.fields['email'].initial = self.instance.user.email
  231. # Ensure user fields are above profile
  232. order = list(self.user_fields)
  233. for field_name in self.fields.keys():
  234. if field_name not in self.user_fields:
  235. order.append(field_name)
  236. self.fields.keyOrder = order
  237. class Meta:
  238. model = Profile
  239. exclude = ('user',)
  240. def save(self, *args, **kwargs):
  241. user = self.instance.user
  242. user.first_name = self.cleaned_data['first_name']
  243. user.last_name = self.cleaned_data['last_name']
  244. user.email = self.cleaned_data['email']
  245. user.save()
  246. return super(ProfileForm, self).save(*args,**kwargs)
  247. ProfileForm = UserAndProfileForm
  248. else:
  249. ProfileForm = UserForm
  250. class ProductAlertForm(forms.ModelForm):
  251. email = forms.EmailField(required=True, label=_(u'Send notification to'),
  252. widget=forms.TextInput(attrs={
  253. 'placeholder': _('Enter your email')
  254. }))
  255. def __init__(self, user, product, *args, **kwargs):
  256. self.user = user
  257. self.product = product
  258. super(ProductAlertForm, self).__init__(*args, **kwargs)
  259. # Only show email field to unauthenticated users
  260. if user and user.is_authenticated():
  261. self.fields['email'].widget = forms.HiddenInput()
  262. self.fields['email'].required = False
  263. def save(self, commit=True):
  264. alert = super(ProductAlertForm, self).save(commit=False)
  265. if self.user.is_authenticated():
  266. alert.user = self.user
  267. alert.product = self.product
  268. if commit:
  269. alert.save()
  270. return alert
  271. def clean(self):
  272. cleaned_data = self.cleaned_data
  273. email = cleaned_data.get('email')
  274. if email:
  275. try:
  276. ProductAlert.objects.get(
  277. product=self.product, email=email,
  278. status=ProductAlert.ACTIVE)
  279. except ProductAlert.DoesNotExist:
  280. pass
  281. else:
  282. raise forms.ValidationError(_(
  283. "There is already an active stock alert for %s") % email)
  284. elif self.user.is_authenticated():
  285. try:
  286. ProductAlert.objects.get(product=self.product,
  287. user=self.user,
  288. status=ProductAlert.ACTIVE)
  289. except ProductAlert.DoesNotExist:
  290. pass
  291. else:
  292. raise forms.ValidationError(_(
  293. "You already have an active alert for this product"))
  294. return cleaned_data
  295. class Meta:
  296. model = ProductAlert
  297. exclude = ('user', 'key',
  298. 'status', 'date_confirmed', 'date_cancelled', 'date_closed',
  299. 'product')