package net.minecraft.network.chat; import com.google.common.primitives.Ints; import com.mojang.serialization.MapCodec; import com.mojang.serialization.codecs.RecordCodecBuilder; import java.security.SignatureException; import java.time.Duration; import java.time.Instant; import java.util.Objects; import java.util.Optional; import java.util.UUID; import java.util.stream.Collectors; import net.minecraft.util.SignatureUpdater; import net.minecraft.util.SignatureValidator; import net.minecraft.util.Util; import org.jspecify.annotations.Nullable; public record PlayerChatMessage( SignedMessageLink link, @Nullable MessageSignature signature, SignedMessageBody signedBody, @Nullable Component unsignedContent, FilterMask filterMask ) { public static final MapCodec MAP_CODEC = RecordCodecBuilder.mapCodec( i -> i.group( SignedMessageLink.CODEC.fieldOf("link").forGetter(PlayerChatMessage::link), MessageSignature.CODEC.optionalFieldOf("signature").forGetter(playerChatMessage -> Optional.ofNullable(playerChatMessage.signature)), SignedMessageBody.MAP_CODEC.forGetter(PlayerChatMessage::signedBody), ComponentSerialization.CODEC.optionalFieldOf("unsigned_content").forGetter(playerChatMessage -> Optional.ofNullable(playerChatMessage.unsignedContent)), FilterMask.CODEC.optionalFieldOf("filter_mask", FilterMask.PASS_THROUGH).forGetter(PlayerChatMessage::filterMask) ) .apply( i, (link, signature, signedBody, unsignedContent, filterMask) -> new PlayerChatMessage( link, (MessageSignature)signature.orElse(null), signedBody, (Component)unsignedContent.orElse(null), filterMask ) ) ); private static final UUID SYSTEM_SENDER = Util.NIL_UUID; public static final Duration MESSAGE_EXPIRES_AFTER_SERVER = Duration.ofMinutes(5L); public static final Duration MESSAGE_EXPIRES_AFTER_CLIENT = MESSAGE_EXPIRES_AFTER_SERVER.plus(Duration.ofMinutes(2L)); public static PlayerChatMessage system(final String content) { return unsigned(SYSTEM_SENDER, content); } public static PlayerChatMessage unsigned(final UUID sender, final String content) { SignedMessageBody body = SignedMessageBody.unsigned(content); SignedMessageLink link = SignedMessageLink.unsigned(sender); return new PlayerChatMessage(link, null, body, null, FilterMask.PASS_THROUGH); } public PlayerChatMessage withUnsignedContent(final Component content) { Component unsignedContent = !content.equals(Component.literal(this.signedContent())) ? content : null; return new PlayerChatMessage(this.link, this.signature, this.signedBody, unsignedContent, this.filterMask); } public PlayerChatMessage removeUnsignedContent() { return this.unsignedContent != null ? new PlayerChatMessage(this.link, this.signature, this.signedBody, null, this.filterMask) : this; } public PlayerChatMessage filter(final FilterMask filterMask) { return this.filterMask.equals(filterMask) ? this : new PlayerChatMessage(this.link, this.signature, this.signedBody, this.unsignedContent, filterMask); } public PlayerChatMessage filter(final boolean filtered) { return this.filter(filtered ? this.filterMask : FilterMask.PASS_THROUGH); } public PlayerChatMessage removeSignature() { SignedMessageBody body = SignedMessageBody.unsigned(this.signedContent()); SignedMessageLink link = SignedMessageLink.unsigned(this.sender()); return new PlayerChatMessage(link, null, body, this.unsignedContent, this.filterMask); } public static void updateSignature(final SignatureUpdater.Output output, final SignedMessageLink link, final SignedMessageBody body) throws SignatureException { output.update(Ints.toByteArray(1)); link.updateSignature(output); body.updateSignature(output); } public boolean verify(final SignatureValidator signatureValidator) { return this.signature != null && this.signature.verify(signatureValidator, output -> updateSignature(output, this.link, this.signedBody)); } public String signedContent() { return this.signedBody.content(); } public Component decoratedContent() { return (Component)Objects.requireNonNullElseGet(this.unsignedContent, () -> Component.literal(this.signedContent())); } public Instant timeStamp() { return this.signedBody.timeStamp(); } public long salt() { return this.signedBody.salt(); } public boolean hasExpiredServer(final Instant now) { return now.isAfter(this.timeStamp().plus(MESSAGE_EXPIRES_AFTER_SERVER)); } public boolean hasExpiredClient(final Instant now) { return now.isAfter(this.timeStamp().plus(MESSAGE_EXPIRES_AFTER_CLIENT)); } public UUID sender() { return this.link.sender(); } public boolean isSystem() { return this.sender().equals(SYSTEM_SENDER); } public boolean hasSignature() { return this.signature != null; } public boolean hasSignatureFrom(final UUID profileId) { return this.hasSignature() && this.link.sender().equals(profileId); } public boolean isFullyFiltered() { return this.filterMask.isFullyFiltered(); } public static String describeSigned(final PlayerChatMessage message) { return "'" + message.signedBody.content() + "' @ " + message.signedBody.timeStamp() + "\n - From: " + message.link.sender() + "/" + message.link.sessionId() + ", message #" + message.link.index() + "\n - Salt: " + message.signedBody.salt() + "\n - Signature: " + MessageSignature.describe(message.signature) + "\n - Last Seen: [\n" + (String)message.signedBody .lastSeen() .entries() .stream() .map(signature -> " " + MessageSignature.describe(signature) + "\n") .collect(Collectors.joining()) + " ]\n"; } }