Trust is a great force multiplier. – Tom Ridge
When you build a web application you have two choices for the pages: they can either be public or protected. Mostly you will build applications using a combination of the two. Some pages will be publicly available, like the landing page of your application, but there will also be pages which are protected. Users need to login to the application to use these pages.
Authentication methods
APEX provides us with a couple of authentication schemes by default. Some of which are:
- Application Express – Every user must exist as an APEX user
- Database Account – Every user must have a database account
- Open Door Credentials
The custom authentication is a good option so you can keep full control over how you want this to work. When you have created your application, or at least the start pages, you go to the shared components page where you go select the Authentication Schemes from the Security section.
Custom authentication scheme
When you want to create your own authentication scheme you must create a (packaged) function that must obey a few rules. The function must accept two parameters: the first is the username, and the second is the password. Both these parameters are varchar2 type. Remember, the username and password parameter are sent as clear text. If you want your application to be more secure, you may want to obfuscate the values before sending them to the authentication function. The simplest form of the authentication function is like this:
1 2 3 4 5 6 7 |
<span class="kwrd">FUNCTION</span> authenticate(username_in <span class="kwrd">IN</span> <span class="kwrd">VARCHAR2</span> ,password_in <span class="kwrd">IN</span> <span class="kwrd">VARCHAR2</span>) <span class="kwrd">RETURN</span> BOOLEAN <span class="kwrd">IS</span> <span class="kwrd">BEGIN</span> <span class="kwrd">RETURN</span> <span class="kwrd">TRUE</span>; <span class="kwrd">END</span> authenticate; |
This function is pretty much the same as the open door version, but it’s the start of a real authentication. You create a new authentication scheme by selecting the create button on the screen with the list of defined schemes for this application.
When you create a new scheme you can choose to create a scheme based on an existing scheme, but in this case we want to create a new scheme based on one of the pre-configured schemes. In this screen you can create the code for the Authentication Function. You can write the code for the function in this screen, but you can also create the function as a stored (package) function so you can use your IDE to create the code.
To create a real authentication scheme you need to do more than just return true for whatever parameters are sent in. You can write code to check the usernames and their passwords, but that would mean you would have to alter the code every time you want to add or remove a user. You are better off using a table which stores the users for the application. The table can look like this: Table USERS
ID | NUMBER(15,0) | Primary key |
USERNAME | VARCHAR2(50) | Unique constraint |
PASSWORD | VARCHAR2(50) |
You can this table to save more information on the user, such as the email address but for this example that is not necessary. The function could be updated to something like this:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 |
<span class="kwrd">FUNCTION</span> authenticate(username_in <span class="kwrd">IN</span> <span class="kwrd">VARCHAR2</span> ,password_in <span class="kwrd">IN</span> <span class="kwrd">VARCHAR2</span>) <span class="kwrd">RETURN</span> BOOLEAN <span class="kwrd">IS</span> l_value <span class="kwrd">NUMBER</span>; l_returnvalue BOOLEAN; <span class="kwrd">BEGIN</span> <span class="kwrd">BEGIN</span> <span class="kwrd">SELECT</span> 1 <span class="kwrd">IN</span>TO l_value <span class="kwrd">FROM</span> users <span class="kwrd">WHERE</span> 1 = 1 <span class="kwrd">AND</span> upper(users.username) = upper(username_in) <span class="kwrd">AND</span> upper(users.password) = upper(password_in); <span class="kwrd">EXCEPT</span>ION <span class="kwrd">WHEN</span> no_data_found <span class="kwrd">OR</span> too_many_rows <span class="kwrd">THEN</span> l_value := 0; <span class="kwrd">WHEN</span> <span class="kwrd">OTHER</span>S <span class="kwrd">THEN</span> l_value := 0; <span class="kwrd">END</span>; l_returnvalue := l_value = 1; <span class="kwrd">RETURN</span> l_returnvalue; <span class="kwrd">END</span>; |
In this example the username and password are validated against the data in the table. The username and password are stored in plain text in the database, which is not a good idea. You may want to obfuscate the password before storing it. This is where creating a package for the authentication functions comes in handy. You want to use the same function for obfuscating the password when you store the data as when you check the credentials. You can do this by creating a private function in the package that does the obfuscating and use the outcome of this function for both storing the data and checking the entered credentials. A package like this could look like this:
1 2 3 4 5 6 |
<span class="kwrd">PACKAGE</span> redgate_authentication <span class="kwrd">IS</span> <span class="kwrd">PROCEDURE</span> adduser(username_in <span class="kwrd">IN</span> <span class="kwrd">VARCHAR2</span> ,password_in <span class="kwrd">IN</span> VARCHAR2); <span class="kwrd">FUNCTION</span> authenticate(username_in <span class="kwrd">IN</span> <span class="kwrd">VARCHAR2</span> ,password_in <span class="kwrd">IN</span> VARCHAR2) <span class="kwrd">RETURN</span> BOOLEAN; <span class="kwrd">END</span> redgate_authentication; |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 |
<span class="kwrd">PACKAGE</span> <span class="kwrd">BODY</span> redgate_authentication <span class="kwrd">IS</span> <span class="rem">-- private functions </span> <span class="rem">/******************************************************************************\ || function : obfuscate || parameters : text_in -=> text to be obfuscated || || return value: obfuscated value || || purpose : Hash the value of text_in || || author : PBA || (C) 2013 : Patrick Barel \******************************************************************************/</span> <span class="kwrd">FUNCTION</span> obfuscate(text_in <span class="kwrd">IN</span> VARCHAR2) <span class="kwrd">RETURN</span> <span class="kwrd">RAW</span> <span class="kwrd">IS</span> l_returnvalue RAW(<span class="str">16</span>); <span class="kwrd">BEGIN</span> dbms_obfuscation_toolkit.md5(input <span class="kwrd">=></span> utl_raw.cast_to_raw(text_in), checksum <span class="kwrd">=></span> l_returnvalue); <span class="kwrd">RETURN</span> l_returnvalue; <span class="kwrd">END</span> obfuscate; <span class="rem">-- public functions </span> <span class="rem">/******************************************************************************\ || procedure : adduser || parameters : username_in -=> Username of the user to be authenticated || password_in -=> Password of the user to be authenticated || || purpose : Add a user to the users table || || author : PBA || (C) 2013 : Patrick Barel \******************************************************************************/</span> <span class="kwrd">PROCEDURE</span> adduser(username_in <span class="kwrd">IN</span> <span class="kwrd">VARCHAR2</span> ,password_in <span class="kwrd">IN</span> VARCHAR2) <span class="kwrd">IS</span> l_obfuscated_password users.password%TYPE; <span class="kwrd">BEGIN</span> l_obfuscated_password := obfuscate(text_in <span class="kwrd">=></span> password_in); <span class="kwrd">INSERT</span> <span class="kwrd">INTO</span> users (id ,username ,password) <span class="kwrd">VALUES</span> (users_seq.nextval ,username_in ,l_obfuscated_password); NULL; <span class="kwrd">END</span> adduser; <span class="rem">/******************************************************************************\ || function : authenticate || parameters : username_in -=> Username of the user to be authenticated || password_in -=> Password of the user to be authenticated || || return value: TRUE -=> User is authenticated || FALSE -=> User is not authenticated || || purpose : Check if a user is authenticated based on the username and || password supplied || || author : PBA || (C) 2013 : Patrick Barel \******************************************************************************/</span> <span class="kwrd">FUNCTION</span> authenticate(username_in <span class="kwrd">IN</span> <span class="kwrd">VARCHAR2</span> ,password_in <span class="kwrd">IN</span> VARCHAR2) <span class="kwrd">RETURN</span> BOOLEAN <span class="kwrd">IS</span> l_obfuscated_password users.password%TYPE; l_value NUMBER; l_returnvalue BOOLEAN; <span class="kwrd">BEGIN</span> l_obfuscated_password := obfuscate(text_in <span class="kwrd">=></span> password_in); <span class="kwrd">BEGIN</span> <span class="kwrd">SELECT</span> 1 <span class="kwrd">INTO</span> l_value <span class="kwrd">FROM</span> users <span class="kwrd">WHERE</span> 1 = 1 <span class="kwrd">AND</span> upper(users.username) = upper(username_in) <span class="kwrd">AND</span> users.password = l_obfuscated_password; EXCEPTION <span class="kwrd">WHEN</span> no_data_found <span class="kwrd">OR</span> too_many_rows <span class="kwrd">THEN</span> l_value := 0; <span class="kwrd">WHEN</span> OTHERS <span class="kwrd">THEN</span> l_value := 0; END; l_returnvalue := l_value = 1; <span class="kwrd">RETURN</span> l_returnvalue; <span class="kwrd">END</span> authenticate; <span class="kwrd">END</span> redgate_authentication; |
Now all you have to do is tell your application that it needs to reference the custom authentication schema. When you created the new authentication schema your application was automatically told to use the new schema, but if you created an authentication scheme in one application and you created a new application where you copied the authentication scheme from an existing application you have to do this yourself. It can also happen that during development you want to use a different authentication scheme than you might in production.
To change the current authentication scheme you go to the Shared components – Authentication Schemes. Select the scheme you want to use and press the ‘Make Current Scheme’ button.
If the scheme you selected is already the current scheme, then this button will not be available.
Load comments