2019 12 04 Wisdom Wednesday Tasks and their Trees - TISTATechnologies/caseflow GitHub Wiki
This talk is from my perspective! I am biased towards backend software, Rails, and team echo's work. Echo Team primarily works on Queue focused tasks and has very little interaction with other types of tasks, such as Hearing Tasks or Intake tasks. There is certainly interesting information to be gleaned from a front end/react view of tasks, and the perspectives of the other teams, both on Queue tasks and on the other task types!
While I'm largely following this doc for the presentation, I have a tendency to speak extemporaneously on things I find interesting without planning to, so feel free to take notes and then add them here afterwards! I think better when speaking out loud, so thank you all for being my rubber ducks today.
Three options (currently):
- Most often: An Appeal's Collection of Tasks - parent/child relationships
- Sometimes: Rails Object Inheritance
- Sometimes: An Appeal's Collection of Tasks - Created order (more properly, this is a appeal task graph)
Or sometimes a combination of these!
Is it a problem that we mean multiple things here? No!
While we need to make sure we're all on the same page for any given conversation, these different ways of looking at tasks in Caseflow reveal different insights into what is happening either on an appeal or in the code. If anything, I want more ways to visualize the Tasks in Caseflow, both those on a particular appeal and across the system generally.
Every appeal has a task tree, sort of. It is created over time as the appeal accumulates tasks. Those tasks are related to each other in a tree structure via the parent_id
on a task. A task only has one parent, but can have multiple children.
The task tree of an appeal is not a thing in and of itself. Rather, it is the collection of tasks and how they behave in relation to each other. We cannot simply 'create' an appeal with a task tree beyond the most trivial examples, but rather have to step through how an appeal would create tasks over its journey in Caseflow. Even if we manage to create good exemplar task trees for our development, testing, or demo needs (which I am incredibly in favor of doing!), they would inherently only be a subset of the reality of the possible trees in production.
Example - a completed hypothetical ES docket appeal with Colocated and QR tasks
This tree structure follows some structural norms, but adherence is reliant on all the engineers implementing those norms on each task's behavior. There is no 'tree' behavior configuration, only the sum of the behavior of the tasks.
- Organization task with a user task child
- Mostly true within Queue tasks
- Exception: Judge & Attorney tasks - no org
- Exception: QR - sometimes doesn't bother with a user task
- Child tasks block Parent task
- Within Queue this is now always true, to the best of my knowledge
- All hearing related tasks live under HearingTask
- Except Colocated & Mail subtypes!
- Mostly true within Queue tasks
How tasks relate to each other as this tree has changed over time! Previously valid task trees are no longer valid for new appeals, but could still exist for appeals that were extant before the change.
When we make changes to how the tasks relate to each other, we have to chose what to do with the existing appeals tasks.
- Sometimes they were updated to match the new behavior (looking for example)
- Sometimes they were intentionally or unavoidably not changed
- Sometimes we're waiting to change things while extant appeals finish
Less visual, but more manageable, grep queries!
- Inherits Task:
git grep "< \<Task\>" app/models/tasks
- Inherits MailTask:
git grep "< \<MailTask\>" app/models/tasks
- Inherits any task type:
git grep "< .*Task\>" app/models/tasks
From examining the Task Inheritance Tree we can discover connections and disconnections between the behavior of various tasks.
Task and LegacyTask are completely unrelated in inheritance. Even in who they inherit from! Task is from ActiveRecord, LegacyTask from ActiveModel.
Following the inheritance tree allows you to find behaviors not implemented directly by task class you're looking at. This is especially true in the case of validation behavior.
A validation pattern we follow: the validation invoked at the class and the conditional on that validation is defined differently in various children classes!
Walkthrough:
self.hide_from_queue_table_view
Visual
Newest lens! For any given appeal, shows what tasks were created before and after the other
Documented in the Data-Driven Documentation of Task Trees Deep Dive.
Gives insight into real production task creation order, and can help reveal behavior.
More in Part 4! BLUF: team echo is super psyched about this.
This section will hopefully become rapidly outdated as we improve the task actions architecture, but for now it would be nice to get us all on the same page, so let's dive in!
Defined initially in Task with a few overrides.
- Task: is the task open and Not an org task with a user child task assigned to this user
- RootTask: Overrides Task's definition; always returns true
- HearingAdminActionTask: Extends via super, also checks the user is a Hearings manager or Hearings Admin
- ChangeHearingDispositionTask: Extends via super, also checks the user is a Hearings Admin
actions_available?(user)
- A wrapper around actions_allowable that intended to disallow doing actions to on_hold tasks. (ish)
Defined initially in Task with a few overrides.
- Task: false if on hold (unless timed hold) AND actions_allowable?
- RootTask: Overrides Task's definition; always returns true
- ColocatedTask: Overrides Task's definition; returns true if task is open
- NoShowHearingTask: Overrides Task's definition; just calls actions_allowable?
A good lesson in smaller, focused PRs. The PR adding this (#7484) touched 36 files and wasn't related to the task serializer, so the reasoning is lost.
We call this method instead of available_actions in a dozen tests. I have done this. It is probably the wrong function, but an understandable mistake!
defined 26 distinct times in the Inheritance model . Sometimes refers to the super
function, sometimes does not.
actions returned often consider some combo of:
- is the task is assigned to the user
- is the task is assigned to the user's Organization
- is the task is assigned to another user in the user's Organization
- is the user is an Admin of the Organization of the assigned to user
- is the parent task assigned to this user
- and more
Echo is aware this is a pain to understand, and would love to refactor it!
Unfortunately, there is no easy answer. You need to interrogate the code for which, if any, of the checks it runs first (actions_available?
or actions_allowable?
), confirm that is true
for your user, and then check available_actions
returns.
There is not an easy way for a non-technical person to answer this question, currently, beyond User Impersonation.
Bat Team! Support needs your help! The user says the appeal automatically got cancelled. What's going on with it?
I largely use the Task Tree through structure_render(:id, :status) to determine the what the general step of the process the appeal is at.
We're at initial translation, on a timed hold
irb> appeal.structure_render(:id, :status)
Appeal 201 [id, status]
└── RootTask 596, on_hold
└── DistributionTask 597, on_hold
└── TranslationTask 598, on_hold
└── TimedHoldTask 630, assigned
We're at Judge Assign, but in Colocated
irb> appeal.structure_render(:id, :status)
Appeal 1 [id, status]
└── RootTask 100, on_hold
├── DistributionTask 101, completed
└── JudgeAssignTask 150, on_hold
└── ExtensionColocatedTask 594, on_hold
└── ExtensionColocatedTask 595, assigned
We're at the Evidence Window stage
irb> appeal.structure_render(:id, :status)
Appeal 201 [id, status]
└── RootTask 596, on_hold
└── EvidenceWindowTask 597, assigned
We got to Dispatch, but the case was returned to the Judge for a correction
irb> appeal.structure_render(:id, :status)
Appeal 11 [id, status]
└── RootTask 31, on_hold
├── DistributionTask 32, completed
├── JudgeAssignTask 100, completed
├── JudgeDecisionReviewTask 501, completed
│ └── AttorneyTask 502, completed
└── BvaDispatchTask 600, on_hold
└── BvaDispatchTask 601, on_hold
└── JudgeDispatchReturnTask 700, assigned
Did the Caseflow User close the last issue on appeal? I would ask for the closed_at value structure_render(:id, :status, :closed_at)
and find the tasks that match/are immediately after the updated_at
value on the last updated issue. Those need reopened!
Are we looking at a Legacy Appeal? We only truly keep ColocatedTasks for LegacyAppeals. LegacyJudgeTasks and LegacvyAttorneyTasks are ephemeral and only created in memory to reflect the state in VACOLs. They do not persist once the appeal moves on.
This is why LegacyTask and Task do not share an inheritence tree! They are fundamentally distinct objects.
Why aren't I showing you this on the dev rails console? Well, because task trees are only actually created by the appeal moving through the system, we don't actually have realistic task trees in dev. There's some effort underway to teach the Appeal factory to create more realistic task trees, but these are currently limited to early stages (post-intake, at distribution) and are still lacking in correct statuses and realistic creation times.
We've begun keeping a Bat Team FAQ for established processes for managing known bugs. But what you see here is a streamlined way to fix a known issue, rather than the actual debugging. I highly suggest riding along with folks on Bat Team (as their time allows!) to see how different folks debug issues.
The Data-Driven Documentation of Task Trees
This is the first time we have the tree structures built from reality as opposed to from theory of behavior in the code! We can actually see what our trees look like, what weird edge cases are happening, and refer to these docs on bat team to clarify "how odd is this task tree I'm looking at here?"
I'm also excited to deep dive on any given Task to investigate and validate what good task trees look like.
For example, let's chat about AttorneyTask!
On team echo we've been collecting Tasks Frustrations; we're no strangers to being confused or frustrated with these objects. Please feel free to add your frustrations to that issue; more insight is welcome! We're formulating solutions small and large.
- Thank you Hunter for helping me get the outline for this talk out of my head and into words
- Thank you Kevin for helping in get the final draft done, and then helping me step away from it
- Thank you to Team Echo for dealing with me fretting about the talk all this week
- Thank you Tomas for
structure_render