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

com.spotify.metrics.core.MetricIdCache Maven / Gradle / Ivy

/*
 * Copyright (c) 2016 Spotify AB.
 *
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you 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.spotify.metrics.core;

import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;
import java.util.concurrent.ExecutionException;

// @formatter:off
/**
 * 

A utility class that simplifies mutating and storing a MetricId.

* *

Example

* *
 * // example using Guava.
 * final MetricIdCache.CacheBuilder cacheBuilder = new MetricIdCache.CacheBuilder() {
 *     public  ConcurrentMap build() {
 *         return CacheBuilder.newBuilder()
 *             .expireAfterWrite(12, TimeUnit.HOURS).build().asMap()
 *     }
 * };
 *
 * final MetricIdCache.Builder builder = MetricIdCache.builder().cacheBuilder(cacheBuilder);
 *
 * final MetricIdCache cache = builder.metricId(MetricId.build().tagged("foo", "bar"))
 * .loader(new MetricIdCache.Loader() {
 *     public MetricId load(MetricId base, String key) {
 *         base.tagged("key", key);
 *     }
 * }).build();
 *
 * cache.get("hello"); // MetricId(foo=bar, key=hello)
 * cache.get("world"); // MetricId(foo=bar, key=world)
 *
 * // subsequent calls with cache.get("hello"), or cache.get("world") will return the same
 * MetricId instance.
 * 
* * @author udoprog * @param The key type for loading the metric id. */ // @formatter:on public class MetricIdCache { private final Cache cache; private final MetricId metricId; private MetricIdCache(Cache cache, MetricId metricId) { this.cache = cache; this.metricId = metricId; } public static Any builder() { return new Any(); } /** * Get the value for the specified key. *

* This method is guaranteed to return the same instance of the associated MetricId, regardless * of how many threads invoke it. *

* The configured Loader could be called multiple times, and should be idempotent to that. * * @param key The given key to get MetricId for. * @return A metric id for the given key. * @throws IllegalStateException if loader returns {@code null}. */ public MetricId get(T key) { try { return cache.get(metricId, key); } catch (ExecutionException e) { throw new RuntimeException("failed to get cache value", e); } } public void invalidate(T key) { cache.invalidate(key); } public void invalidateAll() { cache.invalidateAll(); } public static interface Loader { public MetricId load(MetricId id, T key); } public static interface Cache { public MetricId get(MetricId metricId, T key) throws ExecutionException; public void invalidate(T key); public void invalidateAll(); } public static interface MapBuilder { public ConcurrentMap build(); } public static interface CacheBuilder { public Cache build(Loader loader); } private static final class ConcurrentMapCache implements Cache { private final ConcurrentMap cache; private final Loader loader; private ConcurrentMapCache(ConcurrentMap cache, Loader loader) { this.cache = cache; this.loader = loader; } @Override public MetricId get(MetricId metricId, T key) { final MetricId candidate = cache.get(key); if (candidate != null) { return candidate; } final MetricId addition = loader.load(metricId, key); if (addition == null) { throw new IllegalStateException("loader returned null value"); } final MetricId put; if ((put = cache.putIfAbsent(key, addition)) != null) { return put; } return addition; } @Override public void invalidate(T key) { cache.remove(key); } @Override public void invalidateAll() { cache.clear(); } } private interface TypedCacheBuilder { Cache build(Loader loader); } /** * A builder for which the metric id cache has a given type. * * @param The type of the metric id cache. */ public static class Typed { private final Any any; private final Cache cache; private final Loader loader; /* internal */ private final TypedCacheBuilder typedCacheBuilder; private Typed( Any any, Cache cache, Loader loader, TypedCacheBuilder typedCacheBuilder ) { this.any = any; this.cache = cache; this.loader = loader; this.typedCacheBuilder = typedCacheBuilder; } /** * Configure a base metric id. * * @see Any#metricId(MetricId) */ public Typed metricId(MetricId base) { return any(any.metricId(base)); } /** * Use a shared cache. */ public Typed cache(Cache cache) { if (cache == null) { throw new IllegalArgumentException("'cache' must not be null"); } return new Typed(any, cache, loader, typedCacheBuilder); } /** * Allows an unbounded cache. *

* This could leak memory, since the provided cache implementation will never expire * entries. * * @see Any#unbounded(Loader) */ public Typed unbounded(Loader loader) { return cache(new ConcurrentHashMap(), loader); } /** * Configure a specific cache instance, use this if you wish for the consequent builders to * re-use the same cache instance. */ public Typed cache(final ConcurrentMap map, Loader loader) { if (map == null) { throw new IllegalArgumentException("'cache' must not be null"); } return loader(loader).typedCacheBuilder(new TypedCacheBuilder() { @Override public Cache build(Loader loader) { return new ConcurrentMapCache(map, loader); } }); } public Typed cacheBuilder(CacheBuilder cacheBuilder) { return any(any.cacheBuilder(cacheBuilder)); } public Typed mapBuilder(MapBuilder mapBuilder, Loader loader) { return loader(loader).any(any.mapBuilder(mapBuilder)); } public Typed mapBuilder(MapBuilder mapBuilder) { return any(any.mapBuilder(mapBuilder)); } public Typed loader(Loader loader) { if (cache != null) { throw new IllegalStateException("shared cache is already set"); } return new Typed(any, cache, loader, typedCacheBuilder); } /** * Build a cache with the given MetricId base. */ public MetricIdCache build() { final Cache cache = any.buildCache(this.cache, this.loader, this.typedCacheBuilder); return new MetricIdCache(cache, any.buildMetricId()); } private Typed typedCacheBuilder(TypedCacheBuilder typedCacheBuilder) { return new Typed(any, cache, loader, typedCacheBuilder); } private Typed any(Any any) { return new Typed<>(any, cache, loader, typedCacheBuilder); } } /** * A metric id cache builder that can take any type. *

* Calling one of the type-specific configuration options will returned a regular, typed * Builder. * * @author udoprog */ public static class Any { private final MetricId metricId; private final CacheBuilder cacheBuilder; private Any(MetricId base, CacheBuilder cacheBuilder) { this.metricId = base; this.cacheBuilder = cacheBuilder; } private Any() { this(null, null); } /** * Set the given base metric id for the cache. */ public Any metricId(MetricId metricId) { if (metricId == null) { throw new IllegalArgumentException("'metricId' must not be null"); } return new Any(metricId, cacheBuilder); } public Typed unbounded(Loader loader) { return new Typed(this, null, null, null).unbounded(loader); } public Typed cache(Cache cache) { return new Typed(this, null, null, null).cache(cache); } public Typed cache(ConcurrentMap cache, Loader loader) { return new Typed(this, null, null, null).cache(cache, loader); } public Typed loader(Loader loader) { return new Typed(this, null, null, null).loader(loader); } public Typed mapBuilder(MapBuilder mapBuilder, Loader loader) { return mapBuilder(mapBuilder).loader(loader); } public Any cacheBuilder(CacheBuilder cacheBuilder) { if (cacheBuilder == null) { throw new IllegalArgumentException("'cacheBuilder' must not be null"); } return new Any(metricId, cacheBuilder); } public Any mapBuilder(final MapBuilder mapBuilder) { if (mapBuilder == null) { throw new IllegalArgumentException("'mapBuilder' must not be null"); } return new Any(metricId, new CacheBuilder() { @Override public Cache build(Loader loader) { final ConcurrentMap map = mapBuilder.build(); if (map == null) { throw new IllegalStateException("'mapBuilder' must not return null"); } return new ConcurrentMapCache(map, loader); } }); } private Cache buildCache( Cache cache, Loader loader, TypedCacheBuilder typedCacheBuilder ) { if (cache != null) { return cache; } if (loader == null) { throw new IllegalStateException("'loader' must be set"); } if (cacheBuilder != null) { final Cache c = cacheBuilder.build(loader); if (c == null) { throw new IllegalStateException("'cacheBuilder' must return non-null values"); } return c; } if (typedCacheBuilder != null) { return typedCacheBuilder.build(loader); } throw new IllegalStateException("No cache implementation is configured"); } private MetricId buildMetricId() { if (metricId == null) { return MetricId.EMPTY; } return metricId; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy