| /* |
| * 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); |
| } |
| } |