Only show enduring nodes in command output, encapsulate the Node multimaps in PermissionHolder into separate class

This commit is contained in:
Luck 2018-01-21 19:53:20 +00:00
parent 821dc4ef56
commit 4c3e28ba85
No known key found for this signature in database
GPG Key ID: EFA9B3EC5FD90F8B
11 changed files with 451 additions and 479 deletions

View File

@ -175,7 +175,7 @@ public class LPPermissionAttachment extends PermissionAttachment {
.build();
// convert the constructed node to a transient node instance to refer back to this attachment
ImmutableTransientNode transientNode = ImmutableTransientNode.of(node, this);
ImmutableTransientNode<LPPermissionAttachment> transientNode = ImmutableTransientNode.of(node, this);
// set the transient node
User user = this.permissible.getUser();

View File

@ -41,6 +41,7 @@ import me.lucko.luckperms.api.caching.CachedData;
import me.lucko.luckperms.api.context.ContextSet;
import me.lucko.luckperms.api.context.ImmutableContextSet;
import me.lucko.luckperms.common.model.Group;
import me.lucko.luckperms.common.model.NodeMapType;
import me.lucko.luckperms.common.model.User;
import me.lucko.luckperms.common.node.MetaType;
import me.lucko.luckperms.common.utils.ImmutableCollectors;
@ -161,14 +162,14 @@ public class ApiPermissionHolder implements PermissionHolder {
@Override
public Tristate hasPermission(@Nonnull Node node) {
Objects.requireNonNull(node, "node");
return this.handle.hasPermission(node, false);
return this.handle.hasPermission(node, NodeMapType.ENDURING);
}
@Nonnull
@Override
public Tristate hasTransientPermission(@Nonnull Node node) {
Objects.requireNonNull(node, "node");
return this.handle.hasPermission(node, true);
return this.handle.hasPermission(node, NodeMapType.TRANSIENT);
}
@Nonnull

View File

@ -76,7 +76,7 @@ public class ParentInfo extends SharedSubCommand {
SortMode sortMode = SortMode.determine(args);
// get the holders nodes
List<LocalizedNode> nodes = new ArrayList<>(holder.getOwnNodesSorted());
List<LocalizedNode> nodes = new ArrayList<>(holder.getEnduringData().asSortedSet());
// remove irrelevant types (these are displayed in the other info commands)
nodes.removeIf(node -> !node.isGroupNode() || !node.getValuePrimitive());

View File

@ -76,7 +76,7 @@ public class PermissionInfo extends SharedSubCommand {
SortMode sortMode = SortMode.determine(args);
// get the holders nodes
List<LocalizedNode> nodes = new ArrayList<>(holder.getOwnNodesSorted());
List<LocalizedNode> nodes = new ArrayList<>(holder.getEnduringData().asSortedSet());
// remove irrelevant types (these are displayed in the other info commands)
nodes.removeIf(node ->

View File

@ -60,19 +60,19 @@ public class GroupInfo extends SubCommand<Group> {
group.getName(),
group.getDisplayName().orElse(group.getName()),
group.getWeight().isPresent() ? group.getWeight().getAsInt() : "None",
group.getOwnNodes().size(),
group.getOwnNodes().stream().filter(n -> !(n.isGroupNode() || n.isPrefix() || n.isSuffix() || n.isMeta())).mapToInt(n -> 1).sum(),
group.getOwnNodes().stream().filter(Node::isPrefix).mapToInt(n -> 1).sum(),
group.getOwnNodes().stream().filter(Node::isSuffix).mapToInt(n -> 1).sum(),
group.getOwnNodes().stream().filter(Node::isMeta).mapToInt(n -> 1).sum()
group.getEnduringData().asList().size(),
group.getEnduringData().asList().stream().filter(n -> !(n.isGroupNode() || n.isPrefix() || n.isSuffix() || n.isMeta())).mapToInt(n -> 1).sum(),
group.getEnduringData().asList().stream().filter(Node::isPrefix).mapToInt(n -> 1).sum(),
group.getEnduringData().asList().stream().filter(Node::isSuffix).mapToInt(n -> 1).sum(),
group.getEnduringData().asList().stream().filter(Node::isMeta).mapToInt(n -> 1).sum()
);
Set<Node> parents = group.getOwnNodesSet().stream()
Set<Node> parents = group.getEnduringData().asSet().stream()
.filter(Node::isGroupNode)
.filter(Node::isPermanent)
.collect(Collectors.toSet());
Set<Node> tempParents = group.getOwnNodesSet().stream()
Set<Node> tempParents = group.getEnduringData().asSet().stream()
.filter(Node::isGroupNode)
.filter(Node::isTemporary)
.collect(Collectors.toSet());

View File

@ -66,19 +66,19 @@ public class UserInfo extends SubCommand<User> {
user.getUuid(),
status.asString(plugin.getLocaleManager()),
user.getPrimaryGroup().getValue(),
user.getOwnNodes().size(),
user.getOwnNodes().stream().filter(n -> !(n.isGroupNode() || n.isPrefix() || n.isSuffix() || n.isMeta())).mapToInt(n -> 1).sum(),
user.getOwnNodes().stream().filter(Node::isPrefix).mapToInt(n -> 1).sum(),
user.getOwnNodes().stream().filter(Node::isSuffix).mapToInt(n -> 1).sum(),
user.getOwnNodes().stream().filter(Node::isMeta).mapToInt(n -> 1).sum()
user.getEnduringData().asList().size(),
user.getEnduringData().asList().stream().filter(n -> !(n.isGroupNode() || n.isPrefix() || n.isSuffix() || n.isMeta())).mapToInt(n -> 1).sum(),
user.getEnduringData().asList().stream().filter(Node::isPrefix).mapToInt(n -> 1).sum(),
user.getEnduringData().asList().stream().filter(Node::isSuffix).mapToInt(n -> 1).sum(),
user.getEnduringData().asList().stream().filter(Node::isMeta).mapToInt(n -> 1).sum()
);
Set<Node> parents = user.getOwnNodesSet().stream()
Set<Node> parents = user.getEnduringData().asSet().stream()
.filter(Node::isGroupNode)
.filter(Node::isPermanent)
.collect(Collectors.toSet());
Set<Node> tempParents = user.getOwnNodesSet().stream()
Set<Node> tempParents = user.getEnduringData().asSet().stream()
.filter(Node::isGroupNode)
.filter(Node::isTemporary)
.collect(Collectors.toSet());

View File

@ -0,0 +1,293 @@
/*
* This file is part of LuckPerms, licensed under the MIT License.
*
* Copyright (c) lucko (Luck) <luck@lucko.me>
* 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.common.model;
import com.google.common.collect.ImmutableSetMultimap;
import com.google.common.collect.Multimap;
import com.google.common.collect.MultimapBuilder;
import com.google.common.collect.SortedSetMultimap;
import me.lucko.luckperms.api.LocalizedNode;
import me.lucko.luckperms.api.Node;
import me.lucko.luckperms.api.context.ContextSet;
import me.lucko.luckperms.api.context.ImmutableContextSet;
import me.lucko.luckperms.common.buffers.Cache;
import me.lucko.luckperms.common.contexts.ContextSetComparator;
import me.lucko.luckperms.common.node.ImmutableLocalizedNode;
import me.lucko.luckperms.common.node.NodeComparator;
import me.lucko.luckperms.common.node.NodeWithContextComparator;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Iterator;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.SortedSet;
import java.util.TreeSet;
import java.util.concurrent.locks.ReentrantLock;
import java.util.function.Predicate;
/**
* A map of nodes held by a {@link PermissionHolder}.
*
* Each holder has two of these maps, one for enduring and transient nodes.
*/
public final class NodeMap {
/**
* The holder which this map is for
*/
private final PermissionHolder holder;
/**
* The backing data map.
*
* <p>Nodes are mapped by the result of {@link Node#getFullContexts()}, and keys are sorted by the weight of the
* ContextSet. ContextSets are ordered first by the presence of a server key, then by the presence of a world
* key, and finally by the overall size of the set. Nodes are ordered according to the priority rules
* defined in {@link NodeComparator}.</p>
*/
private final SortedSetMultimap<ImmutableContextSet, Node> map = MultimapBuilder
.treeKeys(ContextSetComparator.reverse())
.treeSetValues(NodeComparator.reverse())
.build();
/**
* The lock which synchronizes the instance
*/
private final ReentrantLock lock = new ReentrantLock();
/**
* A cache which holds an immutable copy of the backing map
*/
private final NodeMapCache cache = new NodeMapCache(this);
NodeMap(PermissionHolder holder) {
this.holder = holder;
}
public List<Node> asList() {
this.lock.lock();
try {
return new ArrayList<>(this.map.values());
} finally {
this.lock.unlock();
}
}
public LinkedHashSet<Node> asSet() {
this.lock.lock();
try {
return new LinkedHashSet<>(this.map.values());
} finally {
this.lock.unlock();
}
}
public SortedSet<LocalizedNode> asSortedSet() {
SortedSet<LocalizedNode> ret = new TreeSet<>(NodeWithContextComparator.reverse());
copyToLocalized(ret);
return ret;
}
public void copyTo(Collection<? super Node> collection) {
this.lock.lock();
try {
collection.addAll(this.map.values());
} finally {
this.lock.unlock();
}
}
public void copyTo(Collection<? super Node> collection, ContextSet filter) {
this.lock.lock();
try {
for (Map.Entry<ImmutableContextSet, Collection<Node>> e : this.map.asMap().entrySet()) {
if (e.getKey().isSatisfiedBy(filter)) {
collection.addAll(e.getValue());
}
}
} finally {
this.lock.unlock();
}
}
public void copyToLocalized(Collection<LocalizedNode> collection) {
this.lock.lock();
try {
for (Node node : this.map.values()) {
collection.add(ImmutableLocalizedNode.of(node, this.holder.getObjectName()));
}
} finally {
this.lock.unlock();
}
}
/**
* Returns an immutable representation of the maps current state.
*
* @return an immutable copy
*/
public ImmutableSetMultimap<ImmutableContextSet, Node> immutable() {
return this.cache.get();
}
/**
* Invalidates the cache
*/
void invalidate() {
this.cache.invalidate();
}
void add(Node node) {
this.lock.lock();
try {
this.map.put(node.getFullContexts().makeImmutable(), node);
} finally {
this.lock.unlock();
}
}
void replace(Node node, Node previous) {
this.lock.lock();
try {
this.map.remove(previous.getFullContexts().makeImmutable(), previous);
this.map.put(node.getFullContexts().makeImmutable(), node);
} finally {
this.lock.unlock();
}
}
void remove(Node node) {
this.lock.lock();
try {
this.map.get(node.getFullContexts().makeImmutable()).removeIf(e -> e.almostEquals(node));
} finally {
this.lock.unlock();
}
}
void clear() {
this.lock.lock();
try {
this.map.clear();
} finally {
this.lock.unlock();
}
}
void clear(ContextSet contextSet) {
this.lock.lock();
try {
this.map.removeAll(contextSet.makeImmutable());
} finally {
this.lock.unlock();
}
}
void setContent(Set<Node> set) {
this.lock.lock();
try {
this.map.clear();
for (Node n : set) {
this.map.put(n.getFullContexts().makeImmutable(), n);
}
} finally {
this.lock.unlock();
}
}
void setContent(Multimap<ImmutableContextSet, Node> multimap) {
this.lock.lock();
try {
this.map.clear();
this.map.putAll(multimap);
} finally {
this.lock.unlock();
}
}
boolean removeIf(Predicate<? super Node> predicate) {
this.lock.lock();
try {
return this.map.values().removeIf(predicate);
} finally {
this.lock.unlock();
}
}
boolean removeIf(ContextSet contextSet, Predicate<? super Node> predicate) {
this.lock.lock();
try {
SortedSet<Node> nodes = this.map.get(contextSet.makeImmutable());
return nodes != null && nodes.removeIf(predicate);
} finally {
this.lock.unlock();
}
}
boolean auditTemporaryNodes(Set<Node> removed) {
boolean work = false;
this.lock.lock();
try {
Iterator<Node> it = this.map.values().iterator();
while (it.hasNext()) {
Node entry = it.next();
if (entry.hasExpired()) {
removed.add(entry);
work = true;
it.remove();
}
}
} finally {
this.lock.unlock();
}
return work;
}
private static final class NodeMapCache extends Cache<ImmutableSetMultimap<ImmutableContextSet, Node>> {
private final NodeMap handle;
private NodeMapCache(NodeMap handle) {
this.handle = handle;
}
@Override
protected ImmutableSetMultimap<ImmutableContextSet, Node> supply() {
this.handle.lock.lock();
try {
return ImmutableSetMultimap.copyOf(this.handle.map);
} finally {
this.handle.lock.unlock();
}
}
}
}

View File

@ -0,0 +1,32 @@
/*
* This file is part of LuckPerms, licensed under the MIT License.
*
* Copyright (c) lucko (Luck) <luck@lucko.me>
* 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.common.model;
public enum NodeMapType {
ENDURING, TRANSIENT
}

View File

@ -31,8 +31,6 @@ import com.google.common.collect.ImmutableSet;
import com.google.common.collect.ImmutableSetMultimap;
import com.google.common.collect.Maps;
import com.google.common.collect.Multimap;
import com.google.common.collect.MultimapBuilder;
import com.google.common.collect.SortedSetMultimap;
import me.lucko.luckperms.api.Contexts;
import me.lucko.luckperms.api.DataMutateResult;
@ -63,11 +61,9 @@ import me.lucko.luckperms.common.references.HolderReference;
import me.lucko.luckperms.common.references.HolderType;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Comparator;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.LinkedHashSet;
import java.util.LinkedList;
import java.util.List;
@ -119,40 +115,8 @@ public abstract class PermissionHolder {
/**
* The holders persistent nodes.
*
* <p>Nodes are mapped by the result of {@link Node#getFullContexts()}, and keys are sorted by the weight of the
* ContextSet. ContextSets are ordered first by the presence of a server key, then by the presence of a world
* key, and finally by the overall size of the set. Nodes are ordered according to the priority rules
* defined in {@link NodeComparator}.</p>
*/
private final SortedSetMultimap<ImmutableContextSet, Node> nodes = MultimapBuilder
.treeKeys(ContextSetComparator.reverse())
.treeSetValues(NodeComparator.reverse())
.build();
protected PermissionHolder(String objectName, LuckPermsPlugin plugin) {
this.objectName = objectName;
this.plugin = plugin;
}
/**
* Caches an immutable copy of the above nodes multimap
*/
private final class NodesCache extends Cache<ImmutableSetMultimap<ImmutableContextSet, Node>> {
@Override
protected ImmutableSetMultimap<ImmutableContextSet, Node> supply() {
PermissionHolder.this.nodesLock.lock();
try {
return ImmutableSetMultimap.copyOf(PermissionHolder.this.nodes);
} finally {
PermissionHolder.this.nodesLock.unlock();
}
}
}
private final NodesCache nodesCopy = new NodesCache();
// used to ensure thread safe access to the backing nodes map
private final ReentrantLock nodesLock = new ReentrantLock();
private final NodeMap enduringNodes = new NodeMap(this);
/**
* The holders transient nodes.
@ -160,46 +124,18 @@ public abstract class PermissionHolder {
* <p>These are nodes which are never stored or persisted to a file, and only
* last until the end of the objects lifetime. (for a group, that's when the server stops, and for a user, it's when
* they log out, or get unloaded.)</p>
*
* <p>Nodes are mapped by the result of {@link Node#getFullContexts()}, and keys are sorted by the weight of the
* ContextSet. ContextSets are ordered first by the presence of a server key, then by the presence of a world
* key, and finally by the overall size of the set. Nodes are ordered according to the priority rules
* defined in {@link NodeComparator}.</p>
*/
private final SortedSetMultimap<ImmutableContextSet, Node> transientNodes = MultimapBuilder
.treeKeys(ContextSetComparator.reverse())
.treeSetValues(NodeComparator.reverse())
.build();
/**
* Caches an immutable copy of the above transientNodes multimap
*/
private final class TransientNodesCache extends Cache<ImmutableSetMultimap<ImmutableContextSet, Node>> {
@Override
protected ImmutableSetMultimap<ImmutableContextSet, Node> supply() {
PermissionHolder.this.transientNodesLock.lock();
try {
return ImmutableSetMultimap.copyOf(PermissionHolder.this.transientNodes);
} finally {
PermissionHolder.this.transientNodesLock.unlock();
}
}
}
private final TransientNodesCache transientNodesCopy = new TransientNodesCache();
private final NodeMap transientNodes = new NodeMap(this);
/**
* Caches the holders weight lookup
*/
private final class WeightCache extends Cache<OptionalInt> {
private final Cache<OptionalInt> weightCache = new Cache<OptionalInt>() {
@Override
protected OptionalInt supply() {
return calculateWeight();
}
}
private final WeightCache weightCache = new WeightCache();
// used to ensure thread safe access to the backing transientNodes map
private final ReentrantLock transientNodesLock = new ReentrantLock();
};
/**
* Lock used by Storage implementations to prevent concurrent read/writes
@ -216,6 +152,11 @@ public abstract class PermissionHolder {
*/
private final Set<StateListener> stateListeners = ConcurrentHashMap.newKeySet();
protected PermissionHolder(String objectName, LuckPermsPlugin plugin) {
this.objectName = objectName;
this.plugin = plugin;
}
public String getObjectName() {
return this.objectName;
}
@ -233,8 +174,8 @@ public abstract class PermissionHolder {
}
private void invalidateCache() {
this.nodesCopy.invalidate();
this.transientNodesCopy.invalidate();
this.enduringNodes.invalidate();
this.transientNodes.invalidate();
this.weightCache.invalidate();
// Invalidate listeners
@ -295,13 +236,32 @@ public abstract class PermissionHolder {
*/
public abstract ApiPermissionHolder getDelegate();
public NodeMap getData(NodeMapType type) {
switch (type) {
case ENDURING:
return this.enduringNodes;
case TRANSIENT:
return this.transientNodes;
default:
throw new AssertionError();
}
}
public NodeMap getEnduringData() {
return this.enduringNodes;
}
public NodeMap getTransientData() {
return this.transientNodes;
}
/**
* Returns an immutable copy of this objects nodes
*
* @return an immutable copy of the multimap storing this objects nodes
*/
public ImmutableSetMultimap<ImmutableContextSet, Node> getEnduringNodes() {
return this.nodesCopy.get();
return this.enduringNodes.immutable();
}
/**
@ -310,7 +270,7 @@ public abstract class PermissionHolder {
* @return an immutable copy of the multimap storing this objects transient nodes
*/
public ImmutableSetMultimap<ImmutableContextSet, Node> getTransientNodes() {
return this.transientNodesCopy.get();
return this.transientNodes.immutable();
}
/**
@ -319,15 +279,7 @@ public abstract class PermissionHolder {
* @param set the set of nodes to apply to the object
*/
public void setEnduringNodes(Set<Node> set) {
this.nodesLock.lock();
try {
this.nodes.clear();
for (Node n : set) {
this.nodes.put(n.getFullContexts().makeImmutable(), n);
}
} finally {
this.nodesLock.unlock();
}
this.enduringNodes.setContent(set);
invalidateCache();
}
@ -337,37 +289,7 @@ public abstract class PermissionHolder {
* @param multimap the replacement multimap
*/
public void replaceEnduringNodes(Multimap<ImmutableContextSet, Node> multimap) {
this.nodesLock.lock();
try {
this.nodes.clear();
this.nodes.putAll(multimap);
} finally {
this.nodesLock.unlock();
}
invalidateCache();
}
public void setTransientNodes(Set<Node> set) {
this.transientNodesLock.lock();
try {
this.transientNodes.clear();
for (Node n : set) {
this.transientNodes.put(n.getFullContexts().makeImmutable(), n);
}
} finally {
this.transientNodesLock.unlock();
}
invalidateCache();
}
public void replaceTransientNodes(Multimap<ImmutableContextSet, Node> multimap) {
this.transientNodesLock.lock();
try {
this.transientNodes.clear();
this.transientNodes.putAll(multimap);
} finally {
this.transientNodesLock.unlock();
}
this.enduringNodes.setContent(multimap);
invalidateCache();
}
@ -378,172 +300,49 @@ public abstract class PermissionHolder {
*/
public LinkedHashSet<Node> getOwnNodesSet() {
LinkedHashSet<Node> ret = new LinkedHashSet<>();
this.transientNodesLock.lock();
try {
ret.addAll(this.transientNodes.values());
} finally {
this.transientNodesLock.unlock();
}
this.nodesLock.lock();
try {
ret.addAll(this.nodes.values());
} finally {
this.nodesLock.unlock();
}
this.transientNodes.copyTo(ret);
this.enduringNodes.copyTo(ret);
return ret;
}
public List<Node> getOwnNodes() {
List<Node> ret = new ArrayList<>();
this.transientNodes.copyTo(ret);
this.enduringNodes.copyTo(ret);
return ret;
}
this.transientNodesLock.lock();
try {
ret.addAll(this.transientNodes.values());
} finally {
this.transientNodesLock.unlock();
}
this.nodesLock.lock();
try {
ret.addAll(this.nodes.values());
} finally {
this.nodesLock.unlock();
}
public List<Node> getOwnNodes(ContextSet filter) {
List<Node> ret = new ArrayList<>();
this.transientNodes.copyTo(ret, filter);
this.enduringNodes.copyTo(ret, filter);
return ret;
}
public SortedSet<LocalizedNode> getOwnNodesSorted() {
SortedSet<LocalizedNode> ret = new TreeSet<>(NodeWithContextComparator.reverse());
this.transientNodesLock.lock();
try {
for (Node node : this.transientNodes.values()) {
ret.add(ImmutableLocalizedNode.of(node, getObjectName()));
}
} finally {
this.transientNodesLock.unlock();
}
this.nodesLock.lock();
try {
for (Node node : this.nodes.values()) {
ret.add(ImmutableLocalizedNode.of(node, getObjectName()));
}
} finally {
this.nodesLock.unlock();
}
return ret;
}
public List<Node> filterEnduringNodes(ContextSet filter) {
return filterEnduringNodes(new ArrayList<>(), filter);
}
public <C extends Collection<Node>> C filterEnduringNodes(C ret, ContextSet filter) {
this.nodesLock.lock();
try {
for (Map.Entry<ImmutableContextSet, Collection<Node>> e : this.nodes.asMap().entrySet()) {
if (e.getKey().isSatisfiedBy(filter)) {
ret.addAll(e.getValue());
}
}
} finally {
this.nodesLock.unlock();
}
return ret;
}
public List<Node> filterTransientNodes(ContextSet filter) {
return filterTransientNodes(new ArrayList<>(), filter);
}
public <C extends Collection<Node>> C filterTransientNodes(C ret, ContextSet filter) {
this.transientNodesLock.lock();
try {
for (Map.Entry<ImmutableContextSet, Collection<Node>> e : this.transientNodes.asMap().entrySet()) {
if (e.getKey().isSatisfiedBy(filter)) {
ret.addAll(e.getValue());
}
}
} finally {
this.transientNodesLock.unlock();
}
return ret;
}
public List<Node> filterNodes(ContextSet filter) {
return filterNodes(new ArrayList<>(), filter);
}
public <C extends Collection<Node>> C filterNodes(C ret, ContextSet filter) {
this.transientNodesLock.lock();
try {
for (Map.Entry<ImmutableContextSet, Collection<Node>> e : this.transientNodes.asMap().entrySet()) {
if (e.getKey().isSatisfiedBy(filter)) {
ret.addAll(e.getValue());
}
}
} finally {
this.transientNodesLock.unlock();
}
this.nodesLock.lock();
try {
for (Map.Entry<ImmutableContextSet, Collection<Node>> e : this.nodes.asMap().entrySet()) {
if (e.getKey().isSatisfiedBy(filter)) {
ret.addAll(e.getValue());
}
}
} finally {
this.nodesLock.unlock();
}
this.transientNodes.copyToLocalized(ret);
this.enduringNodes.copyToLocalized(ret);
return ret;
}
public boolean removeIf(Predicate<Node> predicate) {
boolean result;
ImmutableCollection<Node> before = getEnduringNodes().values();
this.nodesLock.lock();
try {
result = this.nodes.values().removeIf(predicate);
} finally {
this.nodesLock.unlock();
}
if (!result) {
if (!this.enduringNodes.removeIf(predicate)) {
return false;
}
invalidateCache();
ImmutableCollection<Node> after = getEnduringNodes().values();
this.plugin.getEventFactory().handleNodeClear(this, before, after);
return true;
}
public boolean removeIfTransient(Predicate<Node> predicate) {
boolean result;
this.transientNodesLock.lock();
try {
result = this.transientNodes.values().removeIf(predicate);
} finally {
this.transientNodesLock.unlock();
}
boolean result = this.transientNodes.removeIf(predicate);
if (result) {
invalidateCache();
}
return result;
}
@ -568,7 +367,7 @@ public abstract class PermissionHolder {
}
// get and add the objects own nodes
List<Node> nodes = filterNodes(context.getContexts());
List<Node> nodes = getOwnNodes(context.getContexts());
for (Node node : nodes) {
ImmutableLocalizedNode localizedNode = ImmutableLocalizedNode.of(node, getObjectName());
accumulator.add(localizedNode);
@ -702,7 +501,7 @@ public abstract class PermissionHolder {
entries = resolveInheritances(new LinkedList<>(), null, context);
} else {
entries = new LinkedList<>();
for (Node n : filterNodes(context.getContexts())) {
for (Node n : getOwnNodes(context.getContexts())) {
ImmutableLocalizedNode localizedNode = ImmutableLocalizedNode.of(n, getObjectName());
entries.add(localizedNode);
}
@ -727,7 +526,7 @@ public abstract class PermissionHolder {
entries = resolveInheritances(new LinkedList<>(), null, context);
} else {
entries = new LinkedList<>();
for (Node n : filterNodes(context.getContexts())) {
for (Node n : getOwnNodes(context.getContexts())) {
ImmutableLocalizedNode localizedNode = ImmutableLocalizedNode.of(n, getObjectName());
entries.add(localizedNode);
}
@ -795,7 +594,7 @@ public abstract class PermissionHolder {
}
// get and add the objects own nodes
List<Node> nodes = filterNodes(context.getContexts());
List<Node> nodes = getOwnNodes(context.getContexts());
for (Node node : nodes) {
if (!node.getValuePrimitive()) continue;
@ -903,47 +702,11 @@ public abstract class PermissionHolder {
* @return true if permissions had expired and were removed
*/
public boolean auditTemporaryPermissions() {
boolean work = false;
Set<Node> removed = new HashSet<>();
ImmutableSet<Node> before = ImmutableSet.copyOf(getOwnNodesSet());
this.nodesLock.lock();
try {
Iterator<Node> it = this.nodes.values().iterator();
while (it.hasNext()) {
Node entry = it.next();
if (entry.hasExpired()) {
removed.add(entry);
work = true;
it.remove();
}
}
} finally {
this.nodesLock.unlock();
}
if (work) {
invalidateCache();
work = false;
}
this.transientNodesLock.lock();
try {
Iterator<Node> it = this.transientNodes.values().iterator();
while (it.hasNext()) {
Node entry = it.next();
if (entry.hasExpired()) {
removed.add(entry);
work = true;
it.remove();
}
}
} finally {
this.transientNodesLock.unlock();
}
if (work) {
if (this.enduringNodes.auditTemporaryNodes(removed) | this.transientNodes.auditTemporaryNodes(removed)) {
invalidateCache();
}
@ -952,7 +715,6 @@ public abstract class PermissionHolder {
}
ImmutableSet<Node> after = ImmutableSet.copyOf(getOwnNodesSet());
for (Node r : removed) {
this.plugin.getEventFactory().handleNodeRemove(r, this, before, after);
}
@ -960,8 +722,8 @@ public abstract class PermissionHolder {
return true;
}
public Optional<Node> getAlmostEquals(Node node, boolean t) {
for (Node n : t ? getTransientNodes().values() : getEnduringNodes().values()) {
public Optional<Node> getAlmostEquals(Node node, NodeMapType type) {
for (Node n : getData(type).immutable().values()) {
if (n.almostEquals(node)) {
return Optional.of(n);
}
@ -974,19 +736,19 @@ public abstract class PermissionHolder {
* Check if the holder has a permission node
*
* @param node the node to check
* @param checkTransient whether to check transient nodes
* @param type which backing map to check
* @return a tristate
*/
public Tristate hasPermission(Node node, boolean checkTransient) {
public Tristate hasPermission(Node node, NodeMapType type) {
if (this.getType().isGroup() && node.isGroupNode() && node.getGroupName().equalsIgnoreCase(getObjectName())) {
return Tristate.TRUE;
}
return getAlmostEquals(node, checkTransient).map(Node::getTristate).orElse(Tristate.UNDEFINED);
return getAlmostEquals(node, type).map(Node::getTristate).orElse(Tristate.UNDEFINED);
}
public Tristate hasPermission(Node node) {
return hasPermission(node, false);
return hasPermission(node, NodeMapType.ENDURING);
}
/**
@ -1021,20 +783,13 @@ public abstract class PermissionHolder {
* @param node the node to set
*/
public DataMutateResult setPermission(Node node) {
if (hasPermission(node, false) != Tristate.UNDEFINED) {
if (hasPermission(node, NodeMapType.ENDURING) != Tristate.UNDEFINED) {
return DataMutateResult.ALREADY_HAS;
}
ImmutableCollection<Node> before = getEnduringNodes().values();
this.nodesLock.lock();
try {
this.nodes.put(node.getFullContexts().makeImmutable(), node);
} finally {
this.nodesLock.unlock();
}
this.enduringNodes.add(node);
invalidateCache();
ImmutableCollection<Node> after = getEnduringNodes().values();
this.plugin.getEventFactory().handleNodeAdd(node, this, before, after);
@ -1052,7 +807,7 @@ public abstract class PermissionHolder {
if (node.isTemporary()) {
if (modifier == TemporaryModifier.ACCUMULATE) {
// Try to accumulate with an existing node
Optional<Node> existing = getAlmostEquals(node, false);
Optional<Node> existing = getAlmostEquals(node, NodeMapType.ENDURING);
// An existing node was found
if (existing.isPresent()) {
@ -1061,26 +816,19 @@ public abstract class PermissionHolder {
// Create a new node with the same properties, but add the expiry dates together
Node newNode = NodeFactory.builder(node).setExpiry(previous.getExpiryUnixTime() + node.getSecondsTilExpiry()).build();
ImmutableCollection<Node> before = getEnduringNodes().values();
// Remove the old node & add the new one.
this.nodesLock.lock();
try {
this.nodes.remove(previous.getFullContexts().makeImmutable(), previous);
this.nodes.put(newNode.getFullContexts().makeImmutable(), newNode);
} finally {
this.nodesLock.unlock();
}
ImmutableCollection<Node> before = getEnduringNodes().values();
this.enduringNodes.replace(newNode, previous);
invalidateCache();
ImmutableCollection<Node> after = getEnduringNodes().values();
this.plugin.getEventFactory().handleNodeAdd(newNode, this, before, after);
return Maps.immutableEntry(DataMutateResult.SUCCESS, newNode);
}
} else if (modifier == TemporaryModifier.REPLACE) {
// Try to replace an existing node
Optional<Node> existing = getAlmostEquals(node, false);
Optional<Node> existing = getAlmostEquals(node, NodeMapType.ENDURING);
// An existing node was found
if (existing.isPresent()) {
@ -1090,17 +838,10 @@ public abstract class PermissionHolder {
if (node.getExpiryUnixTime() > previous.getExpiryUnixTime()) {
ImmutableCollection<Node> before = getEnduringNodes().values();
this.nodesLock.lock();
try {
this.nodes.remove(previous.getFullContexts().makeImmutable(), previous);
this.nodes.put(node.getFullContexts().makeImmutable(), node);
} finally {
this.nodesLock.unlock();
}
this.enduringNodes.replace(node, previous);
invalidateCache();
ImmutableCollection<Node> after = getEnduringNodes().values();
this.plugin.getEventFactory().handleNodeAdd(node, this, before, after);
return Maps.immutableEntry(DataMutateResult.SUCCESS, node);
}
@ -1120,21 +861,13 @@ public abstract class PermissionHolder {
* @param node the node to set
*/
public DataMutateResult setTransientPermission(Node node) {
if (hasPermission(node, true) != Tristate.UNDEFINED) {
if (hasPermission(node, NodeMapType.TRANSIENT) != Tristate.UNDEFINED) {
return DataMutateResult.ALREADY_HAS;
}
ImmutableCollection<Node> before = getTransientNodes().values();
this.transientNodesLock.lock();
try {
this.transientNodes.put(node.getFullContexts().makeImmutable(), node);
} finally {
this.transientNodesLock.unlock();
}
this.transientNodes.add(node);
invalidateCache();
ImmutableCollection<Node> after = getTransientNodes().values();
this.plugin.getEventFactory().handleNodeAdd(node, this, before, after);
@ -1147,22 +880,15 @@ public abstract class PermissionHolder {
* @param node the node to unset
*/
public DataMutateResult unsetPermission(Node node) {
if (hasPermission(node, false) == Tristate.UNDEFINED) {
if (hasPermission(node, NodeMapType.ENDURING) == Tristate.UNDEFINED) {
return DataMutateResult.LACKS;
}
ImmutableCollection<Node> before = getEnduringNodes().values();
this.nodesLock.lock();
try {
this.nodes.get(node.getFullContexts().makeImmutable()).removeIf(e -> e.almostEquals(node));
} finally {
this.nodesLock.unlock();
}
this.enduringNodes.remove(node);
invalidateCache();
ImmutableCollection<Node> after = getEnduringNodes().values();
this.plugin.getEventFactory().handleNodeRemove(node, this, before, after);
return DataMutateResult.SUCCESS;
}
@ -1173,22 +899,15 @@ public abstract class PermissionHolder {
* @param node the node to unset
*/
public DataMutateResult unsetTransientPermission(Node node) {
if (hasPermission(node, true) == Tristate.UNDEFINED) {
if (hasPermission(node, NodeMapType.TRANSIENT) == Tristate.UNDEFINED) {
return DataMutateResult.LACKS;
}
ImmutableCollection<Node> before = getTransientNodes().values();
this.transientNodesLock.lock();
try {
this.transientNodes.get(node.getFullContexts().makeImmutable()).removeIf(e -> e.almostEquals(node));
} finally {
this.transientNodesLock.unlock();
}
this.transientNodes.remove(node);
invalidateCache();
ImmutableCollection<Node> after = getTransientNodes().values();
this.plugin.getEventFactory().handleNodeRemove(node, this, before, after);
return DataMutateResult.SUCCESS;
}
@ -1206,14 +925,7 @@ public abstract class PermissionHolder {
*/
public boolean clearNodes() {
ImmutableCollection<Node> before = getEnduringNodes().values();
this.nodesLock.lock();
try {
this.nodes.clear();
} finally {
this.nodesLock.unlock();
}
this.enduringNodes.clear();
invalidateCache();
ImmutableCollection<Node> after = getEnduringNodes().values();
@ -1227,13 +939,7 @@ public abstract class PermissionHolder {
public boolean clearNodes(ContextSet contextSet) {
ImmutableCollection<Node> before = getEnduringNodes().values();
this.nodesLock.lock();
try {
this.nodes.removeAll(contextSet.makeImmutable());
} finally {
this.nodesLock.unlock();
}
this.enduringNodes.clear(contextSet);
invalidateCache();
ImmutableCollection<Node> after = getEnduringNodes().values();
@ -1248,22 +954,16 @@ public abstract class PermissionHolder {
public boolean clearParents(boolean giveDefault) {
ImmutableCollection<Node> before = getEnduringNodes().values();
this.nodesLock.lock();
try {
boolean b = this.nodes.values().removeIf(Node::isGroupNode);
if (!b) {
return false;
}
} finally {
this.nodesLock.unlock();
if (!this.enduringNodes.removeIf(Node::isGroupNode)) {
return false;
}
if (this.getType().isUser() && giveDefault) {
this.plugin.getUserManager().giveDefaultIfNeeded((User) this, false);
}
invalidateCache();
ImmutableCollection<Node> after = getEnduringNodes().values();
this.plugin.getEventFactory().handleNodeClear(this, before, after);
return true;
}
@ -1271,125 +971,71 @@ public abstract class PermissionHolder {
public boolean clearParents(ContextSet contextSet, boolean giveDefault) {
ImmutableCollection<Node> before = getEnduringNodes().values();
this.nodesLock.lock();
try {
SortedSet<Node> nodes = this.nodes.get(contextSet.makeImmutable());
if (nodes == null) {
return false;
}
boolean b = nodes.removeIf(Node::isGroupNode);
if (!b) {
return false;
}
} finally {
this.nodesLock.unlock();
if (!this.enduringNodes.removeIf(contextSet, Node::isGroupNode)) {
return false;
}
if (this.getType().isUser() && giveDefault) {
this.plugin.getUserManager().giveDefaultIfNeeded((User) this, false);
}
invalidateCache();
ImmutableCollection<Node> after = getEnduringNodes().values();
this.plugin.getEventFactory().handleNodeClear(this, before, after);
return true;
}
public boolean clearMeta(MetaType type) {
ImmutableCollection<Node> before = getEnduringNodes().values();
this.nodesLock.lock();
try {
if (!this.nodes.values().removeIf(type::matches)) {
return false;
}
} finally {
this.nodesLock.unlock();
if (!this.enduringNodes.removeIf(type::matches)) {
return false;
}
invalidateCache();
ImmutableCollection<Node> after = getEnduringNodes().values();
this.plugin.getEventFactory().handleNodeClear(this, before, after);
return true;
}
public boolean clearMeta(MetaType type, ContextSet contextSet) {
ImmutableCollection<Node> before = getEnduringNodes().values();
this.nodesLock.lock();
try {
SortedSet<Node> nodes = this.nodes.get(contextSet.makeImmutable());
if (nodes == null) {
return false;
}
boolean b = nodes.removeIf(type::matches);
if (!b) {
return false;
}
} finally {
this.nodesLock.unlock();
if (!this.enduringNodes.removeIf(contextSet, type::matches)) {
return false;
}
invalidateCache();
ImmutableCollection<Node> after = getEnduringNodes().values();
this.plugin.getEventFactory().handleNodeClear(this, before, after);
return true;
}
public boolean clearMetaKeys(String key, boolean temp) {
ImmutableCollection<Node> before = getEnduringNodes().values();
this.nodesLock.lock();
try {
boolean b = this.nodes.values().removeIf(n -> n.isMeta() && (n.isTemporary() == temp) && n.getMeta().getKey().equalsIgnoreCase(key));
if (!b) {
return false;
}
} finally {
this.nodesLock.unlock();
if (!this.enduringNodes.removeIf(n -> n.isMeta() && (n.isTemporary() == temp) && n.getMeta().getKey().equalsIgnoreCase(key))) {
return false;
}
invalidateCache();
ImmutableCollection<Node> after = getEnduringNodes().values();
this.plugin.getEventFactory().handleNodeClear(this, before, after);
return true;
}
public boolean clearMetaKeys(String key, ContextSet contextSet, boolean temp) {
ImmutableCollection<Node> before = getEnduringNodes().values();
this.nodesLock.lock();
try {
SortedSet<Node> nodes = this.nodes.get(contextSet.makeImmutable());
if (nodes == null) {
return false;
}
boolean b = nodes.removeIf(n -> n.isMeta() && (n.isTemporary() == temp) && n.getMeta().getKey().equalsIgnoreCase(key));
if (!b) {
return false;
}
} finally {
this.nodesLock.unlock();
if (!this.enduringNodes.removeIf(contextSet, n -> n.isMeta() && (n.isTemporary() == temp) && n.getMeta().getKey().equalsIgnoreCase(key))) {
return false;
}
invalidateCache();
ImmutableCollection<Node> after = getEnduringNodes().values();
this.plugin.getEventFactory().handleNodeClear(this, before, after);
return true;
}
public boolean clearTransientNodes() {
ImmutableCollection<Node> before = getTransientNodes().values();
this.transientNodesLock.lock();
try {
this.transientNodes.clear();
} finally {
this.transientNodesLock.unlock();
}
this.transientNodes.clear();
invalidateCache();
ImmutableCollection<Node> after = getTransientNodes().values();

View File

@ -40,17 +40,17 @@ import javax.annotation.Nonnull;
/**
* Holds a Node and plus an owning object. All calls are passed onto the contained Node instance.
*/
public final class ImmutableTransientNode implements Node {
public static ImmutableTransientNode of(Node node, Object owner) {
public final class ImmutableTransientNode<O> implements Node {
public static <O> ImmutableTransientNode<O> of(Node node, O owner) {
Objects.requireNonNull(node, "node");
Objects.requireNonNull(owner, "owner");
return new ImmutableTransientNode(node, owner);
return new ImmutableTransientNode<>(node, owner);
}
private final Node node;
private final Object owner;
private final O owner;
private ImmutableTransientNode(Node node, Object owner) {
private ImmutableTransientNode(Node node, O owner) {
this.node = node;
this.owner = owner;
}
@ -69,7 +69,7 @@ public final class ImmutableTransientNode implements Node {
return this.node;
}
public Object getOwner() {
public O getOwner() {
return this.owner;
}

View File

@ -46,7 +46,7 @@ public class ParentsByWeightHolder extends CachedPrimaryGroupHolder {
}
Set<Group> groups = new LinkedHashSet<>();
for (Node node : this.user.filterNodes(contexts.getContexts())) {
for (Node node : this.user.getOwnNodes(contexts.getContexts())) {
if (!node.getValuePrimitive() || !node.isGroupNode()) {
continue;
}