Вот что я в итоге использовал.Ключом было не использовать переменную по умолчанию ${val}
и использовать простые старые f-строки Python.
tz = self.tz
days_before = 28
dtmin = local_days_before(tz, days_before) # localized min date
dtmax = datetime.now(utc).astimezone(tz)
datetime_event = colander.SchemaNode(
colander.DateTime(default_tzinfo=dtmax.tzinfo),
widget=deform.widget.DateTimeInputWidget(
date_options={'min': -days_before,
'max': True,
'format': 'yyyy-mm-dd'},
time_options={'format': 'HH:i',
'formatLabel': 'HH:i'},
),
validator=colander.Range(
min=dtmin,
min_err=(f"Datetime must be after "
f"{dtmin:%B %d, %Y, %-I:%M %p} "),
max=dtmax,
max_err=(f"Datetime must be before "
f"{dtmax: %B %d, %Y, %-I:%M %p}")
),
title='Date and Time',
description='Date and time when the event occurred'
)
В этом решении также реализовано форматирование даты и времени, а также минимальных и максимальных дат в пользовательском интерфейсе pickadate..