com.sun.enterprise.admin.servermgmt.stringsubs.impl.StringSubstitutionEngine Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of payara-micro Show documentation
Show all versions of payara-micro Show documentation
Payara-Micro Distribution of the Payara Project
/*
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
*
* Copyright (c) 2013 Oracle and/or its affiliates. All rights reserved.
*
* The contents of this file are subject to the terms of either the GNU
* General Public License Version 2 only ("GPL") or the Common Development
* and Distribution License("CDDL") (collectively, the "License"). You
* may not use this file except in compliance with the License. You can
* obtain a copy of the License at
* https://glassfish.dev.java.net/public/CDDL+GPL_1_1.html
* or packager/legal/LICENSE.txt. See the License for the specific
* language governing permissions and limitations under the License.
*
* When distributing the software, include this License Header Notice in each
* file and include the License file at packager/legal/LICENSE.txt.
*
* GPL Classpath Exception:
* Oracle designates this particular file as subject to the "Classpath"
* exception as provided by Oracle in the GPL Version 2 section of the License
* file that accompanied this code.
*
* Modifications:
* If applicable, add the following below the License Header, with the fields
* enclosed by brackets [] replaced by your own identifying information:
* "Portions Copyright [year] [name of copyright owner]"
*
* Contributor(s):
* If you wish your version of this file to be governed by only the CDDL or
* only the GPL Version 2, indicate your decision by adding "[Contributor]
* elects to include this software in this distribution under the [CDDL or GPL
* Version 2] license." If you don't indicate a single choice of license, a
* recipient has the option to distribute your version of this file under
* either the CDDL, the GPL Version 2 or to extend the choice of license to
* its licensees as provided above. However, if you add GPL Version 2 code
* and therefore, elected the GPL Version 2 license, then the option applies
* only if the new code is made subject to such option by the copyright
* holder.
*/
// Portions Copyright [2018] Payara Foundation and/or affiliates
package com.sun.enterprise.admin.servermgmt.stringsubs.impl;
import java.io.File;
import java.io.InputStream;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.logging.Level;
import java.util.logging.Logger;
import org.glassfish.api.logging.LogHelper;
import com.sun.enterprise.admin.servermgmt.SLogger;
import com.sun.enterprise.admin.servermgmt.stringsubs.AttributePreprocessor;
import com.sun.enterprise.admin.servermgmt.stringsubs.StringSubstitutionException;
import com.sun.enterprise.admin.servermgmt.stringsubs.StringSubstitutor;
import com.sun.enterprise.admin.servermgmt.stringsubs.Substitutable;
import com.sun.enterprise.admin.servermgmt.stringsubs.SubstitutableFactory;
import com.sun.enterprise.admin.servermgmt.stringsubs.SubstitutionAlgorithm;
import com.sun.enterprise.admin.servermgmt.xml.stringsubs.Archive;
import com.sun.enterprise.admin.servermgmt.xml.stringsubs.ChangePair;
import com.sun.enterprise.admin.servermgmt.xml.stringsubs.ChangePairRef;
import com.sun.enterprise.admin.servermgmt.xml.stringsubs.Component;
import com.sun.enterprise.admin.servermgmt.xml.stringsubs.Defaults;
import com.sun.enterprise.admin.servermgmt.xml.stringsubs.FileEntry;
import com.sun.enterprise.admin.servermgmt.xml.stringsubs.Group;
import com.sun.enterprise.admin.servermgmt.xml.stringsubs.GroupRef;
import com.sun.enterprise.admin.servermgmt.xml.stringsubs.ModeType;
import com.sun.enterprise.admin.servermgmt.xml.stringsubs.Property;
import com.sun.enterprise.admin.servermgmt.xml.stringsubs.PropertyType;
import com.sun.enterprise.admin.servermgmt.xml.stringsubs.StringsubsDefinition;
import com.sun.enterprise.universal.i18n.LocalStringsImpl;
/**
* A class to encapsulate string-subs definition. Parse, validate and performs
* String substitution for the given string-subs.xml.
*/
public class StringSubstitutionEngine implements StringSubstitutor {
private static final Logger LOGGER = SLogger.getLogger();
private static final LocalStringsImpl STRINGS = new LocalStringsImpl(StringSubstitutionEngine.class);
private InputStream configInputStream = null;
//Root of JAXB parsed string-subs configuration
private StringsubsDefinition root = null;
private Map changePairsMap = null;
Map defaultProperties = null;
private SubstitutableFactory substitutableFactory = new SubstituableFactoryImpl();
private AttributePreprocessor attrPreprocessor = new AttributePreprocessorImpl();
/**
* Constructs {@link StringSubstitutionEngine} based on the given string-subs
* configuration stream. Engine parse and validate the configuration and build
* the internal representation to perform string substitution.
*
* @param inputStream The string-subs configuration stream.
* @throws StringSubstitutionException If any error occurs in engine initialization.
*/
public StringSubstitutionEngine(InputStream inputStream) throws StringSubstitutionException {
if (inputStream == null) {
throw new StringSubstitutionException("InputStream is null");
}
configInputStream = inputStream;
root = StringSubstitutionParser.parse(configInputStream);
}
@Override
public void setAttributePreprocessor(AttributePreprocessor attributePreprocessor) {
attrPreprocessor = attributePreprocessor;
}
@Override
public void setEntryFactory(SubstitutableFactory factory) {
substitutableFactory = factory;
}
@Override
public void setFileBackupLocation(File backupLocation) {
// TODO Auto-generated method stub
}
@Override
public List getDefaultProperties(PropertyType type) {
Defaults defaults = root.getDefaults();
if (defaults == null) {
return Collections.emptyList();
}
if (type == null) {
return defaults.getProperty();
}
List props = new ArrayList();
for (Property prop : defaults.getProperty()) {
if (prop.getType().equals(type)) {
props.add(prop);
}
}
return props;
}
@Override
public void substituteAll() throws StringSubstitutionException {
for (Component component : root.getComponent()) {
doSubstitution(component);
}
}
@Override
public void substituteComponents(List components)
throws StringSubstitutionException {
if (!isValid(components)) {
throw new StringSubstitutionException(STRINGS.get("missingComponentIdentifiers"));
}
for (String componentId : components) {
Component component = findComponentById(componentId);
if (component == null) {
LOGGER.log(Level.INFO, SLogger.MISSING_COMPONENT, componentId);
continue;
}
doSubstitution(component);
}
}
@Override
public void substituteGroups(List groups)
throws StringSubstitutionException {
if (!isValid(groups)) {
throw new StringSubstitutionException(STRINGS.get("missingGroupIdentifiers"));
}
for (String groupId : groups) {
Group group = findGroupById(groupId);
if (group == null) {
LOGGER.log(Level.WARNING, SLogger.MISSING_GROUP, groupId);
continue;
}
doSubstitution(group);
}
}
@Override
public StringsubsDefinition getStringSubsDefinition() {
return root;
}
/**
* Perform's string substitution for a given component.
*
* @param component {@link Component} for which the string substitution
* has to be performed.
* @throws StringSubstitutionException If any error occurs during
* substitution.
*/
private void doSubstitution(Component component)
throws StringSubstitutionException {
List extends GroupRef> refList = component.getGroupRef();
for (GroupRef ref : refList) {
doSubstitution(findGroupById(ref.getName()));
}
}
/**
* Perform's string substitution for a given group.
*
* @param groups Groups for which the string substitution
* has to be performed.
* @throws StringSubstitutionException If any error occurs during
* substitution.
*/
private void doSubstitution(Group group)
throws StringSubstitutionException {
List extends FileEntry> fileList = group.getFileEntry();
List extends Archive> archiveList = group.getArchive();
if (!isValid(fileList) && !isValid(archiveList)) {
if (LOGGER.isLoggable(Level.FINER)) {
LOGGER.log(Level.FINER, STRINGS.get("noSubstitutableGroupEntry", group.getId()));
}
return;
}
List extends ChangePairRef> refList = group.getChangePairRef();
if (!isValid(refList)) {
if (LOGGER.isLoggable(Level.FINE)) {
LOGGER.log(Level.FINE, STRINGS.get("noChangePairForGroup", group.getId()));
}
return;
}
String groupMode = null;
ModeType modeType = group.getMode();
if (modeType != null) {
groupMode = modeType.value();
}
buildChangePairsMap();
Map substitutionMap = new HashMap();
for (ChangePairRef ref : refList) {
String name = ref.getName();
String localMode = ref.getMode();
// if mode is not specified for this change-pair-ref
// then inherit the mode of the group
if (localMode == null || localMode.length() == 0) {
localMode = groupMode;
}
Pair pair = changePairsMap.get(name);
if (pair == null) {
LOGGER.log(Level.INFO, SLogger.MISSING_CHANGE_PAIR, new Object[] {name, group.getId()});
continue;
}
String beforeString = pair.getBefore();
String afterString = pair.getAfter();
if (localMode == null || localMode.length() == 0) {
if (LOGGER.isLoggable(Level.FINEST)) {
LOGGER.log(Level.FINEST, STRINGS.get("noModeValue", group.getId()));
}
}
else {
try {
afterString = ModeProcessor.processModeType(ModeType.fromValue(localMode), afterString);
} catch (Exception e) {
LOGGER.log(Level.WARNING, SLogger.INVALID_MODE_TYPE, localMode);
}
}
substitutionMap.put(beforeString, afterString);
}
SubstitutionAlgorithm algorithm = new SubstitutionAlgorithmFactory().getAlgorithm(substitutionMap);
for (FileEntry fileEntry : fileList) {
fileEntry.setName(attrPreprocessor.substitutePath(fileEntry.getName()));
List extends Substitutable> substituables = substitutableFactory.getFileEntrySubstituables(fileEntry);
for (Substitutable substituable : substituables) {
algorithm.substitute(substituable);
substituable.finish();
}
}
for (Archive archive : archiveList) {
if (archive == null || archive.getName().isEmpty()) {
continue;
}
try {
archive.setName(attrPreprocessor.substitutePath(archive.getName()));
List extends Substitutable> substituables = substitutableFactory.getArchiveEntrySubstitutable(archive);
if (!isValid(substituables)) {
continue;
}
for (Substitutable substituable : substituables) {
algorithm.substitute(substituable);
substituable.finish();
}
} catch (Exception e) {
LogHelper.log(LOGGER, Level.WARNING, SLogger.ERR_ARCHIVE_SUBSTITUTION, e, archive.getName());
}
}
}
/**
* Build's a HashMap containing an entry for each in the string-subs
* configuration file. The HashMap is created so that elements do not
* need to be re-analyzed each time they're referenced.
*
*/
private void buildChangePairsMap() {
if (changePairsMap == null || changePairsMap.isEmpty()) {
Defaults defaults = root.getDefaults();
if (defaults != null) {
List properties = defaults.getProperty();
if (!properties.isEmpty()) {
defaultProperties = new HashMap(properties.size(), 1);
for (Property prop : properties) {
defaultProperties.put(prop.getKey(), prop);
}
}
}
List extends ChangePair> changePairList = root.getChangePair();
changePairsMap = new HashMap(changePairList.size());
for (ChangePair pair : root.getChangePair()) {
String id = pair.getId();
String beforeValue = pair.getBefore();
String afterValue = pair.getAfter();
if (id == null || beforeValue == null || afterValue == null) {
LOGGER.log(Level.INFO, SLogger.EMPTY_CHANGE_PAIR);
continue;
}
beforeValue = attrPreprocessor.substituteBefore(beforeValue);
afterValue = attrPreprocessor.substituteAfter(afterValue);
changePairsMap.put(id, new Pair(beforeValue, afterValue));
}
}
}
/**
* Find {@link Group} by the given id. Returns null
if no
* group found.
*
* @param id Identifier for a group.
* @return Matched Group.
*/
private Group findGroupById(String id) {
if (id == null) {
return null;
}
List extends Group> groupList = root.getGroup();
if (!isValid(groupList)) {
return null;
}
for (Group group : groupList) {
if (id.equals(group.getId())) {
return group;
}
}
return null;
}
/**
* Find {@link Component} by the given id. Returns null
if no
* component found.
*
* @param id Identifier for a component.
* @return Matched component.
*/
private Component findComponentById(String id) {
if (id == null) {
return null;
}
List extends Component> components = root.getComponent();
if (!isValid(components)) {
return null;
}
for (Component component : components) {
if (id.equals(component.getId())) {
return component;
}
}
return null;
}
/**
* Check's if the give {@link Collection} is valid. A non null and non empty Collection is
* termed as valid Collection.
*
* @param collection Collection to validate
* @return true
for valid Collection and
* false
for invalid Collection.
*/
private boolean isValid(Collection extends Object> collection) {
return collection != null && !collection.isEmpty();
}
/**
* A class to store the before, after tuple.
* Use to store before and after value of change-pair.
*/
private static class Pair {
String before;
String after;
Pair (String before, String after) {
this.before = before;
this.after = after;
}
public String getBefore() {
return before;
}
public String getAfter() {
return after;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy