I’m beginning to think I should change the tagline for my blog from weekly to monthly. One of our Project Managers has been giving me a hard time about my blog getting a bit stale. I told her I was waiting for her to create a post to her (currently non-existent) blog and she only laughed. Where’s the equity in that?!? ;-)
Life continues to be extremely busy here at Click (the real reason it’s been so long since my last post) but good things are happening. Extranet 5.6 is now officially released (Yay!) and exciting work is being done in all groups. My work on our new Animal Operations solution is progressing and I’m excited to see it all come together later this year. Within Professional Services, we’ve been working to drive consistency in our approach to both projects and development, including a major revision to our naming standards and coding guidelines. I hope to make that available to customers very soon so you have the opportunity to adopt the same standards as we have.
Today, I want to talk about an approach to solving the thorny problem of being able to select CDT entities that were previously created as part of the same project. Now I won’t be the first to solve this problem, but the solutions I’ve heard about involve the creation of custom layouts using Site Designer or clever hacks to the standard view controls. Neither of those approaches appealed to me so I set out to come up with an approach that could be done through simple configuration.
Selection versus Data Entry Custom Data Types
Before I go into the technique, it’s important to understand why this is a problem to begin with. To do that, one must understand the difference between Selection and Data Entry custom data types. Selection types serve the purpose of providing a data source that serves as a list of choices. Data Entry custom data types serve as an extension to something else and is not allowed to be referenced by projects or custom data types other than the type it is extending. The distinction is important to the base Framework so that data lifetime, security, and presentation can be effectively managed.
- Data Lifetime
By knowing that an entity belongs to a Data Entry CDT, the base application knows that it is owned by only one project, person or organization thus, if the project is deleted or a reference to the entity is removed, that entity can also be deleted. Selection CDT Entities on the other hand are intended to be referenced by multiple entities so do not get deleted when all references are removed. - Security
Since the a Data Entry entity belongs to a single project, it is subject to the same security rules as the project itself. Selection CDT Entities have no such allegiance to a single project and can be referred to by many projects, or none at all. Their purpose is to serve as the contents of selection lists so are visible to all users. - Presentation
How references or sets of CDT entities are presented differs depending on the CDT being Selection or Data Entry. A Selection CDT entity can only be selected by the end user, never created, modified, or deleted. Data Entry CDT entities are intended to serve as an extension of the data comprising a project, person, or organization so, by their very nature, can be created, edited, and deleted.
So what happens when you need both characteristics in a single set of data?
Implementing a “Hybrid CDT”
You won’t find the term Hybrid CDT anywhere in the product or product documentation. That’s because i just made it up ;-) In fact, the term is a bit misleading in that it makes you think there is a single CDT when, as you’ll see, there are really two. But, conceptually the two types serve a single purpose . I’d gladly consider other name suggestions but, for now, I’m going to use the term out of sheer convenience.
The goal is to define a “type” that can be used for both data entry and selection.
Step 1: The Basic Setup
We’ll need to create two Custom Data Types. One that is a Selection CDT, and the other is a Data Entry CDT.
The selection CDT will define all the attributes you need to hold all the specific detail that needs to be captured. We’ll keep this example simple by only adding a single string attribute called “name”, but any number of attributes could be added.
MyHybridCustomDataTypeSE
- name String
When creating this type, be careful to not select the option to source control the data (entities) as it is our goal to allow the end user to create the entities so that they can be used in selection lists later.
Next, we’ll define a Data Entry Custom Data Type. The only attribute on this type will be a reference to the selection type we just created.
MyHybridCustomDataTypeDE
- data (MyHybridCustomDataTypeSE)
By creating the types in this way, we’ve effectively created a Selection CDT that is “wrapped” by a Data Entry CDT.
Step 2: Creating the Data Entry Form
Now that types are defined, the next step is to create the data entry view on the data entry CDT. To make this all work we want to expose the attributes on the selection CDT into a view on the data entry CDT. With our simple data model, this means the view will only have a single field on it:
“customAttributes.data.customAttributes.name”
Of course, in the view editor, you’ll see the attribute in the tree as
data
-> name
It’s not critical to define a view on the selection CDT but you could if you wanted to tailor the appearance of the Chooser for this type.
Step 3: Insuring proper data lifetime
As mentioned earlier, a Selection CDT entity will not automatically be deleted when there are no other entities referring to it. In most situations, this is exactly what we want because the data is intended to be available for selection across any number of project types and custom data types. In the Hybrid CDT example, however, our intent is that this data is specific to the entity that refers to it and it should be deleted when the referring entity is deleted. There are a few ways to do this, but only one way to do it without having to write any code.
- Launch Entity Manager and connect to your development store
- Open the details of the selection CDT you just created
- At the bottom of the details display you will see a checkbox that allows you to indicate if the extent is “weak”. In this case we want it to be so check the box
- Save your changes
Following these steps will cause any unreferenced entities to be deleted the next time Garbage Collection is run.
Step 4: Putting the “Hybrid CDT” to use
Believe it or not, we now have a “Hybrid CDT”. So now we get to put it to use on a Project. To add color to this, we’ll work through a simple example. We’ll define a project type to manage trivial information about your children. Let’s call it MyChildren. It will allow a user to specify the names and ages of their children and then answer a couple of simple questions requiring them to select one of the kids in response. So, here’s the setup:
MyChildSE (the selection CDT representing a single child)
- name (string)
- age (integer)
MyChildDE (the data entry CDT representing a single child)
- child (a reference to the MyChildSE entity containing information about the child)
MyChildren (the project type)
- children (Set of MyChildDE)
- childMostLikelyToBeARockStar (entity of MyChildSE)
- childrenWithAllergies (Set of MyChildSE)
It’s pretty simple and any real world example will likely be far more complex. The complexity will be in the volume of data, not the structure so hopefully this will clearly demonstrate the technique. I’m also going to overlook that case where there is only one child in which case the questions don’t make much sense (such is the price of a contrived simple example).
Now that the types are defined, we can add the fields to views on the project which can then be used in a SmartForm. The first view will be constructed to allow the user to specify their children. All this requires is to add the children attribute and configure the view control just like you do for any other data entry CDT.
The second view will be used to prompt the user for the following information:
- Which child is most likely to become a Rock Star?
- This will be done by adding the childMostLikelyToBeARockStar attribute to the form and configuring the view control like you would any other reference to a Selection CDT.
- Which children have allergies?
- This will be done by adding the childrenWithAllergies attribute to the form and configuring the view control like you would any other set of Selection CDT entities.
Simple, right? There’s really only one step left and that is to make sure that the list of children presented for selection are just those added on the current project. If this isn’t done, then any child created on any project would be presented and that wouldn’t make much sense. This is accomplished through the use of a new feature in Extranet 5.6: The “Data Bind Override” feature of the View Controls for the selection data type reference and set. You will add this script to the controls for both questions by clicking on the script icon following “Data Bind Override” in the view control properties form
// Return a set of MyChildSE entities that are derived from the
// set of MyChildDE entities in the children set
var browseDataSet = rootEntity.getQualifiedAttribute("customAttributes.children");
if (browseDataSet != null) {
browseDataSet = browseDataSet.dereference("customAttributes.child");
}
These two views can either be used individually or as part of a SmartForm. You will just need to make sure that the children are added before the additional questions are asked or the choice list will be empty.
All Done! We have now defined a Hybrid CDT and put it to use. If you found this to be of value in your own work, please drop me a note. I’d love to hear how it worked for you.
Cheers!
UPDATE: I've posted a follow-up to this post to address what happens when data is deleted by the user: Implementing a Hybrid CDT – UPDATE
UPDATE 2: Another follow-up post to address what happens when the Hybrid CDT is cloned: Hybrid CDTs and Cloning
This was a great example for data types. Don't stop blogging, I don't see anything for October yet and I'm getting anxious!
ReplyDeleteThanks. I'm glad you liked it. This technique has now been put into practice in two different projects and is working extremely well.
ReplyDeleteAs for my next post, it's coming soon. Stay tuned!
works like a charm!
ReplyDeleteThank you Tom!
I'm glad. If you get a chance, drop me a note. I'd love to know what problem you solved with this technique. Thanks!
ReplyDeletehey Tom, I had to add a condition to null if no entities existed since we don't require input, and if nothing exists, it still displays the entities for the other records..
ReplyDelete// Return a set of MyChildSE entities that are derived from the
// set of MyChildDE entities in the children set
var browseDataSet = rootEntity.getQualifiedAttribute("customAttributes.children");
if (browseDataSet != null) {
browseDataSet = browseDataSet.dereference("customAttributes.child");
}
else {
browseDataSet = null;
}
Thanks Deborah! That's a great addition.
ReplyDeleteTom,
ReplyDeleteWe have implemented several "Hybrid CDT" in our IACUC module. The problem is that the data entered through data entry CDT is garbage collected periodically. We did un-checked "Type extent is a weak reference holder" in EM for all related data enter cdt and selection cdt, but it seems not prevent the data be deleted through GC. Could you please let us know how to set up correctly to avoid the data be lost? Thanks.
If the entity is registered and the extend is not weak, Garbage Collection will not identify it as garbage. This leaves me wondering is the entities get unregistered somehow.
ReplyDeleteIf the data entry entities are members of a set referenced by a project, then when a set member is removed through interactive use (via a view), Portal will unregister the data entry CDT. This is correct and expected behavior with how Portal manages data entry CDTs. Is it possible that the entities that you have observed to be deleted by GC were previously unregistered through interactive use?
If that isn't the reason, I encourage you to reach out to your Click Services contact so this can be investigated further.
Cheers!
Tom, thank you very much for your response.
ReplyDeleteI am not sure what we did here will possible unregistered data entries. Here is an example what happened.
In a IACUC protocol, there is a view A to let user to enter agents (agentDataEntery CDT). There is pop up view to select species and their agents from a dropdown list. After data entered, here is how they look like on main view table:
Mouse AgentA
Rat AgentB
Fish AgentC
Next, in a tumor procedure, there is question ask if an agent will be used for this procedure, if yes, the agents will be listed as checkbox (selection cdt, agentAssociatedStudy)
Mouse AgentA
Rat AgentB
Fish AgentC
The procedure is a set, after data entered, it looks like this,
First procedure Mouse AgentA
Second procedure Rat AgentB
During weekend GC, all three agents were deleted from data entry cdt on the view A, in the procedures, all selected agants are still listed in each procedure, but those un-selected were gone. This is in processed testing protocol. Strange thing is that not all testing protocols lost their data.
Can you see anything we did wrong from this example to cause data unregistered?
From the description it's not clear what might be going on. I encourage you to reach out to your Click Services contact so this can be investigated further
ReplyDelete