All Downloads are FREE. Search and download functionalities are using the official Maven repository.

me.lucko.luckperms.api.context.MutableContextSet Maven / Gradle / Ivy

/*
 * This file is part of LuckPerms, licensed under the MIT License.
 *
 *  Copyright (c) lucko (Luck) 
 *  Copyright (c) contributors
 *
 *  Permission is hereby granted, free of charge, to any person obtaining a copy
 *  of this software and associated documentation files (the "Software"), to deal
 *  in the Software without restriction, including without limitation the rights
 *  to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 *  copies of the Software, and to permit persons to whom the Software is
 *  furnished to do so, subject to the following conditions:
 *
 *  The above copyright notice and this permission notice shall be included in all
 *  copies or substantial portions of the Software.
 *
 *  THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 *  IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 *  FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 *  AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 *  LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 *  OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
 *  SOFTWARE.
 */

package me.lucko.luckperms.api.context;

import com.google.common.base.Preconditions;
import com.google.common.collect.HashMultimap;
import com.google.common.collect.ImmutableMap;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.ImmutableSetMultimap;
import com.google.common.collect.Multimap;
import com.google.common.collect.Multimaps;
import com.google.common.collect.SetMultimap;

import java.util.Collection;
import java.util.Map;
import java.util.Set;

import javax.annotation.Nonnull;

import static com.google.common.base.Preconditions.checkNotNull;

/**
 * A mutable implementation of {@link ContextSet}.
 *
 * 

On construction, all keys/values are {@link String#intern()}ed, in order to increase * comparison speed.

* * @since 2.16 */ public final class MutableContextSet implements ContextSet { /** * Make a singleton MutableContextSet from a context pair * * @param key the key * @param value the value * @return a new MutableContextSet containing one KV pair * @throws NullPointerException if key or value is null */ @Nonnull public static MutableContextSet singleton(@Nonnull String key, @Nonnull String value) { checkNotNull(key, "key"); checkNotNull(value, "value"); MutableContextSet set = MutableContextSet.create(); set.add(key, value); return set; } /** * Makes a MutableContextSet from two context pairs * * @param key1 the first key * @param value1 the first value * @param key2 the second key * @param value2 the second value * @return a new MutableContextSet containing the two pairs * @throws NullPointerException if any of the keys or values are null * @since 3.1 */ @Nonnull public static MutableContextSet of(@Nonnull String key1, @Nonnull String value1, @Nonnull String key2, @Nonnull String value2) { checkNotNull(key1, "key1"); checkNotNull(value1, "value1"); checkNotNull(key2, "key2"); checkNotNull(value2, "value2"); MutableContextSet set = MutableContextSet.create(); set.add(key1, value1); set.add(key2, value2); return set; } /** * Creates a MutableContextSet from an existing iterable of Map Entries * * @param iterable the iterable to copy from * @return a new MutableContextSet representing the pairs in the iterable * @throws NullPointerException if the iterable is null */ @Nonnull public static MutableContextSet fromEntries(@Nonnull Iterable> iterable) { checkNotNull(iterable, "iterable"); MutableContextSet set = MutableContextSet.create(); set.addAll(iterable); return set; } /** * Creates a MutableContextSet from an existing map * * @param map the map to copy from * @return a new MutableContextSet representing the pairs from the map * @throws NullPointerException if the map is null */ @Nonnull public static MutableContextSet fromMap(@Nonnull Map map) { checkNotNull(map, "map"); MutableContextSet set = MutableContextSet.create(); set.addAll(map); return set; } /** * Creates a MutableContextSet from an existing multimap * * @param multimap the multimap to copy from * @return a new MutableContextSet representing the pairs in the multimap * @throws NullPointerException if the multimap is null */ @Nonnull public static MutableContextSet fromMultimap(@Nonnull Multimap multimap) { checkNotNull(multimap, "multimap"); MutableContextSet set = MutableContextSet.create(); set.addAll(multimap); return set; } /** * Creates a new MutableContextSet from an existing set. * Only really useful for converting between mutable and immutable types. * * @param contextSet the context set to copy from * @return a new MutableContextSet with the same content and the one provided * @throws NullPointerException if contextSet is null */ @Nonnull public static MutableContextSet fromSet(@Nonnull ContextSet contextSet) { Preconditions.checkNotNull(contextSet, "contextSet"); MutableContextSet set = new MutableContextSet(); set.addAll(contextSet.toSet()); return set; } /** * Creates a new empty MutableContextSet. * * @return a new MutableContextSet */ @Nonnull public static MutableContextSet create() { return new MutableContextSet(); } private final SetMultimap map; public MutableContextSet() { this.map = Multimaps.synchronizedSetMultimap(HashMultimap.create()); } private MutableContextSet(MutableContextSet other) { this.map = Multimaps.synchronizedSetMultimap(HashMultimap.create(other.map)); } @Override public boolean isImmutable() { return false; } @Nonnull @Override public ImmutableContextSet makeImmutable() { return new ImmutableContextSet(ImmutableSetMultimap.copyOf(map)); } @Nonnull @Override public MutableContextSet mutableCopy() { return new MutableContextSet(this); } @Nonnull @Override public Set> toSet() { return ImmutableSet.copyOf(map.entries()); } @Nonnull @Override public Map toMap() { ImmutableMap.Builder m = ImmutableMap.builder(); for (Map.Entry e : map.entries()) { m.put(e.getKey(), e.getValue()); } return m.build(); } @Nonnull @Override public Multimap toMultimap() { return ImmutableSetMultimap.copyOf(map); } @Nonnull @Override public boolean containsKey(@Nonnull String key) { return map.containsKey(checkNotNull(key, "key").toLowerCase().intern()); } @Nonnull @Override public Set getValues(@Nonnull String key) { Collection values = map.get(checkNotNull(key, "key").toLowerCase().intern()); return values != null ? ImmutableSet.copyOf(values) : ImmutableSet.of(); } @Nonnull @Override public boolean has(@Nonnull String key, @Nonnull String value) { return map.containsEntry(checkNotNull(key, "key").toLowerCase().intern(), checkNotNull(value, "value").intern()); } @Nonnull @Override public boolean hasIgnoreCase(@Nonnull String key, @Nonnull String value) { value = checkNotNull(value, "value").intern(); Collection values = map.get(checkNotNull(key, "key").toLowerCase().intern()); if (values == null || values.isEmpty()) { return false; } for (String val : values) { if (val.equalsIgnoreCase(value)) { return true; } } return false; } @Override public boolean isEmpty() { return map.isEmpty(); } @Override public int size() { return map.size(); } /** * Adds a new key value pair to the set * * @param key the key to add * @param value the value to add * @throws NullPointerException if the key or value is null */ public void add(@Nonnull String key, @Nonnull String value) { map.put(checkNotNull(key, "key").toLowerCase().intern(), checkNotNull(value, "value").intern()); } /** * Adds a new key value pair to the set * * @param entry the entry to add * @throws NullPointerException if the entry is null */ public void add(@Nonnull Map.Entry entry) { checkNotNull(entry, "entry"); add(entry.getKey(), entry.getValue()); } /** * Adds an iterable containing contexts to the set * * @param iterable an iterable of key value context pairs * @throws NullPointerException if iterable is null */ public void addAll(@Nonnull Iterable> iterable) { for (Map.Entry e : checkNotNull(iterable, "iterable")) { add(e); } } /** * Adds the entry set of a map to the set * * @param map the map to add from * @throws NullPointerException if the map is null */ public void addAll(@Nonnull Map map) { addAll(checkNotNull(map, "map").entrySet()); } /** * Adds the entries of a multimap to the set * * @param multimap the multimap to add from * @throws NullPointerException if the map is null * @since 3.4 */ public void addAll(@Nonnull Multimap multimap) { addAll(checkNotNull(multimap, "multimap").entries()); } /** * Adds of of the values in another ContextSet to this set * * @param contextSet the set to add from * @throws NullPointerException if the contextSet is null */ public void addAll(@Nonnull ContextSet contextSet) { checkNotNull(contextSet, "contextSet"); if (contextSet instanceof MutableContextSet) { MutableContextSet other = ((MutableContextSet) contextSet); this.map.putAll(other.map); } else { addAll(contextSet.toMultimap()); } } /** * Remove a key value pair from this set * * @param key the key to remove * @param value the value to remove (case sensitive) * @throws NullPointerException if the key or value is null */ public void remove(@Nonnull String key, @Nonnull String value) { String k = checkNotNull(key, "key").toLowerCase().intern(); String v = checkNotNull(value, "value").intern(); //noinspection StringEquality map.entries().removeIf(entry -> entry.getKey() == k && entry.getValue() == v); } /** * Same as {@link #remove(String, String)}, except ignores the case of the value * * @param key the key to remove * @param value the value to remove * @throws NullPointerException if the key or value is null */ public void removeIgnoreCase(@Nonnull String key, @Nonnull String value) { String k = checkNotNull(key, "key").toLowerCase().intern(); String v = checkNotNull(value, "value").intern(); //noinspection StringEquality map.entries().removeIf(e -> e.getKey() == k && e.getValue().equalsIgnoreCase(v)); } /** * Removes all pairs with the given key * * @param key the key to remove * @throws NullPointerException if the key is null */ public void removeAll(@Nonnull String key) { map.removeAll(checkNotNull(key, "key").toLowerCase()); } /** * Clears the set */ public void clear() { map.clear(); } @Override public boolean equals(Object o) { if (o == this) return true; if (!(o instanceof ContextSet)) return false; final ContextSet other = (ContextSet) o; final Multimap otherContexts; if (other instanceof MutableContextSet) { otherContexts = ((MutableContextSet) other).map; } else { otherContexts = other.toMultimap(); } return this.map.equals(otherContexts); } @Override public int hashCode() { return map.hashCode(); } @Override public String toString() { return "MutableContextSet(contexts=" + this.map + ")"; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy