Custom Diagnostics
In addition to the standard Diagnostics it is possible to define Custom Diagnostics.
Custom Diagnostics can inspect Draft and Custom Objects.
A Custom Diagnostic has the following properties:
short_name: must be unique within the Rave URL. Can be up to 20 characters in length. Short names comprising 4 digits (0000 - 9999) are reserved for built-in diagnostics. It may be useful to use a standard prefix, for example 'C001', 'C002', etc.
name: must be unique within the Rave URL
description
object types: the object types which the custom diagnostic will inspect
active: if active is False the Custom Diagnostic will not be displayed in Project Manage Diagnostics and cannot be run against a Draft.
code
Copying Custom Diagnostics
Custom Diagnostics can be copied from one URL into another. If a Custom Diagnostic has object types which do not exist in the target URL then a warning will be displayed. The Custom Diagnostic will be copied but might not function as expected. There is no check that any Custom Properties referenced in the Custom Diagnostic code exist in the target URL.
Custom Diagnostic Code
Custom Diagnostic code is written using a subset of the Python programming language called RestrictedPython. For security, RestrictedPython limits the code which can be used; for example it is not possible to use the 'import', 'exec' or 'eval' statements.
A 'diagnostic' object is passed into the Custom Diagnostic Code. The diagnostic object has the following methods which can be called from the Custom Diagnostic code:
objects.get(plural_name)
returns a list of DiagnosticObjects which be can inspected for the given plural_name (the plural name of the Object Definition). See below for the definition of a DiagnosticObject.
debug(message)
displays the string when testing the Custom Diagnostic. When the Custom Diagnostic has been activated and is run for Drafts no debug messages are displayed.
Parameters:
message (str) – (required). the message to be displayed
report_draft_finding(message)
creates a diagnostic finding for the Draft
Parameters:
message (str) – (required). the finding text
report_object_finding(object_with_finding, property_name, message)
Parameters:
message (DiagnosticObject) – (required). the object associated with the finding
property_name (str) – (required). the name of the property associated with the finding
message (str) – (required). the finding text
A DiagnosticObject has the following property:
identifier
returns the identifier of the object
A DiagnosticObject has the following method:
property(property_name)
returns the value associated with the property of the object
Parameters:
property_name (str) – (required). the name of the DiagnosticObject property
Diagnostic Object Properties
The list of properties available for the selected object types will be displayed in the "Properties" tab.
Custom Objects will have all defined Custom Properties; the data type will always be 'string'.
Draft Objects will have built-in Medidata Rave properties and any Custom Properties which have been created for that type of Draft Object.
For example, if "Matrices" are selected for inclusion in the Custom Diagnostic the following built-in properties will be available:
- Matrices
id (int)
MatrixName (string)
OID (string)
Addable (bool)
Maximum (int)
The "id" property is a unique integer identifying each Matrix in TrialGrid.
"Matrix Folder Forms" objects will also be included, with the following properties:
- Matrix Folder Forms
id (int)
Matrix_id (int)
Folder_id (int)
Form_id (int)
The "id" property is a unique integer identifying each Matrix Folder Form in TrialGrid.
Properties ending with "_id" are "foreign key" references to another object. "Matrix Folder Form" objects have a "Matrix_id" property which references the "Matrix".
The related object can be looked up using this pattern:
matrices_by_id = {matrix.property("id") : matrix for matrix in diagnostic.objects.get('Matrices')}
for mff in diagnostic.objects.get("Matrix Folder Forms"):
matrix = matrices_by_id[mff.property("Matrix_id")]
Examples
# Report a draft finding if the 'ID' property of the 'Specifications' objects is repeated in more than one object
spec_ids = []
all_spec_ids = []
for spec in diagnostic.objects.get('Specifications'):
spec_id = spec.property('ID')
if spec_id not in spec_ids:
spec_ids.append(spec_id)
all_spec_ids.append(spec_id)
for spec_id in spec_ids:
spec_count = all_spec_ids.count(spec_id)
if spec_count > 1:
msg = f"Specification ID {spec_id} is repeated {spec_count} times"
diagnostic.report_draft_finding(msg)
# Report an object finding if the length of the 'DraftFieldName' property of a 'Fields' object is more than 20
for field in diagnostic.objects.get('Fields'):
if len(field.property('DraftFieldName')) > 20:
diagnostic.report_object_finding(field, "DraftFieldName", "is longer than 20 characters")
# Report an object finding if the 'Description' property of a 'Specifications' object contains double-spacing
for spec in diagnostic.objects.get('Specifications'):
if " " in spec.property('Description'):
diagnostic.report_object_finding(spec, "Description", "has double-spacing")