Encoding trace scenarios¶
As an Alloy specification increases in size and complexity, its validation through the exploration of arbitrary instances in the visualizer becomes more and more difficult. A technique that can be used is to combine this interactive exploration with the specification of run
commands that directly search for interesting scenarios. When models have mutable structures, scenarios now have a temporal facet that must also be restricted, which can get difficult to manage. Also, often we don’t want the command to completely fix the trace since we want to be able to explore the design. In those cases, the scenario may be under-specified and the Analyzer will complete it by solving the specified constraints. This topic presents an idiom to encode such trace scenarios.
State-oriented scenario encoding¶
A trace is comprised of a (immutable) configuration and a (mutable) state. The first step of encoding a scenario is to restrict the configuration, and this can be achieved with the same strategy used for encoding test instances, based on some
/disj
patterns. For example, let us try to encode the scenario that was being explored in chapter Behavioral modeling: upload a file, share it twice, download it once, delete it, and finally empty the trash. This trace requires exactly one file and two tokens in the configuration (there are no immutable fields in the file sharing example). Such configuration could be specified as follows.
run scenario_two_shared {
some f : File, disj t1, t2 : Token {
File = f
Token = t1 + t2
// ...
}
} for 1 File, 2 Token
Focusing now on the mutable part of the model, we must extend the above formulas with constraints that encode the different states of the desired execution trace. The main issue here is that with the standard temporal
operators we have seen so far, this would result in either nested
and
/after
formulas, or expressions with several appended
primes '
, neither being a scalable approach. For example, to restrict
the first 4 states of the desired scenario we could write the following.
no uploaded and after (uploaded = f and after (uploaded = f and after uploaded = f))
no shared and after (no shared and after (shared = f->t1 and after shared = f->t1 + f->t2))
no trashed and after (no trashed and after (no trashed and after no trashed))
Or, alternatively, the following.
no uploaded and uploaded' = f and uploaded'' = f and uploaded''' = f
no shared and no shared' and shared'' = f->t1 and shared''' = f->t1 + f->t2
no trashed and no trashed' and no trashed'' and no trashed'''
Some of these formulas could of course be simplified (for instance, we could
just state always no trash
), but they quickly become unmanageable.
The temporal sequence operator¶
To address this kind of rather common formulas, Alloy has a sequential temporal
operator ;
that eases their formulation: a formula p;q
literally means p and after q
. Moreover, this operator has the least
precedence of all logic and temporal operators, further simplifying the
combination of formulas restricting each individual state. Given this operator,
we can now encode our scenario with the following command.
run scenario_two_shared {
some f : File, disj t1, t2 : Token {
File = f
Token = t1 + t2
no uploaded; uploaded = f; uploaded = f; uploaded = f; uploaded = f; uploaded = f; no uploaded
no shared; no shared; shared = f->t1; shared = f->t1 + f->t2; shared = f->t1; no shared; no shared
no trashed; no trashed; no trashed; no trashed; no trashed; trashed = f; no trashed
}
} for 1 File, 2 Token
If we now run this scenario, we will get exactly the trace we were expecting. Below is the first transition (an upload action), and you can see in the trace overview that it has the expected 7 states.
data:image/s3,"s3://crabby-images/36ffb/36ffb8d823822d4c98dddc291efdea9b0dd201ea" alt="Get the code to generate this instance."
Recall that due to the mechanics already explained, the quantified variables are
Skolemized and appear in the visualizer identified by a $
prefix and
the name of the command, as shown above, but they can be omitted in the theme.
This may help identify which atoms in the scenario correspond to the variables
in the command, but they also clutter the visualization. Below is the second
transition of the same scenario (a sharing action) after disabling Show as labels in the
theme for those sets.
data:image/s3,"s3://crabby-images/c6897/c68974bc995a56799b08ee120f6dbe04416fa43e" alt="Get the code to generate this instance."
Since we’ve fully restricted the first 7 states of the trace, we might think that the scenario is fully specified and that any instance exploration operation fails to produce another trace. In fact, if you hit New Trace in the visualizer you’ll find that there are many possible traces that respect the specified scenario, namely any valid trace that extends its behavior beyond the first 7 fixed states. Below is one such alternative, which rather than stuttering after emptying the trash, uploads the deleted file again and deletes it again, going back to state 5.
data:image/s3,"s3://crabby-images/8cf0c/8cf0c3b13c6077ca761784b888b23253539de30b" alt="Get the code to generate this instance."
Exploring the possible continuations of a scenario may or not be something interesting to explore. For instance, you may wish to explore whether it is possible to share the file again (it isn’t because all tokens available in the scenario configuration have been used). In fact, as expected, these formulas can be as under-specified as we wish. For instance, we could leave certain relations unrestricted and just leave them to be solved by the Analyzer. Rather than encoding a single scenario, such commands would encode a “family” of related scenarios.
If we actually wanted the scenario to perform exactly the 6 actions and then
stutter forever, we have to enforce what happens after the last state in the
command’s constraint. The ;
operator can combine arbitrary Alloy
temporal formulas, although nesting temporal formulas quickly become difficult
to manage. In this case we just want to restrict the behavior after the last
step, so we can just specify that the desired last state always
happens from that
point onwards.
run scenario_two_shared_stutter {
some f : File, disj t1, t2 : Token {
File = f
Token = t1 + t2
no uploaded; uploaded = f; uploaded = f; uploaded = f; uploaded = f; uploaded = f; always no uploaded
no shared; no shared; shared = f->t1; shared = f->t1 + f->t2; shared = f->t1; no shared; always no shared
no trashed; no trashed; no trashed; no trashed; no trashed; trashed = f; always no trashed
}
} for 1 File, 2 Token
If we run this command and then ask for a New Trace, the Analyzer will report that there is no such instance.
Notice that, since these scenarios are encoded as plain formulas, we can refactor out common formulas into auxiliary predicates. For instance, although configurations are extremely simple in the file sharing example, the two scenarios above share the same configuration, so we can reuse that portion of the command.
pred two_tokens [f : File, t1, t2 : Token] {
File = f
Token = t1 + t2
}
run scenario_two_shared {
some f : File, disj t1, t2 : Token {
two_tokens[f, t1, t2]
no uploaded; uploaded = f; uploaded = f; uploaded = f; uploaded = f; uploaded = f; no uploaded
no shared; no shared; shared = f->t1; shared = f->t1 + f->t2; shared = f->t1; no shared; no shared
no trashed; no trashed; no trashed; no trashed; no trashed; trashed = f; no trashed
}
} for 1 File, 2 Token
run scenario_two_shared_stutter {
some f : File, disj t1, t2 : Token {
two_tokens[f, t1, t2]
no uploaded; uploaded = f; uploaded = f; uploaded = f; uploaded = f; uploaded = f; always no uploaded
no shared; no shared; shared = f->t1; shared = f->t1 + f->t2; shared = f->t1; no shared; always no shared
no trashed; no trashed; no trashed; no trashed; no trashed; trashed = f; always no trashed
}
} for 1 File, 2 Token
Event-oriented scenario encoding¶
The encodings above followed a “relation-wise” definition of the scenario, in
the sense that we restricted the evolution of each mutable relation
independently. Sometimes it’s easier to reason about scenarios in a “state-wise”
manner, particularly for relations closely related. We could easily adapt this
idiom for that purpose as follows (again, the ;
operator has the lowest
precedence).
run scenario_two_shared_stutter {
some f : File, disj t1, t2 : Token {
two_tokens[f, t1, t2]
no uploaded + trashed and no shared;
uploaded = f and no shared and no trashed;
uploaded = f and shared = f->t1 and no trashed;
uploaded = f and shared = f->t1 + f->t2 and no trashed;
uploaded = f and shared = f->t1 and no trashed;
uploaded = f and no shared and trashed = f;
always (no uploaded + trashed and no shared)
}
} for 1 File, 2 Token
This command represents exactly the same scenario as the previous
“relation-wise” version. Notice that we can’t just say no uploaded +
trashed + shared
because the +
operator must combine relations with
the same arity.
We can take this approach even further. So far we’ve encoded the expected scenario by directly restricting the value of the states. This allows us to exactly restrict the value of the relations in each state, but the result is verbose. An alternative is to instead just use the action predicates and force their occurrence in each desirable state. This results in simpler encodings, but at the cost of disregarding the concrete value of the relations. In particular, if an event is non-deterministic, the same action event may allow different state transitions to occur.
Let us get back to the scenario above. The same command could be written instead as follows.
run scenario_two_shared_event {
some f : File, disj t1, t2 : Token {
two_tokens[f, t1, t2]
upload[f]; share[f, t1]; share[f, t2]; download[t1]; delete[f]; empty; always stutter
}
} for 1 File, 2 Token
For the file sharing application, where all events are deterministic, this command represents exactly the same scenario as the previous versions.
This idiom is the one used throughout the book to encode the presented example traces, which you can access by clicking the figures.