com.regnosys.rosetta.common.model.CachingMethodInterceptor Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of rosetta-common Show documentation
Show all versions of rosetta-common Show documentation
Rune Common is a java library that is utilised by Rosetta Code Generators and models expressed in the Rosetta DSL.
package com.regnosys.rosetta.common.model;
/*-
* ==============
* Rune Common
* ==============
* Copyright (C) 2018 - 2024 REGnosys
* ==============
* 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.
* ==============
*/
import com.google.common.cache.Cache;
import com.google.common.cache.CacheBuilder;
import org.aopalliance.intercept.MethodInterceptor;
import org.aopalliance.intercept.MethodInvocation;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.util.Optional;
import java.util.Set;
public class CachingMethodInterceptor implements MethodInterceptor {
private static final Logger LOGGER = LoggerFactory.getLogger(CachingMethodInterceptor.class);
private static final Object NULL = new Object() {
@Override
public String toString() {
return "null";
}
};
private final Cache memoiseCache;
private final Set debugFunctions;
public CachingMethodInterceptor(CacheBuilder cacheBuilder, Set debugFunctions) {
this.memoiseCache = cacheBuilder.build();
this.debugFunctions = debugFunctions;
}
@Override
public Object invoke(MethodInvocation invocation) throws Throwable {
MemoiseCacheKey key = MemoiseCacheKey.create(invocation.getMethod().toString(), invocation.getArguments());
boolean debugLoggingEnabled = isDebugLoggingEnabled(invocation);
Object ifPresent = memoiseCache.getIfPresent(key);
if (ifPresent == null) {
Object invoked = invocation.proceed();
log(debugLoggingEnabled, "Executed function", invocation, invoked);
if (invoked == null) {
memoiseCache.put(key, NULL);
} else {
memoiseCache.put(key, invoked);
}
return invoked;
}
log(debugLoggingEnabled, "Cached function", invocation, ifPresent);
if (ifPresent == NULL) {
return null;
}
return ifPresent;
}
// /* This method has slightly different null handling - experimental */
// public Object invoke(MethodInvocation invocation) throws Throwable {
// MemoiseCacheKey key = MemoiseCacheKey.create(invocation.getMethod().toString(), invocation.getArguments());
// boolean debugLoggingEnabled = isDebugLoggingEnabled(invocation);
//
// Object ifPresent = memoiseCache.getIfPresent(key);
// if (ifPresent == null) {
// Object functionResult = Optional.ofNullable(invocation.proceed()).orElse(NULL);
// log(debugLoggingEnabled, "Executed function", invocation, functionResult);
// memoiseCache.put(key, functionResult);
// return functionResult;
// }
// log(debugLoggingEnabled, "Cached function", invocation, ifPresent);
// if (ifPresent == NULL) {
// return null;
// }
// return ifPresent;
// }
private boolean isDebugLoggingEnabled(MethodInvocation invocation) {
return debugFunctions.contains(invocation.getMethod().getDeclaringClass().getSimpleName().toUpperCase());
}
private static void log(boolean debugLoggingEnabled, String message, MethodInvocation invocation, Object functionResult) {
if (debugLoggingEnabled) {
LOGGER.debug("{} '{}' Inputs[{}] Output[{}]",
message,
invocation.getMethod().getDeclaringClass().getSimpleName(),
invocation.getArguments(),
functionResult);
}
}
}