org.modeshape.connector.filesystem.LegacySidecarExtraPropertyStore 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.connector.filesystem;
import java.io.File;
import java.io.FileWriter;
import java.io.IOException;
import java.io.Writer;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import javax.jcr.NamespaceRegistry;
import javax.jcr.RepositoryException;
import org.modeshape.common.text.QuoteEncoder;
import org.modeshape.common.text.TextDecoder;
import org.modeshape.common.text.TextEncoder;
import org.modeshape.common.text.XmlNameEncoder;
import org.modeshape.common.util.IoUtil;
import org.modeshape.common.util.StringUtil;
import org.modeshape.jcr.JcrLexicon;
import org.modeshape.jcr.api.Binary;
import org.modeshape.jcr.cache.DocumentStoreException;
import org.modeshape.jcr.federation.spi.ExtraPropertiesStore;
import org.modeshape.jcr.value.Name;
import org.modeshape.jcr.value.NameFactory;
import org.modeshape.jcr.value.Property;
import org.modeshape.jcr.value.PropertyFactory;
import org.modeshape.jcr.value.PropertyType;
import org.modeshape.jcr.value.ValueFactories;
import org.modeshape.jcr.value.ValueFactory;
import org.modeshape.jcr.value.ValueFormatException;
/**
* An {@link ExtraPropertiesStore} implementation that stores extra properties in legacy sidecar files adjacent to the actual file
* or directory corresponding to the external node. The format of these legacy files is compatible with thosed used by the
* ModeShape 2.x file system connector.
*/
class LegacySidecarExtraPropertyStore implements ExtraPropertiesStore {
/**
* The regex pattern string used to parse properties. The capture groups are as follows:
*
* - property name (encoded)
* - property type string
* - a '[' if the value is multi-valued
* - the single value, or comma-separated values
*
*
* The expression is: ([\S]+)\s*[(](\w+)[)]\s*([\[]?)?([^\]]+)[\]]?
*
*/
protected static final String PROPERTY_PATTERN_STRING = "([\\S]+)\\s*[(](\\w+)[)]\\s*([\\[]?)?([^\\]]+)[\\]]?";
protected static final Pattern PROPERTY_PATTERN = Pattern.compile(PROPERTY_PATTERN_STRING);
/**
* The regex pattern string used to parse quoted string property values. This is a repeating expression, and group(0) captures
* the characters within the quotes (including escaped double quotes).
*
* The expression is: \"((((?<=\\)\")|[^"])*)\"
*
*/
protected static final String STRING_VALUE_PATTERN_STRING = "\\\"((((?<=\\\\)\\\")|[^\"])*)\\\"";
protected static final Pattern STRING_VALUE_PATTERN = Pattern.compile(STRING_VALUE_PATTERN_STRING);
/**
* The regex pattern string used to parse non-string property values (including hexadecimal-encoded binary values). This is a
* repeating expression, and group(1) captures the individual values.
*
* The expression is: ([^\s,]+)\s*[,]*\s*
*
*/
protected static final String VALUE_PATTERN_STRING = "([^\\s,]+)\\s*[,]*\\s*";
protected static final Pattern VALUE_PATTERN = Pattern.compile(VALUE_PATTERN_STRING);
public static final String DEFAULT_EXTENSION = ".modeshape";
public static final String DEFAULT_RESOURCE_EXTENSION = ".content.modeshape";
private final FileSystemConnector connector;
private final NamespaceRegistry registry;
private final PropertyFactory propertyFactory;
private final ValueFactories factories;
private final ValueFactory stringFactory;
private final TextEncoder encoder = new XmlNameEncoder();
private final TextDecoder decoder = new XmlNameEncoder();
private final QuoteEncoder quoter = new QuoteEncoder();
protected LegacySidecarExtraPropertyStore( FileSystemConnector connector ) {
this.connector = connector;
this.registry = this.connector.registry();
this.propertyFactory = this.connector.getContext().getPropertyFactory();
this.factories = this.connector.getContext().getValueFactories();
this.stringFactory = factories.getStringFactory();
}
protected String getExclusionPattern() {
return "(.+)\\.(content\\.)?modeshape$";
}
@Override
public Map getProperties( String id ) {
return load(id, sidecarFile(id));
}
@Override
public void storeProperties( String id,
Map properties ) {
write(id, sidecarFile(id), properties);
}
@Override
public void updateProperties( String id,
Map properties ) {
File sidecar = sidecarFile(id);
Map existing = load(id, sidecar);
if (existing == null || existing.isEmpty()) {
write(id, sidecar, properties);
} else {
// Merge the changed properties into the existing ...
for (Map.Entry entry : properties.entrySet()) {
Name name = entry.getKey();
Property property = entry.getValue();
if (property == null) {
existing.remove(name);
} else {
existing.put(name, property);
}
}
// Now write out all of the updated properties ...
write(id, sidecar, existing);
}
}
@Override
public boolean removeProperties( String id ) {
File file = sidecarFile(id);
if (!file.exists()) return false;
file.delete();
return true;
}
protected File sidecarFile( String id ) {
File actualFile = connector.fileFor(id);
String extension = DEFAULT_EXTENSION;
if (connector.isContentNode(id)) {
extension = DEFAULT_RESOURCE_EXTENSION;
}
return new File(actualFile.getAbsolutePath() + extension);
}
protected Map load( String id,
File propertiesFile ) {
if (!propertiesFile.exists() || !propertiesFile.canRead()) return NO_PROPERTIES;
try {
String content = IoUtil.read(propertiesFile);
Map result = new HashMap();
for (String line : StringUtil.splitLines(content)) {
// Parse each line ...
Property property = parse(line, result);
if (property != null) {
result.put(property.getName(), property);
}
}
return result;
} catch (Throwable e) {
throw new DocumentStoreException(id, e);
}
}
protected void write( String id,
File propertiesFile,
Map properties ) {
if (properties.isEmpty()) {
if (propertiesFile.exists()) {
// Delete the file ...
propertiesFile.delete();
}
return;
}
try {
Writer fileWriter = null;
try {
// Write the primary type first ...
Property primaryType = properties.get(JcrLexicon.PRIMARY_TYPE);
if (primaryType != null) {
fileWriter = new FileWriter(propertiesFile);
write(primaryType, fileWriter);
}
// Then write the mixin types ...
Property mixinTypes = properties.get(JcrLexicon.MIXIN_TYPES);
if (mixinTypes != null) {
if (fileWriter == null) fileWriter = new FileWriter(propertiesFile);
write(mixinTypes, fileWriter);
}
// Then write the UUID ...
Property uuid = properties.get(JcrLexicon.UUID);
if (uuid != null) {
if (fileWriter == null) fileWriter = new FileWriter(propertiesFile);
write(uuid, fileWriter);
}
// Then all the others ...
for (Property property : properties.values()) {
if (property == null) continue;
if (property == primaryType || property == mixinTypes || property == uuid) continue;
if (fileWriter == null) fileWriter = new FileWriter(propertiesFile);
write(property, fileWriter);
}
} finally {
if (fileWriter != null) {
fileWriter.close();
} else {
// Nothing was written, so remove the sidecar file ...
propertiesFile.delete();
}
}
} catch (Throwable e) {
throw new DocumentStoreException(id, e);
}
}
protected void write( Property property,
Writer stream ) throws IOException, RepositoryException {
String name = stringFactory.create(property.getName());
stream.append(encoder.encode(name));
if (property.isEmpty()) {
stream.append('\n');
stream.flush();
return;
}
stream.append(" (");
PropertyType type = PropertyType.discoverType(property.getFirstValue());
stream.append(type.getName().toLowerCase());
stream.append(") ");
if (property.isMultiple()) {
stream.append('[');
}
boolean first = true;
boolean quote = type == PropertyType.STRING;
for (Object value : property) {
if (first) first = false;
else stream.append(", ");
String str = null;
if (value instanceof Binary) {
byte[] bytes = IoUtil.readBytes(((Binary)value).getStream());
str = StringUtil.getHexString(bytes);
} else {
str = stringFactory.create(value);
}
if (quote) {
stream.append('"');
stream.append(quoter.encode(str));
stream.append('"');
} else {
stream.append(str);
}
}
if (property.isMultiple()) {
stream.append(']');
}
stream.append('\n');
stream.flush();
}
protected Property parse( String line,
Map result ) throws RepositoryException {
if (line.length() == 0) return null; // blank line
char firstChar = line.charAt(0);
if (firstChar == '#') return null; // comment line
if (firstChar == ' ') return null; // ignore line
Matcher matcher = PROPERTY_PATTERN.matcher(line);
NameFactory nameFactory = factories.getNameFactory();
if (!matcher.matches()) {
// It should be an empty multi-valued property, and the line consists only of the name ...
Name name = nameFactory.create(decoder.decode(line));
return propertyFactory.create(name);
}
String nameString = decoder.decode(matcher.group(1));
String typeString = matcher.group(2);
String valuesString = matcher.group(4);
Name name = null;
try {
name = factories.getNameFactory().create(nameString);
} catch (ValueFormatException e) {
// See MODE-1281. Earlier versions would write out an empty property without the trailing line feed,
// so we need to consider this case now. About the only thing we can do is look for two namespace-prefixed names
// ...
if (nameString.indexOf(':') < nameString.lastIndexOf(':')) {
// This is likely two names smashed together. Use the namespace prefixes to look for where we can break this
// ...
Set prefixes = new LinkedHashSet();
prefixes.add(JcrLexicon.Namespace.PREFIX);
for (String prefix : registry.getPrefixes()) {
prefixes.add(prefix);
}
for (String prefix : prefixes) {
int index = nameString.lastIndexOf(prefix + ":");
if (index <= 0) continue;
// Otherwise, we found a match. Parse the first property name, and create an empty property ...
name = nameFactory.create(nameString.substring(0, index));
result.put(name, propertyFactory.create(name));
// Now parse the name of the next property and continue ...
name = nameFactory.create(nameString.substring(index));
break;
}
} else {
throw e;
}
}
PropertyType type = PropertyType.valueFor(typeString);
Pattern pattern = VALUE_PATTERN;
ValueFactory> valueFactory = factories.getValueFactory(type);
boolean binary = false;
boolean decode = false;
if (type == PropertyType.STRING) {
// Parse the double-quoted value(s) ...
pattern = STRING_VALUE_PATTERN;
decode = true;
} else if (type == PropertyType.BINARY) {
binary = true;
}
Matcher valuesMatcher = pattern.matcher(valuesString);
List
© 2015 - 2025 Weber Informatics LLC | Privacy Policy