/* * Copyright (c) 2014 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.data.impl.schema.tree; import static com.google.common.base.Preconditions.checkArgument; import com.google.common.base.Optional; import com.google.common.base.Preconditions; import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier; import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier.PathArgument; import org.opendaylight.yangtools.yang.data.api.schema.NormalizedNode; import org.opendaylight.yangtools.yang.data.api.schema.tree.IncorrectDataStructureException; import org.opendaylight.yangtools.yang.data.api.schema.tree.ModificationType; import org.opendaylight.yangtools.yang.data.api.schema.tree.spi.TreeNode; import org.opendaylight.yangtools.yang.data.api.schema.tree.spi.TreeNodeFactory; import org.opendaylight.yangtools.yang.data.api.schema.tree.spi.Version; import org.opendaylight.yangtools.yang.model.api.DataSchemaNode; abstract class AbstractValueNodeModificationStrategy<T extends DataSchemaNode> extends SchemaAwareApplyOperation { private final Class<? extends NormalizedNode<?, ?>> nodeClass; private final T schema; protected AbstractValueNodeModificationStrategy(final T schema, final Class<? extends NormalizedNode<?, ?>> nodeClass) { this.nodeClass = Preconditions.checkNotNull(nodeClass); this.schema = schema; } @Override protected final void verifyStructure(final NormalizedNode<?, ?> writtenValue, final boolean verifyChildren) { checkArgument(nodeClass.isInstance(writtenValue), "Node should must be of type %s", nodeClass); } @Override public final Optional<ModificationApplyOperation> getChild(final PathArgument child) { throw new UnsupportedOperationException("Node " + schema.getPath() + "is leaf type node. Child nodes not allowed"); } @Override protected final ChildTrackingPolicy getChildPolicy() { return ChildTrackingPolicy.NONE; } @Override protected final TreeNode applyTouch(final ModifiedNode modification, final TreeNode currentMeta, final Version version) { throw new UnsupportedOperationException("Node " + schema.getPath() + "is leaf type node. Subtree change is not allowed."); } @Override protected final TreeNode applyMerge(final ModifiedNode modification, final TreeNode currentMeta, final Version version) { // Just overwrite whatever was there, but be sure to run validation verifyStructure(modification.getWrittenValue(), true); modification.resolveModificationType(ModificationType.WRITE); return applyWrite(modification, null, version); } @Override protected final TreeNode applyWrite(final ModifiedNode modification, final Optional<TreeNode> currentMeta, final Version version) { return TreeNodeFactory.createTreeNode(modification.getWrittenValue(), version); } @Override protected final void checkTouchApplicable(final YangInstanceIdentifier path, final NodeModification modification, final Optional<TreeNode> current, final Version version) throws IncorrectDataStructureException { throw new IncorrectDataStructureException(path, "Subtree modification is not allowed."); } @Override void mergeIntoModifiedNode(final ModifiedNode node, final NormalizedNode<?, ?> value, final Version version) { switch (node.getOperation()) { // Delete performs a data dependency check on existence of the node. Performing a merge // on DELETE means we // are really performing a write. case DELETE: case WRITE: node.write(value); break; default: node.updateValue(LogicalOperation.MERGE, value); } } @Override void recursivelyVerifyStructure(final NormalizedNode<?, ?> value) { verifyStructure(value, false); } }