/*
* This file is part of Sponge, licensed under the MIT License (MIT).
*
* Copyright (c) SpongePowered <https://www.spongepowered.org>
* Copyright (c) contributors
*
* 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.
*/
package org.spongepowered.server.network;
import static com.google.common.base.Preconditions.checkArgument;
import static com.google.common.base.Preconditions.checkNotNull;
import static com.google.common.base.Preconditions.checkState;
import com.google.common.base.Objects;
import com.google.common.collect.Sets;
import io.netty.buffer.ByteBuf;
import io.netty.buffer.Unpooled;
import it.unimi.dsi.fastutil.bytes.Byte2ObjectMap;
import it.unimi.dsi.fastutil.bytes.Byte2ObjectOpenHashMap;
import net.minecraft.entity.player.EntityPlayerMP;
import net.minecraft.network.PacketBuffer;
import net.minecraft.network.play.server.SPacketCustomPayload;
import org.spongepowered.api.Platform;
import org.spongepowered.api.entity.living.player.Player;
import org.spongepowered.api.network.ChannelBinding;
import org.spongepowered.api.network.ChannelRegistrar;
import org.spongepowered.api.network.Message;
import org.spongepowered.api.network.MessageHandler;
import org.spongepowered.api.network.RemoteConnection;
import org.spongepowered.api.plugin.PluginContainer;
import org.spongepowered.common.SpongeImpl;
import org.spongepowered.common.network.SpongeNetworkManager;
import org.spongepowered.server.interfaces.IMixinNetHandlerPlayServer;
import java.util.HashMap;
import java.util.Map;
import java.util.Set;
public final class VanillaIndexedMessageChannel extends VanillaChannelBinding implements ChannelBinding.IndexedMessageChannel {
private final Byte2ObjectMap<IndexedMessageType<?>> messageIds = new Byte2ObjectOpenHashMap<>();
private final Map<Class<? extends Message>, IndexedMessageType<?>> messageClasses = new HashMap<>();
public VanillaIndexedMessageChannel(ChannelRegistrar registrar, String name, PluginContainer owner) {
super(registrar, name, owner);
}
@Override
public void registerMessage(Class<? extends Message> messageClass, int messageId) {
validate();
checkNotNull(messageClass, "messageClass");
byte id = (byte) messageId;
IndexedMessageType<?> type = this.messageIds.get(id);
checkState(type == null, "Message ID %s is already assigned to %s", id, type);
type = new IndexedMessageType<>(id, messageClass);
this.messageIds.put(id, type);
this.messageClasses.put(messageClass, type);
}
@Override
public <M extends Message> void registerMessage(Class<M> messageClass, int messageId, MessageHandler<M> handler) {
checkNotNull(handler, "handler");
registerMessage(messageClass, messageId);
registerHandler(messageClass, handler);
}
@Override
public <M extends Message> void registerMessage(Class<M> messageClass, int messageId, Platform.Type side, MessageHandler<M> handler) {
checkNotNull(handler, "handler");
registerMessage(messageClass, messageId);
registerHandler(messageClass, side, handler);
}
@SuppressWarnings("unchecked")
private <M extends Message> void registerHandler(Class<M> messageClass, MessageHandler<M> handler) {
IndexedMessageType<M> message = (IndexedMessageType<M>) this.messageClasses.get(messageClass);
checkArgument(message != null, "Unregistered message class: %s", messageClass);
message.handlers.add(handler);
}
private <M extends Message> void registerHandler(Class<M> messageClass, Platform.Type side, MessageHandler<M> handler) {
if (side == Platform.Type.SERVER) {
registerHandler(messageClass, handler);
}
}
@Override
public <M extends Message> void addHandler(Class<M> messageClass, MessageHandler<M> handler) {
validate();
checkNotNull(handler, "handler");
registerHandler(messageClass, handler);
}
@Override
public <M extends Message> void addHandler(Class<M> messageClass, Platform.Type side, MessageHandler<M> handler) {
validate();
checkNotNull(handler, "handler");
registerHandler(messageClass, side, handler);
}
@Override
public void post(RemoteConnection connection, PacketBuffer payload) {
try {
byte id = payload.readByte();
IndexedMessageType<?> type = this.messageIds.get(id);
checkNotNull(type, "Unknown message with id %s", id);
type.post(connection, payload.slice());
} catch (Throwable e) {
getOwner().getLogger().error("Failed to read indexed message for channel {} of {}", getName(), getOwner(), e);
}
}
private SPacketCustomPayload createPacket(Message message) {
Class<? extends Message> messageClass = message.getClass();
IndexedMessageType<?> type = this.messageClasses.get(messageClass);
checkNotNull(type, "Unknown message type %s of %s", messageClass, message);
PacketBuffer buffer = new PacketBuffer(Unpooled.buffer());
buffer.writeByte(type.id);
buffer.markWriterIndex();
message.writeTo(SpongeNetworkManager.toChannelBuf(buffer));
return new SPacketCustomPayload(getName(), buffer);
}
@Override
public void sendTo(Player player, Message message) {
validate();
final EntityPlayerMP playerMP = (EntityPlayerMP) player;
if (((IMixinNetHandlerPlayServer) playerMP.connection).supportsChannel(getName())) {
playerMP.connection.sendPacket(createPacket(message));
}
}
@Override
public void sendToServer(Message message) {
validate();
// Nothing to do here
}
@Override
public void sendToAll(Message message) {
validate();
final String name = getName();
SPacketCustomPayload packet = null;
for (EntityPlayerMP player : SpongeImpl.getServer().getPlayerList().getPlayers()) {
if (((IMixinNetHandlerPlayServer) player.connection).supportsChannel(name)) {
if (packet == null) {
packet = createPacket(message);
}
player.connection.sendPacket(packet);
}
}
}
private final class IndexedMessageType<T extends Message> {
private final byte id;
private final Class<T> messageClass;
private final Set<MessageHandler<T>> handlers = Sets.newIdentityHashSet();
private IndexedMessageType(byte id, Class<T> messageClass) {
this.id = id;
this.messageClass = messageClass;
}
private T read(ByteBuf buf) throws Exception {
// Woo, reflection!
T message = this.messageClass.newInstance();
message.readFrom(SpongeNetworkManager.toChannelBuf(buf));
return message;
}
private void post(RemoteConnection connection, T message) {
for (MessageHandler<T> listener : this.handlers) {
try {
listener.handleMessage(message, connection, Platform.Type.SERVER);
} catch (Throwable e) {
getOwner().getLogger().error("Could not pass indexed message {} on channel '{}' to {}", message, getName(), getOwner(), e);
}
}
}
private void post(RemoteConnection connection, ByteBuf buf) throws Exception {
post(connection, read(buf));
}
@Override
public String toString() {
return Objects.toStringHelper(this)
.add("id", this.id)
.add("class", this.messageClass)
.toString();
}
}
}