Skip to end of metadata
Go to start of metadata

You are viewing an old version of this page. View the current version.

Compare with Current View Page History

« Previous Version 13 Next »

In adding Mobile Toolbox to Bridge, we’re encountering more complex authorization. We have roles like an organization administrator, who can create users who are members of their organization (but who cannot see any other kind of accounts, including study participants in their organization’s studies), and study designers, who can edit some of the things that a developer can edit…but only in the context of the studies they have access to.

Right now these checks are spread out:

  1. Controllers usually check for roles and consent;

  2. AuthUtils methods are called in the services to check the relationship of the caller to entities like studies and organizations.

This is getting messy. I think we could use a more robust alternative to implement this. But first, here are the authorization checks we have implemented or want to implement in the MTB timeframe (described in terms of access to objects in the REST API, rather than through the several endpoints that are needed to expose each object in the API itself, and skipping participant-facing APIs):

Authorization Rules

Object

Assoc(1)

Role

Permissions(2)

AccountSummary

Study (“participant”)

researcher

read

Organization (“member”)

org admin

read

App

all

read

dev, admin

update

superadmin

create, delete

AppConfig

public

read (filtered)

dev

create, read, write, delete

AppConfigElement

dev

create, read, write, delete

Assessment

AssessmentConfig

Enrollment

EnrollmentDetail

ExternalResource

FileMetadata/Revision

HealthDataRecord(Ex3)

MasterScheduleConfig

NotificationMessage

NotificationRegistration

NotificationTopic

OAuthProvider

Organization

RecordExportStatusRequest

ReportData

ReportIndex

RequestInfo

SchedulePlan

SmsTemplate

Study

StudyConsent

StudyParticipant

Subpopulation

Survey

Tag

Template/TemplateRevision

Upload

UploadSchema

(1) = association to another model object.

(2) C = create/write, R = read (list or detail object), U = update/write/delete logically, D = delete (physically)

Here are objects from the v2 domain model that have been designed far enough to think about permissions:

Object

Assoc

Role

Permissions

Note

Protocol/Study Arms

Schedule

Session

Spring Security

This is the most complex option available, but we could use the method-based security via annotations. Our service methods could declare the security using complex expression rules.

Pros

  1. It’s a standardized thing so other developers should have an easier time working with it;

  2. Ultimately, security would be expressed in annotations, which seems nice;

  3. One step closer to implementing a major change to something like storing access rights in a database (so far we haven’t seen anything in Bridge that requires that level of authorization, but)

Cons

  1. Our existing authentication and authorization have all been written from scratch, which means we’d need to overwrite all the core Spring Security classes before any of this would work;

  2. The result would take time and involve learning a lot more about Spring Security;

  3. Security exceptions would appear in places where we are now able to return InvalidEntityExceptions and the like. For example empty/null/imaginary values are going to throw security exceptions. (I’m fine with this).

Implement a DSL

Nothing Spring does is that hard to duplicate with a DSL that uses our existing RequestContext. I think it makes sense to call these in the services, where multiple endpoints might create multiple conditions under which a call is acceptable. It could look something like this:

private static final AuthEvaluator IS_ORG_ADMIN = AuthUtils.orgMember().inRole(ORG_ADMIN);
    
// If we pass arguments in when we check, these are threadsafe and composable
private static final AuthEvaluator IS_ORG_ADMIN_OR_ADMIN = IS_ORG_ADMIN.or().inAnyRole(ADMIN, SUPERADMIN);
    
public void mustBeAnOrgAdmin(String appId, String orgId) {
    IS_ORG_ADMIN.checkAndThrow("orgId", orgId);
        
    // do something
}
    
public void mustBeSomeKindOfAdmin(String appId, String orgId) {
    IS_ORG_ADMIN_OR_ADMIN.checkAndThrow("orgId", orgId);

    // do something
}

Pros:

  1. Easier to implement and understand at this point, when compared with overriding Spring Security’s implementation classes

  2. Arguably, easier to understand because it’ll only contain what it necessary for our application (as opposed to Spring which is always more complicated because it can handle anything, including future requirements).

Cons:

  1. More custom code for new developers to learn (assuming they already know Spring security which is more portable for them anyway);

  2. We’re bearing down further on a custom implementation making that much harder later if we decide to abandon it for anything else. If we’re going to switch, better to switch sooner.

Recommit to implementing authorization in the controllers

Arguably we can continue to use authorization in the controllers if we just create more endpoints that are tailored to our specific authorization scenarios.

Take the example of AuthUtils.checkSelfStudyResearcherOrAdmin check for enrollments:

// self
POST /v5/studies/enrollments/self

// study researcher or admin
POST /v3/studies/enrollments

Another example are org administrators who can list, read, create, and delete administrative accounts in their organizations:

// Instead of the /v3/participants APIs:

GET /v1/organizations/{orgId}/members
POST /v1/organizations/{orgId}/members
GET /v1/organizations/{orgId}/members/{userId}
DELETE /v1/organizations/{orgId}/members/{userId}

Pros:

  1. Consistent with what we’ve done and thus the least code churn;

  2. Possibly easier for external users to understand our APIs;

  3. Possibly easier to secure down the road with Spring Security (using URL-based security rather than method-based security);

Cons

  1. Honestly I was trying to eliminate keywords like “self” that make more endpoints…that’s more to add to the SDK, more to test, more to document, etc. In the second example, all those APIs are redundant with the participant APIs except for the security differences. Putting all the security within the system is easier;

  2. Some security checks might get tedious. You might still want a DSL class even if doing this work in the controller. The delete user is a good example:

    boolean orgAdminDelete = context.isInRole(ORG_ADMIN) && !account.getRoles().isEmpty() && 
            context.getCallerOrgMembership().equals(account.getOrgMembership());
    boolean testDelete = AuthUtils.isSelfResearcherOrAdmin(context.getCallerUserId()) &&
                account.getDataGroups().contains(TEST_USER_GROUP);
    boolean adminDelete = context.isInRole(ADMIN);
    
    if (!orgAdminDelete && !testDelete && !adminDelete) {
        throw new UnauthorizedException();
    }
  • No labels