blob: 67ea85c52107866bb7181fb5015fde8d60e60703 [file] [log] [blame]
/*
* 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 org.apache.johnzon.jsonb;
import org.apache.johnzon.mapper.Cleanable;
import org.apache.johnzon.mapper.util.BeanUtil;
import jakarta.json.bind.annotation.JsonbProperty;
import jakarta.json.bind.annotation.JsonbVisibility;
import jakarta.json.bind.config.PropertyVisibilityStrategy;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;
import static java.util.Optional.ofNullable;
class DefaultPropertyVisibilityStrategy implements jakarta.json.bind.config.PropertyVisibilityStrategy, Cleanable<Class<?>> {
private final ConcurrentMap<Class<?>, PropertyVisibilityStrategy> strategies = new ConcurrentHashMap<>();
private volatile boolean skipGetpackage;
@Override
public boolean isVisible(final Field field) {
return isVisible(field, field.getDeclaringClass(), true);
}
public boolean isVisible(final Field field, final Class<?> root, final boolean useGetter) {
if (field.getAnnotation(JsonbProperty.class) != null) {
return true;
}
final PropertyVisibilityStrategy strategy = strategies.computeIfAbsent(root, this::visibilityStrategy);
return strategy == this ? isFieldVisible(field, root, useGetter) : strategy.isVisible(field);
}
/**
* If the field is not public then it's of course not visible. If the field is public then we need to look at the
* accessors. If there is a private/protected/default accessor for it then it overrides and the field is not visible
* But if there is no accessor for it, then it's visible.
*/
private boolean isFieldVisible(final Field field, final Class<?> root, final boolean useGetter) {
if (!Modifier.isPublic(field.getModifiers())) {
return false;
}
// 3.7.1. Scope and Field access strategy
// note: we should bind the class since a field of a parent class can have a getter in a child
if (!useGetter) {
return !hasMethod(root, BeanUtil.setterName(field.getName()), field.getType());
}
return !hasMethod(root, BeanUtil.getterName(field.getName(), field.getType()));
}
private boolean hasMethod(Class<?> clazz, String methodName, Class<?>... paramTypes) {
try {
final Method declaredMethod = clazz.getDeclaredMethod(methodName, paramTypes);
return !Modifier.isPublic(declaredMethod.getModifiers());
} catch (NoSuchMethodException e) {
final Class<?> superclass = clazz.getSuperclass();
if (superclass == Object.class) {
return false;
}
return hasMethod(superclass, methodName, paramTypes);
}
}
@Override
public boolean isVisible(final Method method) {
final PropertyVisibilityStrategy strategy = strategies.computeIfAbsent(
method.getDeclaringClass(), this::visibilityStrategy);
return strategy == this ? Modifier.isPublic(method.getModifiers()) : strategy.isVisible(method);
}
private PropertyVisibilityStrategy visibilityStrategy(final Class<?> type) { // can be cached
JsonbVisibility visibility = type.getAnnotation(JsonbVisibility.class);
if (visibility != null) {
try {
return visibility.value().newInstance();
} catch (final InstantiationException | IllegalAccessException e) {
throw new IllegalArgumentException(e);
}
}
Package p = type.getPackage();
while (p != null) {
visibility = p.getAnnotation(JsonbVisibility.class);
if (visibility != null) {
try {
return visibility.value().newInstance();
} catch (final InstantiationException | IllegalAccessException e) {
throw new IllegalArgumentException(e);
}
}
final String name = p.getName();
final int end = name.lastIndexOf('.');
if (end < 0) {
break;
}
p = null;
final String parentPack = name.substring(0, end);
if (!skipGetpackage) {
try {
p = Package.getPackage(parentPack);
} catch (final Error unsupported) {
skipGetpackage = true; // graalvm likely
}
}
if (p == null) {
try {
p = ofNullable(type.getClassLoader())
.orElseGet(ClassLoader::getSystemClassLoader)
.loadClass(parentPack + ".package-info").getPackage();
} catch (final ClassNotFoundException e) {
// no-op
}
}
}
return this;
}
@Override
public void clean(final Class<?> clazz) {
strategies.remove(clazz);
}
}