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.krad.devtools.maintainablexml;
017    
018    import java.io.BufferedReader;
019    import java.io.File;
020    import java.io.IOException;
021    import java.io.InputStreamReader;
022    import java.util.Arrays;
023    import java.util.HashMap;
024    import javax.xml.parsers.DocumentBuilder;
025    import javax.xml.parsers.DocumentBuilderFactory;
026    
027    import org.apache.commons.lang.StringUtils;
028    import org.w3c.dom.Document;
029    import org.w3c.dom.Element;
030    import org.w3c.dom.Node;
031    import org.w3c.dom.NodeList;
032    
033    /**
034     * This is a command line utility class which upgrades the maintenance document xml stored in krns_maint_doc_t.doc_cntnt
035     * to be able to still open and use any maintenance documents that were enroute at the time of an upgrade to Rice 2.0.
036     *
037     * <p>Instructions:</p>
038     * <ol>
039     *   <li>Backup database.</li>
040     *   <li>Add the conversion rules to the rules xml file -
041     *       ..\rice\development-tools\src\main\resources\org\kuali\rice\devtools\krad\maintainablexml\MaintainableXMLUpgradeRules.xml
042     *       See comments in the xml file to setup the rules.</li>
043     *   <li>Run this class.  Note that any classes being converted must be on the classpath.</li>
044     *   <li>Enter the rice config file location that has the database connection properties. Only enter the location relative
045     *       to user.dir.  NOTE -- If it is uncommented in ConvertMaintainableXML.java, MaintainableXMLConversionConfig.xml
046     *       in the user.dir will be used as the conversion config file.  runmode, fromRange, and toRange can be
047     *       specified in this file to avoid prompting the user for this information.  An example file can be found in
048     *        ..\rice\development-tools\src\main\resources\org\kuali\rice\devtools\krad\maintainablexml\MaintainableXMLConversionConfig.xml
049     *   </li>
050     *   <li>If it is not specified in the config file, the user will be prompted for a run mode. Mode 1 will do the xml
051     *   upgrade and update the krns_maint_doc_t table with the new xml. CANNOT ROLL BACK AFTER THIS HAS BEEN DONE.
052     *   Mode 2 will only print out the old xml and the new xml - this can be used to test the rules xml setup.</li>
053     *   <li>If it is not specified in the config file, the user will be prompted for the document number to upgrade from.
054     *   Enter "all" in the from range prompt to upgrade all documents.  Otherwise this should be an starting document ID
055     *   such as 1000</li>
056     *   <li>If it is not specified in the config file and the fromRange is not "all", the user will be prompted for the document
057     *   number to upgrade to.  If "all" was entered for the fromRange, toRange will be ignored.  toRange should be a
058     *   document ID such as 9999</li>
059     * </ol>
060     *
061     * @author Kuali Rice Team (rice.collab@kuali.org)
062     */
063    public class ConvertMaintainableXML {
064    
065        private static final String MAINTAINABLE_CONVERSION_CONFIG_FILE = "maintainable.conversion.rule.file";
066    
067        public static void main(String[] args) {
068    
069            // NOTE : If you want the user to be prompted for the config settings file name, Leave this set to ""
070            // Otherwise uncomment out the second line and it will look for the file MaintainableXMLConversionConfig.xml
071    
072            String settingsFile = "";
073            //settingsFile =  "MaintainableXMLConversionConfig.xml";
074    
075            if (StringUtils.isEmpty(settingsFile)) {
076                settingsFile = readInput("Enter config file location relative to Kuali main user directory "
077                    + "(ie. dev/sample-app-config.xml OR MaintainableXMLConversionConfig.xml) : ", null, false);
078            }
079    
080            String filePath = System.getProperty("user.home") + "/" + settingsFile;
081    
082            File file = new File(filePath);
083            if (!file.exists()) {
084                System.out.println("The settings file location is not valid : \n" + filePath);
085                System.exit(1);
086            }
087    
088            try {
089                HashMap settingsMap = getSettings(filePath);
090                System.out.println("Using the following settings : " + settingsMap + "\n");
091    
092                //  RUN MODE
093                String runMode = (String)settingsMap.get("runMode");
094                if (StringUtils.isEmpty(runMode) || !(StringUtils.equals(runMode, "1") || StringUtils.equals(runMode, "2"))) {
095                    runMode = readInput("Run mode :\n1. Update xml in DB\n2. Print old and new xml\nEnter 1 or 2\n",
096                        new String[]{"1", "2"}, false);
097                }
098    
099                //  FROM RANGE
100                String fromRange = (String)settingsMap.get("fromRange");
101                if (StringUtils.isEmpty(fromRange) || (!StringUtils.equals(fromRange, "all") && !(isNumber(fromRange)))) {
102                    fromRange = readInput("Please enter document start range value ('all', empty line for all, or the starting document number such as 1000) :\n",
103                        new String[]{"all", ""}, true);
104                }
105    
106                String toRange = null;
107                boolean hasRangeParameters = false;
108    
109                if (!"".equals(fromRange) && !"all".equals(fromRange)) {
110                    //  TO RANGE, if needed
111                    toRange = (String)settingsMap.get("toRange");
112                    if (StringUtils.isEmpty(toRange) || !(isNumber(toRange))) {
113                        toRange = readInput("Please enter end range value :\n", null, true);
114                    }
115                    System.out.println("Upgrading document numbers from " + fromRange + " to " + toRange + "\n");
116                    hasRangeParameters = true;
117                }
118    
119                FileConverter fileConverter = new FileConverter();
120    
121                // RUN THE CONVERSION!
122                fileConverter.runFileConversion(settingsMap, runMode, fromRange, toRange, hasRangeParameters);
123    
124            } catch (Exception e) {
125                System.out.println("Error executing conversion : " + e.getMessage());
126                e.printStackTrace();
127            }
128        }
129    
130        public static boolean isNumber(String string) {
131            try {
132                Integer.parseInt(string);
133            } catch (Exception e) {
134                return false;
135            }
136            return true;
137        }
138    
139        /**
140         * Displays message in command line and read user input. Checks that entered values are within valid input.
141         *
142         * @param message - the message string to print out
143         * @param validOptions - the allowed user input
144         * @return the string input from the user
145         */
146        private static String readInput(String message, String[] validOptions, boolean numeric) {
147            System.out.print(message);
148            BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
149    
150            String inputString = null;
151    
152            try {
153                inputString = br.readLine();
154                if (numeric && (validOptions == null || !Arrays.asList(validOptions).contains(inputString))) {
155                    Integer.parseInt(inputString);
156                    return inputString;
157                }
158            } catch (IOException ioe) {
159                System.out.println("IO error trying to read input!");
160                System.exit(1);
161            } catch (NumberFormatException nfe) {
162                System.out.println("Invalid Option! Must be numeric.");
163                readInput(message, validOptions, numeric);
164            }
165            if (validOptions != null && !Arrays.asList(validOptions).contains(inputString)) {
166                System.out.println("Invalid Option!");
167                readInput(message, validOptions, numeric);
168            }
169            return inputString;
170        }
171    
172        /**
173         * Parses settings file and put the properties in a map.
174         *
175         * @param filePath - the location of the settings file
176         * @return a HashMap populated with the settings
177         * @throws Exception
178         */
179        public static HashMap getSettings(String filePath) throws Exception {
180            File file = new File(filePath);
181            HashMap params = new HashMap();
182            DocumentBuilderFactory dbf = DocumentBuilderFactory.newInstance();
183            DocumentBuilder db = dbf.newDocumentBuilder();
184            Document doc = db.parse(file);
185            doc.getDocumentElement().normalize();
186            NodeList nodeLst = doc.getElementsByTagName("param");
187    
188            for (int s = 0; s < nodeLst.getLength(); s++) {
189                Node fstNode = nodeLst.item(s);
190                if (fstNode.getNodeType() == Node.ELEMENT_NODE) {
191                    Element fstElmnt = (Element) fstNode;
192                    String paramName = fstElmnt.getAttribute("name");
193                    NodeList textFNList = fstElmnt.getChildNodes();
194                    if (textFNList.item(0) != null) {
195                        String paramValue = ((Node) textFNList.item(0)).getNodeValue().trim();
196                        params.put(paramName, paramValue);
197                    } else {
198                        params.put(paramName, null);
199                    }
200                }
201            }
202            return params;
203        }
204    
205    }