blob: e535de5e1987efb4a2fc732a59b8c17f1aa1590f [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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.apache.chemistry.opencmis.inmemory;
import java.util.ArrayList;
import java.util.Collection;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.UUID;
import org.apache.chemistry.opencmis.commons.data.Ace;
import org.apache.chemistry.opencmis.commons.data.Acl;
import org.apache.chemistry.opencmis.commons.definitions.DocumentTypeDefinition;
import org.apache.chemistry.opencmis.commons.definitions.FolderTypeDefinition;
import org.apache.chemistry.opencmis.commons.definitions.ItemTypeDefinition;
import org.apache.chemistry.opencmis.commons.definitions.PolicyTypeDefinition;
import org.apache.chemistry.opencmis.commons.definitions.PropertyDefinition;
import org.apache.chemistry.opencmis.commons.definitions.RelationshipTypeDefinition;
import org.apache.chemistry.opencmis.commons.definitions.SecondaryTypeDefinition;
import org.apache.chemistry.opencmis.commons.definitions.TypeDefinition;
import org.apache.chemistry.opencmis.commons.definitions.TypeDefinitionContainer;
import org.apache.chemistry.opencmis.commons.enums.Cardinality;
import org.apache.chemistry.opencmis.commons.enums.ContentStreamAllowed;
import org.apache.chemistry.opencmis.commons.enums.Updatability;
import org.apache.chemistry.opencmis.commons.exceptions.CmisConstraintException;
import org.apache.chemistry.opencmis.commons.exceptions.CmisInvalidArgumentException;
import org.apache.chemistry.opencmis.commons.impl.dataobjects.AbstractPropertyDefinition;
import org.apache.chemistry.opencmis.commons.impl.dataobjects.AbstractTypeDefinition;
import org.apache.chemistry.opencmis.commons.impl.dataobjects.AccessControlEntryImpl;
import org.apache.chemistry.opencmis.commons.impl.dataobjects.AccessControlListImpl;
import org.apache.chemistry.opencmis.commons.impl.dataobjects.AccessControlPrincipalDataImpl;
import org.apache.chemistry.opencmis.commons.impl.dataobjects.DocumentTypeDefinitionImpl;
import org.apache.chemistry.opencmis.commons.impl.dataobjects.FolderTypeDefinitionImpl;
import org.apache.chemistry.opencmis.commons.impl.dataobjects.ItemTypeDefinitionImpl;
import org.apache.chemistry.opencmis.commons.impl.dataobjects.PolicyTypeDefinitionImpl;
import org.apache.chemistry.opencmis.commons.impl.dataobjects.RelationshipTypeDefinitionImpl;
import org.apache.chemistry.opencmis.commons.impl.dataobjects.SecondaryTypeDefinitionImpl;
import org.apache.chemistry.opencmis.commons.impl.dataobjects.TypeMutabilityImpl;
import org.apache.chemistry.opencmis.inmemory.types.TypeUtil;
import org.apache.chemistry.opencmis.server.support.TypeManager;
/**
* A helper class doing some consistency checks when new type definitions are
* added to the system.
*/
public final class TypeValidator {
private static final Object CMIS_USER = "cmis:user";
private TypeValidator() {
}
public static void checkType(TypeManager tm, TypeDefinition td) {
if (null == td) {
throw new CmisInvalidArgumentException("Cannot add type, because the type defintion is null.");
}
if (null == tm.getTypeById(td.getParentTypeId())) {
throw new CmisInvalidArgumentException("Cannot add type, because parent with id " + td.getParentTypeId()
+ " does not exist.");
}
if (null != tm.getTypeById(td.getId())) {
throw new CmisInvalidArgumentException("Cannot add type, because type with id " + td.getId()
+ " already exists.");
}
checkTypeId(tm, td.getId());
checkTypeQueryName(tm, td.getQueryName());
checkTypeLocalName(tm, td.getLocalName());
checkBaseAndParentType(td);
if (null != td.getPropertyDefinitions()) {
TypeValidator.checkProperties(tm, td.getPropertyDefinitions().values());
}
}
public static AbstractTypeDefinition completeType(TypeDefinition type) {
if (type instanceof DocumentTypeDefinition) {
return completeTypeDoc((DocumentTypeDefinition) type);
} else if (type instanceof FolderTypeDefinition) {
return completeTypeFolder((FolderTypeDefinition) type);
} else if (type instanceof PolicyTypeDefinition) {
return completeTypePolicy((PolicyTypeDefinition) type);
} else if (type instanceof ItemTypeDefinition) {
return completeTypeItem((ItemTypeDefinition) type);
} else if (type instanceof RelationshipTypeDefinition) {
return completeTypeRelationship((RelationshipTypeDefinition) type);
} else if (type instanceof SecondaryTypeDefinition) {
return completeTypeSecondary((SecondaryTypeDefinition) type);
} else {
return null;
}
}
public static void adjustTypeNamesAndId(AbstractTypeDefinition typeDef) {
if (null == typeDef.getId()) {
typeDef.setId(UUID.randomUUID().toString());
} else {
if (!NameValidator.isValidId(typeDef.getId())) {
// if there are illegal characters adjust them
String newId = replaceInvalidCharacters(typeDef.getId());
typeDef.setId(newId);
}
}
if (!NameValidator.isValidQueryName(typeDef.getQueryName())) {
typeDef.setQueryName(typeDef.getId());
}
if (!NameValidator.isValidLocalName(typeDef.getLocalName())) {
typeDef.setLocalName(typeDef.getId());
}
}
private static void completeAbstractTypeDefinition(AbstractTypeDefinition td) {
if (td.isControllableAcl() == null) {
td.setIsControllableAcl(true);
}
if (td.isControllablePolicy() == null) {
td.setIsControllablePolicy(false);
}
if (td.isCreatable() == null) {
td.setIsCreatable(true);
}
if (td.isFileable() == null) {
td.setIsFileable(true);
}
td.setIsFulltextIndexed(false);
td.setIsIncludedInSupertypeQuery(false);
if (td.isQueryable() == null) {
td.setIsQueryable(true);
}
td.setParentTypeId(td.getParentTypeId());
TypeMutabilityImpl tm = new TypeMutabilityImpl();
tm.setCanCreate(true);
tm.setCanDelete(true);
tm.setCanUpdate(true);
td.setTypeMutability(tm);
td.setExtensions(td.getExtensions());
Map<String, PropertyDefinition<?>> propDefsNew = new LinkedHashMap<String, PropertyDefinition<?>>();
if (null != td.getPropertyDefinitions()) {
Map<String, PropertyDefinition<?>> propDefs = td.getPropertyDefinitions();
for (PropertyDefinition<?> pd : propDefs.values()) {
AbstractPropertyDefinition<?> pdNew = completePropertyDef(pd);
adjustPropertyNamesAndId(pdNew);
propDefsNew.put(pdNew.getId(), pd);
}
}
td.setPropertyDefinitions(propDefsNew);
}
private static void checkProperties(TypeManager tm, Collection<PropertyDefinition<?>> pds) {
Collection<TypeDefinitionContainer> tdl = tm.getTypeDefinitionList();
for (PropertyDefinition<?> pd2 : pds) {
// check id syntax
if (null == pd2.getId()) {
throw new CmisInvalidArgumentException("property id cannot be null.");
}
if (!NameValidator.isValidId(pd2.getId())) {
throw new CmisInvalidArgumentException(NameValidator.ERROR_ILLEGAL_NAME);
}
// check query name syntax
if (null == pd2.getQueryName()) {
throw new CmisInvalidArgumentException("property query name cannot be null.");
}
if (!NameValidator.isValidQueryName(pd2.getQueryName())) {
throw new CmisInvalidArgumentException(NameValidator.ERROR_ILLEGAL_NAME);
}
// check local name syntax
if (null == pd2.getLocalName()) {
throw new CmisInvalidArgumentException("property local name cannot be null.");
}
if (!NameValidator.isValidLocalName(pd2.getLocalName())) {
throw new CmisInvalidArgumentException(NameValidator.ERROR_ILLEGAL_NAME);
}
for (TypeDefinitionContainer tdc : tdl) {
TypeDefinition td = tdc.getTypeDefinition();
if (null != td.getPropertyDefinitions()) {
for (PropertyDefinition<?> pd1 : td.getPropertyDefinitions().values()) {
// check if id is used
if (pd1.getId().equals(pd2.getId())) {
throw new CmisConstraintException("Property id " + pd2.getId() + " already in use in type "
+ td.getId());
}
// check if query name is used
if (pd1.getQueryName().equals(pd2.getQueryName())) {
throw new CmisConstraintException("Property query name " + pd2.getQueryName()
+ " already in use in type " + td.getQueryName());
}
// check if local name is used
if (pd1.getLocalName().equals(pd2.getLocalName())) {
throw new CmisConstraintException("Property local name " + pd2.getLocalName()
+ " already in use in type " + td.getId());
}
}
}
}
}
}
private static void adjustPropertyNamesAndId(AbstractPropertyDefinition<?> propDef) {
if (null == propDef.getId()) {
propDef.setId(UUID.randomUUID().toString());
} else {
if (!NameValidator.isValidId(propDef.getId())) {
String newId = replaceInvalidCharacters(propDef.getId());
propDef.setId(newId);
}
}
if (!NameValidator.isValidQueryName(propDef.getQueryName())) {
propDef.setQueryName(propDef.getId());
}
if (!NameValidator.isValidLocalName(propDef.getLocalName())) {
propDef.setLocalName(propDef.getId());
}
}
private static String replaceInvalidCharacters(String id) {
// if there are illegal characters adjust them
StringBuilder sb = new StringBuilder();
for (int i = 0; i < id.length(); i++) {
if (NameValidator.isValidId(id.substring(i, i + 1))) {
sb.append(id.charAt(i));
} else {
sb.append('_');
}
}
return sb.toString();
}
private static DocumentTypeDefinitionImpl completeTypeDoc(DocumentTypeDefinition type) {
DocumentTypeDefinitionImpl td = TypeUtil.cloneTypeDoc(type);
completeAbstractTypeDefinition(td);
td.setIsVersionable(type.isVersionable());
td.setContentStreamAllowed(type.getContentStreamAllowed());
if (td.isVersionable() == null) {
td.setIsVersionable(false);
}
if (td.getContentStreamAllowed() == null) {
td.setContentStreamAllowed(ContentStreamAllowed.ALLOWED);
}
return td;
}
private static FolderTypeDefinitionImpl completeTypeFolder(FolderTypeDefinition type) {
FolderTypeDefinitionImpl td = TypeUtil.cloneTypeFolder(type);
completeAbstractTypeDefinition(td);
return td;
}
private static RelationshipTypeDefinitionImpl completeTypeRelationship(RelationshipTypeDefinition type) {
RelationshipTypeDefinitionImpl td = TypeUtil.cloneTypeRelationship(type);
completeAbstractTypeDefinition(td);
td.setAllowedSourceTypes(type.getAllowedSourceTypeIds());
td.setAllowedTargetTypes(type.getAllowedTargetTypeIds());
return td;
}
private static ItemTypeDefinitionImpl completeTypeItem(ItemTypeDefinition type) {
ItemTypeDefinitionImpl td = TypeUtil.cloneTypeItem(type);
td.initialize(type);
completeAbstractTypeDefinition(td);
return td;
}
private static SecondaryTypeDefinitionImpl completeTypeSecondary(SecondaryTypeDefinition type) {
SecondaryTypeDefinitionImpl td = TypeUtil.cloneTypeSecondary(type);
completeAbstractTypeDefinition(td);
return td;
}
private static PolicyTypeDefinitionImpl completeTypePolicy(PolicyTypeDefinition type) {
PolicyTypeDefinitionImpl td = TypeUtil.cloneTypePolicy(type);
completeAbstractTypeDefinition(td);
return null;
}
// When creating types PropertyDefinitions may only be partially filled,
// fill all fields
// to make a complete definition
private static AbstractPropertyDefinition<?> completePropertyDef(PropertyDefinition<?> pdSrc) {
AbstractPropertyDefinition<?> newPropDef = TypeUtil.clonePropertyDefinition(pdSrc);
if (null == newPropDef.getPropertyType()) {
throw new CmisInvalidArgumentException("Property " + pdSrc.getId() + "has no property type.");
}
if (null == newPropDef.getId()) {
newPropDef.setId(UUID.randomUUID().toString());
}
if (null == newPropDef.getQueryName()) {
newPropDef.setQueryName(pdSrc.getId());
}
if (null == newPropDef.getLocalName()) {
newPropDef.setLocalName(pdSrc.getId());
}
if (null == newPropDef.getCardinality()) {
newPropDef.setCardinality(Cardinality.SINGLE);
}
if (null == newPropDef.isOrderable()) {
newPropDef.setIsOrderable(true);
}
if (null == newPropDef.isQueryable()) {
newPropDef.setIsQueryable(true);
}
if (null == newPropDef.isRequired()) {
newPropDef.setIsRequired(false);
}
if (null == newPropDef.getUpdatability()) {
newPropDef.setUpdatability(Updatability.READWRITE);
}
return newPropDef;
}
public static Acl expandAclMakros(String user, Acl acl) {
boolean mustCopy = false;
if (user == null || acl == null || acl.getAces() == null) {
return acl;
}
for (Ace ace : acl.getAces()) {
String principal = ace.getPrincipalId();
if (principal != null && principal.equals(CMIS_USER)) {
mustCopy = true;
}
}
if (mustCopy) {
AccessControlListImpl result = new AccessControlListImpl();
List<Ace> list = new ArrayList<Ace>(acl.getAces().size());
for (Ace ace : acl.getAces()) {
String principal = ace.getPrincipalId();
if (principal != null && principal.equals(CMIS_USER)) {
AccessControlEntryImpl ace2 = new AccessControlEntryImpl();
ace2.setPermissions(ace.getPermissions());
ace2.setExtensions(ace.getExtensions());
ace2.setPrincipal(new AccessControlPrincipalDataImpl(user));
list.add(ace2);
} else {
list.add(ace);
}
}
result.setAces(list);
return result;
} else {
return acl;
}
}
private static void checkTypeId(TypeManager tm, String typeId) {
if (null == typeId) {
throw new CmisInvalidArgumentException("Type id cannot be null.");
}
// check name syntax
if (!NameValidator.isValidId(typeId)) {
throw new CmisInvalidArgumentException(NameValidator.ERROR_ILLEGAL_ID);
}
if (null != tm.getTypeById(typeId)) {
throw new CmisInvalidArgumentException("You cannot add type with id " + typeId
+ " because it already exists.");
}
}
private static void checkTypeQueryName(TypeManager tm, String queryName) {
if (null == queryName) {
throw new CmisInvalidArgumentException("Query name cannot be null.");
}
if (!NameValidator.isValidQueryName(queryName)) {
throw new CmisInvalidArgumentException(NameValidator.ERROR_ILLEGAL_NAME);
}
// check set query name is unique in the type system
if (null != tm.getTypeByQueryName(queryName)) {
throw new CmisInvalidArgumentException("You cannot add type with query name " + queryName
+ " because it already exists.");
}
}
private static void checkTypeLocalName(TypeManager tm, String localName) {
if (null == localName) {
throw new CmisInvalidArgumentException("Local name cannot be null.");
}
if (!NameValidator.isValidLocalName(localName)) {
throw new CmisInvalidArgumentException(NameValidator.ERROR_ILLEGAL_NAME);
}
for (TypeDefinitionContainer tdc : tm.getTypeDefinitionList()) {
if (tdc.getTypeDefinition().getLocalName().equals(localName)) {
throw new CmisConstraintException("You cannot add type with local name " + localName
+ " because it already exists.");
}
}
}
private static void checkBaseAndParentType(TypeDefinition td) {
if (null == td.getBaseTypeId()) {
throw new CmisInvalidArgumentException("You cannot create a type without a base type id: " + td.getId());
}
if (null == td.getParentTypeId()) {
throw new CmisInvalidArgumentException("You cannot create a type without a parent type id: " + td.getId());
}
}
}