All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.googlecode.jinahya.sql.SequenceManager Maven / Gradle / Ivy

The newest version!
/*
 * Copyright 2011 Jin Kwon .
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */


package com.googlecode.jinahya.sql;


import java.sql.Connection;
import java.sql.SQLException;

import java.util.Collections;
import java.util.HashMap;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;

import javax.sql.DataSource;


/**
 * Abstract SequenceGenerator implementation.
 *
 * @author Jin Kwon 
 */
public abstract class SequenceManager {


    /**
     * Creates a new instance.
     *
     * @param dataSource data source
     * @param minimumSize minimum count
     * @param maximumSize maximum count
     */
    public SequenceManager(final DataSource dataSource, final int minimumSize,
                           final int maximumSize) {
        super();

        if (dataSource == null) {
            throw new NullPointerException("null dataSource");
        }

        if (minimumSize <= 0) {
            throw new IllegalArgumentException(
                "minimumSize(" + minimumSize + ") <= 0");
        }

        if (maximumSize <= 0) {
            throw new IllegalArgumentException(
                "maximumSize(" + maximumSize + ") <=0");
        }

        if (minimumSize >= maximumSize) {
            throw new IllegalArgumentException(
                "minimumSize(" + maximumSize + ") >= maximumSize("
                + maximumSize + ")");
        }

        this.dataSource = dataSource;
        this.minimumSize = minimumSize;
        this.maximumSize = maximumSize;

        map = Collections.synchronizedMap(new HashMap>());
    }


    /**
     * Returns next id value identified by given name.
     *
     * @param sequenceName sequence name
     * @return next sequence value
     * @throws SQLException if an SQL error occurs.
     */
    public Long getNextValue(final String sequenceName) throws SQLException {

        synchronized (map) {

            if (!map.containsKey(sequenceName)) {
                map.put(sequenceName, new LinkedList());
            }

            final List list = map.get(sequenceName);

            if (list.size() < minimumSize) {
                final Connection connection = dataSource.getConnection();
                try {
                    fetchNextValues(connection, sequenceName, list,
                                    maximumSize - list.size());
                } finally {
                    connection.close();
                }
            }

            return list.remove(0);
        }
    }


    /**
     * Fetch sequence values.
     *
     * @param connection connection
     * @param sequenceName sequence name
     * @param sequenceValues sequence value list
     * @param fetchCount number of value to fetch
     * @throws SQLException if an SQL error occurs.
     */
    protected abstract void fetchNextValues(Connection connection,
                                            String sequenceName,
                                            List sequenceValues,
                                            int fetchCount)
        throws SQLException;


    /**
     * Returns a Connection.
     *
     * @return the Connection
     * @throws SQLException if an SQL error occurs.
     */
    protected final Connection getConnection() throws SQLException {
        return dataSource.getConnection();
    }


    /**
     * Returns minimumSize.
     *
     * @return minimumSize
     */
    public final int getMinimumSize() {
        return minimumSize;
    }


    /**
     * Returns maximumSize.
     *
     * @return maximumSize
     */
    public final int getMaximumSize() {
        return maximumSize;
    }


    /** data source. */
    private final DataSource dataSource;


    /** min count. */
    private final int minimumSize;


    /** max count. */
    private final int maximumSize;


    /** map. */
    private final Map> map;


}





© 2015 - 2025 Weber Informatics LLC | Privacy Policy