org.phoenicis.win32.registry.RegistryParser.java Source code

Java tutorial

Introduction

Here is the source code for org.phoenicis.win32.registry.RegistryParser.java

Source

/*
 * Copyright (C) 2015-2017 PRIS Quentin
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License along
 * with this program; if not, write to the Free Software Foundation, Inc.,
 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
 */

package org.phoenicis.win32.registry;

import org.apache.commons.lang.StringUtils;
import org.phoenicis.configuration.security.Safe;

import java.io.BufferedReader;
import java.io.File;
import java.io.FileReader;
import java.io.IOException;
import java.text.ParseException;

/***
 * Registry parser class If someone wants to improve this code, feel free to do
 * it!
 */
@Safe
public class RegistryParser {
    private static final char QUOTE = '"';
    private static final String PARSE_ERROR_MESSAGE = "Invalid registry file. Error found line %s";

    enum ParseState {
        INITIAL, READING_NAME, SEPARATOR, READING_VALUE
    }

    public RegistryKey parseFile(File registryFile, String rootName) {
        try (BufferedReader bufferReader = new BufferedReader(new FileReader(registryFile))) {
            final RegistryKey root = new RegistryKey(rootName);
            RegistryKey lastNode = null;
            int lineNumber = 1;
            for (String currentLine = bufferReader.readLine(); currentLine != null; currentLine = bufferReader
                    .readLine()) {

                while (currentLine.trim().endsWith("\\")) {
                    currentLine = StringUtils.substring(currentLine.trim(), 0, -1) + bufferReader.readLine().trim();
                }

                if (currentLine.startsWith(";") || currentLine.startsWith("#") || StringUtils.isBlank(currentLine)
                        || currentLine.startsWith("@")) {
                    lineNumber++;
                    continue;
                }

                if (currentLine.startsWith("[")) {
                    lastNode = this.parseDirectoryLine(root, currentLine);
                } else if (lineNumber == 1) {
                    lineNumber++;
                    continue;
                } else if (lastNode == null) {
                    throw new ParseException(String.format(PARSE_ERROR_MESSAGE, lineNumber), 0);
                } else {
                    this.parseValueLine(lastNode, currentLine, lineNumber);
                }

                lineNumber++;
            }

            return root;
        } catch (IOException | ParseException e) {
            throw new IllegalArgumentException("Error while parsing the registry", e);
        }
    }

    private void parseValueLine(RegistryKey lastNode, String currentLine, int lineNumber) throws ParseException {
        if (!currentLine.startsWith("\"")) {
            throw new ParseException(String.format(PARSE_ERROR_MESSAGE, lineNumber), 0);
        }

        final StringBuilder nameBuilder = new StringBuilder();
        final StringBuilder valueBuilder = new StringBuilder();

        ParseState parseState = ParseState.INITIAL;
        Boolean ignoreNextQuote = false;

        for (int i = 0; i < currentLine.length(); i++) {
            char currentChar = currentLine.charAt(i);

            if (parseState == ParseState.INITIAL) {
                if (currentChar == QUOTE) {
                    parseState = ParseState.READING_NAME;
                }
            } else if (parseState == ParseState.READING_NAME) {
                if (currentChar == '"' && !ignoreNextQuote) {
                    parseState = ParseState.SEPARATOR;
                } else if (currentChar == '\\' && !ignoreNextQuote) {
                    ignoreNextQuote = true;
                } else {
                    nameBuilder.append(currentChar);
                    ignoreNextQuote = false;
                }
            } else if (parseState == ParseState.SEPARATOR) {
                if (currentChar != '=') {
                    throw new ParseException(String.format(PARSE_ERROR_MESSAGE, lineNumber), 0);
                } else {
                    parseState = ParseState.READING_VALUE;
                }
            } else {
                valueBuilder.append(currentChar);
            }
        }

        final String name = nameBuilder.toString();
        try {
            RegistryValue<AbstractValueType> value = RegistryValue.fromString(name, valueBuilder.toString());
            lastNode.addChild(value);
        } catch (IllegalArgumentException e) {
            throw new ParseException(String.format("Error on line %s (%s-: %s", lineNumber, currentLine, e), 0);
        }
    }

    private RegistryKey parseDirectoryLine(RegistryKey root, String currentLine) {
        final String extractedLine = extractDirectoryLine(currentLine);
        final String[] splitLine = extractedLine.split("\\\\\\\\");
        RegistryKey currentKey = root;
        for (String registryKeyName : splitLine) {
            RegistryKey childrenSearched = (RegistryKey) currentKey.getChild(registryKeyName);
            if (childrenSearched != null) {
                currentKey = childrenSearched;
            } else {
                RegistryKey newChild = new RegistryKey(registryKeyName);
                currentKey.addChild(newChild);
                currentKey = newChild;
            }
        }
        return currentKey;
    }

    private String extractDirectoryLine(String currentLine) {
        return currentLine.substring(1, currentLine.indexOf(']'));
    }
}