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

analysis.NFAAnalyserMerging Maven / Gradle / Ivy

Go to download

A tool to perform static analysis on regexes to determine whether they are vulnerable to ReDoS.

There is a newer version: 1.0.8
Show newest version
package analysis;

import java.util.HashMap;
import java.util.LinkedList;
import java.util.Map;

import analysis.EdaAnalysisResults.EdaCases;
import analysis.IdaAnalysisResults.IdaCases;
import analysis.NFAAnalyserInterface.EdaAnalysisResultsNoEda;
import analysis.NFAAnalyserInterface.IdaAnalysisResultsNoIda;

import analysis.AnalysisSettings.PriorityRemovalStrategy;

import nfa.NFAGraph;
import nfa.NFAEdge;
import nfa.NFAVertexND;

/**
 * An analyser for finding patterns that can cause catastrophic backtracking.
 * 
 * @author N. H. Weideman
 *
 */
public class NFAAnalyserMerging extends NFAAnalyser {
	
	public NFAAnalyserMerging(PriorityRemovalStrategy priorityRemovalStrategy) {
		super(priorityRemovalStrategy);
	}

	private EdaAnalysisResults testCaseESCC(NFAGraph originalM, LinkedList sccsInOriginal, Map esccs) throws InterruptedException {

		/* mapping SCCs to the ESCC's in them */
		HashMap> sccToMergedESCCStatesMap = new HashMap>();
		for (NFAGraph currentSCCInOriginal : sccsInOriginal) {
			for (NFAVertexND currentVertex : currentSCCInOriginal.vertexSet()) {
				/* if a vertex in this SCC becomes a merged ESCC */
				if (esccs.containsKey(currentVertex)) {
					LinkedList newList;
					if (sccToMergedESCCStatesMap.containsKey(currentVertex)) {
						newList = sccToMergedESCCStatesMap.get(currentVertex);
					} else {
						newList = new LinkedList();
					}
					//LinkedList newList = sccToMergedESCCStatesMap.getOrDefault(currentVertex, new LinkedList());
					newList.add(currentVertex);
					sccToMergedESCCStatesMap.put(currentSCCInOriginal, newList);
				}
			}
		}
		for (NFAGraph currentSCCInOriginal : sccsInOriginal) {
			if (isInterrupted()) {
				throw new InterruptedException();
			}
			/*
			 * find esccs in this scc (in merged). The OrDefault lets it return
			 * an empty list to save a null check
			 */
			LinkedList esscsInCurrentSCC;
			if (sccToMergedESCCStatesMap.containsKey(currentSCCInOriginal)) {
				esscsInCurrentSCC = sccToMergedESCCStatesMap.get(currentSCCInOriginal);
			} else {
				esscsInCurrentSCC = new LinkedList();
			}
			//LinkedList esscsInCurrentSCC = sccToMergedESCCStatesMap.getOrDefault(currentSCCInOriginal, new LinkedList());

			for (NFAVertexND mergedESCCInSCC : esscsInCurrentSCC) {
				if (isInterrupted()) {
					throw new InterruptedException();
				}
				NFAGraph currentESCC = esccs.get(mergedESCCInSCC);

				/*
				 * find all the entrance and exit states of the escc and map
				 * them to their entrance, exit edges
				 */
				HashMap entranceStates = new HashMap();
				HashMap exitStates = new HashMap();

				for (NFAVertexND v : currentESCC.vertexSet()) {
					if (isInterrupted()) {
						throw new InterruptedException();
					}
					/*
					 * we defined an entrance/exit edge to still be in the scc
					 * of the escc
					 */

					for (NFAEdge e : currentSCCInOriginal.incomingEdgesOf(v)) {
						/*
						 * if the edge comes from a vertex not in the escc (but
						 * in the scc), or is a non-epsilon transition.
						 */
						if (!currentESCC.containsEdge(e)) {
							// System.out.println(e.getSourceVertex() + "->" +
							// e.getTargetVertex());
							if (!entranceStates.containsKey(v)) {
								entranceStates.put(v, e);
							}
						}
					}
					for (NFAEdge e : currentSCCInOriginal.outgoingEdgesOf(v)) {
						/*
						 * if the edge goes to a vertex not in the escc (but in
						 * the scc), or is a non-epsilon transition.
						 */
						if (!currentESCC.containsEdge(e)) {
							// System.out.println(e.getSourceVertex() + "->" +
							// e.getTargetVertex());
							if (!exitStates.containsKey(v)) {
								exitStates.put(v, e);
							}
						}
					}
				}

				/*
				 * if there is more than one walk between an entrance to the
				 * escc and and exit to the escc we have EDA. If a vertex is
				 * both an entrance and an exit, we count staying on the same
				 * vertex as a path.
				 */

				for (NFAVertexND start : entranceStates.keySet()) {
					if (isInterrupted()) {
						throw new InterruptedException();
					}
					HashMap pathsToOtherVertices = NFAAnalysisTools.numWalksFrom(currentESCC, start);
					/*
					 * map from other vertex to the number of paths to that
					 * vertex
					 */
					for (Map.Entry kv : pathsToOtherVertices.entrySet()) {
						if (isInterrupted()) {
							throw new InterruptedException();
						}
						NFAVertexND p = kv.getKey();
						/* if the other vertex is an exit vertex */
						if (exitStates.containsKey(p)) {
							if (kv.getValue() > 1) {
								/* building the exploit string */
								/*
								 * find the scc of the merged escc in the merged
								 * graph
								 */
								 
								NFAEdge entranceEdge = entranceStates.get(start);
								NFAEdge exitEdge = exitStates.get(p);
								//System.out.println("Case: More than one walk through escc: " + entranceEdge.getTargetVertex() + " " + exitEdge.getSourceVertex());
								EdaAnalysisResultsESCC resultsObject = new EdaAnalysisResultsESCC(originalM, currentSCCInOriginal, entranceEdge, exitEdge);
								return resultsObject;
							}
						}
					}
				}
			}
		}

		return new EdaAnalysisResultsNoEda(originalM);
	}

	@Override
	protected EdaAnalysisResults calculateEdaAnalysisResults(NFAGraph originalM) throws InterruptedException {
		NFAGraph merged = originalM.copy();
		//merged = NFAAnalysisTools.makeTrim(merged);
		if (isInterrupted()) {
			throw new InterruptedException();
		}
		
		LinkedList sccsInOriginal = NFAAnalysisTools.getStronglyConnectedComponents(originalM);
		if (isInterrupted()) {
			throw new InterruptedException();
		}
		

		Map esccs = NFAAnalysisTools.mergeStronglyConnectedComponents(merged, true);
		if (isInterrupted()) {
			throw new InterruptedException();
		}
		
		LinkedList sccsInMerged = NFAAnalysisTools.getStronglyConnectedComponents(merged);
		if (isInterrupted()) {
			throw new InterruptedException();
		}

		EdaAnalysisResults toReturn = new EdaAnalysisResultsNoEda(originalM);
		/* Testing for parallel edges in scc in merged graph */
		toReturn = edaTestCaseParallel(originalM, sccsInMerged);
		if (isInterrupted()) {
			throw new InterruptedException();
		}
		if (toReturn.edaCase != EdaCases.NO_EDA) {
			return toReturn;
		}
		
		
		/* Testing for multiple paths through ESCCs */
		toReturn = testCaseESCC(originalM, sccsInOriginal, esccs);
		toReturn.setPriorityRemovalStrategy(PriorityRemovalStrategy.IGNORE);
		
		if (isInterrupted()) {
			throw new InterruptedException();
		}
		if (toReturn.edaCase != EdaCases.NO_EDA) {
			return toReturn;
		}
		
		/* Testing for multiple paths in PC */
		toReturn = edaTestCaseFilter(originalM, merged);
		toReturn.setPriorityRemovalStrategy(PriorityRemovalStrategy.IGNORE);
		
		return toReturn;
	}
	
	@Override
	protected EdaAnalysisResults calculateEdaUnprioritisedAnalysisResults(NFAGraph originalM) throws InterruptedException {
		NFAGraph merged = originalM.copy();
		NFAAnalysisTools.mergeStronglyConnectedComponents(merged, true);
		if (isInterrupted()) {
			throw new InterruptedException();
		}
		//merged = NFAAnalysisTools.makeTrim(merged);
		if (isInterrupted()) {
			throw new InterruptedException();
		}

		EdaAnalysisResults toReturn = new EdaAnalysisResultsNoEda(originalM);
		toReturn = edaUnprioritisedAnalysis(merged);

		toReturn.setPriorityRemovalStrategy(PriorityRemovalStrategy.UNPRIORITISE);
		return toReturn;
	}


	@Override
	protected IdaAnalysisResults calculateIdaAnalysisResults(NFAGraph originalM) throws InterruptedException {
		NFAGraph merged = originalM.copy();
		NFAAnalysisTools.mergeStronglyConnectedComponents(merged, true);
		if (isInterrupted()) {
			throw new InterruptedException();
		}
		
		//merged = NFAAnalysisTools.makeTrim(merged);
		if (isInterrupted()) {
			throw new InterruptedException();
		}
		
		IdaAnalysisResults toReturn = new IdaAnalysisResultsNoIda(originalM);
		toReturn.setPriorityRemovalStrategy(PriorityRemovalStrategy.IGNORE);

		/* Testing for multiple paths in PC */
		toReturn = idaTestCaseFilter(originalM, merged);
		toReturn.setPriorityRemovalStrategy(PriorityRemovalStrategy.IGNORE);
				
		return toReturn;
	}

	@Override
	protected IdaAnalysisResults calculateIdaUnprioritisedAnalysisResults(NFAGraph originalM) throws InterruptedException {
		NFAGraph merged = originalM.copy();
		NFAAnalysisTools.mergeStronglyConnectedComponents(merged, true);
		if (isInterrupted()) {
			throw new InterruptedException();
		}
		
		//merged = NFAAnalysisTools.makeTrim(merged);		
		if (isInterrupted()) {
			throw new InterruptedException();
		}

		IdaAnalysisResults toReturn = new IdaAnalysisResultsNoIda(originalM);
		toReturn = idaUnprioritisedAnalysis(merged);

		toReturn.setPriorityRemovalStrategy(PriorityRemovalStrategy.UNPRIORITISE);
		return toReturn;
	}	

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy