org.gradle.model.internal.registry.ModelElementNode Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of gradle-api Show documentation
Show all versions of gradle-api Show documentation
Gradle 6.9.1 API redistribution.
/*
* Copyright 2015 the original author or authors.
*
* Licensed 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.gradle.model.internal.registry;
import com.google.common.base.Predicate;
import com.google.common.collect.Iterables;
import com.google.common.collect.Maps;
import com.google.common.collect.Sets;
import org.gradle.internal.Cast;
import org.gradle.model.RuleSource;
import org.gradle.model.internal.core.DuplicateModelException;
import org.gradle.model.internal.core.EmptyReferenceProjection;
import org.gradle.model.internal.core.ModelAction;
import org.gradle.model.internal.core.ModelActionRole;
import org.gradle.model.internal.core.ModelNode;
import org.gradle.model.internal.core.ModelPath;
import org.gradle.model.internal.core.ModelProjection;
import org.gradle.model.internal.core.ModelRegistration;
import org.gradle.model.internal.core.ModelRegistrations;
import org.gradle.model.internal.core.ModelView;
import org.gradle.model.internal.core.MutableModelNode;
import org.gradle.model.internal.core.NodePredicate;
import org.gradle.model.internal.core.rule.describe.ModelRuleDescriptor;
import org.gradle.model.internal.type.ModelType;
import javax.annotation.Nullable;
import java.util.Collections;
import java.util.Map;
import java.util.Set;
import static org.gradle.model.internal.core.ModelNode.State.Created;
import static org.gradle.model.internal.core.ModelNode.State.Initialized;
class ModelElementNode extends ModelNodeInternal {
private Map links;
private final MutableModelNode parent;
private Object privateData;
private ModelType> privateDataType;
public ModelElementNode(ModelRegistryInternal modelRegistry, ModelRegistration registration, MutableModelNode parent) {
super(modelRegistry, registration);
this.parent = parent;
}
@Override
public MutableModelNode getParent() {
return parent;
}
@Override
public ModelView extends T> asImmutable(ModelType type, @Nullable ModelRuleDescriptor ruleDescriptor) {
ModelView extends T> modelView = getAdapter().asImmutable(type, this, ruleDescriptor);
if (modelView == null) {
throw new IllegalStateException("Model element " + getPath() + " cannot be expressed as a read-only view of type " + type);
}
return modelView;
}
@Override
public ModelView extends T> asMutable(ModelType type, ModelRuleDescriptor ruleDescriptor) {
ModelView extends T> modelView;
if (isMutable()) {
modelView = getAdapter().asMutable(type, this, ruleDescriptor);
} else {
modelView = getAdapter().asImmutable(type, this, ruleDescriptor);
}
if (modelView == null) {
throw new IllegalStateException("Model element " + getPath() + " cannot be expressed as a mutable view of type " + type);
}
return modelView;
}
@Override
public T getPrivateData(Class type) {
return getPrivateData(ModelType.of(type));
}
@Override
public T getPrivateData(ModelType type) {
if (privateData == null) {
return null;
}
if (!type.isAssignableFrom(privateDataType)) {
throw new ClassCastException("Cannot get private data '" + privateData + "' of type '" + privateDataType + "' as type '" + type);
}
return Cast.uncheckedCast(privateData);
}
@Override
public Object getPrivateData() {
return privateData;
}
@Override
public void setPrivateData(Class super T> type, T object) {
setPrivateData(ModelType.of(type), object);
}
@Override
public void setPrivateData(ModelType super T> type, T object) {
if (!isMutable()) {
throw new IllegalStateException(String.format("Cannot set value for model element '%s' as this element is not mutable.", getPath()));
}
this.privateDataType = type;
this.privateData = object;
}
@Override
public boolean hasLink(String name) {
return links != null && links.containsKey(name);
}
@Override
@Nullable
public ModelNodeInternal getLink(String name) {
return links == null ? null : links.get(name);
}
@Override
public Iterable extends ModelNodeInternal> getLinks() {
return links == null ? Collections.emptyList() : links.values();
}
@Override
public int getLinkCount(Predicate super MutableModelNode> predicate) {
return links == null ? 0 : Iterables.size(Iterables.filter(links.values(), predicate));
}
@Override
public Set getLinkNames(Predicate super MutableModelNode> predicate) {
if (links == null) {
return Collections.emptySet();
}
Set names = Sets.newLinkedHashSet();
for (Map.Entry entry : links.entrySet()) {
ModelNodeInternal link = entry.getValue();
if (predicate.apply(link)) {
names.add(entry.getKey());
}
}
return names;
}
public Set getLinkNames() {
return links == null ? Collections.emptySet() : links.keySet();
}
@Override
public Iterable extends MutableModelNode> getLinks(Predicate super MutableModelNode> predicate) {
return links == null ? Collections.emptyList() : Iterables.filter(links.values(), predicate);
}
@Override
public int getLinkCount() {
return links == null ? 0 : links.size();
}
@Override
public boolean hasLink(String name, Predicate super MutableModelNode> predicate) {
ModelNodeInternal linked = getLink(name);
return linked != null && predicate.apply(linked);
}
@Override
public void applyToLink(ModelActionRole type, ModelAction action) {
if (!getPath().isDirectChild(action.getSubject().getPath())) {
throw new IllegalArgumentException(String.format("Linked element action reference has a path (%s) which is not a child of this node (%s).", action.getSubject().getPath(), getPath()));
}
modelRegistry.bind(action.getSubject(), type, action);
}
@Override
public void applyTo(NodePredicate predicate, ModelActionRole role, ModelAction action) {
modelRegistry.configureMatching(predicate.scope(getPath()), role, action);
}
@Override
public void applyTo(NodePredicate predicate, Class extends RuleSource> rules) {
modelRegistry.configureMatching(predicate.scope(getPath()), rules);
}
@Override
public void addReference(String name, ModelType type, ModelNode target, ModelRuleDescriptor descriptor) {
// TODO:LPTR Remove projection for reference node
// This shouldn't be needed, but if there's no actual value referenced, model report can only
// show the type of the node if we do this for now. It should use the schema instead to find
// the type of the property node instead.
ModelProjection projection = new EmptyReferenceProjection(type);
ModelRegistration registration = ModelRegistrations.of(getPath().child(name))
.withProjection(projection)
.descriptor(descriptor).build();
ModelReferenceNode referenceNode = new ModelReferenceNode(modelRegistry, registration, this);
if (target != null) {
referenceNode.setTarget(target);
}
addNode(referenceNode, registration);
}
@Override
public void addLink(ModelRegistration registration) {
addNode(new ModelElementNode(modelRegistry, registration, this), registration);
}
private void addNode(ModelNodeInternal child, ModelRegistration registration) {
ModelPath childPath = child.getPath();
if (!getPath().isDirectChild(childPath)) {
throw new IllegalArgumentException(String.format("Element registration has a path (%s) which is not a child of this node (%s).", childPath, getPath()));
}
ModelNodeInternal currentChild = links == null ? null : links.get(childPath.getName());
if (currentChild != null) {
if (!currentChild.isAtLeast(Created)) {
throw new DuplicateModelException(
String.format(
"Cannot create '%s' using creation rule '%s' as the rule '%s' is already registered to create this model element.",
childPath,
describe(registration.getDescriptor()),
describe(currentChild.getDescriptor())
)
);
}
throw new DuplicateModelException(
String.format(
"Cannot create '%s' using creation rule '%s' as the rule '%s' has already been used to create this model element.",
childPath,
describe(registration.getDescriptor()),
describe(currentChild.getDescriptor())
)
);
}
if (!isMutable()) {
throw new IllegalStateException(
String.format(
"Cannot create '%s' using creation rule '%s' as model element '%s' is no longer mutable.",
childPath,
describe(registration.getDescriptor()),
getPath()
)
);
}
if (links == null) {
links = Maps.newTreeMap();
}
links.put(child.getPath().getName(), child);
modelRegistry.registerNode(child, registration.getActions());
}
@Override
public void removeLink(String name) {
if (links!=null && links.remove(name) != null) {
modelRegistry.remove(getPath().child(name));
}
}
@Override
public void setTarget(ModelNode target) {
throw new UnsupportedOperationException(String.format("This node (%s) is not a reference to another node.", getPath()));
}
@Override
public void ensureUsable() {
ensureAtLeast(Initialized);
}
@Override
public void ensureAtLeast(State state) {
modelRegistry.transition(this, state, true);
}
private static String describe(ModelRuleDescriptor descriptor) {
StringBuilder stringBuilder = new StringBuilder();
descriptor.describeTo(stringBuilder);
return stringBuilder.toString();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy