001/**
002 * Copyright 2005-2015 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 */
016package org.kuali.rice.kns.util;
017
018import org.apache.commons.lang.StringUtils;
019import org.kuali.rice.core.api.CoreApiServiceLocator;
020import org.kuali.rice.core.api.config.property.ConfigurationService;
021import org.kuali.rice.kns.datadictionary.MaintainableCollectionDefinition;
022import org.kuali.rice.kns.datadictionary.MaintainableFieldDefinition;
023import org.kuali.rice.kns.datadictionary.MaintainableItemDefinition;
024import org.kuali.rice.kns.datadictionary.MaintainableSectionDefinition;
025import org.kuali.rice.kns.lookup.LookupUtils;
026import org.kuali.rice.kns.maintenance.Maintainable;
027import org.kuali.rice.kns.service.KNSServiceLocator;
028import org.kuali.rice.kns.service.MaintenanceDocumentDictionaryService;
029import org.kuali.rice.kns.web.ui.Field;
030import org.kuali.rice.kns.web.ui.Row;
031import org.kuali.rice.kns.web.ui.Section;
032import org.kuali.rice.krad.bo.BusinessObject;
033import org.kuali.rice.krad.datadictionary.AttributeSecurity;
034import org.kuali.rice.kns.datadictionary.MaintenanceDocumentEntry;
035import org.kuali.rice.krad.exception.ValidationException;
036import org.kuali.rice.kns.lookup.SelectiveReferenceRefresher;
037import org.kuali.rice.krad.service.DataDictionaryService;
038import org.kuali.rice.krad.service.KRADServiceLocatorWeb;
039import org.kuali.rice.krad.service.KualiExceptionIncidentService;
040import org.kuali.rice.krad.service.MaintenanceDocumentService;
041import org.kuali.rice.krad.util.KRADConstants;
042import org.kuali.rice.krad.workflow.service.WorkflowDocumentService;
043
044import java.util.Collection;
045import java.util.HashMap;
046import java.util.HashSet;
047import java.util.Iterator;
048import java.util.List;
049import java.util.Map;
050import java.util.Set;
051import java.util.StringTokenizer;
052
053/**
054 * @deprecated Use {@link org.kuali.rice.krad.maintenance.MaintenanceUtils}.
055 */
056@Deprecated
057public final class MaintenanceUtils {
058    private static final org.apache.log4j.Logger LOG = org.apache.log4j.Logger.getLogger(MaintenanceUtils.class);
059
060    private static MaintenanceDocumentService maintenanceDocumentService;
061    private static WorkflowDocumentService workflowDocumentService;
062    private static ConfigurationService kualiConfigurationService;
063    private static KualiExceptionIncidentService kualiExceptionIncidentService;
064    private static MaintenanceDocumentDictionaryService maintenanceDocumentDictionaryService;
065    private static DataDictionaryService dataDictionaryService;
066
067    private MaintenanceUtils() {
068        throw new UnsupportedOperationException("do not call");
069    }
070
071    /**
072     * Returns the field templates defined in the maint dictionary xml files. Field templates are used in multiple value lookups.
073     * When doing a MV lookup on a collection, the returned BOs are not necessarily of the same type as the elements of the
074     * collection. Therefore, a means of mapping between the fields for the 2 BOs are necessary. The template attribute of
075     * <maintainableField>s contained within <maintainableCollection>s tells us this mapping. Example: a
076     * <maintainableField name="collectionAttrib" template="lookupBOAttrib"> definition means that when a list of BOs are
077     * returned, the lookupBOAttrib value of the looked up BO will be placed into the collectionAttrib value of the BO added to the
078     * collection
079     *
080     * @param sections       the sections of a document
081     * @param collectionName the name of a collection. May be a nested collection with indices (e.g. collA[1].collB)
082     * @return
083     */
084    public static Map<String, String> generateMultipleValueLookupBOTemplate(List<MaintainableSectionDefinition> sections, String collectionName) {
085        MaintainableCollectionDefinition definition = findMaintainableCollectionDefinition(sections, collectionName);
086        if (definition == null) {
087            return null;
088        }
089        Map<String, String> template = null;
090
091        for (MaintainableFieldDefinition maintainableField : definition.getMaintainableFields()) {
092            String templateString = maintainableField.getTemplate();
093            if (StringUtils.isNotBlank(templateString)) {
094                if (template == null) {
095                    template = new HashMap<String, String>();
096                }
097                template.put(maintainableField.getName(), templateString);
098            }
099        }
100        return template;
101    }
102
103    /**
104     * Finds the MaintainableCollectionDefinition corresponding to the given collection name. For example, if the collection name is
105     * "A.B.C", it will attempt to find the MaintainableCollectionDefinition for C that is nested in B that is nested under A. This
106     * may not work correctly if there are duplicate collection definitions within the sections
107     *
108     * @param sections       the sections of a maint doc
109     * @param collectionName the name of a collection, relative to the root of the BO being maintained. This value may have index
110     *                       values (e.g. [1]), but these are ignored.
111     * @return
112     */
113    public static MaintainableCollectionDefinition findMaintainableCollectionDefinition(List<MaintainableSectionDefinition> sections, String collectionName) {
114        String[] collectionNameParts = StringUtils.split(collectionName, ".");
115        for (MaintainableSectionDefinition section : sections) {
116            MaintainableCollectionDefinition collDefinition = findMaintainableCollectionDefinitionHelper(section.getMaintainableItems(), collectionNameParts, 0);
117            if (collDefinition != null) {
118                return collDefinition;
119            }
120        }
121        return null;
122    }
123
124    private static <E extends MaintainableItemDefinition> MaintainableCollectionDefinition findMaintainableCollectionDefinitionHelper(Collection<E> items, String[] collectionNameParts, int collectionNameIndex) {
125        if (collectionNameParts.length <= collectionNameIndex) {
126            // we've gone too far down the nesting without finding it
127            return null;
128        }
129
130        // we only care about the coll name, and not the index, since the coll definitions do not include the indexing characters,
131        // i.e. [ and ]
132        String collectionToFind = StringUtils.substringBefore(collectionNameParts[collectionNameIndex], "[");
133        for (MaintainableItemDefinition item : items) {
134            if (item instanceof MaintainableCollectionDefinition) {
135                MaintainableCollectionDefinition collection = (MaintainableCollectionDefinition) item;
136                if (collection.getName().equals(collectionToFind)) {
137                    // we found an appropriate coll, now we have to see if we need to recurse even more (more nested collections),
138                    // or just return the one we found.
139                    if (collectionNameIndex == collectionNameParts.length - 1) {
140                        // we're at the last part of the name, so we return
141                        return collection;
142                    } else {
143                        // go deeper
144                        return findMaintainableCollectionDefinitionHelper(collection.getMaintainableCollections(), collectionNameParts, collectionNameIndex + 1);
145                    }
146                }
147            }
148        }
149        return null;
150    }
151
152    /**
153     * Checks to see if there has been an override lookup declared for the maintenance field. If so, the override will be used for
154     * the quickfinder and lookup utils will not be called. If no override was given, LookupUtils.setFieldQuickfinder will be called
155     * to set the system generated quickfinder based on the attributes relationship to the parent business object.
156     *
157     * @return Field with quickfinder set if one was found
158     */
159    public static final Field setFieldQuickfinder(BusinessObject businessObject, String attributeName, MaintainableFieldDefinition maintainableFieldDefinition, Field field, List<String> displayedFieldNames, SelectiveReferenceRefresher srr) {
160        if (maintainableFieldDefinition.getOverrideLookupClass() != null && StringUtils.isNotBlank(maintainableFieldDefinition.getOverrideFieldConversions())) {
161            field.setQuickFinderClassNameImpl(maintainableFieldDefinition.getOverrideLookupClass().getName());
162            field.setFieldConversions(maintainableFieldDefinition.getOverrideFieldConversions());
163            field.setBaseLookupUrl(LookupUtils.getBaseLookupUrl(false));
164            field.setReferencesToRefresh(LookupUtils.convertReferencesToSelectCollectionToString(
165                    srr.getAffectedReferencesFromLookup(businessObject, attributeName, "")));
166            return field;
167        }
168        if (maintainableFieldDefinition.isNoLookup()) {
169            return field;
170        }
171        return LookupUtils.setFieldQuickfinder(businessObject, null, false, 0, attributeName, field, displayedFieldNames, maintainableFieldDefinition.isNoLookup());
172    }
173
174    public static final Field setFieldQuickfinder(BusinessObject businessObject, String collectionName, boolean addLine, int index,
175                                                  String attributeName, Field field, List<String> displayedFieldNames, Maintainable maintainable, MaintainableFieldDefinition maintainableFieldDefinition) {
176        if (maintainableFieldDefinition.getOverrideLookupClass() != null && StringUtils.isNotBlank(maintainableFieldDefinition.getOverrideFieldConversions())) {
177            if (maintainable != null) {
178                String collectionPrefix = "";
179                if (collectionName != null) {
180                    if (addLine) {
181                        collectionPrefix = KRADConstants.MAINTENANCE_ADD_PREFIX + collectionName + ".";
182                    } else {
183                        collectionPrefix = collectionName + "[" + index + "].";
184                    }
185                }
186                field.setQuickFinderClassNameImpl(maintainableFieldDefinition.getOverrideLookupClass().getName());
187
188                String prefixedFieldConversions = prefixFieldConversionsDestinationsWithCollectionPrefix(maintainableFieldDefinition.getOverrideFieldConversions(), collectionPrefix);
189                field.setFieldConversions(prefixedFieldConversions);
190                field.setBaseLookupUrl(LookupUtils.getBaseLookupUrl(false));
191                field.setReferencesToRefresh(LookupUtils.convertReferencesToSelectCollectionToString(
192                        maintainable.getAffectedReferencesFromLookup(businessObject, attributeName, collectionPrefix)));
193            }
194            return field;
195        }
196        if (maintainableFieldDefinition.isNoLookup()) {
197            return field;
198        }
199        return LookupUtils.setFieldQuickfinder(businessObject, collectionName, addLine, index,
200                attributeName, field, displayedFieldNames, maintainable);
201    }
202
203    private static String prefixFieldConversionsDestinationsWithCollectionPrefix(String originalFieldConversions, String collectionPrefix) {
204        StringBuilder buf = new StringBuilder();
205        StringTokenizer tok = new StringTokenizer(originalFieldConversions, KRADConstants.FIELD_CONVERSIONS_SEPARATOR);
206        boolean needsSeparator = false;
207        while (tok.hasMoreTokens()) {
208            String conversionPair = tok.nextToken();
209            if (StringUtils.isBlank(conversionPair)) {
210                continue;
211            }
212
213            String fromValue = StringUtils.substringBefore(conversionPair, KRADConstants.FIELD_CONVERSION_PAIR_SEPARATOR);
214            String toValue = StringUtils.substringAfter(conversionPair, KRADConstants.FIELD_CONVERSION_PAIR_SEPARATOR);
215
216            if (needsSeparator) {
217                buf.append(KRADConstants.FIELD_CONVERSIONS_SEPARATOR);
218            }
219            needsSeparator = true;
220
221            buf.append(fromValue).append(KRADConstants.FIELD_CONVERSION_PAIR_SEPARATOR).append(collectionPrefix).append(toValue);
222        }
223        return buf.toString();
224    }
225
226    public static final void setFieldDirectInquiry(BusinessObject businessObject, String attributeName, MaintainableFieldDefinition maintainableFieldDefinition, Field field, List<String> displayedFieldNames) {
227        LookupUtils.setFieldDirectInquiry(businessObject, attributeName, field);
228    }
229
230    public static final void setFieldDirectInquiry(BusinessObject businessObject, String collectionName, boolean addLine, int index,
231                                                   String attributeName, Field field, List<String> displayedFieldNames, Maintainable maintainable, MaintainableFieldDefinition maintainableFieldDefinition) {
232        LookupUtils.setFieldDirectInquiry(businessObject, attributeName, field);
233    }
234
235    /**
236     * Given a section, returns a comma delimited string of all fields, representing the error keys that exist for a section
237     *
238     * @param section a section
239     * @return
240     */
241    public static String generateErrorKeyForSection(Section section) {
242        Set<String> fieldPropertyNames = new HashSet<String>();
243        addRowsToErrorKeySet(section.getRows(), fieldPropertyNames);
244
245        StringBuilder buf = new StringBuilder();
246        buf.append(section.getSectionId()).append(",");
247
248        Iterator<String> nameIter = fieldPropertyNames.iterator();
249        while (nameIter.hasNext()) {
250            buf.append(nameIter.next());
251            if (nameIter.hasNext()) {
252                buf.append(",");
253            }
254        }
255
256        if (section.getContainedCollectionNames() != null && section.getContainedCollectionNames().size() > 0) {
257            buf.append(",");
258
259            Iterator<String> collectionIter = section.getContainedCollectionNames().iterator();
260            while (collectionIter.hasNext()) {
261                buf.append(KRADConstants.MAINTENANCE_NEW_MAINTAINABLE + collectionIter.next());
262                if (collectionIter.hasNext()) {
263                    buf.append(",");
264                }
265            }
266        }
267
268        return buf.toString();
269    }
270
271    /**
272     * This method recurses through all the fields of the list of rows and adds each field's property name to the set if it starts
273     * with Constants.MAINTENANCE_NEW_MAINTAINABLE
274     *
275     * @param listOfRows
276     * @param errorKeys
277     * @see KRADConstants#MAINTENANCE_NEW_MAINTAINABLE
278     */
279    protected static void addRowsToErrorKeySet(List<Row> listOfRows, Set<String> errorKeys) {
280        if (listOfRows == null) {
281            return;
282        }
283        for (Row row : listOfRows) {
284            List<Field> fields = row.getFields();
285            if (fields == null) {
286                continue;
287            }
288            for (Field field : fields) {
289                String fieldPropertyName = field.getPropertyName();
290                if (fieldPropertyName != null && fieldPropertyName.startsWith(KRADConstants.MAINTENANCE_NEW_MAINTAINABLE)) {
291                    errorKeys.add(field.getPropertyName());
292                }
293                addRowsToErrorKeySet(field.getContainerRows(), errorKeys);
294            }
295        }
296    }
297
298    /**
299     * This method will throw a {@link ValidationException} if there is a valid locking document in existence and throwExceptionIfLocked is true.
300     */
301    public static void checkForLockingDocument(Maintainable maintainable, final boolean throwExceptionIfLocked) {
302        LOG.info("starting checkForLockingDocument (by Maintainable)");
303
304        // get the docHeaderId of the blocking docs, if any are locked and blocking
305        //String blockingDocId = getMaintenanceDocumentService().getLockingDocumentId(maintainable, null);
306        final String blockingDocId = maintainable.getLockingDocumentId();
307
308        org.kuali.rice.krad.maintenance.MaintenanceUtils.checkDocumentBlockingDocumentId(blockingDocId, throwExceptionIfLocked);
309    }
310
311    private static MaintenanceDocumentDictionaryService getMaintenanceDocumentDictionaryService() {
312        if (maintenanceDocumentDictionaryService == null) {
313            maintenanceDocumentDictionaryService = KNSServiceLocator.getMaintenanceDocumentDictionaryService();
314        }
315        return maintenanceDocumentDictionaryService;
316    }
317
318    private static DataDictionaryService getDataDictionaryService() {
319        if (dataDictionaryService == null) {
320            dataDictionaryService = KRADServiceLocatorWeb.getDataDictionaryService();
321        }
322        return dataDictionaryService;
323    }
324
325    public static Map<String, AttributeSecurity> retrievePropertyPathToAttributeSecurityMappings(String docTypeName) {
326        Map<String, AttributeSecurity> results = new HashMap<String, AttributeSecurity>();
327        MaintenanceDocumentEntry entry = getMaintenanceDocumentDictionaryService().getMaintenanceDocumentEntry(docTypeName);
328        String className = entry.getDataObjectClass().getName();
329
330        for (MaintainableSectionDefinition section : entry.getMaintainableSections()) {
331            for (MaintainableItemDefinition item : section.getMaintainableItems()) {
332                if (item instanceof MaintainableFieldDefinition) {
333                    MaintainableFieldDefinition field = (MaintainableFieldDefinition) item;
334                    AttributeSecurity attributeSecurity = getDataDictionaryService().getAttributeSecurity(className, field.getName());
335                    if (attributeSecurity != null) {
336                        results.put(field.getName(), attributeSecurity);
337                    }
338                } else if (item instanceof MaintainableCollectionDefinition) {
339                    addMaintenanceDocumentCollectionPathToSecurityMappings(results, "", (MaintainableCollectionDefinition) item);
340                }
341            }
342        }
343        return results;
344    }
345
346    private static void addMaintenanceDocumentCollectionPathToSecurityMappings(Map<String, AttributeSecurity> mappings, String propertyPathPrefix, MaintainableCollectionDefinition collectionDefinition) {
347        propertyPathPrefix = propertyPathPrefix + collectionDefinition.getName() + ".";
348        String boClassName = collectionDefinition.getBusinessObjectClass().getName();
349        for (MaintainableFieldDefinition field : collectionDefinition.getMaintainableFields()) {
350            AttributeSecurity attributeSecurity = getDataDictionaryService().getAttributeSecurity(boClassName, field.getName());
351            if (attributeSecurity != null) {
352                mappings.put(propertyPathPrefix + field.getName(), attributeSecurity);
353            }
354        }
355        for (MaintainableCollectionDefinition nestedCollection : collectionDefinition.getMaintainableCollections()) {
356            addMaintenanceDocumentCollectionPathToSecurityMappings(mappings, propertyPathPrefix, nestedCollection);
357        }
358    }
359
360    private static MaintenanceDocumentService getMaintenanceDocumentService() {
361        if (maintenanceDocumentService == null) {
362            maintenanceDocumentService = KRADServiceLocatorWeb.getMaintenanceDocumentService();
363        }
364        return maintenanceDocumentService;
365    }
366
367    private static WorkflowDocumentService getWorkflowDocumentService() {
368        if (workflowDocumentService == null) {
369            workflowDocumentService = KRADServiceLocatorWeb.getWorkflowDocumentService();
370        }
371        return workflowDocumentService;
372    }
373
374    private static ConfigurationService getKualiConfigurationService() {
375        if (kualiConfigurationService == null) {
376            kualiConfigurationService = CoreApiServiceLocator.getKualiConfigurationService();
377        }
378        return kualiConfigurationService;
379    }
380
381    private static KualiExceptionIncidentService getKualiExceptionIncidentService() {
382        if (kualiExceptionIncidentService == null) {
383            kualiExceptionIncidentService = KRADServiceLocatorWeb.getKualiExceptionIncidentService();
384        }
385        return kualiExceptionIncidentService;
386    }
387}