Tuesday, March 13, 2018

My top 3 to gather user feedback in an Oracle APEX app


In every Oracle APEX application we create, we add a feedback mechanism. We're not only doing this during the development phase, but we also leave it enabled in production.

We want to give the users the ability to give feedback to the team in a structured way. It should not be an hassle to give feedback, one click for the user where he can say what it's going; if he likes it, if something can be improved, if there's a bug... but at the same time we want as much info as possible at the time the user enters this feedback. What app are they in, which page are they on, what session information was there...

Most of the users really like to be heard and the feedback mechanism in our apps helped us a lot to engage with our customers.

Here're my top 3 ways of providing such feedback mechanism in an Oracle APEX app:

Feedback link and Team Development (pre APEX 18.1)

This method I started to use with Oracle APEX 4.2 and beyond. I blogged about using this feature here and here before.

In APEX you can create a new page; called the Feedback page. It will not only create a page, but also a navigation bar entry. Some screenshots of the wizard:


Which will result in:


I prefere to use the Modal dialog option as it feels more integrated in the app. This page is a normal APEX page, so you can further customise. For example you can make the Application and Page fields hidden.

The feedback that is entered is stored in Team Development - an area in APEX where you can do your project management.



Feedback feature in Blueprint and the new APEX 18.1 app creation wizard

In Oracle APEX 18.1 the application wizard got an overhaul and is now inline with the blueprint feature which was enabled previously on apex.oracle.com.


Enabling feedback is a matter of ticking a checkbox... and the result looks awesome to me :)


The user can share his feeling by selecting a smily, enter some feedback and include an attachment. When you use APEX 18.1, you will see the nice floating label (label inside box) for the feedback text item. Also when you submit feedback you get a nicer message that the feedback was submitted, instead that the dialog will automatically disappear as with the standard feedback page in 5.1 and earlier.

In Blueprint on apex.oracle.com and in APEX 5.2 EA1 (which became now APEX 18.1 EA2) the feedback was stored in its own table, but since APEX 18.1 EA2 the feedback is stored again in Team Development. In the Administration section there are a couple of screens that query the apex_team_feedback view. It looks like many of the Blueprint features that were showcased before, became native APEX features (e.g. email framework, feedback).


When clicking on User Feedback and the pencil you see the details for every entry.


You can enter a response which will update Team Development by using the APEX API  apex_util.reply_to_feedback.

What I find interesting is that APEX 18.1 made a change in their pages; it catalogs the pages now as Component or Feature. A feature contains more than just a page, so the Feedback page became the Feedback feature and will create more pages than just the feedback page itself. Just like if you click the box with Blueprint, it can add the administration section and navigation bar entry.


The feedback mechanism we knew in APEX 5.1 and before, got a nice update in Oracle APEX 18.1, it might be worth doing an upgrade once 18.1 hits production.
One nice addition would be the ability to add a screenshot and annotate the screen like Martin and I build a long time ago. It would save the user creating a screenshot and uploading the file.

Feedback with REST API

If you are using another issue or ticketing system it might be worthwhile to gather the feedback there. You can still create the feedback page, but add some additional processes (or replace the team development process) so the feedback is stored in your favourite tracking system like Bugzilla, Jira, Redmine and others. In the next section I'll show how to integrate with two issue tracking systems we use.

Oracle Developer Cloud Issues

In the projects where we use Oracle Exadata Express, we use Oracle Developer Cloud service, which you get with your Exadata Express account to manage our project.


The nice thing is that you have a Git repo and Issues all available and all the other things around to mange and streamline your project. In the feedback page I added a call to a PL/SQL procedure, so an issue is created whenever feedback is given.

The PL/SQL package I wrote to create an Oracle Developer Cloud issue:

create or replace package ora_dev_cloud_pkg as
/*
Purpose : Oracle Developer Cloud module
history
Version Date By Description
------- ----------- ---------------- ----------------------------------------------
0.1 22-FEB-2018 Dimitri First version, based on Dimitri's blog post
*/
g_username constant varchar2(100) := 'no-reply@apexrnd.be';
g_password constant varchar2(100) := 'password';
g_url constant varchar2(200) := 'https://developer.em2.oraclecloud.com/developerXXX-XXX/rest/developerXXX-XXX_project_XXX';
g_url_issues constant varchar2(200) := g_url || '/issues/v2/issues';
function issue_template(
p_summary in varchar2,
p_description in varchar2,
p_type in varchar2)
return clob;
procedure create_issue(
p_summary in varchar2,
p_description in varchar2,
p_type in varchar2);
end ora_dev_cloud_pkg;
/
create or replace package body ora_dev_cloud_pkg as
function issue_template(
p_summary in varchar2,
p_description in varchar2,
p_type in varchar2)
return clob
as
l_clob clob;
l_json_values apex_json.t_values;
l_issue_template varchar2(32767);
l_return clob;
begin
-- do a request for the create form service to get the definition of the Oracle developer cloud issues
l_clob := apex_web_service.make_rest_request(
p_url => g_url_issues || '/create-form',
p_username => g_username,
p_password => g_password,
p_http_method => 'GET',
p_body => '');
-- parse the result
apex_json.parse(l_json_values, l_clob);
-- get the issues url
--l_issues_url := apex_json.get_varchar2(p_path => 'createIssue.links[%d].href', p0 => 2, p_values => l_json_values);
-- debug
--sys.htp.p(l_issues_url);
-- build the json, probably we need to make this more dynamic later
l_issue_template := '
{
"links": [{
"rel": "self",
"href": "'||g_url_issues||'/create-form"
}, {
"rel": "create",
"href": "'||g_url_issues||'"
}],
"summary": "#SUMMARY#",
"description": "#DESCRIPTION#",
"type": "#TYPE#",
"severity": {
"id": 4,
"value": "normal",
"sortkey": 400
},
"status": {
"id": 1,
"value": "UNCONFIRMED",
"isActive": true,
"isOpen": true,
"sortkey": 100
},
"priority": {
"id": 3,
"value": "Normal",
"sortkey": 300
},
"release": {
"id": 1,
"value": "---",
"sortkey": 0,
"product": {
"id": 1
}
},
"product": {
"id": 1,
"name": "APEX RND",
"description": "APEX RND",
"isActive": true,
"defaultRelease": {
"id": 1,
"value": "---",
"sortkey": 0,
"product": {
"id": 1
}
},
"defaultComponent": {
"id": 1,
"product": {
"id": 1
},
"name": "Default",
"description": "default component"
},
"releases": [{
"id": 1,
"value": "---",
"sortkey": 0,
"product": {
"id": 1
}
}],
"components": [{
"id": 1,
"product": {
"id": 1
},
"name": "Default",
"description": "default component"
}],
"releaseTags": []
}
}
';
l_return := replace(l_issue_template,'#SUMMARY#', p_summary);
l_return := replace(l_return,'#DESCRIPTION#', p_description);
l_return := replace(l_return,'#TYPE#', p_type);
return l_return;
end issue_template;
procedure create_issue(
p_summary in varchar2,
p_description in varchar2,
p_type in varchar2)
as
PRAGMA AUTONOMOUS_TRANSACTION;
l_clob clob;
l_issue_template clob;
begin
l_issue_template := issue_template(
p_summary => p_summary,
p_description => p_description,
p_type => p_type);
-- create a new issue
apex_web_service.g_request_headers(1).name := 'Content-Type';
apex_web_service.g_request_headers(1).value := 'application/json';
l_clob := apex_web_service.make_rest_request(
p_url => g_url_issues,
p_username => g_username,
p_password => g_password,
p_http_method => 'POST',
p_body => l_issue_template);
end create_issue;
end ora_dev_cloud_pkg;
/
Bitbucket Git Issues

When we started in 2015 with the development of APEX Office Print, we used Team Development in Oracle APEX to manage the development and for version control we used Bitbucket (Git repo). We use different technologies like PL/SQL, APEX, Node.js, Markdown, CSS and HTML. A bit later we decided to use the issues in Git for our node.js code as it made it easier to track an issue/feature and a certain commit (we enter the issue number when we commit).


Bitbucket has also some nice features to integrate Trello boards and Bitbucket cards.

At one stage I wanted to add all our Team Development features as Bitbucket issues, so I wrote a small script that calls the REST API and creates the issues for you:

declare
l_clob clob;
l_body clob;
begin
for r in (select feature_name as title, '*'||focus_area||'*' ||feature_description || ' ' || publishable_description || '' as description
from apex_team_features
where nvl(feature_status,0) < 100 and nvl(focus_area,'A') <> 'ON HOLD')
-- change where to what you need
loop
apex_json.initialize_clob_output;
apex_json.open_object;
apex_json.write(p_name => 'title', p_value => r.title);
apex_json.open_object(p_name => 'content');
apex_json.write(p_name => 'raw', p_value => r.description);
apex_json.close_object;
apex_json.close_object;
l_body := apex_json.get_clob_output;
apex_json.free_output;
apex_web_service.g_request_headers(1).name := 'Content-Type';
apex_web_service.g_request_headers(1).value := 'application/json';
l_clob := apex_web_service.make_rest_request(
p_url => 'https://api.bitbucket.org/2.0/repositories/owner/repo/issues',
p_http_method => 'POST',
p_username => 'xxx',
p_password => 'xxx',
p_body => l_body);
end loop;
end;
Nothing stops you to add a process on your Feedback page in Oracle APEX to create a Bitbucket issue automatically. If you copy everything inside the for loop, you are golden.

I hope this post helps you to get user feedback... and don't hesitate to put in the comments what you do to engage with your users.