All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.gs.obevo.impl.reader.TextMarkupDocumentReader Maven / Gradle / Ivy

/**
 * Copyright 2017 Goldman Sachs.
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 * http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing,
 * software distributed under the License is distributed on an
 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
 * KIND, either express or implied.  See the License for the
 * specific language governing permissions and limitations
 * under the License.
 */
package com.gs.obevo.impl.reader;

import java.util.List;

import com.gs.obevo.api.appdata.doc.TextMarkupDocument;
import com.gs.obevo.api.appdata.doc.TextMarkupDocumentSection;
import com.gs.obevo.db.sqlparser.textmarkup.TextMarkupLineSyntaxParserConstants;
import com.gs.obevo.db.sqlparser.textmarkup.TextMarkupParser;
import com.gs.obevo.db.sqlparser.textmarkup.Token;
import com.gs.obevo.impl.text.CommentRemover;
import org.apache.commons.lang3.StringUtils;
import org.eclipse.collections.api.block.function.Function;
import org.eclipse.collections.api.block.predicate.Predicate;
import org.eclipse.collections.api.list.ImmutableList;
import org.eclipse.collections.api.list.MutableList;
import org.eclipse.collections.api.map.ImmutableMap;
import org.eclipse.collections.api.map.MutableMap;
import org.eclipse.collections.api.set.ImmutableSet;
import org.eclipse.collections.api.set.MutableSet;
import org.eclipse.collections.api.tuple.Pair;
import org.eclipse.collections.impl.factory.Lists;
import org.eclipse.collections.impl.factory.Maps;
import org.eclipse.collections.impl.factory.Sets;
import org.eclipse.collections.impl.tuple.Tuples;

public class TextMarkupDocumentReader {
    public static final String TAG_CHANGE = "CHANGE";
    public static final String TAG_METADATA = "METADATA";
    public static final String TAG_BODY = "BODY";
    public static final String TAG_DROP_COMMAND = "DROP_COMMAND";
    public static final String TAG_ROLLBACK = "ROLLBACK";  // not yet used elsewhere, unlike the
    // rollbackIfAlreadyDeployed
    public static final String TAG_ROLLBACK_IF_ALREADY_DEPLOYED = "ROLLBACK-IF-ALREADY-DEPLOYED";
    public static final String TOGGLE_DISABLE_QUOTED_IDENTIFIERS = "DISABLE_QUOTED_IDENTIFIERS";
    public static final String ATTR_UPDATE_TIME_COLUMN = "updateTimeColumn";
    public static final String ATTR_DEPENDENCIES = "dependencies";
    public static final String ATTR_EXCLUDE_DEPENDENCIES = "excludeDependencies";
    public static final String ATTR_INCLUDE_DEPENDENCIES = "includeDependencies";
    public static final String ATTR_PRIMARY_KEYS = "primaryKeys";
    public static final String INCLUDE_ENVS = "includeEnvs";
    public static final String EXCLUDE_ENVS = "excludeEnvs";
    public static final String INCLUDE_PLATFORMS = "includePlatforms";
    public static final String EXCLUDE_PLATFORMS = "excludePlatforms";
    public static final String TAG_PERM_SCHEME = "permissionScheme";

    private final ImmutableList firstLevelElements = Lists.immutable.with(TAG_CHANGE, TAG_DROP_COMMAND, TAG_METADATA, TAG_BODY);
    private final ImmutableList singleLineElements = Lists.immutable.with(TAG_METADATA);
    private final ImmutableList secondLevelElements = Lists.immutable.with(TAG_ROLLBACK_IF_ALREADY_DEPLOYED, TAG_ROLLBACK);

    private final boolean legacyMode;

    public TextMarkupDocumentReader(boolean legacyMode) {
        this.legacyMode = legacyMode;
    }

    public TextMarkupDocument parseString(String text, final TextMarkupDocumentSection otherSection) {
        ImmutableList textMarkupDocumentSections = this.parseString(text, this.firstLevelElements, true, "////");

        if (otherSection != null) {
            TextMarkupDocumentSection thisSection = textMarkupDocumentSections.detect(new Predicate() {
                @Override
                public boolean accept(TextMarkupDocumentSection it) {
                    return it.getName().equals(otherSection.getName());
                }
            });
            if (thisSection != null) {
                thisSection.mergeAttributes(otherSection);
            } else {
                textMarkupDocumentSections = textMarkupDocumentSections.newWith(otherSection);
            }
        }

        return new TextMarkupDocument(textMarkupDocumentSections);
    }

    private ImmutableList parseString(String text, ImmutableList elementsToCheck, final boolean recurse,
            final String elementPrefix) {
        MutableList> outerSections = splitIntoMainSections(text, elementsToCheck, elementPrefix);

        MutableList sections = outerSections.flatCollect(new ConvertOuterSectionToTextSection(recurse, elementPrefix));

        // remove any blank sections
        return sections.toImmutable().reject(new Predicate() {
            @Override
            public boolean accept(TextMarkupDocumentSection each) {
                return recurse && each.getName() == null
                        && (StringUtils.isBlank(each.getContent())
                        || StringUtils.isBlank(CommentRemover.removeComments(each.getContent(), "removing on markup document reader"))  // need comments in a separate clause as CommentRemover returns a "null" string on null; will fix eventually
                );
            }
        });
    }

    private MutableList> splitIntoMainSections(String text, ImmutableList elementsToCheck, String elementPrefix) {
        MutableList> outerSections = Lists.mutable.empty();
        String nextSectionName = null;
        boolean startOfSearch = true;

        // here, we go in a loop searching for the next referenc of "[elementPrefix] [sectionName]", e.g. //// CHANGE
        // By each of those points, we split those into separate text sections and return back to the client.
        // We aim to preserve the line breaks found when parsing the sections
        while (text != null) {
            String currentSectionName = nextSectionName;
            String currentSectionText;

            int earliestIndex = Integer.MAX_VALUE;

            for (String firstLevelElement : elementsToCheck) {
                // on the first search, the text may start w/ the section; hence, we set the search fromIndex param to 0.
                // Subsequently, the index picks up at the beginning of the next section; hence, we must start
                // the search at the next character, so the fromIndex param is 1
                int index = text.indexOf(elementPrefix + " " + firstLevelElement, startOfSearch ? 0 : 1);

                if (index != -1 && index < earliestIndex) {
                    earliestIndex = index;
                    nextSectionName = firstLevelElement;
                }
            }

            startOfSearch = false;

            if (earliestIndex == Integer.MAX_VALUE) {
                currentSectionText = StringUtils.chomp(text);
                text = null;
            } else {
                currentSectionText = StringUtils.chomp(text.substring(0, earliestIndex));
                text = text.substring(earliestIndex);
            }

            outerSections.add(Tuples.pair(currentSectionName, currentSectionText));
        }
        return outerSections;
    }

    private class ConvertOuterSectionToTextSection implements Function, Iterable> {
        private final boolean recurse;
        private final String elementPrefix;

        ConvertOuterSectionToTextSection(boolean recurse, String elementPrefix) {
            this.elementPrefix = elementPrefix;
            this.recurse = recurse;
        }

        @Override
        public Iterable valueOf(Pair outerSection) {
            String currentSectionName = outerSection.getOne();
            if (currentSectionName == null) {
                return Lists.mutable.with(new TextMarkupDocumentSection(null, outerSection.getTwo()));
            } else {
                String[] contents = outerSection.getTwo().split("\\r?\\n", 2);
                String firstLine = contents[0];
                firstLine = firstLine.replaceFirst(elementPrefix + " " + currentSectionName, "");

                Pair, ImmutableSet> attrsTogglesPair = parseAttrsAndToggles(firstLine);
                ImmutableMap attrs = attrsTogglesPair.getOne();
                ImmutableSet toggles = attrsTogglesPair.getTwo();

                String sectionContent = contents.length > 1 ? contents[1] : null;
                if (singleLineElements.contains(currentSectionName)) {
                    TextMarkupDocumentSection metadataSection = new TextMarkupDocumentSection(currentSectionName, null, attrs.toImmutable());
                    metadataSection.setToggles(toggles.toImmutable());
                    return Lists.mutable.with(metadataSection, new TextMarkupDocumentSection(null, sectionContent));
                } else {
                    ImmutableList finalsubsections = Lists.immutable.empty();
                    String finalContent;
                    if (!recurse) {
                        finalContent = sectionContent;
                    } else if (sectionContent != null) {
                        ImmutableList subsections = parseString(sectionContent, secondLevelElements, false, "//");
                        if (subsections.size() == 1) {
                            finalContent = sectionContent;
                        } else {
                            finalContent = subsections.get(0).getContent();
                            finalsubsections = subsections.subList(1, subsections.size());
                        }
                    } else {
                        finalContent = null;
                    }

                    TextMarkupDocumentSection section = new TextMarkupDocumentSection(currentSectionName, finalContent, attrs.toImmutable());
                    section.setToggles(toggles.toImmutable());
                    section.setSubsections(finalsubsections);
                    return Lists.mutable.with(section);
                }
            }
        }
    }

    Pair, ImmutableSet> parseAttrsAndToggles(String line) {
        MutableMap attrs = Maps.mutable.empty();
        MutableSet toggles = Sets.mutable.empty();

        if (!legacyMode) {
            List tokens = TextMarkupParser.parseTokens(line);
            Token curToken = !tokens.isEmpty() ? tokens.get(0) : null;
            while (curToken != null && curToken.kind != TextMarkupLineSyntaxParserConstants.EOF) {
                switch (curToken.kind) {
                case TextMarkupLineSyntaxParserConstants.WHITESPACE:
                    // skip whitespace if encountered
                    break;
                case TextMarkupLineSyntaxParserConstants.QUOTED_LITERAL:
                case TextMarkupLineSyntaxParserConstants.STRING_LITERAL:
                    // let's check if this is a toggle or an attribute
                    if (curToken.next.kind == TextMarkupLineSyntaxParserConstants.ASSIGN) {
                        Token keyToken = curToken;
                        curToken = curToken.next;  // to ASSIGN
                        curToken = curToken.next;  // to the following token
                        switch (curToken.kind) {
                        case TextMarkupLineSyntaxParserConstants.QUOTED_LITERAL:
                        case TextMarkupLineSyntaxParserConstants.STRING_LITERAL:
                            // in this case, we have an attribute value
                            String value = curToken.image;
                            if (value.charAt(0) == '"' && value.charAt(value.length() - 1) == '"') {
                                value = curToken.image.substring(1, curToken.image.length() - 1);
                            }
                            value = value.replaceAll("\\\\\"", "\"");
                            attrs.put(keyToken.image, value);
                            break;
                        case TextMarkupLineSyntaxParserConstants.WHITESPACE:
                        case TextMarkupLineSyntaxParserConstants.EOF:
                            // in this case, we will assume a blank value
                            attrs.put(keyToken.image, "");
                            break;
                        case TextMarkupLineSyntaxParserConstants.ASSIGN:
                        default:
                            throw new IllegalStateException("Not allowed here");
                        }
                    } else {
                        toggles.add(curToken.image);
                    }
                    break;
                case TextMarkupLineSyntaxParserConstants.ASSIGN:
                    toggles.add(curToken.image);
                    break;
                case TextMarkupLineSyntaxParserConstants.EOF:
                default:
                    throw new IllegalStateException("Should not arise");
                }

                curToken = curToken.next;
            }
        } else {
            // keeping this mode for backwards-compatibility until we can guarantee all clients are fine without it
            // This way cannot handle spaces in quotes
            String[] args = StringUtils.splitByWholeSeparator(line, " ");

            for (String arg : args) {
                if (arg.contains("=")) {
                    String[] attr = arg.split("=");
                    if (attr.length > 2) {
                        throw new IllegalArgumentException("Cannot mark = multiple times in a parameter - " + line);
                    }
                    String attrVal = attr[1];
                    if (attrVal.startsWith("\"") && attrVal.endsWith("\"")) {
                        attrVal = attrVal.substring(1, attrVal.length() - 1);
                    }
                    attrs.put(attr[0], attrVal);
                } else if (StringUtils.isNotBlank(arg)) {
                    toggles.add(arg);
                }
            }
        }

        return Tuples.pair(attrs.toImmutable(), toggles.toImmutable());
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy