/* * Copyright (c) 2013 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.builder.impl; import com.google.common.collect.Iterables; import java.util.Collection; import java.util.LinkedHashMap; import java.util.Map; import java.util.Optional; import org.eclipse.jdt.annotation.NonNull; import org.opendaylight.yangtools.util.UnmodifiableCollection; import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier.NodeIdentifier; import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier.NodeIdentifierWithPredicates; import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier.PathArgument; import org.opendaylight.yangtools.yang.data.api.schema.MapEntryNode; import org.opendaylight.yangtools.yang.data.api.schema.MapNode; import org.opendaylight.yangtools.yang.data.api.schema.OrderedMapNode; import org.opendaylight.yangtools.yang.data.impl.schema.builder.api.CollectionNodeBuilder; import org.opendaylight.yangtools.yang.data.impl.schema.builder.api.NormalizedNodeContainerBuilder; import org.opendaylight.yangtools.yang.data.impl.schema.nodes.AbstractImmutableNormalizedNode; public class ImmutableOrderedMapNodeBuilder implements CollectionNodeBuilder { private static final int DEFAULT_CAPACITY = 4; private Map value; private NodeIdentifier nodeIdentifier; private boolean dirty; protected ImmutableOrderedMapNodeBuilder() { this.value = new LinkedHashMap<>(DEFAULT_CAPACITY); this.dirty = false; } protected ImmutableOrderedMapNodeBuilder(final int sizeHint) { if (sizeHint >= 0) { this.value = new LinkedHashMap<>(sizeHint + sizeHint / 3); } else { this.value = new LinkedHashMap<>(DEFAULT_CAPACITY); } this.dirty = false; } protected ImmutableOrderedMapNodeBuilder(final ImmutableOrderedMapNode node) { this.nodeIdentifier = node.getIdentifier(); this.value = node.children; this.dirty = true; } public static @NonNull CollectionNodeBuilder create() { return new ImmutableOrderedMapNodeBuilder(); } public static @NonNull CollectionNodeBuilder create(final int sizeHint) { return new ImmutableOrderedMapNodeBuilder(sizeHint); } public static @NonNull CollectionNodeBuilder create(final MapNode node) { if (!(node instanceof ImmutableOrderedMapNode)) { throw new UnsupportedOperationException(String.format("Cannot initialize from class %s", node.getClass())); } return new ImmutableOrderedMapNodeBuilder((ImmutableOrderedMapNode) node); } private void checkDirty() { if (dirty) { value = new LinkedHashMap<>(value); dirty = false; } } @Override public CollectionNodeBuilder withChild(final MapEntryNode child) { checkDirty(); this.value.put(child.getIdentifier(), child); return this; } @Override public CollectionNodeBuilder withoutChild(final PathArgument key) { checkDirty(); this.value.remove(key); return this; } @Override public CollectionNodeBuilder withValue(final Collection withValue) { // TODO replace or putAll ? for (final MapEntryNode mapEntryNode : withValue) { withChild(mapEntryNode); } return this; } @Override public CollectionNodeBuilder withNodeIdentifier( final NodeIdentifier withNodeIdentifier) { this.nodeIdentifier = withNodeIdentifier; return this; } @Override public OrderedMapNode build() { dirty = true; return new ImmutableOrderedMapNode(nodeIdentifier, value); } @Override public CollectionNodeBuilder addChild( final MapEntryNode child) { return withChild(child); } @Override public NormalizedNodeContainerBuilder removeChild( final PathArgument key) { return withoutChild(key); } protected static final class ImmutableOrderedMapNode extends AbstractImmutableNormalizedNode> implements OrderedMapNode { private final Map children; ImmutableOrderedMapNode(final NodeIdentifier nodeIdentifier, final Map children) { super(nodeIdentifier); this.children = children; } @Override public Optional getChild(final NodeIdentifierWithPredicates child) { return Optional.ofNullable(children.get(child)); } @Override public MapEntryNode getChild(final int position) { return Iterables.get(children.values(), position); } @Override public int size() { return children.size(); } @Override protected int valueHashCode() { return children.hashCode(); } @Override protected boolean valueEquals(final AbstractImmutableNormalizedNode other) { return children.equals(((ImmutableOrderedMapNode) other).children); } @Override public int getSize() { return children.size(); } @Override public Collection getValue() { return UnmodifiableCollection.create(children.values()); } } }