2017-02-15 15:06:47 +00:00
|
|
|
|
import re
|
|
|
|
|
|
|
2016-08-07 09:17:49 +01:00
|
|
|
|
import pytz
|
2017-01-18 15:11:34 +00:00
|
|
|
|
from flask_login import current_user
|
2015-11-27 09:47:29 +00:00
|
|
|
|
from flask_wtf import Form
|
2016-08-07 09:17:49 +01:00
|
|
|
|
from datetime import datetime, timedelta
|
2016-04-14 12:00:55 +01:00
|
|
|
|
from notifications_utils.recipients import (
|
2016-03-31 15:17:05 +01:00
|
|
|
|
validate_phone_number,
|
|
|
|
|
|
InvalidPhoneError
|
|
|
|
|
|
)
|
2016-01-12 10:43:23 +00:00
|
|
|
|
from wtforms import (
|
2016-05-11 09:43:55 +01:00
|
|
|
|
validators,
|
2016-01-12 10:43:23 +00:00
|
|
|
|
StringField,
|
|
|
|
|
|
PasswordField,
|
|
|
|
|
|
ValidationError,
|
|
|
|
|
|
TextAreaField,
|
2016-01-19 15:54:12 +00:00
|
|
|
|
FileField,
|
2016-03-02 15:25:04 +00:00
|
|
|
|
BooleanField,
|
2016-05-11 09:43:55 +01:00
|
|
|
|
HiddenField,
|
2016-06-29 17:10:49 +01:00
|
|
|
|
IntegerField,
|
2016-09-20 12:30:00 +01:00
|
|
|
|
RadioField,
|
2016-12-28 14:44:53 +00:00
|
|
|
|
FieldList,
|
2017-01-18 15:11:34 +00:00
|
|
|
|
DateField,
|
|
|
|
|
|
SelectField)
|
2017-03-14 10:46:38 +00:00
|
|
|
|
from wtforms.fields.html5 import EmailField, TelField, SearchField
|
2016-09-20 12:30:00 +01:00
|
|
|
|
from wtforms.validators import (DataRequired, Email, Length, Regexp, Optional)
|
2016-01-11 15:00:51 +00:00
|
|
|
|
|
2017-02-13 13:11:29 +00:00
|
|
|
|
from app.main.validators import (Blacklist, CsvFileValidator, ValidGovEmail, NoCommasInPlaceHolders, OnlyGSMCharacters)
|
2016-02-01 16:57:40 +00:00
|
|
|
|
|
2015-11-27 09:47:29 +00:00
|
|
|
|
|
2016-08-07 09:17:49 +01:00
|
|
|
|
def get_time_value_and_label(future_time):
|
|
|
|
|
|
return (
|
|
|
|
|
|
future_time.replace(tzinfo=None).isoformat(),
|
2016-10-11 14:11:10 +01:00
|
|
|
|
'{} at {}'.format(
|
|
|
|
|
|
get_human_day(future_time.astimezone(pytz.timezone('Europe/London'))),
|
|
|
|
|
|
get_human_time(future_time.astimezone(pytz.timezone('Europe/London')))
|
|
|
|
|
|
)
|
2016-08-07 09:17:49 +01:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def get_human_time(time):
|
|
|
|
|
|
return {
|
2016-10-11 14:11:10 +01:00
|
|
|
|
'0': 'midnight',
|
|
|
|
|
|
'12': 'midday'
|
2016-08-07 09:17:49 +01:00
|
|
|
|
}.get(
|
|
|
|
|
|
time.strftime('%-H'),
|
|
|
|
|
|
time.strftime('%-I%p').lower()
|
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
|
2016-10-11 17:59:09 +01:00
|
|
|
|
def get_human_day(time, prefix_today_with='T'):
|
2016-10-11 14:11:10 +01:00
|
|
|
|
# Add 1 hour to get ‘midnight today’ instead of ‘midnight tomorrow’
|
|
|
|
|
|
time = (time - timedelta(hours=1)).strftime('%A')
|
|
|
|
|
|
if time == datetime.utcnow().strftime('%A'):
|
2016-10-11 17:59:09 +01:00
|
|
|
|
return '{}oday'.format(prefix_today_with)
|
2016-10-11 14:11:10 +01:00
|
|
|
|
if time == (datetime.utcnow() + timedelta(days=1)).strftime('%A'):
|
|
|
|
|
|
return 'Tomorrow'
|
|
|
|
|
|
return time
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def get_furthest_possible_scheduled_time():
|
|
|
|
|
|
return (datetime.utcnow() + timedelta(days=4)).replace(hour=0)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def get_next_hours_until(until):
|
|
|
|
|
|
now = datetime.utcnow()
|
|
|
|
|
|
hours = int((until - now).total_seconds() / (60 * 60))
|
2016-08-07 09:17:49 +01:00
|
|
|
|
return [
|
|
|
|
|
|
(now + timedelta(hours=i)).replace(minute=0, second=0).replace(tzinfo=pytz.utc)
|
|
|
|
|
|
for i in range(1, hours + 1)
|
|
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
|
|
|
2016-10-11 14:11:10 +01:00
|
|
|
|
def get_next_days_until(until):
|
|
|
|
|
|
now = datetime.utcnow()
|
|
|
|
|
|
days = int((until - now).total_seconds() / (60 * 60 * 24))
|
|
|
|
|
|
return [
|
2016-10-11 17:59:09 +01:00
|
|
|
|
get_human_day(
|
|
|
|
|
|
(now + timedelta(days=i)).replace(tzinfo=pytz.utc),
|
|
|
|
|
|
prefix_today_with='Later t'
|
|
|
|
|
|
)
|
2016-10-11 14:11:10 +01:00
|
|
|
|
for i in range(0, days + 1)
|
|
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
|
|
|
2016-10-26 14:01:01 +01:00
|
|
|
|
def email_address(label='Email address', gov_user=True):
|
|
|
|
|
|
validators = [
|
2016-01-08 12:00:52 +00:00
|
|
|
|
Length(min=5, max=255),
|
2016-04-04 10:42:04 +01:00
|
|
|
|
DataRequired(message='Can’t be empty'),
|
2016-10-26 14:01:01 +01:00
|
|
|
|
Email(message='Enter a valid email address')
|
|
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
|
|
if gov_user:
|
2016-10-28 10:45:05 +01:00
|
|
|
|
validators.append(ValidGovEmail())
|
2016-10-26 14:01:01 +01:00
|
|
|
|
return EmailField(label, validators)
|
2016-01-08 12:00:52 +00:00
|
|
|
|
|
|
|
|
|
|
|
2016-02-15 13:13:57 +00:00
|
|
|
|
class UKMobileNumber(TelField):
|
2016-01-12 18:10:16 +00:00
|
|
|
|
def pre_validate(self, form):
|
2016-02-01 16:57:40 +00:00
|
|
|
|
try:
|
2016-03-08 07:17:39 +00:00
|
|
|
|
validate_phone_number(self.data)
|
2016-02-01 16:57:40 +00:00
|
|
|
|
except InvalidPhoneError as e:
|
2016-12-20 14:38:34 +00:00
|
|
|
|
raise ValidationError(str(e))
|
2016-01-12 18:10:16 +00:00
|
|
|
|
|
|
|
|
|
|
|
2016-01-08 12:00:52 +00:00
|
|
|
|
def mobile_number():
|
2016-04-07 14:12:40 +01:00
|
|
|
|
return UKMobileNumber('Mobile number',
|
2016-04-04 10:42:04 +01:00
|
|
|
|
validators=[DataRequired(message='Can’t be empty')])
|
2016-01-08 12:00:52 +00:00
|
|
|
|
|
|
|
|
|
|
|
2016-04-07 14:12:40 +01:00
|
|
|
|
def password(label='Password'):
|
2016-01-12 11:25:46 +00:00
|
|
|
|
return PasswordField(label,
|
2016-04-04 10:42:04 +01:00
|
|
|
|
validators=[DataRequired(message='Can’t be empty'),
|
2016-09-26 09:29:50 +01:00
|
|
|
|
Length(8, 255, message='Must be at least 8 characters'),
|
2016-09-27 11:37:20 +01:00
|
|
|
|
Blacklist(message='Choose a password that’s harder to guess')])
|
2016-01-08 12:00:52 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def sms_code():
|
|
|
|
|
|
verify_code = '^\d{5}$'
|
2016-02-02 15:41:54 +00:00
|
|
|
|
return StringField('Text message code',
|
2016-04-04 10:42:04 +01:00
|
|
|
|
validators=[DataRequired(message='Can’t be empty'),
|
2016-01-08 12:00:52 +00:00
|
|
|
|
Regexp(regex=verify_code,
|
2017-02-15 14:56:22 +00:00
|
|
|
|
message='Code not found')])
|
2016-01-08 12:00:52 +00:00
|
|
|
|
|
|
|
|
|
|
|
2015-11-27 09:47:29 +00:00
|
|
|
|
class LoginForm(Form):
|
|
|
|
|
|
email_address = StringField('Email address', validators=[
|
2015-11-27 16:25:56 +00:00
|
|
|
|
Length(min=5, max=255),
|
2016-04-04 10:42:04 +01:00
|
|
|
|
DataRequired(message='Can’t be empty'),
|
2016-01-08 12:00:52 +00:00
|
|
|
|
Email(message='Enter a valid email address')
|
2015-11-27 09:47:29 +00:00
|
|
|
|
])
|
2015-12-02 15:23:03 +00:00
|
|
|
|
password = PasswordField('Password', validators=[
|
2016-01-08 12:00:52 +00:00
|
|
|
|
DataRequired(message='Enter your password')
|
2015-11-27 09:47:29 +00:00
|
|
|
|
])
|
2015-12-01 13:23:54 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class RegisterUserForm(Form):
|
2015-12-02 15:23:03 +00:00
|
|
|
|
name = StringField('Full name',
|
2016-04-04 10:42:04 +01:00
|
|
|
|
validators=[DataRequired(message='Can’t be empty')])
|
2016-01-08 12:00:52 +00:00
|
|
|
|
email_address = email_address()
|
|
|
|
|
|
mobile_number = mobile_number()
|
|
|
|
|
|
password = password()
|
2015-12-04 16:21:01 +00:00
|
|
|
|
|
|
|
|
|
|
|
2016-03-02 15:25:04 +00:00
|
|
|
|
class RegisterUserFromInviteForm(Form):
|
|
|
|
|
|
name = StringField('Full name',
|
2016-04-04 10:42:04 +01:00
|
|
|
|
validators=[DataRequired(message='Can’t be empty')])
|
2016-03-02 15:25:04 +00:00
|
|
|
|
mobile_number = mobile_number()
|
|
|
|
|
|
password = password()
|
|
|
|
|
|
service = HiddenField('service')
|
2016-03-08 16:29:05 +00:00
|
|
|
|
email_address = HiddenField('email_address')
|
2016-03-02 15:25:04 +00:00
|
|
|
|
|
|
|
|
|
|
|
2016-03-22 13:18:06 +00:00
|
|
|
|
class PermissionsForm(Form):
|
|
|
|
|
|
send_messages = BooleanField("Send messages from existing templates")
|
2017-02-23 10:48:10 +00:00
|
|
|
|
manage_service = BooleanField("Modify this service, its team, and its templates")
|
2016-03-22 13:18:06 +00:00
|
|
|
|
manage_api_keys = BooleanField("Create and revoke API keys")
|
2016-03-03 13:00:12 +00:00
|
|
|
|
|
2016-02-19 15:02:13 +00:00
|
|
|
|
|
2016-03-22 13:18:06 +00:00
|
|
|
|
class InviteUserForm(PermissionsForm):
|
2016-10-26 14:01:01 +01:00
|
|
|
|
email_address = email_address(gov_user=False)
|
2016-03-09 13:00:52 +00:00
|
|
|
|
|
|
|
|
|
|
def __init__(self, invalid_email_address, *args, **kwargs):
|
|
|
|
|
|
super(InviteUserForm, self).__init__(*args, **kwargs)
|
|
|
|
|
|
self.invalid_email_address = invalid_email_address.lower()
|
|
|
|
|
|
|
|
|
|
|
|
def validate_email_address(self, field):
|
|
|
|
|
|
if field.data.lower() == self.invalid_email_address:
|
2016-04-04 10:42:04 +01:00
|
|
|
|
raise ValidationError("You can’t send an invitation to yourself")
|
2016-03-09 13:00:52 +00:00
|
|
|
|
|
|
|
|
|
|
|
2015-12-07 16:56:11 +00:00
|
|
|
|
class TwoFactorForm(Form):
|
2016-01-27 12:22:32 +00:00
|
|
|
|
def __init__(self, validate_code_func, *args, **kwargs):
|
2016-01-07 12:43:10 +00:00
|
|
|
|
'''
|
|
|
|
|
|
Keyword arguments:
|
2016-01-27 12:22:32 +00:00
|
|
|
|
validate_code_func -- Validates the code with the API.
|
2016-01-07 12:43:10 +00:00
|
|
|
|
'''
|
2016-01-27 12:22:32 +00:00
|
|
|
|
self.validate_code_func = validate_code_func
|
2016-01-07 12:43:10 +00:00
|
|
|
|
super(TwoFactorForm, self).__init__(*args, **kwargs)
|
|
|
|
|
|
|
2016-01-08 12:00:52 +00:00
|
|
|
|
sms_code = sms_code()
|
2015-12-08 12:36:54 +00:00
|
|
|
|
|
2016-01-27 12:22:32 +00:00
|
|
|
|
def validate_sms_code(self, field):
|
|
|
|
|
|
is_valid, reason = self.validate_code_func(field.data)
|
|
|
|
|
|
if not is_valid:
|
|
|
|
|
|
raise ValidationError(reason)
|
|
|
|
|
|
|
2015-12-07 16:56:11 +00:00
|
|
|
|
|
2015-12-15 15:35:30 +00:00
|
|
|
|
class EmailNotReceivedForm(Form):
|
2016-01-08 12:00:52 +00:00
|
|
|
|
email_address = email_address()
|
2015-12-15 15:35:30 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class TextNotReceivedForm(Form):
|
2016-01-08 12:00:52 +00:00
|
|
|
|
mobile_number = mobile_number()
|
2015-12-15 15:35:30 +00:00
|
|
|
|
|
|
|
|
|
|
|
2015-12-14 17:12:28 +00:00
|
|
|
|
class AddServiceForm(Form):
|
2016-01-15 16:10:24 +00:00
|
|
|
|
def __init__(self, names_func, *args, **kwargs):
|
|
|
|
|
|
"""
|
|
|
|
|
|
Keyword arguments:
|
|
|
|
|
|
names_func -- Returns a list of unique service_names already registered
|
|
|
|
|
|
on the system.
|
|
|
|
|
|
"""
|
|
|
|
|
|
self._names_func = names_func
|
2016-01-04 15:31:50 +00:00
|
|
|
|
super(AddServiceForm, self).__init__(*args, **kwargs)
|
|
|
|
|
|
|
2016-01-19 09:49:01 +00:00
|
|
|
|
name = StringField(
|
2016-01-18 10:47:53 +00:00
|
|
|
|
'Service name',
|
|
|
|
|
|
validators=[
|
2016-04-04 10:42:04 +01:00
|
|
|
|
DataRequired(message='Can’t be empty')
|
2016-01-18 10:47:53 +00:00
|
|
|
|
]
|
|
|
|
|
|
)
|
2015-12-15 10:17:43 +00:00
|
|
|
|
|
2016-01-18 17:35:28 +00:00
|
|
|
|
def validate_name(self, a):
|
2016-03-31 15:17:05 +01:00
|
|
|
|
from app.utils import email_safe
|
|
|
|
|
|
# make sure the email_from will be unique to all services
|
|
|
|
|
|
if email_safe(a.data) in self._names_func():
|
2016-02-26 10:46:49 +00:00
|
|
|
|
raise ValidationError('This service name is already in use')
|
2016-01-04 14:00:39 +00:00
|
|
|
|
|
|
|
|
|
|
|
2016-01-11 13:15:10 +00:00
|
|
|
|
class ServiceNameForm(Form):
|
2016-03-10 14:29:31 +00:00
|
|
|
|
def __init__(self, names_func, *args, **kwargs):
|
|
|
|
|
|
"""
|
|
|
|
|
|
Keyword arguments:
|
|
|
|
|
|
names_func -- Returns a list of unique service_names already registered
|
|
|
|
|
|
on the system.
|
|
|
|
|
|
"""
|
|
|
|
|
|
self._names_func = names_func
|
|
|
|
|
|
super(ServiceNameForm, self).__init__(*args, **kwargs)
|
|
|
|
|
|
|
|
|
|
|
|
name = StringField(
|
2016-06-20 13:33:29 +01:00
|
|
|
|
u'Service name',
|
2016-03-10 14:29:31 +00:00
|
|
|
|
validators=[
|
2016-04-04 10:42:04 +01:00
|
|
|
|
DataRequired(message='Can’t be empty')
|
2016-03-10 14:29:31 +00:00
|
|
|
|
])
|
|
|
|
|
|
|
|
|
|
|
|
def validate_name(self, a):
|
2016-03-31 15:17:05 +01:00
|
|
|
|
from app.utils import email_safe
|
|
|
|
|
|
# make sure the email_from will be unique to all services
|
|
|
|
|
|
if email_safe(a.data) in self._names_func():
|
2016-03-10 14:29:31 +00:00
|
|
|
|
raise ValidationError('This service name is already in use')
|
2016-01-11 13:15:10 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class ConfirmPasswordForm(Form):
|
2016-01-22 16:34:36 +00:00
|
|
|
|
def __init__(self, validate_password_func, *args, **kwargs):
|
|
|
|
|
|
self.validate_password_func = validate_password_func
|
|
|
|
|
|
super(ConfirmPasswordForm, self).__init__(*args, **kwargs)
|
|
|
|
|
|
|
2016-01-11 13:15:10 +00:00
|
|
|
|
password = PasswordField(u'Enter password')
|
|
|
|
|
|
|
2016-01-22 16:34:36 +00:00
|
|
|
|
def validate_password(self, field):
|
|
|
|
|
|
if not self.validate_password_func(field.data):
|
|
|
|
|
|
raise ValidationError('Invalid password')
|
|
|
|
|
|
|
2016-01-11 13:15:10 +00:00
|
|
|
|
|
2017-02-15 15:06:47 +00:00
|
|
|
|
class BaseTemplateForm(Form):
|
2016-01-19 15:54:12 +00:00
|
|
|
|
name = StringField(
|
|
|
|
|
|
u'Template name',
|
2016-04-04 10:42:04 +01:00
|
|
|
|
validators=[DataRequired(message="Can’t be empty")])
|
2016-01-22 12:19:15 +00:00
|
|
|
|
|
2016-01-22 12:15:47 +00:00
|
|
|
|
template_content = TextAreaField(
|
2016-06-20 11:37:42 +01:00
|
|
|
|
u'Message',
|
2016-04-07 16:02:06 +01:00
|
|
|
|
validators=[
|
|
|
|
|
|
DataRequired(message="Can’t be empty"),
|
2017-02-15 15:06:47 +00:00
|
|
|
|
NoCommasInPlaceHolders()
|
2016-04-07 16:02:06 +01:00
|
|
|
|
]
|
|
|
|
|
|
)
|
2017-01-19 09:35:34 +00:00
|
|
|
|
process_type = RadioField(
|
|
|
|
|
|
'Use priority queue?',
|
|
|
|
|
|
choices=[
|
|
|
|
|
|
('priority', 'Yes'),
|
|
|
|
|
|
('normal', 'No'),
|
|
|
|
|
|
],
|
|
|
|
|
|
validators=[DataRequired()],
|
|
|
|
|
|
default='normal'
|
|
|
|
|
|
)
|
2016-01-11 13:15:10 +00:00
|
|
|
|
|
|
|
|
|
|
|
2017-02-15 15:06:47 +00:00
|
|
|
|
class SMSTemplateForm(BaseTemplateForm):
|
|
|
|
|
|
def validate_template_content(self, field):
|
|
|
|
|
|
OnlyGSMCharacters()(None, field)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class EmailTemplateForm(BaseTemplateForm):
|
2016-04-14 14:04:41 +01:00
|
|
|
|
subject = TextAreaField(
|
2016-02-22 14:45:13 +00:00
|
|
|
|
u'Subject',
|
2016-04-04 10:42:04 +01:00
|
|
|
|
validators=[DataRequired(message="Can’t be empty")])
|
2016-02-22 14:45:13 +00:00
|
|
|
|
|
|
|
|
|
|
|
2016-11-08 13:12:07 +00:00
|
|
|
|
class LetterTemplateForm(EmailTemplateForm):
|
2017-03-13 10:55:15 +00:00
|
|
|
|
|
|
|
|
|
|
subject = TextAreaField(
|
2017-05-10 11:27:45 +01:00
|
|
|
|
u'Main heading',
|
2017-03-13 10:55:15 +00:00
|
|
|
|
validators=[DataRequired(message="Can’t be empty")])
|
|
|
|
|
|
|
|
|
|
|
|
template_content = TextAreaField(
|
|
|
|
|
|
u'Body',
|
|
|
|
|
|
validators=[
|
|
|
|
|
|
DataRequired(message="Can’t be empty"),
|
|
|
|
|
|
NoCommasInPlaceHolders()
|
|
|
|
|
|
]
|
|
|
|
|
|
)
|
2016-11-08 13:12:07 +00:00
|
|
|
|
|
|
|
|
|
|
|
2016-01-05 17:52:09 +00:00
|
|
|
|
class ForgotPasswordForm(Form):
|
2016-10-26 14:01:01 +01:00
|
|
|
|
email_address = email_address(gov_user=False)
|
2016-01-04 14:00:39 +00:00
|
|
|
|
|
|
|
|
|
|
|
2016-01-06 17:37:07 +00:00
|
|
|
|
class NewPasswordForm(Form):
|
2016-01-08 12:00:52 +00:00
|
|
|
|
new_password = password()
|
2016-01-11 15:00:51 +00:00
|
|
|
|
|
|
|
|
|
|
|
2016-01-12 11:25:46 +00:00
|
|
|
|
class ChangePasswordForm(Form):
|
2016-01-27 12:22:32 +00:00
|
|
|
|
def __init__(self, validate_password_func, *args, **kwargs):
|
|
|
|
|
|
self.validate_password_func = validate_password_func
|
|
|
|
|
|
super(ChangePasswordForm, self).__init__(*args, **kwargs)
|
|
|
|
|
|
|
2016-01-12 11:25:46 +00:00
|
|
|
|
old_password = password('Current password')
|
|
|
|
|
|
new_password = password('New password')
|
|
|
|
|
|
|
2016-01-27 12:22:32 +00:00
|
|
|
|
def validate_old_password(self, field):
|
|
|
|
|
|
if not self.validate_password_func(field.data):
|
|
|
|
|
|
raise ValidationError('Invalid password')
|
|
|
|
|
|
|
2016-01-12 11:25:46 +00:00
|
|
|
|
|
2016-01-11 15:00:51 +00:00
|
|
|
|
class CsvUploadForm(Form):
|
2016-02-04 12:20:24 +00:00
|
|
|
|
file = FileField('Add recipients', validators=[DataRequired(
|
2016-05-11 09:43:55 +01:00
|
|
|
|
message='Please pick a file'), CsvFileValidator()])
|
2016-01-12 10:28:14 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class ChangeNameForm(Form):
|
|
|
|
|
|
new_name = StringField(u'Your name')
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class ChangeEmailForm(Form):
|
2016-01-27 12:22:32 +00:00
|
|
|
|
def __init__(self, validate_email_func, *args, **kwargs):
|
|
|
|
|
|
self.validate_email_func = validate_email_func
|
|
|
|
|
|
super(ChangeEmailForm, self).__init__(*args, **kwargs)
|
|
|
|
|
|
|
2016-01-12 10:28:14 +00:00
|
|
|
|
email_address = email_address()
|
|
|
|
|
|
|
2016-01-27 12:22:32 +00:00
|
|
|
|
def validate_email_address(self, field):
|
|
|
|
|
|
is_valid = self.validate_email_func(field.data)
|
|
|
|
|
|
if not is_valid:
|
|
|
|
|
|
raise ValidationError("The email address is already in use")
|
|
|
|
|
|
|
2016-01-12 10:28:14 +00:00
|
|
|
|
|
|
|
|
|
|
class ChangeMobileNumberForm(Form):
|
|
|
|
|
|
mobile_number = mobile_number()
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class ConfirmMobileNumberForm(Form):
|
2016-01-27 12:22:32 +00:00
|
|
|
|
def __init__(self, validate_code_func, *args, **kwargs):
|
|
|
|
|
|
self.validate_code_func = validate_code_func
|
|
|
|
|
|
super(ConfirmMobileNumberForm, self).__init__(*args, **kwargs)
|
|
|
|
|
|
|
2016-01-12 10:28:14 +00:00
|
|
|
|
sms_code = sms_code()
|
2016-01-19 09:55:13 +00:00
|
|
|
|
|
2016-01-27 12:22:32 +00:00
|
|
|
|
def validate_sms_code(self, field):
|
|
|
|
|
|
is_valid, msg = self.validate_code_func(field.data)
|
|
|
|
|
|
if not is_valid:
|
|
|
|
|
|
raise ValidationError(msg)
|
|
|
|
|
|
|
2016-01-19 09:55:13 +00:00
|
|
|
|
|
2016-08-07 09:17:49 +01:00
|
|
|
|
class ChooseTimeForm(Form):
|
|
|
|
|
|
|
|
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
|
|
super(ChooseTimeForm, self).__init__(*args, **kwargs)
|
|
|
|
|
|
self.scheduled_for.choices = [('', 'Now')] + [
|
2016-10-11 14:11:10 +01:00
|
|
|
|
get_time_value_and_label(hour) for hour in get_next_hours_until(
|
|
|
|
|
|
get_furthest_possible_scheduled_time()
|
|
|
|
|
|
)
|
2016-08-07 09:17:49 +01:00
|
|
|
|
]
|
2016-10-11 14:17:29 +01:00
|
|
|
|
self.scheduled_for.categories = get_next_days_until(get_furthest_possible_scheduled_time())
|
2016-08-07 09:17:49 +01:00
|
|
|
|
|
|
|
|
|
|
scheduled_for = RadioField(
|
|
|
|
|
|
'When should Notify send these messages?',
|
|
|
|
|
|
default='',
|
|
|
|
|
|
validators=[
|
|
|
|
|
|
DataRequired()
|
|
|
|
|
|
]
|
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
|
2016-01-19 09:55:13 +00:00
|
|
|
|
class CreateKeyForm(Form):
|
2016-01-21 14:15:36 +00:00
|
|
|
|
def __init__(self, existing_key_names=[], *args, **kwargs):
|
2016-01-21 16:52:01 +00:00
|
|
|
|
self.existing_key_names = [x.lower() for x in existing_key_names]
|
2016-01-21 14:15:36 +00:00
|
|
|
|
super(CreateKeyForm, self).__init__(*args, **kwargs)
|
|
|
|
|
|
|
2016-06-29 17:10:49 +01:00
|
|
|
|
key_type = RadioField(
|
2017-04-26 13:21:27 +01:00
|
|
|
|
'Type of key',
|
2016-06-29 17:10:49 +01:00
|
|
|
|
validators=[
|
|
|
|
|
|
DataRequired()
|
|
|
|
|
|
]
|
|
|
|
|
|
)
|
|
|
|
|
|
|
2016-01-21 14:15:36 +00:00
|
|
|
|
key_name = StringField(u'Description of key', validators=[
|
|
|
|
|
|
DataRequired(message='You need to give the key a name')
|
|
|
|
|
|
])
|
|
|
|
|
|
|
|
|
|
|
|
def validate_key_name(self, key_name):
|
2016-01-21 16:52:01 +00:00
|
|
|
|
if key_name.data.lower() in self.existing_key_names:
|
2016-01-21 14:15:36 +00:00
|
|
|
|
raise ValidationError('A key with this name already exists')
|
2016-04-19 13:51:16 +01:00
|
|
|
|
|
|
|
|
|
|
|
2016-12-12 11:25:43 +00:00
|
|
|
|
class SupportType(Form):
|
|
|
|
|
|
support_type = RadioField(
|
|
|
|
|
|
'How can we help you?',
|
|
|
|
|
|
choices=[
|
|
|
|
|
|
('problem', 'Report a problem'),
|
|
|
|
|
|
('question', 'Ask a question or give feedback'),
|
|
|
|
|
|
],
|
|
|
|
|
|
validators=[DataRequired()]
|
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
|
2016-12-21 14:55:49 +00:00
|
|
|
|
class Feedback(Form):
|
2016-04-19 13:51:16 +01:00
|
|
|
|
name = StringField('Name')
|
|
|
|
|
|
email_address = StringField('Email address')
|
2016-12-12 11:25:43 +00:00
|
|
|
|
feedback = TextAreaField('Your message', validators=[DataRequired(message="Can’t be empty")])
|
2016-04-26 13:31:57 +01:00
|
|
|
|
|
|
|
|
|
|
|
2016-12-21 14:55:49 +00:00
|
|
|
|
class Problem(Feedback):
|
|
|
|
|
|
email_address = email_address(label='Email address', gov_user=False)
|
|
|
|
|
|
|
|
|
|
|
|
|
2016-12-12 11:44:11 +00:00
|
|
|
|
class Triage(Form):
|
|
|
|
|
|
severe = RadioField(
|
|
|
|
|
|
'Is it an emergency?',
|
|
|
|
|
|
choices=[
|
|
|
|
|
|
('yes', 'Yes'),
|
|
|
|
|
|
('no', 'No'),
|
|
|
|
|
|
],
|
|
|
|
|
|
validators=[DataRequired()]
|
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
|
2016-04-26 13:31:57 +01:00
|
|
|
|
class RequestToGoLiveForm(Form):
|
2016-11-01 11:31:39 +00:00
|
|
|
|
mou = RadioField(
|
|
|
|
|
|
(
|
|
|
|
|
|
'Has your organisation accepted the GOV.UK Notify data sharing and financial '
|
|
|
|
|
|
'agreement (Memorandum of Understanding)?'
|
|
|
|
|
|
),
|
|
|
|
|
|
choices=[
|
|
|
|
|
|
('yes', 'Yes'),
|
2016-11-02 15:46:22 +00:00
|
|
|
|
('no', 'No'),
|
|
|
|
|
|
('don’t know', 'I don’t know')
|
2016-11-01 11:31:39 +00:00
|
|
|
|
],
|
|
|
|
|
|
validators=[DataRequired()]
|
|
|
|
|
|
)
|
2016-10-31 15:05:22 +00:00
|
|
|
|
channel = RadioField(
|
2016-10-31 15:21:29 +00:00
|
|
|
|
'What kind of messages will you be sending?',
|
2016-10-31 15:05:22 +00:00
|
|
|
|
choices=[
|
|
|
|
|
|
('emails', 'Emails'),
|
|
|
|
|
|
('text messages', 'Text messages'),
|
|
|
|
|
|
('emails and text messages', 'Both')
|
|
|
|
|
|
],
|
|
|
|
|
|
validators=[DataRequired()]
|
2016-10-24 15:46:22 +01:00
|
|
|
|
)
|
|
|
|
|
|
start_date = StringField(
|
2016-10-24 16:10:41 +01:00
|
|
|
|
'When will you be ready to start sending messages?',
|
|
|
|
|
|
validators=[DataRequired(message='Can’t be empty')]
|
2016-10-24 15:46:22 +01:00
|
|
|
|
)
|
2016-10-24 13:38:55 +01:00
|
|
|
|
start_volume = StringField(
|
2016-11-01 11:30:03 +00:00
|
|
|
|
'How many messages do you expect to send to start with?',
|
2016-10-24 16:10:41 +01:00
|
|
|
|
validators=[DataRequired(message='Can’t be empty')]
|
2016-10-24 13:38:55 +01:00
|
|
|
|
)
|
|
|
|
|
|
peak_volume = StringField(
|
2016-11-01 11:30:03 +00:00
|
|
|
|
'Will the number of messages increase and when will that start?',
|
2016-10-24 16:10:41 +01:00
|
|
|
|
validators=[DataRequired(message='Can’t be empty')]
|
2016-10-24 13:38:55 +01:00
|
|
|
|
)
|
2016-10-31 15:05:22 +00:00
|
|
|
|
upload_or_api = RadioField(
|
2016-10-31 15:21:29 +00:00
|
|
|
|
'How are you going to send messages?',
|
2016-10-31 15:05:22 +00:00
|
|
|
|
choices=[
|
|
|
|
|
|
('File upload', 'Upload a spreadsheet of recipients'),
|
|
|
|
|
|
('API', 'Integrate with the GOV.UK Notify API'),
|
|
|
|
|
|
('API and file upload', 'Both')
|
|
|
|
|
|
],
|
|
|
|
|
|
validators=[DataRequired()]
|
2016-10-24 13:38:55 +01:00
|
|
|
|
)
|
2016-05-11 09:43:55 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class ProviderForm(Form):
|
|
|
|
|
|
priority = IntegerField('Priority', [validators.NumberRange(min=1, max=100, message="Must be between 1 and 100")])
|
2016-05-16 13:09:58 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class ServiceReplyToEmailFrom(Form):
|
2016-08-23 14:16:07 +01:00
|
|
|
|
email_address = email_address(label='Email reply to address')
|
2016-07-01 13:47:22 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class ServiceSmsSender(Form):
|
2016-08-22 16:10:57 +01:00
|
|
|
|
sms_sender = StringField(
|
|
|
|
|
|
'Text message sender',
|
|
|
|
|
|
validators=[
|
2017-02-27 15:49:42 +00:00
|
|
|
|
Length(max=11, message="Enter 11 characters or fewer")
|
2016-08-22 16:10:57 +01:00
|
|
|
|
]
|
|
|
|
|
|
)
|
2016-07-01 13:47:22 +01:00
|
|
|
|
|
|
|
|
|
|
def validate_sms_sender(form, field):
|
2017-03-02 15:56:28 +00:00
|
|
|
|
if field.data and not re.match(r'^[a-zA-Z0-9\s]+$', field.data):
|
2016-08-22 16:10:57 +01:00
|
|
|
|
raise ValidationError('Use letters and numbers only')
|
2016-08-08 10:28:40 +01:00
|
|
|
|
|
|
|
|
|
|
|
2017-03-02 15:56:28 +00:00
|
|
|
|
class ServiceLetterContactBlock(Form):
|
2017-04-03 10:46:21 +01:00
|
|
|
|
letter_contact_block = TextAreaField()
|
2017-03-02 15:56:28 +00:00
|
|
|
|
|
|
|
|
|
|
def validate_letter_contact_block(form, field):
|
2017-03-03 16:15:15 +00:00
|
|
|
|
line_count = field.data.strip().count('\n')
|
|
|
|
|
|
if line_count >= 10:
|
|
|
|
|
|
raise ValidationError(
|
|
|
|
|
|
'Contains {} lines, maximum is 10'.format(line_count + 1)
|
|
|
|
|
|
)
|
2017-03-02 15:56:28 +00:00
|
|
|
|
|
|
|
|
|
|
|
2016-08-08 10:28:40 +01:00
|
|
|
|
class ServiceBrandingOrg(Form):
|
|
|
|
|
|
|
|
|
|
|
|
def __init__(self, organisations=[], *args, **kwargs):
|
|
|
|
|
|
self.organisation.choices = organisations
|
|
|
|
|
|
super(ServiceBrandingOrg, self).__init__(*args, **kwargs)
|
|
|
|
|
|
|
|
|
|
|
|
branding_type = RadioField(
|
|
|
|
|
|
'Branding',
|
|
|
|
|
|
choices=[
|
|
|
|
|
|
('govuk', 'GOV.UK only'),
|
|
|
|
|
|
('both', 'GOV.UK and organisation'),
|
|
|
|
|
|
('org', 'Organisation only')
|
|
|
|
|
|
],
|
|
|
|
|
|
validators=[
|
|
|
|
|
|
DataRequired()
|
|
|
|
|
|
]
|
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
organisation = RadioField(
|
|
|
|
|
|
'Organisation',
|
|
|
|
|
|
validators=[
|
|
|
|
|
|
DataRequired()
|
|
|
|
|
|
]
|
|
|
|
|
|
)
|
2016-09-20 12:30:00 +01:00
|
|
|
|
|
|
|
|
|
|
|
2017-04-19 16:11:28 +01:00
|
|
|
|
class LetterBranding(Form):
|
|
|
|
|
|
|
|
|
|
|
|
def __init__(self, choices=[], *args, **kwargs):
|
|
|
|
|
|
super().__init__(*args, **kwargs)
|
|
|
|
|
|
self.dvla_org_id.choices = choices
|
|
|
|
|
|
|
|
|
|
|
|
dvla_org_id = RadioField(
|
|
|
|
|
|
'Which logo should this service’s letter have?',
|
|
|
|
|
|
validators=[
|
|
|
|
|
|
DataRequired()
|
|
|
|
|
|
]
|
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
|
2016-09-20 12:30:00 +01:00
|
|
|
|
class Whitelist(Form):
|
|
|
|
|
|
|
|
|
|
|
|
def populate(self, email_addresses, phone_numbers):
|
|
|
|
|
|
for form_field, existing_whitelist in (
|
|
|
|
|
|
(self.email_addresses, email_addresses),
|
|
|
|
|
|
(self.phone_numbers, phone_numbers)
|
|
|
|
|
|
):
|
|
|
|
|
|
for index, value in enumerate(existing_whitelist):
|
|
|
|
|
|
form_field[index].data = value
|
|
|
|
|
|
|
|
|
|
|
|
email_addresses = FieldList(
|
|
|
|
|
|
EmailField(
|
|
|
|
|
|
'',
|
|
|
|
|
|
validators=[
|
|
|
|
|
|
Optional(),
|
|
|
|
|
|
Email(message='Enter valid email addresses')
|
|
|
|
|
|
],
|
|
|
|
|
|
default=''
|
|
|
|
|
|
),
|
|
|
|
|
|
min_entries=5,
|
|
|
|
|
|
max_entries=5,
|
|
|
|
|
|
label="Email addresses"
|
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
phone_numbers = FieldList(
|
|
|
|
|
|
UKMobileNumber(
|
|
|
|
|
|
'',
|
|
|
|
|
|
validators=[
|
|
|
|
|
|
Optional()
|
|
|
|
|
|
],
|
|
|
|
|
|
default=''
|
|
|
|
|
|
),
|
|
|
|
|
|
min_entries=5,
|
|
|
|
|
|
max_entries=5,
|
|
|
|
|
|
label="Mobile numbers"
|
|
|
|
|
|
)
|
2017-01-03 10:45:06 +00:00
|
|
|
|
|
2017-01-03 16:14:25 +00:00
|
|
|
|
|
2017-01-03 10:45:06 +00:00
|
|
|
|
class DateFilterForm(Form):
|
|
|
|
|
|
start_date = DateField("Start Date", [validators.optional()])
|
2017-01-03 16:14:25 +00:00
|
|
|
|
end_date = DateField("End Date", [validators.optional()])
|
|
|
|
|
|
include_from_test_key = BooleanField("Include test keys", default="checked", false_values={"N"})
|
Merge email, text message + letter templates pages
Right now we have separate pages for email and text message templates.
In the future we will also have a separate page for letter templates.
This commit changes Notify to only have one page for all templates.
What is the problem?
---
The left-hand navigation is getting quite crowded, at 8 items for a
service that can send letters. Research suggests that the number of
objects an average human can hold in working memory is 7 ± 2 [1]. So
we’re at the limit of how many items the navigation should have.
In the future we will need to search/sort/filter templates by attributes
other than type, for example:
- show me the ‘confirmation’ templates
- show me the most recently used templates
- show me all templates containing the placeholder `((ref_no))`
These are hypothetical for now, but these needs (or others) may become
real in the future. At this point pre-filtering the list of templates
by type would restrict what searches a user could do. So by making this
change now we’re in a better position to iterate the design in the
future.
What’s the change?
---
This commit replaces the ‘Email templates’, ‘Text message templates’ and
‘Letter templates’ pages with one page called ‘Templates’.
This new templates page shows all the templates for the service, sorted
by most recently created first (as before).
To add a new template there is a new page with a form asking you what
kind of template you want to create. This is necessary because in the
past we knew what kind of template you wanted to create based on the
kind you were looking at.
What’s the impact of this change on new users?
---
This change alters the onboarding process slightly. We still want to
take people through the empty templates page from the call-to-action on
the dashboard because it helps them understand that to send a message
using Notify you need a template. But because we don’t have separate
pages for emails/text messages we will have to send users through the
extra step of choosing what kind of template to create. This is a bit
clunkier on first use but:
- it still gets the point across
- it takes them through the actual flow they will be using to create new
templates in the future (ie they’re learning how to use Notify, not
just being taken through a special onboarding route)
I’m not too worried about this change in terms of the experience for new
users. Furthermore, by making it now we get to validate whether it’s
causing any problems in the lab research booked for next week.
What’s the impact of this change on current services?
---
Looking at the top 15 services by number of templates[2], most are using
either text messages or emails. So this change would not have a
significant impact on these services because the page will not get any
longer. In other words we wouldn’t be making it worse for them.
Those services who do use both are not using as many templates. The
worst-case scenario is SSCS, who have 16 templates, evenly split between
email and text messages. So they would go from having 8 templates per
page to 16, which is still less than half the number that HMPO or
Digital Marketplace are managing.
References
---
1. https://en.wikipedia.org/wiki/The_Magical_Number_Seven,_Plus_or_Minus_Two
2. Template usage by service
Service name | Template count | Template types
---------------------------------------|----------------|---------------
Her Majesty's Passport Office | 40 | sms
Digital Marketplace | 40 | email
GovWifi-Staging | 19 | sms
GovWifi | 18 | sms
Digital Apprenticeship Service | 16 | email
SSCS | 16 | both
Crown Commercial Service MI Collection | 15 | email
Help with Prison Visits | 12 | both
Digital Future | 12 | email
Export Licensing Service | 11 | email
Civil Money Claims | 9 | both
DVLA Drivers Medical Service | 9 | sms
GOV.UK Notify | 8 | both
Manage your benefit overpayments | 8 | both
Tax Renewals | 8 | both
2017-02-28 12:16:35 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class ChooseTemplateType(Form):
|
|
|
|
|
|
|
|
|
|
|
|
template_type = RadioField(
|
|
|
|
|
|
'What kind of template do you want to add?',
|
|
|
|
|
|
validators=[
|
|
|
|
|
|
DataRequired()
|
|
|
|
|
|
]
|
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
def __init__(self, include_letters=False, *args, **kwargs):
|
|
|
|
|
|
|
|
|
|
|
|
super().__init__(*args, **kwargs)
|
|
|
|
|
|
|
|
|
|
|
|
self.template_type.choices = filter(None, [
|
|
|
|
|
|
('email', 'Email'),
|
|
|
|
|
|
('sms', 'Text message'),
|
|
|
|
|
|
('letter', 'Letter') if include_letters else None
|
|
|
|
|
|
])
|
2017-03-14 10:46:38 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class SearchTemplatesForm(Form):
|
|
|
|
|
|
|
|
|
|
|
|
search = SearchField('Search by name')
|