Versions Compared

Key

  • This line was added.
  • This line was removed.
  • Formatting was changed.

Initially we referred to this as compliance, but because “compliance” also has meaning in a governance context, we’re using “adherence” to describe the measurement of the extent to which a participant performs all the assessments they are asked to perform as part of their participation in a study member participation in the study.

Table of Contents
minLevel1
maxLevel7

...

  1. The schedule itself, which is finite in length and described by a timeline;

  2. The set of adherence records for the participant which describe what the participant has done in that timeline (we only work with the session records when calculating adherence);

  3. A set of event timestamps that tell us where the participant is at in the timeline (ie what the user should have done and what they should currently be doing in the timelineschedule).

So far the Bridge server provides this information through separate APIs and does not keep track of the time of the participants. For the sake of efficiency, I think we want the Bridge server to combine this information and provide reports on the status of the account, which will be less resource intensive that serving out all the information to be processed by a client or worker process. The issues I see with this:

  1. For the first time the server will need to have knowledge of the participant’s time and time zone(?);

  2. Because this information depends on the time of the request, it is not very cacheable;

  3. Nevertheless, the The reports probably update infrequently compared to the amount they will be read (the exact amount depends on many factors), while they may be read frequently in a couple of different formats.

Event Day Adherence Report API

Given the designs I have seen so far, I would suggest an “event by day” report which would be available via a single API:

...

Method

...

Path

...

Description

...

GET

...

/v5/studies/{studyId}/participants/{userId}/adherence/eventday
?activeOnly=true|false

...

activeOnly = show only currently available days. This can be expanded to show days before or after, etc. as useful to clients.

The JSON looks as follows (note that we can add whatever further information we need based on the UI— though ultimately, it may make better sense to look up session and assessment metadata in the timeline). The existing designs are currently minimal, with all windows just being represented by a symbol colored to show its current adherence state:

...

languagejson

...

  1. . We will have to devise caching strategies to make the system performant when looking at many participants.

Persistent time windows will be excluded from all adherence reports. Completing assessments that are part of a persistent window do not count for or against adherence.

All of these views operate on the most recent timestamps for all events. Building schedules that rely on a mutable event changing, and triggering a new timeline of sessions to perform, will not work with these adherence APIs. That would be events like “do X when session type Y has been completed.” Since it will show compliance with the most event time stream, it may be sufficient anyway. Past time streams are no longer actionable.

APIs

Method

Path (Under /v5/studies/{studyId}

Description

GET

/participants/{userId}/adherence/eventstream

EventStreamAdherenceReport for one user. This view includes scheduling based on events the user does not have, and is a detailed view of the entire schedule for one user.

GET

/participants/{userId}/adherence/weekly

A WeeklyAdherenceReport for one user. This calculates the “current week” of each event timestamp and aligns it to a seven day schedule.

GET

/participants/adherence/weekly

A paged list where each record is a WeeklyAdherenceReport for one user. If no call has generated this report, it is not generated as part of this call, so a worker process will need to periodically generate these out-of-request.

GET

/v1/apps/{appId}/studies/{studyId}
/participants/{userId}/adherence/weekly

Worker endpoint to create and persist the weekly report for users, to enable the paged view across all accounts.

Each report builds on the latter report.

EventStream Report (all adherence data for one participant)

This report is intended to support the following type of adherence chart:

...

The report is calculated against the caller’s clientTimeZone value (if they have a value set), or else it is calculated against the timeZone value of the Study (to determine the current date). This is what the JSON would like like:

Code Block
languagejson
{
  "timestamp": "2021-11-23T22:00:31.699Z",
  "clientTimeZone": "America/Los_Angeles",
  "adherencePercent": 25,
  "streams": [
    {
      "startEventId": "custom:event1",
      "eventTimestamp": "2021-11-21T20:00:00.000Z",
      "daysSinceEvent": 2,
      "byDayEntries": {
        "0": [
          {
            "sessionGuid": "eRLgI5gfe1kef_XRZDfdFU9I",
            "sessionLabel": "Session #2",
            "sessionSymbol": "2",
            "startDay": 0,
            "startDate": "2021-11-21",
            "timeWindows": [
              {
                "sessionInstanceGuid": "bDBVV02XrjrOG8NgBYSYFg",
                "timeWindowGuid": "KZ1piANVdeD-r8PCHL2bviLh",
                "state": "completed",
                "endDay": 0,
                "endDate": "2021-11-21",
                "type": "EventStreamWindow"
              }
            ],
            "type": "EventStreamDay"
          }
        ],
        "1": [
          {
            "sessionGuid": "eRLgI5gfe1kef_XRZDfdFU9I",
            "sessionLabel": "Session #2",
            "sessionSymbol": "2",
            "startDay": 1,
            "startDate": "2021-11-22",
            "timeWindows": [
              {
                "sessionInstanceGuid": "u-cc8Ou4xqUPVnht4oh-bw",
                "timeWindowGuid": "KZ1piANVdeD-r8PCHL2bviLh",
                "state": "expired",
                "endDay": 1,
                "endDate": "2021-11-22",
                "type": "EventStreamWindow"
              }
            ],
            "type": "EventStreamDay"
          }
        ]
      },
      "type": "EventStream"
    },
    {
      "startEventId": "custom:event2",
      "eventTimestamp": "2021-11-15T20:00:00.000Z",
      "daysSinceEvent": 8,
      "byDayEntries": {
        "0": [
          {
            "sessionGuid": "z_jb4p2Lr9Q56z8AwiYNieqw",
            "sessionLabel": "Session #3",
            "sessionSymbol": "3",
            "startDay": 0,
            "startDate": "2021-11-15",
            "timeWindows": [
              {
                "sessionInstanceGuid": "KAxvwhsX6jSVl89a3-gdKw",
                "timeWindowGuid": "gF6hy-UiipJLXqe7F_yK-wQc",
                "state": "expired",
                "endDay": 2,
                "endDate": "2021-11-17",
                "type": "EventStreamWindow"
              }
            ],
            "type": "EventStreamDay"
          }
        ],
        "3": [
          {
            "sessionGuid": "z_jb4p2Lr9Q56z8AwiYNieqw",
            "sessionLabel": "Session #3",
            "sessionSymbol": "3",
            "startDay": 3,
            "startDate": "2021-11-18",
            "timeWindows": [
              {
                "sessionInstanceGuid": "nQnWs_4ECvLtY4K1gYk67g",
                "timeWindowGuid": "gF6hy-UiipJLXqe7F_yK-wQc",
                "state": "expired",
                "endDay": 5,
                "endDate": "2021-11-20",
                "type": "EventStreamWindow"
              }
            ],
            "type": "EventStreamDay"
          }
        ]
      },
      "type": "EventStream"
    },
    {
      "startEventId": "study_burst:main-sequence:01",
      "eventTimestamp": "2021-11-21T20:00:00.000Z",
      "daysSinceEvent": 2,
      "studyBurstId": "main-sequence",
      "studyBurstNum": 1,
      "byDayEntries": {
        "0": [
          {
            "sessionGuid": "LcWpQFKaGY5FSQ0LT4tnvdO7",
            "sessionLabel": "Session #1",
            "sessionSymbol": "1",
            "startDay": 0,
            "startDate": "2021-11-21",
            "timeWindows": [
              {
                "sessionInstanceGuid": "1aCbUaFYkixIsIJBf9WGpg",
                "timeWindowGuid": "GNp94CnfTTtR-s0OzrFeftrh",
                "state": "expired",
                "endDay": 0,
                "endDate": "2021-11-21",
                "type": "EventStreamWindow"
              },
              {
                "sessionInstanceGuid": "yQnubrShfYMY9ZzOE3zw3Q",
                "timeWindowGuid": "aRaHNKIY0yKgOl5CLuA3ZDHJ",
                "state": "expired",
                "endDay": 0,
                "endDate": "2021-11-21",
                "type": "EventStreamWindow"
              }
            ],
            "type": "EventStreamDay"
          }
        ]
      },
      "type": "EventStream"
    }
  ],
  "type": "EventStreamAdherenceReport"
}

EventStreamWindow includes the state of that session (which is the basis for adherence calculation). This adherence report could be displayed on a calendar since it is specific to a user and the server must know all the calendrical values to calculate it (it's the only report that can take a timestamp to indicate the time at which the report should be calculated, and it is calculated in the user’s local time if possible).

The completion states are:

State

Description

Adherence

not_applicable

Participant does not have this event in their events, so these sessions will not currently ever be shown to the participant.

N/A

not_yet_available

Participant should not have seen or started this session. It’s in the future.

N/A

unstarted

Participant should see the session (they are being asked to do it now), but they have not started it.

unknown

started

Participant has started the session.

unknown

completed

Participant completed the session before it expired.

compliant

abandoned

Participant started or finished at least one assessment in the session, but there was more work to do and it expired before they finished it.

noncompliant

expired

Participant did not start the session and it is now expired.

noncompliant

We can calculate a compliance percentage from these values across a participant’s entire participation in the study. In the weekly report below, we can calculate adherence for that week alone.

Weekly Adherence Report

This report always returns seven days of adherence records for a given user. The report calculates this information by finding the “week since event N” for every event and every session (it will be the week that has the day that falls on “today” as measured in the user’s local time zone). For performance reasons, we will not calculate this week outside of the “current week” so it’s not possible to move forward or backward from this view (at least initially).

Note that the the individual sessions listed in this report are not lined up by calendar date (unless they were triggered by the same event). The structure of this report would be as follows:

Code Block
languagejson
{
  "participant": {
    "firstName": "A-chan",
    "email": "alx.dark+achan@sagebase.org",
    "externalId": "asdfasdf",
    "identifier": "GqYpNUWolebxS2eQudF1hc-a",
    "type": "AccountRef"
  },
  "requestTimestamp": "2021-11-23T21:03:21.356Z",
  "clientTimeZone": "America/Los_Angeles",
  "createdOn": "2021-11-23T22:03:21.356Z",
  "weeklyAdherencePercent": 33,
  // nextActivity, if byDayEntries is empty
  "byDayEntries": {
    "0": [
      {
        "sessionGuid": "eRLgI5gfe1kef_XRZDfdFU9I",
        "sessionLabel": "Session #2",
        "sessionSymbol": "2",
        "week": 1,
        "startDate": "2021-11-21",
        "timeWindows": [
          {
            "sessionInstanceGuid": "bDBVV02XrjrOG8NgBYSYFg",
            "timeWindowGuid": "KZ1piANVdeD-r8PCHL2bviLh",
            "state": "completed",
            "endDate": "2021-11-21",
            "type": "EventStreamWindow"
          }
        ],
        "type": "EventStreamDay"
      },
      {
        "sessionGuid": "LcWpQFKaGY5FSQ0LT4tnvdO7",
        "sessionLabel": "Session #1",
        "sessionSymbol": "1",
        "week": 1,
        "studyBurstId": "main-sequence",
        "studyBurstNum": 1,
        "startDate": "2021-11-21",
        "timeWindows": [
          {
            "sessionInstanceGuid": "1aCbUaFYkixIsIJBf9WGpg",
            "timeWindowGuid": "GNp94CnfTTtR-s0OzrFeftrh",
            "state": "expired",
            "endDate": "2021-11-21",
            "type": "EventStreamWindow"
          },
          {
            "sessionInstanceGuid": "yQnubrShfYMY9ZzOE3zw3Q",
            "timeWindowGuid": "aRaHNKIY0yKgOl5CLuA3ZDHJ",
            "state": "expired",
            "endDate": "2021-11-21",
            "type": "EventStreamWindow"
          }
        ],
        "type": "EventStreamDay"
      }
    ],
    "1": [
      {
        "sessionGuid": "eRLgI5gfe1kef_XRZDfdFU9I",
        "sessionLabel": "Session #2",
        "sessionSymbol": "2",
        "week": 1,
        "startDate": "2021-11-22",
        "timeWindows": [
          {
            "sessionInstanceGuid": "u-cc8Ou4xqUPVnht4oh-bw",
            "timeWindowGuid": "KZ1piANVdeD-r8PCHL2bviLh",
            "state": "expired",
            "endDate": "2021-11-22",
            "type": "EventStreamWindow"
          }
        ],
        "type": "EventStreamDay"
      },
      {
        "sessionGuid": "LcWpQFKaGY5FSQ0LT4tnvdO7",
        "sessionLabel": "Session #1",
        "sessionSymbol": "1",
        "week": 1,
        "studyBurstId": "main-sequence",
        "studyBurstNum": 1,
        "startDate": "2021-11-22",
        "timeWindows": [
          {
            "sessionInstanceGuid": "7HaIdOehYJrJk3VZcGeNxg",
            "timeWindowGuid": "GNp94CnfTTtR-s0OzrFeftrh",
            "state": "completed",
            "endDate": "2021-11-22",
            "type": "EventStreamWindow"
          },
          {
            "sessionInstanceGuid": "R0I7n1fEeIR88fYv6iVPTg",
            "timeWindowGuid": "aRaHNKIY0yKgOl5CLuA3ZDHJ",
            "state": "expired",
            "endDate": "2021-11-22",
            "type": "EventStreamWindow"
          }
        ],
        "type": "EventStreamDay"
      }
    ],
    "2": [
      {
        "sessionGuid": "eRLgI5gfe1kef_XRZDfdFU9I",
        "sessionLabel": "Session #2",
        "sessionSymbol": "2",
        "week": 1,
        "startDate": "2021-11-23",
        "timeWindows": [
          {
            "sessionInstanceGuid": "s7HuTxm4E-ffhfiddVFkXQ",
            "timeWindowGuid": "KZ1piANVdeD-r8PCHL2bviLh",
            "state": "started",
            "endDate": "2021-11-23",
            "type": "EventStreamWindow"
          }
        ],
        "type": "EventStreamDay"
      },
      {
        "sessionGuid": "z_jb4p2Lr9Q56z8AwiYNieqw",
        "sessionLabel": "Session #3",
        "sessionSymbol": "3",
        "week": 2,
        "startDate": "2021-11-24",
        "timeWindows": [
          {
            "sessionInstanceGuid": "WHE_gHE71tk8qFERatcruA",
            "timeWindowGuid": "gF6hy-UiipJLXqe7F_yK-wQc",
            "state": "not_yet_available",
            "endDate": "2021-11-26",
            "type": "EventStreamWindow"
          }
        ],
        "type": "EventStreamDay"
      },
      {
        "sessionGuid": "LcWpQFKaGY5FSQ0LT4tnvdO7",
        "sessionLabel": "Session #1",
        "sessionSymbol": "1",
        "week": 1,
        "studyBurstId": "main-sequence",
        "studyBurstNum": 1,
        "startDate": "2021-11-23",
        "timeWindows": [
          {
            "sessionInstanceGuid": "SgTohgnPkGjB9eE73QRXEA",
            "timeWindowGuid": "GNp94CnfTTtR-s0OzrFeftrh",
            "state": "completed",
            "endDate": "2021-11-23",
            "type": "EventStreamWindow"
          },
          {
            "sessionInstanceGuid": "ezwrtaRiJd5Bfa3w5Xxffw",
            "timeWindowGuid": "aRaHNKIY0yKgOl5CLuA3ZDHJ",
            "state": "unstarted",
            "endDate": "2021-11-23",
            "type": "EventStreamWindow"
          }
        ],
        "type": "EventStreamDay"
      }
    ],
    "3": [
      {
        "sessionGuid": "eRLgI5gfe1kef_XRZDfdFU9I",
        "sessionLabel": "Session #2",
        "sessionSymbol": "2",
        "week": 1,
        "startDate": "2021-11-24",
        "timeWindows": [
          {
            "sessionInstanceGuid": "u78_-kFdyNnKt6wbUmSXcA",
            "timeWindowGuid": "KZ1piANVdeD-r8PCHL2bviLh",
            "state": "not_yet_available",
            "endDate": "2021-11-24",
            "type": "EventStreamWindow"
          }
        ],
        "type": "EventStreamDay"
      },
      {
        "sessionGuid": "LcWpQFKaGY5FSQ0LT4tnvdO7",
        "sessionLabel": "Session #1",
        "sessionSymbol": "1",
        "week": 1,
        "studyBurstId": "main-sequence",
        "studyBurstNum": 1,
        "startDate": "2021-11-24",
        "timeWindows": [
          {
            "sessionInstanceGuid": "mcpv_-sW1vXfk4Vyi5dwgg",
            "timeWindowGuid": "GNp94CnfTTtR-s0OzrFeftrh",
            "state": "not_yet_available",
            "endDate": "2021-11-24",
            "type": "EventStreamWindow"
          },
          {
            "sessionInstanceGuid": "t4AEs7UzOjXYACSOO1REPA",
            "timeWindowGuid": "aRaHNKIY0yKgOl5CLuA3ZDHJ",
            "state": "not_yet_available",
            "endDate": "2021-11-24",
            "type": "EventStreamWindow"
          }
        ],
        "type": "EventStreamDay"
      }
    ],
    "4": [
      {
        "sessionGuid": "eRLgI5gfe1kef_XRZDfdFU9I",
        "sessionLabel": "Session #2",
        "sessionSymbol": "2",
        "week": 1,
        "startDate": "2021-11-25",
        "timeWindows": [
          {
            "sessionInstanceGuid": "tJxcgcQK40X6jL71MORb3g",
            "timeWindowGuid": "KZ1piANVdeD-r8PCHL2bviLh",
            "state": "not_yet_available",
            "endDate": "2021-11-25",
            "type": "EventStreamWindow"
          }
        ],
        "type": "EventStreamDay"
      },
      {
        "sessionGuid": "LcWpQFKaGY5FSQ0LT4tnvdO7",
        "sessionLabel": "Session #1",
        "sessionSymbol": "1",
        "week": 1,
        "studyBurstId": "main-sequence",
        "studyBurstNum": 1,
        "startDate": "2021-11-25",
        "timeWindows": [
          {
            "sessionInstanceGuid": "22nOvDMhYIE7btX6X8zlHw",
            "timeWindowGuid": "GNp94CnfTTtR-s0OzrFeftrh",
            "state": "not_yet_available",
            "endDate": "2021-11-25",
            "type": "EventStreamWindow"
          },
          {
            "sessionInstanceGuid": "H2EG1SV-tF_s8dmKWwvtOQ",
            "timeWindowGuid": "aRaHNKIY0yKgOl5CLuA3ZDHJ",
            "state": "not_yet_available",
            "endDate": "2021-11-25",
            "type": "EventStreamWindow"
          }
        ],
        "type": "EventStreamDay"
      }
    ],
    "5": [
      {
        "sessionGuid": "eRLgI5gfe1kef_XRZDfdFU9I",
        "sessionLabel": "Session #2",
        "sessionSymbol": "2",
        "week": 1,
        "startDate": "2021-11-26",
        "timeWindows": [
          {
            "sessionInstanceGuid": "-OoNi3vhifRUqN6X9WTMCg",
            "timeWindowGuid": "KZ1piANVdeD-r8PCHL2bviLh",
            "state": "not_yet_available",
            "endDate": "2021-11-26",
            "sessionGuidtype": "vZBHBVv_H2_1TBbELF48czjS",
 EventStreamWindow"
          }
        ],
        "labeltype": "Session #1",
 EventStreamDay"
      },
      {
        "symbolsessionGuid": "circlez_jb4p2Lr9Q56z8AwiYNieqw",
          "timeWindowssessionLabel":[ "Session #3",
        "sessionSymbol": "3",
 {       "week": 2,
        "sessionInstanceGuidstartDate": "ePcCf6VmfIiVuU0ckdBeRw2021-11-27",
    
         "timeWindowGuidtimeWindows":"sUaNAasy_LiT3_IYa1Fx_dSv",
 [
          {
  "state":"not_yet_available",
              "typesessionInstanceGuid": "EventDayWindow2O-jPnpWOYZLx0VjBvvO8g",
             }"timeWindowGuid": "gF6hy-UiipJLXqe7F_yK-wQc",
            {
"state": "not_yet_available",
             "sessionInstanceGuidendDate": "DB13D4mO72j6S2021-11-g7PIkI2Q29",
              "timeWindowGuidtype": "Bw6rAAeG6zotqes4cLSgKjh5EventStreamWindow",
          }
   "state":"not_yet_available",     ],
         "type": "EventDayWindowEventStreamDay"
      },
     } {
         ],
"sessionGuid": "LcWpQFKaGY5FSQ0LT4tnvdO7",
         "typesessionLabel": "EventDay"
   Session #1",
    }    "sessionSymbol": "1",
 ],       "2week":[ 1,
       { "studyBurstId": "main-sequence",
        "sessionGuidstudyBurstNum":"vZBHBVv_H2_1TBbELF48czjS" 1,
          "labelstartDate": "Session #12021-11-26",
 
        "symboltimeWindows": "circle",[
          "timeWindows":[{
            {
 "sessionInstanceGuid": "JfyiAbHZ-nlmTZui_N19Tg",
            "sessionInstanceGuidtimeWindowGuid": "wvEV4fJZQ0nfgYGNp94CnfTTtR-TN2LekAs0OzrFeftrh",
 
            "timeWindowGuidstate": "sUaNAasynot_LiT3yet_IYa1Fx_dSvavailable",
              "stateendDate":"not_yet_available "2021-11-26",
              "type": "EventDayWindowEventStreamWindow"
 
          },

           {
 
            "sessionInstanceGuid": "IHDTSoj552vGDv1Qt7nXkgg6kRO-JZgt0lGghDT8v37A",
 
            "timeWindowGuid": "Bw6rAAeG6zotqes4cLSgKjh5aRaHNKIY0yKgOl5CLuA3ZDHJ",

             "state": "not_yet_available",

             "typeendDate": "EventDayWindow2021-11-26",
            }
          ],
          "type": "EventDayEventStreamWindow"
          }
        ]
,
   },     "type": "EventDayAdherenceReportEventStreamDay"
  },    {}
    "startEventId":"study_burst:ClinicVisit:02"],
    "eventTimestamp":"2021-11-16T19:00:00.000Z",6": [
      "entries":{
        "1sessionGuid":[ "eRLgI5gfe1kef_XRZDfdFU9I",
        {
 "sessionLabel": "Session #2",
        "sessionGuidsessionSymbol": "vZBHBVv_H2_1TBbELF48czjS2",
          "labelweek": "Session #1"1,

         "symbolstartDate": "circle2021-11-27",
          "timeWindows": [
            {
              "sessionInstanceGuid": "zk7X4dQCy7Nvnuo2PcnSCA_-1AlFrrN9IfA5tK-OYvag",
              "timeWindowGuid":"sUaNAasy_LiT3_IYa1Fx_dSv "KZ1piANVdeD-r8PCHL2bviLh",
 
            "state": "not_yet_available",
 
            "typeendDate": "EventDayWindow2021-11-27",
            }, "type": "EventStreamWindow"
          }
  {      ],
        "sessionInstanceGuidtype": "rMRne-cbwIN5mkGZLymxzg"EventStreamDay"
      },
      {
        "timeWindowGuidsessionGuid": "Bw6rAAeG6zotqes4cLSgKjh5LcWpQFKaGY5FSQ0LT4tnvdO7",
        "sessionLabel": "Session #1",
        "statesessionSymbol": "not_yet_available1",
        "week": 1,
        "typestudyBurstId": "EventDayWindowmain-sequence",
        "studyBurstNum": 1,
  }           ],
 "startDate": "2021-11-27",
        "typetimeWindows":"EventDay" [
       }      {
],       "2":[      "sessionInstanceGuid": "IUBXsO4ioFgJfi8GA9NSxw",
 {           "sessionGuidtimeWindowGuid":"vZBHBVv_H2_1TBbELF48czjS "GNp94CnfTTtR-s0OzrFeftrh",
            "labelstate": "Session #1not_yet_available",
            "symbolendDate": "circle2021-11-27",
            "type": "EventStreamWindow"
      "timeWindows":[    },
        {  {
            "sessionInstanceGuid": "QXM1cO6yb0gSPWzRwRD8eAWdRjHkDWoqmQx4vxRuCVeg",
              "timeWindowGuid":"sUaNAasy_LiT3_IYa1Fx_dSv "aRaHNKIY0yKgOl5CLuA3ZDHJ",
              "state": "not_yet_available",
              "typeendDate":"EventDayWindow" "2021-11-27",
            "type": "EventStreamWindow"
},          }
  {      ],
        "sessionInstanceGuidtype": "hCXFevxbBnpaUYjH212dsQEventStreamDay",
      }
    ]
  },
  "timeWindowGuidtype": "Bw6rAAeG6zotqes4cLSgKjh5",
              "state":"not_yet_available",
              "type":"EventDayWindow"
            }
          ],
          "type":"EventDay"
        }
      ]
    },
    "type":"EventDayAdherenceReport"
  }
]

All sessions in the timeline are grouped in this view by the event that triggers them, and then the number of days since that event. All potential events in the schedule are included in this report whether they exist for the user or not (we don’t currently have a way to say “count this part of the schedule if the event exists for the user, but don’t count it if the event doesn’t exist for the user). Then the actual “days since each event” are calculated to determine what the state of each time window is (the window state is a session-level measure of the state, derived from the adherence records of the assessments in that session).

Basically we figure out what they should be doing, and then look at their adherence records to figure out if they’re doing it.

The states are:

...

State

...

Description

...

Adherence

...

not_yet_available

...

Participant should not have seen or started this session. It’s in the future.

...

N/A

...

unstarted

...

Participant should see the session (they are being asked to do it now), but they have not started it.

...

unknown

...

started

...

Participant has started the session.

...

unknown

...

completed

...

Participant completed the session before it expired.

...

compliant

...

abandoned

...

Participant started or finished at least one assessment in the session, but there was more work to do and it expired before they finished it.

...

noncompliant

...

expired

...

Participant did not start the session and it is now expired.

...

noncompliant

I would propose that a participant’s noncompliance percentage is equal to noncompliant / (compliant + noncompliant + unknown). We can then set a threshold at which we would want to intervene (from 0% — any failure gets reported — to something less stringent).

TODO: How do we represent persistent time windows? I think I have said elsewhere that performing it one time = compliant, but that might not be correct.

TODO: There’s nothing here about whether or not specific events should be generated for the user…we’re assuming that the user must do everything in the schedule.

TODO: This view operates on the most recent timestamp for each event. Study bursts generate multiple events and will work with this, but do we have cases where a part of the timeline is supposed to repeat due to an event being updated? For example, if the event is “visited the clinic” and the study expects participants to visit four times over the length of the study, we cannot capture this as an expectation. They’d have to create four separate events for this, they couldn’t just update “clinic_visited” four times.

Study-level paged APIs

These reports are not easily cacheable because the states depend on the current time of the request. A single report takes about 1/3 of a second, so paginated lists of user would be prohibitively expensive to calculate. Let’s assume we have a worker process that creates some form of update (probably of just the active days of the report) and caches it for the list view API. What this means is that the list view will be behind the current state by a day or so. Coordinators could go in and look at individual accounts for more up-to-date information.

In past work I have also created “sweepers” for things like sporting event games that just endlessly loop and refresh caches. Given half a second for each record, for example, all of mPower could be updated every 16 minutes.

...

Method

...

Path

...

GET

...

/v5/studies/{studyId}/participants/adherence/eventday?offsetBy=&pageSize=

Here is a visual design for this feature…this is all I got on requirements at this point:

...

Each row represents one participant, presumably seven days around their current day in the study. Some questions to answer about this:

  • Is this the image of the most recent study burst for each user? If so, day 1 is the day 0 of that burst, not just “three days ago.” If so, what if the burst is longer than 7 days, what if the schedule does not use study bursts?

  • If the user is in a “fallow” period of the schedule (like a period of time between study bursts), what do we return?

I would be inclined to simply show X days before and after “now” for the user, across all event streams. That means there could be more than one row of symbols in a row, I think (or maybe the different sessions are just shown side-by-side).

Presumably we will only persist the records we need for this view, and then we can return all records in the paged API. I think one record in this API must be equal to one participant, and it will contain nested information to draw one or more of these rows.

DateAdherenceReport

Here’s one possible report structure for the above (all the sessions and session windows are squished into one day of information, so the above reports are showing seven of these per user…they need then to be summarized for a week because that’s too many individual reports to download:

Code Block
languagejson
{
    "studyId": "foo",
    "userId": "BnfcoJLwm95XXkd_Dxjo_w",
    "date": "2021-10-27",
    "sessions": [
        {
            "sessionGuid": "sePT2TAcQ-1ZBY_xeuoA7w0r",
            "label": "Persistent session",
            "sessionInstanceGuid": "BnfcoJLwm95XXkd_Dxjo_w",
            "state": "expired"
        },
        {
            "sessionGuid": "gDnRq7C1LMeTdT1aCDC4vTOo",
            "label": "Also repeats on enrollment at different cadence",
            "sessionInstanceGuid": "mwHBh8lxaW7zYMMOAhhQKw",
            "state": "expired"
        }
    ]
}

Protocol adherence and notifications

There is definitely a lot of possible ways we could measure adherence and I have some questions I have asked above about this. We can probably be pretty flexible on this since at some point, we have to load the whole state of the user’s participation to generate this report. We could then set a threshold for this (ie. warn administrators if there are any non-compliant assessments, vs. setting a proportion threshold that would need to be hit before notifying administrators). This warning apparently applies to specific participants, so it would be generated as part of the worker process described above.

Messaging APIs

One UI design I have seen shows a list of notifications in an administrative UI, indicating specific participants that are out-of-adherence with the study protocol. This hints at a very large amount of functionality.

We have had requirements to message both participants and study administrators over the years. For example, very early on we embedded a blog in an app for this purpose. Then we implemented support for remote notifications and topics (rarely used). We often message participants through local notifications based on scheduling information. Now we are talking about showing notifications to administrators about participants being out-of-adherence in an administrative UI.

I would like to add a simple but complete messaging system to Bridge which could be used to persist, deliver, and record the consumption of messages. Features would include:

  • API would be push-pull (write a message, request a list of active messages for a recipient; push-push is a lot more complicated);

  • recipients could include individuals or organizations (and other groups once we have them);

  • each recipient could mark the message as “read” to remove it from their UIs, separate from others, or “resolved” to remove it from everyone’s UIs. They would not be deleted so they are auditable;

  • messages could be set to expire;

  • Messages could indicate if they can be read, resolved, or if they expire. For example, an adherence message might only allow you to mark it “resolved” since you don’t want multiple study coordinators all following up with a participant.

  • Messages could be assigned (change recipients), and indicate a status like “busy” along with the user currently working on the message. Now you have a very simple way to hand off work between team members.

Once pull messaging is working, we could support “push” messaging (where we send a message via email or SMS, or possibly through remote or local push notifications). This introduced a lot of additional complexity, however:

...

It needs to be integrated with Amazon’s messaging APIs, where we are recording “sent” status. The advantage is that we’d have a record of such messaging, which is part of the study protocol and something that we’ve put together one-off solutions for in the past for scientists;

...

WeeklyAdherenceReport"
}

This API would calculate and persist this report before returning it in the call, so that this call can be used by a worker process to build a record for every participant. The call could also be used to retrieve up-to-date state information for a specific participant. If this call is not called for a user, it will not show up in the paginated view of these records. We may want to hide a record that hasn’t been updated in the last 24 hours although a worker process will hopefully force a refresh on all accounts.

The SQL table for this report will not attempt to store the report normalized:

Code Block
languagesql
CREATE TABLE `WeeklyAdherenceReports` (
  `appId` varchar(255) NOT NULL,
  `studyId` varchar(255) NOT NULL,
  `userId` varchar(255) NOT NULL,
  `requestTimestamp` bigint(20) unsigned NOT NULL,
  `createdOn` bigint(20) unsigned NOT NULL,
  `labels` text NOT NULL,
  `weeklyAdherencePercent` int(3) unsigned NOT NULL,
  `reportData` mediumtext NOT NULL,
  PRIMARY KEY (`appId`, `studyId`, `userId`),
  CONSTRAINT `WeeklyAdherenceReports-Study-Constraint` 
      FOREIGN KEY (`studyId`, `appId`) 
      REFERENCES `Substudies` (`id`, `studyId`) ON DELETE CASCADE,
  CONSTRAINT `WeeklyAdherenceReports-Account-Constraint` 
      FOREIGN KEY (`userId`) 
      REFERENCES `Accounts` (`id`) ON DELETE CASCADE
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;

The “labels“ columns will pull out the labels from the JSON into a pipe-delimited string, to enable searching for records that match the label in the study-scoped API (searching by label and percentage of adherence are UI design requirements). Note that right now, this would mean the server would need to create the labels and the client would need to search with those server-supplied labels.

Weekly Adherence Reports for Study

This API is intended to support the following kind of adherence overview for the study:

...

The final API will allow an admin to retrieve a paged set of weekly adherence records for all participants in the study. This retrieves the records from the database and does not calculate them. The paging search criteria should be the same as the study participants search API. In addition to the search arguments in the /v5/studies/{studyId}/participants/search API, it should be possible to filter and or sort on the following fields:

  • The adherence percentage;

  • The label of individual windows (through a like query on the labels column).

I would like to reuse the search code for participants, but it may be necessary to join tables in a way where that can’t be done, while still maintaining the desired paging logic. To start, it’s probably possible to just search and sort by adherence and labels.

Protocol adherence and notifications (worker process)

Since all the reports give an adherence percentage (whole study or one week), we can use this to sort records by adherence in the weekly view, or to generate a message to administrators as part of the worker process that will be run nightly to update the weekly reports. Any necessary configuration for this can be added to the Study model (in particular, at what percentage of adherence should we notify administrators that a participant is out of adherence).

A worker process could run hourly, querying Bridge for any studies that had a time zone such that the local time of the study was at a specific time (e.g. 4am). If that study had a schedule, all the participants would be retrieved and the weekly adherence report would be generated for them. Since these reports are being generated on the server’s time, it’s possible in unusual scenarios that this time would give odd results for a user in a far off time zone…if so we can possibly run reports for a study more frequently than every 24 hours (but we needn’t start by implementing to prevent this).

If the worker hit certain criteria, it could also send a message warning about a participant being out of adherence.

Adherence Service

The following methods will be added to the AdherenceService:

Code Block
languagejava
public class AdherenceService {
  public EventStreamAdherenceReport getEventStreamAdherenceReport(
      String appId, String studyId, String userId, DateTime now, boolean showActive) {
  }
  public WeeklyAdherenceReport getWeeklyAdherenceReport(
      String appId, String studyId, String userId, DateTime now) {
  }
  public PagedResourceList<WeeklyAdherenceReport> getPagedWeeklyAdherenceReports(
      AdherenceSearch search) {
  }  
}

public class AdherenceSearch {
  int offsetBy;
  int pageSize;
  searches accounts that are enrolled in a specific study
  String studyId;
  String emailFilter;
  String phoneFilter;
  Set<String> allOfGroups;
  Set<String> noneOfGroups; 
  String language;
  DateTime startTime;
  DateTime endTime;
  String externalIdFilter;
  AccountStatus status;
  String attributeKey;
  String attributeValueFilter;
  String sessionLabel;
  Integer weekNumber;
  String studyBurstId;
  Integer studyBurstNum;
  Integer maxAdherencePercent;
  SearchTermPredicate predicate;
  StringSearchPosition stringSearchPosition;
}

System Messaging APIs

We have had requirements to message both participants and study administrators over the years. For example, very early on we embedded a blog in an app for this purpose. Then we implemented support for remote push notifications and topics (rarely used). Instead in recent studies we have tried using local notifications based on scheduling information. Now we are talking about showing notifications to administrators about participants being out-of-adherence in an administrative UI. Because “notifications” is heavily overloaded for Bridge, I will call these “system messages.”

I would like to add a simple Ωmessaging system to Bridge which could be used to persist, deliver, and record the consumption of messages. Features would include:

  • API would be pub-sub and pull-based (write a message, request a list of active messages for a recipient; push-push via something like SMS or email is a lot more complicated);

  • Recipients should include individuals only, opt-in, to start;

  • Each recipient could mark the message as “read” to hide it from their UIs, or “resolved” to hide it for everyone. They would not be deleted so they are auditable. We should record who deleted or resolved the message;

  • Messages would have a type (“out of adherence,” “study joined,” “study completed”, etc.) and messages could be filtered on the type and/or study;

  • These should be app-scoped because it would be very annoying to have to go into every study to see what needed attention in that study.

This should meet the basics needs of our recent UI designs.

Method

Path

Description

GET

/v1/messages

Get the messages (paged and filterable by type and whether message has been read) for the caller.

POST

/v1/messages

Publish a message to all subscribers.

DELETE

/v1/messages/{guid}

Mark a message as read or resolved (if resolved=true)

GET

/v1/messages/subscriptions

Get the studies and types of messages you are subscribed to

POST

/v1/messages/subscriptions/{studyId}/{type}

Subscribe to receive messages of a specific type in a specific study.

DELETE

/v1/messages/subscriptions/{studyId}/{type}

Unsubscribe from receiving messages of a given type in a given study.

The objects (subscriptions and message):

Code Block
languagejava
public class Subscription {
  String appId;
  String studyId;
  MessageType messageType;
  String recipientId;
}

// As submitted
public class Message {
  String studyId;
  MessageType messageType;
  String text;
}

// As persisted, one record per subscription
public class Message {
  String guid;
  String appId;
  String studyId;
  String userId;
  MessageType messageType;
  String text;
  DateTime createdOn;
  boolean read;
  String readBy;
}

Additional proposed work items

  • Allow study burst and custom event IDs to have spaces, essentially allowing them to be human-readable labels like “Study Burst” or “Clinic Visit.” Right now we prevent the obvious labels from being used but this becomes difficult when we’re going to allow people to search by labels that include these identifiers.

  • Study should add a studyTimeZone field. We may also want to add an adherenceThresholdPercent in anticipation of messaging.