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

org.modeshape.jcr.sequencer.SequencerPathExpression Maven / Gradle / Ivy

/*
 * ModeShape (http://www.modeshape.org)
 * See the COPYRIGHT.txt file distributed with this work for information
 * regarding copyright ownership.  Some portions may be licensed
 * to Red Hat, Inc. under one or more contributor license agreements.
 * See the AUTHORS.txt file in the distribution for a full listing of 
 * individual contributors. 
 *
 * ModeShape is free software. Unless otherwise indicated, all code in ModeShape
 * is licensed to you under the terms of the GNU Lesser General Public License as
 * published by the Free Software Foundation; either version 2.1 of
 * the License, or (at your option) any later version.
 *
 * ModeShape is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with this software; if not, write to the Free
 * Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
 * 02110-1301 USA, or see the FSF site: http://www.fsf.org.
 */
package org.modeshape.jcr.sequencer;

import java.io.Serializable;
import java.util.HashMap;
import java.util.Map;
import java.util.regex.Pattern;
import org.modeshape.common.annotation.Immutable;
import org.modeshape.common.util.CheckArg;
import org.modeshape.common.util.HashCode;
import org.modeshape.common.util.ObjectUtil;
import org.modeshape.jcr.RepositoryI18n;
import org.modeshape.jcr.sequencer.PathExpression.WorkspacePath;

/**
 * An expression that defines a selection of some change in the repository that signals a sequencing operation should be run, and
 * the location where the sequencing output should be placed. Sequencer path expressions are used to determine whether information
 * in the repository needs to be sequenced.
 * 

* A simple example is the following: * *

 *     /a/b/c@title => /d/e/f
 * 
* * which means that a sequencer (that uses this expression in its configuration) should be run any time there is a new or modified * title property on the /a/b/c node, and that the output of the sequencing should be placed at * /d/e/f. *

*/ @Immutable public class SequencerPathExpression implements Serializable { /** */ private static final long serialVersionUID = 229464314137494765L; /** * The pattern used to break the initial input string into the two major parts, the selection and output expressions. Group 1 * contains the selection expression, and group 2 contains the output expression. */ private static final Pattern TWO_PART_PATTERN = Pattern.compile("((?:[^=]|=(?!>))+)(?:=>(.+))?"); protected static final String DEFAULT_OUTPUT_EXPRESSION = "."; private static final String PARENT_PATTERN_STRING = "[^/]+/\\.\\./"; // [^/]+/\.\./ private static final Pattern PARENT_PATTERN = Pattern.compile(PARENT_PATTERN_STRING); private static final String REPLACEMENT_VARIABLE_PATTERN_STRING = "(?" + this.outputExpression; } /** * Determine if this path expression applies to content within the supplied workspace name. * * @param workspaceName the name of the workspace; may not be null * @return true if this path expression matches the workspace name, or false otherwise */ public final boolean appliesToWorkspace( String workspaceName ) { return selectExpression.matchesWorkspace(workspaceName); } /** * Obtain a Matcher that can be used to convert the supplied workspace key and absolute path into an output workspace name and * and output path. * * @param absolutePath the path in the workspace; may not be null * @return the matcher; never null */ public Matcher matcher( String absolutePath ) { PathExpression.Matcher inputMatcher = selectExpression.matcher(absolutePath); String outputPath = null; WorkspacePath wsPath = null; if (inputMatcher.matches()) { // Grab the named groups ... Map replacements = new HashMap(); for (int i = 0, count = inputMatcher.groupCount(); i <= count; ++i) { replacements.put(i, inputMatcher.group(i)); } // Grab the selected path ... String selectedPath = inputMatcher.getSelectedNodePath(); // Find the output path using the groups from the match pattern ... wsPath = PathExpression.parsePathInWorkspace(this.outputExpression); if (wsPath != null) { if (wsPath.workspaceName == null) wsPath = wsPath.withWorkspaceName(inputMatcher.getSelectedWorkspaceName()); outputPath = wsPath.path; if (!DEFAULT_OUTPUT_EXPRESSION.equals(outputPath)) { java.util.regex.Matcher replacementMatcher = REPLACEMENT_VARIABLE_PATTERN.matcher(outputPath); StringBuffer sb = new StringBuffer(); if (replacementMatcher.find()) { do { String variable = replacementMatcher.group(1); String replacement = replacements.get(Integer.valueOf(variable)); if (replacement == null) replacement = replacementMatcher.group(0); replacementMatcher.appendReplacement(sb, replacement); } while (replacementMatcher.find()); replacementMatcher.appendTail(sb); outputPath = sb.toString(); } // Make sure there is a trailing '/' ... if (!outputPath.endsWith("/")) outputPath = outputPath + "/"; // Replace all references to "/./" with "/" ... outputPath = outputPath.replaceAll("/\\./", "/"); // Remove any path segment followed by a parent reference ... java.util.regex.Matcher parentMatcher = PARENT_PATTERN.matcher(outputPath); while (parentMatcher.find()) { outputPath = parentMatcher.replaceAll(""); // Make sure there is a trailing '/' ... if (!outputPath.endsWith("/")) outputPath = outputPath + "/"; parentMatcher = PARENT_PATTERN.matcher(outputPath); } // Remove all multiple occurrences of '/' ... outputPath = outputPath.replaceAll("/{2,}", "/"); // Remove the trailing '/@property' ... outputPath = outputPath.replaceAll("/@[^/\\[\\]]+$", ""); // Remove a trailing '/' ... outputPath = outputPath.replaceAll("/$", ""); // If the output path is blank, then use the default output expression ... if (outputPath.length() == 0) outputPath = DEFAULT_OUTPUT_EXPRESSION; } if (DEFAULT_OUTPUT_EXPRESSION.equals(outputPath)) { // The output path is the default expression, so use the selected path ... outputPath = selectedPath; } wsPath = wsPath.withPath(outputPath); } } return new Matcher(inputMatcher, wsPath); } @Immutable public static class Matcher { private final PathExpression.Matcher inputMatcher; private final WorkspacePath outputPath; private final int hc; protected Matcher( PathExpression.Matcher inputMatcher, WorkspacePath outputPath ) { this.inputMatcher = inputMatcher; this.outputPath = outputPath; this.hc = HashCode.compute(super.hashCode(), this.outputPath); } public boolean matches() { return inputMatcher.matches() && this.outputPath != null; } /** * @return inputPath */ public String getInputPath() { return inputMatcher.getInputPath(); } /** * Get the {@link #getInputPath() input path} in standard JCR form. * * @return the JCR form of the input path; null only if {@link #getInputPath()} is null */ public String getJcrInputPath() { // Conver the input path (which has a '@' to denote a property) to a standard JCR path ... String inputPath = getInputPath(); if (inputPath == null) return null; int index = inputPath.lastIndexOf("/@"); if (index != -1) { inputPath = inputPath.substring(0, index) + '/' + inputPath.substring(index + 2); } return inputPath; } /** * @return selectPattern */ public String getSelectedPath() { return inputMatcher.getSelectedNodePath(); } /** * Get the path in the repository where the sequenced content should be placed. * * @return outputPath the output path, or null if this matcher does not match the input */ public String getOutputPath() { return this.outputPath != null ? this.outputPath.path : null; } /** * Get the name of the workspace where the sequenced content should be placed. * * @return outputPath the output path, or null if this matcher does not match the input */ public String getOutputWorkspaceName() { return this.outputPath != null ? this.outputPath.workspaceName : null; } /** * {@inheritDoc} */ @Override public int hashCode() { return this.hc; } /** * {@inheritDoc} */ @Override public boolean equals( Object obj ) { if (obj == this) return true; if (obj instanceof SequencerPathExpression.Matcher) { SequencerPathExpression.Matcher that = (SequencerPathExpression.Matcher)obj; if (!super.equals(that)) return false; return ObjectUtil.isEqualWithNulls(this.outputPath, that.outputPath); } return false; } /** * {@inheritDoc} */ @Override public String toString() { return inputMatcher + " => " + this.outputPath; } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy