15.13.1Consent Service: JavaScript API

 

The following methods may be implemented in a consent service script:

15.13.2Method: consentStartOperation

 

This method is called once at the start of each request. The purpose is to determine whether a request should be permitted to proceed, and whether to apply the rest of the consent service logic to the request. It may also pre-fetch any relevant data or resources that will be used to provide consent rules in subsequent method calls.

Inputs:

  • theRequestDetails – This object contains details about the FHIR request being made. The object will be of type RequestDetailsJson.
  • theUserSession – This object contains details about the authenticated user and their session. This includes demographics, permissions, approved scopes, etc. The object will be of type UserSessionDetailsJson.
  • theContextServices – This object has utility methods for interacting with the environment, and indicating a consent outcome to the rest of the framework. The object will be of type ScriptConsentContextServices.
  • theClientSession – This object contains details about the authenticated OIDC client and their OAuth2 session. The object will be of type OAuth2ClientSession.

Outputs:

  • This method does not return an output, but must indicate a response mode by calling an appropriate method on theContextServices.

Example:

function consentStartOperation(theRequestDetails, theUserSession, theContextServices, theClientSession) {

	// For superusers, we will skip the rest of the consent service entirely
	if (theUserSession != null && theUserSession.hasAuthority('ROLE_SUPERUSER')) {
		theContextServices.authorized();
		return;
	}

	// By default, this request should continue but still be evaluated by the service
	theContextServices.proceed();

}

15.13.3Method: consentCanSeeResource

 

This method is called once for each resource that may be returned to the user. It has the ability to authorize the release of data, or to block it, or to request that it be passed to consentWillSeeResource for masking.

Note that this method will not be called if theContextServices.authorized() was invoked by the consent script in the consentStartOperation(...) method. This method can carry a significant performance impact, so it should only be used if necessary.

Inputs:

  • theRequestDetails – This object contains details about the FHIR request being made. The object will be of type RequestDetailsJson.
  • theUserSession – This object contains details about the authenticated user and their session. This includes demographics, permissions, approved scopes, etc. The object will be of type UserSessionDetailsJson.
  • theContextServices – This object has utility methods for interacting with the environment, and indicating a consent outcome to the rest of the framework. The object will be of type ScriptConsentContextServices.
  • theResource – The resource that will be returned. The object will be a FHIR resource and can be examined using the techniques described in FHIR Model API.
  • theClientSession – This object contains details about the authenticated OIDC client and their OAuth2 session. The object will be of type OAuth2ClientSession.

Outputs:

  • This method does not return an output, but must indicate a response mode by calling an appropriate method on theContextServices.

Example:

/**
 * If present, this function is called before every resource that may be returned
 * to the user.
 *
 * @param theRequestDetails  Contains details about the request (e.g. the
 *                           FHIR operation being performed, the HTTP method,
 *                           the URL, etc.
 * @param theUserSession     Contains details about the logged in user and
 *                           their session.
 * @param theContextServices Contains various utility methods for accessing
 *                           relevant information about the request, as well
 *                           as providing a response.
 * @package theResource      The resource that will be accessed.
 * @param theClientSession   Contains details about the OIDC client and
 *                           their OAuth2 session.
 */
function consentCanSeeResource(theRequestDetails, theUserSession, theContextServices, theResource, theClientSession) {
	if ( ...some condition... ) {
		theContextServices.reject();
	} else {
		theContextServices.proceed();
	}
}

15.13.4Method: consentWillSeeResource

 

This method is called once for each resource that is being returned to the user. It has the ability to mask specific elements (or even the whole resource) from being returned to the user.

Inputs:

  • theRequestDetails – This object contains details about the FHIR request being made. The object will be of type RequestDetailsJson.
  • theUserSession – This object contains details about the authenticated user and their session. This includes demographics, permissions, approved scopes, etc. The object will be of type UserSessionDetailsJson.
  • theContextServices – This object has utility methods for interacting with the environment, and indicating a consent outcome to the rest of the framework. The object will be of type ScriptConsentContextServices.
  • theResource – The resource that will be returned. The object will be a FHIR resource and can be examined using the techniques described in FHIR Model API.
  • theClientSession – This object contains details about the authenticated OIDC client and their OAuth2 session. The object will be of type OAuth2ClientSession.

Outputs:

  • This method does not return an output, but must indicate a response mode by calling an appropriate method on theContextServices.

Example:

/**
 * If present, this function is called before every resource that may be returned
 * to the user.
 *
 * @param theRequestDetails  Contains details about the request (e.g. the
 *                           FHIR operation being performed, the HTTP method,
 *                           the URL, etc.
 * @param theUserSession     Contains details about the logged in user and
 *                           their session.
 * @param theContextServices Contains various utility methods for accessing
 *                           relevant information about the request, as well
 *                           as providing a response.
 * @package theResource      The resource that will be accessed.
 * @param theClientSession   Contains details about the OIDC client and
 *                           their OAuth2 session.
 */
function consentWillSeeResource(theRequestDetails, theUserSession, theContextServices, theResource, theClientSession) {
	if ( ...some condition... ) { 
	   theResource.clear('value');
	}
}

15.13.5Method: completeOperationSuccess

 

This function will be called exactly once for each request that completes successfully (i.e. returns an HTTP 2xx response). Note that in the case of failed requests (HTTP 4xx or 5xx), completeOperationFailure will be called instead.

Inputs:

  • theRequestDetails – This object contains details about the FHIR request being made. The object will be of type RequestDetailsJson.
  • theUserSession – This object contains details about the authenticated user and their session. This includes demographics, permissions, approved scopes, etc. The object will be of type UserSessionDetailsJson.
  • theContextServices – This object has utility methods for interacting with the environment, and indicating a consent outcome to the rest of the framework. The object will be of type ScriptConsentContextServices.
  • theClientSession – This object contains details about the authenticated OIDC client and their OAuth2 session. The object will be of type OAuth2ClientSession.

Outputs:

  • This method does not return an output.

Example:

/**
 * If present, this function is called when a request has been handled
 * successfully (i.e. an HTTP 2xx response)
 *
 * @param theRequestDetails  Contains details about the request (e.g. the
 *                           FHIR operation being performed, the HTTP method,
 *                           the URL, etc.
 * @param theUserSession     Contains details about the logged in user and
 *                           their session.
 * @param theContextServices Contains various utility methods for accessing
 *                           relevant information about the request, as well
 *                           as providing a response.
 * @param theClientSession   Contains details about the OIDC client and
 *                           their OAuth2 session.
 */
function completeOperationSuccess(theRequestDetails, theUserSession, theContextServices, theClientSession) {
	// some functions...
}

15.13.6Method: completeOperationFailure

 

This function will be called exactly once for each request that completes unsuccessfully (i.e. returns an HTTP 4xx or 5xx response).

Inputs:

  • theRequestDetails – This object contains details about the FHIR request being made. The object will be of type RequestDetailsJson.
  • theUserSession – This object contains details about the authenticated user and their session. This includes demographics, permissions, approved scopes, etc. The object will be of type UserSessionDetailsJson.
  • theContextServices – This object has utility methods for interacting with the environment, and indicating a consent outcome to the rest of the framework. The object will be of type ScriptConsentContextServices.
  • theClientSession – This object contains details about the authenticated OIDC client and their OAuth2 session. The object will be of type OAuth2ClientSession.

Outputs:

  • This method does not return an output.

Example:

/**
 * If present, this function is called when a request has been handled
 * unsuccessfully (i.e. an HTTP 4xx/5xx response)
 *
 * @param theRequestDetails  Contains details about the request (e.g. the
 *                           FHIR operation being performed, the HTTP method,
 *                           the URL, etc.
 * @param theUserSession     Contains details about the logged in user and
 *                           their session.
 * @param theContextServices Contains various utility methods for accessing
 *                           relevant information about the request, as well
 *                           as providing a response.
 * @param theClientSession   Contains details about the OIDC client and
 *                           their OAuth2 session.
 */
function completeOperationFailure(theRequestDetails, theUserSession, theContextServices, theClientSession) {
	// some functions
}

15.13.7Example: Tag-Based Consent Script

 

The following example shows a simple consent script, implementing consent rules based on user permissions and resource security labels.

In this example:

  • Users with the RULE_SUPERUSER permission do not have any consent rules applied
  • Observation resources with the R (restricted) security label will have their Observation.value and Observation.note fields cleared before they are returned to the user.
  • Any resources with the V (very restricted) security label will not be shown to the user at all.
/**
 * If present, this function is called before every request. It serves two
 * primary purposes:
 *
 * 1. It can be used to proactively determine that a request does not need
 *    to have the consent service applied to it. This is good for
 *    performance, since applying the consent service has performance
 *    implications.
 * 2. It can be used to proactively load consent directives, user
 *    information, etc. that will be used in future methods.
 *
 * @param theRequestDetails  Contains details about the request (e.g. the
 *                           FHIR operation being performed, the HTTP method,
 *                           the URL, etc.
 * @param theUserSession     Contains details about the logged in user and
 *                           their session.
 * @param theContextServices Contains various utility methods for accessing
 *                           relevant information about the request, as well
 *                           as providing a response.
 * @param theClientSession   Contains details about the OIDC client and
 *                           their OAuth2 session.
 */
function consentStartOperation(theRequestDetails, theUserSession, theContextServices, theClientSession) {

	// For superusers, we will skip the rest of the consent service entirely
	if (theUserSession != null && theUserSession.hasAuthority('ROLE_SUPERUSER')) {
		theContextServices.authorized();
		return;
	}

	// By default, this request should continue but still be evaluated by the service
	theContextServices.proceed();

}

/**
 * If present, this function is called before every resource that may be returned
 * to the user.
 *
 * @param theRequestDetails  Contains details about the request (e.g. the
 *                           FHIR operation being performed, the HTTP method,
 *                           the URL, etc.
 * @param theUserSession     Contains details about the logged in user and
 *                           their session.
 * @param theContextServices Contains various utility methods for accessing
 *                           relevant information about the request, as well
 *                           as providing a response.
 * @package theResource      The resource that will be accessed.
 * @param theClientSession   Contains details about the OIDC client and
 *                           their OAuth2 session.
 */
function consentCanSeeResource(theRequestDetails, theUserSession, theContextServices, theResource, theClientSession) {

	if (theResource.meta.hasSecurity('https://www.hl7.org/fhir/v3/Confidentiality/cs.html', 'V')) {

		// Any resources with a security label of "V" (very restricted) is rejected,
		// meaning that the client doesn't see them and doesn't even know that they
		// exist
		theContextServices.reject();

	} else if (theResource.meta.hasSecurity('https://www.hl7.org/fhir/v3/Confidentiality/cs.html', 'R')) {

		// Any resources with a security label of "R" (restricted) is allowed to
		// proceed, but will still pass through consentWillSeeResource(...) for
		// filtering
		theContextServices.proceed();

	} else {

		// Any other resources are authorized, meaning that the client will get
		// to see the resources, and consentWillSeeResource(...) is not called. This
		// is good for performance but means that no filtering can take place.
		theContextServices.authorized();
	}

}

/**
 * If present, this function is called before every resource that will be
 * shown to the user.
 *
 * @param theRequestDetails  Contains details about the request (e.g. the
 *                           FHIR operation being performed, the HTTP method,
 *                           the URL, etc.
 * @param theUserSession     Contains details about the logged in user and
 *                           their session.
 * @param theContextServices Contains various utility methods for accessing
 *                           relevant information about the request, as well
 *                           as providing a response.
 * @package theResource      The resource that will be accessed.
 * @param theClientSession   Contains details about the OIDC client and
 *                           their OAuth2 session.
 */
function consentWillSeeResource(theRequestDetails, theUserSession, theContextServices, theResource, theClientSession) {

	// For Observation resources with a confidentiality code of "R" (restricted),
	// we will strip the value and notes from what the user will see
	if (theResource.meta.hasSecurity('https://www.hl7.org/fhir/v3/Confidentiality/cs.html', 'R')) {
		if (theResource.resourceType === 'Observation') {
			theResource.clear('value');
			theResource.clear('note');
		}
	}

	// Let the user see the resource
	theContextServices.proceed();
}

15.13.8Example: Scope-Based Consent Script

 

Another approach to applying consent rules involves using rules based on OpenID Connect Scopes. In this example, a custom scope is defined in the OIDC Client Definition. Unless the application explicitly requests this scope, and the user has approved it, the application will not be able to access SARS-COV-2 Serum/Plasma test results.

The custom scope is named observation_view_covid19. See Custom Scopes Example for more information on how to define this scope and request user approval for it.

/**
 * If present, this function is called before every resource that may be returned
 * to the user.
 *
 * @param theRequestDetails  Contains details about the request (e.g. the
 *                           FHIR operation being performed, the HTTP method,
 *                           the URL, etc.
 * @param theUserSession     Contains details about the logged in user and
 *                           their session.
 * @param theContextServices Contains various utility methods for accessing
 *                           relevant information about the request, as well
 *                           as providing a response.
 * @package theResource      The resource that will be accessed.
 * @param theClientSession   Contains details about the OIDC client and
 *                           their OAuth2 session.
 */
function consentCanSeeResource(theRequestDetails, theUserSession, theContextServices, theResource, theClientSession) {
   
   // In this example, we are applying consent rules only to Observation resources
   if (theResource.resourceType === 'Observation') {

      let resourceSystem = theResource.code.coding[0].system;
      let resourceCode = theResource.code.coding[0].code;

      // Any non-LOINC coded Observations will be allowed
      if (resourceSystem !== 'http://loinc.org') {
         theContextServices.authorized();
         return;
      }

      // The LOINC code 94767-1 corresponds to a SARS-COV-2 Serum/Plasma test. Unless the session
      // has an appropriate scope approved, we'll filter this result.
      if (resourceCode === '94767-1') {
         let approvedScopes = theRequestDetails.approvedScopes;
         if (!approvedScopes.contains('observation_view_covid19')) {
            theContextServices.reject();
            return;
         }
      }

   }

   // Any other resources are authorized, meaning that the client will get
   // to see the resources, and consentWillSeeResource(...) is not called. This
   // is good for performance but means that no filtering can take place.
   theContextServices.authorized();

}