/* * Copyright 2003-2016 JetBrains s.r.o. * * 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 jetbrains.mps.smodel; import jetbrains.mps.logging.Logger; import jetbrains.mps.scope.ErrorScope; import jetbrains.mps.scope.Scope; import jetbrains.mps.smodel.constraints.ModelConstraints; import jetbrains.mps.smodel.legacy.ConceptMetaInfoConverter; import jetbrains.mps.util.annotation.ToRemove; import org.apache.log4j.LogManager; import org.jetbrains.annotations.NotNull; import org.jetbrains.annotations.Nullable; import org.jetbrains.mps.annotations.Immutable; import org.jetbrains.mps.openapi.language.SAbstractConcept; import org.jetbrains.mps.openapi.language.SReferenceLink; import org.jetbrains.mps.openapi.model.SModelName; import org.jetbrains.mps.openapi.model.SModelReference; import org.jetbrains.mps.openapi.model.SNode; import org.jetbrains.mps.openapi.model.SNodeReference; import java.util.ArrayList; import java.util.HashSet; import java.util.List; import java.util.Set; /** * Igor Alshannikov * Dec 10, 2007 */ public class DynamicReference extends SReferenceBase { private static final Logger LOG = Logger.wrap(LogManager.getLogger(DynamicReference.class)); private DynamicReferenceOrigin myOrigin; // this is for tracking loops in dynref resolving, typically arising from interaction // between type system rules and scopes private static final ThreadLocal<Set<DynamicReference>> currentlyResolved = new ThreadLocal<Set<DynamicReference>>() { @Override protected Set<DynamicReference> initialValue() { return new HashSet<DynamicReference>(); } }; // we also keep track of references for which we call reportErrorWithOrigin // we need this because it will call source node's getPresentation() which in turn might resolve us again // we don't want to report loop in this case, rather just return null private static final ThreadLocal<Set<DynamicReference>> currentlySourceNodeLogged = new ThreadLocal<Set<DynamicReference>>() { @Override protected Set<DynamicReference> initialValue() { return new HashSet<DynamicReference>(); } }; private boolean myHasBeenResolve; private SNode myCachedTargetNode; /* * create 'young' reference */ @Deprecated @ToRemove(version = 3.5) //maybe possible to remove in 3.4 public DynamicReference(@NotNull String role, @NotNull SNode sourceNode, @NotNull SNode immatureTargetNode) { super(role, sourceNode, null, immatureTargetNode); } /* * create 'mature' reference */ @Deprecated public DynamicReference(@NotNull String role, @NotNull SNode sourceNode, @Nullable SModelReference targetModelReference, String resolveInfo) { this(((ConceptMetaInfoConverter) sourceNode.getConcept()).convertAssociation(role), sourceNode, targetModelReference == null ? null : targetModelReference.getName(), resolveInfo); } public DynamicReference(@NotNull SReferenceLink role, @NotNull SNode sourceNode, @Nullable SModelReference targetModelReference, String resolveInfo) { this(role, sourceNode, targetModelReference == null ? null : targetModelReference.getName(), resolveInfo); } public static DynamicReference createDynamicReference(@NotNull SReferenceLink role, @NotNull SNode sourceNode, @Nullable String modelName, String resolveInfo) { return new DynamicReference(role, sourceNode, modelName == null ? null : new SModelName(modelName), resolveInfo); } private DynamicReference(@NotNull SReferenceLink role, @NotNull SNode sourceNode, @Nullable SModelName modelName, String resolveInfo) { super(role, sourceNode, null, null); if (modelName != null && !resolveInfo.startsWith(modelName.getLongName()) && isTargetClassifier(role)) { // hack for classifiers resolving with specified targetModelReference. For now (18/04/2012) targetModelReference used only for Classifiers (in stubs and [model]node construction). setResolveInfo(modelName.getLongName() + '.' + resolveInfo); } else { setResolveInfo(resolveInfo); } } private static boolean isTargetClassifier(@NotNull SReferenceLink role) { SAbstractConcept lnkTarget = role.getTargetConcept(); if (lnkTarget == null) { return false; } return lnkTarget.isSubConceptOf(SNodeUtil.concept_Classifier); } @Override protected SNode getTargetNode_internal() { // seems like getTargetNode() doesn't make sense if source node is detached if (mySourceNode.getModel() == null) { assert myHasBeenResolve : "Taking target node of dynamic reference whose source node is not in a model"; return myCachedTargetNode; } final Set<DynamicReference> currentRefs = currentlyResolved.get(); final Set<DynamicReference> loggedRefs = currentlySourceNodeLogged.get(); if (currentRefs.contains(this)) { // loop detected! if (!loggedRefs.contains(this)) { // it's not spurious loop, via logging. it's real, let's complain LOG.errorWithTrace("Loop detected in dynamic references (number of current dyn. refs: " + currentRefs.size() + ")"); } return null; } currentRefs.add(this); try { if (myImmatureTargetNode != null) { synchronized (this) { if (!makeIndirect()) { return myImmatureTargetNode; } } } if (getResolveInfo() == null) { reportErrorWithOrigin("bad reference: no resolve info"); return null; } Scope scope = ModelConstraints.getScope(this); if (scope instanceof ErrorScope) { reportErrorWithOrigin("cannot obtain scope for reference `" + getRole() + "': " + ((ErrorScope) scope).getMessage()); return null; } SNode targetNode = null; try { targetNode = scope.resolve(getSourceNode(), getResolveInfo()); } catch (Throwable t) { LOG.warning("Exception was thrown while dynamic reference resolving", t); } if (targetNode == null) { reportErrorWithOrigin("cannot resolve reference by string: '" + getResolveInfo() + "'"); } myHasBeenResolve = true; myCachedTargetNode = targetNode; return targetNode; } finally { // cleaning up our loop checking stuff currentRefs.remove(this); } } @Override public SNodeReference getTargetNodeReference() { SNode targetNode = getTargetNode_internal(); if (targetNode == null) { return new SNodePointer(null); } return targetNode.getReference(); } private void reportErrorWithOrigin(String message) { Set<DynamicReference> refs = currentlySourceNodeLogged.get(); try { refs.add(this); if (myOrigin != null) { List<ProblemDescription> result = new ArrayList<ProblemDescription>(2); if (myOrigin.getInputNode() != null) { result.add(new ProblemDescription(myOrigin.getInputNode(), " -- was input: " + myOrigin.getInputNode().toString())); } if (myOrigin.getTemplate() != null) { result.add(new ProblemDescription(myOrigin.getTemplate(), " -- was template: " + myOrigin.getTemplate().toString())); } if (result.size() > 0) { error(message, false, result.toArray(new ProblemDescription[result.size()])); return; } } error(message, false); } finally { refs.remove(this); } } @Override public void makeDirect() { } @Override protected synchronized void makeMature() { } @Nullable public DynamicReferenceOrigin getOrigin() { return myOrigin; } public void setOrigin(@Nullable DynamicReferenceOrigin origin) { myOrigin = origin; } @Immutable public static class DynamicReferenceOrigin { private final SNodeReference template; private final SNodeReference inputNode; public DynamicReferenceOrigin(SNodeReference template, SNodeReference inputNode) { this.template = template; this.inputNode = inputNode; } public SNodeReference getTemplate() { return template; } public SNodeReference getInputNode() { return inputNode; } } }