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

net.sf.ehcache.terracotta.KeySnapshotter Maven / Gradle / Ivy

/**
 *  Copyright Terracotta, Inc.
 *
 *  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 net.sf.ehcache.terracotta;

import net.sf.ehcache.Cache;
import net.sf.ehcache.CacheManager;
import net.sf.ehcache.CacheStoreHelper;
import net.sf.ehcache.Ehcache;
import net.sf.ehcache.store.Store;
import net.sf.ehcache.store.TerracottaStore;
import net.sf.ehcache.util.WeakIdentityConcurrentMap;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.io.IOException;
import java.util.Collection;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.ScheduledFuture;
import java.util.concurrent.ScheduledThreadPoolExecutor;
import java.util.concurrent.TimeUnit;

/**
 * A class that will snapshot the local keySet of a Terracotta clustered cache to disk
 *
 * @author Alex Snaps
 */
class KeySnapshotter implements Runnable {

    private static final Logger LOG = LoggerFactory.getLogger(KeySnapshotter.class.getName());
    private static final int POOL_SIZE = Integer.getInteger("net.sf.ehcache.terracotta.KeySnapshotter.threadPoolSize", 10);

    private static final WeakIdentityConcurrentMap INSTANCES =
        new WeakIdentityConcurrentMap(
            new WeakIdentityConcurrentMap.CleanUpTask() {
                public void cleanUp(final ScheduledExecutorService executor) {
                    executor.shutdownNow();
                }
            });

    private final String cacheName;
    private volatile TerracottaStore tcStore;
    private final RotatingSnapshotFile rotatingWriter;
    private final Thread thread;

    private volatile Runnable onSnapshot;
    private final ScheduledFuture scheduledFuture;

    /**
     * Default Constructor
     *
     * @param cache                          the Terracotta clustered Cache to snapshot
     * @param interval                       the interval to do the snapshots on
     * @param doKeySnapshotOnDedicatedThread whether the snapshots have to be done on a dedicated thread
     * @param rotatingWriter                 the RotatingSnapshotFile to write to
     * @throws IllegalArgumentException if interval is less than or equal to zero
     */
    KeySnapshotter(final Ehcache cache, final long interval,
                   final boolean doKeySnapshotOnDedicatedThread,
                   final RotatingSnapshotFile rotatingWriter)
        throws IllegalArgumentException {
        final Store store = new CacheStoreHelper((Cache)cache).getStore();
        if (!(store instanceof TerracottaStore)) {
            throw new IllegalArgumentException("Cache '" + cache.getName() + "' isn't backed by a " + TerracottaStore.class.getSimpleName()
                                               + " but uses a " + store.getClass().getName() + " instead");
        }

        if (interval <= 0) {
            throw new IllegalArgumentException("Interval needs to be a positive & non-zero value");
        }

        if (rotatingWriter == null) {
            throw new NullPointerException();
        }

        this.cacheName = cache.getName();
        this.rotatingWriter = rotatingWriter;
        this.tcStore = (TerracottaStore)store;

        if (doKeySnapshotOnDedicatedThread) {
            scheduledFuture = null;
            thread = new SnapShottingThread(this, interval, "KeySnapshotter for cache " + cacheName);
            thread.start();
        } else {
            scheduledFuture = getScheduledExecutorService(cache.getCacheManager())
                .scheduleWithFixedDelay(this, interval, interval, TimeUnit.SECONDS);
            thread = null;
        }
    }

    private ScheduledExecutorService getScheduledExecutorService(final CacheManager cacheManager) {
        ScheduledExecutorService scheduledExecutorService = INSTANCES.get(cacheManager);
        if (scheduledExecutorService == null) {
            scheduledExecutorService = new ScheduledThreadPoolExecutor(POOL_SIZE);
            final ScheduledExecutorService previous = INSTANCES.putIfAbsent(cacheManager, scheduledExecutorService);
            if (previous != null) {
                scheduledExecutorService.shutdownNow();
                scheduledExecutorService = previous;
            }
        }
        return scheduledExecutorService;
    }

    /**
     * Shuts down the writer thread and cleans up resources
     *
     * @param immediately whether to leave the writer finish or shut down immediately
     */
    void dispose(boolean immediately) {
        if (thread != null) {
            rotatingWriter.setShutdownOnThreadInterrupted(immediately);
            thread.interrupt();
            try {
                thread.join();
            } catch (InterruptedException e) {
                Thread.currentThread().interrupt();
            }
        } else {
            scheduledFuture.cancel(immediately);
        }
        tcStore = null;
    }

    /**
     * {@inheritDoc}
     */
    public void run() {
        try {
            INSTANCES.cleanUp();
            rotatingWriter.writeAll(tcStore.getLocalKeys());
            onSnapshot();
        } catch (Throwable e) {
            LOG.error("Couldn't snapshot local keySet for Cache {}", cacheName, e);
        }
    }

    private void onSnapshot() {
        if (onSnapshot != null) {
            try {
                onSnapshot.run();
            } catch (Exception e) {
                LOG.warn("Error occurred in onSnapshot callback", e);
            }
        }
    }

    /**
     * Accessor to all known cacheManagers (which are also bound to a ScheduledExecutorService)
     *
     * @return the collection of known CacheManagers
     */
    static Collection getKnownCacheManagers() {
        return INSTANCES.keySet();
    }

    /**
     * Calling this method will result in a snapshot being taken or wait for the one in progress to finish
     *
     * @throws IOException On exception being thrown while doing the snapshot
     */
    void doSnapshot() throws IOException {
        rotatingWriter.snapshotNowOrWaitForCurrentToFinish(tcStore.getLocalKeys());
        onSnapshot();
    }

    /**
     * Let register a Runnable that will be called on every snapshot happening
     * @param onSnapshot the runnable
     */
    void setOnSnapshot(final Runnable onSnapshot) {
        this.onSnapshot = onSnapshot;
    }

    /**
     * Returns the name of the underlying cache for which this snapshots
     * @return The name of the cache
     */
    public String getCacheName() {
        return cacheName;
    }

    /**
     * Thread doing background snapshots of the local key set
     */
    private static class SnapShottingThread extends Thread {

        private long lastRun;
        private final long interval;

        public SnapShottingThread(final Runnable runnable, final long interval, final String threadName) {
            super(runnable, threadName);
            this.interval = interval;
            lastRun = System.currentTimeMillis();
            this.setDaemon(true);
        }

        @Override
        public void run() {
            while (!isInterrupted()) {
                final long nextTime = lastRun + TimeUnit.SECONDS.toMillis(interval);
                final long now = System.currentTimeMillis();
                if (nextTime <= now) {
                    super.run();
                    lastRun = System.currentTimeMillis();
                } else {
                    try {
                        sleep(nextTime - now);
                    } catch (InterruptedException e) {
                        interrupt();
                    }
                }
            }
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy