All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.modeshape.jcr.cache.NodeKey Maven / Gradle / Ivy

There is a newer version: 5.4.1.Final
Show newest version
/*
 * ModeShape (http://www.modeshape.org)
 *
 * 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.modeshape.jcr.cache;

import java.io.Serializable;
import java.util.Comparator;
import java.util.UUID;
import org.infinispan.schematic.SchematicDb;
import org.modeshape.common.annotation.Immutable;
import org.modeshape.common.util.ObjectUtil;
import org.modeshape.common.util.SecureHash;
import org.modeshape.common.util.StringUtil;
import org.modeshape.jcr.JcrRepository;

/**
 * An immutable unique key for a node within the {@link JcrRepository repository}'s {@link SchematicDb database}.
 * 

* A node key consists of three parts: *

    *
  1. A multi-character key uniquely identifying the repository's storage source in which this node appears;
  2. *
  3. A multi-character key uniquely identifying the workspace in which the node appears; and
  4. *
  5. A multi-character key representing the JCR identifier of a node, which is usually a UUID.
  6. *
*

*

* This class contains methods for */ @Immutable public final class NodeKey implements Serializable, Comparable { public static final Comparator COMPARATOR = new Comparator() { @Override public int compare( NodeKey key1, NodeKey key2 ) { return ObjectUtil.compareWithNulls(key1, key2); } }; private static final int UUID_LENGTH = UUID.randomUUID().toString().length(); private static final long serialVersionUID = 1L; protected static final int SOURCE_LENGTH = 7; protected static final int WORKSPACE_LENGTH = 7; private static final int SOURCE_START_INDEX = 0; private static final int SOURCE_END_INDEX = SOURCE_START_INDEX + SOURCE_LENGTH; private static final int WORKSPACE_START_INDEX = SOURCE_LENGTH; private static final int WORKSPACE_END_INDEX = WORKSPACE_START_INDEX + WORKSPACE_LENGTH; private static final int IDENTIFIER_START_INDEX = WORKSPACE_END_INDEX; /** * Determine if the supplied string may be a valid identifier. This method returns 'false' only if the identifier is known to * be invalid (e.g., it is not of the correct format). This method may return true even if the identifier itself does not * reference an existing node. * * @param key * @return true if the string is of the correct format for a node key, or false if it not the correct format */ public static boolean isValidFormat( String key ) { if (StringUtil.isBlank(key) || key.length() <= IDENTIFIER_START_INDEX) return false; return true; } /** * Determine if the supplied string is known to be a valid {@link #withRandomId() random} node key identifier. * * @param identifier the identifier * @return true if the string is of the correct format for a node key, or false if it not the correct format */ public static boolean isValidRandomIdentifier( String identifier ) { if (identifier.length() == UUID_LENGTH) { // It's the right length, but see if it's a UUID ... try { UUID.fromString(identifier); return true; } catch (IllegalArgumentException e) { // Nope return false; } } return false; } private final String key; private transient String sourceKey; private transient String workspaceKey; private transient String identifier; /** * Reconstitute a node key from the supplied string. * * @param key the string representation of the key; may not be null */ public NodeKey( String key ) { assert key != null; assert key.length() > IDENTIFIER_START_INDEX; this.key = key; } /** * Reconstitute a node key from the supplied source key, workspace key, and node identifier. * * @param sourceKey the source key; may not be null and must be 4 characters * @param workspaceKey the workspace key; may not be null and must be 4 characters * @param identifier the node identifier; may not be null and must be at least 1 character */ public NodeKey( String sourceKey, String workspaceKey, String identifier ) { assert sourceKey != null; assert workspaceKey != null; assert identifier != null; assert sourceKey.length() == SOURCE_LENGTH; assert workspaceKey.length() == WORKSPACE_LENGTH; assert workspaceKey.length() > 0; this.key = sourceKey + workspaceKey + identifier; } /** * Get the multi-character key uniquely identifying the repository's storage source in which this node appears. * * @return the source key; never null and always contains at least one character */ public String getSourceKey() { if (sourceKey == null) { // Value is idempotent, so it's okay to do this without synchronizing ... sourceKey = key.substring(SOURCE_START_INDEX, SOURCE_END_INDEX); } return sourceKey; } /** * Get the multi-character key uniquely identifying the workspace in which the node appears. * * @return the workspace key; never null and always contains at least one character */ public String getWorkspaceKey() { if (workspaceKey == null) { // Value is idempotent, so it's okay to do this without synchronizing ... workspaceKey = key.substring(WORKSPACE_START_INDEX, WORKSPACE_END_INDEX); } return workspaceKey; } /** * Get the multi-character key representing the JCR identifier of a node, which is usually a UUID. * * @return the JCR identifier for the node; never null and always contains at least one character */ public String getIdentifier() { if (identifier == null) { // Value is idempotent, so it's okay to do this without synchronizing ... identifier = key.substring(IDENTIFIER_START_INDEX); } return identifier; } /** * Get the SHA-1 hash of the {@link #getIdentifier() identifier}. * * @return the hexadecimal representation of the identifier's SHA-1 hash; never null */ public String getIdentifierHash() { return SecureHash.sha1(getIdentifier()); } @Override public int compareTo( NodeKey that ) { if (that == this) return 0; return this.key.compareTo(that.key); } @Override public int hashCode() { return key.hashCode(); } @Override public boolean equals( Object obj ) { if (obj == this) return true; if (obj instanceof NodeKey) { NodeKey that = (NodeKey)obj; return this.key.equals(that.key); } return false; } @Override public String toString() { return key; } public NodeKey withRandomId() { return new NodeKey(getSourceKey(), getWorkspaceKey(), UUID.randomUUID().toString()); } public NodeKey withRandomIdAndWorkspace( String workspaceKey ) { return new NodeKey(getSourceKey(), workspaceKey, UUID.randomUUID().toString()); } public NodeKey withId( String identifier ) { return new NodeKey(getSourceKey(), getWorkspaceKey(), identifier); } public NodeKey withWorkspaceKey( String workspaceKey ) { return new NodeKey(getSourceKey(), workspaceKey, getIdentifier()); } public NodeKey withWorkspaceKeyAndId( String workspaceKey, String identifier ) { return new NodeKey(getSourceKey(), workspaceKey, identifier); } public NodeKey withSourceKeyAndId( String sourceKey, String identifier ) { return new NodeKey(sourceKey, getWorkspaceKey(), identifier); } public static String keyForSourceName( String name ) { return SecureHash.sha1(name).substring(0, NodeKey.SOURCE_LENGTH); } public static String keyForWorkspaceName( String name ) { return SecureHash.sha1(name).substring(0, NodeKey.WORKSPACE_LENGTH); } public static String sourceKey( String key ) { return isValidFormat(key) ? key.substring(SOURCE_START_INDEX, SOURCE_END_INDEX) : null; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy