package org.bukkit; import com.google.common.base.Preconditions; import java.util.Locale; import java.util.UUID; import java.util.regex.Pattern; import org.bukkit.plugin.Plugin; import org.jetbrains.annotations.NotNull; /** * Represents a String based key which consists of two components - a namespace * and a key. * * Namespaces may only contain lowercase alphanumeric characters, periods, * underscores, and hyphens. *

* Keys may only contain lowercase alphanumeric characters, periods, * underscores, hyphens, and forward slashes. * */ public final class NamespacedKey { /** * The namespace representing all inbuilt keys. */ public static final String MINECRAFT = "minecraft"; /** * The namespace representing all keys generated by Bukkit for backwards * compatibility measures. */ public static final String BUKKIT = "bukkit"; // private static final Pattern VALID_NAMESPACE = Pattern.compile("[a-z0-9._-]+"); private static final Pattern VALID_KEY = Pattern.compile("[a-z0-9/._-]+"); // private final String namespace; private final String key; /** * Create a key in a specific namespace. * * @param namespace * @param key * @deprecated should never be used by plugins, for internal use only!! */ @Deprecated public NamespacedKey(@NotNull String namespace, @NotNull String key) { Preconditions.checkArgument(namespace != null && VALID_NAMESPACE.matcher(namespace).matches(), "Invalid namespace. Must be [a-z0-9._-]: %s", namespace); Preconditions.checkArgument(key != null && VALID_KEY.matcher(key).matches(), "Invalid key. Must be [a-z0-9/._-]: %s", key); this.namespace = namespace; this.key = key; String string = toString(); Preconditions.checkArgument(string.length() < 256, "NamespacedKey must be less than 256 characters", string); } /** * Create a key in the plugin's namespace. *

* Namespaces may only contain lowercase alphanumeric characters, periods, * underscores, and hyphens. *

* Keys may only contain lowercase alphanumeric characters, periods, * underscores, hyphens, and forward slashes. * * @param plugin the plugin to use for the namespace * @param key the key to create */ public NamespacedKey(@NotNull Plugin plugin, @NotNull String key) { Preconditions.checkArgument(plugin != null, "Plugin cannot be null"); Preconditions.checkArgument(key != null, "Key cannot be null"); this.namespace = plugin.getName().toLowerCase(Locale.ROOT); this.key = key.toLowerCase().toLowerCase(Locale.ROOT); // Check validity after normalization Preconditions.checkArgument(VALID_NAMESPACE.matcher(this.namespace).matches(), "Invalid namespace. Must be [a-z0-9._-]: %s", this.namespace); Preconditions.checkArgument(VALID_KEY.matcher(this.key).matches(), "Invalid key. Must be [a-z0-9/._-]: %s", this.key); String string = toString(); Preconditions.checkArgument(string.length() < 256, "NamespacedKey must be less than 256 characters (%s)", string); } @NotNull public String getNamespace() { return namespace; } @NotNull public String getKey() { return key; } @Override public int hashCode() { int hash = 5; hash = 47 * hash + this.namespace.hashCode(); hash = 47 * hash + this.key.hashCode(); return hash; } @Override public boolean equals(Object obj) { if (obj == null) { return false; } if (getClass() != obj.getClass()) { return false; } final NamespacedKey other = (NamespacedKey) obj; return this.namespace.equals(other.namespace) && this.key.equals(other.key); } @Override public String toString() { return this.namespace + ":" + this.key; } /** * Return a new random key in the {@link #BUKKIT} namespace. * * @return new key * @deprecated should never be used by plugins, for internal use only!! */ @Deprecated @NotNull public static NamespacedKey randomKey() { return new NamespacedKey(BUKKIT, UUID.randomUUID().toString()); } /** * Get a key in the Minecraft namespace. * * @param key the key to use * @return new key in the Minecraft namespace */ @NotNull public static NamespacedKey minecraft(@NotNull String key) { return new NamespacedKey(MINECRAFT, key); } }