com.groupbyinc.common.jackson.jq.Scope Maven / Gradle / Ivy
package net.thisptr.jackson.jq;
import java.io.IOException;
import java.io.InputStream;
import java.net.URL;
import java.util.ArrayList;
import java.util.Collections;
import java.util.Enumeration;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Optional;
import java.util.ServiceLoader;
import java.util.TreeMap;
import com.fasterxml.jackson.annotation.JsonIgnore;
import com.fasterxml.jackson.annotation.JsonIgnoreProperties;
import com.fasterxml.jackson.annotation.JsonProperty;
import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.MappingIterator;
import com.fasterxml.jackson.databind.ObjectMapper;
import net.thisptr.jackson.jq.exception.JsonQueryException;
import net.thisptr.jackson.jq.internal.BuiltinFunction;
import net.thisptr.jackson.jq.internal.JsonQueryFunction;
public class Scope {
private static final ObjectMapper DEFAULT_MAPPER = new ObjectMapper();
@Deprecated
private static final class RootScopeHolder {
@Deprecated
public static final Scope INSTANCE = new Scope(null);
static {
try {
final ClassLoader classLoader = Optional.ofNullable(Thread.currentThread().getContextClassLoader())
.orElse(Scope.class.getClassLoader());
INSTANCE.loadFunctions(classLoader);
} catch (Exception e) {
throw new RuntimeException("Failed to instantiate default Scope object", e);
}
}
}
@BuiltinFunction("debug_scope/0")
public static class DebugScopeFunction implements Function {
@Override
public List apply(final Scope scope, final List args, final JsonNode in) throws JsonQueryException {
final Map info = new HashMap<>();
info.put("scope", scope);
info.put("input", in);
return Collections.singletonList(DEFAULT_MAPPER.valueToTree(info));
}
}
@JsonProperty("functions")
@SuppressWarnings("unused")
private Map debugFunctions() {
final Map result = new TreeMap<>();
for (final Entry f : functions.entrySet())
result.put(f.getKey(), f.getValue().toString());
return result;
}
@JsonProperty("parent")
private Scope parentScope;
@JsonIgnore
private Map functions = new HashMap<>();
@JsonProperty("variables")
private Map values = new HashMap<>();
@JsonIgnore
private ObjectMapper mapper = DEFAULT_MAPPER;
/**
* Use {@link Scope#newEmptyScope()} instead and explicitly
* call {@link #loadFunctions(ClassLoader)} with the appropriate
* {@link ClassLoader} for your application. E.g.:
*
* final Scope scope = Scope.newEmptyScope();
* scope.loadFunctions(Thread.currentThread().getContextClassLoader());
*
*/
@Deprecated
public Scope() {
this(RootScopeHolder.INSTANCE);
}
@Deprecated
public Scope(final Scope parentScope) {
this.parentScope = parentScope;
}
public static Scope newEmptyScope() {
return new Scope(null);
}
public static Scope newChildScope(final Scope scope) {
return new Scope(scope);
}
public void addFunction(final String name, final int n, final Function q) {
functions.put(name + "/" + n, q);
}
public void addFunction(final String name, final Function q) {
functions.put(name, q);
}
public Function getFunction(final String name, final int nargs) {
final Function f = getFunctionRecursive(name, nargs);
if (f != null)
return f;
return getFunctionRecursive(name);
}
private Function getFunctionRecursive(final String name, final int nargs) {
final Function q = functions.get(name + "/" + nargs);
if (q == null && parentScope != null)
return parentScope.getFunctionRecursive(name, nargs);
return q;
}
private Function getFunctionRecursive(final String name) {
final Function q = functions.get(name);
if (q == null && parentScope != null)
return parentScope.getFunctionRecursive(name);
return q;
}
public void setValue(final String name, final JsonNode value) {
values.put(name, value);
}
public JsonNode getValue(final String name) {
final JsonNode value = values.get(name);
if (value == null && parentScope != null)
return parentScope.getValue(name);
return value;
}
@JsonIgnore
public ObjectMapper getObjectMapper() {
return mapper;
}
@JsonIgnoreProperties(ignoreUnknown = true)
private static class JqJson {
@JsonIgnoreProperties(ignoreUnknown = true)
public static class JqFuncDef {
@JsonProperty("name")
public String name;
@JsonProperty("args")
public List args = new ArrayList<>();
@JsonProperty("body")
public String body;
}
@JsonProperty("functions")
public List functions = new ArrayList<>();
}
@Deprecated
public static Scope rootScope() {
return RootScopeHolder.INSTANCE;
}
/**
* Dynamically resolve the path for a resource as packages may be relocated, e.g. by
* the maven-shade-plugin.
*/
private static final String resolvePath(final Class> clazz, final String name) {
final String base = clazz.getName();
return base.substring(0, base.lastIndexOf('.')).replace('.', '/') + '/' + name;
}
/**
* Load function definitions from the default resource
* from an arbitrary {@link ClassLoader}.
* E.g. in an OSGi context this may be the Bundle's {@link ClassLoader}.
*/
public void loadFunctions(final ClassLoader classLoader) {
loadMacros(classLoader, resolvePath(Scope.class, "jq.json"));
loadBuiltinFunctions(classLoader);
}
private static List loadConfig(final ClassLoader loader, final String path) throws IOException {
final List result = new ArrayList<>();
final Enumeration iter = loader.getResources(path);
while (iter.hasMoreElements()) {
try (final InputStream is = iter.nextElement().openStream()) {
final MappingIterator iter2 = DEFAULT_MAPPER.readValues(DEFAULT_MAPPER.getFactory().createParser(is), JqJson.class);
while (iter2.hasNext()) {
result.add(iter2.next());
}
}
}
return result;
}
private void loadBuiltinFunctions(final ClassLoader classLoader) {
for (final Function fn : ServiceLoader.load(Function.class, classLoader)) {
final BuiltinFunction annotation = fn.getClass().getAnnotation(BuiltinFunction.class);
if (annotation == null)
continue;
for (final String name : annotation.value())
addFunction(name, fn);
}
}
private void loadMacros(final ClassLoader classLoader, final String path) {
try {
final List configs = loadConfig(classLoader, path);
for (final JqJson jqJson : configs) {
for (final JqJson.JqFuncDef def : jqJson.functions)
addFunction(def.name, def.args.size(), new JsonQueryFunction(def.name, def.args, JsonQuery.compile(def.body), this));
}
} catch (final IOException e) {
throw new RuntimeException("Failed to load macros", e);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy