...
This document illustrates techniques for controlling access to data used in action sequences in the Pentaho BI Platform. It assumes familiarity with the Pentaho BI Platform, creating action sequences using the action sequence editor, and SQL database queries. We recommend reading the Creating Pentaho Solutions guide and Getting Started with Design Studio prior to reading this document.
Motivation
When developing an action sequence, you may need to restrict a user's view of the available data based on some criteria. For instance, you may develop an action sequence that queries a database for financial information, and displays the financial information in a web page. However you may want to restrict a user's view of the financial data to the data from the user's region (e.g. Eastern, Southern, Western, Northern). You can use a system action to assist in filtering the data provided to the user.
...
A system action is simply an action sequence that is configured to run either at system start time (i.e. when the application server starts) or session start time (i.e. when the user logs in). The output of a system action is available to other action sequences as input parameters. System actions are configured by adding the appropriate child elements to the <system-action>
element in the pentaho.xml file.
Note | ||
---|---|---|
| ||
In order to reduce the amount of time it takes to start the platform, System Actions configured to run at global scope are run in a lazy manner. The platform postpones running the global System Actions until it absolutely has to run them. It has to run them when the first user session is initiated. When the first user session is initiated, all of the global System Actions run, and then all of the session System Actions run. When the second and following user sessions are initiated, only the session System Actions run. |
...
The first step in configuring a system action is to locate and edit the pentaho.xml. In the Pentaho Preconfigured Install, the pentaho.xml file is located in the pentaho-demo/system
folder.
You configure a system action by adding information about your action sequence to the <system-action>
node of the pentaho.xml file. For instance, if your action sequence is called usernameToRegion.xaction
, it exists in the repository under samples/filters
Configuring System Actions
Anchor | ||||
---|---|---|---|---|
|
The first step in configuring a system action is to locate and edit the sessionStartupActions.xml. sessionStartupActions.xml is located in pentaho-solutions/system
folder.
Assume your action sequence is called session-region-list.xaction
, it exists in the repository under bi-developers/rules, you want the action sequence to run when the user logs in, with the output placed in session scope, and you want the output of the action sequence to be available in a servlet, your addition to sessionStartupActions.xml would look like this:
Code Block | ||||
---|---|---|---|---|
| ||||
<bean class="org.pentaho.platform.engine.core.system.SessionStartupAction">
<property name="sessionType" value="org.pentaho.platform.web.http.session.PentahoHttpSession"/>
<property name="actionPath" value="bi-developers/rules/session-region-list.xaction"/>
<property name="actionOutputScope" value="session"/>
</bean>
|
For another example, assume your action sequence is called global-department-list.xaction
, it exists in the repository under bi-developers/secure
, you want the action sequence to run when the user logs in, system starts up with the output placed in session global scope, and you want the output of the action sequence to be available in a portlet, you would write this xml elementthen your addition to sessionStartupActions.xml would look like this:
Code Block | ||||
---|---|---|---|---|
| ||||
<org <bean class="org.pentaho.platform.engine.uicore.portlet.PentahoPortletSession scope="session"system.SessionStartupAction"> samples/filters/usernameToRegion.xaction </<property name="sessionType" value="org.pentaho.uiplatform.portlet.PentahoPortletSession> |
and place it inside the <system-action>
node in the pentaho.xml.
As another example, suppose your action sequence is called setCompanyName.xaction
, it exists in the repository under samples/filters
, you want the action sequence to run when the system starts up with the output placed in global scope, and you want the output of the action sequence to be available in a Servlet/jsp, you would write this xml element:
Code Block |
---|
<org.pentaho.core.session.PentahoHttpSession scope="global">
samples/filters/setCompanyName.xaction
</org.pentaho.core.session.PentahoHttpSession>
|
Info | ||
---|---|---|
| ||
Let's break it down a bit using the previous examples.
|
Your pentaho.xml file would look something like this:
Code Block | ||
---|---|---|
xml | xml | <pentaho-system> ... <system-actions> <org.pentaho.ui.portlet.PentahoPortletSession scope="session"engine.security.session.TrustedSystemStartupSession"/> <property name="actionPath" value="bi-developers/secure/global-department-list.xaction"/> <property name="actionOutputScope" value="global"/> </bean> |
Let's summarize the previous two examples.
- We make the output of the action sequence available to portlets by specifying a
sessionType
oforg.pentaho.platform.web.portal.PentahoPortletSession
. - We make the output of the action sequence available to servlets/jsps by specifying a
sessionType
oforg.pentaho.platform.web.http.session.PentahoHttpSession
. - We make the action sequence run at system start time and make output available in global scope by specifying an
actionOutputScope
ofglobal
and specify asessionType
oforg.pentaho.platform.engine.security.session.TrustedSystemStartupSession
. - We make the action sequence run at session start time and make output available in session scope by specifying an
actionOutputScope
ofsession
. - We identify the name and location of our action sequence by specifying
actionPath
.
Your pentaho.xml file would look something like this:
Code Block | ||||||
---|---|---|---|---|---|---|
| ||||||
<beans> <bean id="sessionStartupActionsList" class="java.util.ArrayList"> <constructor-arg> <list> <bean class="org.pentaho.platform.engine.core.system.SessionStartupAction"> <property name="sessionType" value=""/> <property name="actionPath" value=""/> <property name="actionOutputScope" value=""/> samples/filters/usernameToRegion.xaction</bean> </org.pentaho.ui.portlet.PentahoPortletSession> <org.pentaho.core.session.PentahoHttpSession scope="global">!-- ...other session startup actions... --> <bean class="org.pentaho.platform.engine.core.system.SessionStartupAction"> samples/filters/setCompanyName.xaction<property name="sessionType" value=""/> </org.pentaho.core.session.PentahoHttpSession> </system-actions><property name="actionPath" value=""/> <property name="actionOutputScope" value=""/> ... </pentaho-system> |
...
bean>
</list>
</constructor-arg>
</bean>
</beans>
|
Now you know how to configure a system action, but what might a system action look like that will assist in controlling access to data?
...
title | What is _Scope_? |
---|
To answer that question, we must first talk about scope. In the context of the Pentaho BI Platform, scope describes the lifetime of parameters in the platform's runtime environment. Parameters can be scoped at one of four different lifetimes:
- request: The lifetime of a parameter in request scope is the duration of the current request/response. A parameter in request scope is only available to the thread executing the current request/response.
- session: The lifetime of a parameter in session scope is a user's session. A session starts when a user logs in, and ends when the user logs out, or the session times out. A parameter in session scope is available to all request/response threads associated with a user's session.
- global: The lifetime of a parameter in global scope begins when the application server initiates the application, and ends when the application server terminates the application. Parameters in global scope are availabe to threads in any user's session.
- runtime: Parameters in runtime scope have an infinite lifetime. Their existence transcends the application server's lifetime. This has certain metaphysical consequences and therefore should be used with appropriate caution. Parameters in runtime scope are available to threads in any user's session.
Developing an Action Sequence to Run As a System Action
...
title | Tools You'll Need to Create and Run the Code |
---|
...
Note: This code was developed to be run
...
against the SampleData
...
datasource that comes with the Preconfigured Install. The code was developed using the action sequence editor. The action sequence editor is part of the
...
Pentaho Design Studio, which is a plugin for the Eclipse IDE.
...
In this example we'll develop an action sequence to generate an HTML report containing financial data for the current user's region. This suggests that when the user logs in, we need to identify the user's region, and store this information someplace In this example we'll develop an action sequence to generate an HTML report containing financial data for the current user's region. This suggests that when the user logs in, we need to identify the user's region, and store this information some place that we can use later in our action sequence.
To identify the user's region, let's create a new action sequence in Eclipse using the action sequence editor.
To create a new action sequence, click on the Design Studio icon in the Eclipse tool bar, and select New Action Sequence. Name the action sequence usernameToRegion.xaction, and place it in the samples/filters folder of the repository. The Container text box in the wizard should contain this text: /pentaho-solutions/samples/filters.
Info | ||
---|---|---|
| ||
The repository is where resources, like action sequence definition documents are stored in the Pentaho BI Platform. The Platform has a variety of repository implementations. The repository used by the Pentaho Preconfigured Install is a file system repository, and it is located in the |
...
Next, in the action sequence editor, add an input called name
to the Process Inputs tree control. Configure it to be of type string
, and define its scope to be session
.
When the user logs into the BI Platform, the platform places a parameter called name
into the session scope. name
is the user name that was used to log into the platform. We can get access to this parameter by defining it as an input to our action sequence.
The DEPARTMENT_MANAGERS
table of the SampleData database has the columns MANAGER_NAME
, REGION
, and EMAIL
. MANAGER_NAME
is the full name of the manager. We need a mechanism for mapping the user's login name to their fullname. We can then use a SQL query in an action sequence to discover the user's region.
We'll do this by adding a JavaScript action to our action sequence. Using the Process Actions tree control in the action sequence editor, click on the Add control (blue + sign with a triangle next to it), and select Generate Data From -> JavaScript. In the JavaScript editor, add this JavaScript code:
Code Block |
---|
fullName = "";
fullNameArray = new Object();
fullNameArray["joe"] = "Joe Pentaho";
fullNameArray["suzy"] = "Suzy Pentaho";
fullNameArray["pat"] = "Pat Pentaho";
fullNameArray["tiffany"] = "Tiffany Pentaho";
fullName = fullNameArray[name];
|
This code simply maps the user's login name to the user's full name, and places the full name in a JavaScript variable called fullName
. fullName
will be available to the next action in our action sequence.
...
title | JavaScript Caveat |
---|
Warning: Generally in JavaScript, it is good practice (but not required) to declare JavaScript variables using the JavaScript
var
keyword. For instance:
var myVar = "Pentaho Inc.";
However JavaScript variables, in an JavaScript action, that are going to be identified as outputs of the action sequence should not be declared with the JavaScriptvar
keyword. This would be the correct way to do it:
myVar = "Pentaho Inc.";
If you find that your JavaScript variable is not showing up as an output parameter, review your JavaScript action and make sure that you haven't declared with the variable with thevar
keyword.
We still need to identify name
from the Process Inputs tree control as an input to our JavaScript action. Click on the Add control in the Script Inputs editor, and select name
.
We also need to identify the fullName
variable from our JavaScript code as an output from our JavaScript action. To do this, click in the Script Outputs editor, type in fullName
, and identify its type as string
.
Now that we have the fullname
as an output parameter, we can use that parameter in the next action to discover the region that the manager is responsible for.
In the Process Actions tree control, click on the Add control and select Generate Data From -> Relational. Make sure the JNDI radio button is selected. Identify the name of the Database Connection as SampleData. Add the following query to the Query editor:
Code Block | ||||
---|---|---|---|---|
| ||||
select REGION from DEPARTMENT_MANAGERS where MANAGER_NAME = \{PREPARE:fullName\} |
Specify the Result Set Name as RS_MANAGERS_REGION
. And make sure Keep Connection Open is unchecked. The values from the result set will be available to us in our next action.
...
title | What does "Keep Connections Open" mean? |
---|
...
We now have the name of the region that the manager is responsible for, but it is still in the result set. We don't want to keep the relatively "heavy" result set to in our session. We only need the string from the result set with the region name. We can use another JavaScript action to get the region string out of our result set, and then store the region string into our session.
In the Process Actions tree control, click on the Add control and select Generate Data From -> JavaScript.
Since we need access to the result set from the previous action, in the Scripts Input, click on the Add control and select RS_MANAGERS_REGION
.
Next, add the following code in the JavaScript editor:
...
This will get the region
string out of the result set and place it in a JavaScript variable called strRegion
.
We want to make the variable strRegion
available to action sequences that the user may execute later in their session. So we need to place strRegion
into the user's session. To do this, add want to make the variable strRegion
available to the Script Outputs, and identify its type as string
. Once we've done this, you'll notice that in the Process Actions tree control, the script output shows up under our JavaScript action. This identifies strRegion
as an output of the JavaScript action, but we need it to be an output of our action sequence.
...
title | Which scope should the action sequence's output be placed in? |
---|
...
action sequences that the user may execute later in their session. So we need to place strRegion
into the user's session. To do this, add the variable strRegion
to the Script Outputs, and identify its type as string
. Once we've done this, you'll notice that in the Process Actions tree control, the script output shows up under our JavaScript action. This identifies strRegion
as an output of the JavaScript action, but we need it to be an output of our action sequence.
Note: If you want all user sessions to have access to the same output, place
...
the output in global scope
...
. However if the parameters in the output are specific to a particular user or group of users, place it in session scope.
To make strRegion
an output of our action sequence, in the Process Actions tree control, drag and drop strRegion
onto the outputs in the Process Outputs tree control. You'll notice that this will open the Process Output editor in the right pane. In the Process Output editor, click in the first cell of the table, this will display a combo box with a list of scopes (e.g. request, session, global, etc.) We want to place strRegion
into the session scope, so select session from the combo box. At the conclusion of the action sequence, strRegion
will be in the user's session, available to any action sequence executed by the user.
...
Code Block |
---|
select QUADRANT_ACTUALS.REGION, QUADRANT_ACTUALS.DEPARTMENT, QUADRANT_ACTUALS.POSITIONTITLE, QUADRANT_ACTUALS.ACTUAL, QUADRANT_ACTUALS.BUDGET, QUADRANT_ACTUALS.VARIANCE from QUADRANT_ACTUALS where QUADRANT_ACTUALS.REGION=\{PREPARE:strRegion\} |
Specify the Result Set Name as REGION_ACTUALS
. And make sure Keep Connection Open is unchecked.
In order place the information in the result set into the output for our HTML page, expand the Relational node in the Process Actions tree control, and drag and drop REGION_ACTUALS onto the outputs node of the Process Outputs tree control.
...
In order for our getActualsForRegion action sequence to run successfully, we need to configure the userNameToRegion action sequence to run when the user logs in. To do this we need to modify the pentaho.xml file. The pentaho.xml file is located in the system folder of the pentaho-solutions folder of the Pentaho Preconfigured Install. You should recall that in the section#Configuring System Actions I discussed adding this xml element to the <system-actions>
elementto run when the user logs in. Add the following snippet to sessionStartupActions.xml:
Code Block | ||||
---|---|---|---|---|
| ||||
<org <bean class="org.pentaho.platform.engine.uicore.portlet.PentahoPortletSession scope="session"> system.SessionStartupAction"> <property name="sessionType" value="org.pentaho.platform.web.portal.PentahoHttpSession"/> <property name="actionPath" value="samples/filters/usernameToRegion.xaction </org.pentaho.ui.portlet.PentahoPortletSession>to the <system-actions> |
to the pentaho.xml file. Let's do that now.
Running our Action Sequence
Info | ||
---|---|---|
| ||
You will need to understand how to configure and start the JBoss application server hosting the platform before continuing with our example. "/>
<property name="actionOutputScope" value="session"/>
</bean>
|
Running our Action Sequence
In order for this example to run, we need to start the Hypersonic database and start the JBoss server that is hosting the Pentaho BI Platform. Next, in a browser access http://localhost:8080/portal/portal/Pentaho/%5B01%5DHome. This is the home page for the portal in the Pentaho Preconfigured Install. Click on the link in the bottom left named "Standard Login".To login, click on the text "click here to login as Joe", this will populate the Login form with Joe's credentials. Click the Login button. the SampleDatadatabase and start the Pentaho BI Platform that is hosting the Pentaho BI Platform. Now login as Joe.
Next we need the URL that runs our action sequence. To get this URL, go back to the action sequence editor, and click on the Test tab in the bottom right corner. Click on the Generate URL button. Copy the generated URL, and paste it into your browser's address bar.
...
It may be interesting to log out, and log back in as Suzy, and run the action sequence again. Notice how the output has changed to deliver financial information for the region that Suzy is responsible for.
Wrap Up
Action sequences that are run as system actions are powerful tools for supplying action sequences with parameters whose values don't change during the lifetime of the user's session, or the lifetime of the application. If you are a Servlet/jsp developer, you should see that this is much like the way you work with parameters in the application context and the session context of Servlets and jsps.
Action sequences that are run as system actions in session scope are useful for caching information that is unique to an individual user. The user's full name, or the region that a manager is responsible are examples of the kind of information you may store in a session parameter. These parameters can be useful for controlling access to information in corporate databases.
...