/**
* Copyright 2015-2017 Linagora, Université Joseph Fourier, Floralis
*
* The present code is developed in the scope of the joint LINAGORA -
* Université Joseph Fourier - Floralis research program and is designated
* as a "Result" pursuant to the terms and conditions of the LINAGORA
* - Université Joseph Fourier - Floralis research program. Each copyright
* holder of Results enumerated here above fully & independently holds complete
* ownership of the complete Intellectual Property rights applicable to the whole
* of said Results, and may freely exploit it in any manner which does not infringe
* the moral rights of the other copyright holders.
*
* 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 net.roboconf.core.commands;
import java.util.ArrayList;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import net.roboconf.core.ErrorCode;
import net.roboconf.core.model.ParsingError;
import net.roboconf.core.model.helpers.ComponentHelpers;
/**
* @author Vincent Zurczak - Linagora
*/
public class BulkCommandInstructions extends AbstractCommandInstruction {
private static final String ALL = "\\s+instances\\s+of";
private String instancePath, componentName;
private ChangeStateInstruction changeStateInstruction;
/**
* Constructor.
* @param context
* @param changeStateInstruction
* @param line
*/
BulkCommandInstructions( Context context, String instruction, int line ) {
super( context, instruction, line );
Matcher m = getPatternForInstancePath().matcher( instruction );
if( m.matches()) {
this.syntaxicallyCorrect = true;
this.instancePath = m.group( 2 ).trim();
this.changeStateInstruction = ChangeStateInstruction.which( m.group( 1 ).trim());
} else if(( m = getPatternForComponentName().matcher( instruction )).matches()) {
this.syntaxicallyCorrect = true;
this.componentName = m.group( 2 ).trim();
this.changeStateInstruction = ChangeStateInstruction.which( m.group( 1 ).trim());
}
}
/**
* @param line a non-null string
* @return true if it matches a supported changeStateInstruction, false otherwise
*/
public static boolean isBulkInstruction( String line ) {
boolean match = false;
Pattern[] patterns = {
getPatternForInstancePath(),
getPatternForComponentName()
};
for( Pattern p : patterns ) {
Matcher m = p.matcher( line );
if( m.matches()
&& ChangeStateInstruction.which( m.group( 1 ).trim()) != null ) {
match = true;
break;
}
}
return match;
}
/**
* @return a pattern to recognize instructions that update a given instance
*/
private static Pattern getPatternForInstancePath() {
return Pattern.compile( "([^/]+)(/.*)", Pattern.CASE_INSENSITIVE );
}
/**
* @return a pattern to recognize instructions that update instances of a given component
*/
private static Pattern getPatternForComponentName() {
return Pattern.compile( "(.+)" + ALL + "(.+)", Pattern.CASE_INSENSITIVE );
}
/*
* (non-Javadoc)
* @see net.roboconf.core.commands.AbstractCommandInstruction#doValidate()
*/
@Override
public List<ParsingError> doValidate() {
List<ParsingError> result = new ArrayList<> ();
if( this.changeStateInstruction == null )
result.add( error( ErrorCode.CMD_UNRECOGNIZED_INSTRUCTION, "Instruction: " + this.changeStateInstruction ));
if( this.instancePath != null
&& ! this.context.instanceExists( this.instancePath ))
result.add( error( ErrorCode.CMD_NO_MATCHING_INSTANCE, "Instance path: " + this.instancePath ));
if( this.componentName != null
&& ComponentHelpers.findComponent( this.context.getApp(), this.componentName ) == null )
result.add( error( ErrorCode.CMD_INEXISTING_COMPONENT, "Instance path: " + this.instancePath ));
return result;
}
/*
* (non-Javadoc)
* @see net.roboconf.core.commands.AbstractCommandInstruction#updateContext()
*/
@Override
public void updateContext() {
if( this.changeStateInstruction == ChangeStateInstruction.DELETE ) {
// Find the path of the instances to remove
Set<String> pathOfInstancesToRemove = new HashSet<> ();
if( this.instancePath != null ) {
pathOfInstancesToRemove.add( this.instancePath );
} else for( Map.Entry<String,String> entry : this.context.instancePathToComponentName.entrySet()) {
if( this.componentName.equals( entry.getValue()))
pathOfInstancesToRemove.add( entry.getKey());
}
// Now, delete the instances that need to eb erased
List<String> keys = new ArrayList<>( this.context.instancePathToComponentName.keySet());
for( String path : keys ) {
for( String pathToRemove : pathOfInstancesToRemove ) {
if( path.equals( pathToRemove )
|| path.startsWith( pathToRemove + "/" ))
this.context.instancePathToComponentName.remove( path );
}
}
}
}
/**
* @return the instancePath
*/
public String getInstancePath() {
return this.instancePath;
}
/**
* @return the componentName
*/
public String getComponentName() {
return this.componentName;
}
/**
* @return the changeStateInstruction
*/
public ChangeStateInstruction getChangeStateInstruction() {
return this.changeStateInstruction;
}
/**
* @author Vincent Zurczak - Linagora
*/
public static enum ChangeStateInstruction {
DELETE,
DEPLOY_AND_START_ALL,
STOP_ALL,
UNDEPLOY_ALL;
/*
* (non-Javadoc)
* @see java.lang.Enum#toString()
*/
@Override
public String toString() {
return super.toString().replace( '_', ' ' ).toLowerCase();
}
/**
* @param s a string (can be null)
* @return null if no changeStateInstruction was recognized, or an changeStateInstruction otherwise
*/
public static ChangeStateInstruction which( String s ) {
ChangeStateInstruction result = null;
for( ChangeStateInstruction elt : ChangeStateInstruction.values()) {
if( elt.toString().equalsIgnoreCase( s )) {
result = elt;
break;
}
// "Delete all" => "delete"
if(( elt.toString() + " all" ).equalsIgnoreCase( s )) {
result = elt;
break;
}
}
return result;
}
}
}