- Created by Jonathan Muse , last modified on Sept 13, 2023
- Mentions
- 0 Associations
You are viewing an old version of this page. View the current version.
Compare with Current View Page History
« Previous Version 6 Next »
When writing large and complicated scripts it can become hard to debug the script. Either it becomes difficult to determine where in the large script the point of failure is contained or it is just difficult to identify why the code is not working. A simple solution to this is to add logic to your script that allows you to break it into smaller sections for easier debugging. Or, perhaps you do not wish to perform certain actions while debugging, like sending emails to users. A "test mode" concept will allow you to execute the script without performing this specific actions.
Constructing the flag
The basic idea behind a testing mode is to use flags to identify if the whole script should be run in test mode, or perhaps which sections of the script should be run in test mode. This is done with simple if/then statements. However, it is a good idea to use a struct to create a single flag that has multiple attributes. Whenever I create a test mode flag I usually start with a structure like this:
struct debug { boolean debug; boolean useLogging; boolean sendEmails; boolean codeAction1; boolean codeAction2; boolean codeAction3; } |
This allows me to create a single flag that controls multiple actions.
Initializing the flag
To start using the structure you must initialize it like the code below. When I create the actual flag (instead of the struct) I do not call it "debug" or "test" because that becomes redundant with the "debug" or "test" property. For example, I don't want the flag to look like "debug.debug" or "test.test". For this reason I call the main flag something else. And, for whatever reason, I typically use "goldBug" from the Richard Scarry children books.
debug goldBug; goldBug.debug = true; |
Using the flag
Now that the flag has been initialized it can be used as a wrapper around your actual code like this:
if(goldBug.debug != true) { //do some live, non-test action } else { //do some test/debug related action } |
Setting up the modes
I typically use the "debug" or "test" property to determine if the script is indeed running in test mode. If it is running in test mode you may have specific settings or sections of code that should or should not run. I use the main flag to help set the test mode as a whole. Also, the flag can be used to set the script to live or production mode as well. In the example below logging will be used in test mode and emails will not be sent to the users. However, in production mode logging will not be used but emails will be sent.
if(goldBug.debug == true) { //test mode goldBug.useLogging = true; goldBug.sendEmails = false; goldBug.codeAction1 = true; goldBug.codeAction2 = true; goldBug.codeAction3 = false; } else { //live mode goldBug.useLogging = false; goldBug.sendEmails = true; goldBug.codeAction1 = true; goldBug.codeAction2 = true; goldBug.codeAction3 = true; } |
Automatically trigger test mode
The potential of forgetting to switch test mode off is very high. However, there are some ways to automatically trigger test mode. Generally, the logical place to test the execution of scripts is directly from the SIL Manager in order to take advantage of the output console and viewing the logs. The concept behind the automatic triggering of test mode is that when the script is run from the SIL Manager itself it will be run in test mode but when the script is triggered by other methods it will be run in live or production mode.
There are two methods of automatically triggering test mode and they are based on the method of execution:
Method 1 - Issue context; This is used for the following types of scripts:
Conditions
Validators
Post functions
Listeners
Custom fields
Method 2 - Arguments; This is used for the following types of scripts:
SIL Runner dashboard gadget
SIL Scheduler
Issue Context
The general idea behind this method is that when running in the SIL Manager the script will not be in the context of an issue. This, of course, assumes that an issue has not been added to the Run Configuration setting. If the script is run by executing an issue action (transitions the issue, updating the issue, etc.) it will be in the context of that issue. By using the isIssueContext() routine we can determine if the script is running in the context of an issue.
if(isIssueContext() == true) { goldBug.debug = false; //production mode } else { goldBug.debug = true; //test mode } |
Arguments
When running scripts from the SIL Runner dashboard gadget or through the SIL Scheduler the script can be passed arguments. By detecting if the arguments are present we can trigger test mode. Of course, there are times when the script does not require arguments. For these instances we can just pass a flag as a bogus parameter that indicates the script is running in production mode.
if(size(argv) > 0) { goldBug.debug = false; //production mode } else { goldBug.debug = true; //test mode } |
What is different about test mode?
Issue context
One valuable use of test mode is switching into the context of a test issue or test project. This way, if the action does not perform correctly it will not update a live issue or potentially notify other end users.
string _key; if(goldBug.debug == false) { _key = key; //live issue key } else { _key = "TST-1234"; //test issue } //example use of using conditional issue context %_key%.summary = "New summary text"; |
Output to the console
Often times you may want more information displayed in the output console when debugging then other times. This is helpful for getting the value of custom fields while the script is running.
if(goldBug.debug == false) { //test mode //write output to console runnerLog(key + ": " + summary); } |
Logging
Just like displaying information to the console you might want to write information to the logs when running in test mode.
if(goldBug.debug == false) { //write output to the log logPrint(currentUser() + " performed some action on " + currentDate(), "INFO"); } |
Restricting actions
There may be times were you don't want an action to occur when testing, like sending an email to a user or writing data to a database. And, there may also be times when you do want this action to occur so you can test that it is behaving correctly. In these instances the action can be controlled using separate flags.
JEmailMessage email; email.to = {"testJiraUser1", "testEmail@cprime.com", "testJiraUser2"}; email.subject = "Email to Santa"; email.message = "Dear Santa, I want a train."; for(string a in attach) { JEmailAttachment att; att.name = a; att.file = getAttachmentPath(key, a); email.attachments += att; } if(goldBug.sendEmail == false) { //live mode sendEmail(email); //send email } else { //test mode //don't spam me bro! } |
Compound flag conditions
Like the previous example, there may be times where you want to toggle an action on or off while testing. However, can you be sure that the flag is set back to the proper setting before going live with the script changes. By using compound conditions with the flags we can ensure that the action is only getting toggled on or off in test mode but will always be on in live mode.
if(goldBug.debug == false) { //live mode sendEmail(email); } else if(goldBug.debug == true && goldBug.sendEmail == true) { //test mode sendEmail(email); } else { //don't send email } |
This example will always send the email in live mode but will only send the email in test mode when the sendEmail flag is set to true.
The example above can be consolidated into a single statement:
if(goldBug.debug == false || (goldBug.debug == true && goldBug.sendEmail == true)) { sendEmail(email); } |
Complete test mode script template
Here is a complete test script template for you to get started with:
struct debug { //define the flag boolean debug; boolean useLogging; boolean sendEmails; boolean codeAction1; boolean codeAction2; boolean codeAction3; } debug goldBug; //initialize //automatically set the flag if(isIssueContext() == true) { goldBug.debug = false; //production mode } else { goldBug.debug = true; //test mode } //build out the modes if(goldBug.debug == true) { //test mode goldBug.useLogging = true; goldBug.sendEmails = false; goldBug.codeAction1 = true; goldBug.codeAction2 = true; goldBug.codeAction3 = false; } else { //live mode goldBug.useLogging = false; goldBug.sendEmails = true; goldBug.codeAction1 = true; goldBug.codeAction2 = true; goldBug.codeAction3 = true; } /*---------------------------------------- -----------End Include File--------------- -----------------------------------------*/ //use the flag in the script if(goldBug.debug == false || (goldBug.debug == true && goldBug.codeAction1 == true)) { //do something //test output if(goldBug.debug == true) { //test mode only runnerLog("See something"); } //write to log if(goldBug,useLogging == true) { //test mode only logPrint(currentUser() + " performed some action on " + currentDate(), "INFO"); } } if(goldBug.debug == false || (goldBug.debug == true && goldBug.codeAction2 == true)) { //do something else } if(goldBug.debug == false || (goldBug.debug == true && goldBug.codeAction3 == true)) { //do something else } |
Or, use an include file
In order to make the script smaller and more readable the bulk of the test mode script can be contained in a separate include file and reused in other script as well. In this example the script from the beginning up to the line that says "End Include File" is stored off as a separate file called testMode.incl.
include "testMode.incl"; if(goldBug.debug == false || (goldBug.debug == true && goldBug.codeAction1 == true)) { //do something //test output if(goldBug.debug == true) { //test mode only runnerLog("See something"); } //write to log if(goldBug,useLogging == true) { //test mode only logPrint(currentUser() + " performed some action on " + currentDate(), "INFO"); } } if(goldBug.debug == false || (goldBug.debug == true && goldBug.codeAction2 == true)) { //do something else } if(goldBug.debug == false || (goldBug.debug == true && goldBug.codeAction3 == true)) { //do something else } |
Contents
Advised: Read at your (and possibly my) own risk. There are some program languages that have native capabilities to more gracefully solve the issues outlined below. However, since SIL is a sudo language it does not necessarily have all the bells and whistles and a complete programming language. What is presented below are methods that can be used to solve the issues right now using the tools that are currently available. If you disagree with these suggestions please suppress the urge to hunt me down in person and share your opinions with your fists (or oversized purse like the last time). Instead, please feel free to send us your thoughts and opinions at anovaproducts@appfire.com.
- No labels