com.squarespace.less.exec.SymbolTable Maven / Gradle / Ivy
/**
* Copyright (c) 2014 SQUARESPACE, 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 com.squarespace.less.exec;
import java.lang.reflect.Field;
import java.lang.reflect.Modifier;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Map;
import java.util.Set;
import com.squarespace.less.core.LessInternalException;
import com.squarespace.less.core.TypeRef;
/**
* Generic table mapping symbol strings to values. Raises an error when an attempt
* is made to register a symbol more than once.
*/
public abstract class SymbolTable {
/**
* Set which tracks the packages that have been registered to
* this instance, to detect accidental re-registrations.
*/
private final Set> packages;
/**
* Mapping of symbol to implementation.
*/
private final Map table;
/**
* Accessor for the generic type parameter.
*/
private final TypeRef typeRef;
/**
* Flag to indicate whether this symbol table is in use. If the symbol table
* is marked "in use", any attempt to modify it will throw an exception.
*/
private boolean inUse = false;
/**
* Constructs a table that accepts implementations of type V and sets the
* initial number of {@link HashMap} buckets.
*/
public SymbolTable(TypeRef typeRef, int numBuckets) {
this.packages = new HashSet<>();
this.table = new HashMap<>(numBuckets);
this.typeRef = typeRef;
}
/**
* Mark this table as being in use.
*/
public void setInUse() {
this.inUse = true;
}
/**
* Register all symbols discovered in the given package.
*
* @param pkg package to scan for implementations
*/
public void register(Registry pkg) {
registerByClass(pkg);
}
public V get(String symbol) {
return table.get(symbol);
}
/**
* Override extraction of the symbol name from the given implementation.
* @param implementation
*/
public abstract void registerSymbol(Object implementation);
/**
* Maps a symbol to its value.
*/
protected void put(String symbol, V value) {
if (inUse) {
throw new LessInternalException("Attempt to add a symbol after table in use.");
}
if (table.containsKey(symbol)) {
throw new LessInternalException("A symbol named '" + symbol + "' is already registered!");
}
table.put(symbol, value);
}
/**
* Registers a package of implementations.
*
* @param pkg package to scan for implementations
*/
private void registerByClass(Registry pkg) {
Class> cls = pkg.getClass();
if (!packages.add(cls)) {
throw new LessInternalException("package " + cls.getName() + " is already registered");
}
Field[] fields = pkg.getClass().getDeclaredFields();
for (Field field : fields) {
if (!Modifier.isStatic(field.getModifiers())) {
continue;
}
Class> type = field.getType();
if (type.equals(typeRef.type())) {
field.setAccessible(true);
try {
registerSymbol(field.get(pkg));
} catch (IllegalAccessException e) {
throw new LessInternalException("Failed to register source " + pkg, e);
}
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy