RequirementsRequirements
See BRIDGE-2485, BRIDGE-647, BRIDGE-769, BRIDGE-453, BRIDGE-2382, BRIDGE-2383
- The ability to select templates (like the email address verification email) by criteria such as the user's language, or the version of the app;
- The ability to clearly specify the template that should be used if the criteria filtering is ambiguous (unlike app configurations where we just take the first one);
- The ability to audit changes to templates (like study consents, which create immutable revisions... just record who created the revision);
- The possibility later to create custom templates that can be used with scheduling or notifications (right now, you have to manage these outside the system and can't schedule anything);
- The possibility later to add additional documents (the privacy document has long been considered, because it's required to submit an app to the App Store);
- Should replace the templates on the study object; could conceivably replace the study consent documents as well;
...
- Custom template implementation (just ensure the design can accommodate it);
- Additional document implementation (just ensure the design can accommodate it);
- Document inclusions or attachments (e.g. images for HTML documents... unless we really want this);
- Variable resolution (that's part of using the template);
- Publishing documents (that's part of the consent system).
Template Model
You can now add any number of templates for a specific type of template, differentiated by their guids, and selectable using Criteria.
...
CREATE TABLE `Template` (
`guid` VARCHAR(60) NOT NULL,
`studyId` VARCHAR(255) NOT NULL,
`type` ENUM('EMAIL_ACCOUNT_EXISTS', 'ACCOUNT_EXISTS_SMS', 'EMAIL_APP_INSTALL_LINK', 'APPEMAIL_INSTALLRESET_LINK_SMSPASSWORD',
'EMAIL_SIGN_IN',
'PHONE_SIGN_IN_SMS', 'RESET_PASSWORD', 'RESET_PASSWORD_SMS', 'SIGNED_CONSENT', 'SIGNED_CONSENT_SMS',
'VERIFY_EMAIL', 'VERIFY_PHONE_SMS'EMAIL_SIGNED_CONSENT', 'EMAIL_VERIFY_EMAIL', 'SMS_ACCOUNT_EXISTS',
) NOT NULL,
'SMS_APP_INSTALL_LINK', 'SMS_PHONE_SIGN_IN', 'SMS_RESET_PASSWORD', 'SMS_SIGNED_CONSENT',
'SMS_VERIFY_PHONE' `name` VARCHAR(255) NULL,
`criteriaKey` VARCHAR(255) NULL`description` TEXT,
`createdOn` BIGINT UNSIGNED NULL,
`modifiedOn` BIGINT UNSIGNED NULL,
`publishedCreatedOn` BIGINT UNSIGNED NULL,
`deleted` BOOLEAN NOT NULL DEFAULT FALSE,
`version` INT UNSIGNED NOT NULL, /* optimistic locking */
PRIMARY KEY (`studyId`, `guid`),
INDEX `type_set_idx` (`studyId`, `type`)
) CHARACTER SET utf8 COLLATE utf8_unicode_ci;
CREATE TABLE `TemplateRevision` (
`studyId` VARCHAR(255) NOT NULL,
`guid` `templateGuid` VARCHAR(60) NOT NULL,
`createdOn` BIGINT UNSIGNED NULL,
`createdBy` VARCHAR(255) NOT NULL,
`storagePath` VARCHAR(255) NOT NULL,
`subject` VARCHAR(255) NULL,
`mimeType` ENUM('HTML', 'TEXT') NULL,
PRIMARY_KEY (`templateGuid`, `createdOn`)
CONSTRAINT `Templates-Guid-Constraint`
FOREIGN KEY (`templateGuid`) REFERENCES `Templates` (`guid`) ON DELETE CASCADE) CHARACTER SET utf8 COLLATE utf8_unicode_ci;
...
GET /v3/templates?type=<type>
get documents of type (type parameter is probably required)POST /v3/templates
create new document with a single default revision with default content. Pre-fill the first revision to help people get started. GET /v3/templates/<guid>
get specific documenttemplatePOST /v3/templates/<guid>
update specific document template (not revisioned, but optimistically locked)DELETE /v3/templates/<guid>?physical=true
delete specific document template (logical or physical); probably just leaves revisions TemplateRevision content on S3GET /v3/templates/<guid>/revisions
get the revisions for a document template (createdOn DESC) for history of editsPOST /v3/templates/<guid>/revisions
create a new revision of the given document template (cannot update an existing revision)GET /v3/templates/<guid>/revisions/<createdOn>
get a specific revision of a documenttemplatePOST /v3/templates/<guid>/revisions/<createdOn>/publish
publish this revision as the revision to use when the document template is selected
Notes
Selection
1) get all the documents of the required type and filter using criteria (note that this might be challenging, we'll have to add criteria filtering to a number of places...we might want to move Criteria and CriteriaContext into the RequestContext ThreadLocal);
2) if 1 and only one is filtered, return published revision of that document;
3) otherwise return the published revision for the default template specified for that type in the Study object;
We should also move the signature block into the consent template so it can be localized... although I don't think there's been a single study that has sent out a consent document in the last year or two (or three?).
TemplateService
Sorry the formatting here is difficult to read, due to Jira.
public interface TemplateService {
/**
* Given a criteria context and a template type, return all the templates that match. If one is found through a
...
* configuration.
*/
TemplateService Template getTemplateForUser(CriteriaContext context, TemplateType type);
/** * Get all the templates for a given type. (I am assuming this will not need to be paged). */
List<TemplateService> List<Template> getTemplatesForType(StudyIdentifier studyId, TemplateType type, boolean includeDeleted);
/** * Get a specific template. */
TemplateService Template getTemplate(StudyIdentifier studyId, String guid);
/** * Create a new template. */
GuidVersionHolder createTemplate(StudyIdentifier studyId, TemplateService template);
/**
* Update a template. You can delete it logically, and change the published revision of the associated document , and you
* can change the default template. If you set this template as the default, it should unset the existing default
* template firstwith this call as well.
*/
GuidVersionHolder updateTemplate(StudyIdentifier studyId, TemplateService template);
/** * Delete Mark a template as deleted. */
void deleteTemplate(StudyIdentifier studyId, String guid);
/** * Physically delete the template (probably leaving the revisions). and all its revisions. */
void deleteTemplatePermanently(StudyIdentifier studyId, String guid);
/** * Get a page of revisions . Currently believing this is DDB. from a SQL-type data store. These would not load the document contents from S3 or would not load the
* relevant column if we end up storing the body content in SQL.
*/
ForwardCursorPagedResourceList<TemplateRevision> PagedResourceList<TemplateRevision> getTemplateRevisions(StudyIdentifier studyId, String guid,
String offsetKey int offsetBy, int pageSize);
/** * Get a specific revision. */
TemplateRevision getTemplateRevision(StudyIdentifier studyId, String guid, long createdOn);
/** * Create a new revision (this fails should fail if the createdOn timestamp for a given studyId and GUID already exists). */
CreatedOnHolder createTemplateRevision(StudyIdentifier studyId, TemplateRevision templateRevision);
}
Migration
Can be done in four deployments:
1) Add the new template system in parallel. When reading a template from the existing study object, defer first to the existing template system and use that if a template exists, falling back to the study object template. When writing templates to the study, write to both places, setting up a default template in the templates system if necessary.
2) Create templates for each study from the study object templates.
3) Remove migration bridges to the study object and use only the template system.
4) Delete the fields in the study object itself and remove the fields in the study object schema.