Tag(sp22)
Today's notes involve two different topics that unrelated. We'll talk about something in base Forge that will help with your current homework. Then we'll transition into more about temporal mode in Forge.
Before we go back to temporal mode, I wanted to talk about field access in Forge. Let's go back to the directed-graph model we used before:
#lang forge
sig Person {
friends: set Person,
followers: set Person
}
one sig Nim, Tim extends Person {}
pred wellformed {
-- friendship is symmetric
all disj p1, p2: Person | p1 in p2.friends implies p2 in p1.friends
-- cannot follow or friend yourself
all p: Person | p not in p.friends and p not in p.followers
}
run {wellformed} for exactly 5 Person
pred reachableIn1To7Hops[to: Person, from: Person, fld: Person->Person] {
to in from.fld or
to in from.fld.fld or
to in from.fld.fld.fld or
to in from.fld.fld.fld.fld or
to in from.fld.fld.fld.fld.fld or
to in from.fld.fld.fld.fld.fld.fld or
to in from.fld.fld.fld.fld.fld.fld.fld
-- ... and so on, for any finite number of hops
-- this is what you use the transitive-closure operator (^)
-- or the reachable built-in predicate for.
}
We said that chaining field access with .
allows us to compute reachability in a certain number of hops. That's how reachableIn1To7Hops
works.
However, there's more to .
than this.
Let's run this model, and open up the evaluator. I'll show the first instance Forge found using the table view:
We saw that Tim.friends
produces the set of Tim
's friends, and that Tim.friends.friends
produces the set of Tim
's friends' friends. But let's try something else. Enter this into the evaluator:
friends.friends
This looks like a nonsense expression: there's no object to reference the friends
field of. But it means something in Forge:
What do you notice about this result? Recall that this is just a parenthetical way to show a set of tuples: it's got
The .
operator in Forge isn't exactly field access. It behaves that way in Froglet, but now that we have sets in the language, it's more powerful. It lets us combine relations in a path-like way.
Here's the precise definition of the relational join operator (.
):
If R
and S
are relations (with R.S
is defined to be the set of
That is, whenever the inner columns of the two relations line up on some value, their join contains some tuple(s) that have the inner columns eliminated.
In a path-finding context, this is why Tim.friends.friends.friends.friends
has one column, and all the intermediate steps have been removed: Tim
has one column, and friends
has 2 columns. Tim.friends
is the Tim
's friends. And so on: every time we join on another friends
, 2 columns are removed.
Let's try this out in the evaluator:
Does this mean that we can write something like followers.Tim
? Yes; it denotes the set of everyone who has Tim
as a follower:
Note that this is very different from Tim.followers
, which is the set of everyone who follows Tim
:
We can use Forge to validate the above definition, for relations with fixed arity. So if we want to check the definition for pairs of binary relations, up to a bound of 10
, we'd run:
test expect {
joinDefinitionForBinary: {
friends.followers =
{p1, p2: Person | some x: Person | p1->x in friends and
x->p2 in followers}
} for 10 Person is theorem
}
Notice that we don't include wellformed
here: if we did, we wouldn't be checking the definition for all possible graphs.
Forge will give you an error message if you try to use join in a way that produces a set with no columns:
or if it detects a type mismatch that would mean the join is necessarily empty:
When you see a parenthesized formula in an error like this, you can read it by interpreting operator names in prefix form. E.g., this one means Int.friends
. (We'll be updating Forge to not show constraints in parenthetical form soon.)
Suppose you're modeling something like Dijkstra's algorithm. You'd need a weighted directed graph, which might be something like this:
sig Node {
edges: Node -> Int
}
But then edges
has three columns, and you won't be able to use either reachable
or ^
on it directly. Instead, you can eliminate the rightmost column with join: edges.Int
, and then use that expression as if it were a set
field.
Now we'll shift gears back to temporal mode.
Let's convert the model from last time into temporal mode. We'll add the necessary options first. Note that options in Forge are usually positional, meaning that it is usually a good idea to have options at the beginning unless you want to vary parameters per run
.
#lang forge
option problem_type temporal
option max_tracelength 10
Be sure to get the underscores right. Unfortunately, Forge's current error if you misspell an option name isn't very friendly.
Also, mixing temporal-mode and normal Forge can be tough, since a state-aware model like we had before won't work well in temporal mode.
In temporal mode, we don't have the ability to talk about specific pre- and post-states, which means we have to change the types of our predicates. For init
, we have:
-- No argument! Temporal mode is implicitly aware of time
pred init {
all p: Process | World.loc[p] = Disinterested
no World.flags
}
The loss of a state is perhaps disorienting. How does Forge evaluate init
without knowing which state we're looking at? In temporal mode, every constraint is evaluated not just about an instance, but also in the context of some moment in time. You don't need to explicitly mention the moment. So no World.flags
is true if, at the current time, there's no flags raised.
Similarly, we'll need to change our transition predicates:
-- Only one argument; same reason as above
pred raise[p: Process] {
// pre.loc[p] = Disinterested
// post.loc[p] = Waiting
// post.flags = pre.flags + p
// all p2: Process - p | post.loc[p2] = pre.loc[p2]
World.loc[p] = Disinterested
World.loc'[p] = Waiting
World.flags' = World.flags + p
all p2: Process - p | World.loc'[p2] = World.loc[p2]
}
I've left the old version commented out, so you can contrast the two. Again, the predicate is true subject to an implicit moment in time. The priming (') operator means "this expression in the next state"; so if raise
holds at some point in time, it means there is a specific relationship between the current and next moments.
We'll convert the other predicates similarly, and then run the model:
run {
-- start in an initial state
init
-- in every state of the lasso, the next state is defined by the transition predicate
always trans
}
There are some threats to success here (like deadlocks!) but we'll return to those on Friday.
When we run, we get this:
In temporal mode, Sterling has 2 "next" buttons, rather than just one. The "Next Trace" button will hold all non-var
relations constant, but ask for a new trace that varies the var
relations. The "Next Config" button forces the non-var
relations to differ. These are useful, since otherwise Forge is free to change the value of any relation, even if it's not what you want changed.
In temporal mode, Sterling shows a "minimap" of the trace in the "Time" tab. You can see the number of states in the trace, as well as where the lasso loops back.
It will always be a lasso, because temporal mode never finds anything but lassos.
You can use the navigation arrows, or click on specific states to move the visualization to that state:
Theming works as normal. For the moment, custom visualizer scripts need to visualize a single state at at time.