From 59a2c48c6a925fce61e7977a42e0b6eeddcb8bd3 Mon Sep 17 00:00:00 2001
From: David Rosca <roscadav@fit.cvut.cz>
Date: Mon, 23 Mar 2015 09:43:07 +0100
Subject: [PATCH] Graph: compare by node/edge values + output to xml/string

---
 alib2data/src/graph/GraphFromStringLexer.cpp  | 30 ++++++
 alib2data/src/graph/GraphFromStringLexer.h    |  3 +
 alib2data/src/graph/GraphFromStringParser.cpp | 91 ++++++++++++++++++-
 alib2data/src/graph/GraphFromStringParser.h   |  6 ++
 alib2data/src/graph/GraphFromXMLParser.cpp    | 57 ++++++++++++
 alib2data/src/graph/GraphFromXMLParser.h      |  6 ++
 alib2data/src/graph/GraphToStringComposer.cpp | 40 ++++++++
 alib2data/src/graph/GraphToStringComposer.h   |  6 ++
 alib2data/src/graph/GraphToXMLComposer.cpp    | 41 +++++++--
 alib2data/src/graph/GraphToXMLComposer.h      | 10 +-
 .../src/graph/directed/DirectedGraph.cpp      | 16 ++++
 alib2data/src/graph/directed/DirectedGraph.h  |  3 +
 .../src/graph/undirected/UndirectedGraph.cpp  | 16 ++++
 .../src/graph/undirected/UndirectedGraph.h    |  3 +
 alib2data/test-src/graph/GraphTest.cpp        | 32 +++++++
 15 files changed, 348 insertions(+), 12 deletions(-)

diff --git a/alib2data/src/graph/GraphFromStringLexer.cpp b/alib2data/src/graph/GraphFromStringLexer.cpp
index 422de85536..9dc82b1abb 100644
--- a/alib2data/src/graph/GraphFromStringLexer.cpp
+++ b/alib2data/src/graph/GraphFromStringLexer.cpp
@@ -39,12 +39,42 @@ L0:
 		token.value += character;
 		token.raw += character;
 		return token;
+	} else if (character == '=') {
+		token.type = TokenType::EQUAL;
+		token.value += character;
+		token.raw += character;
+		return token;
+	} else if (character >= '0' && character <= '9') {
+		token.type = TokenType::INTEGER;
+		token.value += character;
+		token.raw += character;
+		goto L1;
 	} else {
 		in.putback(character);
 		putback(in, std::move(token));
 		token.type = TokenType::ERROR;
 		return token;
 	}
+
+L1:
+	character = in.get();
+	if (in.eof()) {
+		return token;
+	} else if (character >= '0' && character <= '9') {
+		token.value += character;
+		token.raw += character;
+		goto L1;
+	} else {
+		in.unget();
+		return token;
+	}
+}
+
+GraphFromStringLexer::Token GraphFromStringLexer::peek(std::istream &in) const
+{
+	Token token = next(in);
+	putback(in, token);
+	return token;
 }
 
 void GraphFromStringLexer::putback(std::istream &input, GraphFromStringLexer::Token token) const
diff --git a/alib2data/src/graph/GraphFromStringLexer.h b/alib2data/src/graph/GraphFromStringLexer.h
index 5246f9795d..92e7d3f6da 100644
--- a/alib2data/src/graph/GraphFromStringLexer.h
+++ b/alib2data/src/graph/GraphFromStringLexer.h
@@ -14,6 +14,8 @@ public:
 		RPAR,
 		COLON,
 		COMMA,
+		EQUAL,
+		INTEGER,
 		TEOF,
 		ERROR
 	};
@@ -25,6 +27,7 @@ public:
 	};
 
 	Token next(std::istream &input) const;
+	Token peek(std::istream &input) const;
 	void putback(std::istream &input, Token token) const;
 
 	std::string getString(std::istream &input) const;
diff --git a/alib2data/src/graph/GraphFromStringParser.cpp b/alib2data/src/graph/GraphFromStringParser.cpp
index b7941294e4..c341ea9fb2 100644
--- a/alib2data/src/graph/GraphFromStringParser.cpp
+++ b/alib2data/src/graph/GraphFromStringParser.cpp
@@ -41,6 +41,10 @@ DirectedGraph GraphFromStringParser::parseDirectedGraph(std::istream &input) con
 	parseNodes(input, graph);
 	parseDelimiter(input); // :
 	parseDirectedEdges(input, graph);
+	parseDelimiter(input); // :
+	parseNodeValues(input, graph);
+	parseDelimiter(input); // :
+	parseDirectedEdgeValues(input, graph);
 	return graph;
 }
 
@@ -51,6 +55,10 @@ UndirectedGraph GraphFromStringParser::parseUndirectedGraph(std::istream &input)
 	parseNodes(input, graph);
 	parseDelimiter(input); // :
 	parseUndirectedEdges(input, graph);
+	parseDelimiter(input); // :
+	parseNodeValues(input, graph);
+	parseDelimiter(input); // :
+	parseUndirectedEdgeValues(input, graph);
 	return graph;
 }
 
@@ -116,7 +124,6 @@ void GraphFromStringParser::parseDirectedEdges(std::istream &input, DirectedGrap
 	}
 	lexer.putback(input, token);
 
-
 	while (true) {
 		graph.addEdge(parseDirectedEdge(input));
 
@@ -157,4 +164,86 @@ void GraphFromStringParser::parseDelimiter(std::istream &input) const
 	}
 }
 
+int GraphFromStringParser::parseValue(std::istream &input) const
+{
+	GraphFromStringLexer::Token token = lexer.next(input);
+	if (token.type != GraphFromStringLexer::TokenType::EQUAL) {
+		throw exception::AlibException("Invalid input. Expected EQUAL");
+	}
+
+	token = lexer.next(input);
+	if (token.type != GraphFromStringLexer::TokenType::INTEGER) {
+		throw exception::AlibException("Invalid input. Expected INTEGER");
+	}
+
+	return std::stoi(token.value);
+}
+
+template<typename T>
+void GraphFromStringParser::parseNodeValues(std::istream &input, T &graph) const
+{
+	GraphFromStringLexer::Token token = lexer.peek(input);
+
+	// Empty values?
+	if (token.type == GraphFromStringLexer::TokenType::COLON) {
+		return;
+	}
+
+	while (true) {
+		graph::Node node = parseNode(input);
+		int value = parseValue(input);
+		graph.setNodeValue(node, value);
+
+		token = lexer.next(input);
+		if (token.type != GraphFromStringLexer::TokenType::COMMA) {
+			lexer.putback(input, token);
+			return;
+		}
+	}
+}
+
+void GraphFromStringParser::parseDirectedEdgeValues(std::istream &input, DirectedGraph &graph) const
+{
+	GraphFromStringLexer::Token token = lexer.peek(input);
+
+	// Empty values?
+	if (token.type == GraphFromStringLexer::TokenType::RPAR) {
+		return;
+	}
+
+	while (true) {
+		graph::DirectedEdge node = parseDirectedEdge(input);
+		int value = parseValue(input);
+		graph.setEdgeValue(node, value);
+
+		token = lexer.next(input);
+		if (token.type != GraphFromStringLexer::TokenType::COMMA) {
+			lexer.putback(input, token);
+			return;
+		}
+	}
+}
+
+void GraphFromStringParser::parseUndirectedEdgeValues(std::istream &input, UndirectedGraph &graph) const
+{
+	GraphFromStringLexer::Token token = lexer.peek(input);
+
+	// Empty values?
+	if (token.type == GraphFromStringLexer::TokenType::RPAR) {
+		return;
+	}
+
+	while (true) {
+		graph::UndirectedEdge node = parseUndirectedEdge(input);
+		int value = parseValue(input);
+		graph.setEdgeValue(node, value);
+
+		token = lexer.next(input);
+		if (token.type != GraphFromStringLexer::TokenType::COMMA) {
+			lexer.putback(input, token);
+			return;
+		}
+	}
+}
+
 } // namespace graph
diff --git a/alib2data/src/graph/GraphFromStringParser.h b/alib2data/src/graph/GraphFromStringParser.h
index 48e979b143..379ac016f8 100644
--- a/alib2data/src/graph/GraphFromStringParser.h
+++ b/alib2data/src/graph/GraphFromStringParser.h
@@ -36,6 +36,12 @@ private:
 	void parseDirectedEdges(std::istream &input, DirectedGraph &graph) const;
 	void parseUndirectedEdges(std::istream &input, UndirectedGraph &graph) const;
 	void parseDelimiter(std::istream &input) const;
+	int parseValue(std::istream &input) const;
+
+	template<typename T>
+	void parseNodeValues(std::istream &input, T &graph) const;
+	void parseDirectedEdgeValues(std::istream &input, DirectedGraph &graph) const;
+	void parseUndirectedEdgeValues(std::istream &input, UndirectedGraph &graph) const;
 
 	GraphFromStringLexer lexer;
 
diff --git a/alib2data/src/graph/GraphFromXMLParser.cpp b/alib2data/src/graph/GraphFromXMLParser.cpp
index c3bc16786b..cb35275417 100644
--- a/alib2data/src/graph/GraphFromXMLParser.cpp
+++ b/alib2data/src/graph/GraphFromXMLParser.cpp
@@ -32,6 +32,8 @@ DirectedGraph GraphFromXMLParser::parseDirectedGraph(std::list<sax::Token> &inpu
 	DirectedGraph graph(parseRepresentation(input));
 	parseNodes(input, graph);
 	parseDirectedEdges(input, graph);
+	parseNodeValues(input, graph);
+	parseDirectedEdgeValues(input, graph);
 
 	popToken(input, sax::Token::TokenType::END_ELEMENT, alib::Names::GRAPH_DIRECTED_GRAPH);
 	return graph;
@@ -44,6 +46,8 @@ UndirectedGraph GraphFromXMLParser::parseUndirectedGraph(std::list<sax::Token> &
 	UndirectedGraph graph(parseRepresentation(input));
 	parseNodes(input, graph);
 	parseUndirectedEdges(input, graph);
+	parseNodeValues(input, graph);
+	parseUndirectedEdgeValues(input, graph);
 
 	popToken(input, sax::Token::TokenType::END_ELEMENT, alib::Names::GRAPH_UNDIRECTED_GRAPH);
 	return graph;
@@ -105,6 +109,16 @@ UndirectedEdge GraphFromXMLParser::parseUndirectedEdge(std::list<sax::Token> &in
 	return UndirectedEdge(first, second, name);
 }
 
+int GraphFromXMLParser::parseValue(std::list<sax::Token> &input) const
+{
+	popToken(input, sax::Token::TokenType::START_ELEMENT, "value");
+
+	int value = std::stoi(popTokenData(input, sax::Token::TokenType::CHARACTER));
+
+	popToken(input, sax::Token::TokenType::END_ELEMENT, "value");
+	return value;
+}
+
 template<typename T>
 void GraphFromXMLParser::parseNodes(std::list<sax::Token> &input, T &graph) const
 {
@@ -139,5 +153,48 @@ void GraphFromXMLParser::parseUndirectedEdges(std::list<sax::Token> &input, Undi
 	popToken(input, sax::Token::TokenType::END_ELEMENT, "edges");
 }
 
+template<typename T>
+void GraphFromXMLParser::parseNodeValues(std::list<sax::Token> &input, T &graph) const
+{
+	popToken(input, sax::Token::TokenType::START_ELEMENT, "nodevalues");
+
+	while (isTokenType(input, sax::Token::TokenType::START_ELEMENT)) {
+		graph::Node node = parseNode(input);
+		int value = parseValue(input);
+
+		graph.setNodeValue(node, value);
+	}
+
+	popToken(input, sax::Token::TokenType::END_ELEMENT, "nodevalues");
+}
+
+void GraphFromXMLParser::parseDirectedEdgeValues(std::list<sax::Token> &input, DirectedGraph &graph) const
+{
+	popToken(input, sax::Token::TokenType::START_ELEMENT, "edgevalues");
+
+	while (isTokenType(input, sax::Token::TokenType::START_ELEMENT)) {
+		graph::DirectedEdge edge = parseDirectedEdge(input);
+		int value = parseValue(input);
+
+		graph.setEdgeValue(edge, value);
+	}
+
+	popToken(input, sax::Token::TokenType::END_ELEMENT, "edgevalues");
+}
+
+void GraphFromXMLParser::parseUndirectedEdgeValues(std::list<sax::Token> &input, UndirectedGraph &graph) const
+{
+	popToken(input, sax::Token::TokenType::START_ELEMENT, "edgevalues");
+
+	while (isTokenType(input, sax::Token::TokenType::START_ELEMENT)) {
+		graph::UndirectedEdge edge = parseUndirectedEdge(input);
+		int value = parseValue(input);
+
+		graph.setEdgeValue(edge, value);
+	}
+
+	popToken(input, sax::Token::TokenType::END_ELEMENT, "edgevalues");
+}
+
 } // namespace graph
 
diff --git a/alib2data/src/graph/GraphFromXMLParser.h b/alib2data/src/graph/GraphFromXMLParser.h
index 472dfc921c..6dff758831 100644
--- a/alib2data/src/graph/GraphFromXMLParser.h
+++ b/alib2data/src/graph/GraphFromXMLParser.h
@@ -33,12 +33,18 @@ private:
 	Node parseNode(std::list<sax::Token> &input) const;
 	DirectedEdge parseDirectedEdge(std::list<sax::Token> &input) const;
 	UndirectedEdge parseUndirectedEdge(std::list<sax::Token> &input) const;
+	int parseValue(std::list<sax::Token> &input) const;
 
 	template<typename T>
 	void parseNodes(std::list<sax::Token> &input, T &graph) const;
 	void parseDirectedEdges(std::list<sax::Token> &input, DirectedGraph &graph) const;
 	void parseUndirectedEdges(std::list<sax::Token> &input, UndirectedGraph &graph) const;
 
+	template<typename T>
+	void parseNodeValues(std::list<sax::Token> &input, T &graph) const;
+	void parseDirectedEdgeValues(std::list<sax::Token> &input, DirectedGraph &graph) const;
+	void parseUndirectedEdgeValues(std::list<sax::Token> &input, UndirectedGraph &graph) const;
+
 	template<typename T> friend class alib::xmlApi;
 };
 
diff --git a/alib2data/src/graph/GraphToStringComposer.cpp b/alib2data/src/graph/GraphToStringComposer.cpp
index 226da7fd55..83ffdb679e 100644
--- a/alib2data/src/graph/GraphToStringComposer.cpp
+++ b/alib2data/src/graph/GraphToStringComposer.cpp
@@ -22,6 +22,10 @@ void GraphToStringComposer::Visit(void *data, const DirectedGraph &graph) const
 	composeNodes(out, graph.getNodes());
 	out << ":";
 	composeDirectedEdges(out, graph.getEdges());
+	out << ":";
+	composeNodeValues(out, graph);
+	out << ":";
+	composeEdgeValues(out, graph);
 }
 
 void GraphToStringComposer::Visit(void *data, const UndirectedGraph &graph) const
@@ -34,6 +38,10 @@ void GraphToStringComposer::Visit(void *data, const UndirectedGraph &graph) cons
 	composeNodes(out, graph.getNodes());
 	out << ":";
 	composeUndirectedEdges(out, graph.getEdges());
+	out << ":";
+	composeNodeValues(out, graph);
+	out << ":";
+	composeEdgeValues(out, graph);
 }
 
 void GraphToStringComposer::Visit(void *data, const Node &node) const
@@ -108,4 +116,36 @@ void GraphToStringComposer::composeUndirectedEdges(std::ostream &out, const std:
 	}
 }
 
+template<typename T>
+void GraphToStringComposer::composeNodeValues(std::ostream &out, const T &graph) const
+{
+	bool first = true;
+
+	for (auto i : graph.nodeValues) {
+		if (first) {
+			first = false;
+		} else {
+			out << ",";
+		}
+		i.first.Accept(static_cast<void*>(&out), *this);
+		out << "=" << i.second;
+	}
+}
+
+template<typename T>
+void GraphToStringComposer::composeEdgeValues(std::ostream &out, const T &graph) const
+{
+	bool first = true;
+
+	for (auto i : graph.edgeValues) {
+		if (first) {
+			first = false;
+		} else {
+			out << ",";
+		}
+		i.first.Accept(static_cast<void*>(&out), *this);
+		out << "=" << i.second;
+	}
+}
+
 } // namespace graph
diff --git a/alib2data/src/graph/GraphToStringComposer.h b/alib2data/src/graph/GraphToStringComposer.h
index 6a7c211b99..54ecda9ddc 100644
--- a/alib2data/src/graph/GraphToStringComposer.h
+++ b/alib2data/src/graph/GraphToStringComposer.h
@@ -31,6 +31,12 @@ private:
 	void composeNodes(std::ostream &out, const std::set<Node> &nodes) const;
 	void composeDirectedEdges(std::ostream &out, const std::set<DirectedEdge> &edges) const;
 	void composeUndirectedEdges(std::ostream &out, const std::set<UndirectedEdge> &edges) const;
+
+	template<typename T>
+	void composeNodeValues(std::ostream &out, const T &graph) const;
+	template<typename T>
+	void composeEdgeValues(std::ostream &out, const T &graph) const;
+
 };
 
 } // namespace graph
diff --git a/alib2data/src/graph/GraphToXMLComposer.cpp b/alib2data/src/graph/GraphToXMLComposer.cpp
index 5421105b43..47b8bcde2d 100644
--- a/alib2data/src/graph/GraphToXMLComposer.cpp
+++ b/alib2data/src/graph/GraphToXMLComposer.cpp
@@ -2,6 +2,7 @@
 #include "GraphRepresentation.h"
 
 #include "../XmlApi.hpp"
+#include "../std/itos.h"
 
 namespace graph {
 
@@ -68,7 +69,9 @@ void GraphToXMLComposer::compose(std::list<sax::Token> &out, const DirectedGraph
 
 	composeRepresentation(out, representationToString(graph.getRepresentation()));
 	composeNodes(out, graph.getNodes());
-	composeDirectedEdges(out, graph.getEdges());
+	composeEdges(out, graph.getEdges());
+	composeNodeValues(out, graph);
+	composeEdgeValues(out, graph);
 
 	out.push_back(sax::Token(alib::Names::GRAPH_DIRECTED_GRAPH, sax::Token::TokenType::END_ELEMENT));
 }
@@ -79,7 +82,9 @@ void GraphToXMLComposer::compose(std::list<sax::Token> &out, const UndirectedGra
 
 	composeRepresentation(out, representationToString(graph.getRepresentation()));
 	composeNodes(out, graph.getNodes());
-	composeUndirectedEdges(out, graph.getEdges());
+	composeEdges(out, graph.getEdges());
+	composeNodeValues(out, graph);
+	composeEdgeValues(out, graph);
 
 	out.push_back(sax::Token(alib::Names::GRAPH_UNDIRECTED_GRAPH, sax::Token::TokenType::END_ELEMENT));
 }
@@ -100,22 +105,40 @@ void GraphToXMLComposer::composeNodes(std::list<sax::Token> &out, const std::set
 	out.push_back(sax::Token("nodes", sax::Token::TokenType::END_ELEMENT));
 }
 
-void GraphToXMLComposer::composeDirectedEdges(std::list<sax::Token> &out, const std::set<DirectedEdge> &edges) const
+template<typename T>
+void GraphToXMLComposer::composeEdges(std::list<sax::Token> &out, const std::set<T> &edges) const
 {
 	out.push_back(sax::Token("edges", sax::Token::TokenType::START_ELEMENT));
-	for (const DirectedEdge &edge : edges) {
+	for (const auto &edge : edges) {
 		edge.Accept(static_cast<void*>(&out), *this);
 	}
 	out.push_back(sax::Token("edges", sax::Token::TokenType::END_ELEMENT));
 }
 
-void GraphToXMLComposer::composeUndirectedEdges(std::list<sax::Token> &out, const std::set<UndirectedEdge> &edges) const
+template<typename T>
+void GraphToXMLComposer::composeNodeValues(std::list<sax::Token> &out, const T &graph) const
 {
-	out.push_back(sax::Token("edges", sax::Token::TokenType::START_ELEMENT));
-	for (const UndirectedEdge &edge : edges) {
-		edge.Accept(static_cast<void*>(&out), *this);
+	out.push_back(sax::Token("nodevalues", sax::Token::TokenType::START_ELEMENT));
+	for (auto i : graph.nodeValues) {
+		i.first.Accept(static_cast<void*>(&out), *this);
+		out.push_back(sax::Token("value", sax::Token::TokenType::START_ELEMENT));
+		out.push_back(sax::Token(std::itos(i.second), sax::Token::TokenType::CHARACTER));
+		out.push_back(sax::Token("value", sax::Token::TokenType::END_ELEMENT));
 	}
-	out.push_back(sax::Token("edges", sax::Token::TokenType::END_ELEMENT));
+	out.push_back(sax::Token("nodevalues", sax::Token::TokenType::END_ELEMENT));
+}
+
+template<typename T>
+void GraphToXMLComposer::composeEdgeValues(std::list<sax::Token> &out, const T &graph) const
+{
+	out.push_back(sax::Token("edgevalues", sax::Token::TokenType::START_ELEMENT));
+	for (auto &i : graph.edgeValues) {
+		i.first.Accept(static_cast<void*>(&out), *this);
+		out.push_back(sax::Token("value", sax::Token::TokenType::START_ELEMENT));
+		out.push_back(sax::Token(std::itos(i.second), sax::Token::TokenType::CHARACTER));
+		out.push_back(sax::Token("value", sax::Token::TokenType::END_ELEMENT));
+	}
+	out.push_back(sax::Token("edgevalues", sax::Token::TokenType::END_ELEMENT));
 }
 
 } // namespace graph
diff --git a/alib2data/src/graph/GraphToXMLComposer.h b/alib2data/src/graph/GraphToXMLComposer.h
index 0fd0e1d8c5..79bd1df882 100644
--- a/alib2data/src/graph/GraphToXMLComposer.h
+++ b/alib2data/src/graph/GraphToXMLComposer.h
@@ -35,8 +35,14 @@ private:
 
 	void composeRepresentation(std::list<sax::Token> &out, const std::string &representation) const;
 	void composeNodes(std::list<sax::Token> &out, const std::set<Node> &nodes) const;
-	void composeDirectedEdges(std::list<sax::Token> &out, const std::set<DirectedEdge> &edges) const;
-	void composeUndirectedEdges(std::list<sax::Token> &out, const std::set<UndirectedEdge> &edges) const;
+
+	template<typename T>
+	void composeEdges(std::list<sax::Token> &out, const std::set<T> &edges) const;
+
+	template<typename T>
+	void composeNodeValues(std::list<sax::Token> &out, const T &graph) const;
+	template<typename T>
+	void composeEdgeValues(std::list<sax::Token> &out, const T &graph) const;
 
 	template<typename T> friend class alib::xmlApi;
 };
diff --git a/alib2data/src/graph/directed/DirectedGraph.cpp b/alib2data/src/graph/directed/DirectedGraph.cpp
index af07c466a6..c7b2349c38 100644
--- a/alib2data/src/graph/directed/DirectedGraph.cpp
+++ b/alib2data/src/graph/directed/DirectedGraph.cpp
@@ -23,6 +23,8 @@ DirectedGraph::~DirectedGraph() noexcept
 
 DirectedGraph::DirectedGraph(const DirectedGraph &other)
 	: representation(other.representation)
+	, nodeValues(other.nodeValues)
+	, edgeValues(other.edgeValues)
 {
 	init();
 	impl->copy(other.impl);
@@ -31,6 +33,8 @@ DirectedGraph::DirectedGraph(const DirectedGraph &other)
 DirectedGraph::DirectedGraph(DirectedGraph &&other) noexcept
 	: representation(other.representation)
 	, impl(other.impl)
+	, nodeValues(std::move(other.nodeValues))
+	, edgeValues(std::move(other.edgeValues))
 {
 	other.impl = 0;
 }
@@ -49,6 +53,8 @@ DirectedGraph &DirectedGraph::operator=(DirectedGraph &&other) noexcept
 {
 	this->representation = other.representation;
 	std::swap(this->impl, other.impl);
+	std::swap(this->nodeValues, other.nodeValues);
+	std::swap(this->edgeValues, other.edgeValues);
 	return *this;
 }
 
@@ -184,6 +190,16 @@ int DirectedGraph::compare(const DirectedGraph &other) const
 		return static_cast<int>(representation) - static_cast<int>(other.representation);
 	}
 
+	auto first = std::tie(nodeValues, edgeValues);
+	auto second = std::tie(other.nodeValues, other.edgeValues);
+
+	std::compare<decltype(first)> comp;
+	int res = comp(first, second);
+
+	if (res != 0) {
+		return res;
+	}
+
 	return impl->compare(other.impl);
 }
 
diff --git a/alib2data/src/graph/directed/DirectedGraph.h b/alib2data/src/graph/directed/DirectedGraph.h
index 0844695284..5aa24d5703 100644
--- a/alib2data/src/graph/directed/DirectedGraph.h
+++ b/alib2data/src/graph/directed/DirectedGraph.h
@@ -79,6 +79,9 @@ private:
 
 	std::unordered_map<Node, int> nodeValues;
 	std::unordered_map<DirectedEdge, int> edgeValues;
+
+	friend class GraphToXMLComposer;
+	friend class GraphToStringComposer;
 };
 
 } // namespace graph
diff --git a/alib2data/src/graph/undirected/UndirectedGraph.cpp b/alib2data/src/graph/undirected/UndirectedGraph.cpp
index 4209cf1674..2c35fa8600 100644
--- a/alib2data/src/graph/undirected/UndirectedGraph.cpp
+++ b/alib2data/src/graph/undirected/UndirectedGraph.cpp
@@ -23,6 +23,8 @@ UndirectedGraph::~UndirectedGraph() noexcept
 
 UndirectedGraph::UndirectedGraph(const UndirectedGraph &other)
 	: representation(other.representation)
+	, nodeValues(other.nodeValues)
+	, edgeValues(other.edgeValues)
 {
 	init();
 	impl->copy(other.impl);
@@ -31,6 +33,8 @@ UndirectedGraph::UndirectedGraph(const UndirectedGraph &other)
 UndirectedGraph::UndirectedGraph(UndirectedGraph &&other) noexcept
 	: representation(other.representation)
 	, impl(other.impl)
+	, nodeValues(std::move(other.nodeValues))
+	, edgeValues(std::move(other.edgeValues))
 {
 	other.impl = 0;
 }
@@ -49,6 +53,8 @@ UndirectedGraph &UndirectedGraph::operator=(UndirectedGraph &&other) noexcept
 {
 	this->representation = other.representation;
 	std::swap(this->impl, other.impl);
+	std::swap(this->nodeValues, other.nodeValues);
+	std::swap(this->edgeValues, other.edgeValues);
 	return *this;
 }
 
@@ -184,6 +190,16 @@ int UndirectedGraph::compare(const UndirectedGraph &other) const
 		return static_cast<int>(representation) - static_cast<int>(other.representation);
 	}
 
+	auto first = std::tie(nodeValues, edgeValues);
+	auto second = std::tie(other.nodeValues, other.edgeValues);
+
+	std::compare<decltype(first)> comp;
+	int res = comp(first, second);
+
+	if (res != 0) {
+		return res;
+	}
+
 	return impl->compare(other.impl);
 }
 
diff --git a/alib2data/src/graph/undirected/UndirectedGraph.h b/alib2data/src/graph/undirected/UndirectedGraph.h
index 8476e4d970..0fc7dad5f0 100644
--- a/alib2data/src/graph/undirected/UndirectedGraph.h
+++ b/alib2data/src/graph/undirected/UndirectedGraph.h
@@ -79,6 +79,9 @@ private:
 
 	std::unordered_map<Node, int> nodeValues;
 	std::unordered_map<UndirectedEdge, int> edgeValues;
+
+	friend class GraphToXMLComposer;
+	friend class GraphToStringComposer;
 };
 
 } // namespace graph
diff --git a/alib2data/test-src/graph/GraphTest.cpp b/alib2data/test-src/graph/GraphTest.cpp
index 5352db77fd..f39266d61a 100644
--- a/alib2data/test-src/graph/GraphTest.cpp
+++ b/alib2data/test-src/graph/GraphTest.cpp
@@ -182,6 +182,14 @@ void GraphTest::testXMLParser_impl(graph::REPRESENTATION representation)
 	graph::Graph dg2 = alib::XmlDataFactory::fromString<graph::Graph>(tmp);
 	CPPUNIT_ASSERT(dg1 == dg2);
 
+	dg.setEdgeValue(graph::DirectedEdge(n1, n2), 1);
+	dg.setEdgeValue(graph::DirectedEdge(n1, n3), 2);
+
+	graph::Graph dg1_2(dg);
+	tmp = alib::XmlDataFactory::toString(dg1_2);
+	graph::Graph dg2_2 = alib::XmlDataFactory::fromString<graph::Graph>(tmp);
+	CPPUNIT_ASSERT(dg1_2 == dg2_2);
+
 	// Undirected
 	graph::UndirectedGraph ug(representation);
 	ug.addNode(n1);
@@ -194,6 +202,14 @@ void GraphTest::testXMLParser_impl(graph::REPRESENTATION representation)
 	tmp = alib::XmlDataFactory::toString(ug1);
 	graph::Graph ug2 = alib::XmlDataFactory::fromString<graph::Graph>(tmp);
 	CPPUNIT_ASSERT(ug1 == ug2);
+
+	ug.setEdgeValue(graph::UndirectedEdge(n1, n2), 1);
+	ug.setEdgeValue(graph::UndirectedEdge(n1, n3), 2);
+
+	graph::Graph ug1_2(ug);
+	tmp = alib::XmlDataFactory::toString(ug1_2);
+	graph::Graph ug2_2 = alib::XmlDataFactory::fromString<graph::Graph>(tmp);
+	CPPUNIT_ASSERT(ug1_2 == ug2_2);
 }
 
 void GraphTest::testStringParser_impl(graph::REPRESENTATION representation)
@@ -216,6 +232,14 @@ void GraphTest::testStringParser_impl(graph::REPRESENTATION representation)
 	graph::Graph dg2 = alib::StringDataFactory::fromString<graph::Graph>(tmp);
 	CPPUNIT_ASSERT(dg1 == dg2);
 
+	dg.setEdgeValue(graph::DirectedEdge(n1, n2), 1);
+	dg.setEdgeValue(graph::DirectedEdge(n1, n3), 2);
+
+	graph::Graph dg1_2(dg);
+	tmp = alib::StringDataFactory::toString(dg1_2);
+	graph::Graph dg2_2 = alib::StringDataFactory::fromString<graph::Graph>(tmp);
+	CPPUNIT_ASSERT(dg1_2 == dg2_2);
+
 	// Undirected
 	graph::UndirectedGraph ug(representation);
 	ug.addNode(n1);
@@ -228,6 +252,14 @@ void GraphTest::testStringParser_impl(graph::REPRESENTATION representation)
 	tmp = alib::StringDataFactory::toString(ug1);
 	graph::Graph ug2 = alib::StringDataFactory::fromString<graph::Graph>(tmp);
 	CPPUNIT_ASSERT(ug1 == ug2);
+
+	ug.setEdgeValue(graph::UndirectedEdge(n1, n2), 1);
+	ug.setEdgeValue(graph::UndirectedEdge(n1, n3), 2);
+
+	graph::Graph ug1_2(ug);
+	tmp = alib::StringDataFactory::toString(ug1_2);
+	graph::Graph ug2_2 = alib::StringDataFactory::fromString<graph::Graph>(tmp);
+	CPPUNIT_ASSERT(ug1_2 == ug2_2);
 }
 
 void GraphTest::testAddEdge_impl(graph::REPRESENTATION representation)
-- 
GitLab