mirror of
https://github.com/Alfresco/alfresco-community-repo.git
synced 2025-06-30 18:15:39 +00:00
18023: RM: groundwork for custom metadata delete 18071: MT - fix ETHREEOH-3730 (reject invitation shows "Invitation not found" instead of Yes / No options) 18078: MT - fix ETHREEOH-3892 (it is not possible to create or manage any Tenants once the DOD5015 RM AMP is installed) 18903: RM Caveats - fix ALF-1894 (improvement for ESC - remove hardcoded "rmc" model/namespace) 19046: Improve dictionary debug logging (ALF-587) 19096: Merging PATCHES/V3.2.r to V3.2 19075: Merging DEV/BELARUS/V3.2-2010_02_24 to PATCHES/V3.2.r 18881: ALF-587: MT Upgrades to 3.2r fail unable to find Alfresco content types 19085: ALF-587 - test/build fix (follow on for r18881->r19075) 19145: Dynamic Models - follow-on for ALF-587 (& ALFCOM-2977) + additional unit tests 19176: Remove Java 6'ism 19198: Build/test fix (follow-on to r19145 - do not validate model delete of version nodes) git-svn-id: https://svn.alfresco.com/repos/alfresco-enterprise/alfresco/HEAD/root@19260 c4b6b30b-aa2e-2d43-bbcb-ca4b014f7261
485 lines
17 KiB
Java
485 lines
17 KiB
Java
/*
|
|
* Copyright (C) 2005-2010 Alfresco Software Limited.
|
|
*
|
|
* This file is part of Alfresco
|
|
*
|
|
* Alfresco is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU Lesser General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* Alfresco 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 Alfresco. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
package org.alfresco.repo.dictionary;
|
|
|
|
import java.util.ArrayList;
|
|
import java.util.Collection;
|
|
import java.util.List;
|
|
|
|
import org.alfresco.repo.dictionary.constraint.ListOfValuesConstraint;
|
|
import org.alfresco.repo.dictionary.constraint.NumericRangeConstraint;
|
|
import org.alfresco.repo.dictionary.constraint.RegexConstraint;
|
|
import org.alfresco.repo.dictionary.constraint.RegisteredConstraint;
|
|
import org.alfresco.repo.dictionary.constraint.StringLengthConstraint;
|
|
import org.alfresco.service.cmr.dictionary.Constraint;
|
|
import org.alfresco.service.cmr.dictionary.ConstraintDefinition;
|
|
import org.alfresco.service.cmr.dictionary.DictionaryException;
|
|
import org.alfresco.service.cmr.dictionary.ModelDefinition;
|
|
import org.alfresco.service.namespace.NamespacePrefixResolver;
|
|
import org.alfresco.service.namespace.QName;
|
|
import org.alfresco.util.EqualsHelper;
|
|
import org.springframework.beans.BeanWrapper;
|
|
import org.springframework.beans.BeanWrapperImpl;
|
|
import org.springframework.beans.InvalidPropertyException;
|
|
import org.springframework.beans.PropertyAccessException;
|
|
|
|
/**
|
|
* Compiled Property Constraint
|
|
*
|
|
* @author Derek Hulley. janv
|
|
*/
|
|
/* package */class M2ConstraintDefinition implements ConstraintDefinition
|
|
{
|
|
private static final String PROP_SHORT_NAME = "shortName";
|
|
private static final String PROP_TITLE = "title";
|
|
private static final String PROP_DESCRIPTION = "description";
|
|
|
|
public static final String ERR_CYCLIC_REF = "d_dictionary.constraint.err.cyclic_ref";
|
|
public static final String ERR_TYPE_AND_REF = "d_dictionary.constraint.err.type_and_ref";
|
|
public static final String ERR_TYPE_OR_REF = "d_dictionary.constraint.err.type_or_ref";
|
|
public static final String ERR_REF_NOT_FOUND = "d_dictionary.constraint.err.ref_not_found";
|
|
public static final String ERR_ANON_NEEDS_PROPERTY = "d_dictionary.constraint.err.anon_needs_property";
|
|
public static final String ERR_INVALID_TYPE = "d_dictionary.constraint.err.invalid_type";
|
|
public static final String ERR_SIMPLE_AND_LIST = "d_dictionary.constraint.err.property_simple_and_list";
|
|
public static final String ERR_CONSTRUCT_FAILURE = "d_dictionary.constraint.err.construct_failure";
|
|
public static final String ERR_PROPERTY_MISMATCH = "d_dictionary.constraint.err.property_mismatch";
|
|
public static final String ERR_RESERVED_PROPERTY = "d_dictionary.constraint.err.reserved_property";
|
|
|
|
private static int anonPropCount = 0;
|
|
|
|
private ModelDefinition model;
|
|
private NamespacePrefixResolver prefixResolver;
|
|
private M2Constraint m2Constraint;
|
|
private QName name;
|
|
private Constraint constraint;
|
|
private boolean resolving;
|
|
|
|
/* package */M2ConstraintDefinition(M2PropertyDefinition m2PropertyDef, M2Constraint m2Constraint,
|
|
NamespacePrefixResolver prefixResolver)
|
|
{
|
|
this(m2PropertyDef.getModel(), m2PropertyDef, m2Constraint, prefixResolver);
|
|
}
|
|
|
|
/* package */M2ConstraintDefinition(ModelDefinition modelDefinition, M2PropertyDefinition m2PropertyDef,
|
|
M2Constraint m2Constraint, NamespacePrefixResolver prefixResolver)
|
|
{
|
|
this.model = modelDefinition;
|
|
this.m2Constraint = m2Constraint;
|
|
this.prefixResolver = prefixResolver;
|
|
|
|
String constraintName = m2Constraint.getName();
|
|
if (constraintName == null)
|
|
{
|
|
// the constraint is anonymous, so it has to be defined within the context of a property
|
|
if (m2PropertyDef == null)
|
|
{
|
|
throw new DictionaryException(ERR_ANON_NEEDS_PROPERTY);
|
|
}
|
|
// pick the name up from the property and some anonymous value
|
|
String localName = m2PropertyDef.getName().getLocalName() + "_anon_" + (++anonPropCount);
|
|
this.name = QName.createQName(m2PropertyDef.getName().getNamespaceURI(), localName);
|
|
m2Constraint.setName(this.name.getPrefixedQName(prefixResolver).toPrefixString());
|
|
}
|
|
else
|
|
{
|
|
this.name = QName.createQName(m2Constraint.getName(), prefixResolver);
|
|
}
|
|
}
|
|
|
|
/* package */synchronized void resolveDependencies(ModelQuery query)
|
|
{
|
|
if (resolving)
|
|
{
|
|
throw new DictionaryException(ERR_CYCLIC_REF, name.toPrefixString());
|
|
}
|
|
// prevent circular references
|
|
try
|
|
{
|
|
resolving = true;
|
|
resolveInternal(query);
|
|
}
|
|
finally
|
|
{
|
|
resolving = false;
|
|
}
|
|
}
|
|
|
|
private synchronized void resolveInternal(ModelQuery query)
|
|
{
|
|
if (constraint != null)
|
|
{
|
|
// already been resolved
|
|
return;
|
|
}
|
|
|
|
String shortName = name.toPrefixString();
|
|
String ref = m2Constraint.getRef();
|
|
String type = m2Constraint.getType();
|
|
|
|
String title = m2Constraint.getTitle();
|
|
String description = m2Constraint.getDescription();
|
|
|
|
if (ref != null && type != null)
|
|
{
|
|
throw new DictionaryException(ERR_TYPE_AND_REF, shortName);
|
|
}
|
|
else if (ref == null && type == null)
|
|
{
|
|
throw new DictionaryException(ERR_TYPE_OR_REF, shortName);
|
|
}
|
|
else if (ref != null)
|
|
{
|
|
// resolve the reference name
|
|
QName qnameRef = QName.createQName(ref, prefixResolver);
|
|
// ensure that the reference exists in the model
|
|
M2ConstraintDefinition constraintDef = (M2ConstraintDefinition) query.getConstraint(qnameRef);
|
|
if (constraintDef == null)
|
|
{
|
|
throw new DictionaryException(ERR_REF_NOT_FOUND, ref, shortName);
|
|
}
|
|
// make sure that the constraint definition has itself been resolved
|
|
constraintDef.resolveDependencies(query);
|
|
// just use the constraint provided by the referenced definition
|
|
this.constraint = constraintDef.getConstraint();
|
|
|
|
//use real constraint name instead of anonymous name
|
|
//TODO Fix backed out - breaks DictionaryDAOTest
|
|
//this.name = constraintDef.getName();
|
|
|
|
|
|
if (m2Constraint.getTitle() == null)
|
|
{
|
|
m2Constraint.setTitle(constraintDef.getTitle());
|
|
}
|
|
|
|
if (m2Constraint.getDescription() == null)
|
|
{
|
|
m2Constraint.setDescription(constraintDef.getDescription());
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// we have to build the constraint from the type
|
|
try
|
|
{
|
|
ConstraintType constraintType = ConstraintType.valueOf(type);
|
|
constraint = constraintType.newInstance();
|
|
}
|
|
catch (IllegalArgumentException e)
|
|
{
|
|
// try to establish it as a class
|
|
try
|
|
{
|
|
@SuppressWarnings("unchecked")
|
|
Class clazz = Class.forName(type);
|
|
constraint = (Constraint) clazz.newInstance();
|
|
}
|
|
catch (ClassNotFoundException ee)
|
|
{
|
|
throw new DictionaryException(ERR_INVALID_TYPE, type, shortName);
|
|
}
|
|
catch (ClassCastException ee)
|
|
{
|
|
throw new DictionaryException(ERR_INVALID_TYPE, type, shortName);
|
|
}
|
|
catch (Exception ee)
|
|
{
|
|
throw new DictionaryException(ERR_CONSTRUCT_FAILURE, type, shortName);
|
|
}
|
|
}
|
|
|
|
// property setters
|
|
BeanWrapper beanWrapper = new BeanWrapperImpl(constraint);
|
|
List<M2NamedValue> constraintNamedValues = m2Constraint.getParameters();
|
|
|
|
if (constraintNamedValues != null)
|
|
{
|
|
for (M2NamedValue namedValue : constraintNamedValues)
|
|
{
|
|
String namedValueName = namedValue.getName();
|
|
// Check for reserved properties
|
|
if (namedValueName.equals(PROP_SHORT_NAME))
|
|
{
|
|
throw new DictionaryException(ERR_RESERVED_PROPERTY, PROP_SHORT_NAME, namedValueName);
|
|
}
|
|
|
|
Object value = null;
|
|
if (namedValue.getSimpleValue() != null && namedValue.getListValue() != null)
|
|
{
|
|
throw new DictionaryException(ERR_SIMPLE_AND_LIST, shortName, namedValue.getName());
|
|
}
|
|
else if (namedValue.getSimpleValue() != null)
|
|
{
|
|
value = namedValue.getSimpleValue();
|
|
}
|
|
else if (namedValue.getListValue() != null)
|
|
{
|
|
value = namedValue.getListValue();
|
|
}
|
|
try
|
|
{
|
|
beanWrapper.setPropertyValue(namedValueName, value);
|
|
}
|
|
catch (PropertyAccessException e)
|
|
{
|
|
throw new DictionaryException(ERR_PROPERTY_MISMATCH, e, namedValueName, shortName);
|
|
}
|
|
catch (InvalidPropertyException e)
|
|
{
|
|
throw new DictionaryException(ERR_PROPERTY_MISMATCH, e, namedValueName, shortName);
|
|
}
|
|
}
|
|
|
|
// Pass in the short name as a special property, if it is available
|
|
if (beanWrapper.isWritableProperty(PROP_SHORT_NAME))
|
|
{
|
|
try
|
|
{
|
|
beanWrapper.setPropertyValue(PROP_SHORT_NAME, shortName);
|
|
}
|
|
catch (PropertyAccessException e)
|
|
{
|
|
throw new DictionaryException(ERR_PROPERTY_MISMATCH, e, shortName, shortName);
|
|
}
|
|
catch (InvalidPropertyException e)
|
|
{
|
|
throw new DictionaryException(ERR_PROPERTY_MISMATCH, e, shortName, shortName);
|
|
}
|
|
}
|
|
|
|
if ((title != null) && (beanWrapper.isWritableProperty(PROP_TITLE)))
|
|
{
|
|
beanWrapper.setPropertyValue(PROP_TITLE, title);
|
|
}
|
|
|
|
if ((title != null) && (beanWrapper.isWritableProperty(PROP_DESCRIPTION)))
|
|
{
|
|
beanWrapper.setPropertyValue(PROP_DESCRIPTION, description);
|
|
}
|
|
}
|
|
|
|
// now initialize
|
|
constraint.initialize();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @see #getName()
|
|
*/
|
|
@Override
|
|
public String toString()
|
|
{
|
|
return getName().toString();
|
|
}
|
|
|
|
public ModelDefinition getModel()
|
|
{
|
|
return model;
|
|
}
|
|
|
|
public QName getName()
|
|
{
|
|
return name;
|
|
}
|
|
|
|
public String getTitle()
|
|
{
|
|
String value = M2Label.getLabel(model, "constraint", name, "title");
|
|
if (value == null)
|
|
{
|
|
value = m2Constraint.getTitle();
|
|
}
|
|
return value;
|
|
}
|
|
|
|
public String getDescription()
|
|
{
|
|
String value = M2Label.getLabel(model, "constraint", name, "description");
|
|
if (value == null)
|
|
{
|
|
value = m2Constraint.getDescription();
|
|
}
|
|
return value;
|
|
}
|
|
|
|
public Constraint getConstraint()
|
|
{
|
|
return constraint;
|
|
}
|
|
|
|
public QName getRef()
|
|
{
|
|
QName refQName = null;
|
|
String ref = m2Constraint.getRef();
|
|
if (ref != null)
|
|
{
|
|
refQName = QName.createQName(ref, prefixResolver);
|
|
}
|
|
return refQName;
|
|
}
|
|
|
|
/**
|
|
* Well-known constraint types
|
|
*/
|
|
public static enum ConstraintType
|
|
{
|
|
REGISTERED
|
|
{
|
|
@Override
|
|
protected Constraint newInstance()
|
|
{
|
|
return new RegisteredConstraint();
|
|
}
|
|
},
|
|
REGEX
|
|
{
|
|
@Override
|
|
protected Constraint newInstance()
|
|
{
|
|
return new RegexConstraint();
|
|
}
|
|
},
|
|
MINMAX
|
|
{
|
|
@Override
|
|
protected Constraint newInstance()
|
|
{
|
|
return new NumericRangeConstraint();
|
|
}
|
|
},
|
|
LENGTH
|
|
{
|
|
@Override
|
|
protected Constraint newInstance()
|
|
{
|
|
return new StringLengthConstraint();
|
|
}
|
|
},
|
|
LIST
|
|
{
|
|
@Override
|
|
protected Constraint newInstance()
|
|
{
|
|
return new ListOfValuesConstraint();
|
|
}
|
|
};
|
|
|
|
/**
|
|
* @return Returns the constraint implementation
|
|
*/
|
|
protected abstract Constraint newInstance();
|
|
}
|
|
|
|
/* package */ M2ModelDiff diffConstraint(ConstraintDefinition conDef)
|
|
{
|
|
M2ModelDiff modelDiff = null;
|
|
boolean isUpdated = false;
|
|
boolean isUpdatedIncrementally = false;
|
|
|
|
if (this == conDef)
|
|
{
|
|
modelDiff = new M2ModelDiff(name, M2ModelDiff.TYPE_CONSTRAINT, M2ModelDiff.DIFF_UNCHANGED);
|
|
return modelDiff;
|
|
}
|
|
|
|
// check name - cannot be null
|
|
if (! name.equals(conDef.getName()))
|
|
{
|
|
isUpdated = true;
|
|
}
|
|
|
|
// check title
|
|
if (! EqualsHelper.nullSafeEquals(getTitle(), conDef.getTitle(), false))
|
|
{
|
|
isUpdatedIncrementally = true;
|
|
}
|
|
|
|
// check description
|
|
if (! EqualsHelper.nullSafeEquals(getDescription(), conDef.getDescription(), false))
|
|
{
|
|
isUpdatedIncrementally = true;
|
|
}
|
|
|
|
// check type string
|
|
if (! EqualsHelper.nullSafeEquals(getConstraint().getType(), conDef.getConstraint().getType()))
|
|
{
|
|
isUpdated = true;
|
|
}
|
|
|
|
if (isUpdated)
|
|
{
|
|
modelDiff = new M2ModelDiff(name, M2ModelDiff.TYPE_CONSTRAINT, M2ModelDiff.DIFF_UPDATED);
|
|
}
|
|
else if (isUpdatedIncrementally)
|
|
{
|
|
modelDiff = new M2ModelDiff(name, M2ModelDiff.TYPE_CONSTRAINT, M2ModelDiff.DIFF_UPDATED_INC);
|
|
}
|
|
else
|
|
{
|
|
modelDiff = new M2ModelDiff(name, M2ModelDiff.TYPE_CONSTRAINT, M2ModelDiff.DIFF_UNCHANGED);
|
|
}
|
|
|
|
return modelDiff;
|
|
}
|
|
|
|
/*package*/ static Collection<M2ModelDiff> diffConstraintLists(Collection<ConstraintDefinition> previousConstraints, Collection<ConstraintDefinition> newConstraints)
|
|
{
|
|
List<M2ModelDiff> modelDiffs = new ArrayList<M2ModelDiff>();
|
|
|
|
for (ConstraintDefinition previousConstraint : previousConstraints)
|
|
{
|
|
boolean found = false;
|
|
for (ConstraintDefinition newConstraint : newConstraints)
|
|
{
|
|
if (newConstraint.getName().equals(previousConstraint.getName()))
|
|
{
|
|
modelDiffs.add(((M2ConstraintDefinition)previousConstraint).diffConstraint(previousConstraint));
|
|
found = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (! found)
|
|
{
|
|
modelDiffs.add(new M2ModelDiff(previousConstraint.getName(), M2ModelDiff.TYPE_CONSTRAINT, M2ModelDiff.DIFF_DELETED));
|
|
}
|
|
}
|
|
|
|
for (ConstraintDefinition newConstraint : newConstraints)
|
|
{
|
|
boolean found = false;
|
|
for (ConstraintDefinition previousConstraint : previousConstraints)
|
|
{
|
|
if (newConstraint.getName().equals(previousConstraint.getName()))
|
|
{
|
|
found = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (! found)
|
|
{
|
|
modelDiffs.add(new M2ModelDiff(newConstraint.getName(), M2ModelDiff.TYPE_CONSTRAINT, M2ModelDiff.DIFF_CREATED));
|
|
}
|
|
}
|
|
|
|
return modelDiffs;
|
|
}
|
|
}
|