All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.curioswitch.common.testing.assertj.proto.FieldScopeLogicMap Maven / Gradle / Ivy

There is a newer version: 0.2.1
Show newest version
/*
 * MIT License
 *
 * Copyright (c) 2018 Choko ([email protected])
 *
 * Permission is hereby granted, free of charge, to any person obtaining a copy
 * of this software and associated documentation files (the "Software"), to deal
 * in the Software without restriction, including without limitation the rights
 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 * copies of the Software, and to permit persons to whom the Software is
 * furnished to do so, subject to the following conditions:
 *
 * The above copyright notice and this permission notice shall be included in all
 * copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
 * SOFTWARE.
 */
/*
 * Copyright (c) 2018 Google, Inc.
 *
 * 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 org.curioswitch.common.testing.assertj.proto;

import com.google.auto.value.AutoValue;
import com.google.common.base.Optional;
import com.google.common.collect.ImmutableList;
import com.google.protobuf.Descriptors.Descriptor;

/**
 * An immutable map of {@link FieldScopeLogic} to an arbitrary value type.
 *
 * 

This is the field-path based analog to {@link com.google.common.collect.RangeMap}. Setting a * value {@code v} for a {@link FieldScopeLogic} {@code l} on the {@code FieldScopeLogicMap} means * that, for any field path {@code f} contained in {@code l}, the value mapped to {@code f} is * {@code v}. This overrides any previous {FieldScopeLogic -> value} pairs/mappings. * *

Unlike {@link com.google.common.collect.RangeMap}, this class does not support analysis of its * internals, only {@link #get} operations, and it is not performant. All {@code get} and {@code * with} operations are O(N), where N = number of mappings, so this class is intended only for small * numbers of entries. */ class FieldScopeLogicMap implements FieldScopeLogicContainer> { @AutoValue abstract static class Entry { abstract FieldScopeLogic fieldScopeLogic(); abstract V value(); static Entry of(FieldScopeLogic fieldScopeLogic, V value) { return new AutoValue_FieldScopeLogicMap_Entry<>(fieldScopeLogic, value); } } private static final FieldScopeLogicMap EMPTY_INSTANCE = new FieldScopeLogicMap<>(ImmutableList.>of()); // Key -> value mappings for this map. Earlier entries override later ones. private final ImmutableList> entries; private FieldScopeLogicMap(Iterable> entries) { this.entries = ImmutableList.copyOf(entries); } public boolean isEmpty() { return entries.isEmpty(); } public Optional get( Descriptor rootDescriptor, FieldDescriptorOrUnknown fieldDescriptorOrUnknown) { // Earlier entries override later ones, so we don't need to iterate backwards. for (Entry entry : entries) { if (entry.fieldScopeLogic().contains(rootDescriptor, fieldDescriptorOrUnknown)) { return Optional.of(entry.value()); } } return Optional.absent(); } /** Returns a new immutable map that adds the given fields -> value mapping. */ public FieldScopeLogicMap with(FieldScopeLogic fieldScopeLogic, V value) { ImmutableList.Builder> newEntries = ImmutableList.builder(); // Earlier entries override later ones, so we insert the new one at the front of the list. newEntries.add(Entry.of(fieldScopeLogic, value)); newEntries.addAll(entries); return new FieldScopeLogicMap<>(newEntries.build()); } @Override public FieldScopeLogicMap subScope( Descriptor rootDescriptor, FieldDescriptorOrUnknown fieldDescriptorOrUnknown) { ImmutableList.Builder> newEntries = ImmutableList.builderWithExpectedSize(entries.size()); for (Entry entry : entries) { newEntries.add( Entry.of( entry.fieldScopeLogic().subScope(rootDescriptor, fieldDescriptorOrUnknown), entry.value())); } return new FieldScopeLogicMap<>(newEntries.build()); } @Override public void validate( Descriptor rootDescriptor, FieldDescriptorValidator fieldDescriptorValidator) { for (Entry entry : entries) { entry.fieldScopeLogic().validate(rootDescriptor, fieldDescriptorValidator); } } @SuppressWarnings("unchecked") // Implementation is fully variant. public static FieldScopeLogicMap empty() { return (FieldScopeLogicMap) EMPTY_INSTANCE; } /** Returns a map which maps all fields to the given value by default. */ public static FieldScopeLogicMap defaultValue(V value) { return new FieldScopeLogicMap<>(ImmutableList.of(Entry.of(FieldScopeLogic.all(), value))); } }