Monday, June 15, 2015

Installing Oracle XE, ORDS and Apex on CentOS - Part Two: Installation

This is part two in a series of blog posts about how to install Oracle 11g Express Edition (XE) with Oracle Application Express (Apex) on a CentOS Linux server, with Apex served by Oracle REST Data Services (ORDS) running on top of Tomcat and Apache.

Let's get right to it, starting with Oracle XE.

Installing Oracle XE

This is actually quite straightforward, assuming you have followed the prerequisite steps in part one.

# install Oracle XE 11g on CentOS
# assumes the installation file has already been copied to /u01/download
# create Oracle user and groups
groupadd oinstall
groupadd dba
useradd -g oinstall -G dba,oinstall oracle
chown -R oracle:oinstall /u01
# change the password
passwd oracle
# unzip the installation files
cd /u01/download
unzip oracle-xe-11.2.0-1.0.x86_64.rpm.zip
cd /u01/download/Disk1
# run installation
rpm -ivh oracle-xe-11.2.0-1.0.x86_64.rpm
# run the configuration script
# accept the defaults (unless you know what you are doing!)
/etc/init.d/oracle-xe configure
view raw xe_install.sh hosted with ❤ by GitHub


As part of the installation, we created a user called oracle. It is useful to set up the default environment of this user to include the path to the sqlplus executable, so we can start sqlplus from anywhere.

# setup Oracle environment in bash profile to be able to access sqlplus from anywhere
# http://stackoverflow.com/questions/16823591/how-to-add-lines-to-end-of-file-linux
# this adds the Oracle environment variables to the "oracle" user, but you may also want to add it to root or any other users
echo '. /u01/app/oracle/product/11.2.0/xe/bin/oracle_env.sh' >> /home/oracle/.bash_profile


Now, let's log in to Oracle as SYS and check that everything looks OK:

# initial test of Oracle XE database after install
# become the "oracle" user
su - oracle
# connect to database
sqlplus /nolog
connect sys as sysdba
-- basic query to see stuff working
select sysdate, sys_context('userenv', 'server_host') from dual;
-- check components and versions
select comp_id, version, status
from dba_registry;
-- check users
select username, account_status
from dba_users
order by 1;


At this point you have an Oracle XE instance running, which also includes Apex 4.0 and the Embedded PL/SQL Gateway (EPG) running on port 8080. That is nice, but it's an old Apex version and the EPG web server is not really suited for heavy usage. We want the latest Apex version, and we want to use ORDS. Read on...

Installing Java


ORDS and Tomcat are both Java applications, so we need to install Java. Actually, we need the Java JDK (Java Development Kit), as opposed to just the Java JRE (Java Runtime Environment). There may already be something called the OpenJDK on the CentOS server, but we want the Oracle-supplied JDK, so let's remove OpenJDK and install the JDK that we downloaded from Oracle:

# install Oracle Java JDK
# see http://stackoverflow.com/questions/20901442/how-to-install-jdk-in-centos
# remove OpenJDK
yum remove java*
# assume .rpm file is already copied to /u01/download
cd /u01/download
rpm -ivh jdk-7u79-linux-x64.rpm
# see the java_env.sh script for how to add the Java environment to the bash profile
# check the java version installed
java -version
To have the Java binaries available from anywhere, we add the Java path to the bash profile of the root user:

# setup Java environment in bash profile to be able to access java from anywhere
# http://stackoverflow.com/questions/16823591/how-to-add-lines-to-end-of-file-linux
# http://www.davidghedini.com/pg/entry/install_tomcat_7_on_centos
echo "JAVA_HOME=/usr/java/latest" >> /root/.bash_profile
echo "export JAVA_HOME" >> /root/.bash_profile
echo "PATH=$JAVA_HOME/bin:$PATH" >> /root/.bash_profile
echo "export PATH" >> /root/.bash_profile
view raw java_env.sh hosted with ❤ by GitHub

Installing Tomcat


To install Tomcat, we will download the installation file directly to the server using the wget command, and then unzip it. Create a tomcat user to run the tomcat process.

# download and install Tomcat
cd /u01/download
# see https://tomcat.apache.org/download-70.cgi for latest version
wget "http://www.eu.apache.org/dist/tomcat/tomcat-7/v7.0.64/bin/apache-tomcat-7.0.64.zip"
# copy file and unzip it
cp apache-tomcat-7.0.64.zip /usr/share/apache-tomcat-7.0.64.zip
cd /usr/share
mkdir tomcat7
unzip apache-tomcat-7.0.64.zip -d tomcat7
# make a symbolic link so we can reference the latest version using /latest instead of a specific version
cd /usr/share/tomcat7
ln -s apache-tomcat-7.0.64 latest
# create tomcat group and user
groupadd tomcat
useradd -s /bin/bash -g tomcat tomcat
chown -Rf tomcat:tomcat /usr/share/tomcat7/apache-tomcat-7.0.64/
# set password
passwd tomcat
To avoid conflicts with Oracle XE running the Embedded PL/SQL Gateway on port 8080, change Tomcat's default port number to 8090 by editing the server.xml file. (Note: Because we will put Apache in front of Tomcat, we won't ever access Tomcat directly on port 8090, and we will soon disable EPG, but let's just avoid possible conflicts anyway by assigning different ports.) It's also important to set the URIEncoding to UTF-8.

<Connector port="8090" protocol="HTTP/1.1"
connectionTimeout="20000"
maxThreads="400"
compression="on"
compressableMimeType="text/html,text/xml,text/plain,text/css,text/javascript"
redirectPort="8443"
server="whateverFakeName"
URIEncoding="UTF-8" />
Next we need to create a script that can be used to start, stop and restart Tomcat as a service. Save the following as tomcat under /etc/init.d/

#!/bin/bash
# description: Tomcat Start Stop Restart
# processname: tomcat
# chkconfig: 234 20 80
JAVA_HOME=/usr/java/latest
export JAVA_HOME
PATH=$JAVA_HOME/bin:$PATH
export PATH
CATALINA_HOME=/usr/share/tomcat7/latest
case $1 in
start)
/bin/su tomcat $CATALINA_HOME/bin/startup.sh
;;
stop)
/bin/su tomcat $CATALINA_HOME/bin/shutdown.sh
;;
restart)
/bin/su tomcat $CATALINA_HOME/bin/shutdown.sh
/bin/su tomcat $CATALINA_HOME/bin/startup.sh
;;
esac
exit 0

Then we need to set up the above script to run automatically if the server is rebooted.

# assumes you have created a service script at /etc/init.d/tomcat
cd /etc/init.d/
# make the script executable
chmod 755 tomcat
# set to start at boot time
chkconfig --add tomcat
chkconfig --level 234 tomcat on
# verify it
chkconfig --list tomcat
# start tomcat
service tomcat start
# note: this gave error "Cannot find /usr/share/tomcat7/latest/bin/catalina.sh
# The file is absent or does not have execute permission"
# see http://louis-sawtell.com/content/tomcat-cannot-find-bincatalinash
cd /usr/share/tomcat7/latest/bin
chmod +x *.sh
chmod +x *.jar
# stop tomcat
service tomcat stop
# restart tomcat
service tomcat restart
# check logfile and look for any errors
cat /usr/share/tomcat7/latest/logs/catalina.out

Installing ORDS


The Oracle Rest Data Services (ORDS) installation consists of unzipping the installation file, running the configuration to specify database details, and then copying the ords.war file into the Tomcat webapps folder.

# install Oracle Rest Data Services (ORDS)
# assumes you have already downloaded ORDS from the Oracle website
# and copied the ORDS zip file to the server
# unzip the files, and move them into a dedicated directory
cd /u01/download
mkdir ords210
unzip ords.2.0.10.289.08.09.zip -d ords210
mv /u01/download/ords210 /u01/ords
# Run the ORDS configuration before deployment
cd /u01/ords
java -jar ords.war
# the ORDS configuration "wizard" will now start
# when prompted for ORDS configuration directory, enter /u01/ords/config
# then provide the necessary connection info (server, port, sid, passwords, etc)
# the values get stored in /u01/ords/config/defaults.xml and may be modified there
# IMPORTANT: "RESTful Services" is required by Apex 5,
# so enable this by specifying passwords for the APEX_LISTENER and APEX_REST_PUBLIC_USER when prompted
# for a DEV environment, open the config file and set "debug.printDebugToScreen" to "true" to get detailed errors on screen
# for a production environment, this setting should be left as "false"
# the "tomcat" user (created as part of Tomcat install) needs write access to the /config/ folder
chown -R tomcat:tomcat /u01/ords/config
# copy the .war into the webapps folder
cp ords.war /usr/share/tomcat7/latest/webapps/ords.war
# restart Tomcat to deploy the .war
service tomcat restart
# if using Tomcat standalone, copy the Apex image files to Tomcat "i" folder
# NOTE: no need to do this if using Apache as web server in front of Tomcat
# cp -r /u01/download/apex/images /usr/share/tomcat7/latest/webapps/i/

Installing Apache


The last step in completing our web stack is to install the Apache HTTP server and place it "in front of" Tomcat. This means that all requests to the server go to Apache first. Requests for static files (images, Javascript and CSS) is served directly by Apache. Requests for dynamic content (ie the actual HTML pages generated by Apex via ORDS) is served by Tomcat, using Apache as a proxy.

Installing Apache is very straightforward:

# install Apache on CentOS
# see https://www.linode.com/docs/websites/apache/apache-2-web-server-on-centos-6
# install
yum install httpd -y
# start the server
service httpd start
# set to start automatically on boot
chkconfig httpd on
Then we need to add our custom configuration. By default, Apache is set up to read any .conf file placed in the /conf.d/ subfolder, so let's create an apex.conf file there. Note that these additional config files are read and processed in alphabetical order, so name your custom config accordingly if you use multiple config files.

# customized Apache configuration
# add this to the end of /etc/httpd/conf/httpd.conf
# or put it in a separate file such as /etc/httpd/conf.d/apex.conf
# disable sensitive version info
ServerSignature Off
ServerTokens Prod
# standard alias for Apex image files
Alias /i/ "/var/www/apex/images/"
# forward dynamic (ORDS) requests to Tomcat
<VirtualHost *:80>
ProxyRequests Off
ProxyPreserveHost On
<Proxy *>
Order deny,allow
Allow from all
</Proxy>
ProxyPass /ords ajp://localhost:8009/ords
ProxyPassReverse /ords ajp://localhost:8009/ords
</VirtualHost>
# enable compression of static content
<IfModule mod_deflate.c>
SetOutputFilter DEFLATE
AddOutputFilterByType DEFLATE text/plain text/html text/xml text/css text/javascript
</IfModule>
# enable client caching of static content
<IfModule mod_expires.c>
ExpiresActive On
ExpiresByType image/gif "access plus 7 days"
ExpiresByType image/jpeg "access plus 7 days"
ExpiresByType image/png "access plus 7 days"
ExpiresByType text/css "access plus 7 days"
ExpiresByType text/javascript "access plus 7 days"
ExpiresByType application/javascript "access plus 7 days"
ExpiresByType application/x-javascript "access plus 7 days"
</IfModule>

Installing (upgrading to) latest Apex version


Finally, we need to upgrade the Apex installation that came bundled with Oracle XE to the latest and greatest Apex version (version 5.0 at the time of writing).

This is done by unzipping the Apex installation file, then running the Apex installation script via sqlplus. There are two different Apex installations to choose from: Either a full installation that includes the Application Builder (suitable for a development environment), and a more lightweight and secure "runtime-only" installation (suitable for test and production environments). Running the full installation on the standard 1GB server at DigitalOcean should take about 12-15 minutes.

# install (or upgrade) Apex on Oracle XE on CentOS
# change to the "oracle" user (created as part of XE installation)
su - oracle
# assumes you have already copied Apex installation file to server
cd /u01/download
unzip apex_5.0.1_en.zip
cd apex
# start sqlplus and run the installation script
sqlplus /nolog
# connect sys as sysdba
# -- run the following for a full installation (including the Application Builder)
# @apexins.sql SYSAUX SYSAUX TEMP /i/
# -- run the following for a runtime-only installation (for production environments)
# @apxrtins.sql SYSAUX SYSAUX TEMP /i/
# check the installation log (last 200 lines)
tail -n 200 *.log
# rename the folder so we can have multiple versions downloaded (apex501, apex502, apex60, etc)
cd /u01/download/
mv apex apex501
# now remember to copy the Apex images files into the web server "/i/" folder
view raw apex_install.sh hosted with ❤ by GitHub

We also need to make sure the apex_public_user schema is unlocked (and stays that way!).

-- run as SYS
-- set up service profile to avoid expiring passwords
create profile web_service_profile
limit password_life_time unlimited;
alter user apex_public_user profile web_service_profile;
alter user apex_listener profile web_service_profile;
alter user apex_rest_public_user profile web_service_profile;
-- unlock the apex public user and set a password
alter user apex_public_user account unlock;
alter user apex_public_user identified by pick_a_password;
-- remove previous versions of Apex, if desired
-- drop user apex_040000 cascade;
-- drop user apex_040100 cascade;
-- drop user apex_040200 cascade;
We need to copy all the static Apex files (images, CSS, Javascript) to the Apache web folder.

# copy the Apex image files to corresponding Apache web folder
# assumes you have unzipped the Apex files already
mkdir -p /var/www/apex/images
cp -rf /u01/download/apex501/images/ /var/www/apex
# note: make sure you have set up an Apache alias "/i/" to the /images folder above
When running on top of ORDS, Apex 5 uses the "RESTful Service" feature to serve any application-specific or workspace-specific static files, so we need to configure Apex with REST:

# configure ORDS RESTful Services for Apex
# see https://docs.oracle.com/cd/E59726_01/install.50/e39144/listener.htm#HTMIG29335
su - oracle
cd /u01/download/apex501
sqlplus /nolog
connect sys as sysdba
@apex_rest_config.sql
Now (finally!), if everything works, we should be able to access the new Apex installation by going to the following URL:

  http://servername/ords/apex

If everything works, you should see this familiar page:



Did it work? Great, now enjoy Apex 5! But wait, we are not fully done yet! In the next part of this series, I will describe various additional configuration that you should perform for a more secure and scalable server.

Stay tuned!

20 comments:

Anonymous said...

if SELINUX is activated then

chcon -R --reference=/var/www /var/www/apex/images

Rob van Wijk said...

Hi Morten,

Thank you for saving me -and likely other people as well- lots of time figuring out all these details. Great job!

Extra tip for other readers: don't be stubborn like me to choose ORDS 3.0 instead of 2.0.10 for now. Using 3.0 will lead to a 404 with the top line of the Java stack saying "oracle.dbtools.http.errors.NotFoundException"

Regards,
Rob.

Kai Donato said...

Thank you for this awesome tutorial.
I got it working with Tomcat 8 and ORDS 3.0 - with some modifications of course.

Best Regards!
Kai

Niall Mc Phillips - Long Acre said...

Thanks for these great notes.

In my case, Centos7 required libaio - this was easily fixed with the "yum install libaio -y" command

rpm -ivh oracle-xe-11.2.0-1.0.x86_64.rpm
error: Failed dependencies:
libaio >= 0.3.104 is needed by oracle-xe-11.2.0-1.0.x86_64

Georg said...

Thanks for this tutorial!
Please tell me what adjustments need to be made to make it working with ORDS 3.0.2?
Thanks in advance

Georg said...

Thanks for this tutorial!

Could you please tell my how to get this system running with the latest ORDS 3.0.2 and tomcat 8 (java jdk 8.66) installed?
Requesting /ords leads to a 404 http error.
Thanks in advance.

Peter said...

two things in your tutorial concerning the server ports are not quite clear to me:

1) Tomcat is configured for port 8090, but the default port offered by the ords
installation is 8080!?
2) The Apex port in the Apache configuration is 8009 - is this a typo??

it's a miracle to me - please help!

Thanks in Advance
Peter

Morten Braten said...

@Peter: To clarify regarding the port numbers:

* Port 80: Apache listens on this port. This is the standard port for HTTP requests.

* Port 443: Apache listens on this port if you have enabled SSL. This is the standard port for HTTPS requests.

* Port 8080: This is the default port for the Embedded PL/SQL Gateway (aka DBMS_EPG). The EPG should not be used in production deployments, and should be disabled.

* Port 8090: In this tutorial, Tomcat is set up to use port 8090, since the default for Tomcat is port 8080, and this would conflict with EPG. It is explained in the article: "To avoid conflicts with Oracle XE running the Embedded PL/SQL Gateway on port 8080, change Tomcat's default port number to 8090 by editing the server.xml file. (Note: Because we will put Apache in front of Tomcat, we won't ever access Tomcat directly on port 8090, and we will soon disable EPG, but let's just avoid possible conflicts anyway by assigning different ports.)"

* Port 8009: This is not a typo. This is the port used by Tomcat to talk to the Apache server via the AJP protocol. Port number 8009 is the default for the AJP protocol in Tomcat. We therefore add this port number to the Apache configuration so it knows which port it should talk with Tomcat on. All traffic to Apache for the "/ords" folder will be redirected to Tomcat on port 8009.

Remember that the only ports that should be open in the firewall to the outside world is port 80 (for http) and port 443 (for https). And since you need to be able to login via ssh to administer the machine, you need to open port 22 (or whatever port you have assigned for ssh), but for this port you could also set additional restrictions such as only allowing certain IP addresses to login via ssh.

- Morten

Nikola said...

Hi Morten.
What should i chage in apex.conf http config file to get url redirect from my domain to apex?

Example: I enter example.com in url, and it's redirected to https://example.com/apex/

I tried to add this at begin, but not working:

DocumentRoot "/var/www/html/"
ServerName example.com

Redirect / https://example.com/apex/

Thanks

Morten Braten said...

@Nikola: Try the following, it should redirect any request on the root website on port 80 to the full URL (such as an APEX application, if desired).



# forward root to default website
RedirectMatch 302 ^/$ https://www.example.com/ords/f?p=100




- Morten

Unknown said...

Hey!, I'm getting an error when i try to open ip_address/ords/apex and ip_address/ords after all the steps, AuthConfigFactory error: java.lang.reflect.InvocationTargetException

Any idea on what it could be?

Morten Braten said...

@Hugo: I would check the Tomcat log (/logs/catalina.out) for any errors first. Then perhaps enable debug error messages to screen in ORDS as described here: https://docs.oracle.com/cd/E37099_01/doc.20/e25066/trouble.htm#AELIG7207

- Morten

Unknown said...

Hi Morton,
great cookbook. it enabled me to get the ords in connection with tomcat and apache up and running very quick.
what confuses me and what took the most time to figure out is the defintion of the image directory.
on my windows box i have to define it like that Alias /i/ "D:\Temp\apex\apex_5.1.1\apex\images/"
including the trailing "/" otherwise i will get "GET /i/css/apex_ui.min.css?v=5.1.1.00.08 HTTP/1.1" 404 .... in the sccess log.
do you know the reason or is that just an odd behavior of windows-apache


greetings
peter

Unknown said...

- just to get follow up mails ;-)

Morten Braten said...

@Peter: The Apache docs (https://httpd.apache.org/docs/trunk/platform/windows.html) have this to say about the slashes: "The directives that accept filenames as arguments must use Windows filenames instead of Unix ones. However, because Apache may interpret backslashes as an "escape character" sequence, you should consistently use forward slashes in path names, not backslashes."

- Morten

Sergei said...

Hi.
With java -jar ords.war install advanced, I can configure my remote database connection with port 1521. But how do I configure the ords connection to an SSL Listener via port 2484 with oracle wallet?

Morten Braten said...

@Sergei: Perhaps somebody on the ORDS discussion forum can help you with advanced ORDS setup:

https://community.oracle.com/community/database/developer-tools/oracle_rest_data_services/content

- Morten

Anonymous said...

Hello!

Great tutorial!

In section # enable client caching of static content

Will performance improve with
FileETag none ?

Best regards Ola

Ade said...

Hello Morten

Great tutorial for the steps to learn Linux basics and APEX with Secure Apache Web Server install.

Instead of DigitalOcean, I provisioned my Cloud server using Upcloud.com CentOS 6.9. https://www.upcloud.com/vs/digitalocean/

After completing Part Two: Installation I can access the initial APEX Designer page using my server IP address and url :

http://83.136.255.138/ords/apex

I really want to access it using my server name

http://bsd.com/ords/apex

...but receive error : server IP address could not be found.

My hosts file entry is :

83.136.255.138 bsd.com bsd.com

Any assistance regarding server name access @Morten? or anybody else with successful installation?

Kind Regards
Ade

Morten Braten said...

@Ade: To use a domain name instead of your IP address, you must own/control the domain name, and you typically use a control panel at the provider where you bought the domain name to map it against an IP address. Ask you domain name provider for the specifics.

- Morten