com.hazelcast.sql.impl.plan.cache.PlanCache Maven / Gradle / Ivy
/*
* Copyright 2024 Hazelcast Inc.
*
* Licensed under the Hazelcast Community License (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://hazelcast.com/hazelcast-community-license
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.hazelcast.sql.impl.plan.cache;
import com.hazelcast.sql.impl.optimizer.PlanCheckContext;
import com.hazelcast.sql.impl.optimizer.PlanKey;
import com.hazelcast.sql.impl.optimizer.SqlPlan;
import java.util.TreeMap;
import java.util.concurrent.ConcurrentHashMap;
/**
* Cache for plans.
*/
public class PlanCache implements CachedPlanInvalidationCallback {
private final int maxSize;
private final ConcurrentHashMap plans = new ConcurrentHashMap<>();
public PlanCache(int maxSize) {
assert maxSize > 0;
this.maxSize = maxSize;
}
public SqlPlan get(PlanKey key) {
SqlPlan plan = plans.get(key);
if (plan != null) {
plan.onPlanUsed();
return plan;
} else {
return null;
}
}
public void put(PlanKey key, SqlPlan plan) {
plan.onPlanUsed();
plans.put(key, plan);
shrinkIfNeeded();
}
public void invalidate(SqlPlan plan) {
remove(plan);
}
public void clear() {
plans.clear();
}
public int size() {
return plans.size();
}
public void check(PlanCheckContext context) {
plans.values().removeIf(plan -> !plan.isPlanValid(context));
}
private void shrinkIfNeeded() {
int oversize = plans.size() - maxSize;
if (oversize <= 0) {
return;
}
// Sort plans according to their last used timestamps
TreeMap sorted = new TreeMap<>();
for (SqlPlan plan : plans.values()) {
sorted.put(plan.getPlanLastUsed(), plan);
}
// Remove oldest plans
for (SqlPlan plan : sorted.values()) {
boolean removed = remove(plan);
if (removed) {
if (--oversize == 0) {
break;
}
}
}
}
/**
* Removes the plan from the cache, decreasing the size.
*
* @param plan Plan.
*/
private boolean remove(SqlPlan plan) {
return plans.remove(plan.getPlanKey(), plan);
}
/**
* For testing only.
*/
public ConcurrentHashMap getPlans() {
return plans;
}
}
© 2015 - 2024 Weber Informatics LLC | Privacy Policy