Package org.openid4java.server

Source Code of org.openid4java.server.JdbcServerAssociationStore

/*
* Copyright 2006-2008 Sxip Identity Corporation
*/

package org.openid4java.server;

import org.openid4java.association.Association;
import org.openid4java.association.AssociationException;

import java.util.*;
import org.springframework.jdbc.core.support.JdbcDaoSupport;
import org.springframework.jdbc.core.JdbcTemplate;
import org.springframework.dao.DataAccessException;
import org.springframework.dao.IncorrectResultSizeDataAccessException;
import org.apache.commons.codec.binary.Base64;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;

/**
* JDBC implementation for the ServerAssociationStore interface.
* <p>
* The JdbcServerAssociation store requires a javax.sql.DataSource to be
* configured and passed in to it with the setDataSource setter method.
* The table name also needs to be specified, either through the constructor,
* or through the setTableName setter.
* <p>
* The specified table must have the following structure:
* <ul>
* <li>handle : string : primary key</li>
* <li>type : string</li>
* <li>mackey : string</li>
* <li>expdate : date</li>
* </ul>
*
* @author Marius Scurtescu, Johnny Bufu
*/
public class JdbcServerAssociationStore extends JdbcDaoSupport
        implements ServerAssociationStore
{
    private static Log _log = LogFactory.getLog(JdbcServerAssociationStore.class);
    private static final boolean DEBUG = _log.isDebugEnabled();

    private static Random _random = new Random(System.currentTimeMillis());

    private static final int CLEANUP_INTERVAL = 60 * 1000; // 1 min in millis
    private static long _lastCleanup = 0;

    private String _tableName;

    public JdbcServerAssociationStore()
    {
    }

    public JdbcServerAssociationStore(String tableName)
    {
        _tableName = tableName;
    }

    public String getTableName()
    {
        return _tableName;
    }

    public void setTableName(String tableName)
    {
        this._tableName = tableName;
    }

    public Association generate(String type, int expiryIn)
            throws AssociationException
    {
        cleanupExpired();
       
        String sql = "INSERT INTO " + _tableName +
                " (handle, type, mackey, expdate) VALUES (?,?,?,?)";

        JdbcTemplate jdbcTemplate = getJdbcTemplate();

        int attemptsLeft = 5;

        while (attemptsLeft > 0)
        {
            try
            {
                String handle = Long.toHexString(_random.nextLong());

                Association association =
                        Association.generate(type, handle, expiryIn);

                int cnt = jdbcTemplate.update(sql,
                        new Object[] {
                                association.getHandle(),
                                association.getType(),
                                new String(Base64.encodeBase64(
                                        association.getMacKey().getEncoded())),
                                association.getExpiry()
                        });

                if (cnt == 1)
                {
                    if (DEBUG)
                        _log.debug("Generated association, handle: " + handle +
                                   " type: " + type +
                                   " expires in: " + expiryIn + " seconds.");

                    return association;
                }
            }
            catch (DataAccessException e)
            {
                _log.error("Error generating association; attempts left: "
                           + (attemptsLeft-1), e);
            }

            attemptsLeft--;
        }

        throw new AssociationException(
                "JDBCServerAssociationStore: Error generating association.");
    }

    public Association load(String handle)
    {
        try
        {
            String sql = "SELECT type,mackey,expdate FROM " + _tableName +
                    " WHERE handle=?";

            JdbcTemplate jdbcTemplate = getJdbcTemplate();

            Map res = jdbcTemplate.queryForMap(sql, new Object[] {handle});

            String type = (String) res.get("type");
            String macKey = (String) res.get("mackey");
            Date expDate = (Date) res.get("expdate");

            if (type == null || macKey == null || expDate == null)
                throw new AssociationException("Invalid association data " +
                        "retrived from database; cannot create Association " +
                        "object for handle: " + handle);

            Association assoc;

            if (Association.TYPE_HMAC_SHA1.equals(type))
                assoc = Association.createHmacSha1(handle,
                        Base64.decodeBase64(macKey.getBytes() ), expDate);

            else if (Association.TYPE_HMAC_SHA256.equals(type))
                assoc = Association.createHmacSha256(handle,
                        Base64.decodeBase64(macKey.getBytes() ), expDate);

            else
                throw new AssociationException("Invalid association type " +
                        "retrieved from database: " + type);

            if (DEBUG)
                _log.debug("Retrieved association for handle: " + handle +
                           " from table: " + _tableName);

            return assoc;
        }
        catch (AssociationException ase )
        {
            _log.error("Error retrieving association from table: " + _tableName, ase);
            return null;
        }
        catch (IncorrectResultSizeDataAccessException rse)
        {
            _log.warn("Association not found for handle: " + handle +
                      " in the table: " + _tableName);
            return null;
        }
        catch (DataAccessException dae)
        {
            _log.error("Error retrieving association for handle: " + handle +
                       "from table: " + _tableName, dae);
            return null;
        }
    }

    public void remove(String handle)
    {
        try
        {
            String sql = "DELETE FROM " + _tableName + " WHERE handle=?";

            JdbcTemplate jdbcTemplate = getJdbcTemplate();

            int cnt = jdbcTemplate.update(sql, new Object[] { handle } );

            if (cnt == 1 && DEBUG)
                _log.debug("Removed association, handle: " + handle +
                           " from table: " + _tableName);

            if (cnt != 1)
                _log.warn("Trying to remove handle: " + handle + " from table: "
                          + _tableName + "; affected entries: " + cnt);
        }
        catch (Exception e)
        {
            _log.error("Error removing association from table: " + _tableName, e);
        }
    }

    private void cleanupExpired()
    {
        if (System.currentTimeMillis() - _lastCleanup < CLEANUP_INTERVAL)
            return;

        try
      {
        String sql = "DELETE FROM " + _tableName + " WHERE expdate<?";

            JdbcTemplate jdbcTemplate = getJdbcTemplate();

            Date now = new Date ( ) ;
            int cnt = jdbcTemplate.update(sql, new Object[] { now } );

            _log.debug("Cleaned " + cnt + " expired associations from table: "
                          + _tableName);

            _lastCleanup = System.currentTimeMillis();
        }
      catch (Exception e)
        {
            _log.error("Error removing expired associations from table: " + _tableName, e);
        }
    }
}
TOP

Related Classes of org.openid4java.server.JdbcServerAssociationStore

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.