ProxiedMap.java

// Generated by delombok at Mon Jan 06 07:19:11 UTC 2025
package de.larssh.utils.collection;

import static java.util.stream.Collectors.toList;
import java.util.Collection;
import java.util.Iterator;
import java.util.Map;
import java.util.Set;
import java.util.function.BooleanSupplier;
import java.util.function.Supplier;
import de.larssh.utils.Finals;
import de.larssh.utils.Nullables;
import edu.umd.cs.findbugs.annotations.CheckForNull;
import edu.umd.cs.findbugs.annotations.Nullable;

/**
 * An abstract {@link Map} implementation pointing to a given map object.
 *
 * <p>
 * Therefore it allows implementing maps based on invisible map types, such as
 * {@link java.util.Collections#unmodifiableMap(Map)} or
 * {@link java.util.Collections#synchronizedMap(Map)}.
 *
 * @param <K> the type of keys in this map
 * @param <V> the type of values in this map
 */
public abstract class ProxiedMap<K, V> implements Map<K, V> {
	/**
	 * Wrapped map
	 */
	private final Map<K, V> map;
	/**
	 * The set of entries as defined by {@link Map#entrySet()}.
	 */
	@SuppressWarnings({"checkstyle:AnonInnerLength", "PMD.AvoidFieldNameMatchingMethodName"})
	private final Supplier<Set<Entry<K, V>>> entrySet = Finals.lazy(() -> new ProxiedSet<Entry<K, V>>(getProxiedForRead().entrySet()) {
		/**
		 * {@inheritDoc}
		 */
		@Override
		public boolean isModifiable() {
			return ProxiedMap.this.isModifiable();
		}
		/**
		 * {@inheritDoc}
		 */
		@Override
		public Iterator<Entry<K, V>> iterator() {
			return new ProxiedIterator<Entry<K, V>>(getWrappedForRead().iterator()) {
				/**
				 * {@inheritDoc}
				 */
				@Override
				public boolean isModifiable() {
					return ProxiedMap.this.isModifiable();
				}
				/**
				 * {@inheritDoc}
				 */
				@Nullable
				@Override
				public Entry<K, V> next() {
					return new ProxiedEntry<>(this::isModifiable, Nullables.orElseThrow(super.next()));
				}
			};
		}
		/**
		 * {@inheritDoc}
		 */
		@Override
		public Object[] toArray() {
			return  //
			stream().map(entry -> new ProxiedEntry<>(this::isModifiable, entry)).toArray();
		}
		/**
		 * {@inheritDoc}
		 */
		@Override
		public <T> T[] toArray(@Nullable final T[] array) {
			return  //
			stream().map(entry -> new ProxiedEntry<>(this::isModifiable, entry)).collect(toList()).toArray(array);
		}
	});
	/**
	 * The set of keys as defined by {@link Map#keySet()}.
	 */
	@SuppressWarnings("PMD.AvoidFieldNameMatchingMethodName")
	private final Supplier<Set<K>> keySet = Finals.lazy(() -> new ProxiedSet<K>(getProxiedForRead().keySet()) {
		/**
		 * {@inheritDoc}
		 */
		@Override
		public boolean isModifiable() {
			return ProxiedMap.this.isModifiable();
		}
	});
	/**
	 * The collection of values as defined by {@link Map#values()}.
	 */
	@SuppressWarnings("PMD.AvoidFieldNameMatchingMethodName")
	private final Supplier<Collection<V>> values = Finals.lazy(() -> new ProxiedCollection<V>(getProxiedForRead().values()) {
		/**
		 * {@inheritDoc}
		 */
		@Override
		public boolean isModifiable() {
			return ProxiedMap.this.isModifiable();
		}
	});

	/**
	 * {@inheritDoc}
	 */
	@Override
	public void clear() {
		getProxiedIfModifiable().clear();
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public boolean containsKey(@Nullable final Object key) {
		return getProxiedForRead().containsKey(key);
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public boolean containsValue(@Nullable final Object value) {
		return getProxiedForRead().containsValue(value);
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public Set<Entry<K, V>> entrySet() {
		return entrySet.get();
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public boolean equals(@CheckForNull final Object object) {
		return getProxiedForRead().equals(object);
	}

	/**
	 * {@inheritDoc}
	 */
	@Nullable
	@Override
	public V get(@Nullable final Object key) {
		return getProxiedForRead().get(key);
	}

	/**
	 * Verifies if this object is modifiable and either returns the wrapped map or
	 * throws an appropriate exception.
	 *
	 * @return the wrapped map if this object is modifiable
	 * @throws UnsupportedOperationException if this object is unmodifiable
	 */
	protected Map<K, V> getProxiedIfModifiable() {
		if (isModifiable()) {
			return map;
		}
		throw new UnsupportedOperationException();
	}

	/**
	 * Returns the wrapped map without verifying if modifying it is prohibited.
	 *
	 * @return the wrapped map
	 */
	protected Map<K, V> getProxiedForRead() {
		return map;
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public int hashCode() {
		return getProxiedForRead().hashCode();
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public boolean isEmpty() {
		return getProxiedForRead().isEmpty();
	}

	/**
	 * Flag specifying if this instance can be modified
	 *
	 * @return {@code true} if this instance is modifiable, else {@code false}
	 */
	public abstract boolean isModifiable();

	/**
	 * {@inheritDoc}
	 */
	@Override
	public Set<K> keySet() {
		return keySet.get();
	}

	/**
	 * {@inheritDoc}
	 */
	@Nullable
	@Override
	public V put(@Nullable final K key, @Nullable final V value) {
		return getProxiedIfModifiable().put(key, value);
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public void putAll(@Nullable final Map<? extends K, ? extends V> map) {
		getProxiedIfModifiable().putAll(map);
	}

	/**
	 * {@inheritDoc}
	 */
	@Nullable
	@Override
	public V remove(@Nullable final Object key) {
		return getProxiedIfModifiable().remove(key);
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public int size() {
		return getProxiedForRead().size();
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public String toString() {
		return getProxiedForRead().toString();
	}

	/**
	 * {@inheritDoc}
	 */
	@Override
	public Collection<V> values() {
		return values.get();
	}


	/**
	 * An abstract {@link Entry} implementation pointing to a given entry object.
	 *
	 * @param <K> the type of keys in this map
	 * @param <V> the type of values in this map
	 */
	private static class ProxiedEntry<K, V> implements Entry<K, V> {
		/**
		 * Supplies whether this instance is modifiable or not
		 */
		@SuppressWarnings("PMD.LinguisticNaming")
		private final BooleanSupplier isModifiable;
		/**
		 * Wrapped entry
		 */
		private final Entry<K, V> entry;

		/**
		 * {@inheritDoc}
		 */
		@Override
		public boolean equals(@CheckForNull final Object object) {
			return getEntry().equals(object);
		}

		/**
		 * {@inheritDoc}
		 */
		@Nullable
		@Override
		public K getKey() {
			return getEntry().getKey();
		}

		/**
		 * {@inheritDoc}
		 */
		@Nullable
		@Override
		public V getValue() {
			return getEntry().getValue();
		}

		/**
		 * {@inheritDoc}
		 */
		@Override
		public int hashCode() {
			return getEntry().hashCode();
		}

		/**
		 * {@inheritDoc}
		 */
		@Nullable
		@Override
		public V setValue(@Nullable final V value) {
			if (isModifiable.getAsBoolean()) {
				return getEntry().setValue(value);
			}
			throw new UnsupportedOperationException();
		}

		/**
		 * Creates a new {@code ProxiedEntry} instance.
		 *
		 * @param isModifiable Supplies whether this instance is modifiable or not
		 * @param entry Wrapped entry
		 */
		@java.lang.SuppressWarnings("all")
		@edu.umd.cs.findbugs.annotations.SuppressFBWarnings(justification = "generated code")
		@lombok.Generated
		public ProxiedEntry(final BooleanSupplier isModifiable, final Entry<K, V> entry) {
			this.isModifiable = isModifiable;
			this.entry = entry;
		}

		/**
		 * Wrapped entry
		 *
		 * @return the wrapped entry
		 */
		@java.lang.SuppressWarnings("all")
		@edu.umd.cs.findbugs.annotations.SuppressFBWarnings(justification = "generated code")
		@lombok.Generated
		public Entry<K, V> getEntry() {
			return this.entry;
		}
	}

	/**
	 * Creates a new {@code ProxiedMap} instance.
	 *
	 * @param map Wrapped map
	 */
	@java.lang.SuppressWarnings("all")
	@edu.umd.cs.findbugs.annotations.SuppressFBWarnings(justification = "generated code")
	@lombok.Generated
	public ProxiedMap(final Map<K, V> map) {
		this.map = map;
	}
}