/* * 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.Maps; import java.util.Collection; import java.util.HashMap; import java.util.Map; import org.opendaylight.yangtools.util.ModifiableMapPhase; import org.opendaylight.yangtools.util.UnmodifiableMapPhase; import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier.PathArgument; import org.opendaylight.yangtools.yang.data.api.schema.DataContainerChild; import org.opendaylight.yangtools.yang.data.api.schema.DataContainerNode; import org.opendaylight.yangtools.yang.data.impl.schema.builder.api.DataContainerNodeBuilder; import org.opendaylight.yangtools.yang.data.impl.schema.builder.api.NormalizedNodeContainerBuilder; import org.opendaylight.yangtools.yang.data.impl.schema.nodes.AbstractImmutableDataContainerNode; import org.opendaylight.yangtools.yang.data.impl.schema.nodes.CloneableMap; abstract class AbstractImmutableDataContainerNodeBuilder> implements DataContainerNodeBuilder { private static final int DEFAULT_CAPACITY = 4; private Map> value; private I nodeIdentifier; /* * Tracks whether the builder is dirty, e.g. whether the value map has been used * to construct a child. If it has, we detect this condition before any further * modification and create a new value map with same contents. This way we do not * force a map copy if the builder is not reused. */ private boolean dirty; protected AbstractImmutableDataContainerNodeBuilder() { this.value = new HashMap<>(DEFAULT_CAPACITY); this.dirty = false; } protected AbstractImmutableDataContainerNodeBuilder(final int sizeHint) { if (sizeHint >= 0) { this.value = Maps.newHashMapWithExpectedSize(sizeHint); } else { this.value = new HashMap<>(DEFAULT_CAPACITY); } this.dirty = false; } protected AbstractImmutableDataContainerNodeBuilder(final AbstractImmutableDataContainerNode node) { this.nodeIdentifier = node.getIdentifier(); /* * This quite awkward. What we actually want to be saying here is: give me * a copy-on-write view of your children. The API involved in that could be * a bit hairy, so we do the next best thing and rely on the fact that the * returned object implements a specific interface, which leaks the functionality * we need. */ this.value = node.getChildren(); this.dirty = true; } protected final I getNodeIdentifier() { return nodeIdentifier; } protected final DataContainerChild getChild(final PathArgument child) { return value.get(child); } protected final Map> buildValue() { if (value instanceof ModifiableMapPhase) { return ((ModifiableMapPhase>)value) .toUnmodifiableMap(); } dirty = true; return value; } private void checkDirty() { if (dirty) { if (value instanceof UnmodifiableMapPhase) { value = ((UnmodifiableMapPhase>) value) .toModifiableMap(); } else if (value instanceof CloneableMap) { value = ((CloneableMap>) value) .createMutableClone(); } else { value = new HashMap<>(value); } dirty = false; } } @Override public DataContainerNodeBuilder withValue( final Collection> withValue) { // TODO Replace or putAll ? for (final DataContainerChild dataContainerChild : withValue) { withChild(dataContainerChild); } return this; } @Override public DataContainerNodeBuilder withChild(final DataContainerChild child) { checkDirty(); this.value.put(child.getIdentifier(), child); return this; } @Override public DataContainerNodeBuilder withoutChild(final PathArgument key) { checkDirty(); this.value.remove(key); return this; } @Override public DataContainerNodeBuilder withNodeIdentifier(final I withNodeIdentifier) { this.nodeIdentifier = withNodeIdentifier; return this; } @Override public DataContainerNodeBuilder addChild( final DataContainerChild child) { return withChild(child); } @Override public NormalizedNodeContainerBuilder, R> removeChild(final PathArgument key) { return withoutChild(key); } }