Thursday, January 20, 2011

Connecting to a Remote Oracle Database with PHP and the Oracle Instant Client

Connecting to a remote Oracle database via PHP on Windows is relatively straightforward. Keep in mind I'm stating that after having gone through the process a couple of times.

First off, you'll need to download the Oracle Instant Client. I downloaded it directly from this page after agreeing to some licensing mumbo jumbo. I went with the Instant Client Package - Basic Lite package since I wasn't interested in non-English error messages and the like. I'd also recommend downloading Instant Client Package - SQL*Plus from that same location, because it's a good way to do a sanity check before you move on to the PHP-specific stuff.

Once you've downloaded those packages, unzip them both to the same directory. I chose C:\Program Files\Oracle Instant Client, so I'll be referring to that location for the remainder of these instructions. When you're done, you should\ have a series of .dll files and a few others, along with the SQL*Plus executable, sqlplus.exe. Next thing you'll want to do is create a new file in that same directory named tnsnames.ora. This is an important configuration file that allows you to create kind of a local reference to a remote Oracle database server. Paste the following into that new file:

RDF = (DESCRIPTION =
(ADDRESS_LIST =
(ADDRESS = (PROTOCOL = TCP)(HOST = REMOTESERVER)(PORT = 1521))
)
(CONNECT_DATA =
(SERVICE_NAME = LOCALNAME)
)
)

Where REMOTESERVER is the name of the remote server that has Oracle running on it and LOCALNAME is the local name by which you wish to refer to the connection with that server.

This is where the sanity check with SQL*Plus comes in. Open up a Windows command prompt and change to the directory where you unzipped all your Oracle Instant Client files (e.g. C:\Program Files\Oracle Instant Client). From there, run the following command to test your setup so far:

sqlplus username/password@LOCALNAME

Where username is a user account with access to the remote Oracle server, password is the password for that user account, and LOCALNAME is, you guessed it, the local name by which you've chosen to refer to the connection with the remote server. If you're presented with an SQL> prompt at this point, you're good to go, knowing that your Oracle connection is in a good state and you're ready to move on to the PHP setup.

To get your PHP scripts talking to the remote Oracle server, you'll use the oci_connect() function. Before you can do that, though, you'll need to enable support for the oci8 module. To do this, open up your php.ini file and uncomment (remove the leading semicolon from) the line that reads:

;extension=php_oci8.dll

Once you've done that, you'll need to update your computer's environment variables. First, modify your PATH environment variable so that it has a reference to C:\Program Files\Oracle Instant Client. Then, create a new environment variable named TNS_ADMIN that points to the same directory. This environment variable ensures that PHP will be able to find your tnsnames.ora file when it tries to connect via oci_connect(), so it's important. Once you've set these environment variables correctly, it's probably a good idea to restart the machine your web server is running on. I had to do this before the changes would register (you can easily check your path if you make a call to phpinfo() from a script). However, it might be possible to only restart Apache (or whatever web server you're running), so you might give that a shot first just to see if you're lucky.

At this point, you should be ready to start connecting to the remote Oracle server from your PHP scripts. To make sure, you can run the following script, keeping in mind that the settings should match what you've set in the steps above:

// try connecting to the database
$conn = oci_connect('username', 'password', 'LOCALNAME');

// check for any errors
if (!$conn)
{
$e = oci_error();
print htmlentities($e['message']);
exit;
}

// else there weren't any errors
else
{
echo 'I am an Oracle mack daddy.';
}

This is all off the top of my head after going through this process a couple times. If your mileage varies and you think I missed something, be sure to leave a comment and I'll see if I can't correct any mistakes.