org.pkl.thirdparty.msgpack.value.impl.ImmutableMapValueImpl Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of pkl-config-java-all Show documentation
Show all versions of pkl-config-java-all Show documentation
Shaded fat Jar for pkl-config-java, a Java config library based on the Pkl config language.
//
// MessagePack for Java
//
// 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.pkl.thirdparty.msgpack.value.impl;
import org.pkl.thirdparty.msgpack.core.MessagePacker;
import org.pkl.thirdparty.msgpack.value.ImmutableMapValue;
import org.pkl.thirdparty.msgpack.value.MapValue;
import org.pkl.thirdparty.msgpack.value.Value;
import org.pkl.thirdparty.msgpack.value.ValueType;
import java.io.IOException;
import java.util.AbstractCollection;
import java.util.AbstractMap;
import java.util.AbstractSet;
import java.util.Arrays;
import java.util.Collection;
import java.util.Iterator;
import java.util.Map;
import java.util.NoSuchElementException;
import java.util.Set;
/**
* {@code ImmutableMapValueImpl} Implements {@code ImmutableMapValue} using a {@code Value[]} field.
*
* @see org.pkl.thirdparty.msgpack.value.MapValue
*/
public class ImmutableMapValueImpl
extends AbstractImmutableValue
implements ImmutableMapValue
{
private static final ImmutableMapValueImpl EMPTY = new ImmutableMapValueImpl(new Value[0]);
public static ImmutableMapValue empty()
{
return EMPTY;
}
private final Value[] kvs;
public ImmutableMapValueImpl(Value[] kvs)
{
this.kvs = kvs;
}
@Override
public ValueType getValueType()
{
return ValueType.MAP;
}
@Override
public ImmutableMapValue immutableValue()
{
return this;
}
@Override
public ImmutableMapValue asMapValue()
{
return this;
}
@Override
public Value[] getKeyValueArray()
{
return Arrays.copyOf(kvs, kvs.length);
}
@Override
public int size()
{
return kvs.length / 2;
}
@Override
public Set keySet()
{
return new KeySet(kvs);
}
@Override
public Set> entrySet()
{
return new EntrySet(kvs);
}
@Override
public Collection values()
{
return new ValueCollection(kvs);
}
@Override
public Map map()
{
return new ImmutableMapValueMap(kvs);
}
@Override
public void writeTo(MessagePacker pk)
throws IOException
{
pk.packMapHeader(kvs.length / 2);
for (int i = 0; i < kvs.length; i++) {
kvs[i].writeTo(pk);
}
}
@Override
public boolean equals(Object o)
{
if (o == this) {
return true;
}
if (!(o instanceof Value)) {
return false;
}
Value v = (Value) o;
if (!v.isMapValue()) {
return false;
}
MapValue mv = v.asMapValue();
return map().equals(mv.map());
}
@Override
public int hashCode()
{
int h = 0;
for (int i = 0; i < kvs.length; i += 2) {
h += kvs[i].hashCode() ^ kvs[i + 1].hashCode();
}
return h;
}
@Override
public String toJson()
{
if (kvs.length == 0) {
return "{}";
}
StringBuilder sb = new StringBuilder();
sb.append("{");
appendJsonKey(sb, kvs[0]);
sb.append(":");
sb.append(kvs[1].toJson());
for (int i = 2; i < kvs.length; i += 2) {
sb.append(",");
appendJsonKey(sb, kvs[i]);
sb.append(":");
sb.append(kvs[i + 1].toJson());
}
sb.append("}");
return sb.toString();
}
private static void appendJsonKey(StringBuilder sb, Value key)
{
if (key.isRawValue()) {
sb.append(key.toJson());
}
else {
ImmutableStringValueImpl.appendJsonString(sb, key.toString());
}
}
@Override
public String toString()
{
if (kvs.length == 0) {
return "{}";
}
StringBuilder sb = new StringBuilder();
sb.append("{");
appendString(sb, kvs[0]);
sb.append(":");
appendString(sb, kvs[1]);
for (int i = 2; i < kvs.length; i += 2) {
sb.append(",");
appendString(sb, kvs[i]);
sb.append(":");
appendString(sb, kvs[i + 1]);
}
sb.append("}");
return sb.toString();
}
private static void appendString(StringBuilder sb, Value value)
{
if (value.isRawValue()) {
sb.append(value.toJson());
}
else {
sb.append(value.toString());
}
}
private static class ImmutableMapValueMap
extends AbstractMap
{
private final Value[] kvs;
public ImmutableMapValueMap(Value[] kvs)
{
this.kvs = kvs;
}
@Override
public Set> entrySet()
{
return new EntrySet(kvs);
}
}
private static class EntrySet
extends AbstractSet>
{
private final Value[] kvs;
EntrySet(Value[] kvs)
{
this.kvs = kvs;
}
@Override
public int size()
{
return kvs.length / 2;
}
@Override
public Iterator> iterator()
{
return new EntrySetIterator(kvs);
}
}
private static class EntrySetIterator
implements Iterator>
{
private final Value[] kvs;
private int index;
EntrySetIterator(Value[] kvs)
{
this.kvs = kvs;
this.index = 0;
}
@Override
public boolean hasNext()
{
return index < kvs.length;
}
@Override
public Map.Entry next()
{
if (index >= kvs.length) {
throw new NoSuchElementException(); // TODO message
}
Value key = kvs[index];
Value value = kvs[index + 1];
Map.Entry pair = new AbstractMap.SimpleImmutableEntry(key, value);
index += 2;
return pair;
}
@Override
public void remove()
{
throw new UnsupportedOperationException(); // TODO message
}
}
private static class KeySet
extends AbstractSet
{
private Value[] kvs;
KeySet(Value[] kvs)
{
this.kvs = kvs;
}
@Override
public int size()
{
return kvs.length / 2;
}
@Override
public Iterator iterator()
{
return new EntryIterator(kvs, 0);
}
}
private static class ValueCollection
extends AbstractCollection
{
private Value[] kvs;
ValueCollection(Value[] kvs)
{
this.kvs = kvs;
}
@Override
public int size()
{
return kvs.length / 2;
}
@Override
public Iterator iterator()
{
return new EntryIterator(kvs, 1);
}
}
private static class EntryIterator
implements Iterator
{
private Value[] kvs;
private int index;
public EntryIterator(Value[] kvs, int offset)
{
this.kvs = kvs;
this.index = offset;
}
@Override
public boolean hasNext()
{
return index < kvs.length;
}
@Override
public Value next()
{
int i = index;
if (i >= kvs.length) {
throw new NoSuchElementException();
}
index = i + 2;
return kvs[i];
}
@Override
public void remove()
{
throw new UnsupportedOperationException();
}
}
}
© 2015 - 2024 Weber Informatics LLC | Privacy Policy