lemon/lgf_reader.h
author Alpar Juttner <alpar@cs.elte.hu>
Fri, 05 Aug 2011 00:12:23 +0200
branch1.0
changeset 426 8276e0ee48b1
parent 376 d3524090d5e2
parent 422 54464584b157
child 427 c59bdcc8e33e
permissions -rw-r--r--
Update NEWS file
alpar@209
     1
/* -*- mode: C++; indent-tabs-mode: nil; -*-
deba@127
     2
 *
alpar@209
     3
 * This file is a part of LEMON, a generic C++ optimization library.
deba@127
     4
 *
alpar@422
     5
 * Copyright (C) 2003-2011
deba@127
     6
 * Egervary Jeno Kombinatorikus Optimalizalasi Kutatocsoport
deba@127
     7
 * (Egervary Research Group on Combinatorial Optimization, EGRES).
deba@127
     8
 *
deba@127
     9
 * Permission to use, modify and distribute this software is granted
deba@127
    10
 * provided that this copyright notice appears in all copies. For
deba@127
    11
 * precise terms see the accompanying LICENSE file.
deba@127
    12
 *
deba@127
    13
 * This software is provided "AS IS" with no warranty of any kind,
deba@127
    14
 * express or implied, and with no claim as to its suitability for any
deba@127
    15
 * purpose.
deba@127
    16
 *
deba@127
    17
 */
deba@127
    18
deba@127
    19
///\ingroup lemon_io
deba@127
    20
///\file
ladanyi@236
    21
///\brief \ref lgf-format "LEMON Graph Format" reader.
deba@127
    22
deba@127
    23
deba@127
    24
#ifndef LEMON_LGF_READER_H
deba@127
    25
#define LEMON_LGF_READER_H
deba@127
    26
deba@127
    27
#include <iostream>
deba@127
    28
#include <fstream>
deba@127
    29
#include <sstream>
deba@127
    30
deba@127
    31
#include <set>
deba@127
    32
#include <map>
deba@127
    33
deba@220
    34
#include <lemon/core.h>
deba@127
    35
deba@127
    36
#include <lemon/lgf_writer.h>
deba@127
    37
deba@127
    38
#include <lemon/concept_check.h>
deba@127
    39
#include <lemon/concepts/maps.h>
deba@127
    40
deba@127
    41
namespace lemon {
deba@127
    42
deba@127
    43
  namespace _reader_bits {
deba@127
    44
deba@127
    45
    template <typename Value>
deba@127
    46
    struct DefaultConverter {
deba@127
    47
      Value operator()(const std::string& str) {
alpar@209
    48
        std::istringstream is(str);
alpar@209
    49
        Value value;
deba@290
    50
        if (!(is >> value)) {
deba@290
    51
          throw FormatError("Cannot read token");
deba@290
    52
        }
alpar@209
    53
alpar@209
    54
        char c;
alpar@209
    55
        if (is >> std::ws >> c) {
deba@290
    56
          throw FormatError("Remaining characters in token");
alpar@209
    57
        }
alpar@209
    58
        return value;
deba@127
    59
      }
deba@127
    60
    };
deba@127
    61
deba@127
    62
    template <>
deba@127
    63
    struct DefaultConverter<std::string> {
deba@127
    64
      std::string operator()(const std::string& str) {
alpar@209
    65
        return str;
deba@127
    66
      }
deba@127
    67
    };
deba@127
    68
alpar@209
    69
    template <typename _Item>
deba@127
    70
    class MapStorageBase {
deba@127
    71
    public:
deba@127
    72
      typedef _Item Item;
deba@127
    73
deba@127
    74
    public:
deba@127
    75
      MapStorageBase() {}
deba@127
    76
      virtual ~MapStorageBase() {}
deba@127
    77
deba@127
    78
      virtual void set(const Item& item, const std::string& value) = 0;
deba@127
    79
deba@127
    80
    };
deba@127
    81
alpar@209
    82
    template <typename _Item, typename _Map,
alpar@209
    83
              typename _Converter = DefaultConverter<typename _Map::Value> >
deba@127
    84
    class MapStorage : public MapStorageBase<_Item> {
deba@127
    85
    public:
deba@127
    86
      typedef _Map Map;
deba@127
    87
      typedef _Converter Converter;
deba@127
    88
      typedef _Item Item;
alpar@209
    89
deba@127
    90
    private:
deba@127
    91
      Map& _map;
deba@127
    92
      Converter _converter;
deba@127
    93
deba@127
    94
    public:
alpar@209
    95
      MapStorage(Map& map, const Converter& converter = Converter())
alpar@209
    96
        : _map(map), _converter(converter) {}
deba@127
    97
      virtual ~MapStorage() {}
deba@127
    98
deba@127
    99
      virtual void set(const Item& item ,const std::string& value) {
alpar@209
   100
        _map.set(item, _converter(value));
deba@127
   101
      }
deba@127
   102
    };
deba@127
   103
alpar@209
   104
    template <typename _Graph, bool _dir, typename _Map,
alpar@209
   105
              typename _Converter = DefaultConverter<typename _Map::Value> >
deba@165
   106
    class GraphArcMapStorage : public MapStorageBase<typename _Graph::Edge> {
deba@165
   107
    public:
deba@165
   108
      typedef _Map Map;
deba@165
   109
      typedef _Converter Converter;
deba@165
   110
      typedef _Graph Graph;
deba@165
   111
      typedef typename Graph::Edge Item;
deba@165
   112
      static const bool dir = _dir;
alpar@209
   113
deba@165
   114
    private:
deba@165
   115
      const Graph& _graph;
deba@165
   116
      Map& _map;
deba@165
   117
      Converter _converter;
deba@165
   118
deba@165
   119
    public:
alpar@209
   120
      GraphArcMapStorage(const Graph& graph, Map& map,
alpar@209
   121
                         const Converter& converter = Converter())
alpar@209
   122
        : _graph(graph), _map(map), _converter(converter) {}
deba@165
   123
      virtual ~GraphArcMapStorage() {}
deba@165
   124
deba@165
   125
      virtual void set(const Item& item ,const std::string& value) {
alpar@209
   126
        _map.set(_graph.direct(item, dir), _converter(value));
deba@165
   127
      }
deba@165
   128
    };
deba@165
   129
deba@127
   130
    class ValueStorageBase {
deba@127
   131
    public:
deba@127
   132
      ValueStorageBase() {}
deba@127
   133
      virtual ~ValueStorageBase() {}
deba@127
   134
deba@127
   135
      virtual void set(const std::string&) = 0;
deba@127
   136
    };
deba@127
   137
deba@127
   138
    template <typename _Value, typename _Converter = DefaultConverter<_Value> >
deba@127
   139
    class ValueStorage : public ValueStorageBase {
deba@127
   140
    public:
deba@127
   141
      typedef _Value Value;
deba@127
   142
      typedef _Converter Converter;
deba@127
   143
deba@127
   144
    private:
deba@127
   145
      Value& _value;
deba@127
   146
      Converter _converter;
deba@127
   147
deba@127
   148
    public:
deba@127
   149
      ValueStorage(Value& value, const Converter& converter = Converter())
kpeter@212
   150
        : _value(value), _converter(converter) {}
deba@127
   151
deba@127
   152
      virtual void set(const std::string& value) {
alpar@209
   153
        _value = _converter(value);
deba@127
   154
      }
deba@127
   155
    };
deba@127
   156
deba@127
   157
    template <typename Value>
deba@127
   158
    struct MapLookUpConverter {
deba@127
   159
      const std::map<std::string, Value>& _map;
deba@127
   160
deba@127
   161
      MapLookUpConverter(const std::map<std::string, Value>& map)
deba@127
   162
        : _map(map) {}
deba@127
   163
deba@127
   164
      Value operator()(const std::string& str) {
deba@127
   165
        typename std::map<std::string, Value>::const_iterator it =
deba@127
   166
          _map.find(str);
deba@127
   167
        if (it == _map.end()) {
deba@127
   168
          std::ostringstream msg;
deba@127
   169
          msg << "Item not found: " << str;
deba@290
   170
          throw FormatError(msg.str());
deba@127
   171
        }
deba@127
   172
        return it->second;
deba@127
   173
      }
deba@127
   174
    };
deba@127
   175
deba@165
   176
    template <typename Graph>
deba@165
   177
    struct GraphArcLookUpConverter {
deba@165
   178
      const Graph& _graph;
deba@165
   179
      const std::map<std::string, typename Graph::Edge>& _map;
alpar@209
   180
alpar@209
   181
      GraphArcLookUpConverter(const Graph& graph,
alpar@209
   182
                              const std::map<std::string,
alpar@209
   183
                                             typename Graph::Edge>& map)
alpar@209
   184
        : _graph(graph), _map(map) {}
alpar@209
   185
deba@165
   186
      typename Graph::Arc operator()(const std::string& str) {
alpar@209
   187
        if (str.empty() || (str[0] != '+' && str[0] != '-')) {
deba@290
   188
          throw FormatError("Item must start with '+' or '-'");
alpar@209
   189
        }
alpar@209
   190
        typename std::map<std::string, typename Graph::Edge>
alpar@209
   191
          ::const_iterator it = _map.find(str.substr(1));
alpar@209
   192
        if (it == _map.end()) {
deba@290
   193
          throw FormatError("Item not found");
alpar@209
   194
        }
alpar@209
   195
        return _graph.direct(it->second, str[0] == '+');
deba@165
   196
      }
deba@165
   197
    };
deba@165
   198
deba@197
   199
    inline bool isWhiteSpace(char c) {
alpar@209
   200
      return c == ' ' || c == '\t' || c == '\v' ||
alpar@209
   201
        c == '\n' || c == '\r' || c == '\f';
deba@127
   202
    }
alpar@209
   203
deba@197
   204
    inline bool isOct(char c) {
alpar@209
   205
      return '0' <= c && c <='7';
deba@127
   206
    }
alpar@209
   207
deba@197
   208
    inline int valueOct(char c) {
deba@127
   209
      LEMON_ASSERT(isOct(c), "The character is not octal.");
deba@127
   210
      return c - '0';
deba@127
   211
    }
deba@127
   212
deba@197
   213
    inline bool isHex(char c) {
alpar@209
   214
      return ('0' <= c && c <= '9') ||
alpar@209
   215
        ('a' <= c && c <= 'z') ||
alpar@209
   216
        ('A' <= c && c <= 'Z');
deba@127
   217
    }
alpar@209
   218
deba@197
   219
    inline int valueHex(char c) {
deba@127
   220
      LEMON_ASSERT(isHex(c), "The character is not hexadecimal.");
deba@127
   221
      if ('0' <= c && c <= '9') return c - '0';
deba@127
   222
      if ('a' <= c && c <= 'z') return c - 'a' + 10;
deba@127
   223
      return c - 'A' + 10;
deba@127
   224
    }
deba@127
   225
deba@197
   226
    inline bool isIdentifierFirstChar(char c) {
deba@127
   227
      return ('a' <= c && c <= 'z') ||
alpar@209
   228
        ('A' <= c && c <= 'Z') || c == '_';
deba@127
   229
    }
deba@127
   230
deba@197
   231
    inline bool isIdentifierChar(char c) {
deba@127
   232
      return isIdentifierFirstChar(c) ||
alpar@209
   233
        ('0' <= c && c <= '9');
deba@127
   234
    }
deba@127
   235
deba@197
   236
    inline char readEscape(std::istream& is) {
deba@127
   237
      char c;
deba@127
   238
      if (!is.get(c))
deba@290
   239
        throw FormatError("Escape format error");
deba@127
   240
deba@127
   241
      switch (c) {
deba@127
   242
      case '\\':
alpar@209
   243
        return '\\';
deba@127
   244
      case '\"':
alpar@209
   245
        return '\"';
deba@127
   246
      case '\'':
alpar@209
   247
        return '\'';
deba@127
   248
      case '\?':
alpar@209
   249
        return '\?';
deba@127
   250
      case 'a':
alpar@209
   251
        return '\a';
deba@127
   252
      case 'b':
alpar@209
   253
        return '\b';
deba@127
   254
      case 'f':
alpar@209
   255
        return '\f';
deba@127
   256
      case 'n':
alpar@209
   257
        return '\n';
deba@127
   258
      case 'r':
alpar@209
   259
        return '\r';
deba@127
   260
      case 't':
alpar@209
   261
        return '\t';
deba@127
   262
      case 'v':
alpar@209
   263
        return '\v';
deba@127
   264
      case 'x':
alpar@209
   265
        {
alpar@209
   266
          int code;
alpar@209
   267
          if (!is.get(c) || !isHex(c))
deba@290
   268
            throw FormatError("Escape format error");
alpar@209
   269
          else if (code = valueHex(c), !is.get(c) || !isHex(c)) is.putback(c);
alpar@209
   270
          else code = code * 16 + valueHex(c);
alpar@209
   271
          return code;
alpar@209
   272
        }
deba@127
   273
      default:
alpar@209
   274
        {
alpar@209
   275
          int code;
alpar@209
   276
          if (!isOct(c))
deba@290
   277
            throw FormatError("Escape format error");
alpar@209
   278
          else if (code = valueOct(c), !is.get(c) || !isOct(c))
alpar@209
   279
            is.putback(c);
alpar@209
   280
          else if (code = code * 8 + valueOct(c), !is.get(c) || !isOct(c))
alpar@209
   281
            is.putback(c);
alpar@209
   282
          else code = code * 8 + valueOct(c);
alpar@209
   283
          return code;
alpar@209
   284
        }
alpar@209
   285
      }
deba@127
   286
    }
alpar@209
   287
deba@197
   288
    inline std::istream& readToken(std::istream& is, std::string& str) {
deba@127
   289
      std::ostringstream os;
deba@127
   290
deba@127
   291
      char c;
deba@127
   292
      is >> std::ws;
alpar@209
   293
alpar@209
   294
      if (!is.get(c))
alpar@209
   295
        return is;
deba@127
   296
deba@127
   297
      if (c == '\"') {
alpar@209
   298
        while (is.get(c) && c != '\"') {
alpar@209
   299
          if (c == '\\')
alpar@209
   300
            c = readEscape(is);
alpar@209
   301
          os << c;
alpar@209
   302
        }
alpar@209
   303
        if (!is)
deba@290
   304
          throw FormatError("Quoted format error");
deba@127
   305
      } else {
alpar@209
   306
        is.putback(c);
alpar@209
   307
        while (is.get(c) && !isWhiteSpace(c)) {
alpar@209
   308
          if (c == '\\')
alpar@209
   309
            c = readEscape(is);
alpar@209
   310
          os << c;
alpar@209
   311
        }
alpar@209
   312
        if (!is) {
alpar@209
   313
          is.clear();
alpar@209
   314
        } else {
alpar@209
   315
          is.putback(c);
alpar@209
   316
        }
deba@127
   317
      }
deba@127
   318
      str = os.str();
deba@127
   319
      return is;
deba@127
   320
    }
deba@162
   321
deba@162
   322
    class Section {
deba@162
   323
    public:
deba@162
   324
      virtual ~Section() {}
deba@162
   325
      virtual void process(std::istream& is, int& line_num) = 0;
deba@162
   326
    };
deba@162
   327
deba@162
   328
    template <typename Functor>
deba@162
   329
    class LineSection : public Section {
deba@162
   330
    private:
deba@162
   331
deba@162
   332
      Functor _functor;
deba@162
   333
deba@162
   334
    public:
alpar@209
   335
deba@162
   336
      LineSection(const Functor& functor) : _functor(functor) {}
deba@162
   337
      virtual ~LineSection() {}
deba@162
   338
deba@162
   339
      virtual void process(std::istream& is, int& line_num) {
alpar@209
   340
        char c;
alpar@209
   341
        std::string line;
alpar@209
   342
        while (is.get(c) && c != '@') {
alpar@209
   343
          if (c == '\n') {
alpar@209
   344
            ++line_num;
alpar@209
   345
          } else if (c == '#') {
alpar@209
   346
            getline(is, line);
alpar@209
   347
            ++line_num;
alpar@209
   348
          } else if (!isWhiteSpace(c)) {
alpar@209
   349
            is.putback(c);
alpar@209
   350
            getline(is, line);
alpar@209
   351
            _functor(line);
alpar@209
   352
            ++line_num;
alpar@209
   353
          }
alpar@209
   354
        }
alpar@209
   355
        if (is) is.putback(c);
alpar@209
   356
        else if (is.eof()) is.clear();
deba@162
   357
      }
deba@162
   358
    };
deba@162
   359
deba@162
   360
    template <typename Functor>
deba@162
   361
    class StreamSection : public Section {
deba@162
   362
    private:
deba@162
   363
deba@162
   364
      Functor _functor;
deba@162
   365
deba@162
   366
    public:
alpar@209
   367
deba@162
   368
      StreamSection(const Functor& functor) : _functor(functor) {}
alpar@209
   369
      virtual ~StreamSection() {}
deba@162
   370
deba@162
   371
      virtual void process(std::istream& is, int& line_num) {
alpar@209
   372
        _functor(is, line_num);
alpar@209
   373
        char c;
alpar@209
   374
        std::string line;
alpar@209
   375
        while (is.get(c) && c != '@') {
alpar@209
   376
          if (c == '\n') {
alpar@209
   377
            ++line_num;
alpar@209
   378
          } else if (!isWhiteSpace(c)) {
alpar@209
   379
            getline(is, line);
alpar@209
   380
            ++line_num;
alpar@209
   381
          }
alpar@209
   382
        }
alpar@209
   383
        if (is) is.putback(c);
alpar@209
   384
        else if (is.eof()) is.clear();
deba@162
   385
      }
deba@162
   386
    };
alpar@209
   387
deba@127
   388
  }
alpar@156
   389
deba@190
   390
  template <typename Digraph>
deba@190
   391
  class DigraphReader;
deba@190
   392
deba@190
   393
  template <typename Digraph>
kpeter@374
   394
  DigraphReader<Digraph> digraphReader(Digraph& digraph, 
kpeter@374
   395
                                       std::istream& is = std::cin);
deba@190
   396
  template <typename Digraph>
kpeter@374
   397
  DigraphReader<Digraph> digraphReader(Digraph& digraph, const std::string& fn);
deba@190
   398
  template <typename Digraph>
kpeter@374
   399
  DigraphReader<Digraph> digraphReader(Digraph& digraph, const char *fn);
deba@190
   400
alpar@156
   401
  /// \ingroup lemon_io
alpar@209
   402
  ///
kpeter@192
   403
  /// \brief \ref lgf-format "LGF" reader for directed graphs
alpar@156
   404
  ///
alpar@156
   405
  /// This utility reads an \ref lgf-format "LGF" file.
alpar@156
   406
  ///
alpar@156
   407
  /// The reading method does a batch processing. The user creates a
alpar@156
   408
  /// reader object, then various reading rules can be added to the
alpar@156
   409
  /// reader, and eventually the reading is executed with the \c run()
alpar@156
   410
  /// member function. A map reading rule can be added to the reader
alpar@156
   411
  /// with the \c nodeMap() or \c arcMap() members. An optional
deba@162
   412
  /// converter parameter can also be added as a standard functor
kpeter@192
   413
  /// converting from \c std::string to the value type of the map. If it
deba@162
   414
  /// is set, it will determine how the tokens in the file should be
kpeter@192
   415
  /// converted to the value type of the map. If the functor is not set,
deba@162
   416
  /// then a default conversion will be used. One map can be read into
deba@162
   417
  /// multiple map objects at the same time. The \c attribute(), \c
deba@162
   418
  /// node() and \c arc() functions are used to add attribute reading
deba@162
   419
  /// rules.
alpar@156
   420
  ///
alpar@156
   421
  ///\code
kpeter@293
   422
  /// DigraphReader<Digraph>(digraph, std::cin).
kpeter@192
   423
  ///   nodeMap("coordinates", coord_map).
kpeter@192
   424
  ///   arcMap("capacity", cap_map).
kpeter@192
   425
  ///   node("source", src).
kpeter@192
   426
  ///   node("target", trg).
kpeter@192
   427
  ///   attribute("caption", caption).
kpeter@192
   428
  ///   run();
alpar@156
   429
  ///\endcode
alpar@156
   430
  ///
alpar@156
   431
  /// By default the reader uses the first section in the file of the
alpar@156
   432
  /// proper type. If a section has an optional name, then it can be
deba@162
   433
  /// selected for reading by giving an optional name parameter to the
deba@189
   434
  /// \c nodes(), \c arcs() or \c attributes() functions.
alpar@156
   435
  ///
alpar@156
   436
  /// The \c useNodes() and \c useArcs() functions are used to tell the reader
alpar@156
   437
  /// that the nodes or arcs should not be constructed (added to the
alpar@156
   438
  /// graph) during the reading, but instead the label map of the items
alpar@156
   439
  /// are given as a parameter of these functions. An
kpeter@192
   440
  /// application of these functions is multipass reading, which is
kpeter@192
   441
  /// important if two \c \@arcs sections must be read from the
kpeter@192
   442
  /// file. In this case the first phase would read the node set and one
alpar@156
   443
  /// of the arc sets, while the second phase would read the second arc
alpar@156
   444
  /// set into an \e ArcSet class (\c SmartArcSet or \c ListArcSet).
alpar@156
   445
  /// The previously read label node map should be passed to the \c
alpar@156
   446
  /// useNodes() functions. Another application of multipass reading when
alpar@210
   447
  /// paths are given as a node map or an arc map.
alpar@210
   448
  /// It is impossible to read this in
alpar@156
   449
  /// a single pass, because the arcs are not constructed when the node
alpar@156
   450
  /// maps are read.
deba@127
   451
  template <typename _Digraph>
deba@127
   452
  class DigraphReader {
deba@127
   453
  public:
deba@127
   454
deba@127
   455
    typedef _Digraph Digraph;
deba@148
   456
    TEMPLATE_DIGRAPH_TYPEDEFS(Digraph);
alpar@209
   457
deba@127
   458
  private:
deba@127
   459
deba@127
   460
deba@127
   461
    std::istream* _is;
deba@127
   462
    bool local_is;
deba@290
   463
    std::string _filename;
deba@127
   464
deba@127
   465
    Digraph& _digraph;
deba@127
   466
deba@127
   467
    std::string _nodes_caption;
deba@127
   468
    std::string _arcs_caption;
deba@127
   469
    std::string _attributes_caption;
deba@127
   470
deba@127
   471
    typedef std::map<std::string, Node> NodeIndex;
deba@127
   472
    NodeIndex _node_index;
deba@127
   473
    typedef std::map<std::string, Arc> ArcIndex;
deba@127
   474
    ArcIndex _arc_index;
alpar@209
   475
alpar@209
   476
    typedef std::vector<std::pair<std::string,
alpar@209
   477
      _reader_bits::MapStorageBase<Node>*> > NodeMaps;
alpar@209
   478
    NodeMaps _node_maps;
deba@127
   479
deba@127
   480
    typedef std::vector<std::pair<std::string,
deba@127
   481
      _reader_bits::MapStorageBase<Arc>*> >ArcMaps;
deba@127
   482
    ArcMaps _arc_maps;
deba@127
   483
alpar@209
   484
    typedef std::multimap<std::string, _reader_bits::ValueStorageBase*>
deba@127
   485
      Attributes;
deba@127
   486
    Attributes _attributes;
deba@127
   487
deba@127
   488
    bool _use_nodes;
deba@127
   489
    bool _use_arcs;
deba@127
   490
deba@188
   491
    bool _skip_nodes;
deba@188
   492
    bool _skip_arcs;
deba@188
   493
deba@127
   494
    int line_num;
deba@127
   495
    std::istringstream line;
deba@127
   496
deba@127
   497
  public:
deba@127
   498
alpar@156
   499
    /// \brief Constructor
alpar@156
   500
    ///
alpar@156
   501
    /// Construct a directed graph reader, which reads from the given
alpar@156
   502
    /// input stream.
kpeter@293
   503
    DigraphReader(Digraph& digraph, std::istream& is = std::cin)
deba@127
   504
      : _is(&is), local_is(false), _digraph(digraph),
alpar@209
   505
        _use_nodes(false), _use_arcs(false),
alpar@209
   506
        _skip_nodes(false), _skip_arcs(false) {}
deba@127
   507
alpar@156
   508
    /// \brief Constructor
alpar@156
   509
    ///
alpar@156
   510
    /// Construct a directed graph reader, which reads from the given
alpar@156
   511
    /// file.
kpeter@293
   512
    DigraphReader(Digraph& digraph, const std::string& fn)
deba@290
   513
      : _is(new std::ifstream(fn.c_str())), local_is(true),
deba@290
   514
        _filename(fn), _digraph(digraph),
kpeter@212
   515
        _use_nodes(false), _use_arcs(false),
deba@290
   516
        _skip_nodes(false), _skip_arcs(false) {
deba@295
   517
      if (!(*_is)) {
deba@295
   518
        delete _is;
deba@295
   519
        throw IoError("Cannot open file", fn);
deba@295
   520
      }
deba@290
   521
    }
alpar@209
   522
alpar@156
   523
    /// \brief Constructor
alpar@156
   524
    ///
alpar@156
   525
    /// Construct a directed graph reader, which reads from the given
alpar@156
   526
    /// file.
kpeter@293
   527
    DigraphReader(Digraph& digraph, const char* fn)
deba@290
   528
      : _is(new std::ifstream(fn)), local_is(true),
deba@290
   529
        _filename(fn), _digraph(digraph),
kpeter@212
   530
        _use_nodes(false), _use_arcs(false),
deba@290
   531
        _skip_nodes(false), _skip_arcs(false) {
deba@295
   532
      if (!(*_is)) {
deba@295
   533
        delete _is;
deba@295
   534
        throw IoError("Cannot open file", fn);
deba@295
   535
      }
deba@290
   536
    }
deba@127
   537
alpar@156
   538
    /// \brief Destructor
deba@127
   539
    ~DigraphReader() {
alpar@209
   540
      for (typename NodeMaps::iterator it = _node_maps.begin();
alpar@209
   541
           it != _node_maps.end(); ++it) {
alpar@209
   542
        delete it->second;
deba@127
   543
      }
deba@127
   544
alpar@209
   545
      for (typename ArcMaps::iterator it = _arc_maps.begin();
alpar@209
   546
           it != _arc_maps.end(); ++it) {
alpar@209
   547
        delete it->second;
deba@127
   548
      }
deba@127
   549
alpar@209
   550
      for (typename Attributes::iterator it = _attributes.begin();
alpar@209
   551
           it != _attributes.end(); ++it) {
alpar@209
   552
        delete it->second;
deba@127
   553
      }
deba@127
   554
deba@127
   555
      if (local_is) {
alpar@209
   556
        delete _is;
deba@127
   557
      }
deba@127
   558
deba@127
   559
    }
deba@127
   560
deba@127
   561
  private:
deba@190
   562
kpeter@374
   563
    template <typename DGR>
kpeter@374
   564
    friend DigraphReader<DGR> digraphReader(DGR& digraph, std::istream& is);
kpeter@374
   565
    template <typename DGR>
kpeter@374
   566
    friend DigraphReader<DGR> digraphReader(DGR& digraph, 
kpeter@374
   567
                                            const std::string& fn);
kpeter@374
   568
    template <typename DGR>
kpeter@374
   569
    friend DigraphReader<DGR> digraphReader(DGR& digraph, const char *fn);
alpar@209
   570
alpar@209
   571
    DigraphReader(DigraphReader& other)
deba@190
   572
      : _is(other._is), local_is(other.local_is), _digraph(other._digraph),
alpar@209
   573
        _use_nodes(other._use_nodes), _use_arcs(other._use_arcs),
alpar@209
   574
        _skip_nodes(other._skip_nodes), _skip_arcs(other._skip_arcs) {
deba@190
   575
deba@190
   576
      other._is = 0;
deba@190
   577
      other.local_is = false;
alpar@209
   578
deba@190
   579
      _node_index.swap(other._node_index);
deba@190
   580
      _arc_index.swap(other._arc_index);
deba@190
   581
deba@190
   582
      _node_maps.swap(other._node_maps);
deba@190
   583
      _arc_maps.swap(other._arc_maps);
deba@190
   584
      _attributes.swap(other._attributes);
deba@190
   585
deba@190
   586
      _nodes_caption = other._nodes_caption;
deba@190
   587
      _arcs_caption = other._arcs_caption;
deba@190
   588
      _attributes_caption = other._attributes_caption;
deba@190
   589
deba@190
   590
    }
deba@190
   591
deba@127
   592
    DigraphReader& operator=(const DigraphReader&);
deba@127
   593
deba@127
   594
  public:
deba@127
   595
alpar@156
   596
    /// \name Reading rules
alpar@156
   597
    /// @{
alpar@209
   598
alpar@156
   599
    /// \brief Node map reading rule
alpar@156
   600
    ///
alpar@156
   601
    /// Add a node map reading rule to the reader.
deba@127
   602
    template <typename Map>
deba@127
   603
    DigraphReader& nodeMap(const std::string& caption, Map& map) {
deba@127
   604
      checkConcept<concepts::WriteMap<Node, typename Map::Value>, Map>();
alpar@209
   605
      _reader_bits::MapStorageBase<Node>* storage =
alpar@209
   606
        new _reader_bits::MapStorage<Node, Map>(map);
deba@127
   607
      _node_maps.push_back(std::make_pair(caption, storage));
deba@127
   608
      return *this;
deba@127
   609
    }
deba@127
   610
alpar@156
   611
    /// \brief Node map reading rule
alpar@156
   612
    ///
alpar@156
   613
    /// Add a node map reading rule with specialized converter to the
alpar@156
   614
    /// reader.
deba@127
   615
    template <typename Map, typename Converter>
alpar@209
   616
    DigraphReader& nodeMap(const std::string& caption, Map& map,
alpar@209
   617
                           const Converter& converter = Converter()) {
deba@127
   618
      checkConcept<concepts::WriteMap<Node, typename Map::Value>, Map>();
alpar@209
   619
      _reader_bits::MapStorageBase<Node>* storage =
alpar@209
   620
        new _reader_bits::MapStorage<Node, Map, Converter>(map, converter);
deba@127
   621
      _node_maps.push_back(std::make_pair(caption, storage));
deba@127
   622
      return *this;
deba@127
   623
    }
deba@127
   624
alpar@156
   625
    /// \brief Arc map reading rule
alpar@156
   626
    ///
alpar@156
   627
    /// Add an arc map reading rule to the reader.
deba@127
   628
    template <typename Map>
deba@127
   629
    DigraphReader& arcMap(const std::string& caption, Map& map) {
deba@127
   630
      checkConcept<concepts::WriteMap<Arc, typename Map::Value>, Map>();
alpar@209
   631
      _reader_bits::MapStorageBase<Arc>* storage =
alpar@209
   632
        new _reader_bits::MapStorage<Arc, Map>(map);
deba@127
   633
      _arc_maps.push_back(std::make_pair(caption, storage));
deba@127
   634
      return *this;
deba@127
   635
    }
deba@127
   636
alpar@156
   637
    /// \brief Arc map reading rule
alpar@156
   638
    ///
alpar@156
   639
    /// Add an arc map reading rule with specialized converter to the
alpar@156
   640
    /// reader.
deba@127
   641
    template <typename Map, typename Converter>
alpar@209
   642
    DigraphReader& arcMap(const std::string& caption, Map& map,
alpar@209
   643
                          const Converter& converter = Converter()) {
deba@127
   644
      checkConcept<concepts::WriteMap<Arc, typename Map::Value>, Map>();
alpar@209
   645
      _reader_bits::MapStorageBase<Arc>* storage =
alpar@209
   646
        new _reader_bits::MapStorage<Arc, Map, Converter>(map, converter);
deba@127
   647
      _arc_maps.push_back(std::make_pair(caption, storage));
deba@127
   648
      return *this;
deba@127
   649
    }
deba@127
   650
alpar@156
   651
    /// \brief Attribute reading rule
alpar@156
   652
    ///
alpar@156
   653
    /// Add an attribute reading rule to the reader.
deba@127
   654
    template <typename Value>
deba@127
   655
    DigraphReader& attribute(const std::string& caption, Value& value) {
alpar@209
   656
      _reader_bits::ValueStorageBase* storage =
alpar@209
   657
        new _reader_bits::ValueStorage<Value>(value);
deba@127
   658
      _attributes.insert(std::make_pair(caption, storage));
deba@127
   659
      return *this;
deba@127
   660
    }
deba@127
   661
alpar@156
   662
    /// \brief Attribute reading rule
alpar@156
   663
    ///
alpar@156
   664
    /// Add an attribute reading rule with specialized converter to the
alpar@156
   665
    /// reader.
deba@127
   666
    template <typename Value, typename Converter>
alpar@209
   667
    DigraphReader& attribute(const std::string& caption, Value& value,
alpar@209
   668
                             const Converter& converter = Converter()) {
alpar@209
   669
      _reader_bits::ValueStorageBase* storage =
alpar@209
   670
        new _reader_bits::ValueStorage<Value, Converter>(value, converter);
deba@127
   671
      _attributes.insert(std::make_pair(caption, storage));
deba@127
   672
      return *this;
deba@127
   673
    }
deba@127
   674
alpar@156
   675
    /// \brief Node reading rule
alpar@156
   676
    ///
alpar@156
   677
    /// Add a node reading rule to reader.
deba@127
   678
    DigraphReader& node(const std::string& caption, Node& node) {
deba@127
   679
      typedef _reader_bits::MapLookUpConverter<Node> Converter;
deba@127
   680
      Converter converter(_node_index);
alpar@209
   681
      _reader_bits::ValueStorageBase* storage =
alpar@209
   682
        new _reader_bits::ValueStorage<Node, Converter>(node, converter);
deba@127
   683
      _attributes.insert(std::make_pair(caption, storage));
deba@127
   684
      return *this;
deba@127
   685
    }
deba@127
   686
alpar@156
   687
    /// \brief Arc reading rule
alpar@156
   688
    ///
alpar@156
   689
    /// Add an arc reading rule to reader.
deba@127
   690
    DigraphReader& arc(const std::string& caption, Arc& arc) {
deba@127
   691
      typedef _reader_bits::MapLookUpConverter<Arc> Converter;
deba@127
   692
      Converter converter(_arc_index);
alpar@209
   693
      _reader_bits::ValueStorageBase* storage =
alpar@209
   694
        new _reader_bits::ValueStorage<Arc, Converter>(arc, converter);
deba@127
   695
      _attributes.insert(std::make_pair(caption, storage));
deba@127
   696
      return *this;
deba@127
   697
    }
deba@127
   698
alpar@156
   699
    /// @}
alpar@156
   700
alpar@156
   701
    /// \name Select section by name
alpar@156
   702
    /// @{
alpar@156
   703
alpar@156
   704
    /// \brief Set \c \@nodes section to be read
alpar@156
   705
    ///
alpar@156
   706
    /// Set \c \@nodes section to be read
deba@127
   707
    DigraphReader& nodes(const std::string& caption) {
deba@127
   708
      _nodes_caption = caption;
deba@127
   709
      return *this;
deba@127
   710
    }
deba@127
   711
alpar@156
   712
    /// \brief Set \c \@arcs section to be read
alpar@156
   713
    ///
alpar@156
   714
    /// Set \c \@arcs section to be read
deba@127
   715
    DigraphReader& arcs(const std::string& caption) {
deba@127
   716
      _arcs_caption = caption;
deba@127
   717
      return *this;
deba@127
   718
    }
deba@127
   719
alpar@156
   720
    /// \brief Set \c \@attributes section to be read
alpar@156
   721
    ///
alpar@156
   722
    /// Set \c \@attributes section to be read
deba@127
   723
    DigraphReader& attributes(const std::string& caption) {
deba@127
   724
      _attributes_caption = caption;
deba@127
   725
      return *this;
deba@127
   726
    }
deba@127
   727
alpar@156
   728
    /// @}
alpar@156
   729
alpar@156
   730
    /// \name Using previously constructed node or arc set
alpar@156
   731
    /// @{
alpar@156
   732
alpar@156
   733
    /// \brief Use previously constructed node set
alpar@156
   734
    ///
alpar@156
   735
    /// Use previously constructed node set, and specify the node
alpar@156
   736
    /// label map.
deba@127
   737
    template <typename Map>
deba@127
   738
    DigraphReader& useNodes(const Map& map) {
deba@127
   739
      checkConcept<concepts::ReadMap<Node, typename Map::Value>, Map>();
alpar@209
   740
      LEMON_ASSERT(!_use_nodes, "Multiple usage of useNodes() member");
deba@127
   741
      _use_nodes = true;
deba@127
   742
      _writer_bits::DefaultConverter<typename Map::Value> converter;
deba@127
   743
      for (NodeIt n(_digraph); n != INVALID; ++n) {
alpar@209
   744
        _node_index.insert(std::make_pair(converter(map[n]), n));
deba@127
   745
      }
deba@127
   746
      return *this;
deba@127
   747
    }
deba@127
   748
alpar@156
   749
    /// \brief Use previously constructed node set
alpar@156
   750
    ///
alpar@156
   751
    /// Use previously constructed node set, and specify the node
alpar@156
   752
    /// label map and a functor which converts the label map values to
kpeter@192
   753
    /// \c std::string.
deba@127
   754
    template <typename Map, typename Converter>
alpar@209
   755
    DigraphReader& useNodes(const Map& map,
alpar@209
   756
                            const Converter& converter = Converter()) {
deba@127
   757
      checkConcept<concepts::ReadMap<Node, typename Map::Value>, Map>();
alpar@209
   758
      LEMON_ASSERT(!_use_nodes, "Multiple usage of useNodes() member");
deba@127
   759
      _use_nodes = true;
deba@127
   760
      for (NodeIt n(_digraph); n != INVALID; ++n) {
alpar@209
   761
        _node_index.insert(std::make_pair(converter(map[n]), n));
deba@127
   762
      }
deba@127
   763
      return *this;
deba@127
   764
    }
deba@127
   765
alpar@156
   766
    /// \brief Use previously constructed arc set
alpar@156
   767
    ///
alpar@156
   768
    /// Use previously constructed arc set, and specify the arc
alpar@156
   769
    /// label map.
deba@127
   770
    template <typename Map>
deba@127
   771
    DigraphReader& useArcs(const Map& map) {
deba@127
   772
      checkConcept<concepts::ReadMap<Arc, typename Map::Value>, Map>();
deba@127
   773
      LEMON_ASSERT(!_use_arcs, "Multiple usage of useArcs() member");
deba@127
   774
      _use_arcs = true;
deba@127
   775
      _writer_bits::DefaultConverter<typename Map::Value> converter;
deba@127
   776
      for (ArcIt a(_digraph); a != INVALID; ++a) {
alpar@209
   777
        _arc_index.insert(std::make_pair(converter(map[a]), a));
deba@127
   778
      }
deba@127
   779
      return *this;
deba@127
   780
    }
deba@127
   781
alpar@156
   782
    /// \brief Use previously constructed arc set
alpar@156
   783
    ///
alpar@156
   784
    /// Use previously constructed arc set, and specify the arc
alpar@156
   785
    /// label map and a functor which converts the label map values to
kpeter@192
   786
    /// \c std::string.
deba@127
   787
    template <typename Map, typename Converter>
alpar@209
   788
    DigraphReader& useArcs(const Map& map,
alpar@209
   789
                           const Converter& converter = Converter()) {
deba@127
   790
      checkConcept<concepts::ReadMap<Arc, typename Map::Value>, Map>();
alpar@209
   791
      LEMON_ASSERT(!_use_arcs, "Multiple usage of useArcs() member");
deba@127
   792
      _use_arcs = true;
deba@127
   793
      for (ArcIt a(_digraph); a != INVALID; ++a) {
alpar@209
   794
        _arc_index.insert(std::make_pair(converter(map[a]), a));
deba@127
   795
      }
deba@127
   796
      return *this;
deba@127
   797
    }
deba@127
   798
deba@188
   799
    /// \brief Skips the reading of node section
deba@188
   800
    ///
deba@188
   801
    /// Omit the reading of the node section. This implies that each node
kpeter@192
   802
    /// map reading rule will be abandoned, and the nodes of the graph
deba@188
   803
    /// will not be constructed, which usually cause that the arc set
kpeter@192
   804
    /// could not be read due to lack of node name resolving.
kpeter@192
   805
    /// Therefore \c skipArcs() function should also be used, or
kpeter@192
   806
    /// \c useNodes() should be used to specify the label of the nodes.
deba@188
   807
    DigraphReader& skipNodes() {
alpar@209
   808
      LEMON_ASSERT(!_skip_nodes, "Skip nodes already set");
deba@188
   809
      _skip_nodes = true;
deba@188
   810
      return *this;
deba@188
   811
    }
deba@188
   812
deba@188
   813
    /// \brief Skips the reading of arc section
deba@188
   814
    ///
deba@188
   815
    /// Omit the reading of the arc section. This implies that each arc
kpeter@192
   816
    /// map reading rule will be abandoned, and the arcs of the graph
deba@188
   817
    /// will not be constructed.
deba@188
   818
    DigraphReader& skipArcs() {
alpar@209
   819
      LEMON_ASSERT(!_skip_arcs, "Skip arcs already set");
deba@188
   820
      _skip_arcs = true;
deba@188
   821
      return *this;
deba@188
   822
    }
deba@188
   823
alpar@156
   824
    /// @}
alpar@156
   825
deba@127
   826
  private:
deba@127
   827
deba@127
   828
    bool readLine() {
deba@127
   829
      std::string str;
deba@127
   830
      while(++line_num, std::getline(*_is, str)) {
alpar@209
   831
        line.clear(); line.str(str);
alpar@209
   832
        char c;
alpar@209
   833
        if (line >> std::ws >> c && c != '#') {
alpar@209
   834
          line.putback(c);
alpar@209
   835
          return true;
alpar@209
   836
        }
deba@127
   837
      }
deba@127
   838
      return false;
deba@127
   839
    }
deba@127
   840
deba@127
   841
    bool readSuccess() {
deba@127
   842
      return static_cast<bool>(*_is);
deba@127
   843
    }
alpar@209
   844
deba@127
   845
    void skipSection() {
deba@127
   846
      char c;
deba@127
   847
      while (readSuccess() && line >> c && c != '@') {
alpar@209
   848
        readLine();
deba@127
   849
      }
deba@347
   850
      if (readSuccess()) {
deba@347
   851
        line.putback(c);
deba@347
   852
      }
deba@127
   853
    }
deba@127
   854
deba@127
   855
    void readNodes() {
deba@127
   856
deba@127
   857
      std::vector<int> map_index(_node_maps.size());
deba@127
   858
      int map_num, label_index;
deba@127
   859
deba@186
   860
      char c;
deba@186
   861
      if (!readLine() || !(line >> c) || c == '@') {
alpar@209
   862
        if (readSuccess() && line) line.putback(c);
alpar@209
   863
        if (!_node_maps.empty())
deba@290
   864
          throw FormatError("Cannot find map names");
alpar@209
   865
        return;
deba@186
   866
      }
deba@186
   867
      line.putback(c);
deba@186
   868
deba@127
   869
      {
alpar@209
   870
        std::map<std::string, int> maps;
alpar@209
   871
alpar@209
   872
        std::string map;
alpar@209
   873
        int index = 0;
alpar@209
   874
        while (_reader_bits::readToken(line, map)) {
alpar@209
   875
          if (maps.find(map) != maps.end()) {
alpar@209
   876
            std::ostringstream msg;
alpar@209
   877
            msg << "Multiple occurence of node map: " << map;
deba@290
   878
            throw FormatError(msg.str());
alpar@209
   879
          }
alpar@209
   880
          maps.insert(std::make_pair(map, index));
alpar@209
   881
          ++index;
alpar@209
   882
        }
alpar@209
   883
alpar@209
   884
        for (int i = 0; i < static_cast<int>(_node_maps.size()); ++i) {
alpar@209
   885
          std::map<std::string, int>::iterator jt =
alpar@209
   886
            maps.find(_node_maps[i].first);
alpar@209
   887
          if (jt == maps.end()) {
alpar@209
   888
            std::ostringstream msg;
kpeter@291
   889
            msg << "Map not found: " << _node_maps[i].first;
deba@290
   890
            throw FormatError(msg.str());
alpar@209
   891
          }
alpar@209
   892
          map_index[i] = jt->second;
alpar@209
   893
        }
alpar@209
   894
alpar@209
   895
        {
alpar@209
   896
          std::map<std::string, int>::iterator jt = maps.find("label");
alpar@209
   897
          if (jt != maps.end()) {
alpar@209
   898
            label_index = jt->second;
alpar@209
   899
          } else {
alpar@209
   900
            label_index = -1;
alpar@209
   901
          }
alpar@209
   902
        }
alpar@209
   903
        map_num = maps.size();
deba@127
   904
      }
deba@127
   905
deba@127
   906
      while (readLine() && line >> c && c != '@') {
alpar@209
   907
        line.putback(c);
alpar@209
   908
alpar@209
   909
        std::vector<std::string> tokens(map_num);
alpar@209
   910
        for (int i = 0; i < map_num; ++i) {
alpar@209
   911
          if (!_reader_bits::readToken(line, tokens[i])) {
alpar@209
   912
            std::ostringstream msg;
alpar@209
   913
            msg << "Column not found (" << i + 1 << ")";
deba@290
   914
            throw FormatError(msg.str());
alpar@209
   915
          }
alpar@209
   916
        }
alpar@209
   917
        if (line >> std::ws >> c)
kpeter@291
   918
          throw FormatError("Extra character at the end of line");
alpar@209
   919
alpar@209
   920
        Node n;
alpar@209
   921
        if (!_use_nodes) {
alpar@209
   922
          n = _digraph.addNode();
alpar@209
   923
          if (label_index != -1)
alpar@209
   924
            _node_index.insert(std::make_pair(tokens[label_index], n));
alpar@209
   925
        } else {
alpar@209
   926
          if (label_index == -1)
kpeter@291
   927
            throw FormatError("Label map not found");
alpar@209
   928
          typename std::map<std::string, Node>::iterator it =
alpar@209
   929
            _node_index.find(tokens[label_index]);
alpar@209
   930
          if (it == _node_index.end()) {
alpar@209
   931
            std::ostringstream msg;
alpar@209
   932
            msg << "Node with label not found: " << tokens[label_index];
deba@290
   933
            throw FormatError(msg.str());
alpar@209
   934
          }
alpar@209
   935
          n = it->second;
alpar@209
   936
        }
alpar@209
   937
alpar@209
   938
        for (int i = 0; i < static_cast<int>(_node_maps.size()); ++i) {
alpar@209
   939
          _node_maps[i].second->set(n, tokens[map_index[i]]);
alpar@209
   940
        }
deba@127
   941
deba@127
   942
      }
deba@127
   943
      if (readSuccess()) {
alpar@209
   944
        line.putback(c);
deba@127
   945
      }
deba@127
   946
    }
deba@127
   947
deba@127
   948
    void readArcs() {
deba@127
   949
deba@127
   950
      std::vector<int> map_index(_arc_maps.size());
deba@127
   951
      int map_num, label_index;
deba@127
   952
deba@186
   953
      char c;
deba@186
   954
      if (!readLine() || !(line >> c) || c == '@') {
alpar@209
   955
        if (readSuccess() && line) line.putback(c);
alpar@209
   956
        if (!_arc_maps.empty())
deba@290
   957
          throw FormatError("Cannot find map names");
alpar@209
   958
        return;
deba@186
   959
      }
deba@186
   960
      line.putback(c);
alpar@209
   961
deba@127
   962
      {
alpar@209
   963
        std::map<std::string, int> maps;
alpar@209
   964
alpar@209
   965
        std::string map;
alpar@209
   966
        int index = 0;
alpar@209
   967
        while (_reader_bits::readToken(line, map)) {
alpar@422
   968
          if(map == "-") {
alpar@422
   969
              if(index!=0)
alpar@422
   970
                throw FormatError("'-' is not allowed as a map name");
alpar@422
   971
              else if (line >> std::ws >> c)
alpar@422
   972
                throw FormatError("Extra character at the end of line");
alpar@422
   973
              else break;
alpar@422
   974
            }
alpar@209
   975
          if (maps.find(map) != maps.end()) {
alpar@209
   976
            std::ostringstream msg;
alpar@209
   977
            msg << "Multiple occurence of arc map: " << map;
deba@290
   978
            throw FormatError(msg.str());
alpar@209
   979
          }
alpar@209
   980
          maps.insert(std::make_pair(map, index));
alpar@209
   981
          ++index;
alpar@209
   982
        }
alpar@209
   983
alpar@209
   984
        for (int i = 0; i < static_cast<int>(_arc_maps.size()); ++i) {
alpar@209
   985
          std::map<std::string, int>::iterator jt =
alpar@209
   986
            maps.find(_arc_maps[i].first);
alpar@209
   987
          if (jt == maps.end()) {
alpar@209
   988
            std::ostringstream msg;
kpeter@291
   989
            msg << "Map not found: " << _arc_maps[i].first;
deba@290
   990
            throw FormatError(msg.str());
alpar@209
   991
          }
alpar@209
   992
          map_index[i] = jt->second;
alpar@209
   993
        }
alpar@209
   994
alpar@209
   995
        {
alpar@209
   996
          std::map<std::string, int>::iterator jt = maps.find("label");
alpar@209
   997
          if (jt != maps.end()) {
alpar@209
   998
            label_index = jt->second;
alpar@209
   999
          } else {
alpar@209
  1000
            label_index = -1;
alpar@209
  1001
          }
alpar@209
  1002
        }
alpar@209
  1003
        map_num = maps.size();
deba@127
  1004
      }
deba@127
  1005
deba@127
  1006
      while (readLine() && line >> c && c != '@') {
alpar@209
  1007
        line.putback(c);
alpar@209
  1008
alpar@209
  1009
        std::string source_token;
alpar@209
  1010
        std::string target_token;
alpar@209
  1011
alpar@209
  1012
        if (!_reader_bits::readToken(line, source_token))
deba@290
  1013
          throw FormatError("Source not found");
alpar@209
  1014
alpar@209
  1015
        if (!_reader_bits::readToken(line, target_token))
deba@290
  1016
          throw FormatError("Target not found");
alpar@209
  1017
alpar@209
  1018
        std::vector<std::string> tokens(map_num);
alpar@209
  1019
        for (int i = 0; i < map_num; ++i) {
alpar@209
  1020
          if (!_reader_bits::readToken(line, tokens[i])) {
alpar@209
  1021
            std::ostringstream msg;
alpar@209
  1022
            msg << "Column not found (" << i + 1 << ")";
deba@290
  1023
            throw FormatError(msg.str());
alpar@209
  1024
          }
alpar@209
  1025
        }
alpar@209
  1026
        if (line >> std::ws >> c)
kpeter@291
  1027
          throw FormatError("Extra character at the end of line");
alpar@209
  1028
alpar@209
  1029
        Arc a;
alpar@209
  1030
        if (!_use_arcs) {
deba@127
  1031
deba@127
  1032
          typename NodeIndex::iterator it;
alpar@209
  1033
deba@127
  1034
          it = _node_index.find(source_token);
deba@127
  1035
          if (it == _node_index.end()) {
deba@127
  1036
            std::ostringstream msg;
deba@127
  1037
            msg << "Item not found: " << source_token;
deba@290
  1038
            throw FormatError(msg.str());
deba@127
  1039
          }
deba@127
  1040
          Node source = it->second;
deba@127
  1041
deba@127
  1042
          it = _node_index.find(target_token);
alpar@209
  1043
          if (it == _node_index.end()) {
alpar@209
  1044
            std::ostringstream msg;
deba@127
  1045
            msg << "Item not found: " << target_token;
deba@290
  1046
            throw FormatError(msg.str());
alpar@209
  1047
          }
alpar@209
  1048
          Node target = it->second;
alpar@209
  1049
alpar@209
  1050
          a = _digraph.addArc(source, target);
alpar@209
  1051
          if (label_index != -1)
alpar@209
  1052
            _arc_index.insert(std::make_pair(tokens[label_index], a));
alpar@209
  1053
        } else {
alpar@209
  1054
          if (label_index == -1)
kpeter@291
  1055
            throw FormatError("Label map not found");
alpar@209
  1056
          typename std::map<std::string, Arc>::iterator it =
alpar@209
  1057
            _arc_index.find(tokens[label_index]);
alpar@209
  1058
          if (it == _arc_index.end()) {
alpar@209
  1059
            std::ostringstream msg;
alpar@209
  1060
            msg << "Arc with label not found: " << tokens[label_index];
deba@290
  1061
            throw FormatError(msg.str());
alpar@209
  1062
          }
alpar@209
  1063
          a = it->second;
alpar@209
  1064
        }
alpar@209
  1065
alpar@209
  1066
        for (int i = 0; i < static_cast<int>(_arc_maps.size()); ++i) {
alpar@209
  1067
          _arc_maps[i].second->set(a, tokens[map_index[i]]);
alpar@209
  1068
        }
deba@127
  1069
deba@127
  1070
      }
deba@127
  1071
      if (readSuccess()) {
alpar@209
  1072
        line.putback(c);
deba@127
  1073
      }
deba@127
  1074
    }
deba@127
  1075
deba@127
  1076
    void readAttributes() {
deba@127
  1077
deba@127
  1078
      std::set<std::string> read_attr;
deba@127
  1079
deba@127
  1080
      char c;
deba@127
  1081
      while (readLine() && line >> c && c != '@') {
alpar@209
  1082
        line.putback(c);
alpar@209
  1083
alpar@209
  1084
        std::string attr, token;
alpar@209
  1085
        if (!_reader_bits::readToken(line, attr))
deba@290
  1086
          throw FormatError("Attribute name not found");
alpar@209
  1087
        if (!_reader_bits::readToken(line, token))
deba@290
  1088
          throw FormatError("Attribute value not found");
alpar@209
  1089
        if (line >> c)
kpeter@291
  1090
          throw FormatError("Extra character at the end of line");
alpar@209
  1091
alpar@209
  1092
        {
alpar@209
  1093
          std::set<std::string>::iterator it = read_attr.find(attr);
alpar@209
  1094
          if (it != read_attr.end()) {
alpar@209
  1095
            std::ostringstream msg;
kpeter@291
  1096
            msg << "Multiple occurence of attribute: " << attr;
deba@290
  1097
            throw FormatError(msg.str());
alpar@209
  1098
          }
alpar@209
  1099
          read_attr.insert(attr);
alpar@209
  1100
        }
alpar@209
  1101
alpar@209
  1102
        {
alpar@209
  1103
          typename Attributes::iterator it = _attributes.lower_bound(attr);
alpar@209
  1104
          while (it != _attributes.end() && it->first == attr) {
alpar@209
  1105
            it->second->set(token);
alpar@209
  1106
            ++it;
alpar@209
  1107
          }
alpar@209
  1108
        }
deba@127
  1109
deba@127
  1110
      }
deba@127
  1111
      if (readSuccess()) {
alpar@209
  1112
        line.putback(c);
deba@127
  1113
      }
deba@127
  1114
      for (typename Attributes::iterator it = _attributes.begin();
alpar@209
  1115
           it != _attributes.end(); ++it) {
alpar@209
  1116
        if (read_attr.find(it->first) == read_attr.end()) {
alpar@209
  1117
          std::ostringstream msg;
kpeter@291
  1118
          msg << "Attribute not found: " << it->first;
deba@290
  1119
          throw FormatError(msg.str());
alpar@209
  1120
        }
deba@127
  1121
      }
deba@127
  1122
    }
deba@127
  1123
deba@127
  1124
  public:
alpar@156
  1125
alpar@209
  1126
    /// \name Execution of the reader
alpar@156
  1127
    /// @{
alpar@156
  1128
alpar@156
  1129
    /// \brief Start the batch processing
alpar@156
  1130
    ///
alpar@156
  1131
    /// This function starts the batch processing
deba@127
  1132
    void run() {
deba@127
  1133
      LEMON_ASSERT(_is != 0, "This reader assigned to an other reader");
alpar@209
  1134
deba@188
  1135
      bool nodes_done = _skip_nodes;
deba@188
  1136
      bool arcs_done = _skip_arcs;
deba@127
  1137
      bool attributes_done = false;
deba@127
  1138
alpar@209
  1139
      line_num = 0;
deba@127
  1140
      readLine();
deba@172
  1141
      skipSection();
deba@127
  1142
deba@127
  1143
      while (readSuccess()) {
alpar@209
  1144
        try {
alpar@209
  1145
          char c;
alpar@209
  1146
          std::string section, caption;
alpar@209
  1147
          line >> c;
alpar@209
  1148
          _reader_bits::readToken(line, section);
alpar@209
  1149
          _reader_bits::readToken(line, caption);
alpar@209
  1150
alpar@209
  1151
          if (line >> c)
kpeter@291
  1152
            throw FormatError("Extra character at the end of line");
alpar@209
  1153
alpar@209
  1154
          if (section == "nodes" && !nodes_done) {
alpar@209
  1155
            if (_nodes_caption.empty() || _nodes_caption == caption) {
alpar@209
  1156
              readNodes();
alpar@209
  1157
              nodes_done = true;
alpar@209
  1158
            }
alpar@209
  1159
          } else if ((section == "arcs" || section == "edges") &&
alpar@209
  1160
                     !arcs_done) {
alpar@209
  1161
            if (_arcs_caption.empty() || _arcs_caption == caption) {
alpar@209
  1162
              readArcs();
alpar@209
  1163
              arcs_done = true;
alpar@209
  1164
            }
alpar@209
  1165
          } else if (section == "attributes" && !attributes_done) {
alpar@209
  1166
            if (_attributes_caption.empty() || _attributes_caption == caption) {
alpar@209
  1167
              readAttributes();
alpar@209
  1168
              attributes_done = true;
alpar@209
  1169
            }
alpar@209
  1170
          } else {
alpar@209
  1171
            readLine();
alpar@209
  1172
            skipSection();
alpar@209
  1173
          }
deba@290
  1174
        } catch (FormatError& error) {
alpar@209
  1175
          error.line(line_num);
deba@290
  1176
          error.file(_filename);
alpar@209
  1177
          throw;
alpar@209
  1178
        }
deba@127
  1179
      }
deba@127
  1180
deba@127
  1181
      if (!nodes_done) {
deba@290
  1182
        throw FormatError("Section @nodes not found");
deba@127
  1183
      }
deba@127
  1184
deba@127
  1185
      if (!arcs_done) {
deba@290
  1186
        throw FormatError("Section @arcs not found");
deba@127
  1187
      }
deba@127
  1188
deba@127
  1189
      if (!attributes_done && !_attributes.empty()) {
deba@290
  1190
        throw FormatError("Section @attributes not found");
deba@127
  1191
      }
deba@127
  1192
deba@127
  1193
    }
alpar@156
  1194
alpar@156
  1195
    /// @}
alpar@209
  1196
deba@127
  1197
  };
deba@127
  1198
kpeter@374
  1199
  /// \brief Return a \ref DigraphReader class
kpeter@374
  1200
  ///
kpeter@374
  1201
  /// This function just returns a \ref DigraphReader class.
kpeter@374
  1202
  /// \relates DigraphReader
kpeter@374
  1203
  template <typename Digraph>
kpeter@374
  1204
  DigraphReader<Digraph> digraphReader(Digraph& digraph, std::istream& is) {
kpeter@374
  1205
    DigraphReader<Digraph> tmp(digraph, is);
kpeter@374
  1206
    return tmp;
kpeter@374
  1207
  }
kpeter@374
  1208
kpeter@374
  1209
  /// \brief Return a \ref DigraphReader class
kpeter@374
  1210
  ///
kpeter@374
  1211
  /// This function just returns a \ref DigraphReader class.
kpeter@374
  1212
  /// \relates DigraphReader
kpeter@374
  1213
  template <typename Digraph>
kpeter@374
  1214
  DigraphReader<Digraph> digraphReader(Digraph& digraph,
kpeter@374
  1215
                                       const std::string& fn) {
kpeter@374
  1216
    DigraphReader<Digraph> tmp(digraph, fn);
kpeter@374
  1217
    return tmp;
kpeter@374
  1218
  }
kpeter@374
  1219
kpeter@374
  1220
  /// \brief Return a \ref DigraphReader class
kpeter@374
  1221
  ///
kpeter@374
  1222
  /// This function just returns a \ref DigraphReader class.
kpeter@374
  1223
  /// \relates DigraphReader
kpeter@374
  1224
  template <typename Digraph>
kpeter@374
  1225
  DigraphReader<Digraph> digraphReader(Digraph& digraph, const char* fn) {
kpeter@374
  1226
    DigraphReader<Digraph> tmp(digraph, fn);
kpeter@374
  1227
    return tmp;
kpeter@374
  1228
  }
kpeter@374
  1229
ladanyi@307
  1230
  template <typename Graph>
ladanyi@307
  1231
  class GraphReader;
kpeter@374
  1232
 
ladanyi@307
  1233
  template <typename Graph>
kpeter@374
  1234
  GraphReader<Graph> graphReader(Graph& graph, 
kpeter@374
  1235
                                 std::istream& is = std::cin);
ladanyi@307
  1236
  template <typename Graph>
kpeter@374
  1237
  GraphReader<Graph> graphReader(Graph& graph, const std::string& fn);
ladanyi@307
  1238
  template <typename Graph>
kpeter@374
  1239
  GraphReader<Graph> graphReader(Graph& graph, const char *fn);
deba@165
  1240
deba@165
  1241
  /// \ingroup lemon_io
alpar@209
  1242
  ///
kpeter@192
  1243
  /// \brief \ref lgf-format "LGF" reader for undirected graphs
deba@165
  1244
  ///
deba@165
  1245
  /// This utility reads an \ref lgf-format "LGF" file.
kpeter@192
  1246
  ///
kpeter@192
  1247
  /// It can be used almost the same way as \c DigraphReader.
kpeter@192
  1248
  /// The only difference is that this class can handle edges and
kpeter@192
  1249
  /// edge maps as well as arcs and arc maps.
deba@201
  1250
  ///
deba@201
  1251
  /// The columns in the \c \@edges (or \c \@arcs) section are the
deba@201
  1252
  /// edge maps. However, if there are two maps with the same name
deba@201
  1253
  /// prefixed with \c '+' and \c '-', then these can be read into an
deba@201
  1254
  /// arc map.  Similarly, an attribute can be read into an arc, if
deba@201
  1255
  /// it's value is an edge label prefixed with \c '+' or \c '-'.
deba@165
  1256
  template <typename _Graph>
deba@165
  1257
  class GraphReader {
deba@165
  1258
  public:
deba@165
  1259
deba@165
  1260
    typedef _Graph Graph;
deba@165
  1261
    TEMPLATE_GRAPH_TYPEDEFS(Graph);
alpar@209
  1262
deba@165
  1263
  private:
deba@165
  1264
deba@165
  1265
    std::istream* _is;
deba@165
  1266
    bool local_is;
deba@290
  1267
    std::string _filename;
deba@165
  1268
deba@165
  1269
    Graph& _graph;
deba@165
  1270
deba@165
  1271
    std::string _nodes_caption;
deba@165
  1272
    std::string _edges_caption;
deba@165
  1273
    std::string _attributes_caption;
deba@165
  1274
deba@165
  1275
    typedef std::map<std::string, Node> NodeIndex;
deba@165
  1276
    NodeIndex _node_index;
deba@165
  1277
    typedef std::map<std::string, Edge> EdgeIndex;
deba@165
  1278
    EdgeIndex _edge_index;
alpar@209
  1279
alpar@209
  1280
    typedef std::vector<std::pair<std::string,
alpar@209
  1281
      _reader_bits::MapStorageBase<Node>*> > NodeMaps;
alpar@209
  1282
    NodeMaps _node_maps;
deba@165
  1283
deba@165
  1284
    typedef std::vector<std::pair<std::string,
deba@165
  1285
      _reader_bits::MapStorageBase<Edge>*> > EdgeMaps;
deba@165
  1286
    EdgeMaps _edge_maps;
deba@165
  1287
alpar@209
  1288
    typedef std::multimap<std::string, _reader_bits::ValueStorageBase*>
deba@165
  1289
      Attributes;
deba@165
  1290
    Attributes _attributes;
deba@165
  1291
deba@165
  1292
    bool _use_nodes;
deba@165
  1293
    bool _use_edges;
deba@165
  1294
deba@188
  1295
    bool _skip_nodes;
deba@188
  1296
    bool _skip_edges;
deba@188
  1297
deba@165
  1298
    int line_num;
deba@165
  1299
    std::istringstream line;
deba@165
  1300
deba@165
  1301
  public:
deba@165
  1302
deba@165
  1303
    /// \brief Constructor
deba@165
  1304
    ///
kpeter@192
  1305
    /// Construct an undirected graph reader, which reads from the given
deba@165
  1306
    /// input stream.
kpeter@293
  1307
    GraphReader(Graph& graph, std::istream& is = std::cin)
deba@165
  1308
      : _is(&is), local_is(false), _graph(graph),
alpar@209
  1309
        _use_nodes(false), _use_edges(false),
alpar@209
  1310
        _skip_nodes(false), _skip_edges(false) {}
deba@165
  1311
deba@165
  1312
    /// \brief Constructor
deba@165
  1313
    ///
kpeter@192
  1314
    /// Construct an undirected graph reader, which reads from the given
deba@165
  1315
    /// file.
kpeter@293
  1316
    GraphReader(Graph& graph, const std::string& fn)
deba@290
  1317
      : _is(new std::ifstream(fn.c_str())), local_is(true),
deba@290
  1318
        _filename(fn), _graph(graph),
kpeter@212
  1319
        _use_nodes(false), _use_edges(false),
deba@290
  1320
        _skip_nodes(false), _skip_edges(false) {
deba@295
  1321
      if (!(*_is)) {
deba@295
  1322
        delete _is;
deba@295
  1323
        throw IoError("Cannot open file", fn);
deba@295
  1324
      }
deba@290
  1325
    }
alpar@209
  1326
deba@165
  1327
    /// \brief Constructor
deba@165
  1328
    ///
kpeter@192
  1329
    /// Construct an undirected graph reader, which reads from the given
deba@165
  1330
    /// file.
kpeter@293
  1331
    GraphReader(Graph& graph, const char* fn)
deba@290
  1332
      : _is(new std::ifstream(fn)), local_is(true),
deba@290
  1333
        _filename(fn), _graph(graph),
kpeter@212
  1334
        _use_nodes(false), _use_edges(false),
deba@290
  1335
        _skip_nodes(false), _skip_edges(false) {
deba@295
  1336
      if (!(*_is)) {
deba@295
  1337
        delete _is;
deba@295
  1338
        throw IoError("Cannot open file", fn);
deba@295
  1339
      }
deba@290
  1340
    }
deba@165
  1341
deba@165
  1342
    /// \brief Destructor
deba@165
  1343
    ~GraphReader() {
alpar@209
  1344
      for (typename NodeMaps::iterator it = _node_maps.begin();
alpar@209
  1345
           it != _node_maps.end(); ++it) {
alpar@209
  1346
        delete it->second;
deba@165
  1347
      }
deba@165
  1348
alpar@209
  1349
      for (typename EdgeMaps::iterator it = _edge_maps.begin();
alpar@209
  1350
           it != _edge_maps.end(); ++it) {
alpar@209
  1351
        delete it->second;
deba@165
  1352
      }
deba@165
  1353
alpar@209
  1354
      for (typename Attributes::iterator it = _attributes.begin();
alpar@209
  1355
           it != _attributes.end(); ++it) {
alpar@209
  1356
        delete it->second;
deba@165
  1357
      }
deba@165
  1358
deba@165
  1359
      if (local_is) {
alpar@209
  1360
        delete _is;
deba@165
  1361
      }
deba@165
  1362
deba@165
  1363
    }
deba@165
  1364
deba@165
  1365
  private:
kpeter@374
  1366
    template <typename GR>
kpeter@374
  1367
    friend GraphReader<GR> graphReader(GR& graph, std::istream& is);
kpeter@374
  1368
    template <typename GR>
kpeter@374
  1369
    friend GraphReader<GR> graphReader(GR& graph, const std::string& fn); 
kpeter@374
  1370
    template <typename GR>
kpeter@374
  1371
    friend GraphReader<GR> graphReader(GR& graph, const char *fn);
alpar@209
  1372
alpar@209
  1373
    GraphReader(GraphReader& other)
deba@190
  1374
      : _is(other._is), local_is(other.local_is), _graph(other._graph),
alpar@209
  1375
        _use_nodes(other._use_nodes), _use_edges(other._use_edges),
alpar@209
  1376
        _skip_nodes(other._skip_nodes), _skip_edges(other._skip_edges) {
deba@190
  1377
deba@190
  1378
      other._is = 0;
deba@190
  1379
      other.local_is = false;
alpar@209
  1380
deba@190
  1381
      _node_index.swap(other._node_index);
deba@190
  1382
      _edge_index.swap(other._edge_index);
deba@190
  1383
deba@190
  1384
      _node_maps.swap(other._node_maps);
deba@190
  1385
      _edge_maps.swap(other._edge_maps);
deba@190
  1386
      _attributes.swap(other._attributes);
deba@190
  1387
deba@190
  1388
      _nodes_caption = other._nodes_caption;
deba@190
  1389
      _edges_caption = other._edges_caption;
deba@190
  1390
      _attributes_caption = other._attributes_caption;
deba@190
  1391
deba@190
  1392
    }
deba@190
  1393
deba@165
  1394
    GraphReader& operator=(const GraphReader&);
deba@165
  1395
deba@165
  1396
  public:
deba@165
  1397
deba@165
  1398
    /// \name Reading rules
deba@165
  1399
    /// @{
alpar@209
  1400
deba@165
  1401
    /// \brief Node map reading rule
deba@165
  1402
    ///
deba@165
  1403
    /// Add a node map reading rule to the reader.
deba@165
  1404
    template <typename Map>
deba@165
  1405
    GraphReader& nodeMap(const std::string& caption, Map& map) {
deba@165
  1406
      checkConcept<concepts::WriteMap<Node, typename Map::Value>, Map>();
alpar@209
  1407
      _reader_bits::MapStorageBase<Node>* storage =
alpar@209
  1408
        new _reader_bits::MapStorage<Node, Map>(map);
deba@165
  1409
      _node_maps.push_back(std::make_pair(caption, storage));
deba@165
  1410
      return *this;
deba@165
  1411
    }
deba@165
  1412
deba@165
  1413
    /// \brief Node map reading rule
deba@165
  1414
    ///
deba@165
  1415
    /// Add a node map reading rule with specialized converter to the
deba@165
  1416
    /// reader.
deba@165
  1417
    template <typename Map, typename Converter>
alpar@209
  1418
    GraphReader& nodeMap(const std::string& caption, Map& map,
alpar@209
  1419
                           const Converter& converter = Converter()) {
deba@165
  1420
      checkConcept<concepts::WriteMap<Node, typename Map::Value>, Map>();
alpar@209
  1421
      _reader_bits::MapStorageBase<Node>* storage =
alpar@209
  1422
        new _reader_bits::MapStorage<Node, Map, Converter>(map, converter);
deba@165
  1423
      _node_maps.push_back(std::make_pair(caption, storage));
deba@165
  1424
      return *this;
deba@165
  1425
    }
deba@165
  1426
deba@165
  1427
    /// \brief Edge map reading rule
deba@165
  1428
    ///
deba@165
  1429
    /// Add an edge map reading rule to the reader.
deba@165
  1430
    template <typename Map>
deba@165
  1431
    GraphReader& edgeMap(const std::string& caption, Map& map) {
deba@165
  1432
      checkConcept<concepts::WriteMap<Edge, typename Map::Value>, Map>();
alpar@209
  1433
      _reader_bits::MapStorageBase<Edge>* storage =
alpar@209
  1434
        new _reader_bits::MapStorage<Edge, Map>(map);
deba@165
  1435
      _edge_maps.push_back(std::make_pair(caption, storage));
deba@165
  1436
      return *this;
deba@165
  1437
    }
deba@165
  1438
deba@165
  1439
    /// \brief Edge map reading rule
deba@165
  1440
    ///
deba@165
  1441
    /// Add an edge map reading rule with specialized converter to the
deba@165
  1442
    /// reader.
deba@165
  1443
    template <typename Map, typename Converter>
alpar@209
  1444
    GraphReader& edgeMap(const std::string& caption, Map& map,
alpar@209
  1445
                          const Converter& converter = Converter()) {
deba@165
  1446
      checkConcept<concepts::WriteMap<Edge, typename Map::Value>, Map>();
alpar@209
  1447
      _reader_bits::MapStorageBase<Edge>* storage =
alpar@209
  1448
        new _reader_bits::MapStorage<Edge, Map, Converter>(map, converter);
deba@165
  1449
      _edge_maps.push_back(std::make_pair(caption, storage));
deba@165
  1450
      return *this;
deba@165
  1451
    }
deba@165
  1452
deba@165
  1453
    /// \brief Arc map reading rule
deba@165
  1454
    ///
deba@165
  1455
    /// Add an arc map reading rule to the reader.
deba@165
  1456
    template <typename Map>
deba@165
  1457
    GraphReader& arcMap(const std::string& caption, Map& map) {
deba@165
  1458
      checkConcept<concepts::WriteMap<Arc, typename Map::Value>, Map>();
alpar@209
  1459
      _reader_bits::MapStorageBase<Edge>* forward_storage =
alpar@209
  1460
        new _reader_bits::GraphArcMapStorage<Graph, true, Map>(_graph, map);
deba@165
  1461
      _edge_maps.push_back(std::make_pair('+' + caption, forward_storage));
alpar@209
  1462
      _reader_bits::MapStorageBase<Edge>* backward_storage =
alpar@209
  1463
        new _reader_bits::GraphArcMapStorage<Graph, false, Map>(_graph, map);
deba@165
  1464
      _edge_maps.push_back(std::make_pair('-' + caption, backward_storage));
deba@165
  1465
      return *this;
deba@165
  1466
    }
deba@165
  1467
deba@165
  1468
    /// \brief Arc map reading rule
deba@165
  1469
    ///
deba@165
  1470
    /// Add an arc map reading rule with specialized converter to the
deba@165
  1471
    /// reader.
deba@165
  1472
    template <typename Map, typename Converter>
alpar@209
  1473
    GraphReader& arcMap(const std::string& caption, Map& map,
alpar@209
  1474
                          const Converter& converter = Converter()) {
deba@165
  1475
      checkConcept<concepts::WriteMap<Arc, typename Map::Value>, Map>();
alpar@209
  1476
      _reader_bits::MapStorageBase<Edge>* forward_storage =
alpar@209
  1477
        new _reader_bits::GraphArcMapStorage<Graph, true, Map, Converter>
alpar@209
  1478
        (_graph, map, converter);
deba@165
  1479
      _edge_maps.push_back(std::make_pair('+' + caption, forward_storage));
alpar@209
  1480
      _reader_bits::MapStorageBase<Edge>* backward_storage =
alpar@209
  1481
        new _reader_bits::GraphArcMapStorage<Graph, false, Map, Converter>
alpar@209
  1482
        (_graph, map, converter);
deba@165
  1483
      _edge_maps.push_back(std::make_pair('-' + caption, backward_storage));
deba@165
  1484
      return *this;
deba@165
  1485
    }
deba@165
  1486
deba@165
  1487
    /// \brief Attribute reading rule
deba@165
  1488
    ///
deba@165
  1489
    /// Add an attribute reading rule to the reader.
deba@165
  1490
    template <typename Value>
deba@165
  1491
    GraphReader& attribute(const std::string& caption, Value& value) {
alpar@209
  1492
      _reader_bits::ValueStorageBase* storage =
alpar@209
  1493
        new _reader_bits::ValueStorage<Value>(value);
deba@165
  1494
      _attributes.insert(std::make_pair(caption, storage));
deba@165
  1495
      return *this;
deba@165
  1496
    }
deba@165
  1497
deba@165
  1498
    /// \brief Attribute reading rule
deba@165
  1499
    ///
deba@165
  1500
    /// Add an attribute reading rule with specialized converter to the
deba@165
  1501
    /// reader.
deba@165
  1502
    template <typename Value, typename Converter>
alpar@209
  1503
    GraphReader& attribute(const std::string& caption, Value& value,
alpar@209
  1504
                             const Converter& converter = Converter()) {
alpar@209
  1505
      _reader_bits::ValueStorageBase* storage =
alpar@209
  1506
        new _reader_bits::ValueStorage<Value, Converter>(value, converter);
deba@165
  1507
      _attributes.insert(std::make_pair(caption, storage));
deba@165
  1508
      return *this;
deba@165
  1509
    }
deba@165
  1510
deba@165
  1511
    /// \brief Node reading rule
deba@165
  1512
    ///
deba@165
  1513
    /// Add a node reading rule to reader.
deba@165
  1514
    GraphReader& node(const std::string& caption, Node& node) {
deba@165
  1515
      typedef _reader_bits::MapLookUpConverter<Node> Converter;
deba@165
  1516
      Converter converter(_node_index);
alpar@209
  1517
      _reader_bits::ValueStorageBase* storage =
alpar@209
  1518
        new _reader_bits::ValueStorage<Node, Converter>(node, converter);
deba@165
  1519
      _attributes.insert(std::make_pair(caption, storage));
deba@165
  1520
      return *this;
deba@165
  1521
    }
deba@165
  1522
deba@165
  1523
    /// \brief Edge reading rule
deba@165
  1524
    ///
deba@165
  1525
    /// Add an edge reading rule to reader.
deba@165
  1526
    GraphReader& edge(const std::string& caption, Edge& edge) {
deba@165
  1527
      typedef _reader_bits::MapLookUpConverter<Edge> Converter;
deba@165
  1528
      Converter converter(_edge_index);
alpar@209
  1529
      _reader_bits::ValueStorageBase* storage =
alpar@209
  1530
        new _reader_bits::ValueStorage<Edge, Converter>(edge, converter);
deba@165
  1531
      _attributes.insert(std::make_pair(caption, storage));
deba@165
  1532
      return *this;
deba@165
  1533
    }
deba@165
  1534
deba@165
  1535
    /// \brief Arc reading rule
deba@165
  1536
    ///
deba@165
  1537
    /// Add an arc reading rule to reader.
deba@165
  1538
    GraphReader& arc(const std::string& caption, Arc& arc) {
deba@165
  1539
      typedef _reader_bits::GraphArcLookUpConverter<Graph> Converter;
deba@165
  1540
      Converter converter(_graph, _edge_index);
alpar@209
  1541
      _reader_bits::ValueStorageBase* storage =
alpar@209
  1542
        new _reader_bits::ValueStorage<Arc, Converter>(arc, converter);
deba@165
  1543
      _attributes.insert(std::make_pair(caption, storage));
deba@165
  1544
      return *this;
deba@165
  1545
    }
deba@165
  1546
deba@165
  1547
    /// @}
deba@165
  1548
deba@165
  1549
    /// \name Select section by name
deba@165
  1550
    /// @{
deba@165
  1551
deba@165
  1552
    /// \brief Set \c \@nodes section to be read
deba@165
  1553
    ///
kpeter@192
  1554
    /// Set \c \@nodes section to be read.
deba@165
  1555
    GraphReader& nodes(const std::string& caption) {
deba@165
  1556
      _nodes_caption = caption;
deba@165
  1557
      return *this;
deba@165
  1558
    }
deba@165
  1559
deba@165
  1560
    /// \brief Set \c \@edges section to be read
deba@165
  1561
    ///
kpeter@192
  1562
    /// Set \c \@edges section to be read.
deba@165
  1563
    GraphReader& edges(const std::string& caption) {
deba@165
  1564
      _edges_caption = caption;
deba@165
  1565
      return *this;
deba@165
  1566
    }
deba@165
  1567
deba@165
  1568
    /// \brief Set \c \@attributes section to be read
deba@165
  1569
    ///
kpeter@192
  1570
    /// Set \c \@attributes section to be read.
deba@165
  1571
    GraphReader& attributes(const std::string& caption) {
deba@165
  1572
      _attributes_caption = caption;
deba@165
  1573
      return *this;
deba@165
  1574
    }
deba@165
  1575
deba@165
  1576
    /// @}
deba@165
  1577
deba@165
  1578
    /// \name Using previously constructed node or edge set
deba@165
  1579
    /// @{
deba@165
  1580
deba@165
  1581
    /// \brief Use previously constructed node set
deba@165
  1582
    ///
deba@165
  1583
    /// Use previously constructed node set, and specify the node
deba@165
  1584
    /// label map.
deba@165
  1585
    template <typename Map>
deba@165
  1586
    GraphReader& useNodes(const Map& map) {
deba@165
  1587
      checkConcept<concepts::ReadMap<Node, typename Map::Value>, Map>();
alpar@209
  1588
      LEMON_ASSERT(!_use_nodes, "Multiple usage of useNodes() member");
deba@165
  1589
      _use_nodes = true;
deba@165
  1590
      _writer_bits::DefaultConverter<typename Map::Value> converter;
deba@165
  1591
      for (NodeIt n(_graph); n != INVALID; ++n) {
alpar@209
  1592
        _node_index.insert(std::make_pair(converter(map[n]), n));
deba@165
  1593
      }
deba@165
  1594
      return *this;
deba@165
  1595
    }
deba@165
  1596
deba@165
  1597
    /// \brief Use previously constructed node set
deba@165
  1598
    ///
deba@165
  1599
    /// Use previously constructed node set, and specify the node
deba@165
  1600
    /// label map and a functor which converts the label map values to
kpeter@192
  1601
    /// \c std::string.
deba@165
  1602
    template <typename Map, typename Converter>
alpar@209
  1603
    GraphReader& useNodes(const Map& map,
alpar@209
  1604
                            const Converter& converter = Converter()) {
deba@165
  1605
      checkConcept<concepts::ReadMap<Node, typename Map::Value>, Map>();
alpar@209
  1606
      LEMON_ASSERT(!_use_nodes, "Multiple usage of useNodes() member");
deba@165
  1607
      _use_nodes = true;
deba@165
  1608
      for (NodeIt n(_graph); n != INVALID; ++n) {
alpar@209
  1609
        _node_index.insert(std::make_pair(converter(map[n]), n));
deba@165
  1610
      }
deba@165
  1611
      return *this;
deba@165
  1612
    }
deba@165
  1613
deba@165
  1614
    /// \brief Use previously constructed edge set
deba@165
  1615
    ///
deba@165
  1616
    /// Use previously constructed edge set, and specify the edge
deba@165
  1617
    /// label map.
deba@165
  1618
    template <typename Map>
deba@165
  1619
    GraphReader& useEdges(const Map& map) {
deba@165
  1620
      checkConcept<concepts::ReadMap<Edge, typename Map::Value>, Map>();
deba@165
  1621
      LEMON_ASSERT(!_use_edges, "Multiple usage of useEdges() member");
deba@165
  1622
      _use_edges = true;
deba@165
  1623
      _writer_bits::DefaultConverter<typename Map::Value> converter;
deba@165
  1624
      for (EdgeIt a(_graph); a != INVALID; ++a) {
alpar@209
  1625
        _edge_index.insert(std::make_pair(converter(map[a]), a));
deba@165
  1626
      }
deba@165
  1627
      return *this;
deba@165
  1628
    }
deba@165
  1629
deba@165
  1630
    /// \brief Use previously constructed edge set
deba@165
  1631
    ///
deba@165
  1632
    /// Use previously constructed edge set, and specify the edge
deba@165
  1633
    /// label map and a functor which converts the label map values to
kpeter@192
  1634
    /// \c std::string.
deba@165
  1635
    template <typename Map, typename Converter>
alpar@209
  1636
    GraphReader& useEdges(const Map& map,
alpar@209
  1637
                            const Converter& converter = Converter()) {
deba@165
  1638
      checkConcept<concepts::ReadMap<Edge, typename Map::Value>, Map>();
alpar@209
  1639
      LEMON_ASSERT(!_use_edges, "Multiple usage of useEdges() member");
deba@165
  1640
      _use_edges = true;
deba@165
  1641
      for (EdgeIt a(_graph); a != INVALID; ++a) {
alpar@209
  1642
        _edge_index.insert(std::make_pair(converter(map[a]), a));
deba@165
  1643
      }
deba@165
  1644
      return *this;
deba@165
  1645
    }
deba@165
  1646
kpeter@192
  1647
    /// \brief Skip the reading of node section
deba@188
  1648
    ///
deba@188
  1649
    /// Omit the reading of the node section. This implies that each node
kpeter@192
  1650
    /// map reading rule will be abandoned, and the nodes of the graph
deba@188
  1651
    /// will not be constructed, which usually cause that the edge set
deba@188
  1652
    /// could not be read due to lack of node name
kpeter@192
  1653
    /// could not be read due to lack of node name resolving.
kpeter@192
  1654
    /// Therefore \c skipEdges() function should also be used, or
kpeter@192
  1655
    /// \c useNodes() should be used to specify the label of the nodes.
deba@188
  1656
    GraphReader& skipNodes() {
alpar@209
  1657
      LEMON_ASSERT(!_skip_nodes, "Skip nodes already set");
deba@188
  1658
      _skip_nodes = true;
deba@188
  1659
      return *this;
deba@188
  1660
    }
deba@188
  1661
kpeter@192
  1662
    /// \brief Skip the reading of edge section
deba@188
  1663
    ///
deba@188
  1664
    /// Omit the reading of the edge section. This implies that each edge
kpeter@192
  1665
    /// map reading rule will be abandoned, and the edges of the graph
deba@188
  1666
    /// will not be constructed.
deba@188
  1667
    GraphReader& skipEdges() {
alpar@209
  1668
      LEMON_ASSERT(!_skip_edges, "Skip edges already set");
deba@188
  1669
      _skip_edges = true;
deba@188
  1670
      return *this;
deba@188
  1671
    }
deba@188
  1672
deba@165
  1673
    /// @}
deba@165
  1674
deba@165
  1675
  private:
deba@165
  1676
deba@165
  1677
    bool readLine() {
deba@165
  1678
      std::string str;
deba@165
  1679
      while(++line_num, std::getline(*_is, str)) {
alpar@209
  1680
        line.clear(); line.str(str);
alpar@209
  1681
        char c;
alpar@209
  1682
        if (line >> std::ws >> c && c != '#') {
alpar@209
  1683
          line.putback(c);
alpar@209
  1684
          return true;
alpar@209
  1685
        }
deba@165
  1686
      }
deba@165
  1687
      return false;
deba@165
  1688
    }
deba@165
  1689
deba@165
  1690
    bool readSuccess() {
deba@165
  1691
      return static_cast<bool>(*_is);
deba@165
  1692
    }
alpar@209
  1693
deba@165
  1694
    void skipSection() {
deba@165
  1695
      char c;
deba@165
  1696
      while (readSuccess() && line >> c && c != '@') {
alpar@209
  1697
        readLine();
deba@165
  1698
      }
deba@347
  1699
      if (readSuccess()) {
deba@347
  1700
        line.putback(c);
deba@347
  1701
      }
deba@165
  1702
    }
deba@165
  1703
deba@165
  1704
    void readNodes() {
deba@165
  1705
deba@165
  1706
      std::vector<int> map_index(_node_maps.size());
deba@165
  1707
      int map_num, label_index;
deba@165
  1708
deba@186
  1709
      char c;
deba@186
  1710
      if (!readLine() || !(line >> c) || c == '@') {
alpar@209
  1711
        if (readSuccess() && line) line.putback(c);
alpar@209
  1712
        if (!_node_maps.empty())
deba@290
  1713
          throw FormatError("Cannot find map names");
alpar@209
  1714
        return;
deba@186
  1715
      }
deba@186
  1716
      line.putback(c);
alpar@209
  1717
deba@165
  1718
      {
alpar@209
  1719
        std::map<std::string, int> maps;
alpar@209
  1720
alpar@209
  1721
        std::string map;
alpar@209
  1722
        int index = 0;
alpar@209
  1723
        while (_reader_bits::readToken(line, map)) {
alpar@209
  1724
          if (maps.find(map) != maps.end()) {
alpar@209
  1725
            std::ostringstream msg;
alpar@209
  1726
            msg << "Multiple occurence of node map: " << map;
deba@290
  1727
            throw FormatError(msg.str());
alpar@209
  1728
          }
alpar@209
  1729
          maps.insert(std::make_pair(map, index));
alpar@209
  1730
          ++index;
alpar@209
  1731
        }
alpar@209
  1732
alpar@209
  1733
        for (int i = 0; i < static_cast<int>(_node_maps.size()); ++i) {
alpar@209
  1734
          std::map<std::string, int>::iterator jt =
alpar@209
  1735
            maps.find(_node_maps[i].first);
alpar@209
  1736
          if (jt == maps.end()) {
alpar@209
  1737
            std::ostringstream msg;
kpeter@291
  1738
            msg << "Map not found: " << _node_maps[i].first;
deba@290
  1739
            throw FormatError(msg.str());
alpar@209
  1740
          }
alpar@209
  1741
          map_index[i] = jt->second;
alpar@209
  1742
        }
alpar@209
  1743
alpar@209
  1744
        {
alpar@209
  1745
          std::map<std::string, int>::iterator jt = maps.find("label");
alpar@209
  1746
          if (jt != maps.end()) {
alpar@209
  1747
            label_index = jt->second;
alpar@209
  1748
          } else {
alpar@209
  1749
            label_index = -1;
alpar@209
  1750
          }
alpar@209
  1751
        }
alpar@209
  1752
        map_num = maps.size();
deba@165
  1753
      }
deba@165
  1754
deba@165
  1755
      while (readLine() && line >> c && c != '@') {
alpar@209
  1756
        line.putback(c);
alpar@209
  1757
alpar@209
  1758
        std::vector<std::string> tokens(map_num);
alpar@209
  1759
        for (int i = 0; i < map_num; ++i) {
alpar@209
  1760
          if (!_reader_bits::readToken(line, tokens[i])) {
alpar@209
  1761
            std::ostringstream msg;
alpar@209
  1762
            msg << "Column not found (" << i + 1 << ")";
deba@290
  1763
            throw FormatError(msg.str());
alpar@209
  1764
          }
alpar@209
  1765
        }
alpar@209
  1766
        if (line >> std::ws >> c)
kpeter@291
  1767
          throw FormatError("Extra character at the end of line");
alpar@209
  1768
alpar@209
  1769
        Node n;
alpar@209
  1770
        if (!_use_nodes) {
alpar@209
  1771
          n = _graph.addNode();
alpar@209
  1772
          if (label_index != -1)
alpar@209
  1773
            _node_index.insert(std::make_pair(tokens[label_index], n));
alpar@209
  1774
        } else {
alpar@209
  1775
          if (label_index == -1)
kpeter@291
  1776
            throw FormatError("Label map not found");
alpar@209
  1777
          typename std::map<std::string, Node>::iterator it =
alpar@209
  1778
            _node_index.find(tokens[label_index]);
alpar@209
  1779
          if (it == _node_index.end()) {
alpar@209
  1780
            std::ostringstream msg;
alpar@209
  1781
            msg << "Node with label not found: " << tokens[label_index];
deba@290
  1782
            throw FormatError(msg.str());
alpar@209
  1783
          }
alpar@209
  1784
          n = it->second;
alpar@209
  1785
        }
alpar@209
  1786
alpar@209
  1787
        for (int i = 0; i < static_cast<int>(_node_maps.size()); ++i) {
alpar@209
  1788
          _node_maps[i].second->set(n, tokens[map_index[i]]);
alpar@209
  1789
        }
deba@165
  1790
deba@165
  1791
      }
deba@165
  1792
      if (readSuccess()) {
alpar@209
  1793
        line.putback(c);
deba@165
  1794
      }
deba@165
  1795
    }
deba@165
  1796
deba@165
  1797
    void readEdges() {
deba@165
  1798
deba@165
  1799
      std::vector<int> map_index(_edge_maps.size());
deba@165
  1800
      int map_num, label_index;
deba@165
  1801
deba@186
  1802
      char c;
deba@186
  1803
      if (!readLine() || !(line >> c) || c == '@') {
alpar@209
  1804
        if (readSuccess() && line) line.putback(c);
alpar@209
  1805
        if (!_edge_maps.empty())
deba@290
  1806
          throw FormatError("Cannot find map names");
alpar@209
  1807
        return;
deba@186
  1808
      }
deba@186
  1809
      line.putback(c);
alpar@209
  1810
deba@165
  1811
      {
alpar@209
  1812
        std::map<std::string, int> maps;
alpar@209
  1813
alpar@209
  1814
        std::string map;
alpar@209
  1815
        int index = 0;
alpar@209
  1816
        while (_reader_bits::readToken(line, map)) {
alpar@422
  1817
          if(map == "-") {
alpar@422
  1818
              if(index!=0)
alpar@422
  1819
                throw FormatError("'-' is not allowed as a map name");
alpar@422
  1820
              else if (line >> std::ws >> c)
alpar@422
  1821
                throw FormatError("Extra character at the end of line");
alpar@422
  1822
              else break;
alpar@422
  1823
            }
alpar@209
  1824
          if (maps.find(map) != maps.end()) {
alpar@209
  1825
            std::ostringstream msg;
alpar@209
  1826
            msg << "Multiple occurence of edge map: " << map;
deba@290
  1827
            throw FormatError(msg.str());
alpar@209
  1828
          }
alpar@209
  1829
          maps.insert(std::make_pair(map, index));
alpar@209
  1830
          ++index;
alpar@209
  1831
        }
alpar@209
  1832
alpar@209
  1833
        for (int i = 0; i < static_cast<int>(_edge_maps.size()); ++i) {
alpar@209
  1834
          std::map<std::string, int>::iterator jt =
alpar@209
  1835
            maps.find(_edge_maps[i].first);
alpar@209
  1836
          if (jt == maps.end()) {
alpar@209
  1837
            std::ostringstream msg;
kpeter@291
  1838
            msg << "Map not found: " << _edge_maps[i].first;
deba@290
  1839
            throw FormatError(msg.str());
alpar@209
  1840
          }
alpar@209
  1841
          map_index[i] = jt->second;
alpar@209
  1842
        }
alpar@209
  1843
alpar@209
  1844
        {
alpar@209
  1845
          std::map<std::string, int>::iterator jt = maps.find("label");
alpar@209
  1846
          if (jt != maps.end()) {
alpar@209
  1847
            label_index = jt->second;
alpar@209
  1848
          } else {
alpar@209
  1849
            label_index = -1;
alpar@209
  1850
          }
alpar@209
  1851
        }
alpar@209
  1852
        map_num = maps.size();
deba@165
  1853
      }
deba@165
  1854
deba@165
  1855
      while (readLine() && line >> c && c != '@') {
alpar@209
  1856
        line.putback(c);
alpar@209
  1857
alpar@209
  1858
        std::string source_token;
alpar@209
  1859
        std::string target_token;
alpar@209
  1860
alpar@209
  1861
        if (!_reader_bits::readToken(line, source_token))
deba@290
  1862
          throw FormatError("Node u not found");
alpar@209
  1863
alpar@209
  1864
        if (!_reader_bits::readToken(line, target_token))
deba@290
  1865
          throw FormatError("Node v not found");
alpar@209
  1866
alpar@209
  1867
        std::vector<std::string> tokens(map_num);
alpar@209
  1868
        for (int i = 0; i < map_num; ++i) {
alpar@209
  1869
          if (!_reader_bits::readToken(line, tokens[i])) {
alpar@209
  1870
            std::ostringstream msg;
alpar@209
  1871
            msg << "Column not found (" << i + 1 << ")";
deba@290
  1872
            throw FormatError(msg.str());
alpar@209
  1873
          }
alpar@209
  1874
        }
alpar@209
  1875
        if (line >> std::ws >> c)
kpeter@291
  1876
          throw FormatError("Extra character at the end of line");
alpar@209
  1877
alpar@209
  1878
        Edge e;
alpar@209
  1879
        if (!_use_edges) {
deba@165
  1880
deba@165
  1881
          typename NodeIndex::iterator it;
alpar@209
  1882
deba@165
  1883
          it = _node_index.find(source_token);
deba@165
  1884
          if (it == _node_index.end()) {
deba@165
  1885
            std::ostringstream msg;
deba@165
  1886
            msg << "Item not found: " << source_token;
deba@290
  1887
            throw FormatError(msg.str());
deba@165
  1888
          }
deba@165
  1889
          Node source = it->second;
deba@165
  1890
deba@165
  1891
          it = _node_index.find(target_token);
alpar@209
  1892
          if (it == _node_index.end()) {
alpar@209
  1893
            std::ostringstream msg;
deba@165
  1894
            msg << "Item not found: " << target_token;
deba@290
  1895
            throw FormatError(msg.str());
alpar@209
  1896
          }
alpar@209
  1897
          Node target = it->second;
alpar@209
  1898
alpar@209
  1899
          e = _graph.addEdge(source, target);
alpar@209
  1900
          if (label_index != -1)
alpar@209
  1901
            _edge_index.insert(std::make_pair(tokens[label_index], e));
alpar@209
  1902
        } else {
alpar@209
  1903
          if (label_index == -1)
kpeter@291
  1904
            throw FormatError("Label map not found");
alpar@209
  1905
          typename std::map<std::string, Edge>::iterator it =
alpar@209
  1906
            _edge_index.find(tokens[label_index]);
alpar@209
  1907
          if (it == _edge_index.end()) {
alpar@209
  1908
            std::ostringstream msg;
alpar@209
  1909
            msg << "Edge with label not found: " << tokens[label_index];
deba@290
  1910
            throw FormatError(msg.str());
alpar@209
  1911
          }
alpar@209
  1912
          e = it->second;
alpar@209
  1913
        }
alpar@209
  1914
alpar@209
  1915
        for (int i = 0; i < static_cast<int>(_edge_maps.size()); ++i) {
alpar@209
  1916
          _edge_maps[i].second->set(e, tokens[map_index[i]]);
alpar@209
  1917
        }
deba@165
  1918
deba@165
  1919
      }
deba@165
  1920
      if (readSuccess()) {
alpar@209
  1921
        line.putback(c);
deba@165
  1922
      }
deba@165
  1923
    }
deba@165
  1924
deba@165
  1925
    void readAttributes() {
deba@165
  1926
deba@165
  1927
      std::set<std::string> read_attr;
deba@165
  1928
deba@165
  1929
      char c;
deba@165
  1930
      while (readLine() && line >> c && c != '@') {
alpar@209
  1931
        line.putback(c);
alpar@209
  1932
alpar@209
  1933
        std::string attr, token;
alpar@209
  1934
        if (!_reader_bits::readToken(line, attr))
deba@290
  1935
          throw FormatError("Attribute name not found");
alpar@209
  1936
        if (!_reader_bits::readToken(line, token))
deba@290
  1937
          throw FormatError("Attribute value not found");
alpar@209
  1938
        if (line >> c)
kpeter@291
  1939
          throw FormatError("Extra character at the end of line");
alpar@209
  1940
alpar@209
  1941
        {
alpar@209
  1942
          std::set<std::string>::iterator it = read_attr.find(attr);
alpar@209
  1943
          if (it != read_attr.end()) {
alpar@209
  1944
            std::ostringstream msg;
kpeter@291
  1945
            msg << "Multiple occurence of attribute: " << attr;
deba@290
  1946
            throw FormatError(msg.str());
alpar@209
  1947
          }
alpar@209
  1948
          read_attr.insert(attr);
alpar@209
  1949
        }
alpar@209
  1950
alpar@209
  1951
        {
alpar@209
  1952
          typename Attributes::iterator it = _attributes.lower_bound(attr);
alpar@209
  1953
          while (it != _attributes.end() && it->first == attr) {
alpar@209
  1954
            it->second->set(token);
alpar@209
  1955
            ++it;
alpar@209
  1956
          }
alpar@209
  1957
        }
deba@165
  1958
deba@165
  1959
      }
deba@165
  1960
      if (readSuccess()) {
alpar@209
  1961
        line.putback(c);
deba@165
  1962
      }
deba@165
  1963
      for (typename Attributes::iterator it = _attributes.begin();
alpar@209
  1964
           it != _attributes.end(); ++it) {
alpar@209
  1965
        if (read_attr.find(it->first) == read_attr.end()) {
alpar@209
  1966
          std::ostringstream msg;
kpeter@291
  1967
          msg << "Attribute not found: " << it->first;
deba@290
  1968
          throw FormatError(msg.str());
alpar@209
  1969
        }
deba@165
  1970
      }
deba@165
  1971
    }
deba@165
  1972
deba@165
  1973
  public:
deba@165
  1974
alpar@209
  1975
    /// \name Execution of the reader
deba@165
  1976
    /// @{
deba@165
  1977
deba@165
  1978
    /// \brief Start the batch processing
deba@165
  1979
    ///
deba@165
  1980
    /// This function starts the batch processing
deba@165
  1981
    void run() {
alpar@209
  1982
deba@165
  1983
      LEMON_ASSERT(_is != 0, "This reader assigned to an other reader");
alpar@209
  1984
deba@188
  1985
      bool nodes_done = _skip_nodes;
deba@188
  1986
      bool edges_done = _skip_edges;
deba@165
  1987
      bool attributes_done = false;
deba@165
  1988
alpar@209
  1989
      line_num = 0;
deba@165
  1990
      readLine();
deba@172
  1991
      skipSection();
deba@165
  1992
deba@165
  1993
      while (readSuccess()) {
alpar@209
  1994
        try {
alpar@209
  1995
          char c;
alpar@209
  1996
          std::string section, caption;
alpar@209
  1997
          line >> c;
alpar@209
  1998
          _reader_bits::readToken(line, section);
alpar@209
  1999
          _reader_bits::readToken(line, caption);
alpar@209
  2000
alpar@209
  2001
          if (line >> c)
kpeter@291
  2002
            throw FormatError("Extra character at the end of line");
alpar@209
  2003
alpar@209
  2004
          if (section == "nodes" && !nodes_done) {
alpar@209
  2005
            if (_nodes_caption.empty() || _nodes_caption == caption) {
alpar@209
  2006
              readNodes();
alpar@209
  2007
              nodes_done = true;
alpar@209
  2008
            }
alpar@209
  2009
          } else if ((section == "edges" || section == "arcs") &&
alpar@209
  2010
                     !edges_done) {
alpar@209
  2011
            if (_edges_caption.empty() || _edges_caption == caption) {
alpar@209
  2012
              readEdges();
alpar@209
  2013
              edges_done = true;
alpar@209
  2014
            }
alpar@209
  2015
          } else if (section == "attributes" && !attributes_done) {
alpar@209
  2016
            if (_attributes_caption.empty() || _attributes_caption == caption) {
alpar@209
  2017
              readAttributes();
alpar@209
  2018
              attributes_done = true;
alpar@209
  2019
            }
alpar@209
  2020
          } else {
alpar@209
  2021
            readLine();
alpar@209
  2022
            skipSection();
alpar@209
  2023
          }
deba@290
  2024
        } catch (FormatError& error) {
alpar@209
  2025
          error.line(line_num);
deba@290
  2026
          error.file(_filename);
alpar@209
  2027
          throw;
alpar@209
  2028
        }
deba@165
  2029
      }
deba@165
  2030
deba@165
  2031
      if (!nodes_done) {
deba@290
  2032
        throw FormatError("Section @nodes not found");
deba@165
  2033
      }
deba@165
  2034
deba@165
  2035
      if (!edges_done) {
deba@290
  2036
        throw FormatError("Section @edges not found");
deba@165
  2037
      }
deba@165
  2038
deba@165
  2039
      if (!attributes_done && !_attributes.empty()) {
deba@290
  2040
        throw FormatError("Section @attributes not found");
deba@165
  2041
      }
deba@165
  2042
deba@165
  2043
    }
deba@165
  2044
deba@165
  2045
    /// @}
alpar@209
  2046
deba@165
  2047
  };
deba@165
  2048
kpeter@374
  2049
  /// \brief Return a \ref GraphReader class
kpeter@374
  2050
  ///
kpeter@374
  2051
  /// This function just returns a \ref GraphReader class.
kpeter@374
  2052
  /// \relates GraphReader
kpeter@374
  2053
  template <typename Graph>
kpeter@374
  2054
  GraphReader<Graph> graphReader(Graph& graph, std::istream& is) {
kpeter@374
  2055
    GraphReader<Graph> tmp(graph, is);
kpeter@374
  2056
    return tmp;
kpeter@374
  2057
  }
kpeter@374
  2058
kpeter@374
  2059
  /// \brief Return a \ref GraphReader class
kpeter@374
  2060
  ///
kpeter@374
  2061
  /// This function just returns a \ref GraphReader class.
kpeter@374
  2062
  /// \relates GraphReader
kpeter@374
  2063
  template <typename Graph>
kpeter@374
  2064
  GraphReader<Graph> graphReader(Graph& graph, const std::string& fn) {
kpeter@374
  2065
    GraphReader<Graph> tmp(graph, fn);
kpeter@374
  2066
    return tmp;
kpeter@374
  2067
  }
kpeter@374
  2068
kpeter@374
  2069
  /// \brief Return a \ref GraphReader class
kpeter@374
  2070
  ///
kpeter@374
  2071
  /// This function just returns a \ref GraphReader class.
kpeter@374
  2072
  /// \relates GraphReader
kpeter@374
  2073
  template <typename Graph>
kpeter@374
  2074
  GraphReader<Graph> graphReader(Graph& graph, const char* fn) {
kpeter@374
  2075
    GraphReader<Graph> tmp(graph, fn);
kpeter@374
  2076
    return tmp;
kpeter@374
  2077
  }
kpeter@374
  2078
deba@190
  2079
  class SectionReader;
deba@190
  2080
deba@190
  2081
  SectionReader sectionReader(std::istream& is);
deba@190
  2082
  SectionReader sectionReader(const std::string& fn);
deba@190
  2083
  SectionReader sectionReader(const char* fn);
alpar@209
  2084
kpeter@192
  2085
  /// \ingroup lemon_io
kpeter@192
  2086
  ///
deba@189
  2087
  /// \brief Section reader class
deba@189
  2088
  ///
alpar@209
  2089
  /// In the \ref lgf-format "LGF" file extra sections can be placed,
kpeter@192
  2090
  /// which contain any data in arbitrary format. Such sections can be
alpar@209
  2091
  /// read with this class. A reading rule can be added to the class
kpeter@192
  2092
  /// with two different functions. With the \c sectionLines() function a
kpeter@192
  2093
  /// functor can process the section line-by-line, while with the \c
deba@189
  2094
  /// sectionStream() member the section can be read from an input
deba@189
  2095
  /// stream.
deba@189
  2096
  class SectionReader {
deba@189
  2097
  private:
alpar@209
  2098
deba@189
  2099
    std::istream* _is;
deba@189
  2100
    bool local_is;
deba@290
  2101
    std::string _filename;
deba@189
  2102
deba@189
  2103
    typedef std::map<std::string, _reader_bits::Section*> Sections;
deba@189
  2104
    Sections _sections;
deba@189
  2105
deba@189
  2106
    int line_num;
deba@189
  2107
    std::istringstream line;
deba@189
  2108
deba@189
  2109
  public:
deba@189
  2110
deba@189
  2111
    /// \brief Constructor
deba@189
  2112
    ///
deba@189
  2113
    /// Construct a section reader, which reads from the given input
deba@189
  2114
    /// stream.
alpar@209
  2115
    SectionReader(std::istream& is)
deba@189
  2116
      : _is(&is), local_is(false) {}
deba@189
  2117
deba@189
  2118
    /// \brief Constructor
deba@189
  2119
    ///
deba@189
  2120
    /// Construct a section reader, which reads from the given file.
alpar@209
  2121
    SectionReader(const std::string& fn)
deba@290
  2122
      : _is(new std::ifstream(fn.c_str())), local_is(true),
deba@290
  2123
        _filename(fn) {
deba@295
  2124
      if (!(*_is)) {
deba@295
  2125
        delete _is;
deba@295
  2126
        throw IoError("Cannot open file", fn);
deba@295
  2127
      }
deba@290
  2128
    }
alpar@209
  2129
deba@189
  2130
    /// \brief Constructor
deba@189
  2131
    ///
deba@189
  2132
    /// Construct a section reader, which reads from the given file.
alpar@209
  2133
    SectionReader(const char* fn)
deba@290
  2134
      : _is(new std::ifstream(fn)), local_is(true),
deba@290
  2135
        _filename(fn) {
deba@295
  2136
      if (!(*_is)) {
deba@295
  2137
        delete _is;
deba@295
  2138
        throw IoError("Cannot open file", fn);
deba@295
  2139
      }
deba@290
  2140
    }
deba@189
  2141
deba@189
  2142
    /// \brief Destructor
deba@189
  2143
    ~SectionReader() {
alpar@209
  2144
      for (Sections::iterator it = _sections.begin();
alpar@209
  2145
           it != _sections.end(); ++it) {
alpar@209
  2146
        delete it->second;
deba@189
  2147
      }
deba@189
  2148
deba@189
  2149
      if (local_is) {
alpar@209
  2150
        delete _is;
deba@189
  2151
      }
deba@189
  2152
deba@189
  2153
    }
deba@189
  2154
deba@189
  2155
  private:
deba@190
  2156
deba@190
  2157
    friend SectionReader sectionReader(std::istream& is);
deba@190
  2158
    friend SectionReader sectionReader(const std::string& fn);
deba@190
  2159
    friend SectionReader sectionReader(const char* fn);
deba@190
  2160
alpar@209
  2161
    SectionReader(SectionReader& other)
deba@190
  2162
      : _is(other._is), local_is(other.local_is) {
deba@190
  2163
deba@190
  2164
      other._is = 0;
deba@190
  2165
      other.local_is = false;
alpar@209
  2166
deba@190
  2167
      _sections.swap(other._sections);
deba@190
  2168
    }
alpar@209
  2169
deba@189
  2170
    SectionReader& operator=(const SectionReader&);
deba@189
  2171
deba@189
  2172
  public:
deba@189
  2173
deba@189
  2174
    /// \name Section readers
deba@189
  2175
    /// @{
deba@189
  2176
deba@189
  2177
    /// \brief Add a section processor with line oriented reading
deba@189
  2178
    ///
deba@189
  2179
    /// The first parameter is the type descriptor of the section, the
deba@189
  2180
    /// second is a functor, which takes just one \c std::string
deba@189
  2181
    /// parameter. At the reading process, each line of the section
deba@189
  2182
    /// will be given to the functor object. However, the empty lines
deba@189
  2183
    /// and the comment lines are filtered out, and the leading
deba@189
  2184
    /// whitespaces are trimmed from each processed string.
deba@189
  2185
    ///
deba@189
  2186
    /// For example let's see a section, which contain several
deba@189
  2187
    /// integers, which should be inserted into a vector.
deba@189
  2188
    ///\code
deba@189
  2189
    ///  @numbers
deba@189
  2190
    ///  12 45 23
deba@189
  2191
    ///  4
deba@189
  2192
    ///  23 6
deba@189
  2193
    ///\endcode
deba@189
  2194
    ///
kpeter@192
  2195
    /// The functor is implemented as a struct:
deba@189
  2196
    ///\code
deba@189
  2197
    ///  struct NumberSection {
deba@189
  2198
    ///    std::vector<int>& _data;
deba@189
  2199
    ///    NumberSection(std::vector<int>& data) : _data(data) {}
deba@189
  2200
    ///    void operator()(const std::string& line) {
deba@189
  2201
    ///      std::istringstream ls(line);
deba@189
  2202
    ///      int value;
deba@189
  2203
    ///      while (ls >> value) _data.push_back(value);
deba@189
  2204
    ///    }
deba@189
  2205
    ///  };
deba@189
  2206
    ///
deba@189
  2207
    ///  // ...
deba@189
  2208
    ///
alpar@209
  2209
    ///  reader.sectionLines("numbers", NumberSection(vec));
deba@189
  2210
    ///\endcode
deba@189
  2211
    template <typename Functor>
deba@189
  2212
    SectionReader& sectionLines(const std::string& type, Functor functor) {
kpeter@192
  2213
      LEMON_ASSERT(!type.empty(), "Type is empty.");
alpar@209
  2214
      LEMON_ASSERT(_sections.find(type) == _sections.end(),
alpar@209
  2215
                   "Multiple reading of section.");
alpar@209
  2216
      _sections.insert(std::make_pair(type,
deba@189
  2217
        new _reader_bits::LineSection<Functor>(functor)));
deba@189
  2218
      return *this;
deba@189
  2219
    }
deba@189
  2220
deba@189
  2221
deba@189
  2222
    /// \brief Add a section processor with stream oriented reading
deba@189
  2223
    ///
deba@189
  2224
    /// The first parameter is the type of the section, the second is
kpeter@192
  2225
    /// a functor, which takes an \c std::istream& and an \c int&
deba@189
  2226
    /// parameter, the latter regard to the line number of stream. The
deba@189
  2227
    /// functor can read the input while the section go on, and the
deba@189
  2228
    /// line number should be modified accordingly.
deba@189
  2229
    template <typename Functor>
deba@189
  2230
    SectionReader& sectionStream(const std::string& type, Functor functor) {
kpeter@192
  2231
      LEMON_ASSERT(!type.empty(), "Type is empty.");
alpar@209
  2232
      LEMON_ASSERT(_sections.find(type) == _sections.end(),
alpar@209
  2233
                   "Multiple reading of section.");
alpar@209
  2234
      _sections.insert(std::make_pair(type,
alpar@209
  2235
         new _reader_bits::StreamSection<Functor>(functor)));
deba@189
  2236
      return *this;
alpar@209
  2237
    }
alpar@209
  2238
deba@189
  2239
    /// @}
deba@189
  2240
deba@189
  2241
  private:
deba@189
  2242
deba@189
  2243
    bool readLine() {
deba@189
  2244
      std::string str;
deba@189
  2245
      while(++line_num, std::getline(*_is, str)) {
alpar@209
  2246
        line.clear(); line.str(str);
alpar@209
  2247
        char c;
alpar@209
  2248
        if (line >> std::ws >> c && c != '#') {
alpar@209
  2249
          line.putback(c);
alpar@209
  2250
          return true;
alpar@209
  2251
        }
deba@189
  2252
      }
deba@189
  2253
      return false;
deba@189
  2254
    }
deba@189
  2255
deba@189
  2256
    bool readSuccess() {
deba@189
  2257
      return static_cast<bool>(*_is);
deba@189
  2258
    }
alpar@209
  2259
deba@189
  2260
    void skipSection() {
deba@189
  2261
      char c;
deba@189
  2262
      while (readSuccess() && line >> c && c != '@') {
alpar@209
  2263
        readLine();
deba@189
  2264
      }
deba@347
  2265
      if (readSuccess()) {
deba@347
  2266
        line.putback(c);
deba@347
  2267
      }
deba@189
  2268
    }
deba@189
  2269
deba@189
  2270
  public:
deba@189
  2271
deba@189
  2272
alpar@209
  2273
    /// \name Execution of the reader
deba@189
  2274
    /// @{
deba@189
  2275
deba@189
  2276
    /// \brief Start the batch processing
deba@189
  2277
    ///
kpeter@192
  2278
    /// This function starts the batch processing.
deba@189
  2279
    void run() {
alpar@209
  2280
deba@189
  2281
      LEMON_ASSERT(_is != 0, "This reader assigned to an other reader");
alpar@209
  2282
deba@189
  2283
      std::set<std::string> extra_sections;
deba@189
  2284
alpar@209
  2285
      line_num = 0;
deba@189
  2286
      readLine();
deba@189
  2287
      skipSection();
deba@189
  2288
deba@189
  2289
      while (readSuccess()) {
alpar@209
  2290
        try {
alpar@209
  2291
          char c;
alpar@209
  2292
          std::string section, caption;
alpar@209
  2293
          line >> c;
alpar@209
  2294
          _reader_bits::readToken(line, section);
alpar@209
  2295
          _reader_bits::readToken(line, caption);
alpar@209
  2296
alpar@209
  2297
          if (line >> c)
kpeter@291
  2298
            throw FormatError("Extra character at the end of line");
alpar@209
  2299
alpar@209
  2300
          if (extra_sections.find(section) != extra_sections.end()) {
alpar@209
  2301
            std::ostringstream msg;
kpeter@291
  2302
            msg << "Multiple occurence of section: " << section;
deba@290
  2303
            throw FormatError(msg.str());
alpar@209
  2304
          }
alpar@209
  2305
          Sections::iterator it = _sections.find(section);
alpar@209
  2306
          if (it != _sections.end()) {
alpar@209
  2307
            extra_sections.insert(section);
alpar@209
  2308
            it->second->process(*_is, line_num);
alpar@209
  2309
          }
alpar@209
  2310
          readLine();
alpar@209
  2311
          skipSection();
deba@290
  2312
        } catch (FormatError& error) {
alpar@209
  2313
          error.line(line_num);
deba@290
  2314
          error.file(_filename);
alpar@209
  2315
          throw;
alpar@209
  2316
        }
deba@189
  2317
      }
deba@189
  2318
      for (Sections::iterator it = _sections.begin();
alpar@209
  2319
           it != _sections.end(); ++it) {
alpar@209
  2320
        if (extra_sections.find(it->first) == extra_sections.end()) {
alpar@209
  2321
          std::ostringstream os;
alpar@209
  2322
          os << "Cannot find section: " << it->first;
deba@290
  2323
          throw FormatError(os.str());
alpar@209
  2324
        }
deba@189
  2325
      }
deba@189
  2326
    }
deba@189
  2327
deba@189
  2328
    /// @}
alpar@209
  2329
deba@189
  2330
  };
deba@189
  2331
kpeter@192
  2332
  /// \brief Return a \ref SectionReader class
alpar@209
  2333
  ///
kpeter@192
  2334
  /// This function just returns a \ref SectionReader class.
deba@189
  2335
  /// \relates SectionReader
deba@189
  2336
  inline SectionReader sectionReader(std::istream& is) {
deba@189
  2337
    SectionReader tmp(is);
deba@189
  2338
    return tmp;
deba@189
  2339
  }
deba@189
  2340
kpeter@192
  2341
  /// \brief Return a \ref SectionReader class
alpar@209
  2342
  ///
kpeter@192
  2343
  /// This function just returns a \ref SectionReader class.
deba@189
  2344
  /// \relates SectionReader
deba@189
  2345
  inline SectionReader sectionReader(const std::string& fn) {
deba@189
  2346
    SectionReader tmp(fn);
deba@189
  2347
    return tmp;
deba@189
  2348
  }
deba@189
  2349
kpeter@192
  2350
  /// \brief Return a \ref SectionReader class
alpar@209
  2351
  ///
kpeter@192
  2352
  /// This function just returns a \ref SectionReader class.
deba@189
  2353
  /// \relates SectionReader
deba@189
  2354
  inline SectionReader sectionReader(const char* fn) {
deba@189
  2355
    SectionReader tmp(fn);
deba@189
  2356
    return tmp;
deba@189
  2357
  }
deba@189
  2358
deba@173
  2359
  /// \ingroup lemon_io
deba@173
  2360
  ///
alpar@209
  2361
  /// \brief Reader for the contents of the \ref lgf-format "LGF" file
deba@173
  2362
  ///
deba@173
  2363
  /// This class can be used to read the sections, the map names and
ladanyi@236
  2364
  /// the attributes from a file. Usually, the LEMON programs know
deba@173
  2365
  /// that, which type of graph, which maps and which attributes
deba@173
  2366
  /// should be read from a file, but in general tools (like glemon)
alpar@179
  2367
  /// the contents of an LGF file should be guessed somehow. This class
deba@173
  2368
  /// reads the graph and stores the appropriate information for
deba@173
  2369
  /// reading the graph.
deba@173
  2370
  ///
alpar@209
  2371
  ///\code
alpar@209
  2372
  /// LgfContents contents("graph.lgf");
alpar@179
  2373
  /// contents.run();
deba@173
  2374
  ///
kpeter@192
  2375
  /// // Does it contain any node section and arc section?
alpar@179
  2376
  /// if (contents.nodeSectionNum() == 0 || contents.arcSectionNum()) {
kpeter@192
  2377
  ///   std::cerr << "Failure, cannot find graph." << std::endl;
deba@173
  2378
  ///   return -1;
deba@173
  2379
  /// }
alpar@209
  2380
  /// std::cout << "The name of the default node section: "
alpar@179
  2381
  ///           << contents.nodeSection(0) << std::endl;
alpar@209
  2382
  /// std::cout << "The number of the arc maps: "
alpar@179
  2383
  ///           << contents.arcMaps(0).size() << std::endl;
alpar@209
  2384
  /// std::cout << "The name of second arc map: "
alpar@179
  2385
  ///           << contents.arcMaps(0)[1] << std::endl;
deba@173
  2386
  ///\endcode
alpar@209
  2387
  class LgfContents {
deba@173
  2388
  private:
deba@173
  2389
deba@173
  2390
    std::istream* _is;
deba@173
  2391
    bool local_is;
deba@173
  2392
deba@173
  2393
    std::vector<std::string> _node_sections;
deba@173
  2394
    std::vector<std::string> _edge_sections;
deba@173
  2395
    std::vector<std::string> _attribute_sections;
deba@173
  2396
    std::vector<std::string> _extra_sections;
deba@173
  2397
deba@173
  2398
    std::vector<bool> _arc_sections;
deba@173
  2399
deba@173
  2400
    std::vector<std::vector<std::string> > _node_maps;
deba@173
  2401
    std::vector<std::vector<std::string> > _edge_maps;
deba@173
  2402
deba@173
  2403
    std::vector<std::vector<std::string> > _attributes;
deba@173
  2404
deba@173
  2405
deba@173
  2406
    int line_num;
deba@173
  2407
    std::istringstream line;
alpar@209
  2408
deba@173
  2409
  public:
deba@173
  2410
deba@173
  2411
    /// \brief Constructor
deba@173
  2412
    ///
alpar@179
  2413
    /// Construct an \e LGF contents reader, which reads from the given
deba@173
  2414
    /// input stream.
alpar@209
  2415
    LgfContents(std::istream& is)
deba@173
  2416
      : _is(&is), local_is(false) {}
deba@173
  2417
deba@173
  2418
    /// \brief Constructor
deba@173
  2419
    ///
alpar@179
  2420
    /// Construct an \e LGF contents reader, which reads from the given
deba@173
  2421
    /// file.
alpar@209
  2422
    LgfContents(const std::string& fn)
deba@290
  2423
      : _is(new std::ifstream(fn.c_str())), local_is(true) {
deba@295
  2424
      if (!(*_is)) {
deba@295
  2425
        delete _is;
deba@295
  2426
        throw IoError("Cannot open file", fn);
deba@295
  2427
      }
deba@290
  2428
    }
deba@173
  2429
deba@173
  2430
    /// \brief Constructor
deba@173
  2431
    ///
alpar@179
  2432
    /// Construct an \e LGF contents reader, which reads from the given
deba@173
  2433
    /// file.
alpar@179
  2434
    LgfContents(const char* fn)
deba@290
  2435
      : _is(new std::ifstream(fn)), local_is(true) {
deba@295
  2436
      if (!(*_is)) {
deba@295
  2437
        delete _is;
deba@295
  2438
        throw IoError("Cannot open file", fn);
deba@295
  2439
      }
deba@290
  2440
    }
alpar@209
  2441
deba@173
  2442
    /// \brief Destructor
alpar@179
  2443
    ~LgfContents() {
deba@173
  2444
      if (local_is) delete _is;
deba@173
  2445
    }
deba@173
  2446
deba@190
  2447
  private:
alpar@209
  2448
deba@190
  2449
    LgfContents(const LgfContents&);
deba@190
  2450
    LgfContents& operator=(const LgfContents&);
deba@190
  2451
deba@190
  2452
  public:
deba@190
  2453
deba@173
  2454
deba@173
  2455
    /// \name Node sections
deba@173
  2456
    /// @{
deba@173
  2457
deba@173
  2458
    /// \brief Gives back the number of node sections in the file.
deba@173
  2459
    ///
deba@173
  2460
    /// Gives back the number of node sections in the file.
deba@173
  2461
    int nodeSectionNum() const {
deba@173
  2462
      return _node_sections.size();
deba@173
  2463
    }
deba@173
  2464
alpar@209
  2465
    /// \brief Returns the node section name at the given position.
deba@173
  2466
    ///
alpar@209
  2467
    /// Returns the node section name at the given position.
deba@173
  2468
    const std::string& nodeSection(int i) const {
deba@173
  2469
      return _node_sections[i];
deba@173
  2470
    }
deba@173
  2471
deba@173
  2472
    /// \brief Gives back the node maps for the given section.
deba@173
  2473
    ///
deba@173
  2474
    /// Gives back the node maps for the given section.
alpar@182
  2475
    const std::vector<std::string>& nodeMapNames(int i) const {
deba@173
  2476
      return _node_maps[i];
deba@173
  2477
    }
deba@173
  2478
deba@173
  2479
    /// @}
deba@173
  2480
alpar@209
  2481
    /// \name Arc/Edge sections
deba@173
  2482
    /// @{
deba@173
  2483
alpar@181
  2484
    /// \brief Gives back the number of arc/edge sections in the file.
deba@173
  2485
    ///
alpar@181
  2486
    /// Gives back the number of arc/edge sections in the file.
alpar@181
  2487
    /// \note It is synonym of \c edgeSectionNum().
deba@173
  2488
    int arcSectionNum() const {
deba@173
  2489
      return _edge_sections.size();
deba@173
  2490
    }
deba@173
  2491
alpar@209
  2492
    /// \brief Returns the arc/edge section name at the given position.
deba@173
  2493
    ///
alpar@209
  2494
    /// Returns the arc/edge section name at the given position.
alpar@181
  2495
    /// \note It is synonym of \c edgeSection().
deba@173
  2496
    const std::string& arcSection(int i) const {
deba@173
  2497
      return _edge_sections[i];
deba@173
  2498
    }
deba@173
  2499
alpar@181
  2500
    /// \brief Gives back the arc/edge maps for the given section.
deba@173
  2501
    ///
alpar@181
  2502
    /// Gives back the arc/edge maps for the given section.
alpar@182
  2503
    /// \note It is synonym of \c edgeMapNames().
alpar@182
  2504
    const std::vector<std::string>& arcMapNames(int i) const {
deba@173
  2505
      return _edge_maps[i];
deba@173
  2506
    }
deba@173
  2507
deba@173
  2508
    /// @}
deba@173
  2509
alpar@181
  2510
    /// \name Synonyms
deba@173
  2511
    /// @{
deba@173
  2512
alpar@181
  2513
    /// \brief Gives back the number of arc/edge sections in the file.
deba@173
  2514
    ///
alpar@181
  2515
    /// Gives back the number of arc/edge sections in the file.
alpar@181
  2516
    /// \note It is synonym of \c arcSectionNum().
deba@173
  2517
    int edgeSectionNum() const {
deba@173
  2518
      return _edge_sections.size();
deba@173
  2519
    }
deba@173
  2520
alpar@209
  2521
    /// \brief Returns the section name at the given position.
deba@173
  2522
    ///
alpar@209
  2523
    /// Returns the section name at the given position.
alpar@181
  2524
    /// \note It is synonym of \c arcSection().
deba@173
  2525
    const std::string& edgeSection(int i) const {
deba@173
  2526
      return _edge_sections[i];
deba@173
  2527
    }
deba@173
  2528
deba@173
  2529
    /// \brief Gives back the edge maps for the given section.
deba@173
  2530
    ///
deba@173
  2531
    /// Gives back the edge maps for the given section.
alpar@182
  2532
    /// \note It is synonym of \c arcMapNames().
alpar@182
  2533
    const std::vector<std::string>& edgeMapNames(int i) const {
deba@173
  2534
      return _edge_maps[i];
deba@173
  2535
    }
deba@173
  2536
deba@173
  2537
    /// @}
deba@173
  2538
alpar@209
  2539
    /// \name Attribute sections
deba@173
  2540
    /// @{
deba@173
  2541
deba@173
  2542
    /// \brief Gives back the number of attribute sections in the file.
deba@173
  2543
    ///
deba@173
  2544
    /// Gives back the number of attribute sections in the file.
deba@173
  2545
    int attributeSectionNum() const {
deba@173
  2546
      return _attribute_sections.size();
deba@173
  2547
    }
deba@173
  2548
alpar@209
  2549
    /// \brief Returns the attribute section name at the given position.
deba@173
  2550
    ///
alpar@209
  2551
    /// Returns the attribute section name at the given position.
alpar@182
  2552
    const std::string& attributeSectionNames(int i) const {
deba@173
  2553
      return _attribute_sections[i];
deba@173
  2554
    }
deba@173
  2555
deba@173
  2556
    /// \brief Gives back the attributes for the given section.
deba@173
  2557
    ///
deba@173
  2558
    /// Gives back the attributes for the given section.
deba@173
  2559
    const std::vector<std::string>& attributes(int i) const {
deba@173
  2560
      return _attributes[i];
deba@173
  2561
    }
deba@173
  2562
deba@173
  2563
    /// @}
deba@173
  2564
alpar@209
  2565
    /// \name Extra sections
deba@173
  2566
    /// @{
deba@173
  2567
deba@173
  2568
    /// \brief Gives back the number of extra sections in the file.
deba@173
  2569
    ///
deba@173
  2570
    /// Gives back the number of extra sections in the file.
deba@173
  2571
    int extraSectionNum() const {
deba@173
  2572
      return _extra_sections.size();
deba@173
  2573
    }
deba@173
  2574
alpar@209
  2575
    /// \brief Returns the extra section type at the given position.
deba@173
  2576
    ///
alpar@209
  2577
    /// Returns the section type at the given position.
deba@173
  2578
    const std::string& extraSection(int i) const {
deba@173
  2579
      return _extra_sections[i];
deba@173
  2580
    }
deba@173
  2581
deba@173
  2582
    /// @}
deba@173
  2583
deba@173
  2584
  private:
deba@173
  2585
deba@173
  2586
    bool readLine() {
deba@173
  2587
      std::string str;
deba@173
  2588
      while(++line_num, std::getline(*_is, str)) {
alpar@209
  2589
        line.clear(); line.str(str);
alpar@209
  2590
        char c;
alpar@209
  2591
        if (line >> std::ws >> c && c != '#') {
alpar@209
  2592
          line.putback(c);
alpar@209
  2593
          return true;
alpar@209
  2594
        }
deba@173
  2595
      }
deba@173
  2596
      return false;
deba@173
  2597
    }
deba@173
  2598
deba@173
  2599
    bool readSuccess() {
deba@173
  2600
      return static_cast<bool>(*_is);
deba@173
  2601
    }
deba@173
  2602
deba@173
  2603
    void skipSection() {
deba@173
  2604
      char c;
deba@173
  2605
      while (readSuccess() && line >> c && c != '@') {
alpar@209
  2606
        readLine();
deba@173
  2607
      }
deba@347
  2608
      if (readSuccess()) {
deba@347
  2609
        line.putback(c);
deba@347
  2610
      }
deba@173
  2611
    }
deba@173
  2612
deba@173
  2613
    void readMaps(std::vector<std::string>& maps) {
deba@186
  2614
      char c;
deba@186
  2615
      if (!readLine() || !(line >> c) || c == '@') {
alpar@209
  2616
        if (readSuccess() && line) line.putback(c);
alpar@209
  2617
        return;
deba@186
  2618
      }
deba@186
  2619
      line.putback(c);
deba@173
  2620
      std::string map;
deba@173
  2621
      while (_reader_bits::readToken(line, map)) {
alpar@209
  2622
        maps.push_back(map);
deba@173
  2623
      }
deba@173
  2624
    }
deba@173
  2625
deba@173
  2626
    void readAttributes(std::vector<std::string>& attrs) {
deba@173
  2627
      readLine();
deba@173
  2628
      char c;
deba@173
  2629
      while (readSuccess() && line >> c && c != '@') {
alpar@209
  2630
        line.putback(c);
alpar@209
  2631
        std::string attr;
alpar@209
  2632
        _reader_bits::readToken(line, attr);
alpar@209
  2633
        attrs.push_back(attr);
alpar@209
  2634
        readLine();
deba@173
  2635
      }
deba@173
  2636
      line.putback(c);
deba@173
  2637
    }
deba@173
  2638
deba@173
  2639
  public:
deba@173
  2640
alpar@209
  2641
    /// \name Execution of the contents reader
deba@173
  2642
    /// @{
deba@173
  2643
kpeter@192
  2644
    /// \brief Starts the reading
deba@173
  2645
    ///
kpeter@192
  2646
    /// This function starts the reading.
deba@173
  2647
    void run() {
deba@173
  2648
deba@173
  2649
      readLine();
deba@173
  2650
      skipSection();
deba@173
  2651
deba@173
  2652
      while (readSuccess()) {
deba@173
  2653
alpar@209
  2654
        char c;
alpar@209
  2655
        line >> c;
alpar@209
  2656
alpar@209
  2657
        std::string section, caption;
alpar@209
  2658
        _reader_bits::readToken(line, section);
alpar@209
  2659
        _reader_bits::readToken(line, caption);
alpar@209
  2660
alpar@209
  2661
        if (section == "nodes") {
alpar@209
  2662
          _node_sections.push_back(caption);
alpar@209
  2663
          _node_maps.push_back(std::vector<std::string>());
alpar@209
  2664
          readMaps(_node_maps.back());
alpar@209
  2665
          readLine(); skipSection();
alpar@209
  2666
        } else if (section == "arcs" || section == "edges") {
alpar@209
  2667
          _edge_sections.push_back(caption);
alpar@209
  2668
          _arc_sections.push_back(section == "arcs");
alpar@209
  2669
          _edge_maps.push_back(std::vector<std::string>());
alpar@209
  2670
          readMaps(_edge_maps.back());
alpar@209
  2671
          readLine(); skipSection();
alpar@209
  2672
        } else if (section == "attributes") {
alpar@209
  2673
          _attribute_sections.push_back(caption);
alpar@209
  2674
          _attributes.push_back(std::vector<std::string>());
alpar@209
  2675
          readAttributes(_attributes.back());
alpar@209
  2676
        } else {
alpar@209
  2677
          _extra_sections.push_back(section);
alpar@209
  2678
          readLine(); skipSection();
alpar@209
  2679
        }
deba@173
  2680
      }
deba@173
  2681
    }
deba@173
  2682
deba@173
  2683
    /// @}
alpar@209
  2684
deba@173
  2685
  };
deba@127
  2686
}
deba@127
  2687
deba@127
  2688
#endif