blob: 38cdb4a781e55f599f014244ef249d9f2bd10fc5 [file] [log] [blame]
/**
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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.apache.maven.mercury.artifact;
import java.io.IOException;
import java.io.StringWriter;
import java.io.Writer;
import java.util.ArrayList;
import java.util.List;
import java.util.TreeSet;
import org.codehaus.plexus.lang.DefaultLanguage;
import org.codehaus.plexus.lang.Language;
/**
* metadata [dirty] Tree
*
* @author <a href="oleg@codehaus.org">Oleg Gusakov</a>
*/
public class MetadataTreeNode
{
/** prevailing # of queries (dependencies) in the dirty tree node */
private static final int DEFAULT_QUERY_COUNT = 8;
/** prevailing # of children in the dirty tree node. If all queries are ranges - 2 hits per range */
private static final int DEFAULT_CHILDREN_COUNT = 16;
private static final Language LANG = new DefaultLanguage( MetadataTreeNode.class );
/**
* this node's artifact MD
*/
ArtifactMetadata md;
/**
* fail resolution if it could not be found?
*/
boolean optional = false;
/**
* is there a real artifact behind this node, or it's just a helper ?
*/
boolean real = true;
/**
* parent node
*/
MetadataTreeNode parent;
/**
* node unique id, used to identify this node in external tree manipulations, such as
*/
int id;
/**
* query node - the one that originated this actual node
*/
ArtifactMetadata query;
/**
* queries - one per POM dependency
*/
List<ArtifactMetadata> queries;
/**
* actual found versions
*/
List<MetadataTreeNode> children;
/** unique name of this node. Used in SAT solver */
String name;
// ------------------------------------------------------------------------
public int countNodes()
{
return countNodes( this );
}
// ------------------------------------------------------------------------
public static int countNodes( MetadataTreeNode node )
{
int res = 1;
if ( node.children != null && node.children.size() > 0 )
{
for ( MetadataTreeNode child : node.children )
{
res += countNodes( child );
}
}
return res;
}
// ------------------------------------------------------------------------
public int countDistinctNodes()
{
TreeSet<String> nodes = new TreeSet<String>();
getDistinctNodes( this, nodes );
return nodes.size();
}
// ------------------------------------------------------------------------
public static void getDistinctNodes( MetadataTreeNode node, TreeSet<String> nodes )
{
if ( node.getMd() == null )
throw new IllegalArgumentException( "tree node without metadata" );
nodes.add( node.getMd().getGAV() );
if ( node.children != null && node.children.size() > 0 )
for ( MetadataTreeNode child : node.children )
getDistinctNodes( child, nodes );
}
// ------------------------------------------------------------------------
public MetadataTreeNode()
{
}
// ------------------------------------------------------------------------
/**
* pointers to parent and query are a must.
*/
public MetadataTreeNode( ArtifactMetadata md, MetadataTreeNode parent, ArtifactMetadata query )
{
if ( md != null )
{
md.setArtifactScope( ArtifactScopeEnum.checkScope( md.getArtifactScope() ) );
}
this.md = md;
this.parent = parent;
this.query = query;
}
// ------------------------------------------------------------------------
/**
* dependencies are ordered in the POM - they should be added in the POM order
*/
public MetadataTreeNode addChild( MetadataTreeNode kid )
{
if ( kid == null )
{
return this;
}
if ( children == null )
{
children = new ArrayList<MetadataTreeNode>( DEFAULT_CHILDREN_COUNT );
}
kid.setParent( this );
children.add( kid );
return this;
}
// ------------------------------------------------------------------------
/**
* dependencies are ordered in the POM - they should be added in the POM order
*/
public MetadataTreeNode addQuery( ArtifactMetadata query )
{
if ( query == null )
{
return this;
}
if ( queries == null )
{
queries = new ArrayList<ArtifactMetadata>( DEFAULT_QUERY_COUNT );
}
queries.add( query );
return this;
}
// ------------------------------------------------------------------
@Override
public String toString()
{
return md == null ? "no metadata, parent " + ( parent == null ? "null" : parent.toString() ) : md.toString()
+ ":d=" + getDepth();
}
// ------------------------------------------------------------------------
public boolean hasChildren()
{
return children != null;
}
// ------------------------------------------------------------------------
public ArtifactMetadata getMd()
{
return md;
}
public MetadataTreeNode getParent()
{
return parent;
}
public int getDepth()
{
int depth = 0;
for ( MetadataTreeNode p = parent; p != null; p = p.parent )
++depth;
return depth;
}
public int getMaxDepth( int depth )
{
int res = 0;
if ( !hasChildren() )
return depth + 1;
for ( MetadataTreeNode kid : children )
{
int kidDepth = kid.getMaxDepth( depth + 1 );
if ( kidDepth > res )
res = kidDepth;
}
return res;
}
public void setParent( MetadataTreeNode parent )
{
this.parent = parent;
}
public List<MetadataTreeNode> getChildren()
{
return children;
}
public boolean isOptional()
{
return optional;
}
public boolean isReal()
{
return real;
}
public void setReal( boolean real )
{
this.real = real;
}
public ArtifactMetadata getQuery()
{
return query;
}
public List<ArtifactMetadata> getQueries()
{
return queries;
}
// ------------------------------------------------------------------------
public static final MetadataTreeNode deepCopy( MetadataTreeNode node )
{
MetadataTreeNode res = new MetadataTreeNode( node.getMd(), node.getParent(), node.getQuery() );
res.setId( node.getId() );
if ( node.hasChildren() )
for ( MetadataTreeNode kid : node.children )
{
MetadataTreeNode deepKid = deepCopy( kid );
res.addChild( deepKid );
}
return res;
}
// ----------------------------------------------------------------
/**
* helper method to print the tree into a Writer
*/
public static final void showNode( MetadataTreeNode n, int level, Writer wr )
throws IOException
{
if ( n == null )
{
wr.write( "null node" );
return;
}
for ( int i = 0; i < level; i++ )
wr.write( " " );
wr.write( level + " " + n.getMd() + "\n" );
if ( n.hasChildren() )
{
for ( MetadataTreeNode kid : n.getChildren() )
showNode( kid, level + 1, wr );
}
}
// ----------------------------------------------------------------
/**
* helper method to print the tree into sysout
*/
public static final void showNode( MetadataTreeNode n, int level )
throws IOException
{
StringWriter sw = new StringWriter();
MetadataTreeNode.showNode( n, 0, sw );
System.out.println( sw.toString() );
}
// ------------------------------------------------------------------------
public int getId()
{
return id;
}
public void setId( int id )
{
this.id = id;
}
// ------------------------------------------------------------------------
public static void reNumber( MetadataTreeNode node, int startNum )
{
reNum( node, new Counter( startNum ) );
}
// ------------------------------------------------------------------------
private static void reNum( MetadataTreeNode node, Counter num )
{
node.setId( num.next() );
if ( node.hasChildren() )
for ( MetadataTreeNode kid : node.getChildren() )
reNum( kid, num );
}
public String getName()
{
return name;
}
public void createNames( int level, int seq )
{
name = md.toScopedString() + ":" + level + "." + seq;
if ( hasChildren() )
{
int no = 0;
for ( MetadataTreeNode kid : children )
kid.createNames( level + 1, no++ );
}
}
// ------------------------------------------------------------------------
// ------------------------------------------------------------------------
}
// ------------------------------------------------------------------------
class Counter
{
int n;
public Counter( int n )
{
this.n = n;
}
int next()
{
return n++;
}
}