001 /** 002 * Copyright 2005-2013 The Kuali Foundation 003 * 004 * Licensed under the Educational Community License, Version 2.0 (the "License"); 005 * you may not use this file except in compliance with the License. 006 * You may obtain a copy of the License at 007 * 008 * http://www.opensource.org/licenses/ecl2.php 009 * 010 * Unless required by applicable law or agreed to in writing, software 011 * distributed under the License is distributed on an "AS IS" BASIS, 012 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 013 * See the License for the specific language governing permissions and 014 * limitations under the License. 015 */ 016 package org.kuali.rice.kew.xml; 017 018 import org.jdom.Document; 019 import org.jdom.Element; 020 import org.jdom.JDOMException; 021 import org.kuali.rice.core.api.util.xml.XmlException; 022 import org.kuali.rice.core.api.util.xml.XmlHelper; 023 import org.kuali.rice.kew.api.KewApiConstants; 024 import org.kuali.rice.kew.util.Utilities; 025 import org.kuali.rice.kim.api.KimConstants; 026 import org.kuali.rice.kim.api.group.Group; 027 import org.kuali.rice.kim.api.group.GroupService; 028 import org.kuali.rice.kim.api.identity.IdentityService; 029 import org.kuali.rice.kim.api.identity.principal.Principal; 030 import org.kuali.rice.kim.api.services.KimApiServiceLocator; 031 import org.kuali.rice.kim.api.type.KimType; 032 import org.kuali.rice.kim.api.type.KimTypeAttribute; 033 import org.xml.sax.SAXException; 034 035 import javax.xml.parsers.ParserConfigurationException; 036 import java.io.IOException; 037 import java.io.InputStream; 038 import java.util.ArrayList; 039 import java.util.HashMap; 040 import java.util.List; 041 import java.util.Map; 042 043 import static org.kuali.rice.core.api.impex.xml.XmlConstants.*; 044 045 046 /** 047 * Parses groups from XML. 048 * 049 * @see Group 050 * 051 * @author Kuali Rice Team (rice.collab@kuali.org) 052 * 053 */ 054 public class GroupXmlParser { 055 private static final org.apache.log4j.Logger LOG = org.apache.log4j.Logger.getLogger(GroupXmlParser.class); 056 private static final boolean DEFAULT_ACTIVE_VALUE = true; 057 private static final String DEFAULT_GROUP_DESCRIPTION = ""; 058 private HashMap<String, List<String>> memberGroupIds = new HashMap<String, List<String>>(); 059 private HashMap<String, List<String>> memberGroupNames = new HashMap<String, List<String>>(); 060 private HashMap<String, List<String>> memberPrincipalIds = new HashMap<String, List<String>>(); 061 private Map<String, String> groupAttributes = new HashMap<String, String>(); 062 063 public List<Group> parseGroups(InputStream input) throws IOException, XmlException { 064 try { 065 Document doc = XmlHelper.trimSAXXml(input); 066 Element root = doc.getRootElement(); 067 return parseGroups(root); 068 } catch (JDOMException e) { 069 throw new XmlException("Parse error.", e); 070 } catch (SAXException e){ 071 throw new XmlException("Parse error.",e); 072 } catch(ParserConfigurationException e){ 073 throw new XmlException("Parse error.",e); 074 } 075 } 076 077 078 /** 079 * Parses and saves groups 080 * @param element top-level 'data' element which should contain a <groups> child element 081 * @return a list of parsed and saved, current, groups; 082 * @throws XmlException 083 */ 084 @SuppressWarnings("unchecked") 085 public List<Group> parseGroups(Element element) throws XmlException { 086 List<Group> groups = new ArrayList<Group>(); 087 for (Element groupsElement: (List<Element>) element.getChildren(GROUPS, GROUP_NAMESPACE)) { 088 089 for (Element groupElement: (List<Element>) groupsElement.getChildren(GROUP, GROUP_NAMESPACE)) { 090 groups.add(parseGroup(groupElement)); 091 } 092 } 093 for (Group group : groups) { 094 GroupService groupService = KimApiServiceLocator.getGroupService(); 095 // check if group already exists 096 Group foundGroup = groupService.getGroupByNamespaceCodeAndName(group.getNamespaceCode(), group.getName()); 097 098 if (foundGroup == null) { 099 if ( LOG.isInfoEnabled() ) { 100 LOG.info("Group named '" + group.getName() + "' not found, creating new group named '" + group.getName() + "'"); 101 } 102 try { 103 Group newGroup = groupService.createGroup(group); 104 105 String key = newGroup.getNamespaceCode().trim() + KewApiConstants.KIM_GROUP_NAMESPACE_NAME_DELIMITER_CHARACTER + newGroup.getName().trim(); 106 addGroupMembers(newGroup, key); 107 } catch (Exception e) { 108 throw new RuntimeException("Error creating group with name '" + group.getName() + "'", e); 109 } 110 } else { 111 if ( LOG.isInfoEnabled() ) { 112 LOG.info("Group named '" + group.getName() + "' found, creating a new version"); 113 } 114 try { 115 Group.Builder builder = Group.Builder.create(foundGroup); 116 builder.setActive(group.isActive()); 117 builder.setDescription(group.getDescription()); 118 builder.setKimTypeId(group.getKimTypeId()); 119 120 //builder.setVersionNumber(foundGroup.getVersionNumber()); 121 group = builder.build(); 122 groupService.updateGroup(foundGroup.getId(), group); 123 124 //delete existing group members and replace with new 125 groupService.removeAllMembers(foundGroup.getId()); 126 127 String key = group.getNamespaceCode().trim() + KewApiConstants.KIM_GROUP_NAMESPACE_NAME_DELIMITER_CHARACTER + group.getName().trim(); 128 addGroupMembers(group, key); 129 130 } catch (Exception e) { 131 throw new RuntimeException("Error updating group.", e); 132 } 133 } 134 } 135 return groups; 136 } 137 138 @SuppressWarnings("unchecked") 139 private Group parseGroup(Element element) throws XmlException { 140 141 142 // Type element and children (namespace and name) 143 144 String typeId = null; 145 KimType kimTypeInfo; 146 List<KimTypeAttribute> kimTypeAttributes = new ArrayList<KimTypeAttribute>(); 147 if (element.getChild(TYPE, GROUP_NAMESPACE) != null) { 148 Element typeElement = element.getChild(TYPE, GROUP_NAMESPACE); 149 String typeNamespace = typeElement.getChildText(NAMESPACE, GROUP_NAMESPACE); 150 String typeName = typeElement.getChildText(NAME, GROUP_NAMESPACE); 151 kimTypeInfo = KimApiServiceLocator.getKimTypeInfoService().findKimTypeByNameAndNamespace(typeNamespace, typeName); 152 if (kimTypeInfo != null) { 153 typeId = kimTypeInfo.getId(); 154 kimTypeAttributes = kimTypeInfo.getAttributeDefinitions(); 155 } else { 156 throw new XmlException("Invalid type name and namespace specified."); 157 } 158 } else { //set to default type 159 kimTypeInfo = KimApiServiceLocator.getKimTypeInfoService().findKimTypeByNameAndNamespace(KimConstants.KIM_TYPE_DEFAULT_NAMESPACE, KimConstants.KIM_TYPE_DEFAULT_NAME); 160 if (kimTypeInfo != null) { 161 typeId = kimTypeInfo.getId(); 162 kimTypeAttributes = kimTypeInfo.getAttributeDefinitions(); 163 } else { 164 throw new RuntimeException("Failed to locate the 'Default' group type! Please ensure that it's in your database."); 165 } 166 } 167 //groupInfo.setKimTypeId(typeId); 168 169 String groupNamespace = element.getChildText(NAMESPACE, GROUP_NAMESPACE); 170 if (groupNamespace == null) { 171 throw new XmlException("Namespace must have a value."); 172 } 173 174 String groupName = element.getChildText(NAME, GROUP_NAMESPACE); 175 if (groupName == null) { 176 throw new XmlException("Name must have a value."); 177 } 178 179 Group.Builder groupInfo = Group.Builder.create(groupNamespace, groupName, typeId); 180 IdentityService identityService = KimApiServiceLocator.getIdentityService(); 181 //groupInfo.setGroupName(element.getChildText(NAME, GROUP_NAMESPACE)); 182 183 String id = element.getChildText(ID, GROUP_NAMESPACE); 184 if (id != null) { 185 groupInfo.setId(id.trim()); 186 } else { 187 188 } 189 190 String description = element.getChildText(DESCRIPTION, GROUP_NAMESPACE); 191 if (description != null && !description.trim().equals("")) { 192 groupInfo.setDescription(description); 193 } 194 195 //Active Indicator 196 groupInfo.setActive(DEFAULT_ACTIVE_VALUE); 197 if (element.getChildText(ACTIVE, GROUP_NAMESPACE) != null) { 198 String active = element.getChildText(ACTIVE, GROUP_NAMESPACE).trim(); 199 if (active.toUpperCase().equals("N") || active.toUpperCase().equals("FALSE")) { 200 groupInfo.setActive(false); 201 } 202 } 203 204 //Get list of attribute keys 205 List<String> validAttributeKeys = new ArrayList<String>(); 206 for (KimTypeAttribute attribute : kimTypeAttributes) { 207 validAttributeKeys.add(attribute.getKimAttribute().getAttributeName()); 208 } 209 //Group attributes 210 if (element.getChild(ATTRIBUTES, GROUP_NAMESPACE) != null) { 211 List<Element> attributes = element.getChild(ATTRIBUTES, GROUP_NAMESPACE).getChildren(); 212 213 Map<String, String> attrMap = new HashMap<String, String>(); 214 for (Element attr : attributes ) { 215 attrMap.put(attr.getAttributeValue(KEY), attr.getAttributeValue(VALUE)); 216 if (!validAttributeKeys.contains(attr.getAttributeValue(KEY))) { 217 throw new XmlException("Invalid attribute specified."); 218 } 219 } 220 Map<String, String> groupAttributes = attrMap; 221 if (!groupAttributes.isEmpty()) { 222 groupInfo.setAttributes(groupAttributes); 223 } 224 } 225 226 //Group members 227 228 List<Element> members = null; 229 if(element.getChild(MEMBERS, GROUP_NAMESPACE) == null) { 230 members = new ArrayList<Element>(); 231 } 232 else { 233 members = element.getChild(MEMBERS, GROUP_NAMESPACE).getChildren(); 234 } 235 for (Element member : members) { 236 String elementName = member.getName().trim(); 237 if (elementName.equals(PRINCIPAL_NAME)) { 238 String principalName = member.getText().trim(); 239 Principal principal = identityService.getPrincipalByPrincipalName(principalName); 240 if (principal != null) { 241 addPrincipalToGroup(groupInfo.getNamespaceCode(), groupInfo.getName(), principal.getPrincipalId()); 242 } else { 243 throw new XmlException("Principal Name "+principalName+" cannot be found."); 244 } 245 } else if (elementName.equals(PRINCIPAL_ID)) { 246 String xmlPrincipalId = member.getText().trim(); 247 Principal principal = identityService.getPrincipal(xmlPrincipalId); 248 if (principal != null) { 249 addPrincipalToGroup(groupInfo.getNamespaceCode(), groupInfo.getName(), principal.getPrincipalId()); 250 } else { 251 throw new XmlException("Principal Id "+xmlPrincipalId+" cannot be found."); 252 } 253 // Groups are handled differently since the member group may not be saved yet. Therefore they need to be validated after the groups are saved. 254 } else if (elementName.equals(GROUP_ID)) { 255 String xmlGroupId = member.getText().trim(); 256 addGroupToGroup(groupInfo.getNamespaceCode(), groupInfo.getName(), xmlGroupId); 257 } else if (elementName.equals(GROUP_NAME)) { 258 String xmlGroupName = member.getChildText(NAME, GROUP_NAMESPACE).trim(); 259 String xmlGroupNamespace = member.getChildText(NAMESPACE, GROUP_NAMESPACE).trim(); 260 addGroupNameToGroup(groupInfo.getNamespaceCode(), groupInfo.getName(), xmlGroupNamespace, xmlGroupName); 261 } else { 262 LOG.error("Unknown member element: " + elementName); 263 } 264 265 266 } 267 268 return groupInfo.build(); 269 270 } 271 272 private void addPrincipalToGroup(String groupNamespace, String groupName, String principalId) { 273 String key = groupNamespace.trim() + KewApiConstants.KIM_GROUP_NAMESPACE_NAME_DELIMITER_CHARACTER + groupName.trim(); 274 List<String> principalIds = memberPrincipalIds.get(key); 275 if (principalIds == null) { 276 principalIds = new ArrayList<String>(); 277 } 278 principalIds.add(principalId); 279 memberPrincipalIds.put(key, principalIds); 280 } 281 282 private void addGroupToGroup(String groupNamespace, String groupName, String groupId) { 283 String key = groupNamespace.trim() + KewApiConstants.KIM_GROUP_NAMESPACE_NAME_DELIMITER_CHARACTER + groupName.trim(); 284 List<String> groupIds = memberGroupIds.get(key); 285 if (groupIds == null) { 286 groupIds = new ArrayList<String>(); 287 } 288 groupIds.add(groupId); 289 memberGroupIds.put(key, groupIds); 290 } 291 292 private void addGroupNameToGroup(String groupNamespace, String groupName, String memberGroupNamespace, String memberGroupName) { 293 String key = groupNamespace.trim() + KewApiConstants.KIM_GROUP_NAMESPACE_NAME_DELIMITER_CHARACTER + groupName.trim(); 294 List<String> groupNames = memberGroupNames.get(key); 295 if (groupNames == null) { 296 groupNames = new ArrayList<String>(); 297 } 298 groupNames.add(memberGroupNamespace.trim() + KewApiConstants.KIM_GROUP_NAMESPACE_NAME_DELIMITER_CHARACTER + memberGroupName.trim()); 299 memberGroupNames.put(key, groupNames); 300 } 301 302 private void addGroupMembers(Group groupInfo, String key) throws XmlException { 303 GroupService groupService = KimApiServiceLocator.getGroupService(); 304 List<String> groupIds = memberGroupIds.get(key); 305 if (groupIds != null) { 306 for (String groupId : groupIds) { 307 Group group = groupService.getGroup(groupId); 308 if (group != null) { 309 groupService.addGroupToGroup(group.getId(), groupInfo.getId()); 310 } else { 311 throw new XmlException("Group Id "+groupId+" cannot be found."); 312 } 313 } 314 } 315 List<String> groupNames = memberGroupNames.get(key); 316 if (groupNames != null) { 317 for (String groupName : groupNames) { 318 Group group = groupService.getGroupByNamespaceCodeAndName(Utilities.parseGroupNamespaceCode(groupName), 319 Utilities.parseGroupName(groupName)); 320 if (group != null) { 321 groupService.addGroupToGroup(group.getId(), groupInfo.getId()); 322 } else { 323 throw new XmlException("Group "+groupName+" cannot be found."); 324 } 325 } 326 } 327 List<String> principalIds = memberPrincipalIds.get(key); 328 if (principalIds != null) { 329 for (String principalId : principalIds) { 330 groupService.addPrincipalToGroup(principalId, groupInfo.getId()); 331 } 332 } 333 334 } 335 }