/**
* 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.HashMap;
import java.util.List;
import java.util.Map;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import net.roboconf.core.ErrorCode;
import net.roboconf.core.dsl.ParsingConstants;
import net.roboconf.core.model.ParsingError;
import net.roboconf.core.model.helpers.InstanceHelpers;
import net.roboconf.core.utils.Utils;
/**
* @author Vincent Zurczak - Linagora
*/
public class ReplicateCommandInstruction extends AbstractCommandInstruction {
static final String PREFIX = "replicate";
private String newInstanceName, replicatedInstancePath;
/**
* Constructor.
* @param context
* @param instruction
* @param line
*/
ReplicateCommandInstruction( Context context, String instruction, int line ) {
super( context, instruction, line );
Pattern p = Pattern.compile( PREFIX + "\\s+(/.*)\\s*as\\b(.*)", Pattern.CASE_INSENSITIVE );
Matcher m = p.matcher( instruction );
if( m.matches()) {
this.syntaxicallyCorrect = true;
this.newInstanceName = m.group( 2 ).trim();
this.replicatedInstancePath = m.group( 1 ).trim();
}
}
/*
* (non-Javadoc)
* @see net.roboconf.core.commands.AbstractCommandInstruction#doValidate()
*/
@Override
public List<ParsingError> doValidate() {
List<ParsingError> result = new ArrayList<ParsingError> ();
if( Utils.isEmptyOrWhitespaces( this.newInstanceName ))
result.add( error( ErrorCode.CMD_MISSING_INSTANCE_NAME ));
else if( ! this.newInstanceName.matches( ParsingConstants.PATTERN_FLEX_ID ))
result.add( error( ErrorCode.CMD_INVALID_INSTANCE_NAME ));
if( ! this.context.instanceExists( this.replicatedInstancePath ))
result.add( error( ErrorCode.CMD_NO_MATCHING_INSTANCE, "Instance path: " + this.replicatedInstancePath ));
else if( InstanceHelpers.countInstances( this.replicatedInstancePath ) > 1 )
result.add( error( ErrorCode.CMD_NOT_A_ROOT_INSTANCE, "Only root instances can be replicated." ));
return result;
}
/*
* (non-Javadoc)
* @see net.roboconf.core.commands.AbstractCommandInstruction#updateContext()
*/
@Override
public void updateContext() {
// Find the path to add
String copyInstancePath = "/" + this.replicatedInstancePath.replaceFirst( "/[^/]+$", this.newInstanceName );
Map<String,String> pathToAdd = new HashMap<> ();
for( Map.Entry<String,String> entry : this.context.instancePathToComponentName.entrySet()) {
if( ! entry.getKey().startsWith( this.replicatedInstancePath + "/" )
&& ! entry.getKey().equals( this.replicatedInstancePath ))
continue;
String updatedPath = entry.getKey().replace( this.replicatedInstancePath, copyInstancePath );
pathToAdd.put( updatedPath, entry.getValue());
}
// Update them
this.context.instancePathToComponentName.putAll( pathToAdd );
}
/**
* @return the newInstanceName
*/
public String getNewInstanceName() {
return this.newInstanceName;
}
/**
* @return the replicatedInstancePath
*/
public String getReplicatedInstancePath() {
return this.replicatedInstancePath;
}
}