Problem description
If you have any complex application in APEX you probably need to generate some untrivial reports and may be in several formats (HTML / PDF / XLSX). It's very painfull to try to do it in pure PLSQL and very expensive to use such enterprise software as Oracle BI Publisher. At the other hand you can use any open report server system like JasperReports but here you have big problem with integration external servers with APEX security system to allow access reports only for authorized users. It's good enougth to solve this if you have tons of regular changing reports so you can spend much time for enviroment and less time for reports support in the future but what can you do otherwise?
Possible solution
Obviously that any complex report can be generated at backend with any high level programing language like Java, Python, Ruby e.t.c. But how to quick and accurate integrate such report script into an APEX application? Here you can find the way that successfully works for me. Below external PDF report generator integration example.
So solution consists of three main steps:
- Add URL/Button for start report generating routine that ends with browser prompt for download report file
- Run command line script with report generation
- Generate correct MIME header to allow client browser to download report file
Run OS command line script from Oracle
Oracle have not easy way to run OS command line script. You can easy find some workarounds. Here I provide dbms_scheduler method that works well for commercial and for XE editions. (Please note that you must create my-temp-script.sh script file in the /tmp/ dir for use this example) Source code on github
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
CREATE DIRECTORY TMP AS '/tmp/'; | |
create or replace | |
PROCEDURE system_run(cmd IN varchar2) | |
IS | |
script_file varchar2(40) := 'my-temp-script.sh'; | |
script_data varchar2(4000); | |
MyFile utl_file.file_type; | |
d varchar2(4000); | |
dump_file varchar2(40) := 'my-temp-file.dat'; | |
dump_type utl_file.file_type; | |
BEGIN | |
-- Open file | |
MyFile := utl_file.fopen('TMP',script_file,'w'); | |
-- Write data to file | |
script_data := '#!/bin/bash' || chr(10) || cmd||'>'||'/tmp/'||dump_file; | |
utl_file.put_line(MyFile, script_data, FALSE); | |
-- Close file | |
utl_file.fflush(MyFile); | |
utl_file.fclose(MyFile); | |
-- Purge old logs, no fun anyway | |
dbms_scheduler.purge_log(JOB_NAME=>'TEST_RUN'); | |
-- Execute script | |
-- The job is created as disabled as | |
-- we execute it manually and will | |
-- drop itself once executed. | |
dbms_scheduler.create_job( | |
job_name => 'TEST_RUN', | |
job_type => 'EXECUTABLE', | |
job_action => '/bin/bash', | |
number_of_arguments => 1, | |
start_date => SYSTIMESTAMP, | |
enabled => FALSE); | |
dbms_scheduler.set_job_argument_value('TEST_RUN', 1, '/tmp/'||script_file); | |
dbms_scheduler.enable('TEST_RUN'); | |
-- Wait for the job to be executed | |
-- usually done within 1 second but | |
-- I set it to 2 just in case. | |
dbms_lock.sleep(2); | |
-- Open the output file and | |
-- print the result. | |
dump_type := utl_file.fopen('TMP',dump_file,'r'); | |
loop | |
begin | |
utl_file.get_line(dump_type,d); | |
dbms_output.put_line(d); | |
exception | |
when others then | |
exit; | |
end; | |
end loop; | |
utl_file.fclose(dump_type); | |
-- Clean up our temp files | |
utl_file.fremove('TMP', script_file); | |
utl_file.fremove('TMP', dump_file); | |
END; |
Generate MIME header
- Create blank page in APEX with blank HTML region
- Add "on load: before header" branch with following code. (don't forget to create EXPORT_DIR directory in Oracle DB like in previous code sample) Source code on githubThis file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters. Learn more about bidirectional Unicode characters
DECLARE l_filename BFILE; v_export_dir VARCHAR2(2000); v_length integer; l_blob blob; des_offset number := 1; src_offset number := 1; x number; BEGIN system_run('/path/to/report/generation/script.sh -r '); /*Run report generator script*/ l_filename := bfilename('EXPORT_DIR','/expected_report_file_name.pdf'); /*Please create EXPORT_DIR in Oracle before use*/ v_length := dbms_lob.getlength(l_filename); DBMS_LOB.createtemporary(l_blob, FALSE); dbms_lob.open(l_filename, dbms_lob.lob_readonly); DBMS_LOB.LOADBLOBFROMFILE(dest_lob=>l_blob, src_bfile=>l_filename, amount=>v_length, dest_offset=>des_offset, src_offset=>src_offset); owa_util.mime_header('application/pdf', FALSE ); -- set the size so the browser knows how much to download htp.p('Content-length: ' || v_length); -- the filename will be used by the browser if the users does a save as htp.p('Content-Disposition: attachment; filename="expected_report_file_name.pdf"'); -- close the headers owa_util.http_header_close; -- download the BLOB wpg_docload.download_file(l_blob); DBMS_LOB.freetemporary(l_blob); EXCEPTION WHEN OTHERS THEN DBMS_LOB.freetemporary(l_blob); RAISE; END;
Add button for starting all this stuff
Nothing to describe here. Just create button on report page that redirect to "MIME header" page from previous paragraph.
That's all folks!
No comments:
Post a Comment