001/*
002 * Copyright 2006 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.ole.coa.document.validation.impl;
017
018import java.sql.Date;
019import java.sql.Timestamp;
020import java.util.Calendar;
021import java.util.HashMap;
022import java.util.Iterator;
023import java.util.List;
024import java.util.Map;
025
026import org.apache.commons.lang.StringUtils;
027import org.apache.commons.lang.time.DateUtils;
028import org.kuali.ole.coa.businessobject.Account;
029import org.kuali.ole.coa.businessobject.Organization;
030import org.kuali.ole.coa.service.OrganizationService;
031import org.kuali.ole.sys.OLEConstants;
032import org.kuali.ole.sys.OLEKeyConstants;
033import org.kuali.ole.sys.OLEPropertyConstants;
034import org.kuali.ole.sys.context.SpringContext;
035import org.kuali.ole.sys.identity.OleKimAttributes;
036import org.kuali.rice.core.api.parameter.ParameterEvaluatorService;
037import org.kuali.rice.coreservice.framework.parameter.ParameterService;
038import org.kuali.rice.kim.api.KimConstants;
039import org.kuali.rice.kim.api.identity.Person;
040import org.kuali.rice.kim.api.services.IdentityManagementService;
041import org.kuali.rice.kns.document.MaintenanceDocument;
042import org.kuali.rice.kns.maintenance.rules.MaintenanceDocumentRuleBase;
043import org.kuali.rice.krad.util.GlobalVariables;
044import org.kuali.rice.krad.util.ObjectUtils;
045
046/**
047 * This class implements the business rules specific to the {@link Org} Maintenance Document.
048 */
049public class OrgRule extends MaintenanceDocumentRuleBase {
050
051    protected static org.apache.log4j.Logger LOG = org.apache.log4j.Logger.getLogger(OrgRule.class);
052
053
054    protected static OrganizationService orgService;
055
056    protected Organization oldOrg;
057    protected Organization newOrg;
058    protected boolean isHrmsOrgActivated;
059
060    /**
061     * Constructs a OrgRule and pseudo-injects services
062     */
063    public OrgRule() {
064        super();
065
066        // Pseudo-inject some services.
067        //
068        // This approach is being used to make it simpler to convert the Rule classes
069        // to spring-managed with these services injected by Spring at some later date.
070        // When this happens, just remove these calls to the setters with
071        // SpringContext, and configure the bean defs for spring.
072        if (orgService == null) {
073            orgService = SpringContext.getBean(OrganizationService.class);
074        }
075    }
076
077    /**
078     * This performs the following checks on document approve:
079     * <ul>
080     * <li>{@link OrgRule#checkExistenceAndActive()}</li>
081     * <li>{@link OrgRule#checkOrgClosureRules(MaintenanceDocument)}</li>
082     * <li>{@link OrgRule#checkSimpleRules(MaintenanceDocument)}</li>
083     * <li>{@link OrgRule#checkDefaultAccountNumber(MaintenanceDocument)}</li>
084     * </ul>
085     * This rule fails on rule failure
086     *
087     * @see org.kuali.rice.kns.maintenance.rules.MaintenanceDocumentRuleBase#processCustomApproveDocumentBusinessRules(org.kuali.rice.kns.document.MaintenanceDocument)
088     */
089    @Override
090    protected boolean processCustomApproveDocumentBusinessRules(MaintenanceDocument document) {
091
092        boolean success = true;
093
094        LOG.debug("Entering processCustomApproveDocumentBusinessRules()");
095
096        // determine whether HRMS ORG is activated in this app instance
097        isHrmsOrgActivated = isHrmsOrgActivated();
098
099        // check that all sub-objects whose keys are specified have matching objects in the db
100        success &= checkExistenceAndActive();
101
102        success &= checkOrgClosureRules(document);
103
104        // check that end date is greater than begin date and Reports To Chart/Org should not be same as this Chart/Org
105        success &= checkSimpleRules(document);
106
107        // check that defaultAccount is present unless
108        // ( (orgType = U or C) and ( document is a "create new" ))
109
110        //Code changes for JIRA OLE2344 don't check default account number as account no is made optional
111        //success &= checkDefaultAccountNumber(document);
112
113        return success;
114    }
115
116    /**
117     * This performs the following checks on document route:
118     * <ul>
119     * <li>{@link OrgRule#checkExistenceAndActive()}</li>
120     * <li>{@link OrgRule#checkOrgClosureRules(MaintenanceDocument)}</li>
121     * <li>{@link OrgRule#checkSimpleRules(MaintenanceDocument)}</li>
122     * <li>{@link OrgRule#checkDefaultAccountNumber(MaintenanceDocument)}</li>
123     * </ul>
124     * This rule fails on rule failure
125     *
126     * @see org.kuali.rice.kns.maintenance.rules.MaintenanceDocumentRuleBase#processCustomRouteDocumentBusinessRules(org.kuali.rice.kns.document.MaintenanceDocument)
127     */
128    @Override
129    protected boolean processCustomRouteDocumentBusinessRules(MaintenanceDocument document) {
130
131        boolean success = true;
132
133        LOG.debug("Entering processCustomRouteDocumentBusinessRules()");
134
135        // determine whether HRMS ORG is activated in this app instance
136        isHrmsOrgActivated = isHrmsOrgActivated();
137
138        // check that all sub-objects whose keys are specified have matching objects in the db
139        success &= checkExistenceAndActive();
140
141        // check that end date is greater than begin date and Reports To Chart/Org should not be same as this Chart/Org
142        success &= checkSimpleRules(document);
143
144        // check that defaultAccount is present unless
145        // ( (orgType = U or C) and ( document is a "create new" ))
146
147        //Code changes for JIRA OLE2344 don't check default account number as account no is made optional
148        //success &= checkDefaultAccountNumber(document);
149
150        success &= checkOrgClosureRules(document);
151
152        return success;
153    }
154
155    /**
156     * This performs the following checks on document save:
157     * <ul>
158     * <li>{@link OrgRule#checkExistenceAndActive()}</li>
159     * <li>{@link OrgRule#checkOrgClosureRules(MaintenanceDocument)}</li>
160     * <li>{@link OrgRule#checkSimpleRules(MaintenanceDocument)}</li>
161     * <li>{@link OrgRule#checkDefaultAccountNumber(MaintenanceDocument)}</li>
162     * </ul>
163     * This rule does not fail on rule failure
164     *
165     * @see org.kuali.rice.kns.maintenance.rules.MaintenanceDocumentRuleBase#processCustomSaveDocumentBusinessRules(org.kuali.rice.kns.document.MaintenanceDocument)
166     */
167    @Override
168    protected boolean processCustomSaveDocumentBusinessRules(MaintenanceDocument document) {
169
170        LOG.debug("Entering processCustomSaveDocumentBusinessRules()");
171
172        // determine whether HRMS ORG is activated in this app instance
173        isHrmsOrgActivated = isHrmsOrgActivated();
174
175        // check that all sub-objects whose keys are specified have matching objects in the db
176        checkExistenceAndActive();
177
178        checkOrgClosureRules(document);
179
180        // check that end date is greater than begin date and Reports To Chart/Org should not be same as this Chart/Org
181        checkSimpleRules(document);
182
183        // check that defaultAccount is present unless
184        // ( (orgType = U or C) and ( document is a "create new" ))
185
186        //Code changes for JIRA OLE2344 don't check default account number as account no is made optional
187        //checkDefaultAccountNumber(document);
188
189        return true;
190    }
191
192    /**
193     * This checks to see if the org is active
194     *
195     * @return true if the org is inactive or false otherwise
196     */
197    protected boolean checkExistenceAndActive() {
198
199        LOG.debug("Entering checkExistenceAndActive()");
200        boolean success = true;
201
202        // shortcut out with no enforcement if this org is closed
203        if (!newOrg.isActive()) {
204            return success;
205        }
206
207        // Disabling plant attribute checking
208//        success &= checkPlantAttributes();
209
210        return success;
211    }
212
213    /**
214     * This checks to see if a user is authorized for plant fields modification. If not then it returns true (without activating
215     * fields). If the org does not have to report to itself then it checks to see if the plant fields have been filled out
216     * correctly and fails if they haven't
217     *
218     * @return false if user can edit plant fields but they have not been filled out correctly
219     */
220    protected boolean checkPlantAttributes() {
221
222        boolean success = true;
223
224        /*
225         * KULCOA-1132 - exit if the user is not a member of the plant maintainer work group.
226         */
227
228        // get user
229        Person user = GlobalVariables.getUserSession().getPerson();
230
231        // if not authroized to edit plant fields, exit with true
232        if (isPlantAuthorized(user) == false) {
233            return true;
234        }
235
236        // relax this edit for
237        if (!getOrgMustReportToSelf(newOrg)) {
238            // require Org Plant ChartCode
239            success &= checkEmptyBOField("organizationPlantChartCode", newOrg.getOrganizationPlantChartCode(), "Organization Plant Chart of Accounts Code");
240
241            // require Org Plant AccountNumber
242            success &= checkEmptyBOField("organizationPlantAccountNumber", newOrg.getOrganizationPlantAccountNumber(), "Organization Plant Account Number");
243
244            // require Campus Plant ChartCode
245            success &= checkEmptyBOField("campusPlantChartCode", newOrg.getCampusPlantChartCode(), "Campus Plant Chart of Accounts Code");
246
247            // require Org Plant ChartCode
248            success &= checkEmptyBOField("campusPlantAccountNumber", newOrg.getCampusPlantAccountNumber(), "Campus Plant Account Number");
249
250            // validate Org Plant Account
251            success &= getDictionaryValidationService().validateReferenceExistsAndIsActive(newOrg, "organizationPlantAccount", MAINTAINABLE_ERROR_PREFIX + "organizationPlantAccountNumber", "Organization Plant Account");
252
253            // validate Campus Plant Account
254            success &= getDictionaryValidationService().validateReferenceExistsAndIsActive(newOrg, "campusPlantAccount", MAINTAINABLE_ERROR_PREFIX + "campusPlantAccountNumber", "Campus Plant Account");
255        }
256
257        return success;
258    }
259
260    /**
261     * This method enforces the business rules surrounding when an Org becomes closed/inactive. If we are editing and switching the
262     * org to inactive or if it is a new doc and it is marked as inactive then we assume we are closing the org. If we are not then
263     * we return true. If we are then we return false if there are still active accounts tied to the org
264     *
265     * @param document
266     * @return false if trying to close org but it still has accounts that are active linked to it
267     */
268    protected boolean checkOrgClosureRules(MaintenanceDocument document) {
269
270        boolean success = true;
271        boolean orgBeingClosed = false;
272        boolean checkForChildObjects = true;
273
274        // if its an edit, and its being closed
275        if (document.isEdit()) {
276            if (oldOrg.isActive() && !newOrg.isActive()) {
277                orgBeingClosed = true;
278            }
279        }
280
281        // if its new, and is being created as closed
282        if (document.isNew()) {
283            if (!newOrg.isActive()) {
284                orgBeingClosed = true;
285                // Since it's new, we don't need to check for accounts and orgs
286                checkForChildObjects = false;
287            }
288        }
289
290        // if the org isnt being closed, stop processing here
291        if (!orgBeingClosed) {
292            return success;
293        }
294
295        // FROM HERE ON WE'RE ASSUMING THE ORG IS BEING CLOSED
296
297        // do not allow the org to be closed while there are active accounts tied
298        // to this org
299        if (checkForChildObjects) {
300            List childAccounts = orgService.getActiveAccountsByOrg(newOrg.getChartOfAccountsCode(),
301                    newOrg.getOrganizationCode());
302            if (childAccounts.size() > 0) {
303
304                // get the first three accounts on the list for display
305                StringBuffer childAccountList = new StringBuffer();
306                int count = 0;
307                String delim = "";
308                for (Iterator iter = childAccounts.iterator(); iter.hasNext();) {
309                    Account account = (Account) iter.next();
310                    childAccountList
311                            .append(delim + account.getChartOfAccountsCode() + "-" + account.getAccountNumber());
312                    count++;
313                    if (count >= 1) {
314                        delim = ", ";
315                    }
316                    if (count >= 3) {
317                        break;
318                    }
319                }
320                if (childAccounts.size() > count) {
321                    childAccountList.append(", ... (" + (childAccounts.size() - count) + " more)");
322                }
323
324                putGlobalError(OLEKeyConstants.ERROR_DOCUMENT_ORGMAINT_OPEN_CHILD_ACCOUNTS_ON_ORG_CLOSURE,
325                        childAccountList.toString());
326                success &= false;
327            }
328
329            // do not allow this org to be closed while there are still active orgs
330            // that have this org as their reportsToOrg
331            List childOrgs = orgService.getActiveChildOrgs(newOrg.getChartOfAccountsCode(),
332                    newOrg.getOrganizationCode());
333            if (childOrgs.size() > 0) {
334
335                // get the first three orgs on the list for display
336                StringBuffer childOrgsList = new StringBuffer();
337                int count = 0;
338                String delim = "";
339                for (Iterator iter = childOrgs.iterator(); iter.hasNext();) {
340                    Organization org = (Organization) iter.next();
341                    childOrgsList.append(delim + org.getChartOfAccountsCode() + "-" + org.getOrganizationCode());
342                    count++;
343                    if (count >= 1) {
344                        delim = ", ";
345                    }
346                    if (count >= 3) {
347                        break;
348                    }
349                }
350                if (childOrgs.size() > count) {
351                    childOrgsList.append(", ... (" + (childOrgs.size() - count) + " more)");
352                }
353
354                putGlobalError(OLEKeyConstants.ERROR_DOCUMENT_ORGMAINT_OPEN_CHILD_ORGS_ON_ORG_CLOSURE,
355                        childOrgsList.toString());
356                success &= false;
357            }
358        }
359
360        // if org is being closed, end-date must be valid and present
361        if (ObjectUtils.isNull(newOrg.getOrganizationEndDate())) {
362            success &= false;
363            putFieldError("organizationEndDate", OLEKeyConstants.ERROR_DOCUMENT_ORGMAINT_END_DATE_REQUIRED_ON_ORG_CLOSURE);
364        }
365        return success;
366
367    }
368
369    /**
370     * This checks to see if the org is active and if it the HRMS org is active
371     *
372     * @param document
373     * @return true if either the org is inactive or isHrmsOrgActivated is false
374     */
375    protected boolean checkHrmsOrgRules(MaintenanceDocument document) {
376
377        boolean success = true;
378
379        // shortcut out with no enforcement if this org is closed
380        if (!newOrg.isActive()) {
381            return success;
382        }
383
384        // short circuit and fail if HRMSOrg is turned off
385        if (!isHrmsOrgActivated) {
386            return success;
387        }
388
389        // if the system has a HRMS Org record attached to this org record, then prompt the
390        // user to fill out the HRMS Org info
391
392        // HRMS Org Campus == Org Campus
393        // HRMS Org campus code must be the same as Org campus code
394
395
396        // if the
397        return success;
398    }
399
400    /**
401     * This checks our {@link Parameter} rules to see if this org needs to report to itself
402     *
403     * @param organization
404     * @return true if it does
405     */
406    protected boolean getOrgMustReportToSelf(Organization organization) {
407        return /*REFACTORME*/SpringContext.getBean(ParameterEvaluatorService.class).getParameterEvaluator(Organization.class, OLEConstants.ChartApcParms.ORG_MUST_REPORT_TO_SELF_ORG_TYPES, organization.getOrganizationTypeCode()).evaluationSucceeds();
408    }
409
410    /**
411     * This checks the following conditions:
412     * <ul>
413     * <li>begin date must be greater than or equal to end date</li>
414     * <li>start date must be greater than or equal to today if new Document</li>
415     * <li>Reports To Chart/Org should not be same as this Chart/Org</li>
416     * </ul>
417     *
418     * @param document
419     * @return true if it passes all the rules, false otherwise
420     */
421    protected boolean checkSimpleRules(MaintenanceDocument document) {
422
423        boolean success = true;
424        String lastReportsToChartOfAccountsCode;
425        String lastReportsToOrganizationCode;
426        boolean continueSearch;
427        Organization tempOrg;
428        Integer loopCount;
429        Integer maxLoopCount = 40;
430
431        // begin date must be greater than or equal to end date
432        if ((ObjectUtils.isNotNull(newOrg.getOrganizationBeginDate()) && (ObjectUtils.isNotNull(newOrg.getOrganizationEndDate())))) {
433
434            Date beginDate = newOrg.getOrganizationBeginDate();
435            Date endDate = newOrg.getOrganizationEndDate();
436
437            if (endDate.before(beginDate)) {
438                putFieldError("organizationEndDate", OLEKeyConstants.ERROR_DOCUMENT_ORGMAINT_END_DATE_GREATER_THAN_BEGIN_DATE);
439                success &= false;
440            }
441        }
442
443        // start date must be greater than or equal to today if new Document
444        if ((ObjectUtils.isNotNull(newOrg.getOrganizationBeginDate()) && (document.isNew()))) {
445            Timestamp today = getDateTimeService().getCurrentTimestamp();
446            today.setTime(DateUtils.truncate(today, Calendar.DAY_OF_MONTH).getTime());
447            if (newOrg.getOrganizationBeginDate().before(today)) {
448                putFieldError("organizationBeginDate", OLEKeyConstants.ERROR_DOCUMENT_ORGMAINT_STARTDATE_IN_PAST);
449                success &= false;
450            }
451        }
452
453        // Reports To Chart/Org should not be same as this Chart/Org
454        // However, allow special case where organization type is listed in the business rules
455        if (ObjectUtils.isNotNull(newOrg.getReportsToChartOfAccountsCode()) && ObjectUtils.isNotNull(newOrg.getReportsToOrganizationCode()) && ObjectUtils.isNotNull(newOrg.getChartOfAccountsCode()) && ObjectUtils.isNotNull(newOrg.getOrganizationCode())) {
456            if (!getOrgMustReportToSelf(newOrg)) {
457
458                if ((newOrg.getReportsToChartOfAccountsCode().equals(newOrg.getChartOfAccountsCode())) && (newOrg.getReportsToOrganizationCode().equals(newOrg.getOrganizationCode()))) {
459                    putFieldError("reportsToOrganizationCode", OLEKeyConstants.ERROR_DOCUMENT_ORGMAINT_REPORTING_ORG_CANNOT_BE_SAME_ORG);
460                    success = false;
461                }
462                else {
463                    // Don't allow a circular reference on Reports to Chart/Org
464                    // terminate the search when a top-level org is found
465                    lastReportsToChartOfAccountsCode = newOrg.getReportsToChartOfAccountsCode();
466                    lastReportsToOrganizationCode = newOrg.getReportsToOrganizationCode();
467                    continueSearch = true;
468                    loopCount = 0;
469                    do {
470                        tempOrg = orgService.getByPrimaryId(lastReportsToChartOfAccountsCode, lastReportsToOrganizationCode);
471                        loopCount++;
472                        ;
473                        if (ObjectUtils.isNull(tempOrg)) {
474                            continueSearch = false;
475                            // if a null is returned on the first iteration, then the reports-to org does not exist
476                            // fail the validation
477                            if (loopCount == 1) {
478                                putFieldError("reportsToOrganizationCode", OLEKeyConstants.ERROR_DOCUMENT_ORGMAINT_REPORTING_ORG_MUST_EXIST);
479                                success = false;
480                            }
481                        }
482                        else {
483                            // on the first iteration, check whether the reports-to organization is active
484                            if (loopCount == 1 && !tempOrg.isActive()) {
485                                putFieldError("reportsToOrganizationCode", OLEKeyConstants.ERROR_DOCUMENT_ORGMAINT_REPORTING_ORG_MUST_EXIST);
486                                success = false;
487                                continueSearch = false;
488                            }
489                            else {
490                                // LOG.info("Found Org = " + lastReportsToChartOfAccountsCode + "/" +
491                                // lastReportsToOrganizationCode);
492                                lastReportsToChartOfAccountsCode = tempOrg.getReportsToChartOfAccountsCode();
493                                lastReportsToOrganizationCode = tempOrg.getReportsToOrganizationCode();
494
495                                if ((tempOrg.getReportsToChartOfAccountsCode().equals(newOrg.getChartOfAccountsCode())) && (tempOrg.getReportsToOrganizationCode().equals(newOrg.getOrganizationCode()))) {
496                                    putFieldError("reportsToOrganizationCode", OLEKeyConstants.ERROR_DOCUMENT_ORGMAINT_REPORTING_ORG_CANNOT_BE_CIRCULAR_REF_TO_SAME_ORG);
497                                    success = false;
498                                    continueSearch = false;
499                                }
500                            }
501                        }
502                        if (loopCount > maxLoopCount) {
503                            continueSearch = false;
504                        }
505                        // stop the search if we reach an org that must report to itself
506                        if (continueSearch && /*REFACTORME*/SpringContext.getBean(ParameterEvaluatorService.class).getParameterEvaluator(Organization.class, OLEConstants.ChartApcParms.ORG_MUST_REPORT_TO_SELF_ORG_TYPES, tempOrg.getOrganizationTypeCode()).evaluationSucceeds()) {
507                            continueSearch = false;
508                        }
509
510                    } while (continueSearch == true);
511                } // end else (checking for circular ref)
512            }
513            else { // org must report to self (university level organization)
514                if (!(newOrg.getReportsToChartOfAccountsCode().equals(newOrg.getChartOfAccountsCode()) && newOrg.getReportsToOrganizationCode().equals(newOrg.getOrganizationCode()))) {
515                    putFieldError("reportsToOrganizationCode", OLEKeyConstants.ERROR_DOCUMENT_ORGMAINT_REPORTING_ORG_MUST_BE_SAME_ORG);
516                    success = false;
517                }
518                // org must be the only one of that type
519                String topLevelOrgTypeCode = SpringContext.getBean(ParameterService.class).getParameterValueAsString(Organization.class, OLEConstants.ChartApcParms.ORG_MUST_REPORT_TO_SELF_ORG_TYPES);
520                List<Organization> topLevelOrgs = orgService.getActiveOrgsByType(topLevelOrgTypeCode);
521                if (!topLevelOrgs.isEmpty()) {
522                    // is the new org in the topLevelOrgs list? If not, then there's an error; if so, we're editing the top level
523                    // org
524                    if (!topLevelOrgs.contains(newOrg)) {
525                        putFieldError("organizationTypeCode", OLEKeyConstants.ERROR_DOCUMENT_ORGMAINT_ONLY_ONE_TOP_LEVEL_ORG, topLevelOrgs.get(0).getChartOfAccountsCode() + "-" + topLevelOrgs.get(0).getOrganizationCode());
526                        success = false;
527                    }
528                }
529            }
530        }
531
532
533        return success;
534    }
535
536
537    /**
538     * This checks that defaultAccount is present unless ( (orgType = U or C) and ( document is a "create new" or "edit" ))
539     *
540     * @param document
541     * @return false if missing default account number and it is not an exempt type code
542     */
543    protected boolean checkDefaultAccountNumber(MaintenanceDocument document) {
544
545        boolean success = true;
546        boolean exemptOrganizationTypeCode = false;
547        boolean missingDefaultAccountNumber = StringUtils.isBlank(newOrg.getOrganizationDefaultAccountNumber());
548
549        if (ObjectUtils.isNotNull(newOrg.getOrganizationTypeCode())) {
550            String organizationTypeCode = newOrg.getOrganizationTypeCode();
551            if (/*REFACTORME*/SpringContext.getBean(ParameterEvaluatorService.class).getParameterEvaluator(Organization.class, OLEConstants.ChartApcParms.DEFAULT_ACCOUNT_NOT_REQUIRED_ORG_TYPES, newOrg.getOrganizationTypeCode()).evaluationSucceeds()) {
552                exemptOrganizationTypeCode = true;
553            }
554        }
555
556        if (missingDefaultAccountNumber && (!exemptOrganizationTypeCode || (!document.isNew() && !document.isEdit()))) {
557            putFieldError("organizationDefaultAccountNumber", OLEKeyConstants.ERROR_DOCUMENT_ORGMAINT_DEFAULT_ACCOUNT_NUMBER_REQUIRED);
558            success &= false;
559        }
560
561        return success;
562    }
563
564    /**
565     * This method compares an old and new value, and determines if they've changed. If the old was null/blank, and the new is not,
566     * return true. If the old had a value, and the new is null/blank, return true. If both old and new had a value, and the values
567     * are different (excluding trailing or leading whitespaces, and excluding case changes), return true. If none of the above,
568     * return false.
569     *
570     * @param oldValue - Old value to test.
571     * @param newValue - New value to test.
572     * @return true or false, based on the algorithm described above.
573     */
574    protected boolean fieldsHaveChanged(String oldValue, String newValue) {
575
576        // if old was null/blank and new is not
577        if (StringUtils.isBlank(oldValue) && StringUtils.isNotBlank(newValue)) {
578            return true;
579        }
580
581        // if old had a value, but new is null/blank
582        if (StringUtils.isNotBlank(oldValue) && StringUtils.isBlank(newValue)) {
583            return true;
584        }
585
586        // at this point, we know that we had a value before, and we have a
587        // value now, so we need to test whether this value has changed
588        if (oldValue != null && newValue != null) {
589            if (!oldValue.trim().equalsIgnoreCase(newValue.trim())) {
590                return true;
591            }
592        }
593
594        // if we've made it to here, then no changes have happened to the values
595        return false;
596    }
597
598    /**
599     * This method looks up in the ParameterService whether ther HRMS Org system is turned on.
600     *
601     * @return true or false depending on the app configuration
602     */
603    protected boolean isHrmsOrgActivated() {
604        return SpringContext.getBean(ParameterService.class).getParameterValueAsBoolean(Organization.class, OLEConstants.ChartApcParms.APC_HRMS_ACTIVE_KEY);
605    }
606
607    /**
608     * This method sets the convenience objects like newOrg and oldOrg, so you have short and easy handles to the new and old
609     * objects contained in the maintenance document. It also calls the BusinessObjectBase.refresh(), which will attempt to load all
610     * sub-objects from the DB by their primary keys, if available.
611     *
612     * @param document - the maintenanceDocument being evaluated
613     */
614    @Override
615    public void setupConvenienceObjects() {
616
617        // setup oldAccount convenience objects, make sure all possible sub-objects are populated
618        oldOrg = (Organization) super.getOldBo();
619
620        // setup newAccount convenience objects, make sure all possible sub-objects are populated
621        newOrg = (Organization) super.getNewBo();
622    }
623
624    /**
625     * This method tests whether the specified user is part of the group that grants authorization to the Plant fields.
626     *
627     * @param user - the user to test
628     * @return true if user is part of the group, false otherwise
629     */
630    protected boolean isPlantAuthorized(Person user) {
631        String principalId = user.getPrincipalId();
632        String namespaceCode = OLEConstants.ParameterNamespaces.KNS;
633        String permissionTemplateName = KimConstants.PermissionTemplateNames.MODIFY_FIELD;
634
635        Map<String,String> roleQualifiers = new HashMap<String,String>();
636        roleQualifiers.put(OleKimAttributes.CHART_OF_ACCOUNTS_CODE, newOrg.getChartOfAccountsCode());
637
638        Map<String,String> permissionDetails = new HashMap<String,String>();
639        permissionDetails.put(KimConstants.AttributeConstants.COMPONENT_NAME, Organization.class.getSimpleName());
640        permissionDetails.put(KimConstants.AttributeConstants.PROPERTY_NAME, OLEPropertyConstants.ORGANIZATION_PLANT_CHART_CODE);
641
642        IdentityManagementService identityManagementService = SpringContext.getBean(IdentityManagementService.class);
643        Boolean isAuthorized = identityManagementService.isAuthorizedByTemplateName(principalId, namespaceCode, permissionTemplateName, permissionDetails, roleQualifiers);
644        if (!isAuthorized) {
645            if(LOG.isDebugEnabled()) {
646                LOG.debug("User '" + user.getPrincipalName() + "' has no access to the Plant Chart.");
647            }
648            return false;
649        }
650
651        permissionDetails.put(KimConstants.AttributeConstants.PROPERTY_NAME, OLEPropertyConstants.ORGANIZATION_PLANT_ACCOUNT_NUMBER);
652        isAuthorized = identityManagementService.isAuthorizedByTemplateName(principalId, namespaceCode, permissionTemplateName, permissionDetails, roleQualifiers);
653        if (!isAuthorized) {
654            if(LOG.isDebugEnabled()) {
655                LOG.debug("User '" + user.getPrincipalName() + "' has no access to the Plant account.");
656            }
657            return false;
658        }
659
660        if(LOG.isDebugEnabled()) {
661            LOG.debug("User '" + user.getPrincipalName() + "' has access to the Plant fields.");
662        }
663        return true;
664    }
665}