/* * Copyright (c) 2015 Cisco Systems, Inc. and others. All rights reserved. * * This program and the accompanying materials are made available under the * terms of the Eclipse Public License v1.0 which accompanies this distribution, * and is available at http://www.eclipse.org/legal/epl-v10.html */ package org.opendaylight.yangtools.yang.parser.rfc7950.stmt.extension; import java.util.ArrayDeque; import java.util.Collection; import java.util.Deque; import java.util.Optional; import org.eclipse.jdt.annotation.NonNull; import org.eclipse.jdt.annotation.Nullable; import org.opendaylight.yangtools.util.RecursiveObjectLeaker; import org.opendaylight.yangtools.yang.common.QName; import org.opendaylight.yangtools.yang.model.api.ExtensionDefinition; import org.opendaylight.yangtools.yang.model.api.SchemaPath; import org.opendaylight.yangtools.yang.model.api.meta.EffectiveStatement; import org.opendaylight.yangtools.yang.model.api.stmt.ArgumentEffectiveStatement; import org.opendaylight.yangtools.yang.model.api.stmt.ExtensionEffectiveStatement; import org.opendaylight.yangtools.yang.model.api.stmt.ExtensionStatement; import org.opendaylight.yangtools.yang.model.api.stmt.YinElementEffectiveStatement; import org.opendaylight.yangtools.yang.parser.rfc7950.stmt.AbstractEffectiveDocumentedNode; import org.opendaylight.yangtools.yang.parser.spi.meta.StmtContext; final class ExtensionEffectiveStatementImpl extends AbstractEffectiveDocumentedNode implements ExtensionDefinition, ExtensionEffectiveStatement { private static final class RecursionDetector extends ThreadLocal> { boolean check(final ExtensionEffectiveStatementImpl current) { final Deque stack = get(); if (stack != null) { for (ExtensionEffectiveStatementImpl s : stack) { if (s == current) { return true; } } } return false; } void push(final ExtensionEffectiveStatementImpl current) { Deque stack = get(); if (stack == null) { stack = new ArrayDeque<>(1); set(stack); } stack.push(current); } void pop() { Deque stack = get(); stack.pop(); if (stack.isEmpty()) { remove(); } } } private static final RecursionDetector TOSTRING_DETECTOR = new RecursionDetector(); private final @NonNull QName qname; private final @Nullable String argument; private final @NonNull SchemaPath schemaPath; private final boolean yin; private ExtensionEffectiveStatementImpl( final StmtContext ctx) { super(ctx); this.qname = ctx.coerceStatementArgument(); this.schemaPath = ctx.getSchemaPath().get(); // initFields final Optional optArgumentSubstatement = findFirstEffectiveSubstatement( ArgumentEffectiveStatement.class); if (optArgumentSubstatement.isPresent()) { final ArgumentEffectiveStatement argumentStatement = optArgumentSubstatement.get(); this.argument = argumentStatement.argument().getLocalName(); this.yin = argumentStatement.findFirstEffectiveSubstatement(YinElementEffectiveStatement.class) .map(YinElementEffectiveStatement::argument).orElse(Boolean.FALSE).booleanValue(); } else { this.argument = null; this.yin = false; } } /** * Create a new ExtensionEffectiveStatement, dealing with potential recursion. * * @param ctx Statement context * @return A potentially under-initialized instance */ static ExtensionEffectiveStatement create( final StmtContext ctx) { // Look at the thread-local leak in case we are invoked recursively final ExtensionEffectiveStatementImpl existing = RecursiveObjectLeaker.lookup(ctx, ExtensionEffectiveStatementImpl.class); if (existing != null) { // Careful! this object is not fully initialized! return existing; } RecursiveObjectLeaker.beforeConstructor(ctx); try { // This result is fine, we know it has been completely initialized return new ExtensionEffectiveStatementImpl(ctx); } finally { RecursiveObjectLeaker.afterConstructor(ctx); } } @Override protected Collection> initSubstatements( final Collection> substatementsInit) { // WARNING: this leaks an incompletely-initialized object RecursiveObjectLeaker.inConstructor(this); return super.initSubstatements(substatementsInit); } @Override public QName getQName() { return qname; } @Override @Deprecated public SchemaPath getPath() { return schemaPath; } @Override public String getArgument() { return argument; } @Override public boolean isYinElement() { return yin; } @Override public String toString() { if (TOSTRING_DETECTOR.check(this)) { return recursedToString(); } TOSTRING_DETECTOR.push(this); try { return ExtensionEffectiveStatementImpl.class.getSimpleName() + "[" + "argument=" + argument + ", qname=" + qname + ", schemaPath=" + schemaPath + ", extensionSchemaNodes=" + getUnknownSchemaNodes() + ", yin=" + yin + "]"; } finally { TOSTRING_DETECTOR.pop(); } } private String recursedToString() { return ExtensionEffectiveStatementImpl.class.getSimpleName() + "[" + "argument=" + argument + ", qname=" + qname + ", schemaPath=" + schemaPath + ", yin=" + yin + " ]"; } }