695 lines
24 KiB
Python
695 lines
24 KiB
Python
"""Commands are the building blocks of the filling out of the form.
|
|
|
|
When filling out the form the user can get things asked and some data
|
|
is automagically created. Both can be modelled by commands.
|
|
|
|
"""
|
|
|
|
import datetime
|
|
import formfield as ff
|
|
from constants import (SUBINFO_SEP, VARIABLE_BEGIN_SEP, VARIABLE_END_SEP,
|
|
VARIABLE_PARTS_SEP, VARIABLE_DEFAULT_SEP,
|
|
DATE_END_SEP, DATE_BEGIN_SEP,
|
|
ConfigError, ON, OFF)
|
|
|
|
|
|
class Command():
|
|
"""A command is one action used to create some data for a form.
|
|
|
|
Information about this command can be accessed via [..] notation.
|
|
This access the reason for the necessary information.
|
|
|
|
Attributes:
|
|
name (str):
|
|
the name of the command as used in the
|
|
info of reason (reason must have an info "<command>-name"
|
|
where <command> is the specific type of command used).
|
|
reason (FormField):
|
|
all information for a command is saved in a FormField.
|
|
This FormField is the reason for this Command.
|
|
Usually this command populates the reason with information.
|
|
fieldList ([FormField]):
|
|
list with all fields of the current form
|
|
data ({str: [FormField]}:
|
|
dictionary with lists of FormFields from
|
|
other forms that contain helpful data
|
|
|
|
"""
|
|
|
|
def __init__(self, name, reason, fieldList, data):
|
|
"""Create a command.
|
|
|
|
Attributes:
|
|
name (str): the name of the command as used in the
|
|
info of reason (reason must have an info "<command>: name"
|
|
where <command> is the specific type of command used).
|
|
reason, fieldList, data: see class-Docstring
|
|
|
|
# Raises:
|
|
# NotImplementedError: Command cannot be instantiated. Use a
|
|
# subclass.
|
|
"""
|
|
self.name = name
|
|
self.reason = reason
|
|
self.fieldList = fieldList
|
|
self.data = data
|
|
# raise NotImplementedError("Command cannot be instantiated.")
|
|
|
|
@classmethod
|
|
def extractCommands(cls, reason, fieldList, data):
|
|
"""Find all commands saved in reason.
|
|
|
|
The information used must have the key equal to the subclass.
|
|
|
|
Attributes:
|
|
reason: the field where the commands get extracted.
|
|
fieldList, data: see class-docstring
|
|
Returns:
|
|
a list of commands.
|
|
|
|
"""
|
|
commands = []
|
|
for key in reason.iterkeys():
|
|
for clss in cls.__subclasses__():
|
|
if key == clss.__name__:
|
|
commands.extend([clss(commandname, reason,
|
|
fieldList, data)
|
|
for commandname in reason[key]])
|
|
# reason[key] should be a list
|
|
# see MobileRegex in readformdata
|
|
return commands
|
|
|
|
def __getitem__(self, key):
|
|
"""Get some information.
|
|
|
|
self.reason must have this information in the form
|
|
self["name" + "-" + key].
|
|
|
|
"""
|
|
return self.reason[self.name + SUBINFO_SEP + key]
|
|
|
|
def __setitem__(self, key, value):
|
|
"""Set some information.
|
|
|
|
Similar as __getitem__.
|
|
|
|
"""
|
|
self.reason[self.name + SUBINFO_SEP + key] = value
|
|
|
|
def __contains__(self, key):
|
|
"""Give it makes sense to ask for this key.
|
|
|
|
Returns:
|
|
self.name + SUBINFO_SEP + key in self.reason
|
|
|
|
"""
|
|
return self.name + SUBINFO_SEP + key in self.reason
|
|
|
|
def __lt__(self, other):
|
|
"""Has self a lower priority than other.
|
|
|
|
Returns:
|
|
If only one has a priority, it is "less".
|
|
If both have no priority, True is returned.
|
|
|
|
"""
|
|
try:
|
|
myprio = float(self["Priority"])
|
|
except KeyError:
|
|
return False
|
|
except ValueError:
|
|
raise ConfigError("Priority in Command " + self.name +
|
|
" of field " + self.reason.name +
|
|
" is not a float.")
|
|
try:
|
|
otherprio = float(other["Priority"])
|
|
except KeyError:
|
|
return True
|
|
except ValueError:
|
|
raise ConfigError("Priority in Command " + other.name +
|
|
" of field " + other.reason.name +
|
|
" is not a float.")
|
|
return myprio < otherprio
|
|
|
|
def interpretVariables(self, value):
|
|
"""Interpret variables in values.
|
|
|
|
Replace occurences of {FieldName[#Default]} by the value saved in
|
|
field FieldName.
|
|
Replace occurences of {REF|FieldName[#Default]} by the value saved
|
|
in field FieldName in the data self.data[REF].
|
|
[] means that this part is optional.
|
|
|
|
If nothing is saved raise ConfigError except if a #Default is
|
|
given. Then use the default
|
|
|
|
Variables are replaced as long as there are any.
|
|
Hence if there are variables in the new strings they get replaced
|
|
as well. Hope no one uses this for recursion. ^^^
|
|
|
|
Attributes:
|
|
value (str): the string in which variables are replaced.
|
|
default (str): if some Field has no value saved, use default.
|
|
If default is None (default for this attribute),
|
|
raise a ConfigError.
|
|
|
|
Raises:
|
|
ConfigError:
|
|
if some FieldName does not exist and has no default
|
|
|
|
"""
|
|
while True:
|
|
start = value.find(VARIABLE_BEGIN_SEP)
|
|
end = value.find(VARIABLE_END_SEP, start)
|
|
if start == -1 or end == -1: # not found
|
|
return value
|
|
sep = value.find(VARIABLE_PARTS_SEP, start)
|
|
replaced = value[start:end + len(VARIABLE_END_SEP)]
|
|
if sep == -1: # {FieldName}
|
|
fieldList = self.fieldList
|
|
fieldname = value[start + len(VARIABLE_BEGIN_SEP):end]
|
|
else: # {spec|FieldName}
|
|
fieldList = self.data[value[
|
|
start + len(VARIABLE_BEGIN_SEP):sep]]
|
|
fieldname = value[sep + len(VARIABLE_PARTS_SEP):end]
|
|
fieldnameparts = fieldname.split(VARIABLE_DEFAULT_SEP, maxsplit=1)
|
|
if len(fieldnameparts) == 2:
|
|
# fieldname#default
|
|
fieldname = fieldnameparts[0]
|
|
default = fieldnameparts[1]
|
|
else: # len(fieldnameparts) == 1:
|
|
fieldname = fieldnameparts[0]
|
|
default = None
|
|
try:
|
|
field = ff.FormField.findByFieldName(fieldList, fieldname,
|
|
include=lambda x: True)
|
|
except KeyError as e:
|
|
raise ConfigError("Variable cannot be replaced due to " +
|
|
"missing FormField: " + fieldname
|
|
+ " (" + str(e) + ")")
|
|
else: # normal case: Field found
|
|
try:
|
|
value = value.replace(replaced, field["Value"])
|
|
except KeyError as e:
|
|
if default is None:
|
|
raise ConfigError("Variable cannot be replaced" +
|
|
" due to missing value: "
|
|
+ fieldname + " (" + str(e) + ")")
|
|
else:
|
|
value = value.replace(replaced, default)
|
|
|
|
def interpretDates(self, value):
|
|
"""Interpret dates in value.
|
|
|
|
Replace occurences of {TODAY|format} with the current day
|
|
in the format format.
|
|
format is described in
|
|
https://docs.python.org/3.5/library/datetime.html#strftime-strptime-behavior
|
|
|
|
"""
|
|
while True:
|
|
start = value.find(DATE_BEGIN_SEP)
|
|
end = value.find(DATE_END_SEP, start)
|
|
if start == -1 or end == -1:
|
|
return value
|
|
replaced = value[start: end + len(DATE_END_SEP)]
|
|
formatstr = value[start + len(DATE_BEGIN_SEP): end]
|
|
formatted = datetime.datetime.now().strftime(formatstr)
|
|
value = value.replace(replaced, formatted)
|
|
|
|
def interpret(self, value):
|
|
"""Replace variables and dates with their correct values.
|
|
|
|
See description of interpretDates and interpretVariables.
|
|
|
|
"""
|
|
value = self.interpretVariables(value)
|
|
value = self.interpretDates(value)
|
|
return value
|
|
|
|
def __call__(self): # todo: refactor, make smaller, externalise conditions
|
|
"""Run the command.
|
|
|
|
The subclasses must implement do() because __call__ calls do.
|
|
|
|
There can be conditions on doing actions. Conditions are and-connected.
|
|
That means that all must say yes.
|
|
1) even if exist:
|
|
Usually commands are not run if there is already a value.
|
|
If Ifexist exists as an info, the command is also run if there is
|
|
already a value.
|
|
2) if unequal:
|
|
Do not execute if two values are equal. Syntax:
|
|
name-Ifunequal: x
|
|
name-x-A: Blub
|
|
name-x-B: Bla
|
|
Usually Blub and/or Bla contain variables.
|
|
3) if equal:
|
|
Similarly to Ifunequal
|
|
|
|
"""
|
|
if "Value" in self.reason and "Ifexist" not in self:
|
|
# Do not do anything
|
|
return
|
|
for key, test in [("Ifunequal", lambda a, b: a == b),
|
|
("Ifequal", lambda a, b: a != b)]:
|
|
try:
|
|
for comp in self[key]:
|
|
# there can be sevaral tests of the same kind
|
|
# hence stored in a list self[key]
|
|
comps = {}
|
|
# the values to be compared are to be stored in a dict
|
|
# with the keys "A" and "B":
|
|
for comparevalue in ("A", "B"):
|
|
try:
|
|
# saved in infos of the kind y-a-A, y-a-B
|
|
comps[comparevalue] = self[comp + SUBINFO_SEP +
|
|
comparevalue]
|
|
except KeyError as e:
|
|
# values y-a-A are missing
|
|
raise ConfigError("A " + key +
|
|
" is missing a value " +
|
|
comparevalue + ".(" + str(e) +
|
|
")")
|
|
if test(self.interpret(comps["A"]),
|
|
self.interpret(comps["B"])):
|
|
# do not do anything
|
|
return
|
|
except KeyError:
|
|
# do nothing, apparently key does not exist
|
|
assert key not in self # there should be no other KeyError
|
|
# pass # necessary without assert statement
|
|
|
|
self.do()
|
|
|
|
def userinput(self, prompt=None):
|
|
"""Ask the user a question.
|
|
|
|
Use Prompt if existent or otherwise reason.name for informing the user.
|
|
|
|
Attributes:
|
|
prompt: Text shown to the user to ask him to type something.
|
|
Default (None)= Take "Prompt" info or name.
|
|
": " is added.
|
|
|
|
Returns:
|
|
If user presses Ctrl+D (EOFError), return None.
|
|
Otherwise return what user typed. (Can be "").
|
|
|
|
Raises:
|
|
KeyboardInterrupt:
|
|
if user presses Ctrl+C, wanting to cancel the program.
|
|
"""
|
|
if prompt is None:
|
|
try:
|
|
prompt = self["Prompt"]
|
|
except KeyError:
|
|
prompt = self.reason.name
|
|
else:
|
|
# take the argument
|
|
pass
|
|
prompt = self.interpret(prompt)
|
|
prompt += ": "
|
|
try:
|
|
return input(prompt)
|
|
except EOFError:
|
|
return None
|
|
|
|
def showHelp(self, extrainfo='(Ctrl+D = save nothing)'):
|
|
"""Show the help for this question.
|
|
|
|
Show "Help" or if not existing, the Description or if not existing
|
|
telling the user that.
|
|
|
|
Attributes:
|
|
extrainfo: Text that is shown additionally to the help.
|
|
|
|
"""
|
|
print("")
|
|
try:
|
|
message = self["Help"]
|
|
except KeyError:
|
|
# no help supplied, use field description
|
|
try:
|
|
message = self.reason["Description"]
|
|
except KeyError:
|
|
message = "No help available."
|
|
message = self.interpret(message)
|
|
print(message)
|
|
print(extrainfo)
|
|
|
|
|
|
class Question(Command):
|
|
"""The user gets asked a question.
|
|
|
|
No __init__ method since the Command.__init__ is sufficiant.
|
|
|
|
Uses infos:
|
|
Prompt: what the user sees while typing.
|
|
Help: what the user sees if asking for further information
|
|
|
|
"""
|
|
|
|
def do(self):
|
|
"""Ask the user a question and save the result in the field.
|
|
|
|
If the user writes '?', a help is shown and the user is asked again.
|
|
|
|
If the user only writes nothing (aka an empty string) nothing (not even
|
|
an empty string) is saved. This sounds like a plausible default.
|
|
|
|
If the user writes self.EMPTY, an empty string is saved.
|
|
|
|
Uses the name of self.reason if no other question was supplied.
|
|
|
|
"""
|
|
while True: # while no usable input
|
|
print("")
|
|
text = self.userinput()
|
|
if text == "?":
|
|
self.showHelp()
|
|
# ask again
|
|
else:
|
|
# None deletes what is there
|
|
self.reason["Value"] = text
|
|
break
|
|
|
|
|
|
class Choice(Command):
|
|
"""The user gets asked a question with a limited number of answer options.
|
|
|
|
One possibility is a on-off Button.
|
|
Another is a choice of several buttons from which one can be chosen.
|
|
|
|
|
|
Uses infos:
|
|
Prompt: what the user sees before typing.
|
|
Help: what the user sees if asking for further information
|
|
reason[FieldStateOption]: For the options that can be used.
|
|
Option-1/2/3/On/Off/...: for each option a description what it means.
|
|
Enter: if present tells what should be used if the user just presses
|
|
Enter.
|
|
|
|
"""
|
|
|
|
@property
|
|
def options(self):
|
|
"""Return the list of options of the field of this choice.
|
|
|
|
Raises:
|
|
ConfigError: if this info does not exist.
|
|
|
|
"""
|
|
try:
|
|
return self.reason["FieldStateOption"]
|
|
except KeyError:
|
|
raise ConfigError(
|
|
"A Choice question was asked but there are no options" +
|
|
" specified. Field: " + self.reason.name +
|
|
" Choice: " + self.name)
|
|
|
|
def getEnterDefault(self):
|
|
"""Get the default value.
|
|
|
|
The one that should be saved on pressing Enter without input.
|
|
|
|
Interprets variables.
|
|
|
|
Raises:
|
|
ConfigError: If it is not a valid option.
|
|
|
|
Returns:
|
|
None if no default was specified.
|
|
The default if it was specified.
|
|
|
|
"""
|
|
try:
|
|
default = self["Enter"]
|
|
except KeyError:
|
|
return None
|
|
default = self.interpret(default)
|
|
if default in self.options:
|
|
return default
|
|
else:
|
|
raise ConfigError(
|
|
"Default on Enter - value not a possible option " +
|
|
"(" + self.name + " in " + self.reason.name + ")")
|
|
|
|
def sortOptions(self):
|
|
"""Sort the options in a reasonable way.
|
|
|
|
Change the order of the elements of self.reason["FieldStateOption"].
|
|
|
|
If default (self["Enter"]) exists, use this as first.
|
|
If Yes/On/Ja/ja exists, take this.
|
|
If natural numbers exist, take them in the natural order.
|
|
If No/Off/Nein/nein exists, take this.
|
|
Sort rest alphabetically.
|
|
|
|
"""
|
|
oldorder = self.options[:]
|
|
neworder = []
|
|
enter = self.getEnterDefault()
|
|
if enter is not None:
|
|
print("debug: enter found:", enter)
|
|
# check that it is indeed an option already included
|
|
neworder.append(enter)
|
|
oldorder.remove(enter)
|
|
for yes in ON:
|
|
if yes in oldorder:
|
|
print("debug: yes found:", yes)
|
|
neworder.append(yes)
|
|
oldorder.remove(yes)
|
|
numbers = sorted([number for number in oldorder if number.isdigit()])
|
|
neworder.extend(numbers)
|
|
for number in numbers:
|
|
print("debug: number found:", number)
|
|
oldorder.remove(number)
|
|
for no in OFF:
|
|
if no in oldorder:
|
|
print("debug: no found:", no)
|
|
neworder.append(no)
|
|
oldorder.remove(no)
|
|
print("debug: remaining options:", oldorder)
|
|
print("debug: new order:", neworder)
|
|
neworder.extend(sorted(oldorder)) # add what remained
|
|
assert set(neworder) == set(self.options) # check that we have not
|
|
# lost anything
|
|
self.reason["FieldStateOption"] = neworder
|
|
|
|
def optionDescription(self, option):
|
|
"""Build descriptive option.
|
|
|
|
Return:
|
|
if option is default
|
|
option = description (Default on Enter)
|
|
or if not default:
|
|
option = description
|
|
or if no description is existing:
|
|
option (No help)
|
|
|
|
"""
|
|
default = ("; " if self.getEnterDefault() != option
|
|
else " (Default on Enter); ")
|
|
try:
|
|
return (option + " = " +
|
|
self["Option" + SUBINFO_SEP + option] +
|
|
default)
|
|
except KeyError:
|
|
return option + " (No help) " + default
|
|
|
|
def optionDescriptionInterpreted(self, option):
|
|
"""Build descriptive option.
|
|
|
|
Interpret variables.
|
|
|
|
Return:
|
|
if option is default
|
|
option = description (Default on Enter)
|
|
or if not default:
|
|
option = description
|
|
or if no description is existing:
|
|
option (No help)
|
|
|
|
"""
|
|
return self.interpret(self.optionDescription(option))
|
|
|
|
def showPrompt(self):
|
|
"""Show the prompt/ name specified and the options."""
|
|
print("")
|
|
try:
|
|
print(self["Prompt"], "Options: ", sep="\n", end="")
|
|
except KeyError:
|
|
# no question supplied, hence use field name
|
|
print(self.reason.name, "Options:", sep="\n")
|
|
# show Default as first option -> included in sorting
|
|
self.sortOptions()
|
|
for option in self.options:
|
|
print(self.optionDescriptionInterpreted(option))
|
|
|
|
def usedefault(self):
|
|
"""Save the default value.
|
|
|
|
The default value is saved in "Enter".
|
|
|
|
Show help if no default is given.
|
|
|
|
Raises:
|
|
ConfigError: if the default is not a valid option.
|
|
|
|
Returns:
|
|
whether a value is saved aka if the user is not to be asked again
|
|
|
|
"""
|
|
default = self.getEnterDefault()
|
|
if default is None:
|
|
self.showHelp()
|
|
return False
|
|
else:
|
|
self.reason["Value"] = default
|
|
print(self.optionDescription(default))
|
|
return True
|
|
|
|
def do(self):
|
|
"""Ask the user a question and save the result in the field.
|
|
|
|
Tell the user of the different options.
|
|
|
|
If the user writes '?', a help is shown and the user is asked again.
|
|
|
|
The answer is interpreted in the following manner:
|
|
if the answer is empty:
|
|
if "Enter" is present, take this value
|
|
otherwise show help and reask
|
|
if the answer is ?, show help and reask
|
|
if it is equal to one of the options, take this. Otherwise
|
|
if one of the descriptions start with the answer, (ignore case):
|
|
take this one.
|
|
if none of the above, reask
|
|
if the user aborts with Ctrl+D (EOFError), do not save anything
|
|
|
|
Uses the name of self.reason if no other question was supplied.
|
|
Uses the description of self.reason if no other help was supplied.
|
|
|
|
"""
|
|
while True: # while no usable input
|
|
self.showPrompt()
|
|
text = self.userinput(prompt="Choice")
|
|
if text is None:
|
|
# user aborts, nothing to be saved
|
|
self.reason["Value"] = text # = None = delete if existent
|
|
print("Chose nothing.")
|
|
break
|
|
text = text.strip()
|
|
if text == "":
|
|
if self.usedefault():
|
|
# value is saved
|
|
break
|
|
else:
|
|
# ask again aka continue
|
|
pass
|
|
elif text == "?":
|
|
self.showHelp()
|
|
elif text in self.options:
|
|
self.reason["Value"] = text
|
|
print("Chose", self.optionDescription(text))
|
|
break
|
|
else:
|
|
for option in self.options:
|
|
if ("Option" + SUBINFO_SEP + option in self
|
|
# option help exists
|
|
and self["Option" + SUBINFO_SEP + option].lower(
|
|
).startswith(text.lower())):
|
|
self.reason["Value"] = option
|
|
print("Chose", self.optionDescription(option))
|
|
break
|
|
else:
|
|
# if the loop went through, hence nothing found
|
|
# ask again
|
|
print("Answer not understood. Type '?' for help.")
|
|
continue
|
|
# otherwise end outer loop
|
|
break
|
|
|
|
|
|
class Default(Command):
|
|
"""A default value is saved.
|
|
|
|
No __init__ method since the Command.__init__ is sufficiant.
|
|
|
|
Uses infos:
|
|
Value: what should be saved. Can include variables.
|
|
Addhours, Addmin: see addTime(time)
|
|
|
|
"""
|
|
|
|
def do(self):
|
|
"""Save the default value.
|
|
|
|
Understand variables.
|
|
|
|
If it boils down to an empty string, save nothing (and do not
|
|
overwrite any existing value).
|
|
TODO: Check if this behavior is useful.
|
|
|
|
Raises:
|
|
ConfigError: if understanding variables failed.
|
|
|
|
"""
|
|
if "Value" not in self:
|
|
raise ConfigError("A Default needs a default Value. " +
|
|
"Add a line " + self.name + "-Value: " +
|
|
"<your Default value> to the field " +
|
|
self.reason.name +
|
|
" (" + self.reason["FieldName"] + ")")
|
|
value = self.interpret(self["Value"])
|
|
if value != "" and value is not None:
|
|
# is not None should not happen but who knows
|
|
self.reason["Value"] = self.addTime(value)
|
|
|
|
def addTime(self, time):
|
|
"""Add a specified timeshift to time.
|
|
|
|
If there is a timeshift specified, add this to time.
|
|
|
|
Attributes:
|
|
time (str):
|
|
If time is of the form hh:mm, all right.
|
|
Otherwise print error message if a time shift is specified.
|
|
|
|
Uses infos:
|
|
Addhours:
|
|
must be integer, can be negative. Add this number of hours.
|
|
Addmin:
|
|
must be integer, can be negative. Add this number of minutes.
|
|
|
|
Returns:
|
|
if time is of the form hh:mm, return (hh+Addhours):(mm+Addmin),
|
|
otherwise return time.
|
|
"""
|
|
try:
|
|
split = time.split(":", maxsplit=1)
|
|
hours = int(split[0])
|
|
minutes = int(split[1])
|
|
except (IndexError, ValueError):
|
|
if "Addhours" in self or "Addmin" in self:
|
|
print("Adding a timeshift for field " +
|
|
self.reason.name + " to time '" + time + "' failed" +
|
|
" because the original time is not of the form hh:mm.")
|
|
return time
|
|
else:
|
|
try:
|
|
hours += int(self["Addhours"])
|
|
except ValueError:
|
|
raise ConfigError("Option Addhours of field " +
|
|
self.reason.name + " is not an integer.")
|
|
except KeyError:
|
|
pass # do not change hours apparently
|
|
try:
|
|
minutes += int(self["Addmin"])
|
|
except ValueError:
|
|
raise ConfigError("Option Addmin of field " +
|
|
self.reason.name + " is not an integer.")
|
|
except KeyError:
|
|
pass # do not change minutes apparently
|
|
return str(hours).zfill(2) + ":" + str(minutes).zfill(2)
|