Allow path on view tags

Signed-off-by: TheMode <themode@outlook.fr>
This commit is contained in:
TheMode 2022-05-02 21:30:22 +02:00
parent 68fc705cd3
commit d8a1003368
2 changed files with 123 additions and 43 deletions

View File

@ -13,6 +13,7 @@ import org.jglrxavpok.hephaistos.nbt.NBTType;
import org.jglrxavpok.hephaistos.nbt.mutable.MutableNBTCompound; import org.jglrxavpok.hephaistos.nbt.mutable.MutableNBTCompound;
import java.lang.invoke.VarHandle; import java.lang.invoke.VarHandle;
import java.util.function.Supplier;
import java.util.function.UnaryOperator; import java.util.function.UnaryOperator;
final class TagHandlerImpl implements TagHandler { final class TagHandlerImpl implements TagHandler {
@ -41,21 +42,16 @@ final class TagHandlerImpl implements TagHandler {
@Override @Override
public <T> @UnknownNullability T getTag(@NotNull Tag<T> tag) { public <T> @UnknownNullability T getTag(@NotNull Tag<T> tag) {
if (tag.isView()) return tag.read(asCompound()); return read(entries, tag, this::asCompound);
return read(entries, tag);
} }
@Override @Override
public <T> void setTag(@NotNull Tag<T> tag, @Nullable T value) { public <T> void setTag(@NotNull Tag<T> tag, @Nullable T value) {
if (tag.isView()) { TagHandlerImpl local = this;
MutableNBTCompound viewCompound = new MutableNBTCompound();
tag.write(viewCompound, value);
updateContent(viewCompound);
} else {
final int tagIndex = tag.index;
final Tag.PathEntry[] paths = tag.path; final Tag.PathEntry[] paths = tag.path;
final boolean present = value != null; final boolean present = value != null;
TagHandlerImpl local = this; final int tagIndex = tag.index;
final boolean isView = tag.isView();
synchronized (this) { synchronized (this) {
if (paths != null) { if (paths != null) {
if ((local = traversePathWrite(this, paths, present)) == null) if ((local = traversePathWrite(this, paths, present)) == null)
@ -63,10 +59,19 @@ final class TagHandlerImpl implements TagHandler {
} }
SPMCMap entries = local.entries; SPMCMap entries = local.entries;
if (present) { if (present) {
if (!isView) {
entries.put(tagIndex, valueToEntry(local, tag, value)); entries.put(tagIndex, valueToEntry(local, tag, value));
} else {
local.updateContent((NBTCompound) tag.entry.write(value));
return;
}
} else { } else {
// Remove recursively // Remove recursively
if (!isView) {
if (entries.remove(tagIndex) == null) return; if (entries.remove(tagIndex) == null) return;
} else {
entries.clear();
}
if (paths != null) { if (paths != null) {
TagHandlerImpl tmp = local; TagHandlerImpl tmp = local;
int i = paths.length; int i = paths.length;
@ -81,7 +86,6 @@ final class TagHandlerImpl implements TagHandler {
assert !local.entries.rehashed; assert !local.entries.rehashed;
} }
} }
}
private <T> Entry<?> valueToEntry(TagHandlerImpl parent, Tag<T> tag, @NotNull T value) { private <T> Entry<?> valueToEntry(TagHandlerImpl parent, Tag<T> tag, @NotNull T value) {
if (value instanceof NBT nbt) { if (value instanceof NBT nbt) {
@ -165,13 +169,21 @@ final class TagHandlerImpl implements TagHandler {
return cache; return cache;
} }
private static <T> T read(Int2ObjectOpenHashMap<Entry<?>> entries, Tag<T> tag) { private static <T> T read(Int2ObjectOpenHashMap<Entry<?>> entries, Tag<T> tag,
Supplier<NBTCompound> rootCompoundSupplier) {
final Tag.PathEntry[] paths = tag.path; final Tag.PathEntry[] paths = tag.path;
TagHandlerImpl pathHandler = null;
if (paths != null) { if (paths != null) {
// Must be a path-able entry if ((pathHandler = traversePathRead(paths, entries)) == null)
if ((entries = traversePathRead(paths, entries)) == null) return tag.createDefault(); // Must be a path-able entry, but not present
return tag.createDefault(); entries = pathHandler.entries;
} }
if (tag.isView()) {
return tag.read(pathHandler != null ?
pathHandler.asCompound() : rootCompoundSupplier.get());
}
final Entry<?> entry; final Entry<?> entry;
if ((entry = entries.get(tag.index)) == null) { if ((entry = entries.get(tag.index)) == null) {
return tag.createDefault(); return tag.createDefault();
@ -189,25 +201,29 @@ final class TagHandlerImpl implements TagHandler {
return type == null || type == nbt.getID() ? serializerEntry.read(nbt) : tag.createDefault(); return type == null || type == nbt.getID() ? serializerEntry.read(nbt) : tag.createDefault();
} }
private static Int2ObjectOpenHashMap<Entry<?>> traversePathRead(Tag.PathEntry[] paths, private static TagHandlerImpl traversePathRead(Tag.PathEntry[] paths,
Int2ObjectOpenHashMap<Entry<?>> entries) { Int2ObjectOpenHashMap<Entry<?>> entries) {
assert paths != null && paths.length > 0;
TagHandlerImpl result = null;
for (var path : paths) { for (var path : paths) {
final Entry<?> entry; final Entry<?> entry;
if ((entry = entries.get(path.index())) == null) if ((entry = entries.get(path.index())) == null)
return null; return null;
if (entry instanceof PathEntry pathEntry) { if (entry instanceof PathEntry pathEntry) {
entries = pathEntry.value.entries; result = pathEntry.value;
} else if (entry.updatedNbt() instanceof NBTCompound compound) { } else if (entry.updatedNbt() instanceof NBTCompound compound) {
// Slow path forcing a conversion of the structure to NBTCompound // Slow path forcing a conversion of the structure to NBTCompound
// TODO should the handler be cached inside the entry? // TODO should the handler be cached inside the entry?
TagHandlerImpl tmp = fromCompound(compound); result = fromCompound(compound);
entries = tmp.entries;
} else { } else {
// Entry is not path-able // Entry is not path-able
return null; return null;
} }
assert result != null;
entries = result.entries;
} }
return entries; assert result != null;
return result;
} }
private record Cache(Int2ObjectOpenHashMap<Entry<?>> entries, NBTCompound compound) implements TagReadable { private record Cache(Int2ObjectOpenHashMap<Entry<?>> entries, NBTCompound compound) implements TagReadable {
@ -215,8 +231,7 @@ final class TagHandlerImpl implements TagHandler {
@Override @Override
public <T> @UnknownNullability T getTag(@NotNull Tag<T> tag) { public <T> @UnknownNullability T getTag(@NotNull Tag<T> tag) {
if (tag.isView()) return tag.read(compound); return read(entries, tag, () -> compound);
return read(entries, tag);
} }
} }

View File

@ -81,6 +81,71 @@ public class TagViewTest {
""", handler.asCompound()); """, handler.asCompound());
} }
@Test
public void empty() {
var handler = TagHandler.newHandler();
var tag = Tag.View(new TagSerializer<Entry>() {
@Override
public @Nullable Entry read(@NotNull TagReadable reader) {
// Empty
return null;
}
@Override
public void write(@NotNull TagWritable writer, @NotNull Entry value) {
// Empty
}
});
assertNull(handler.getTag(tag));
assertFalse(handler.hasTag(tag));
var entry = new Entry("hello");
handler.setTag(tag, entry);
assertNull(handler.getTag(tag));
assertFalse(handler.hasTag(tag));
assertEqualsSNBT("{}", handler.asCompound());
handler.removeTag(tag);
assertFalse(handler.hasTag(tag));
assertNull(handler.getTag(VIEW_TAG));
assertEqualsSNBT("{}", handler.asCompound());
}
@Test
public void path() {
var handler = TagHandler.newHandler();
var tag = VIEW_TAG.path("path");
assertNull(handler.getTag(tag));
assertFalse(handler.hasTag(tag));
var entry = new Entry("hello");
handler.setTag(tag, entry);
assertTrue(handler.hasTag(tag));
assertEquals(entry, handler.getTag(tag));
handler.removeTag(tag);
assertFalse(handler.hasTag(tag));
assertNull(handler.getTag(tag));
}
@Test
public void pathSnbt() {
var handler = TagHandler.newHandler();
var tag = VIEW_TAG.path("path");
var entry = new Entry("hello");
handler.setTag(tag, entry);
assertEqualsSNBT("""
{
"path":{
"value":"hello"
}
}
""", handler.asCompound());
handler.removeTag(tag);
assertEqualsSNBT("{}", handler.asCompound());
}
@Test @Test
public void compoundSerializer() { public void compoundSerializer() {
var tag = Tag.View(TagSerializer.COMPOUND); var tag = Tag.View(TagSerializer.COMPOUND);