All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.eviware.soapui.security.scan.MalformedXmlSecurityScan Maven / Gradle / Ivy

The newest version!
/*
 *  soapUI, copyright (C) 2004-2011 smartbear.com 
 *
 *  soapUI is free software; you can redistribute it and/or modify it under the 
 *  terms of version 2.1 of the GNU Lesser General Public License as published by 
 *  the Free Software Foundation.
 *
 *  soapUI is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without 
 *  even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. 
 *  See the GNU Lesser General Public License for more details at gnu.org.
 */
package com.eviware.soapui.security.scan;

import java.io.IOException;
import java.util.ArrayList;
import java.util.Collection;
import java.util.HashMap;
import java.util.Map;

import javax.swing.JComponent;

import org.apache.xmlbeans.XmlException;
import org.apache.xmlbeans.XmlOptions;

import com.eviware.soapui.SoapUI;
import com.eviware.soapui.config.MalformedXmlAttributeConfig;
import com.eviware.soapui.config.MalformedXmlConfig;
import com.eviware.soapui.config.SecurityScanConfig;
import com.eviware.soapui.config.StrategyTypeConfig;
import com.eviware.soapui.model.ModelItem;
import com.eviware.soapui.model.iface.MessageExchange;
import com.eviware.soapui.model.security.SecurityCheckedParameter;
import com.eviware.soapui.model.testsuite.TestCaseRunner;
import com.eviware.soapui.model.testsuite.TestProperty;
import com.eviware.soapui.model.testsuite.TestStep;
import com.eviware.soapui.security.SecurityTestRunContext;
import com.eviware.soapui.security.SecurityTestRunner;
import com.eviware.soapui.security.ui.MalformedXmlAdvancedSettingsPanel;
import com.eviware.soapui.support.types.StringToStringMap;
import com.eviware.soapui.support.xml.XmlObjectTreeModel;
import com.eviware.soapui.support.xml.XmlObjectTreeModel.AttributeXmlTreeNode;
import com.eviware.soapui.support.xml.XmlObjectTreeModel.XmlTreeNode;
import com.eviware.soapui.support.xml.XmlUtils;

public class MalformedXmlSecurityScan extends AbstractSecurityScanWithProperties
{

	public static final String TYPE = "MalformedXmlSecurityScan";
	public static final String NAME = "Malformed XML";
	private Map> parameterMutations = new HashMap>();
	private boolean mutation;
	private MalformedXmlConfig malformedXmlConfig;
	private MalformedXmlAttributeConfig malformedAttributeConfig;
	private MalformedXmlAdvancedSettingsPanel advancedSettingsPanel;

	public MalformedXmlSecurityScan( TestStep testStep, SecurityScanConfig config, ModelItem parent, String icon )
	{
		super( testStep, config, parent, icon );
		if( config.getConfig() == null || !( config.getConfig() instanceof MalformedXmlConfig ) )
			initMalformedXmlConfig();
		else
		{
			malformedXmlConfig = ( ( MalformedXmlConfig )config.getConfig() );
			malformedAttributeConfig = malformedXmlConfig.getAttributeMutation();
		}
	}

	/**
	 * Default malformed xml configuration
	 */
	protected void initMalformedXmlConfig()
	{
		getConfig().setConfig( MalformedXmlConfig.Factory.newInstance() );
		malformedXmlConfig = ( MalformedXmlConfig )getConfig().getConfig();

		malformedXmlConfig.addNewAttributeMutation();

		// init default configuration
		malformedXmlConfig.setInsertNewElement( true );
		malformedXmlConfig.setNewElementValue( "xml   " );
		malformedXmlConfig.setChangeTagName( true );
		malformedXmlConfig.setLeaveTagOpen( true );
		malformedXmlConfig.setInsertInvalidCharacter( true );

		malformedAttributeConfig = malformedXmlConfig.getAttributeMutation();
		malformedAttributeConfig.setMutateAttributes( true );
		malformedAttributeConfig.setInsertInvalidChars( true );
		malformedAttributeConfig.setLeaveAttributeOpen( true );
		malformedAttributeConfig.setAddNewAttribute( true );
		malformedAttributeConfig.setNewAttributeName( "newAttribute" );
		malformedAttributeConfig.setNewAttributeValue( "XXX" );
	}

	@Override
	protected void execute( SecurityTestRunner runner, TestStep testStep, SecurityTestRunContext context )
	{
		try
		{
			StringToStringMap paramsUpdated = update( testStep, context );
			MessageExchange message = ( MessageExchange )testStep.run( ( TestCaseRunner )runner, context );
			createMessageExchange( paramsUpdated, message, context );
		}
		catch( XmlException e )
		{
			SoapUI.logError( e, "[MalformedXmlSecurityScan]XPath seems to be invalid!" );
			reportSecurityScanException( "Property value is not XML or XPath is wrong!" );
		}
		catch( Exception e )
		{
			SoapUI.logError( e, "[MalformedXmlSecurityScan]Property value is not valid xml!" );
			reportSecurityScanException( "Property value is not XML or XPath is wrong!" );
		}
	}

	protected StringToStringMap update( TestStep testStep, SecurityTestRunContext context ) throws XmlException,
			Exception
	{
		StringToStringMap params = new StringToStringMap();

		if( parameterMutations.size() == 0 )
			mutateParameters( testStep, context );

		if( getExecutionStrategy().getStrategy() == StrategyTypeConfig.ONE_BY_ONE )
		{
			/*
			 * Idea is to drain for each parameter mutations.
			 */
			for( SecurityCheckedParameter param : getParameterHolder().getParameterList() )
			{
				if( parameterMutations.containsKey( param ) )
					if( parameterMutations.get( param ).size() > 0 )
					{
						TestProperty property = testStep.getProperties().get( param.getName() );
						String value = context.expand( property.getValue() );
						if( param.getXpath() == null || param.getXpath().trim().length() == 0 )
						{
							// no xpath ignore
						}
						else
						{
							// no value, do nothing.
							if( value == null || value.trim().equals( "" ) )
								continue;
							// XmlObjectTreeModel model = new XmlObjectTreeModel(
							// property.getSchemaType().getTypeSystem(),
							// XmlObject.Factory.parse( value ) );
							XmlObjectTreeModel model = new XmlObjectTreeModel( property.getSchemaType().getTypeSystem(),
									XmlUtils.createXmlObject( value ) );
							XmlTreeNode[] nodes = model.selectTreeNodes( context.expand( param.getXpath() ) );
							StringBuffer buffer = new StringBuffer( value );
							for( int cnt = 0; cnt < nodes.length; cnt++ )
							{
								// find right node
								// this finds where node that needs updateing begins
								int start = value.indexOf( "<" + nodes[cnt].getNodeName() ); // keeps
								// node
								// start
								int cnt2 = 0;
								// if have more than one node that matches xpath, find
								// next one.
								while( cnt2 < cnt )
								{
									start = value.indexOf( "<" + nodes[cnt].getNodeName(), start + 1 );
									cnt2++ ;
								}
								// get node xml
								String nodeXml = getXmlForNode( nodes[cnt] );
								// find end of target xml node
								int end = value.indexOf( "<" + nodes[cnt].getNodeName(), start + 1 );
								if( end <= 0 )
								{
									if( nodeXml.endsWith( "" ) )
									{
										end = value.indexOf( "" )
												+ ( "" ).length();
									}
									else
									{
										end = value.indexOf( ">", value.indexOf( "/", start ) );
									}
								}
								if( end <= 0 || end <= start )
									break;
								// replace node with right value
								buffer.replace( start, end + 1, parameterMutations.get( param ).get( 0 ) );
							}
							params.put( param.getLabel(), parameterMutations.get( param ).get( 0 ) );
							parameterMutations.get( param ).remove( 0 );

							testStep.getProperties().get( param.getName() ).setValue( buffer.toString() );
						}

						break;
					}
			}
		}
		else
		{
			for( TestProperty property : testStep.getPropertyList() )
			{

				String value = context.expand( property.getValue() );
				if( XmlUtils.seemsToBeXml( value ) )
				{
					StringBuffer buffer = new StringBuffer( value );
					XmlObjectTreeModel model = null;
					// model = new XmlObjectTreeModel(
					// property.getSchemaType().getTypeSystem(),
					// XmlObject.Factory.parse( value ) );
					model = new XmlObjectTreeModel( property.getSchemaType().getTypeSystem(),
							XmlUtils.createXmlObject( value ) );
					for( SecurityCheckedParameter param : getParameterHolder().getParameterList() )
					{
						if( param.getXpath() == null || param.getXpath().trim().length() == 0 )
						{
							if( parameterMutations.containsKey( param ) )
							{
								testStep.getProperties().get( param.getName() )
										.setValue( parameterMutations.get( param ).get( 0 ) );
								params.put( param.getLabel(), parameterMutations.get( param ).get( 0 ) );
								parameterMutations.get( param ).remove( 0 );
							}
						}
						else
						{
							// no value, do nothing.
							if( value == null || value.trim().equals( "" ) )
								continue;
							if( param.getName().equals( property.getName() ) )
							{
								XmlTreeNode[] nodes = model.selectTreeNodes( context.expand( param.getXpath() ) );
								if( parameterMutations.containsKey( param ) )
									if( parameterMutations.get( param ).size() > 0 )
									{
										for( int cnt = 0; cnt < nodes.length; cnt++ )
										{
											// find right node
											// keeps node start
											int start = value.indexOf( "<" + nodes[cnt].getNodeName() );
											int cnt2 = 0;
											while( cnt2 < cnt )
											{
												start = value.indexOf( "<" + nodes[cnt].getNodeName(), start + 1 );
												cnt2++ ;
											}
											String nodeXml = getXmlForNode( nodes[cnt] );
											int end = value.indexOf( "<" + nodes[cnt].getNodeName(), start + 1 );
											if( end <= 0 )
											{
												if( nodeXml.endsWith( "" ) )
												{
													end = value.indexOf( "" );
												}
												else
												{
													end = value.indexOf( ">", value.indexOf( "/", start ) );
												}
											}
											if( end <= 0 || end <= start )
												break;
											buffer.replace( start, end + 1, parameterMutations.get( param ).get( 0 ) );
										}
										params.put( param.getLabel(), parameterMutations.get( param ).get( 0 ) );
										parameterMutations.get( param ).remove( 0 );
									}
							}
						}
					}
					if( model != null )
					{
						property.setValue( buffer.toString() );
						model.release();
					}
				}

			}
		}
		return params;
	}

	protected void mutateParameters( TestStep testStep, SecurityTestRunContext context ) throws XmlException,
			IOException
	{
		mutation = true;
		// for each parameter
		for( SecurityCheckedParameter parameter : getParameterHolder().getParameterList() )
		{
			if( parameter.isChecked() )
			{
				TestProperty property = getTestStep().getProperties().get( parameter.getName() );
				// check parameter does not have any xpath
				if( parameter.getXpath() == null || parameter.getXpath().trim().length() == 0 )
				{
					/*
					 * parameter xpath is not set ignore than ignore this parameter
					 */
				}
				else
				{
					// we have xpath but do we have xml which need to mutate
					// ignore if there is no value, since than we'll get exception
					if( !( property.getValue() == null && property.getDefaultValue() == null ) )
					{
						// get value of that property
						String value = context.expand( property.getValue() );

						// we have something that looks like xpath, or hope so.

						// XmlObjectTreeModel model = new XmlObjectTreeModel(
						// property.getSchemaType().getTypeSystem(),
						// XmlObject.Factory.parse( value ) );
						XmlObjectTreeModel model = new XmlObjectTreeModel( property.getSchemaType().getTypeSystem(),
								XmlUtils.createXmlObject( value ) );
						XmlTreeNode[] nodes = model.selectTreeNodes( context.expand( parameter.getXpath() ) );

						if( nodes.length > 0 && !( nodes[0] instanceof AttributeXmlTreeNode ) )
						{
							if( !parameterMutations.containsKey( parameter ) )
								parameterMutations.put( parameter, new ArrayList() );
							parameterMutations.get( parameter ).addAll( mutateNode( nodes[0], value ) );
						}

						model.release();
					}
				}
			}
		}
	}

	protected Collection mutateNode( XmlTreeNode node, String xml ) throws IOException
	{

		ArrayList result = new ArrayList();
		String nodeXml = getXmlForNode( node );
		// insert new element
		if( malformedXmlConfig.getInsertNewElement() )
		{
			StringBuffer buffer = new StringBuffer( nodeXml );
			if( nodeXml.endsWith( "" ) )
			{
				buffer.insert( nodeXml.indexOf( ">" ) + 1, malformedXmlConfig.getNewElementValue() );
			}
			else
			{
				buffer.delete( nodeXml.lastIndexOf( "/" ), nodeXml.length() );
				buffer.append( ">" + malformedXmlConfig.getNewElementValue() + "" );
			}
			result.add( buffer.toString() );
		}
		// change name
		if( malformedXmlConfig.getChangeTagName() )
		{
			String original = node.getNodeName();

			if( original.toUpperCase().equals( original ) )
			{
				result.add( nodeXml.replaceAll( original, original.toLowerCase() ) );
			}
			else if( original.toLowerCase().equals( original ) )
			{
				result.add( nodeXml.replaceAll( original, original.toUpperCase() ) );
			}
			else
			{
				StringBuffer buffer = new StringBuffer();
				// kewl
				for( char ch : original.toCharArray() )
				{
					if( Character.isUpperCase( ch ) )
						buffer.append( Character.toLowerCase( ch ) );
					else
						buffer.append( Character.toUpperCase( ch ) );
				}
				result.add( nodeXml.replaceAll( original, buffer.toString() ) );

				// add '_' before upper case and make uppercase lowercase
				// just start tag change
				buffer = new StringBuffer();
				for( char ch : original.toCharArray() )
				{
					if( Character.isUpperCase( ch ) )
						buffer.append( "_" ).append( Character.toLowerCase( ch ) );
					else
						buffer.append( ch );
				}
				result.add( nodeXml.replaceAll( original, buffer.toString() ) );
			}

		}
		// leave tag open
		if( malformedXmlConfig.getLeaveTagOpen() )
		{
			if( nodeXml.endsWith( "" ) )
			{
				// cut end tag
				StringBuffer buffer = new StringBuffer( nodeXml );
				buffer.delete( buffer.indexOf( "" ), buffer.length() );
				result.add( buffer.toString() );

				// cut start tag
				buffer = new StringBuffer( nodeXml );
				buffer.delete( 0, buffer.indexOf( ">" ) + 1 );
				result.add( buffer.toString() );

				// cut start tag and remove '/' from end tag
				buffer = new StringBuffer( nodeXml );
				buffer.delete( 0, buffer.indexOf( ">" ) + 1 );
				buffer.delete( buffer.indexOf( "" ) + 1,
						buffer.indexOf( "" ) + 2 );
				result.add( buffer.toString() );
			}
			else
			{
				// remove '/>' from end of tag
				StringBuffer buffer = new StringBuffer( nodeXml );
				buffer.delete( nodeXml.lastIndexOf( "/" ), nodeXml.length() );
				result.add( buffer.toString() );
			}
		}
		if( malformedXmlConfig.getInsertInvalidCharacter() )
		{
			for( char ch : new char[] { '<', '>', '&' } )
			{
				StringBuffer buffer = new StringBuffer( nodeXml );
				if( nodeXml.endsWith( "" ) )
				{
					buffer.insert( buffer.indexOf( "" ), ch );
				}
				else
				{
					buffer.delete( nodeXml.lastIndexOf( "/" ), nodeXml.length() );
					buffer.append( '>' ).append( ch ).append( "" );
				}
				result.add( buffer.toString() );
			}
		}

		// mutate attributes
		if( malformedAttributeConfig.getMutateAttributes() )
		{
			if( malformedAttributeConfig.getAddNewAttribute() )
			{
				if( malformedAttributeConfig.getNewAttributeName().trim().length() > 0 )
				{
					// insert new attribute just after node tag
					StringBuffer buffer = new StringBuffer( nodeXml );
					buffer.insert( node.getNodeName().length() + 1, " " + malformedAttributeConfig.getNewAttributeName()
							+ "=" + "\"" + malformedAttributeConfig.getNewAttributeValue() + "\" " );
					result.add( buffer.toString() );
				}
			}
			if( malformedAttributeConfig.getInsertInvalidChars() )
			{
				if( node.getDomNode().hasAttributes() )
				{
					for( char ch : new char[] { '"', '\'', '<', '>', '&' } )
					{
						// add it at beggining of attribute value
						StringBuffer buffer = new StringBuffer( nodeXml );
						buffer.insert( buffer.indexOf( "=" ) + 3, ch );
						result.add( buffer.toString() );
					}
				}
			}
			if( malformedAttributeConfig.getLeaveAttributeOpen() )
			{
				if( node.getDomNode().hasAttributes() )
				{
					StringBuffer buffer = new StringBuffer( nodeXml );
					buffer.delete( buffer.indexOf( "=" ) + 1, buffer.indexOf( "=" ) + 2 );
					result.add( buffer.toString() );
				}
			}
		}
		return result;
	}

	private String getXmlForNode( XmlTreeNode nodes )
	{
		XmlOptions options = new XmlOptions();
		options.setSaveOuter();
		options.setSavePrettyPrint();

		String xml = nodes.getXmlObject().xmlText( options );

		return XmlUtils.removeUnneccessaryNamespaces( xml );
	}

	@Override
	public String getConfigDescription()
	{
		return "Configures Malformed XML Security Scan";
	}

	@Override
	public String getConfigName()
	{
		return "Malformed XML Security Scan";
	}

	@Override
	public String getHelpURL()
	{
		return "http://soapui.org/Security/malformed-xml.html";
	}

	@Override
	public String getType()
	{
		return TYPE;
	}

	@Override
	protected boolean hasNext( TestStep testStep, SecurityTestRunContext context )
	{
		boolean hasNext = false;
		if( ( parameterMutations == null || parameterMutations.size() == 0 ) && !mutation )
		{
			if( getParameterHolder().getParameterList().size() > 0 )
				hasNext = true;
			else
				hasNext = false;
		}
		else
		{
			for( SecurityCheckedParameter param : parameterMutations.keySet() )
			{
				if( parameterMutations.get( param ).size() > 0 )
				{
					hasNext = true;
					break;
				}
			}
		}
		if( !hasNext )
		{
			parameterMutations.clear();
			mutation = false;
		}
		return hasNext;
	}

	@Override
	protected void clear()
	{
		parameterMutations.clear();
		mutation = false;
	}

	@Override
	public JComponent getAdvancedSettingsPanel()
	{
		if( advancedSettingsPanel == null )
			advancedSettingsPanel = new MalformedXmlAdvancedSettingsPanel( malformedXmlConfig );

		return advancedSettingsPanel.getPanel();
	}

	@Override
	public void release()
	{
		if( advancedSettingsPanel != null )
			advancedSettingsPanel.release();

		super.release();
	}

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy