org.jgrapht.ext.DIMACSImporter Maven / Gradle / Ivy
/*
* (C) Copyright 2010-2016, by Michael Behrisch, Joris Kinable, Dimitrios
* Michail and Contributors.
*
* JGraphT : a free Java graph-theory library
*
* This program and the accompanying materials are dual-licensed under
* either
*
* (a) the terms of the GNU Lesser General Public License version 2.1
* as published by the Free Software Foundation, or (at your option) any
* later version.
*
* or (per the licensee's choosing)
*
* (b) the terms of the Eclipse Public License v1.0 as published by
* the Eclipse Foundation.
*/
package org.jgrapht.ext;
import java.io.*;
import java.util.*;
import org.jgrapht.*;
/**
* Imports a graph specified in DIMACS format (http://mat.gsia.cmu.edu/COLOR/general/ccformat.ps).
* In summary, graphs specified in DIMACS format adhere to the following structure:
*
*
* {@code
* DIMACS G {
* c
* e
* e
* e
* e
* ...
* }
* }
*
*
* Although not specified directly in the DIMACS format documentation, this implementation also
* allows for the a weighted variant:
*
*
* {@code
* e
* }
*
*
* Note: the current implementation does not fully implement the DIMACS specifications! Special
* (rarely used) fields specified as 'Optional Descriptors' are currently not supported.
*
* @author Michael Behrisch (adaptation of GraphReader class)
* @author Joris Kinable
* @author Dimitrios Michail
*
* @param the graph vertex type
* @param the graph edge type
*/
public class DIMACSImporter
implements GraphImporter
{
private VertexProvider vertexProvider;
private EdgeProvider edgeProvider;
private final double defaultWeight;
// ~ Constructors ----------------------------------------------------------
/**
* Construct a new DIMACSImporter
*
* @param vertexProvider provider for the generation of vertices. Must not be null.
* @param edgeProvider provider for the generation of edges. Must not be null.
* @param defaultWeight default edge weight
*/
public DIMACSImporter(
VertexProvider vertexProvider, EdgeProvider edgeProvider, double defaultWeight)
{
if (vertexProvider == null) {
throw new IllegalArgumentException("Vertex provider cannot be null");
}
this.vertexProvider = vertexProvider;
if (edgeProvider == null) {
throw new IllegalArgumentException("Edge provider cannot be null");
}
this.edgeProvider = edgeProvider;
this.defaultWeight = defaultWeight;
}
/**
* Construct a new DIMACSImporter
*
* @param vertexProvider provider for the generation of vertices. Must not be null.
* @param edgeProvider provider for the generation of edges. Must not be null.
*/
public DIMACSImporter(VertexProvider vertexProvider, EdgeProvider edgeProvider)
{
this(vertexProvider, edgeProvider, WeightedGraph.DEFAULT_EDGE_WEIGHT);
}
// ~ Methods ---------------------------------------------------------------
/**
* Get the vertex provider
*
* @return the vertex provider
*/
public VertexProvider getVertexProvider()
{
return vertexProvider;
}
/**
* Set the vertex provider
*
* @param vertexProvider the new vertex provider. Must not be null.
*/
public void setVertexProvider(VertexProvider vertexProvider)
{
if (vertexProvider == null) {
throw new IllegalArgumentException("Vertex provider cannot be null");
}
this.vertexProvider = vertexProvider;
}
/**
* Get the edge provider
*
* @return The edge provider
*/
public EdgeProvider getEdgeProvider()
{
return edgeProvider;
}
/**
* Set the edge provider.
*
* @param edgeProvider the new edge provider. Must not be null.
*/
public void setEdgeProvider(EdgeProvider edgeProvider)
{
if (edgeProvider == null) {
throw new IllegalArgumentException("Edge provider cannot be null");
}
this.edgeProvider = edgeProvider;
}
/**
* Import a graph.
*
*
* The provided graph must be able to support the features of the graph that is read. For
* example if the file contains self-loops then the graph provided must also support self-loops.
* The same for multiple edges.
*
*
* If the provided graph is a weighted graph, the importer also reads edge weights. Otherwise
* edge weights are ignored.
*
* @param graph the output graph
* @param input the input reader
* @throws ImportException in case an error occurs, such as I/O or parse error
*/
@Override
public void importGraph(Graph graph, Reader input)
throws ImportException
{
// convert to buffered
BufferedReader in;
if (input instanceof BufferedReader) {
in = (BufferedReader) input;
} else {
in = new BufferedReader(input);
}
// add nodes
final int size = readNodeCount(in);
Map map = new HashMap();
for (int i = 0; i < size; i++) {
Integer id = Integer.valueOf(i + 1);
V vertex = vertexProvider.buildVertex(id.toString(), new HashMap());
map.put(id, vertex);
graph.addVertex(vertex);
}
// add edges
String[] cols = skipComments(in);
while (cols != null) {
if (cols[0].equals("e")) {
if (cols.length < 3) {
throw new ImportException("Failed to parse edge:" + Arrays.toString(cols));
}
Integer source;
try {
source = Integer.parseInt(cols[1]);
} catch (NumberFormatException e) {
throw new ImportException(
"Failed to parse edge source node:" + e.getMessage(), e);
}
Integer target;
try {
target = Integer.parseInt(cols[2]);
} catch (NumberFormatException e) {
throw new ImportException(
"Failed to parse edge target node:" + e.getMessage(), e);
}
String label = "e_" + source + "_" + target;
V from = map.get(source);
if (from == null) {
throw new ImportException("Node " + source + " does not exist");
}
V to = map.get(target);
if (to == null) {
throw new ImportException("Node " + target + " does not exist");
}
try {
E e = edgeProvider.buildEdge(from, to, label, new HashMap());
graph.addEdge(from, to, e);
if (graph instanceof WeightedGraph, ?>) {
double weight = defaultWeight;
if (cols.length > 3) {
weight = Double.parseDouble(cols[3]);
}
((WeightedGraph) graph).setEdgeWeight(e, weight);
}
} catch (IllegalArgumentException e) {
throw new ImportException("Failed to import DIMACS graph:" + e.getMessage(), e);
}
}
cols = skipComments(in);
}
}
private String[] split(final String src)
{
if (src == null) {
return null;
}
return src.split("\\s+");
}
private String[] skipComments(BufferedReader input)
{
String[] cols = null;
try {
cols = split(input.readLine());
while ((cols != null)
&& ((cols.length == 0) || cols[0].equals("c") || cols[0].startsWith("%")))
{
cols = split(input.readLine());
}
} catch (IOException e) {
// ignore
}
return cols;
}
private int readNodeCount(BufferedReader input)
throws ImportException
{
final String[] cols = skipComments(input);
if (cols[0].equals("p")) {
if (cols.length < 3) {
throw new ImportException("Failed to read number of vertices.");
}
Integer nodes;
try {
nodes = Integer.parseInt(cols[2]);
} catch (NumberFormatException e) {
throw new ImportException("Failed to read number of vertices.");
}
if (nodes < 0) {
throw new ImportException("Negative number of vertices.");
}
return nodes;
}
throw new ImportException("Failed to read number of vertices.");
}
}