Description
I was looking at something with @balacij and @NoahCardoso, and we noticed something a bit off. In all of our current input patterns, we follow this pattern:
- Parse all inputs
- Calculate derived values (where derived values are intermediate values with constraints)
- Validate inputs and derived values
- Intermediate calculations
- Output important values
As @balacij pointed out, we should not be calculating derived values before validating inputs, for both security and performance reasons. This is actually also a problem with logging: we shouldn't print a value to the console or a file before we know it is a valid value.
There are two solutions to this problem:
- Once we have
Interleaved
implemented, this will solve that issue. We can sort statements to do all constraint checks that can be done before calculating any derived values that can be calculated. - If we still want options that aren't fully interleaved, we need to split up our constraint-checks to checks on parsed inputs and checks on derived values. This would lead to the following order:
- Parse all inputs
- Validate inputs
- Calculate derived values
- Validate derived values
- Intermediate calculations
- Output important values
The other change we'll need to make is to tie logging to constraint-checks rather than to parsing.
I'll try to make a follow-up with more technical details tomorrow, but I believe that most of the changes we need to make will be in Modules.hs, as well as the readData function in Import.hs.
As a side note, I wonder if we should separate 'safety' constraints from 'modelling' constraints - we need to be much more careful about when and where we check for malicious inputs than inputs that are harmless but incorrect.