blob: 57a4ac27059257d4aac36a5acd5295c5d2ff58d9 [file] [log] [blame]
/*
* Copyright (c) 2009, Rickard Öberg. All Rights Reserved.
*
* 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 org.qi4j.migration.assembly;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import org.qi4j.api.util.ListMap;
/**
* Set of migration rules.
*/
public class MigrationRules<T extends AbstractMigrationRule>
{
// to-version -> List of from-versions
ListMap<String, String> versionChanges = new ListMap<String, String>();
// key=fromversion->toversion value=list of rules for that transition
ListMap<String, T> rules = new ListMap<String, T>();
public void addRule( T migrationRule )
{
versionChanges.add( migrationRule.toVersion(), migrationRule.fromVersion() );
rules.add( migrationRule.fromVersion() + "->" + migrationRule.toVersion(), migrationRule );
}
public Iterable<T> rulesBetweenVersions( String fromVersion, String toVersion )
{
try
{
String ruleToVersion = findHighestToVersion( toVersion );
return migrationRules( fromVersion, ruleToVersion );
}
catch( IllegalArgumentException e )
{
return null;
}
}
private List<T> migrationRules( String fromVersion, String toVersion )
{
List<String> list = versionChanges.get( toVersion );
if( list == null )
{
return null; // No possible rules for this transition
}
for( String possibleFromVersion : list )
{
if( fromVersion.equals( possibleFromVersion ) )
{
// We found the end of the version transitions - return rules, but filter on entity type
return new ArrayList<T>( getRulesForTransition( fromVersion, toVersion ) );
}
else
{
List<T> migrationRules = migrationRules( fromVersion, possibleFromVersion );
if( migrationRules == null )
{
continue; // Wrong transition - try another one
}
// Add entity-filtered rules from this part of the version transition
migrationRules.addAll( getRulesForTransition( possibleFromVersion, toVersion ) );
return migrationRules;
}
}
return null;
}
/**
* Find highest version below the given to-version for which there are rules registered.
*
* @param toVersion The ending toVersion in the search.
*
* @return The highest version below the given version for which there are rules registered.
*/
private String findHighestToVersion( String toVersion )
{
if( versionChanges.get( toVersion ) == null )
{
List<String> toVersions = new ArrayList<String>( versionChanges.keySet() );
Collections.sort( toVersions, Collections.reverseOrder() );
for( String version : toVersions )
{
if( version.compareTo( toVersion ) <= 0 )
{
// Found version to change to
return version;
}
}
throw new IllegalArgumentException( "No version found in rules that matches the given to-version:" + toVersion );
}
else
{
return toVersion;
}
}
private List<T> getRulesForTransition( String fromVersion, String toVersion )
{
return rules.get( fromVersion + "->" + toVersion );
}
}