coreBOS 5.5 has brought a new tool to our powerful business operating system : coreBOS Updater.
Since we have a very stable and tested code base, and thanks to the powerful code administration and distribution system that GitHub gives us, it just seems natural to put your coreBOS install under code versioning with git (be that private or open) and point a remote to the public, open-source coreBOS project.
Once you have that setup, updating your code with our (constant) changes is fairly easy, as git carries all the heavy weight during merge, leaving us to fix the minor details.
So with the approach above and the fact that the majority of changes we make are also tested and stable, you really don't have to wait for us to release an official download to get the new features, you just update your code and apply the database changes the new code may require.
Enter coreBOS Updater.
coreBOS Updater is a tool that will control the state of our database with respect to the code and will be able to automatically apply the pending changes that are needed. This way the upgrade process is converted into these steps:
You are finished! That easy! coreBOS updater has taken care of all the pending database changes and your code and database are in sync and ready for you to start working.
Going forward all future releases of coreBOS will follow this procedure to apply database changes for the upgrades.
If you have a look at the module you will see that each changeset is created as a record in the module. You can see the details of the change and also undo some of them, all those which are not pure system updates. The system type changesets are mandatory for the application to work correctly.
From a developer's point of view, the coreBOS Updater tool is a clean way of recording database changes in code. This permits us to control (to some extent) the set of database changes that are needed in our version control system.
Since we are actually programming the changes in files, we then version those changes along with the rest of file changes that define the new feature or bug fix and it all gets registered in the versioning system.
To help us prepare the database changes coreBOS updater is going to take care of all the dirty work it can for us, so we basically just have to write the changes that are needed.
Create a new file in modules/cbupdater/cbupdates/yourproject.xml and add your definition to the list of changes.
The XML changeSet element contains various entries of which only two are mandatory, but all recommended:
author | optional | Name of the person or company adding this changeset |
description | optional | Explanation of the goals of this changeset |
filename | mandatory | Name of the file which contains the changes. This file can be anywhere on the file system as long as it is accessible by coreBOS. We propose and recommend that these changesets be saved all together in build/changeSets directory |
classname | mandatory | Name of the class that implements the changes (explained next) |
systemupdate | optional | True if the changeset is mandatory, false if it can be undone |
perspective | optional | True if the changeset is a perspective (see below) |
continuous | optional | True if the changeset must/can be executed always. Certain changeset are very general and can be (or must be) applied many times. |
An example:
<changeSet>
<author>joebordes</author>
<description>Custom field support on FAQ module</description>
<filename>build/changeSets/cffaq.php</filename>
<classname>cffaq</classname>
<systemupdate>true</systemupdate>
<perspective>false</perspective>
<continuous>false</continuous>
</changeSet>
The file modules/cbupdater/cbupdater.xsd defines the official schema accepted for the XML changeset file and it is used by coreBOS Updater to validate the XML format.
You can find a whole bunch of real examples of different types in the build/changeSets directory.
Really!!: Test, validate and version the change.
coreBOS Updater changes are implemented by extending the cbupdaterWorker class. This class gives us all the functionality to plug into the coreBOS system and do any type of action that may be required.
Basically, the only method we need to override is the applyChange() method. This method implements the changes that represent the changeset. If your change can be undone then you must also override the undoChange() method and add the necessary code to undo the change.
The rest of the methods that the cbupdaterWorker class implements are helper methods and some auxiliar methods for functionality:
__construct | will load the changeset definition from the database, if it isn't found there an error will be shown and the process will stop. If all goes well this method starts the output formatting on screen |
isApplied | boolean method which tells us the applied state as recorded in the database |
isBlocked | boolean method which tells us if the change set is blocked. If a changeset is blocked it cannot be applied nor undone. It will automatically be ignored |
isContinuous | boolean method which tells us if the changeset is of type Continuous. Continuous changesets can be applied many times without breaking the system and will be picked up when launching an "Apply All" |
isSystemUpdate | boolean method which tells us if the current changeset is a system changeset |
hasError | boolean method which indicates if there has been an error executing the changeset |
markApplied($stoponerror=true) | helper method which does the work of cleaning up after the changeset has been applied. This must be executed at the end of a correct execution of applyChange() |
markUndone($stoponerror=true) | helper method which does the work of cleaning up after the changeset has been undone. This must be executed at the end of a correct execution of undoChange() |
sendMsg($msg) | put the given successful message on screen following the coreBOS Updater on-screen formatting |
sendMsgError($msg) | put the given error message on screen following the coreBOS Updater on-screen formatting |
finishExecution | closes the on screen formatting and outputs the results of the execution |
ExecuteQuery($query,$params=array()) | helper method that permits us to send SQL commands to the database. This method shows the result on-screen and counts updates the SQL command succees/failure counters. With this method, it is not necessary to execute direct SQL commands against the global $adb object, although you can if you need/want to |
deleteWorkflow($wfid) | helper method that permits us to delete workflows |
installManifestModule($module) | helper method that simplifies the task of installing a new module |
isModuleInstalled($module) | helper method that looks in the vtiger_tab table to see if a given module is installed, be it active or not |
deleteAllPicklistValues($tableName, $moduleName) | will eliminate all the values in the given picklist |
deletePicklistValues($values, $tableName, $moduleName) | will eliminate the given values from the given picklist |
setQuickCreateFields($moduleName, $qcfields) | will set the given fields to appear in the quick create form |
massCreateFields($fieldLayout) | an easy structure to mass create blocks and fields in modules |
massHideFields($fieldLayout) | an array of fields to hide |
massDeleteFields($fieldLayout) | an array of fields to delete |
massMoveFieldsToBlock($fieldLayout, $newblock) | move fields around |
orderFieldsInBlocks($fieldLayout) | order fields |
convertTextFieldToPicklist($fieldname, $module, $multiple = false) | convert the given text field into a picklist. it will deduplicate and set the existing values |
setTooltip($tooltips) | set the tooltip for the given fields |
removeAllMenuEntries($module) | remove the given module name from all menu entries it appears in |
After reading the above functionality we can see that the applyChange() method template prepares everything so we just have to write our changes:
function applyChange() {
if ($this->isBlocked()) return true;
if ($this->hasError()) $this->sendError();
if ($this->isApplied()) {
$this->sendMsg('Changeset '.get_class($this).' already applied!');
} else {
// do your magic here
$this->sendMsg('Changeset '.get_class($this).' applied!');
$this->markApplied(); // this should not be done if changeset is coninuous
}
$this->finishExecution();
}
First it makes sure there is no error and it isn't blocked, then it checks if it is already applied, if that is ok, then it lets us do whatever we need to do and it ends up marking the changeset done, informing the user and finishing the changeset.
After reading the above functionality we can see that the undoChange() method template prepares everything so we just have to write code to undo our changes:
function undoChange() {
if ($this->isBlocked()) return true;
if ($this->hasError()) $this->sendError();
if ($this->isSystemUpdate()) {
$this->sendMsg('Changeset '.get_class($this).' is a system update, it cannot be undone!');
} else {
if ($this->isApplied()) {
// undo your magic here
$this->sendMsg('Changeset '.get_class($this).' undone!');
$this->markUndone(); // this should not be done if changeset is coninuous
} else {
$this->sendMsg('Changeset '.get_class($this).' not applied!');
}
}
$this->finishExecution();
}
First, it makes sure there is no error and it isn't blocked, then it checks if it is a system update in which case it stops the execution as system changesets cannot be undone. Next, it looks if it is applied because if it isn't it can't be undone. If that is ok, then it lets us do whatever we need to do and it ends up marking the changeset undone, informing the user and finishing the changeset.
You can find a whole bunch of real examples of different types in the build/changeSets directory, but I am going to comment on a few of the interesting ones.
add_workflow_tags.php | this one is interesting for two things: one is to see how a new workflow method is added using the VTTaskType::registerTaskType() method, and the other because it implements its own isApplied() because it looks for the new workflow tasks it wants to create before permitting it to be created |
installcyp.php | interesting to see how a module is installed and deactivated when undone |
PotentialForecastAmount.php | interesting to see how to add new fields and also a workflow associated with the field |
workflow_contactassignedto.php | interesting to see how to add a workflow custom method and check if it can be undoone |
coreBOS updater has a set of scripts that permit loading and applying changesets from the command line:
You can use this script in your composer.json file to apply changesets after loading some modules:
...
},
"scripts": {
"post-install-cmd": [
"tsolucio\\ComposerInstall\\ComposerInstall::postPackageInstall",
"php modules/cbupdater/loadapplychanges.php modules/ConfigEditor/composer.xml"
],
"post-update-cmd": "tsolucio\\ComposerInstall\\ComposerInstall::postPackageUpdate"
}
}
The coreBOS Perspective changeset loader is an extension for distributing coreBOS updater changesets. It gives the developer the infrastructure to simply create the set of changesets that need to be applied and distribute and apply them easily into any existing coreBOS application.
Once defined the .xml change file and created all the changesets you can send this extension to any coreBOS application or include it as part of a perspective, to have the changes applied.
We have created a template project on github which you can use as a starting point for your changes or simply to study how it is done.
Ideally, you will create one of these extensions for your perspective in such a way that it will require first for all the new modules to be installed and then, the last one will be this extension which will apply the final changes to get all the functionality into place.
coreBOS Updater is one fundamental step towards an important concept that we have on our roadmap: Perspectives.
A perspective is a set of code and database changes that literally convert a stock coreBOS system into a verticalization prepared for a clearly defined market segment.
The other big step upon which the perspective concept stands is the code structure and packaging system. Since we have laid out the code in such a way that we do not need to update packages and all the code is simply there, in place, ready to be modified we can easily define a perspective as:
The patch will add the changes that the code needs while the coreBOS updater .xml changeset file and the composer.json file will define the set of database changes and new modules that are needed to convert the raw application into the verticalization we need.
Read this blog post for more information
The majority of changes in the updater will come from the coreBOS project development effort. You will update your code and get new changesets as required. These changeset records will be marked as "Application Changesets" and you will not be able to edit nor delete them.
The other type of changesets are the ones you can create as any other record in the application. These "manual" changesets support a very restricted set of actions but can turn out to be very useful for customizing a coreBOS install for your users in certain cases.
Some of these restrictions are:
The way these changesets work is by defining the action to execute with a JSON object saved in the description field. The other fields are either not used or maintain their normal meaning.
All the JSON objects have a set of common fields:
Next, we document the JSON format for each of the supported actions with some examples.
Will delete ALL the values in a picklist.
{
"name": "del values industry",
"description": "delete all the current values in the industry picklist of Accounts",
"operation": "deleteAllPicklistValues",
"setting": {
"table": "industry",
"module": "Accounts"
}
}
{
"name": "del values building",
"description": "delete all the current values in the building picklist of Products",
"operation": "deleteAllPicklistValues",
"setting": {
"table": "building",
"module": "Products"
}
}
Will delete the given values from a picklist.
{
"name": "del values industry",
"description": "delete some values from the industry picklist of Accounts",
"operation": "deletePicklistValues",
"setting": {
"table": "industry",
"module": "Accounts",
"values": ["Telecommunications", "Banking"]
}
}
{
"name": "del values prodtype",
"description": "delete some values from the prodtype picklist of Products",
"operation": "deletePicklistValues",
"setting": {
"table": "prodtype",
"module": "Products",
"values": ["Sell", "Rent Holiday"]
}
}
Given an array of field names, this method will activate them in Quick Create in the given order. All the other fields will be deactivated.
{
"operation": "setQuickCreateFields",
"setting": {
"fields": ["fieldname1", "fieldname2", "fieldname3"],
"module": "Accounts"
}
}
{
"operation": "setQuickCreateFields",
"setting": {
"fields": ["floor", "internet", "building"],
"module": "Products"
}
}
Given an array of field definitions, this method will create or activate the fields. The layout is an array of Module Name, Block Name, and Field Definition.
{
"operation": "massCreateFields",
"setting":
{
"module":{
"block name or ID":{
"fieldname1": {
"columntype": "decimal(10,3)",
"typeofdata": "NN~O",
"uitype": "1",
"displaytype": "3",
"label": "", // optional, if empty fieldname will be used
"massedit": 0 | 1 // optional, if empty 0 will be set
"mods": [module names], // used if uitype 10
"vals": [picklist values], // used if uitype 15, 16 or 33
},
"fieldname2": {
"columntype": "decimal(10,3)",
"typeofdata": "NN~O",
"uitype": "1",
"displaytype": "3",
"label": "", // optional, if empty fieldname will be used
"massedit": 0 | 1 // optional, if empty 0 will be set
"mods": [module names], // used if uitype 10
"vals": [picklist values], // used if uitype 15, 16 or 33
}
}
}
}
}
{
"operation": "massCreateFields",
"setting":
{
"Vendors": {
"Custom Information": {
"cal": {
"columntype": "int(11)",
"typeofdata": "V~O",
"uitype": "10",
"displaytype": "1",
"label": "Sabi",
"massedit": "1",
"mods":"Products"
},
"volume": {
"columntype": "varchar(100)",
"typeofdata": "V~O",
"uitype": "1",
"displaytype": "1",
"label": "Volume",
"massedit": "1"
}
}
}
}
}
{
"operation": "massCreateFields",
"setting":
{
"Vendors": {
"Custom Information": {
"palm": {
"columntype": "int(11)",
"typeofdata": "V~O",
"uitype": "10",
"displaytype": "1",
"label": "Palm",
"massedit": "1",
"mods":"Products"
},
"pressures": {
"columntype": "varchar(100)",
"typeofdata": "V~O",
"uitype": "1",
"displaytype": "1",
"label": "Pressures",
"massedit": "1"
}
}
},
"Products": {
"Custom Information": {
"uniqueapplicationfieldname": {
"columntype": "int(11)",
"typeofdata": "V~O",
"uitype": "15",
"displaytype": "1",
"label": "Values",
"massedit": "1",
"vals":["Accounts","Bank","VICOBA"]
},
"volume": {
"columntype": "varchar(100)",
"typeofdata": "V~O",
"uitype": "1",
"displaytype": "1",
"label": "Volume",
"massedit": "1"
}
}
}
}
}
Given an array of field names this method will hide the fields.
{
"operation": "massHideFields",
"setting": {
"fields": ["fieldname1", "fieldname2", "fieldname3"],
"module": "Accounts"
}
}
{
"operation": "massHideFields",
"setting": {
"fields": ["bill_street", "ship_street", "ownership"],
"module": "Accounts"
}
}
Given an array of field names this method will delete the fields.
{
"operation": "massDeleteFields",
"setting": {
"fields": ["fieldname1", "fieldname2", "fieldname3"],
"module": "Accounts"
}
}
{
"operation": "massDeleteFields",
"setting": {
"fields": ["website", "email1", "bill_city"],
"module": "Accounts"
}
}
Given an array of field names this method will move the fields to the specified Block.
{
"operation": "massMoveFieldsToBlock",
"setting": {
"fields": ["fieldname1", "fieldname2", "fieldname3"],
"module": "Accounts",
"block": "label or ID of the destination block"
}
}
{
"operation": "massMoveFieldsToBlock",
"setting": {
"fields": ["city", "street"],
"module": "Vendors",
"block": "LBL_VENDOR_INFORMATION"
}
}
Given an array of blocks and field names, this method will sort the fields in the given order.
{
"operation": "orderFieldsInBlocks",
"setting":
{
"Module":{
"block label or ID": ["fieldname1", "fieldname2", "fieldname3"]
}
}
}
{
"operation": "orderFieldsInBlocks",
"setting":
{
"Vendors":{
"LBL_VENDOR_INFORMATION": ["vendorname", "vendor_no","glacct"]
}
}
}
{
"operation": "orderFieldsInBlocks",
"setting":
{
"Vendors":{
"LBL_VENDOR_INFORMATION": ["vendor_no", "vendorname","glacct"]
},
"Products":{
"LBL_PRODUCT_INFORMATION": ["product_no", "productname","expiry_date"]
}
}
}
Convert a text field into a picklist. It will fill the picklist with the distinct values in the text field. The multiple parameter indicates if the new picklist with be multiple (true) or simple (false)
{
"operation": "convertTextFieldToPicklist",
"setting": {
"module": "Accounts",
"field": "industry",
"multiple": false
}
}
{
"operation": "convertTextFieldToPicklist",
"setting": {
"module": "Invoice",
"field": "salescommission",
"multiple": false
}
}
Set the tooltip configuration for fields. The layout is an array of Module Name, hover fields and Field Names
{
"operation": "setTooltip",
"setting": [
{
"module": "Accounts",
"hoverfield": "fieldname that triggers the tooltip",
"fields2show": ["fieldname1", "fieldname2", "fieldname3"]
}
]
}
{
"operation": "setTooltip",
"setting": [
{
"module": "Vendors",
"hoverfield": "vendorname",
"fields2show": ["state", "street"]
}
]
}
Will delete all menu entries of a given module.
{
"name": "del Accounts from menu",
"description": "delete all Accounts module entries in the menu",
"operation": "removeAllMenuEntries",
"setting": {
"module": "Accounts"
}
}
{
"name": "del invoice module from menu",
"description": "delete all invoice module entries in the menu",
"operation": "removeAllMenuEntries",
"setting": {
"module": "Invoice"
}
}