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

com.redhat.lightblue.crud.mongo.MongoSequenceGenerator Maven / Gradle / Ivy

The newest version!
/*
 Copyright 2013 Red Hat, Inc. and/or its affiliates.

 This file is part of lightblue.

 This program is free software: you can redistribute it and/or modify
 it under the terms of the GNU General Public License as published by
 the Free Software Foundation, either version 3 of the License, or
 (at your option) any later version.

 This program is distributed in the hope that it will be useful,
 but WITHOUT ANY WARRANTY; without even the implied warranty of
 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 GNU General Public License for more details.

 You should have received a copy of the GNU General Public License
 along with this program.  If not, see .
 */
package com.redhat.lightblue.crud.mongo;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import com.mongodb.BasicDBObject;
import com.mongodb.DBCollection;
import com.mongodb.DBObject;
import com.mongodb.WriteConcern;

/**
 * Sequence generation using a MongoDB collection. 
 *
 * Each sequence is a document uniquely identified by the sequence name. The document contains
 * initial value for the sequence, the increment, and the value.
 */
public class MongoSequenceGenerator {

    public static final String NAME="name";
    public static final String INIT="initialValue";
    public static final String INC="increment";
    public static final String VALUE="value";

    private static final Logger LOGGER=LoggerFactory.getLogger(MongoSequenceGenerator.class);
    
    private DBCollection coll;

    public MongoSequenceGenerator(DBCollection coll) {
        this.coll=coll;
    }

    private void initIndex() {
        // Make sure we have a unique index on name
        BasicDBObject keys=new BasicDBObject(NAME,1);
        BasicDBObject options=new BasicDBObject("unique",1);
        coll.ensureIndex(keys,options);
    }

    /**
     * Atomically increments and returns the sequence value. If this
     * is the first use of the sequence, the sequence is created
     *
     * @param name The sequence name
     * @param init The initial value of the sequence. Used only if
     * the sequence does not exists prior to this call
     * @param inc The increment, Could be negative or positive. If 0,
     * it is assumed to be 1. Used only if the sequence does not exist
     * prior to this call
     *
     * If the sequence already exists, the init and
     * inc are ignored.
     *
     * @return The value of the sequence before the call
     */
    public long getNextSequenceValue(String name,long init,long inc) {
        LOGGER.debug("getNextSequenceValue({})",name);
        // Read the sequence document
        BasicDBObject q=new BasicDBObject(NAME,name);
        DBObject doc=coll.findOne(q);
        if(doc==null) {
            // Sequence document does not exist. Insert a new document using the init and inc
            LOGGER.debug("inserting sequence record name={}, init={}, inc={}",name,init,inc);
            if(inc==0)
                inc=1;
            // Here, we also make sure we have the indexes setup properly
            initIndex();
            BasicDBObject u=new BasicDBObject().
                append(NAME,name).
                append(INIT,init).
                append(INC,inc).
                append(VALUE,init);
            try {
                coll.insert(u, WriteConcern.SAFE);
            } catch (Exception e) {
                // Someone else might have inserted already, try to re-read
                LOGGER.debug("Insertion failed with {}, trying to read",e);
            }
            doc=coll.findOne(q);
            if(doc==null)
                throw new RuntimeException("Cannot generate value for "+name);
        }
        LOGGER.debug("Sequence doc={}",doc);
        Long increment=(Long)doc.get(INC);
        BasicDBObject u=new BasicDBObject().
            append("$inc",new BasicDBObject(VALUE,increment));
        // This call returns the unmodified document
        doc=coll.findAndModify(q,u);
        Long l=(Long)doc.get(VALUE);
        LOGGER.debug("{} -> {}",name,l);
        return l;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy