lemon/lgf_reader.h
author Balazs Dezso <deba@inf.elte.hu>
Wed, 11 Jan 2012 22:58:05 +0100
changeset 1028 4441b066368c
parent 1024 b84e68af8248
child 1029 374a9519986b
permissions -rw-r--r--
Doc fix in BpGraphs (#69)
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@949
     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
deba@598
   104
    template <typename _GR, bool _dir, typename _Map,
alpar@209
   105
              typename _Converter = DefaultConverter<typename _Map::Value> >
deba@598
   106
    class GraphArcMapStorage : public MapStorageBase<typename _GR::Edge> {
deba@165
   107
    public:
deba@165
   108
      typedef _Map Map;
deba@165
   109
      typedef _Converter Converter;
deba@598
   110
      typedef _GR GR;
deba@598
   111
      typedef typename GR::Edge Item;
deba@165
   112
      static const bool dir = _dir;
alpar@209
   113
deba@165
   114
    private:
deba@598
   115
      const GR& _graph;
deba@165
   116
      Map& _map;
deba@165
   117
      Converter _converter;
deba@165
   118
deba@165
   119
    public:
deba@598
   120
      GraphArcMapStorage(const GR& 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@598
   176
    template <typename GR>
deba@165
   177
    struct GraphArcLookUpConverter {
deba@598
   178
      const GR& _graph;
deba@598
   179
      const std::map<std::string, typename GR::Edge>& _map;
deba@598
   180
deba@598
   181
      GraphArcLookUpConverter(const GR& graph,
alpar@209
   182
                              const std::map<std::string,
deba@598
   183
                                             typename GR::Edge>& map)
alpar@209
   184
        : _graph(graph), _map(map) {}
alpar@209
   185
deba@598
   186
      typename GR::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
        }
deba@598
   190
        typename std::map<std::string, typename GR::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@598
   390
  template <typename DGR>
deba@190
   391
  class DigraphReader;
deba@190
   392
deba@598
   393
  template <typename TDGR>
deba@598
   394
  DigraphReader<TDGR> digraphReader(TDGR& digraph, std::istream& is = std::cin);
deba@598
   395
  template <typename TDGR>
deba@598
   396
  DigraphReader<TDGR> digraphReader(TDGR& digraph, const std::string& fn);
deba@598
   397
  template <typename TDGR>
deba@598
   398
  DigraphReader<TDGR> digraphReader(TDGR& digraph, const char *fn);
deba@190
   399
alpar@156
   400
  /// \ingroup lemon_io
alpar@209
   401
  ///
kpeter@192
   402
  /// \brief \ref lgf-format "LGF" reader for directed graphs
alpar@156
   403
  ///
alpar@156
   404
  /// This utility reads an \ref lgf-format "LGF" file.
alpar@156
   405
  ///
alpar@156
   406
  /// The reading method does a batch processing. The user creates a
alpar@156
   407
  /// reader object, then various reading rules can be added to the
alpar@156
   408
  /// reader, and eventually the reading is executed with the \c run()
alpar@156
   409
  /// member function. A map reading rule can be added to the reader
alpar@156
   410
  /// with the \c nodeMap() or \c arcMap() members. An optional
deba@162
   411
  /// converter parameter can also be added as a standard functor
kpeter@192
   412
  /// converting from \c std::string to the value type of the map. If it
deba@162
   413
  /// is set, it will determine how the tokens in the file should be
kpeter@192
   414
  /// converted to the value type of the map. If the functor is not set,
deba@162
   415
  /// then a default conversion will be used. One map can be read into
deba@162
   416
  /// multiple map objects at the same time. The \c attribute(), \c
deba@162
   417
  /// node() and \c arc() functions are used to add attribute reading
deba@162
   418
  /// rules.
alpar@156
   419
  ///
alpar@156
   420
  ///\code
deba@598
   421
  /// DigraphReader<DGR>(digraph, std::cin).
kpeter@192
   422
  ///   nodeMap("coordinates", coord_map).
kpeter@192
   423
  ///   arcMap("capacity", cap_map).
kpeter@192
   424
  ///   node("source", src).
kpeter@192
   425
  ///   node("target", trg).
kpeter@192
   426
  ///   attribute("caption", caption).
kpeter@192
   427
  ///   run();
alpar@156
   428
  ///\endcode
alpar@156
   429
  ///
kpeter@786
   430
  /// By default, the reader uses the first section in the file of the
alpar@156
   431
  /// proper type. If a section has an optional name, then it can be
deba@162
   432
  /// selected for reading by giving an optional name parameter to the
deba@189
   433
  /// \c nodes(), \c arcs() or \c attributes() functions.
alpar@156
   434
  ///
alpar@156
   435
  /// The \c useNodes() and \c useArcs() functions are used to tell the reader
alpar@156
   436
  /// that the nodes or arcs should not be constructed (added to the
alpar@156
   437
  /// graph) during the reading, but instead the label map of the items
alpar@156
   438
  /// are given as a parameter of these functions. An
kpeter@192
   439
  /// application of these functions is multipass reading, which is
kpeter@192
   440
  /// important if two \c \@arcs sections must be read from the
kpeter@192
   441
  /// file. In this case the first phase would read the node set and one
alpar@156
   442
  /// of the arc sets, while the second phase would read the second arc
alpar@156
   443
  /// set into an \e ArcSet class (\c SmartArcSet or \c ListArcSet).
alpar@156
   444
  /// The previously read label node map should be passed to the \c
alpar@156
   445
  /// useNodes() functions. Another application of multipass reading when
alpar@210
   446
  /// paths are given as a node map or an arc map.
alpar@210
   447
  /// It is impossible to read this in
alpar@156
   448
  /// a single pass, because the arcs are not constructed when the node
alpar@156
   449
  /// maps are read.
deba@598
   450
  template <typename DGR>
deba@127
   451
  class DigraphReader {
deba@127
   452
  public:
deba@127
   453
deba@598
   454
    typedef DGR Digraph;
kpeter@559
   455
kpeter@559
   456
  private:
kpeter@559
   457
deba@598
   458
    TEMPLATE_DIGRAPH_TYPEDEFS(DGR);
alpar@209
   459
deba@127
   460
    std::istream* _is;
deba@127
   461
    bool local_is;
deba@290
   462
    std::string _filename;
deba@127
   463
deba@598
   464
    DGR& _digraph;
deba@127
   465
deba@127
   466
    std::string _nodes_caption;
deba@127
   467
    std::string _arcs_caption;
deba@127
   468
    std::string _attributes_caption;
deba@127
   469
deba@127
   470
    typedef std::map<std::string, Node> NodeIndex;
deba@127
   471
    NodeIndex _node_index;
deba@127
   472
    typedef std::map<std::string, Arc> ArcIndex;
deba@127
   473
    ArcIndex _arc_index;
alpar@209
   474
alpar@209
   475
    typedef std::vector<std::pair<std::string,
alpar@209
   476
      _reader_bits::MapStorageBase<Node>*> > NodeMaps;
alpar@209
   477
    NodeMaps _node_maps;
deba@127
   478
deba@127
   479
    typedef std::vector<std::pair<std::string,
deba@127
   480
      _reader_bits::MapStorageBase<Arc>*> >ArcMaps;
deba@127
   481
    ArcMaps _arc_maps;
deba@127
   482
alpar@209
   483
    typedef std::multimap<std::string, _reader_bits::ValueStorageBase*>
deba@127
   484
      Attributes;
deba@127
   485
    Attributes _attributes;
deba@127
   486
deba@127
   487
    bool _use_nodes;
deba@127
   488
    bool _use_arcs;
deba@127
   489
deba@188
   490
    bool _skip_nodes;
deba@188
   491
    bool _skip_arcs;
deba@188
   492
deba@127
   493
    int line_num;
deba@127
   494
    std::istringstream line;
deba@127
   495
deba@127
   496
  public:
deba@127
   497
alpar@156
   498
    /// \brief Constructor
alpar@156
   499
    ///
alpar@156
   500
    /// Construct a directed graph reader, which reads from the given
alpar@156
   501
    /// input stream.
deba@598
   502
    DigraphReader(DGR& digraph, std::istream& is = std::cin)
deba@127
   503
      : _is(&is), local_is(false), _digraph(digraph),
alpar@209
   504
        _use_nodes(false), _use_arcs(false),
alpar@209
   505
        _skip_nodes(false), _skip_arcs(false) {}
deba@127
   506
alpar@156
   507
    /// \brief Constructor
alpar@156
   508
    ///
alpar@156
   509
    /// Construct a directed graph reader, which reads from the given
alpar@156
   510
    /// file.
deba@598
   511
    DigraphReader(DGR& digraph, const std::string& fn)
deba@290
   512
      : _is(new std::ifstream(fn.c_str())), local_is(true),
deba@290
   513
        _filename(fn), _digraph(digraph),
kpeter@212
   514
        _use_nodes(false), _use_arcs(false),
deba@290
   515
        _skip_nodes(false), _skip_arcs(false) {
deba@295
   516
      if (!(*_is)) {
deba@295
   517
        delete _is;
deba@295
   518
        throw IoError("Cannot open file", fn);
deba@295
   519
      }
deba@290
   520
    }
alpar@209
   521
alpar@156
   522
    /// \brief Constructor
alpar@156
   523
    ///
alpar@156
   524
    /// Construct a directed graph reader, which reads from the given
alpar@156
   525
    /// file.
deba@598
   526
    DigraphReader(DGR& digraph, const char* fn)
deba@290
   527
      : _is(new std::ifstream(fn)), local_is(true),
deba@290
   528
        _filename(fn), _digraph(digraph),
kpeter@212
   529
        _use_nodes(false), _use_arcs(false),
deba@290
   530
        _skip_nodes(false), _skip_arcs(false) {
deba@295
   531
      if (!(*_is)) {
deba@295
   532
        delete _is;
deba@295
   533
        throw IoError("Cannot open file", fn);
deba@295
   534
      }
deba@290
   535
    }
deba@127
   536
alpar@156
   537
    /// \brief Destructor
deba@127
   538
    ~DigraphReader() {
alpar@209
   539
      for (typename NodeMaps::iterator it = _node_maps.begin();
alpar@209
   540
           it != _node_maps.end(); ++it) {
alpar@209
   541
        delete it->second;
deba@127
   542
      }
deba@127
   543
alpar@209
   544
      for (typename ArcMaps::iterator it = _arc_maps.begin();
alpar@209
   545
           it != _arc_maps.end(); ++it) {
alpar@209
   546
        delete it->second;
deba@127
   547
      }
deba@127
   548
alpar@209
   549
      for (typename Attributes::iterator it = _attributes.begin();
alpar@209
   550
           it != _attributes.end(); ++it) {
alpar@209
   551
        delete it->second;
deba@127
   552
      }
deba@127
   553
deba@127
   554
      if (local_is) {
alpar@209
   555
        delete _is;
deba@127
   556
      }
deba@127
   557
deba@127
   558
    }
deba@127
   559
deba@127
   560
  private:
deba@190
   561
deba@598
   562
    template <typename TDGR>
deba@598
   563
    friend DigraphReader<TDGR> digraphReader(TDGR& digraph, std::istream& is);
deba@598
   564
    template <typename TDGR>
alpar@877
   565
    friend DigraphReader<TDGR> digraphReader(TDGR& digraph,
deba@598
   566
                                             const std::string& fn);
deba@598
   567
    template <typename TDGR>
deba@598
   568
    friend DigraphReader<TDGR> digraphReader(TDGR& digraph, const char *fn);
alpar@209
   569
alpar@209
   570
    DigraphReader(DigraphReader& other)
deba@190
   571
      : _is(other._is), local_is(other.local_is), _digraph(other._digraph),
alpar@209
   572
        _use_nodes(other._use_nodes), _use_arcs(other._use_arcs),
alpar@209
   573
        _skip_nodes(other._skip_nodes), _skip_arcs(other._skip_arcs) {
deba@190
   574
deba@190
   575
      other._is = 0;
deba@190
   576
      other.local_is = false;
alpar@209
   577
deba@190
   578
      _node_index.swap(other._node_index);
deba@190
   579
      _arc_index.swap(other._arc_index);
deba@190
   580
deba@190
   581
      _node_maps.swap(other._node_maps);
deba@190
   582
      _arc_maps.swap(other._arc_maps);
deba@190
   583
      _attributes.swap(other._attributes);
deba@190
   584
deba@190
   585
      _nodes_caption = other._nodes_caption;
deba@190
   586
      _arcs_caption = other._arcs_caption;
deba@190
   587
      _attributes_caption = other._attributes_caption;
deba@190
   588
deba@190
   589
    }
deba@190
   590
deba@127
   591
    DigraphReader& operator=(const DigraphReader&);
deba@127
   592
deba@127
   593
  public:
deba@127
   594
kpeter@584
   595
    /// \name Reading Rules
alpar@156
   596
    /// @{
alpar@209
   597
alpar@156
   598
    /// \brief Node map reading rule
alpar@156
   599
    ///
alpar@156
   600
    /// Add a node map reading rule to the reader.
deba@127
   601
    template <typename Map>
deba@127
   602
    DigraphReader& nodeMap(const std::string& caption, Map& map) {
deba@127
   603
      checkConcept<concepts::WriteMap<Node, typename Map::Value>, Map>();
alpar@209
   604
      _reader_bits::MapStorageBase<Node>* storage =
alpar@209
   605
        new _reader_bits::MapStorage<Node, Map>(map);
deba@127
   606
      _node_maps.push_back(std::make_pair(caption, storage));
deba@127
   607
      return *this;
deba@127
   608
    }
deba@127
   609
alpar@156
   610
    /// \brief Node map reading rule
alpar@156
   611
    ///
alpar@156
   612
    /// Add a node map reading rule with specialized converter to the
alpar@156
   613
    /// reader.
deba@127
   614
    template <typename Map, typename Converter>
alpar@209
   615
    DigraphReader& nodeMap(const std::string& caption, Map& map,
alpar@209
   616
                           const Converter& converter = Converter()) {
deba@127
   617
      checkConcept<concepts::WriteMap<Node, typename Map::Value>, Map>();
alpar@209
   618
      _reader_bits::MapStorageBase<Node>* storage =
alpar@209
   619
        new _reader_bits::MapStorage<Node, Map, Converter>(map, converter);
deba@127
   620
      _node_maps.push_back(std::make_pair(caption, storage));
deba@127
   621
      return *this;
deba@127
   622
    }
deba@127
   623
alpar@156
   624
    /// \brief Arc map reading rule
alpar@156
   625
    ///
alpar@156
   626
    /// Add an arc map reading rule to the reader.
deba@127
   627
    template <typename Map>
deba@127
   628
    DigraphReader& arcMap(const std::string& caption, Map& map) {
deba@127
   629
      checkConcept<concepts::WriteMap<Arc, typename Map::Value>, Map>();
alpar@209
   630
      _reader_bits::MapStorageBase<Arc>* storage =
alpar@209
   631
        new _reader_bits::MapStorage<Arc, Map>(map);
deba@127
   632
      _arc_maps.push_back(std::make_pair(caption, storage));
deba@127
   633
      return *this;
deba@127
   634
    }
deba@127
   635
alpar@156
   636
    /// \brief Arc map reading rule
alpar@156
   637
    ///
alpar@156
   638
    /// Add an arc map reading rule with specialized converter to the
alpar@156
   639
    /// reader.
deba@127
   640
    template <typename Map, typename Converter>
alpar@209
   641
    DigraphReader& arcMap(const std::string& caption, Map& map,
alpar@209
   642
                          const Converter& converter = Converter()) {
deba@127
   643
      checkConcept<concepts::WriteMap<Arc, typename Map::Value>, Map>();
alpar@209
   644
      _reader_bits::MapStorageBase<Arc>* storage =
alpar@209
   645
        new _reader_bits::MapStorage<Arc, Map, Converter>(map, converter);
deba@127
   646
      _arc_maps.push_back(std::make_pair(caption, storage));
deba@127
   647
      return *this;
deba@127
   648
    }
deba@127
   649
alpar@156
   650
    /// \brief Attribute reading rule
alpar@156
   651
    ///
alpar@156
   652
    /// Add an attribute reading rule to the reader.
deba@127
   653
    template <typename Value>
deba@127
   654
    DigraphReader& attribute(const std::string& caption, Value& value) {
alpar@209
   655
      _reader_bits::ValueStorageBase* storage =
alpar@209
   656
        new _reader_bits::ValueStorage<Value>(value);
deba@127
   657
      _attributes.insert(std::make_pair(caption, storage));
deba@127
   658
      return *this;
deba@127
   659
    }
deba@127
   660
alpar@156
   661
    /// \brief Attribute reading rule
alpar@156
   662
    ///
alpar@156
   663
    /// Add an attribute reading rule with specialized converter to the
alpar@156
   664
    /// reader.
deba@127
   665
    template <typename Value, typename Converter>
alpar@209
   666
    DigraphReader& attribute(const std::string& caption, Value& value,
alpar@209
   667
                             const Converter& converter = Converter()) {
alpar@209
   668
      _reader_bits::ValueStorageBase* storage =
alpar@209
   669
        new _reader_bits::ValueStorage<Value, Converter>(value, converter);
deba@127
   670
      _attributes.insert(std::make_pair(caption, storage));
deba@127
   671
      return *this;
deba@127
   672
    }
deba@127
   673
alpar@156
   674
    /// \brief Node reading rule
alpar@156
   675
    ///
alpar@156
   676
    /// Add a node reading rule to reader.
deba@127
   677
    DigraphReader& node(const std::string& caption, Node& node) {
deba@127
   678
      typedef _reader_bits::MapLookUpConverter<Node> Converter;
deba@127
   679
      Converter converter(_node_index);
alpar@209
   680
      _reader_bits::ValueStorageBase* storage =
alpar@209
   681
        new _reader_bits::ValueStorage<Node, Converter>(node, converter);
deba@127
   682
      _attributes.insert(std::make_pair(caption, storage));
deba@127
   683
      return *this;
deba@127
   684
    }
deba@127
   685
alpar@156
   686
    /// \brief Arc reading rule
alpar@156
   687
    ///
alpar@156
   688
    /// Add an arc reading rule to reader.
deba@127
   689
    DigraphReader& arc(const std::string& caption, Arc& arc) {
deba@127
   690
      typedef _reader_bits::MapLookUpConverter<Arc> Converter;
deba@127
   691
      Converter converter(_arc_index);
alpar@209
   692
      _reader_bits::ValueStorageBase* storage =
alpar@209
   693
        new _reader_bits::ValueStorage<Arc, Converter>(arc, converter);
deba@127
   694
      _attributes.insert(std::make_pair(caption, storage));
deba@127
   695
      return *this;
deba@127
   696
    }
deba@127
   697
alpar@156
   698
    /// @}
alpar@156
   699
kpeter@584
   700
    /// \name Select Section by Name
alpar@156
   701
    /// @{
alpar@156
   702
alpar@156
   703
    /// \brief Set \c \@nodes section to be read
alpar@156
   704
    ///
alpar@156
   705
    /// Set \c \@nodes section to be read
deba@127
   706
    DigraphReader& nodes(const std::string& caption) {
deba@127
   707
      _nodes_caption = caption;
deba@127
   708
      return *this;
deba@127
   709
    }
deba@127
   710
alpar@156
   711
    /// \brief Set \c \@arcs section to be read
alpar@156
   712
    ///
alpar@156
   713
    /// Set \c \@arcs section to be read
deba@127
   714
    DigraphReader& arcs(const std::string& caption) {
deba@127
   715
      _arcs_caption = caption;
deba@127
   716
      return *this;
deba@127
   717
    }
deba@127
   718
alpar@156
   719
    /// \brief Set \c \@attributes section to be read
alpar@156
   720
    ///
alpar@156
   721
    /// Set \c \@attributes section to be read
deba@127
   722
    DigraphReader& attributes(const std::string& caption) {
deba@127
   723
      _attributes_caption = caption;
deba@127
   724
      return *this;
deba@127
   725
    }
deba@127
   726
alpar@156
   727
    /// @}
alpar@156
   728
kpeter@584
   729
    /// \name Using Previously Constructed Node or Arc Set
alpar@156
   730
    /// @{
alpar@156
   731
alpar@156
   732
    /// \brief Use previously constructed node set
alpar@156
   733
    ///
alpar@156
   734
    /// Use previously constructed node set, and specify the node
alpar@156
   735
    /// label map.
deba@127
   736
    template <typename Map>
deba@127
   737
    DigraphReader& useNodes(const Map& map) {
deba@127
   738
      checkConcept<concepts::ReadMap<Node, typename Map::Value>, Map>();
alpar@209
   739
      LEMON_ASSERT(!_use_nodes, "Multiple usage of useNodes() member");
deba@127
   740
      _use_nodes = true;
deba@127
   741
      _writer_bits::DefaultConverter<typename Map::Value> converter;
deba@127
   742
      for (NodeIt n(_digraph); n != INVALID; ++n) {
alpar@209
   743
        _node_index.insert(std::make_pair(converter(map[n]), n));
deba@127
   744
      }
deba@127
   745
      return *this;
deba@127
   746
    }
deba@127
   747
alpar@156
   748
    /// \brief Use previously constructed node set
alpar@156
   749
    ///
alpar@156
   750
    /// Use previously constructed node set, and specify the node
alpar@156
   751
    /// label map and a functor which converts the label map values to
kpeter@192
   752
    /// \c std::string.
deba@127
   753
    template <typename Map, typename Converter>
alpar@209
   754
    DigraphReader& useNodes(const Map& map,
alpar@209
   755
                            const Converter& converter = Converter()) {
deba@127
   756
      checkConcept<concepts::ReadMap<Node, typename Map::Value>, Map>();
alpar@209
   757
      LEMON_ASSERT(!_use_nodes, "Multiple usage of useNodes() member");
deba@127
   758
      _use_nodes = true;
deba@127
   759
      for (NodeIt n(_digraph); n != INVALID; ++n) {
alpar@209
   760
        _node_index.insert(std::make_pair(converter(map[n]), n));
deba@127
   761
      }
deba@127
   762
      return *this;
deba@127
   763
    }
deba@127
   764
alpar@156
   765
    /// \brief Use previously constructed arc set
alpar@156
   766
    ///
alpar@156
   767
    /// Use previously constructed arc set, and specify the arc
alpar@156
   768
    /// label map.
deba@127
   769
    template <typename Map>
deba@127
   770
    DigraphReader& useArcs(const Map& map) {
deba@127
   771
      checkConcept<concepts::ReadMap<Arc, typename Map::Value>, Map>();
deba@127
   772
      LEMON_ASSERT(!_use_arcs, "Multiple usage of useArcs() member");
deba@127
   773
      _use_arcs = true;
deba@127
   774
      _writer_bits::DefaultConverter<typename Map::Value> converter;
deba@127
   775
      for (ArcIt a(_digraph); a != INVALID; ++a) {
alpar@209
   776
        _arc_index.insert(std::make_pair(converter(map[a]), a));
deba@127
   777
      }
deba@127
   778
      return *this;
deba@127
   779
    }
deba@127
   780
alpar@156
   781
    /// \brief Use previously constructed arc set
alpar@156
   782
    ///
alpar@156
   783
    /// Use previously constructed arc set, and specify the arc
alpar@156
   784
    /// label map and a functor which converts the label map values to
kpeter@192
   785
    /// \c std::string.
deba@127
   786
    template <typename Map, typename Converter>
alpar@209
   787
    DigraphReader& useArcs(const Map& map,
alpar@209
   788
                           const Converter& converter = Converter()) {
deba@127
   789
      checkConcept<concepts::ReadMap<Arc, typename Map::Value>, Map>();
alpar@209
   790
      LEMON_ASSERT(!_use_arcs, "Multiple usage of useArcs() member");
deba@127
   791
      _use_arcs = true;
deba@127
   792
      for (ArcIt a(_digraph); a != INVALID; ++a) {
alpar@209
   793
        _arc_index.insert(std::make_pair(converter(map[a]), a));
deba@127
   794
      }
deba@127
   795
      return *this;
deba@127
   796
    }
deba@127
   797
deba@188
   798
    /// \brief Skips the reading of node section
deba@188
   799
    ///
deba@188
   800
    /// Omit the reading of the node section. This implies that each node
kpeter@192
   801
    /// map reading rule will be abandoned, and the nodes of the graph
deba@188
   802
    /// will not be constructed, which usually cause that the arc set
kpeter@192
   803
    /// could not be read due to lack of node name resolving.
kpeter@192
   804
    /// Therefore \c skipArcs() function should also be used, or
kpeter@192
   805
    /// \c useNodes() should be used to specify the label of the nodes.
deba@188
   806
    DigraphReader& skipNodes() {
alpar@209
   807
      LEMON_ASSERT(!_skip_nodes, "Skip nodes already set");
deba@188
   808
      _skip_nodes = true;
deba@188
   809
      return *this;
deba@188
   810
    }
deba@188
   811
deba@188
   812
    /// \brief Skips the reading of arc section
deba@188
   813
    ///
deba@188
   814
    /// Omit the reading of the arc section. This implies that each arc
kpeter@192
   815
    /// map reading rule will be abandoned, and the arcs of the graph
deba@188
   816
    /// will not be constructed.
deba@188
   817
    DigraphReader& skipArcs() {
alpar@209
   818
      LEMON_ASSERT(!_skip_arcs, "Skip arcs already set");
deba@188
   819
      _skip_arcs = true;
deba@188
   820
      return *this;
deba@188
   821
    }
deba@188
   822
alpar@156
   823
    /// @}
alpar@156
   824
deba@127
   825
  private:
deba@127
   826
deba@127
   827
    bool readLine() {
deba@127
   828
      std::string str;
deba@127
   829
      while(++line_num, std::getline(*_is, str)) {
alpar@209
   830
        line.clear(); line.str(str);
alpar@209
   831
        char c;
alpar@209
   832
        if (line >> std::ws >> c && c != '#') {
alpar@209
   833
          line.putback(c);
alpar@209
   834
          return true;
alpar@209
   835
        }
deba@127
   836
      }
deba@127
   837
      return false;
deba@127
   838
    }
deba@127
   839
deba@127
   840
    bool readSuccess() {
deba@127
   841
      return static_cast<bool>(*_is);
deba@127
   842
    }
alpar@209
   843
deba@127
   844
    void skipSection() {
deba@127
   845
      char c;
deba@127
   846
      while (readSuccess() && line >> c && c != '@') {
alpar@209
   847
        readLine();
deba@127
   848
      }
deba@427
   849
      if (readSuccess()) {
deba@427
   850
        line.putback(c);
deba@427
   851
      }
deba@127
   852
    }
deba@127
   853
deba@127
   854
    void readNodes() {
deba@127
   855
deba@127
   856
      std::vector<int> map_index(_node_maps.size());
deba@127
   857
      int map_num, label_index;
deba@127
   858
deba@186
   859
      char c;
deba@186
   860
      if (!readLine() || !(line >> c) || c == '@') {
alpar@209
   861
        if (readSuccess() && line) line.putback(c);
alpar@209
   862
        if (!_node_maps.empty())
deba@290
   863
          throw FormatError("Cannot find map names");
alpar@209
   864
        return;
deba@186
   865
      }
deba@186
   866
      line.putback(c);
deba@186
   867
deba@127
   868
      {
alpar@209
   869
        std::map<std::string, int> maps;
alpar@209
   870
alpar@209
   871
        std::string map;
alpar@209
   872
        int index = 0;
alpar@209
   873
        while (_reader_bits::readToken(line, map)) {
alpar@209
   874
          if (maps.find(map) != maps.end()) {
alpar@209
   875
            std::ostringstream msg;
alpar@209
   876
            msg << "Multiple occurence of node map: " << map;
deba@290
   877
            throw FormatError(msg.str());
alpar@209
   878
          }
alpar@209
   879
          maps.insert(std::make_pair(map, index));
alpar@209
   880
          ++index;
alpar@209
   881
        }
alpar@209
   882
alpar@209
   883
        for (int i = 0; i < static_cast<int>(_node_maps.size()); ++i) {
alpar@209
   884
          std::map<std::string, int>::iterator jt =
alpar@209
   885
            maps.find(_node_maps[i].first);
alpar@209
   886
          if (jt == maps.end()) {
alpar@209
   887
            std::ostringstream msg;
kpeter@291
   888
            msg << "Map not found: " << _node_maps[i].first;
deba@290
   889
            throw FormatError(msg.str());
alpar@209
   890
          }
alpar@209
   891
          map_index[i] = jt->second;
alpar@209
   892
        }
alpar@209
   893
alpar@209
   894
        {
alpar@209
   895
          std::map<std::string, int>::iterator jt = maps.find("label");
alpar@209
   896
          if (jt != maps.end()) {
alpar@209
   897
            label_index = jt->second;
alpar@209
   898
          } else {
alpar@209
   899
            label_index = -1;
alpar@209
   900
          }
alpar@209
   901
        }
alpar@209
   902
        map_num = maps.size();
deba@127
   903
      }
deba@127
   904
deba@127
   905
      while (readLine() && line >> c && c != '@') {
alpar@209
   906
        line.putback(c);
alpar@209
   907
alpar@209
   908
        std::vector<std::string> tokens(map_num);
alpar@209
   909
        for (int i = 0; i < map_num; ++i) {
alpar@209
   910
          if (!_reader_bits::readToken(line, tokens[i])) {
alpar@209
   911
            std::ostringstream msg;
alpar@209
   912
            msg << "Column not found (" << i + 1 << ")";
deba@290
   913
            throw FormatError(msg.str());
alpar@209
   914
          }
alpar@209
   915
        }
alpar@209
   916
        if (line >> std::ws >> c)
kpeter@291
   917
          throw FormatError("Extra character at the end of line");
alpar@209
   918
alpar@209
   919
        Node n;
alpar@209
   920
        if (!_use_nodes) {
alpar@209
   921
          n = _digraph.addNode();
alpar@209
   922
          if (label_index != -1)
alpar@209
   923
            _node_index.insert(std::make_pair(tokens[label_index], n));
alpar@209
   924
        } else {
alpar@209
   925
          if (label_index == -1)
kpeter@291
   926
            throw FormatError("Label map not found");
alpar@209
   927
          typename std::map<std::string, Node>::iterator it =
alpar@209
   928
            _node_index.find(tokens[label_index]);
alpar@209
   929
          if (it == _node_index.end()) {
alpar@209
   930
            std::ostringstream msg;
alpar@209
   931
            msg << "Node with label not found: " << tokens[label_index];
deba@290
   932
            throw FormatError(msg.str());
alpar@209
   933
          }
alpar@209
   934
          n = it->second;
alpar@209
   935
        }
alpar@209
   936
alpar@209
   937
        for (int i = 0; i < static_cast<int>(_node_maps.size()); ++i) {
alpar@209
   938
          _node_maps[i].second->set(n, tokens[map_index[i]]);
alpar@209
   939
        }
deba@127
   940
deba@127
   941
      }
deba@127
   942
      if (readSuccess()) {
alpar@209
   943
        line.putback(c);
deba@127
   944
      }
deba@127
   945
    }
deba@127
   946
deba@127
   947
    void readArcs() {
deba@127
   948
deba@127
   949
      std::vector<int> map_index(_arc_maps.size());
deba@127
   950
      int map_num, label_index;
deba@127
   951
deba@186
   952
      char c;
deba@186
   953
      if (!readLine() || !(line >> c) || c == '@') {
alpar@209
   954
        if (readSuccess() && line) line.putback(c);
alpar@209
   955
        if (!_arc_maps.empty())
deba@290
   956
          throw FormatError("Cannot find map names");
alpar@209
   957
        return;
deba@186
   958
      }
deba@186
   959
      line.putback(c);
alpar@209
   960
deba@127
   961
      {
alpar@209
   962
        std::map<std::string, int> maps;
alpar@209
   963
alpar@209
   964
        std::string map;
alpar@209
   965
        int index = 0;
alpar@209
   966
        while (_reader_bits::readToken(line, map)) {
alpar@949
   967
          if(map == "-") {
alpar@949
   968
              if(index!=0)
alpar@949
   969
                throw FormatError("'-' is not allowed as a map name");
alpar@949
   970
              else if (line >> std::ws >> c)
alpar@949
   971
                throw FormatError("Extra character at the end of line");
alpar@949
   972
              else break;
alpar@949
   973
            }
alpar@209
   974
          if (maps.find(map) != maps.end()) {
alpar@209
   975
            std::ostringstream msg;
alpar@209
   976
            msg << "Multiple occurence of arc map: " << map;
deba@290
   977
            throw FormatError(msg.str());
alpar@209
   978
          }
alpar@209
   979
          maps.insert(std::make_pair(map, index));
alpar@209
   980
          ++index;
alpar@209
   981
        }
alpar@209
   982
alpar@209
   983
        for (int i = 0; i < static_cast<int>(_arc_maps.size()); ++i) {
alpar@209
   984
          std::map<std::string, int>::iterator jt =
alpar@209
   985
            maps.find(_arc_maps[i].first);
alpar@209
   986
          if (jt == maps.end()) {
alpar@209
   987
            std::ostringstream msg;
kpeter@291
   988
            msg << "Map not found: " << _arc_maps[i].first;
deba@290
   989
            throw FormatError(msg.str());
alpar@209
   990
          }
alpar@209
   991
          map_index[i] = jt->second;
alpar@209
   992
        }
alpar@209
   993
alpar@209
   994
        {
alpar@209
   995
          std::map<std::string, int>::iterator jt = maps.find("label");
alpar@209
   996
          if (jt != maps.end()) {
alpar@209
   997
            label_index = jt->second;
alpar@209
   998
          } else {
alpar@209
   999
            label_index = -1;
alpar@209
  1000
          }
alpar@209
  1001
        }
alpar@209
  1002
        map_num = maps.size();
deba@127
  1003
      }
deba@127
  1004
deba@127
  1005
      while (readLine() && line >> c && c != '@') {
alpar@209
  1006
        line.putback(c);
alpar@209
  1007
alpar@209
  1008
        std::string source_token;
alpar@209
  1009
        std::string target_token;
alpar@209
  1010
alpar@209
  1011
        if (!_reader_bits::readToken(line, source_token))
deba@290
  1012
          throw FormatError("Source not found");
alpar@209
  1013
alpar@209
  1014
        if (!_reader_bits::readToken(line, target_token))
deba@290
  1015
          throw FormatError("Target not found");
alpar@209
  1016
alpar@209
  1017
        std::vector<std::string> tokens(map_num);
alpar@209
  1018
        for (int i = 0; i < map_num; ++i) {
alpar@209
  1019
          if (!_reader_bits::readToken(line, tokens[i])) {
alpar@209
  1020
            std::ostringstream msg;
alpar@209
  1021
            msg << "Column not found (" << i + 1 << ")";
deba@290
  1022
            throw FormatError(msg.str());
alpar@209
  1023
          }
alpar@209
  1024
        }
alpar@209
  1025
        if (line >> std::ws >> c)
kpeter@291
  1026
          throw FormatError("Extra character at the end of line");
alpar@209
  1027
alpar@209
  1028
        Arc a;
alpar@209
  1029
        if (!_use_arcs) {
deba@127
  1030
deba@127
  1031
          typename NodeIndex::iterator it;
alpar@209
  1032
deba@127
  1033
          it = _node_index.find(source_token);
deba@127
  1034
          if (it == _node_index.end()) {
deba@127
  1035
            std::ostringstream msg;
deba@127
  1036
            msg << "Item not found: " << source_token;
deba@290
  1037
            throw FormatError(msg.str());
deba@127
  1038
          }
deba@127
  1039
          Node source = it->second;
deba@127
  1040
deba@127
  1041
          it = _node_index.find(target_token);
alpar@209
  1042
          if (it == _node_index.end()) {
alpar@209
  1043
            std::ostringstream msg;
deba@127
  1044
            msg << "Item not found: " << target_token;
deba@290
  1045
            throw FormatError(msg.str());
alpar@209
  1046
          }
alpar@209
  1047
          Node target = it->second;
alpar@209
  1048
alpar@209
  1049
          a = _digraph.addArc(source, target);
alpar@209
  1050
          if (label_index != -1)
alpar@209
  1051
            _arc_index.insert(std::make_pair(tokens[label_index], a));
alpar@209
  1052
        } else {
alpar@209
  1053
          if (label_index == -1)
kpeter@291
  1054
            throw FormatError("Label map not found");
alpar@209
  1055
          typename std::map<std::string, Arc>::iterator it =
alpar@209
  1056
            _arc_index.find(tokens[label_index]);
alpar@209
  1057
          if (it == _arc_index.end()) {
alpar@209
  1058
            std::ostringstream msg;
alpar@209
  1059
            msg << "Arc with label not found: " << tokens[label_index];
deba@290
  1060
            throw FormatError(msg.str());
alpar@209
  1061
          }
alpar@209
  1062
          a = it->second;
alpar@209
  1063
        }
alpar@209
  1064
alpar@209
  1065
        for (int i = 0; i < static_cast<int>(_arc_maps.size()); ++i) {
alpar@209
  1066
          _arc_maps[i].second->set(a, tokens[map_index[i]]);
alpar@209
  1067
        }
deba@127
  1068
deba@127
  1069
      }
deba@127
  1070
      if (readSuccess()) {
alpar@209
  1071
        line.putback(c);
deba@127
  1072
      }
deba@127
  1073
    }
deba@127
  1074
deba@127
  1075
    void readAttributes() {
deba@127
  1076
deba@127
  1077
      std::set<std::string> read_attr;
deba@127
  1078
deba@127
  1079
      char c;
deba@127
  1080
      while (readLine() && line >> c && c != '@') {
alpar@209
  1081
        line.putback(c);
alpar@209
  1082
alpar@209
  1083
        std::string attr, token;
alpar@209
  1084
        if (!_reader_bits::readToken(line, attr))
deba@290
  1085
          throw FormatError("Attribute name not found");
alpar@209
  1086
        if (!_reader_bits::readToken(line, token))
deba@290
  1087
          throw FormatError("Attribute value not found");
alpar@209
  1088
        if (line >> c)
kpeter@291
  1089
          throw FormatError("Extra character at the end of line");
alpar@209
  1090
alpar@209
  1091
        {
alpar@209
  1092
          std::set<std::string>::iterator it = read_attr.find(attr);
alpar@209
  1093
          if (it != read_attr.end()) {
alpar@209
  1094
            std::ostringstream msg;
kpeter@291
  1095
            msg << "Multiple occurence of attribute: " << attr;
deba@290
  1096
            throw FormatError(msg.str());
alpar@209
  1097
          }
alpar@209
  1098
          read_attr.insert(attr);
alpar@209
  1099
        }
alpar@209
  1100
alpar@209
  1101
        {
alpar@209
  1102
          typename Attributes::iterator it = _attributes.lower_bound(attr);
alpar@209
  1103
          while (it != _attributes.end() && it->first == attr) {
alpar@209
  1104
            it->second->set(token);
alpar@209
  1105
            ++it;
alpar@209
  1106
          }
alpar@209
  1107
        }
deba@127
  1108
deba@127
  1109
      }
deba@127
  1110
      if (readSuccess()) {
alpar@209
  1111
        line.putback(c);
deba@127
  1112
      }
deba@127
  1113
      for (typename Attributes::iterator it = _attributes.begin();
alpar@209
  1114
           it != _attributes.end(); ++it) {
alpar@209
  1115
        if (read_attr.find(it->first) == read_attr.end()) {
alpar@209
  1116
          std::ostringstream msg;
kpeter@291
  1117
          msg << "Attribute not found: " << it->first;
deba@290
  1118
          throw FormatError(msg.str());
alpar@209
  1119
        }
deba@127
  1120
      }
deba@127
  1121
    }
deba@127
  1122
deba@127
  1123
  public:
alpar@156
  1124
kpeter@584
  1125
    /// \name Execution of the Reader
alpar@156
  1126
    /// @{
alpar@156
  1127
alpar@156
  1128
    /// \brief Start the batch processing
alpar@156
  1129
    ///
alpar@156
  1130
    /// This function starts the batch processing
deba@127
  1131
    void run() {
deba@127
  1132
      LEMON_ASSERT(_is != 0, "This reader assigned to an other reader");
alpar@209
  1133
deba@188
  1134
      bool nodes_done = _skip_nodes;
deba@188
  1135
      bool arcs_done = _skip_arcs;
deba@127
  1136
      bool attributes_done = false;
deba@127
  1137
alpar@209
  1138
      line_num = 0;
deba@127
  1139
      readLine();
deba@172
  1140
      skipSection();
deba@127
  1141
deba@127
  1142
      while (readSuccess()) {
alpar@209
  1143
        try {
alpar@209
  1144
          char c;
alpar@209
  1145
          std::string section, caption;
alpar@209
  1146
          line >> c;
alpar@209
  1147
          _reader_bits::readToken(line, section);
alpar@209
  1148
          _reader_bits::readToken(line, caption);
alpar@209
  1149
alpar@209
  1150
          if (line >> c)
kpeter@291
  1151
            throw FormatError("Extra character at the end of line");
alpar@209
  1152
alpar@209
  1153
          if (section == "nodes" && !nodes_done) {
alpar@209
  1154
            if (_nodes_caption.empty() || _nodes_caption == caption) {
alpar@209
  1155
              readNodes();
alpar@209
  1156
              nodes_done = true;
alpar@209
  1157
            }
alpar@209
  1158
          } else if ((section == "arcs" || section == "edges") &&
alpar@209
  1159
                     !arcs_done) {
alpar@209
  1160
            if (_arcs_caption.empty() || _arcs_caption == caption) {
alpar@209
  1161
              readArcs();
alpar@209
  1162
              arcs_done = true;
alpar@209
  1163
            }
alpar@209
  1164
          } else if (section == "attributes" && !attributes_done) {
alpar@209
  1165
            if (_attributes_caption.empty() || _attributes_caption == caption) {
alpar@209
  1166
              readAttributes();
alpar@209
  1167
              attributes_done = true;
alpar@209
  1168
            }
alpar@209
  1169
          } else {
alpar@209
  1170
            readLine();
alpar@209
  1171
            skipSection();
alpar@209
  1172
          }
deba@290
  1173
        } catch (FormatError& error) {
alpar@209
  1174
          error.line(line_num);
deba@290
  1175
          error.file(_filename);
alpar@209
  1176
          throw;
alpar@209
  1177
        }
deba@127
  1178
      }
deba@127
  1179
deba@127
  1180
      if (!nodes_done) {
deba@290
  1181
        throw FormatError("Section @nodes not found");
deba@127
  1182
      }
deba@127
  1183
deba@127
  1184
      if (!arcs_done) {
deba@290
  1185
        throw FormatError("Section @arcs not found");
deba@127
  1186
      }
deba@127
  1187
deba@127
  1188
      if (!attributes_done && !_attributes.empty()) {
deba@290
  1189
        throw FormatError("Section @attributes not found");
deba@127
  1190
      }
deba@127
  1191
deba@127
  1192
    }
alpar@156
  1193
alpar@156
  1194
    /// @}
alpar@209
  1195
deba@127
  1196
  };
alpar@877
  1197
deba@598
  1198
  /// \ingroup lemon_io
deba@598
  1199
  ///
deba@598
  1200
  /// \brief Return a \ref DigraphReader class
deba@598
  1201
  ///
deba@598
  1202
  /// This function just returns a \ref DigraphReader class.
deba@598
  1203
  ///
alpar@877
  1204
  /// With this function a digraph can be read from an
deba@598
  1205
  /// \ref lgf-format "LGF" file or input stream with several maps and
deba@598
  1206
  /// attributes. For example, there is network flow problem on a
deba@598
  1207
  /// digraph, i.e. a digraph with a \e capacity map on the arcs and
deba@598
  1208
  /// \e source and \e target nodes. This digraph can be read with the
deba@598
  1209
  /// following code:
deba@598
  1210
  ///
deba@598
  1211
  ///\code
deba@598
  1212
  ///ListDigraph digraph;
deba@598
  1213
  ///ListDigraph::ArcMap<int> cm(digraph);
deba@598
  1214
  ///ListDigraph::Node src, trg;
deba@598
  1215
  ///digraphReader(digraph, std::cin).
deba@598
  1216
  ///  arcMap("capacity", cap).
deba@598
  1217
  ///  node("source", src).
deba@598
  1218
  ///  node("target", trg).
deba@598
  1219
  ///  run();
deba@598
  1220
  ///\endcode
deba@598
  1221
  ///
deba@598
  1222
  /// For a complete documentation, please see the \ref DigraphReader
deba@598
  1223
  /// class documentation.
deba@598
  1224
  /// \warning Don't forget to put the \ref DigraphReader::run() "run()"
deba@598
  1225
  /// to the end of the parameter list.
deba@598
  1226
  /// \relates DigraphReader
deba@598
  1227
  /// \sa digraphReader(TDGR& digraph, const std::string& fn)
deba@598
  1228
  /// \sa digraphReader(TDGR& digraph, const char* fn)
deba@598
  1229
  template <typename TDGR>
deba@598
  1230
  DigraphReader<TDGR> digraphReader(TDGR& digraph, std::istream& is) {
deba@598
  1231
    DigraphReader<TDGR> tmp(digraph, is);
deba@598
  1232
    return tmp;
deba@598
  1233
  }
deba@127
  1234
kpeter@498
  1235
  /// \brief Return a \ref DigraphReader class
kpeter@498
  1236
  ///
kpeter@498
  1237
  /// This function just returns a \ref DigraphReader class.
kpeter@498
  1238
  /// \relates DigraphReader
deba@598
  1239
  /// \sa digraphReader(TDGR& digraph, std::istream& is)
deba@598
  1240
  template <typename TDGR>
deba@598
  1241
  DigraphReader<TDGR> digraphReader(TDGR& digraph, const std::string& fn) {
deba@598
  1242
    DigraphReader<TDGR> tmp(digraph, fn);
kpeter@498
  1243
    return tmp;
kpeter@498
  1244
  }
kpeter@498
  1245
kpeter@498
  1246
  /// \brief Return a \ref DigraphReader class
kpeter@498
  1247
  ///
kpeter@498
  1248
  /// This function just returns a \ref DigraphReader class.
kpeter@498
  1249
  /// \relates DigraphReader
deba@598
  1250
  /// \sa digraphReader(TDGR& digraph, std::istream& is)
deba@598
  1251
  template <typename TDGR>
deba@598
  1252
  DigraphReader<TDGR> digraphReader(TDGR& digraph, const char* fn) {
deba@598
  1253
    DigraphReader<TDGR> tmp(digraph, fn);
kpeter@498
  1254
    return tmp;
kpeter@498
  1255
  }
kpeter@498
  1256
deba@598
  1257
  template <typename GR>
ladanyi@303
  1258
  class GraphReader;
alpar@877
  1259
deba@598
  1260
  template <typename TGR>
deba@598
  1261
  GraphReader<TGR> graphReader(TGR& graph, std::istream& is = std::cin);
deba@598
  1262
  template <typename TGR>
deba@598
  1263
  GraphReader<TGR> graphReader(TGR& graph, const std::string& fn);
deba@598
  1264
  template <typename TGR>
deba@598
  1265
  GraphReader<TGR> graphReader(TGR& graph, const char *fn);
deba@165
  1266
deba@165
  1267
  /// \ingroup lemon_io
alpar@209
  1268
  ///
kpeter@192
  1269
  /// \brief \ref lgf-format "LGF" reader for undirected graphs
deba@165
  1270
  ///
deba@165
  1271
  /// This utility reads an \ref lgf-format "LGF" file.
kpeter@192
  1272
  ///
kpeter@192
  1273
  /// It can be used almost the same way as \c DigraphReader.
kpeter@192
  1274
  /// The only difference is that this class can handle edges and
kpeter@192
  1275
  /// edge maps as well as arcs and arc maps.
deba@201
  1276
  ///
deba@201
  1277
  /// The columns in the \c \@edges (or \c \@arcs) section are the
deba@201
  1278
  /// edge maps. However, if there are two maps with the same name
deba@201
  1279
  /// prefixed with \c '+' and \c '-', then these can be read into an
deba@201
  1280
  /// arc map.  Similarly, an attribute can be read into an arc, if
deba@201
  1281
  /// it's value is an edge label prefixed with \c '+' or \c '-'.
kpeter@559
  1282
  template <typename GR>
deba@165
  1283
  class GraphReader {
deba@165
  1284
  public:
deba@165
  1285
kpeter@559
  1286
    typedef GR Graph;
kpeter@559
  1287
kpeter@559
  1288
  private:
kpeter@559
  1289
deba@598
  1290
    TEMPLATE_GRAPH_TYPEDEFS(GR);
alpar@209
  1291
deba@165
  1292
    std::istream* _is;
deba@165
  1293
    bool local_is;
deba@290
  1294
    std::string _filename;
deba@165
  1295
deba@598
  1296
    GR& _graph;
deba@165
  1297
deba@165
  1298
    std::string _nodes_caption;
deba@165
  1299
    std::string _edges_caption;
deba@165
  1300
    std::string _attributes_caption;
deba@165
  1301
deba@165
  1302
    typedef std::map<std::string, Node> NodeIndex;
deba@165
  1303
    NodeIndex _node_index;
deba@165
  1304
    typedef std::map<std::string, Edge> EdgeIndex;
deba@165
  1305
    EdgeIndex _edge_index;
alpar@209
  1306
alpar@209
  1307
    typedef std::vector<std::pair<std::string,
alpar@209
  1308
      _reader_bits::MapStorageBase<Node>*> > NodeMaps;
alpar@209
  1309
    NodeMaps _node_maps;
deba@165
  1310
deba@165
  1311
    typedef std::vector<std::pair<std::string,
deba@165
  1312
      _reader_bits::MapStorageBase<Edge>*> > EdgeMaps;
deba@165
  1313
    EdgeMaps _edge_maps;
deba@165
  1314
alpar@209
  1315
    typedef std::multimap<std::string, _reader_bits::ValueStorageBase*>
deba@165
  1316
      Attributes;
deba@165
  1317
    Attributes _attributes;
deba@165
  1318
deba@165
  1319
    bool _use_nodes;
deba@165
  1320
    bool _use_edges;
deba@165
  1321
deba@188
  1322
    bool _skip_nodes;
deba@188
  1323
    bool _skip_edges;
deba@188
  1324
deba@165
  1325
    int line_num;
deba@165
  1326
    std::istringstream line;
deba@165
  1327
deba@165
  1328
  public:
deba@165
  1329
deba@165
  1330
    /// \brief Constructor
deba@165
  1331
    ///
kpeter@192
  1332
    /// Construct an undirected graph reader, which reads from the given
deba@165
  1333
    /// input stream.
deba@598
  1334
    GraphReader(GR& graph, std::istream& is = std::cin)
deba@165
  1335
      : _is(&is), local_is(false), _graph(graph),
alpar@209
  1336
        _use_nodes(false), _use_edges(false),
alpar@209
  1337
        _skip_nodes(false), _skip_edges(false) {}
deba@165
  1338
deba@165
  1339
    /// \brief Constructor
deba@165
  1340
    ///
kpeter@192
  1341
    /// Construct an undirected graph reader, which reads from the given
deba@165
  1342
    /// file.
deba@598
  1343
    GraphReader(GR& graph, const std::string& fn)
deba@290
  1344
      : _is(new std::ifstream(fn.c_str())), local_is(true),
deba@290
  1345
        _filename(fn), _graph(graph),
kpeter@212
  1346
        _use_nodes(false), _use_edges(false),
deba@290
  1347
        _skip_nodes(false), _skip_edges(false) {
deba@295
  1348
      if (!(*_is)) {
deba@295
  1349
        delete _is;
deba@295
  1350
        throw IoError("Cannot open file", fn);
deba@295
  1351
      }
deba@290
  1352
    }
alpar@209
  1353
deba@165
  1354
    /// \brief Constructor
deba@165
  1355
    ///
kpeter@192
  1356
    /// Construct an undirected graph reader, which reads from the given
deba@165
  1357
    /// file.
deba@598
  1358
    GraphReader(GR& graph, const char* fn)
deba@290
  1359
      : _is(new std::ifstream(fn)), local_is(true),
deba@290
  1360
        _filename(fn), _graph(graph),
kpeter@212
  1361
        _use_nodes(false), _use_edges(false),
deba@290
  1362
        _skip_nodes(false), _skip_edges(false) {
deba@295
  1363
      if (!(*_is)) {
deba@295
  1364
        delete _is;
deba@295
  1365
        throw IoError("Cannot open file", fn);
deba@295
  1366
      }
deba@290
  1367
    }
deba@165
  1368
deba@165
  1369
    /// \brief Destructor
deba@165
  1370
    ~GraphReader() {
alpar@209
  1371
      for (typename NodeMaps::iterator it = _node_maps.begin();
alpar@209
  1372
           it != _node_maps.end(); ++it) {
alpar@209
  1373
        delete it->second;
deba@165
  1374
      }
deba@165
  1375
alpar@209
  1376
      for (typename EdgeMaps::iterator it = _edge_maps.begin();
alpar@209
  1377
           it != _edge_maps.end(); ++it) {
alpar@209
  1378
        delete it->second;
deba@165
  1379
      }
deba@165
  1380
alpar@209
  1381
      for (typename Attributes::iterator it = _attributes.begin();
alpar@209
  1382
           it != _attributes.end(); ++it) {
alpar@209
  1383
        delete it->second;
deba@165
  1384
      }
deba@165
  1385
deba@165
  1386
      if (local_is) {
alpar@209
  1387
        delete _is;
deba@165
  1388
      }
deba@165
  1389
deba@165
  1390
    }
deba@165
  1391
deba@165
  1392
  private:
deba@598
  1393
    template <typename TGR>
deba@598
  1394
    friend GraphReader<TGR> graphReader(TGR& graph, std::istream& is);
deba@598
  1395
    template <typename TGR>
alpar@877
  1396
    friend GraphReader<TGR> graphReader(TGR& graph, const std::string& fn);
deba@598
  1397
    template <typename TGR>
deba@598
  1398
    friend GraphReader<TGR> graphReader(TGR& graph, const char *fn);
alpar@209
  1399
alpar@209
  1400
    GraphReader(GraphReader& other)
deba@190
  1401
      : _is(other._is), local_is(other.local_is), _graph(other._graph),
alpar@209
  1402
        _use_nodes(other._use_nodes), _use_edges(other._use_edges),
alpar@209
  1403
        _skip_nodes(other._skip_nodes), _skip_edges(other._skip_edges) {
deba@190
  1404
deba@190
  1405
      other._is = 0;
deba@190
  1406
      other.local_is = false;
alpar@209
  1407
deba@190
  1408
      _node_index.swap(other._node_index);
deba@190
  1409
      _edge_index.swap(other._edge_index);
deba@190
  1410
deba@190
  1411
      _node_maps.swap(other._node_maps);
deba@190
  1412
      _edge_maps.swap(other._edge_maps);
deba@190
  1413
      _attributes.swap(other._attributes);
deba@190
  1414
deba@190
  1415
      _nodes_caption = other._nodes_caption;
deba@190
  1416
      _edges_caption = other._edges_caption;
deba@190
  1417
      _attributes_caption = other._attributes_caption;
deba@190
  1418
deba@190
  1419
    }
deba@190
  1420
deba@165
  1421
    GraphReader& operator=(const GraphReader&);
deba@165
  1422
deba@165
  1423
  public:
deba@165
  1424
kpeter@584
  1425
    /// \name Reading Rules
deba@165
  1426
    /// @{
alpar@209
  1427
deba@165
  1428
    /// \brief Node map reading rule
deba@165
  1429
    ///
deba@165
  1430
    /// Add a node map reading rule to the reader.
deba@165
  1431
    template <typename Map>
deba@165
  1432
    GraphReader& nodeMap(const std::string& caption, Map& map) {
deba@165
  1433
      checkConcept<concepts::WriteMap<Node, typename Map::Value>, Map>();
alpar@209
  1434
      _reader_bits::MapStorageBase<Node>* storage =
alpar@209
  1435
        new _reader_bits::MapStorage<Node, Map>(map);
deba@165
  1436
      _node_maps.push_back(std::make_pair(caption, storage));
deba@165
  1437
      return *this;
deba@165
  1438
    }
deba@165
  1439
deba@165
  1440
    /// \brief Node map reading rule
deba@165
  1441
    ///
deba@165
  1442
    /// Add a node map reading rule with specialized converter to the
deba@165
  1443
    /// reader.
deba@165
  1444
    template <typename Map, typename Converter>
alpar@209
  1445
    GraphReader& nodeMap(const std::string& caption, Map& map,
alpar@209
  1446
                           const Converter& converter = Converter()) {
deba@165
  1447
      checkConcept<concepts::WriteMap<Node, typename Map::Value>, Map>();
alpar@209
  1448
      _reader_bits::MapStorageBase<Node>* storage =
alpar@209
  1449
        new _reader_bits::MapStorage<Node, Map, Converter>(map, converter);
deba@165
  1450
      _node_maps.push_back(std::make_pair(caption, storage));
deba@165
  1451
      return *this;
deba@165
  1452
    }
deba@165
  1453
deba@165
  1454
    /// \brief Edge map reading rule
deba@165
  1455
    ///
deba@165
  1456
    /// Add an edge map reading rule to the reader.
deba@165
  1457
    template <typename Map>
deba@165
  1458
    GraphReader& edgeMap(const std::string& caption, Map& map) {
deba@165
  1459
      checkConcept<concepts::WriteMap<Edge, typename Map::Value>, Map>();
alpar@209
  1460
      _reader_bits::MapStorageBase<Edge>* storage =
alpar@209
  1461
        new _reader_bits::MapStorage<Edge, Map>(map);
deba@165
  1462
      _edge_maps.push_back(std::make_pair(caption, storage));
deba@165
  1463
      return *this;
deba@165
  1464
    }
deba@165
  1465
deba@165
  1466
    /// \brief Edge map reading rule
deba@165
  1467
    ///
deba@165
  1468
    /// Add an edge map reading rule with specialized converter to the
deba@165
  1469
    /// reader.
deba@165
  1470
    template <typename Map, typename Converter>
alpar@209
  1471
    GraphReader& edgeMap(const std::string& caption, Map& map,
alpar@209
  1472
                          const Converter& converter = Converter()) {
deba@165
  1473
      checkConcept<concepts::WriteMap<Edge, typename Map::Value>, Map>();
alpar@209
  1474
      _reader_bits::MapStorageBase<Edge>* storage =
alpar@209
  1475
        new _reader_bits::MapStorage<Edge, Map, Converter>(map, converter);
deba@165
  1476
      _edge_maps.push_back(std::make_pair(caption, storage));
deba@165
  1477
      return *this;
deba@165
  1478
    }
deba@165
  1479
deba@165
  1480
    /// \brief Arc map reading rule
deba@165
  1481
    ///
deba@165
  1482
    /// Add an arc map reading rule to the reader.
deba@165
  1483
    template <typename Map>
deba@165
  1484
    GraphReader& arcMap(const std::string& caption, Map& map) {
deba@165
  1485
      checkConcept<concepts::WriteMap<Arc, typename Map::Value>, Map>();
alpar@209
  1486
      _reader_bits::MapStorageBase<Edge>* forward_storage =
alpar@209
  1487
        new _reader_bits::GraphArcMapStorage<Graph, true, Map>(_graph, map);
deba@165
  1488
      _edge_maps.push_back(std::make_pair('+' + caption, forward_storage));
alpar@209
  1489
      _reader_bits::MapStorageBase<Edge>* backward_storage =
deba@598
  1490
        new _reader_bits::GraphArcMapStorage<GR, false, Map>(_graph, map);
deba@165
  1491
      _edge_maps.push_back(std::make_pair('-' + caption, backward_storage));
deba@165
  1492
      return *this;
deba@165
  1493
    }
deba@165
  1494
deba@165
  1495
    /// \brief Arc map reading rule
deba@165
  1496
    ///
deba@165
  1497
    /// Add an arc map reading rule with specialized converter to the
deba@165
  1498
    /// reader.
deba@165
  1499
    template <typename Map, typename Converter>
alpar@209
  1500
    GraphReader& arcMap(const std::string& caption, Map& map,
alpar@209
  1501
                          const Converter& converter = Converter()) {
deba@165
  1502
      checkConcept<concepts::WriteMap<Arc, typename Map::Value>, Map>();
alpar@209
  1503
      _reader_bits::MapStorageBase<Edge>* forward_storage =
deba@598
  1504
        new _reader_bits::GraphArcMapStorage<GR, true, Map, Converter>
alpar@209
  1505
        (_graph, map, converter);
deba@165
  1506
      _edge_maps.push_back(std::make_pair('+' + caption, forward_storage));
alpar@209
  1507
      _reader_bits::MapStorageBase<Edge>* backward_storage =
deba@598
  1508
        new _reader_bits::GraphArcMapStorage<GR, false, Map, Converter>
alpar@209
  1509
        (_graph, map, converter);
deba@165
  1510
      _edge_maps.push_back(std::make_pair('-' + caption, backward_storage));
deba@165
  1511
      return *this;
deba@165
  1512
    }
deba@165
  1513
deba@165
  1514
    /// \brief Attribute reading rule
deba@165
  1515
    ///
deba@165
  1516
    /// Add an attribute reading rule to the reader.
deba@165
  1517
    template <typename Value>
deba@165
  1518
    GraphReader& attribute(const std::string& caption, Value& value) {
alpar@209
  1519
      _reader_bits::ValueStorageBase* storage =
alpar@209
  1520
        new _reader_bits::ValueStorage<Value>(value);
deba@165
  1521
      _attributes.insert(std::make_pair(caption, storage));
deba@165
  1522
      return *this;
deba@165
  1523
    }
deba@165
  1524
deba@165
  1525
    /// \brief Attribute reading rule
deba@165
  1526
    ///
deba@165
  1527
    /// Add an attribute reading rule with specialized converter to the
deba@165
  1528
    /// reader.
deba@165
  1529
    template <typename Value, typename Converter>
alpar@209
  1530
    GraphReader& attribute(const std::string& caption, Value& value,
alpar@209
  1531
                             const Converter& converter = Converter()) {
alpar@209
  1532
      _reader_bits::ValueStorageBase* storage =
alpar@209
  1533
        new _reader_bits::ValueStorage<Value, Converter>(value, converter);
deba@165
  1534
      _attributes.insert(std::make_pair(caption, storage));
deba@165
  1535
      return *this;
deba@165
  1536
    }
deba@165
  1537
deba@165
  1538
    /// \brief Node reading rule
deba@165
  1539
    ///
deba@165
  1540
    /// Add a node reading rule to reader.
deba@165
  1541
    GraphReader& node(const std::string& caption, Node& node) {
deba@165
  1542
      typedef _reader_bits::MapLookUpConverter<Node> Converter;
deba@165
  1543
      Converter converter(_node_index);
alpar@209
  1544
      _reader_bits::ValueStorageBase* storage =
alpar@209
  1545
        new _reader_bits::ValueStorage<Node, Converter>(node, converter);
deba@165
  1546
      _attributes.insert(std::make_pair(caption, storage));
deba@165
  1547
      return *this;
deba@165
  1548
    }
deba@165
  1549
deba@165
  1550
    /// \brief Edge reading rule
deba@165
  1551
    ///
deba@165
  1552
    /// Add an edge reading rule to reader.
deba@165
  1553
    GraphReader& edge(const std::string& caption, Edge& edge) {
deba@165
  1554
      typedef _reader_bits::MapLookUpConverter<Edge> Converter;
deba@165
  1555
      Converter converter(_edge_index);
alpar@209
  1556
      _reader_bits::ValueStorageBase* storage =
alpar@209
  1557
        new _reader_bits::ValueStorage<Edge, Converter>(edge, converter);
deba@165
  1558
      _attributes.insert(std::make_pair(caption, storage));
deba@165
  1559
      return *this;
deba@165
  1560
    }
deba@165
  1561
deba@165
  1562
    /// \brief Arc reading rule
deba@165
  1563
    ///
deba@165
  1564
    /// Add an arc reading rule to reader.
deba@165
  1565
    GraphReader& arc(const std::string& caption, Arc& arc) {
deba@598
  1566
      typedef _reader_bits::GraphArcLookUpConverter<GR> Converter;
deba@165
  1567
      Converter converter(_graph, _edge_index);
alpar@209
  1568
      _reader_bits::ValueStorageBase* storage =
alpar@209
  1569
        new _reader_bits::ValueStorage<Arc, Converter>(arc, converter);
deba@165
  1570
      _attributes.insert(std::make_pair(caption, storage));
deba@165
  1571
      return *this;
deba@165
  1572
    }
deba@165
  1573
deba@165
  1574
    /// @}
deba@165
  1575
kpeter@584
  1576
    /// \name Select Section by Name
deba@165
  1577
    /// @{
deba@165
  1578
deba@165
  1579
    /// \brief Set \c \@nodes section to be read
deba@165
  1580
    ///
kpeter@192
  1581
    /// Set \c \@nodes section to be read.
deba@165
  1582
    GraphReader& nodes(const std::string& caption) {
deba@165
  1583
      _nodes_caption = caption;
deba@165
  1584
      return *this;
deba@165
  1585
    }
deba@165
  1586
deba@165
  1587
    /// \brief Set \c \@edges section to be read
deba@165
  1588
    ///
kpeter@192
  1589
    /// Set \c \@edges section to be read.
deba@165
  1590
    GraphReader& edges(const std::string& caption) {
deba@165
  1591
      _edges_caption = caption;
deba@165
  1592
      return *this;
deba@165
  1593
    }
deba@165
  1594
deba@165
  1595
    /// \brief Set \c \@attributes section to be read
deba@165
  1596
    ///
kpeter@192
  1597
    /// Set \c \@attributes section to be read.
deba@165
  1598
    GraphReader& attributes(const std::string& caption) {
deba@165
  1599
      _attributes_caption = caption;
deba@165
  1600
      return *this;
deba@165
  1601
    }
deba@165
  1602
deba@165
  1603
    /// @}
deba@165
  1604
kpeter@584
  1605
    /// \name Using Previously Constructed Node or Edge Set
deba@165
  1606
    /// @{
deba@165
  1607
deba@165
  1608
    /// \brief Use previously constructed node set
deba@165
  1609
    ///
deba@165
  1610
    /// Use previously constructed node set, and specify the node
deba@165
  1611
    /// label map.
deba@165
  1612
    template <typename Map>
deba@165
  1613
    GraphReader& useNodes(const Map& map) {
deba@165
  1614
      checkConcept<concepts::ReadMap<Node, typename Map::Value>, Map>();
alpar@209
  1615
      LEMON_ASSERT(!_use_nodes, "Multiple usage of useNodes() member");
deba@165
  1616
      _use_nodes = true;
deba@165
  1617
      _writer_bits::DefaultConverter<typename Map::Value> converter;
deba@165
  1618
      for (NodeIt n(_graph); n != INVALID; ++n) {
alpar@209
  1619
        _node_index.insert(std::make_pair(converter(map[n]), n));
deba@165
  1620
      }
deba@165
  1621
      return *this;
deba@165
  1622
    }
deba@165
  1623
deba@165
  1624
    /// \brief Use previously constructed node set
deba@165
  1625
    ///
deba@165
  1626
    /// Use previously constructed node set, and specify the node
deba@165
  1627
    /// label map and a functor which converts the label map values to
kpeter@192
  1628
    /// \c std::string.
deba@165
  1629
    template <typename Map, typename Converter>
alpar@209
  1630
    GraphReader& useNodes(const Map& map,
alpar@209
  1631
                            const Converter& converter = Converter()) {
deba@165
  1632
      checkConcept<concepts::ReadMap<Node, typename Map::Value>, Map>();
alpar@209
  1633
      LEMON_ASSERT(!_use_nodes, "Multiple usage of useNodes() member");
deba@165
  1634
      _use_nodes = true;
deba@165
  1635
      for (NodeIt n(_graph); n != INVALID; ++n) {
alpar@209
  1636
        _node_index.insert(std::make_pair(converter(map[n]), n));
deba@165
  1637
      }
deba@165
  1638
      return *this;
deba@165
  1639
    }
deba@165
  1640
deba@165
  1641
    /// \brief Use previously constructed edge set
deba@165
  1642
    ///
deba@165
  1643
    /// Use previously constructed edge set, and specify the edge
deba@165
  1644
    /// label map.
deba@165
  1645
    template <typename Map>
deba@165
  1646
    GraphReader& useEdges(const Map& map) {
deba@165
  1647
      checkConcept<concepts::ReadMap<Edge, typename Map::Value>, Map>();
deba@165
  1648
      LEMON_ASSERT(!_use_edges, "Multiple usage of useEdges() member");
deba@165
  1649
      _use_edges = true;
deba@165
  1650
      _writer_bits::DefaultConverter<typename Map::Value> converter;
deba@165
  1651
      for (EdgeIt a(_graph); a != INVALID; ++a) {
alpar@209
  1652
        _edge_index.insert(std::make_pair(converter(map[a]), a));
deba@165
  1653
      }
deba@165
  1654
      return *this;
deba@165
  1655
    }
deba@165
  1656
deba@165
  1657
    /// \brief Use previously constructed edge set
deba@165
  1658
    ///
deba@165
  1659
    /// Use previously constructed edge set, and specify the edge
deba@165
  1660
    /// label map and a functor which converts the label map values to
kpeter@192
  1661
    /// \c std::string.
deba@165
  1662
    template <typename Map, typename Converter>
alpar@209
  1663
    GraphReader& useEdges(const Map& map,
alpar@209
  1664
                            const Converter& converter = Converter()) {
deba@165
  1665
      checkConcept<concepts::ReadMap<Edge, typename Map::Value>, Map>();
alpar@209
  1666
      LEMON_ASSERT(!_use_edges, "Multiple usage of useEdges() member");
deba@165
  1667
      _use_edges = true;
deba@165
  1668
      for (EdgeIt a(_graph); a != INVALID; ++a) {
alpar@209
  1669
        _edge_index.insert(std::make_pair(converter(map[a]), a));
deba@165
  1670
      }
deba@165
  1671
      return *this;
deba@165
  1672
    }
deba@165
  1673
kpeter@192
  1674
    /// \brief Skip the reading of node section
deba@188
  1675
    ///
deba@188
  1676
    /// Omit the reading of the node section. This implies that each node
kpeter@192
  1677
    /// map reading rule will be abandoned, and the nodes of the graph
deba@188
  1678
    /// will not be constructed, which usually cause that the edge set
deba@188
  1679
    /// could not be read due to lack of node name
kpeter@192
  1680
    /// could not be read due to lack of node name resolving.
kpeter@192
  1681
    /// Therefore \c skipEdges() function should also be used, or
kpeter@192
  1682
    /// \c useNodes() should be used to specify the label of the nodes.
deba@188
  1683
    GraphReader& skipNodes() {
alpar@209
  1684
      LEMON_ASSERT(!_skip_nodes, "Skip nodes already set");
deba@188
  1685
      _skip_nodes = true;
deba@188
  1686
      return *this;
deba@188
  1687
    }
deba@188
  1688
kpeter@192
  1689
    /// \brief Skip the reading of edge section
deba@188
  1690
    ///
deba@188
  1691
    /// Omit the reading of the edge section. This implies that each edge
kpeter@192
  1692
    /// map reading rule will be abandoned, and the edges of the graph
deba@188
  1693
    /// will not be constructed.
deba@188
  1694
    GraphReader& skipEdges() {
alpar@209
  1695
      LEMON_ASSERT(!_skip_edges, "Skip edges already set");
deba@188
  1696
      _skip_edges = true;
deba@188
  1697
      return *this;
deba@188
  1698
    }
deba@188
  1699
deba@165
  1700
    /// @}
deba@165
  1701
deba@165
  1702
  private:
deba@165
  1703
deba@165
  1704
    bool readLine() {
deba@165
  1705
      std::string str;
deba@165
  1706
      while(++line_num, std::getline(*_is, str)) {
alpar@209
  1707
        line.clear(); line.str(str);
alpar@209
  1708
        char c;
alpar@209
  1709
        if (line >> std::ws >> c && c != '#') {
alpar@209
  1710
          line.putback(c);
alpar@209
  1711
          return true;
alpar@209
  1712
        }
deba@165
  1713
      }
deba@165
  1714
      return false;
deba@165
  1715
    }
deba@165
  1716
deba@165
  1717
    bool readSuccess() {
deba@165
  1718
      return static_cast<bool>(*_is);
deba@165
  1719
    }
alpar@209
  1720
deba@165
  1721
    void skipSection() {
deba@165
  1722
      char c;
deba@165
  1723
      while (readSuccess() && line >> c && c != '@') {
alpar@209
  1724
        readLine();
deba@165
  1725
      }
deba@427
  1726
      if (readSuccess()) {
deba@427
  1727
        line.putback(c);
deba@427
  1728
      }
deba@165
  1729
    }
deba@165
  1730
deba@165
  1731
    void readNodes() {
deba@165
  1732
deba@165
  1733
      std::vector<int> map_index(_node_maps.size());
deba@165
  1734
      int map_num, label_index;
deba@165
  1735
deba@186
  1736
      char c;
deba@186
  1737
      if (!readLine() || !(line >> c) || c == '@') {
alpar@209
  1738
        if (readSuccess() && line) line.putback(c);
alpar@209
  1739
        if (!_node_maps.empty())
deba@290
  1740
          throw FormatError("Cannot find map names");
alpar@209
  1741
        return;
deba@186
  1742
      }
deba@186
  1743
      line.putback(c);
alpar@209
  1744
deba@165
  1745
      {
alpar@209
  1746
        std::map<std::string, int> maps;
alpar@209
  1747
alpar@209
  1748
        std::string map;
alpar@209
  1749
        int index = 0;
alpar@209
  1750
        while (_reader_bits::readToken(line, map)) {
alpar@209
  1751
          if (maps.find(map) != maps.end()) {
alpar@209
  1752
            std::ostringstream msg;
alpar@209
  1753
            msg << "Multiple occurence of node map: " << map;
deba@290
  1754
            throw FormatError(msg.str());
alpar@209
  1755
          }
alpar@209
  1756
          maps.insert(std::make_pair(map, index));
alpar@209
  1757
          ++index;
alpar@209
  1758
        }
alpar@209
  1759
alpar@209
  1760
        for (int i = 0; i < static_cast<int>(_node_maps.size()); ++i) {
alpar@209
  1761
          std::map<std::string, int>::iterator jt =
alpar@209
  1762
            maps.find(_node_maps[i].first);
alpar@209
  1763
          if (jt == maps.end()) {
alpar@209
  1764
            std::ostringstream msg;
kpeter@291
  1765
            msg << "Map not found: " << _node_maps[i].first;
deba@290
  1766
            throw FormatError(msg.str());
alpar@209
  1767
          }
alpar@209
  1768
          map_index[i] = jt->second;
alpar@209
  1769
        }
alpar@209
  1770
alpar@209
  1771
        {
alpar@209
  1772
          std::map<std::string, int>::iterator jt = maps.find("label");
alpar@209
  1773
          if (jt != maps.end()) {
alpar@209
  1774
            label_index = jt->second;
alpar@209
  1775
          } else {
alpar@209
  1776
            label_index = -1;
alpar@209
  1777
          }
alpar@209
  1778
        }
alpar@209
  1779
        map_num = maps.size();
deba@165
  1780
      }
deba@165
  1781
deba@165
  1782
      while (readLine() && line >> c && c != '@') {
alpar@209
  1783
        line.putback(c);
alpar@209
  1784
alpar@209
  1785
        std::vector<std::string> tokens(map_num);
alpar@209
  1786
        for (int i = 0; i < map_num; ++i) {
alpar@209
  1787
          if (!_reader_bits::readToken(line, tokens[i])) {
alpar@209
  1788
            std::ostringstream msg;
alpar@209
  1789
            msg << "Column not found (" << i + 1 << ")";
deba@290
  1790
            throw FormatError(msg.str());
alpar@209
  1791
          }
alpar@209
  1792
        }
alpar@209
  1793
        if (line >> std::ws >> c)
kpeter@291
  1794
          throw FormatError("Extra character at the end of line");
alpar@209
  1795
alpar@209
  1796
        Node n;
alpar@209
  1797
        if (!_use_nodes) {
alpar@209
  1798
          n = _graph.addNode();
alpar@209
  1799
          if (label_index != -1)
alpar@209
  1800
            _node_index.insert(std::make_pair(tokens[label_index], n));
alpar@209
  1801
        } else {
alpar@209
  1802
          if (label_index == -1)
kpeter@291
  1803
            throw FormatError("Label map not found");
alpar@209
  1804
          typename std::map<std::string, Node>::iterator it =
alpar@209
  1805
            _node_index.find(tokens[label_index]);
alpar@209
  1806
          if (it == _node_index.end()) {
alpar@209
  1807
            std::ostringstream msg;
alpar@209
  1808
            msg << "Node with label not found: " << tokens[label_index];
deba@290
  1809
            throw FormatError(msg.str());
alpar@209
  1810
          }
alpar@209
  1811
          n = it->second;
alpar@209
  1812
        }
alpar@209
  1813
alpar@209
  1814
        for (int i = 0; i < static_cast<int>(_node_maps.size()); ++i) {
alpar@209
  1815
          _node_maps[i].second->set(n, tokens[map_index[i]]);
alpar@209
  1816
        }
deba@165
  1817
deba@165
  1818
      }
deba@165
  1819
      if (readSuccess()) {
alpar@209
  1820
        line.putback(c);
deba@165
  1821
      }
deba@165
  1822
    }
deba@165
  1823
deba@165
  1824
    void readEdges() {
deba@165
  1825
deba@165
  1826
      std::vector<int> map_index(_edge_maps.size());
deba@165
  1827
      int map_num, label_index;
deba@165
  1828
deba@186
  1829
      char c;
deba@186
  1830
      if (!readLine() || !(line >> c) || c == '@') {
alpar@209
  1831
        if (readSuccess() && line) line.putback(c);
alpar@209
  1832
        if (!_edge_maps.empty())
deba@290
  1833
          throw FormatError("Cannot find map names");
alpar@209
  1834
        return;
deba@186
  1835
      }
deba@186
  1836
      line.putback(c);
alpar@209
  1837
deba@165
  1838
      {
alpar@209
  1839
        std::map<std::string, int> maps;
alpar@209
  1840
alpar@209
  1841
        std::string map;
alpar@209
  1842
        int index = 0;
alpar@209
  1843
        while (_reader_bits::readToken(line, map)) {
alpar@949
  1844
          if(map == "-") {
alpar@949
  1845
              if(index!=0)
alpar@949
  1846
                throw FormatError("'-' is not allowed as a map name");
alpar@949
  1847
              else if (line >> std::ws >> c)
alpar@949
  1848
                throw FormatError("Extra character at the end of line");
alpar@949
  1849
              else break;
alpar@949
  1850
            }
alpar@209
  1851
          if (maps.find(map) != maps.end()) {
alpar@209
  1852
            std::ostringstream msg;
alpar@209
  1853
            msg << "Multiple occurence of edge map: " << map;
deba@290
  1854
            throw FormatError(msg.str());
alpar@209
  1855
          }
alpar@209
  1856
          maps.insert(std::make_pair(map, index));
alpar@209
  1857
          ++index;
alpar@209
  1858
        }
alpar@209
  1859
alpar@209
  1860
        for (int i = 0; i < static_cast<int>(_edge_maps.size()); ++i) {
alpar@209
  1861
          std::map<std::string, int>::iterator jt =
alpar@209
  1862
            maps.find(_edge_maps[i].first);
alpar@209
  1863
          if (jt == maps.end()) {
alpar@209
  1864
            std::ostringstream msg;
kpeter@291
  1865
            msg << "Map not found: " << _edge_maps[i].first;
deba@290
  1866
            throw FormatError(msg.str());
alpar@209
  1867
          }
alpar@209
  1868
          map_index[i] = jt->second;
alpar@209
  1869
        }
alpar@209
  1870
alpar@209
  1871
        {
alpar@209
  1872
          std::map<std::string, int>::iterator jt = maps.find("label");
alpar@209
  1873
          if (jt != maps.end()) {
alpar@209
  1874
            label_index = jt->second;
alpar@209
  1875
          } else {
alpar@209
  1876
            label_index = -1;
alpar@209
  1877
          }
alpar@209
  1878
        }
alpar@209
  1879
        map_num = maps.size();
deba@165
  1880
      }
deba@165
  1881
deba@165
  1882
      while (readLine() && line >> c && c != '@') {
alpar@209
  1883
        line.putback(c);
alpar@209
  1884
alpar@209
  1885
        std::string source_token;
alpar@209
  1886
        std::string target_token;
alpar@209
  1887
alpar@209
  1888
        if (!_reader_bits::readToken(line, source_token))
deba@290
  1889
          throw FormatError("Node u not found");
alpar@209
  1890
alpar@209
  1891
        if (!_reader_bits::readToken(line, target_token))
deba@290
  1892
          throw FormatError("Node v not found");
alpar@209
  1893
alpar@209
  1894
        std::vector<std::string> tokens(map_num);
alpar@209
  1895
        for (int i = 0; i < map_num; ++i) {
alpar@209
  1896
          if (!_reader_bits::readToken(line, tokens[i])) {
alpar@209
  1897
            std::ostringstream msg;
alpar@209
  1898
            msg << "Column not found (" << i + 1 << ")";
deba@290
  1899
            throw FormatError(msg.str());
alpar@209
  1900
          }
alpar@209
  1901
        }
alpar@209
  1902
        if (line >> std::ws >> c)
kpeter@291
  1903
          throw FormatError("Extra character at the end of line");
alpar@209
  1904
alpar@209
  1905
        Edge e;
alpar@209
  1906
        if (!_use_edges) {
deba@165
  1907
deba@165
  1908
          typename NodeIndex::iterator it;
alpar@209
  1909
deba@165
  1910
          it = _node_index.find(source_token);
deba@165
  1911
          if (it == _node_index.end()) {
deba@165
  1912
            std::ostringstream msg;
deba@165
  1913
            msg << "Item not found: " << source_token;
deba@290
  1914
            throw FormatError(msg.str());
deba@165
  1915
          }
deba@165
  1916
          Node source = it->second;
deba@165
  1917
deba@165
  1918
          it = _node_index.find(target_token);
alpar@209
  1919
          if (it == _node_index.end()) {
alpar@209
  1920
            std::ostringstream msg;
deba@165
  1921
            msg << "Item not found: " << target_token;
deba@290
  1922
            throw FormatError(msg.str());
alpar@209
  1923
          }
alpar@209
  1924
          Node target = it->second;
alpar@209
  1925
alpar@209
  1926
          e = _graph.addEdge(source, target);
alpar@209
  1927
          if (label_index != -1)
alpar@209
  1928
            _edge_index.insert(std::make_pair(tokens[label_index], e));
alpar@209
  1929
        } else {
alpar@209
  1930
          if (label_index == -1)
kpeter@291
  1931
            throw FormatError("Label map not found");
alpar@209
  1932
          typename std::map<std::string, Edge>::iterator it =
alpar@209
  1933
            _edge_index.find(tokens[label_index]);
alpar@209
  1934
          if (it == _edge_index.end()) {
alpar@209
  1935
            std::ostringstream msg;
alpar@209
  1936
            msg << "Edge with label not found: " << tokens[label_index];
deba@290
  1937
            throw FormatError(msg.str());
alpar@209
  1938
          }
alpar@209
  1939
          e = it->second;
alpar@209
  1940
        }
alpar@209
  1941
alpar@209
  1942
        for (int i = 0; i < static_cast<int>(_edge_maps.size()); ++i) {
alpar@209
  1943
          _edge_maps[i].second->set(e, tokens[map_index[i]]);
alpar@209
  1944
        }
deba@165
  1945
deba@165
  1946
      }
deba@165
  1947
      if (readSuccess()) {
alpar@209
  1948
        line.putback(c);
deba@165
  1949
      }
deba@165
  1950
    }
deba@165
  1951
deba@165
  1952
    void readAttributes() {
deba@165
  1953
deba@165
  1954
      std::set<std::string> read_attr;
deba@165
  1955
deba@165
  1956
      char c;
deba@165
  1957
      while (readLine() && line >> c && c != '@') {
alpar@209
  1958
        line.putback(c);
alpar@209
  1959
alpar@209
  1960
        std::string attr, token;
alpar@209
  1961
        if (!_reader_bits::readToken(line, attr))
deba@290
  1962
          throw FormatError("Attribute name not found");
alpar@209
  1963
        if (!_reader_bits::readToken(line, token))
deba@290
  1964
          throw FormatError("Attribute value not found");
alpar@209
  1965
        if (line >> c)
kpeter@291
  1966
          throw FormatError("Extra character at the end of line");
alpar@209
  1967
alpar@209
  1968
        {
alpar@209
  1969
          std::set<std::string>::iterator it = read_attr.find(attr);
alpar@209
  1970
          if (it != read_attr.end()) {
alpar@209
  1971
            std::ostringstream msg;
kpeter@291
  1972
            msg << "Multiple occurence of attribute: " << attr;
deba@290
  1973
            throw FormatError(msg.str());
alpar@209
  1974
          }
alpar@209
  1975
          read_attr.insert(attr);
alpar@209
  1976
        }
alpar@209
  1977
alpar@209
  1978
        {
alpar@209
  1979
          typename Attributes::iterator it = _attributes.lower_bound(attr);
alpar@209
  1980
          while (it != _attributes.end() && it->first == attr) {
alpar@209
  1981
            it->second->set(token);
alpar@209
  1982
            ++it;
alpar@209
  1983
          }
alpar@209
  1984
        }
deba@165
  1985
deba@165
  1986
      }
deba@165
  1987
      if (readSuccess()) {
alpar@209
  1988
        line.putback(c);
deba@165
  1989
      }
deba@165
  1990
      for (typename Attributes::iterator it = _attributes.begin();
alpar@209
  1991
           it != _attributes.end(); ++it) {
alpar@209
  1992
        if (read_attr.find(it->first) == read_attr.end()) {
alpar@209
  1993
          std::ostringstream msg;
kpeter@291
  1994
          msg << "Attribute not found: " << it->first;
deba@290
  1995
          throw FormatError(msg.str());
alpar@209
  1996
        }
deba@165
  1997
      }
deba@165
  1998
    }
deba@165
  1999
deba@165
  2000
  public:
deba@165
  2001
kpeter@584
  2002
    /// \name Execution of the Reader
deba@165
  2003
    /// @{
deba@165
  2004
deba@165
  2005
    /// \brief Start the batch processing
deba@165
  2006
    ///
deba@165
  2007
    /// This function starts the batch processing
deba@165
  2008
    void run() {
alpar@209
  2009
deba@165
  2010
      LEMON_ASSERT(_is != 0, "This reader assigned to an other reader");
alpar@209
  2011
deba@188
  2012
      bool nodes_done = _skip_nodes;
deba@188
  2013
      bool edges_done = _skip_edges;
deba@165
  2014
      bool attributes_done = false;
deba@165
  2015
alpar@209
  2016
      line_num = 0;
deba@165
  2017
      readLine();
deba@172
  2018
      skipSection();
deba@165
  2019
deba@165
  2020
      while (readSuccess()) {
alpar@209
  2021
        try {
alpar@209
  2022
          char c;
alpar@209
  2023
          std::string section, caption;
alpar@209
  2024
          line >> c;
alpar@209
  2025
          _reader_bits::readToken(line, section);
alpar@209
  2026
          _reader_bits::readToken(line, caption);
alpar@209
  2027
alpar@209
  2028
          if (line >> c)
kpeter@291
  2029
            throw FormatError("Extra character at the end of line");
alpar@209
  2030
alpar@209
  2031
          if (section == "nodes" && !nodes_done) {
alpar@209
  2032
            if (_nodes_caption.empty() || _nodes_caption == caption) {
alpar@209
  2033
              readNodes();
alpar@209
  2034
              nodes_done = true;
alpar@209
  2035
            }
alpar@209
  2036
          } else if ((section == "edges" || section == "arcs") &&
alpar@209
  2037
                     !edges_done) {
alpar@209
  2038
            if (_edges_caption.empty() || _edges_caption == caption) {
alpar@209
  2039
              readEdges();
alpar@209
  2040
              edges_done = true;
alpar@209
  2041
            }
alpar@209
  2042
          } else if (section == "attributes" && !attributes_done) {
alpar@209
  2043
            if (_attributes_caption.empty() || _attributes_caption == caption) {
alpar@209
  2044
              readAttributes();
alpar@209
  2045
              attributes_done = true;
alpar@209
  2046
            }
alpar@209
  2047
          } else {
alpar@209
  2048
            readLine();
alpar@209
  2049
            skipSection();
alpar@209
  2050
          }
deba@290
  2051
        } catch (FormatError& error) {
alpar@209
  2052
          error.line(line_num);
deba@290
  2053
          error.file(_filename);
alpar@209
  2054
          throw;
alpar@209
  2055
        }
deba@165
  2056
      }
deba@165
  2057
deba@165
  2058
      if (!nodes_done) {
deba@290
  2059
        throw FormatError("Section @nodes not found");
deba@165
  2060
      }
deba@165
  2061
deba@165
  2062
      if (!edges_done) {
deba@290
  2063
        throw FormatError("Section @edges not found");
deba@165
  2064
      }
deba@165
  2065
deba@165
  2066
      if (!attributes_done && !_attributes.empty()) {
deba@290
  2067
        throw FormatError("Section @attributes not found");
deba@165
  2068
      }
deba@165
  2069
deba@165
  2070
    }
deba@165
  2071
deba@165
  2072
    /// @}
alpar@209
  2073
deba@165
  2074
  };
deba@165
  2075
deba@598
  2076
  /// \ingroup lemon_io
deba@598
  2077
  ///
deba@598
  2078
  /// \brief Return a \ref GraphReader class
deba@598
  2079
  ///
alpar@877
  2080
  /// This function just returns a \ref GraphReader class.
deba@598
  2081
  ///
alpar@877
  2082
  /// With this function a graph can be read from an
deba@598
  2083
  /// \ref lgf-format "LGF" file or input stream with several maps and
deba@598
  2084
  /// attributes. For example, there is weighted matching problem on a
deba@598
  2085
  /// graph, i.e. a graph with a \e weight map on the edges. This
deba@598
  2086
  /// graph can be read with the following code:
deba@598
  2087
  ///
deba@598
  2088
  ///\code
deba@598
  2089
  ///ListGraph graph;
deba@598
  2090
  ///ListGraph::EdgeMap<int> weight(graph);
deba@598
  2091
  ///graphReader(graph, std::cin).
deba@598
  2092
  ///  edgeMap("weight", weight).
deba@598
  2093
  ///  run();
deba@598
  2094
  ///\endcode
deba@598
  2095
  ///
deba@598
  2096
  /// For a complete documentation, please see the \ref GraphReader
deba@598
  2097
  /// class documentation.
deba@598
  2098
  /// \warning Don't forget to put the \ref GraphReader::run() "run()"
deba@598
  2099
  /// to the end of the parameter list.
deba@598
  2100
  /// \relates GraphReader
deba@598
  2101
  /// \sa graphReader(TGR& graph, const std::string& fn)
deba@598
  2102
  /// \sa graphReader(TGR& graph, const char* fn)
deba@598
  2103
  template <typename TGR>
deba@598
  2104
  GraphReader<TGR> graphReader(TGR& graph, std::istream& is) {
deba@598
  2105
    GraphReader<TGR> tmp(graph, is);
deba@598
  2106
    return tmp;
deba@598
  2107
  }
deba@598
  2108
kpeter@498
  2109
  /// \brief Return a \ref GraphReader class
kpeter@498
  2110
  ///
kpeter@498
  2111
  /// This function just returns a \ref GraphReader class.
kpeter@498
  2112
  /// \relates GraphReader
deba@598
  2113
  /// \sa graphReader(TGR& graph, std::istream& is)
deba@598
  2114
  template <typename TGR>
deba@598
  2115
  GraphReader<TGR> graphReader(TGR& graph, const std::string& fn) {
deba@598
  2116
    GraphReader<TGR> tmp(graph, fn);
kpeter@498
  2117
    return tmp;
kpeter@498
  2118
  }
kpeter@498
  2119
kpeter@498
  2120
  /// \brief Return a \ref GraphReader class
kpeter@498
  2121
  ///
kpeter@498
  2122
  /// This function just returns a \ref GraphReader class.
kpeter@498
  2123
  /// \relates GraphReader
deba@598
  2124
  /// \sa graphReader(TGR& graph, std::istream& is)
deba@598
  2125
  template <typename TGR>
deba@598
  2126
  GraphReader<TGR> graphReader(TGR& graph, const char* fn) {
deba@598
  2127
    GraphReader<TGR> tmp(graph, fn);
kpeter@498
  2128
    return tmp;
kpeter@498
  2129
  }
kpeter@498
  2130
deba@1024
  2131
  template <typename BGR>
deba@1024
  2132
  class BpGraphReader;
deba@1024
  2133
deba@1024
  2134
  template <typename TBGR>
deba@1024
  2135
  BpGraphReader<TBGR> bpGraphReader(TBGR& graph, std::istream& is = std::cin);
deba@1024
  2136
  template <typename TBGR>
deba@1024
  2137
  BpGraphReader<TBGR> bpGraphReader(TBGR& graph, const std::string& fn);
deba@1024
  2138
  template <typename TBGR>
deba@1024
  2139
  BpGraphReader<TBGR> bpGraphReader(TBGR& graph, const char *fn);
deba@1024
  2140
deba@1024
  2141
  /// \ingroup lemon_io
deba@1024
  2142
  ///
deba@1024
  2143
  /// \brief \ref lgf-format "LGF" reader for bipartite graphs
deba@1024
  2144
  ///
deba@1024
  2145
  /// This utility reads an \ref lgf-format "LGF" file.
deba@1024
  2146
  ///
deba@1024
  2147
  /// It can be used almost the same way as \c GraphReader, but it
deba@1024
  2148
  /// reads the red and blue nodes from separate sections, and these
deba@1024
  2149
  /// sections can contain different set of maps.
deba@1024
  2150
  ///
deba@1026
  2151
  /// The red and blue node maps are read from the corresponding
deba@1024
  2152
  /// sections. If a map is defined with the same name in both of
deba@1024
  2153
  /// these sections, then it can be read as a node map.
deba@1024
  2154
  template <typename BGR>
deba@1024
  2155
  class BpGraphReader {
deba@1024
  2156
  public:
deba@1024
  2157
deba@1024
  2158
    typedef BGR Graph;
deba@1024
  2159
deba@1024
  2160
  private:
deba@1024
  2161
deba@1024
  2162
    TEMPLATE_BPGRAPH_TYPEDEFS(BGR);
deba@1024
  2163
deba@1024
  2164
    std::istream* _is;
deba@1024
  2165
    bool local_is;
deba@1024
  2166
    std::string _filename;
deba@1024
  2167
deba@1024
  2168
    BGR& _graph;
deba@1024
  2169
deba@1024
  2170
    std::string _nodes_caption;
deba@1024
  2171
    std::string _edges_caption;
deba@1024
  2172
    std::string _attributes_caption;
deba@1024
  2173
deba@1024
  2174
    typedef std::map<std::string, Node> NodeIndex;
deba@1024
  2175
    NodeIndex _node_index;
deba@1024
  2176
    typedef std::map<std::string, Edge> EdgeIndex;
deba@1024
  2177
    EdgeIndex _edge_index;
deba@1024
  2178
deba@1024
  2179
    typedef std::vector<std::pair<std::string,
deba@1024
  2180
      _reader_bits::MapStorageBase<Node>*> > NodeMaps;
deba@1024
  2181
    NodeMaps _red_maps;
deba@1024
  2182
    NodeMaps _blue_maps;
deba@1024
  2183
deba@1024
  2184
    typedef std::vector<std::pair<std::string,
deba@1024
  2185
      _reader_bits::MapStorageBase<Edge>*> > EdgeMaps;
deba@1024
  2186
    EdgeMaps _edge_maps;
deba@1024
  2187
deba@1024
  2188
    typedef std::multimap<std::string, _reader_bits::ValueStorageBase*>
deba@1024
  2189
      Attributes;
deba@1024
  2190
    Attributes _attributes;
deba@1024
  2191
deba@1024
  2192
    bool _use_nodes;
deba@1024
  2193
    bool _use_edges;
deba@1024
  2194
deba@1024
  2195
    bool _skip_nodes;
deba@1024
  2196
    bool _skip_edges;
deba@1024
  2197
deba@1024
  2198
    int line_num;
deba@1024
  2199
    std::istringstream line;
deba@1024
  2200
deba@1024
  2201
  public:
deba@1024
  2202
deba@1024
  2203
    /// \brief Constructor
deba@1024
  2204
    ///
deba@1024
  2205
    /// Construct an undirected graph reader, which reads from the given
deba@1024
  2206
    /// input stream.
deba@1024
  2207
    BpGraphReader(BGR& graph, std::istream& is = std::cin)
deba@1024
  2208
      : _is(&is), local_is(false), _graph(graph),
deba@1024
  2209
        _use_nodes(false), _use_edges(false),
deba@1024
  2210
        _skip_nodes(false), _skip_edges(false) {}
deba@1024
  2211
deba@1024
  2212
    /// \brief Constructor
deba@1024
  2213
    ///
deba@1024
  2214
    /// Construct an undirected graph reader, which reads from the given
deba@1024
  2215
    /// file.
deba@1024
  2216
    BpGraphReader(BGR& graph, const std::string& fn)
deba@1024
  2217
      : _is(new std::ifstream(fn.c_str())), local_is(true),
deba@1024
  2218
        _filename(fn), _graph(graph),
deba@1024
  2219
        _use_nodes(false), _use_edges(false),
deba@1024
  2220
        _skip_nodes(false), _skip_edges(false) {
deba@1024
  2221
      if (!(*_is)) {
deba@1024
  2222
        delete _is;
deba@1024
  2223
        throw IoError("Cannot open file", fn);
deba@1024
  2224
      }
deba@1024
  2225
    }
deba@1024
  2226
deba@1024
  2227
    /// \brief Constructor
deba@1024
  2228
    ///
deba@1024
  2229
    /// Construct an undirected graph reader, which reads from the given
deba@1024
  2230
    /// file.
deba@1024
  2231
    BpGraphReader(BGR& graph, const char* fn)
deba@1024
  2232
      : _is(new std::ifstream(fn)), local_is(true),
deba@1024
  2233
        _filename(fn), _graph(graph),
deba@1024
  2234
        _use_nodes(false), _use_edges(false),
deba@1024
  2235
        _skip_nodes(false), _skip_edges(false) {
deba@1024
  2236
      if (!(*_is)) {
deba@1024
  2237
        delete _is;
deba@1024
  2238
        throw IoError("Cannot open file", fn);
deba@1024
  2239
      }
deba@1024
  2240
    }
deba@1024
  2241
deba@1024
  2242
    /// \brief Destructor
deba@1024
  2243
    ~BpGraphReader() {
deba@1024
  2244
      for (typename NodeMaps::iterator it = _red_maps.begin();
deba@1024
  2245
           it != _red_maps.end(); ++it) {
deba@1024
  2246
        delete it->second;
deba@1024
  2247
      }
deba@1024
  2248
deba@1024
  2249
      for (typename NodeMaps::iterator it = _blue_maps.begin();
deba@1024
  2250
           it != _blue_maps.end(); ++it) {
deba@1024
  2251
        delete it->second;
deba@1024
  2252
      }
deba@1024
  2253
deba@1024
  2254
      for (typename EdgeMaps::iterator it = _edge_maps.begin();
deba@1024
  2255
           it != _edge_maps.end(); ++it) {
deba@1024
  2256
        delete it->second;
deba@1024
  2257
      }
deba@1024
  2258
deba@1024
  2259
      for (typename Attributes::iterator it = _attributes.begin();
deba@1024
  2260
           it != _attributes.end(); ++it) {
deba@1024
  2261
        delete it->second;
deba@1024
  2262
      }
deba@1024
  2263
deba@1024
  2264
      if (local_is) {
deba@1024
  2265
        delete _is;
deba@1024
  2266
      }
deba@1024
  2267
deba@1024
  2268
    }
deba@1024
  2269
deba@1024
  2270
  private:
deba@1024
  2271
    template <typename TBGR>
deba@1024
  2272
    friend BpGraphReader<TBGR> bpGraphReader(TBGR& graph, std::istream& is);
deba@1024
  2273
    template <typename TBGR>
deba@1024
  2274
    friend BpGraphReader<TBGR> bpGraphReader(TBGR& graph,
deba@1024
  2275
                                             const std::string& fn);
deba@1024
  2276
    template <typename TBGR>
deba@1024
  2277
    friend BpGraphReader<TBGR> bpGraphReader(TBGR& graph, const char *fn);
deba@1024
  2278
deba@1024
  2279
    BpGraphReader(BpGraphReader& other)
deba@1024
  2280
      : _is(other._is), local_is(other.local_is), _graph(other._graph),
deba@1024
  2281
        _use_nodes(other._use_nodes), _use_edges(other._use_edges),
deba@1024
  2282
        _skip_nodes(other._skip_nodes), _skip_edges(other._skip_edges) {
deba@1024
  2283
deba@1024
  2284
      other._is = 0;
deba@1024
  2285
      other.local_is = false;
deba@1024
  2286
deba@1024
  2287
      _node_index.swap(other._node_index);
deba@1024
  2288
      _edge_index.swap(other._edge_index);
deba@1024
  2289
deba@1024
  2290
      _red_maps.swap(other._red_maps);
deba@1024
  2291
      _blue_maps.swap(other._blue_maps);
deba@1024
  2292
      _edge_maps.swap(other._edge_maps);
deba@1024
  2293
      _attributes.swap(other._attributes);
deba@1024
  2294
deba@1024
  2295
      _nodes_caption = other._nodes_caption;
deba@1024
  2296
      _edges_caption = other._edges_caption;
deba@1024
  2297
      _attributes_caption = other._attributes_caption;
deba@1024
  2298
deba@1024
  2299
    }
deba@1024
  2300
deba@1024
  2301
    BpGraphReader& operator=(const BpGraphReader&);
deba@1024
  2302
deba@1024
  2303
  public:
deba@1024
  2304
deba@1024
  2305
    /// \name Reading Rules
deba@1024
  2306
    /// @{
deba@1024
  2307
deba@1024
  2308
    /// \brief Node map reading rule
deba@1024
  2309
    ///
deba@1024
  2310
    /// Add a node map reading rule to the reader.
deba@1024
  2311
    template <typename Map>
deba@1024
  2312
    BpGraphReader& nodeMap(const std::string& caption, Map& map) {
deba@1024
  2313
      checkConcept<concepts::WriteMap<Node, typename Map::Value>, Map>();
deba@1024
  2314
      _reader_bits::MapStorageBase<Node>* red_storage =
deba@1024
  2315
        new _reader_bits::MapStorage<Node, Map>(map);
deba@1024
  2316
      _red_maps.push_back(std::make_pair(caption, red_storage));
deba@1024
  2317
      _reader_bits::MapStorageBase<Node>* blue_storage =
deba@1024
  2318
        new _reader_bits::MapStorage<Node, Map>(map);
deba@1024
  2319
      _blue_maps.push_back(std::make_pair(caption, blue_storage));
deba@1024
  2320
      return *this;
deba@1024
  2321
    }
deba@1024
  2322
deba@1024
  2323
    /// \brief Node map reading rule
deba@1024
  2324
    ///
deba@1024
  2325
    /// Add a node map reading rule with specialized converter to the
deba@1024
  2326
    /// reader.
deba@1024
  2327
    template <typename Map, typename Converter>
deba@1024
  2328
    BpGraphReader& nodeMap(const std::string& caption, Map& map,
deba@1024
  2329
                           const Converter& converter = Converter()) {
deba@1024
  2330
      checkConcept<concepts::WriteMap<Node, typename Map::Value>, Map>();
deba@1024
  2331
      _reader_bits::MapStorageBase<Node>* red_storage =
deba@1024
  2332
        new _reader_bits::MapStorage<Node, Map, Converter>(map, converter);
deba@1024
  2333
      _red_maps.push_back(std::make_pair(caption, red_storage));
deba@1024
  2334
      _reader_bits::MapStorageBase<Node>* blue_storage =
deba@1024
  2335
        new _reader_bits::MapStorage<Node, Map, Converter>(map, converter);
deba@1024
  2336
      _blue_maps.push_back(std::make_pair(caption, blue_storage));
deba@1024
  2337
      return *this;
deba@1024
  2338
    }
deba@1024
  2339
deba@1026
  2340
    /// Add a red node map reading rule to the reader.
deba@1024
  2341
    template <typename Map>
deba@1026
  2342
    BpGraphReader& redNodeMap(const std::string& caption, Map& map) {
deba@1024
  2343
      checkConcept<concepts::WriteMap<Node, typename Map::Value>, Map>();
deba@1024
  2344
      _reader_bits::MapStorageBase<Node>* storage =
deba@1024
  2345
        new _reader_bits::MapStorage<Node, Map>(map);
deba@1024
  2346
      _red_maps.push_back(std::make_pair(caption, storage));
deba@1024
  2347
      return *this;
deba@1024
  2348
    }
deba@1024
  2349
deba@1026
  2350
    /// \brief Red node map reading rule
deba@1024
  2351
    ///
deba@1026
  2352
    /// Add a red node map node reading rule with specialized converter to
deba@1026
  2353
    /// the reader.
deba@1024
  2354
    template <typename Map, typename Converter>
deba@1026
  2355
    BpGraphReader& redNodeMap(const std::string& caption, Map& map,
deba@1026
  2356
                              const Converter& converter = Converter()) {
deba@1024
  2357
      checkConcept<concepts::WriteMap<Node, typename Map::Value>, Map>();
deba@1024
  2358
      _reader_bits::MapStorageBase<Node>* storage =
deba@1024
  2359
        new _reader_bits::MapStorage<Node, Map, Converter>(map, converter);
deba@1024
  2360
      _red_maps.push_back(std::make_pair(caption, storage));
deba@1024
  2361
      return *this;
deba@1024
  2362
    }
deba@1024
  2363
deba@1026
  2364
    /// Add a blue node map reading rule to the reader.
deba@1024
  2365
    template <typename Map>
deba@1026
  2366
    BpGraphReader& blueNodeMap(const std::string& caption, Map& map) {
deba@1024
  2367
      checkConcept<concepts::WriteMap<Node, typename Map::Value>, Map>();
deba@1024
  2368
      _reader_bits::MapStorageBase<Node>* storage =
deba@1024
  2369
        new _reader_bits::MapStorage<Node, Map>(map);
deba@1024
  2370
      _blue_maps.push_back(std::make_pair(caption, storage));
deba@1024
  2371
      return *this;
deba@1024
  2372
    }
deba@1024
  2373
deba@1026
  2374
    /// \brief Blue node map reading rule
deba@1024
  2375
    ///
deba@1026
  2376
    /// Add a blue node map reading rule with specialized converter to
deba@1026
  2377
    /// the reader.
deba@1024
  2378
    template <typename Map, typename Converter>
deba@1026
  2379
    BpGraphReader& blueNodeMap(const std::string& caption, Map& map,
deba@1026
  2380
                               const Converter& converter = Converter()) {
deba@1024
  2381
      checkConcept<concepts::WriteMap<Node, typename Map::Value>, Map>();
deba@1024
  2382
      _reader_bits::MapStorageBase<Node>* storage =
deba@1024
  2383
        new _reader_bits::MapStorage<Node, Map, Converter>(map, converter);
deba@1024
  2384
      _blue_maps.push_back(std::make_pair(caption, storage));
deba@1024
  2385
      return *this;
deba@1024
  2386
    }
deba@1024
  2387
deba@1024
  2388
    /// \brief Edge map reading rule
deba@1024
  2389
    ///
deba@1024
  2390
    /// Add an edge map reading rule to the reader.
deba@1024
  2391
    template <typename Map>
deba@1024
  2392
    BpGraphReader& edgeMap(const std::string& caption, Map& map) {
deba@1024
  2393
      checkConcept<concepts::WriteMap<Edge, typename Map::Value>, Map>();
deba@1024
  2394
      _reader_bits::MapStorageBase<Edge>* storage =
deba@1024
  2395
        new _reader_bits::MapStorage<Edge, Map>(map);
deba@1024
  2396
      _edge_maps.push_back(std::make_pair(caption, storage));
deba@1024
  2397
      return *this;
deba@1024
  2398
    }
deba@1024
  2399
deba@1024
  2400
    /// \brief Edge map reading rule
deba@1024
  2401
    ///
deba@1024
  2402
    /// Add an edge map reading rule with specialized converter to the
deba@1024
  2403
    /// reader.
deba@1024
  2404
    template <typename Map, typename Converter>
deba@1024
  2405
    BpGraphReader& edgeMap(const std::string& caption, Map& map,
deba@1024
  2406
                          const Converter& converter = Converter()) {
deba@1024
  2407
      checkConcept<concepts::WriteMap<Edge, typename Map::Value>, Map>();
deba@1024
  2408
      _reader_bits::MapStorageBase<Edge>* storage =
deba@1024
  2409
        new _reader_bits::MapStorage<Edge, Map, Converter>(map, converter);
deba@1024
  2410
      _edge_maps.push_back(std::make_pair(caption, storage));
deba@1024
  2411
      return *this;
deba@1024
  2412
    }
deba@1024
  2413
deba@1024
  2414
    /// \brief Arc map reading rule
deba@1024
  2415
    ///
deba@1024
  2416
    /// Add an arc map reading rule to the reader.
deba@1024
  2417
    template <typename Map>
deba@1024
  2418
    BpGraphReader& arcMap(const std::string& caption, Map& map) {
deba@1024
  2419
      checkConcept<concepts::WriteMap<Arc, typename Map::Value>, Map>();
deba@1024
  2420
      _reader_bits::MapStorageBase<Edge>* forward_storage =
deba@1024
  2421
        new _reader_bits::GraphArcMapStorage<Graph, true, Map>(_graph, map);
deba@1024
  2422
      _edge_maps.push_back(std::make_pair('+' + caption, forward_storage));
deba@1024
  2423
      _reader_bits::MapStorageBase<Edge>* backward_storage =
deba@1024
  2424
        new _reader_bits::GraphArcMapStorage<BGR, false, Map>(_graph, map);
deba@1024
  2425
      _edge_maps.push_back(std::make_pair('-' + caption, backward_storage));
deba@1024
  2426
      return *this;
deba@1024
  2427
    }
deba@1024
  2428
deba@1024
  2429
    /// \brief Arc map reading rule
deba@1024
  2430
    ///
deba@1024
  2431
    /// Add an arc map reading rule with specialized converter to the
deba@1024
  2432
    /// reader.
deba@1024
  2433
    template <typename Map, typename Converter>
deba@1024
  2434
    BpGraphReader& arcMap(const std::string& caption, Map& map,
deba@1024
  2435
                          const Converter& converter = Converter()) {
deba@1024
  2436
      checkConcept<concepts::WriteMap<Arc, typename Map::Value>, Map>();
deba@1024
  2437
      _reader_bits::MapStorageBase<Edge>* forward_storage =
deba@1024
  2438
        new _reader_bits::GraphArcMapStorage<BGR, true, Map, Converter>
deba@1024
  2439
        (_graph, map, converter);
deba@1024
  2440
      _edge_maps.push_back(std::make_pair('+' + caption, forward_storage));
deba@1024
  2441
      _reader_bits::MapStorageBase<Edge>* backward_storage =
deba@1024
  2442
        new _reader_bits::GraphArcMapStorage<BGR, false, Map, Converter>
deba@1024
  2443
        (_graph, map, converter);
deba@1024
  2444
      _edge_maps.push_back(std::make_pair('-' + caption, backward_storage));
deba@1024
  2445
      return *this;
deba@1024
  2446
    }
deba@1024
  2447
deba@1024
  2448
    /// \brief Attribute reading rule
deba@1024
  2449
    ///
deba@1024
  2450
    /// Add an attribute reading rule to the reader.
deba@1024
  2451
    template <typename Value>
deba@1024
  2452
    BpGraphReader& attribute(const std::string& caption, Value& value) {
deba@1024
  2453
      _reader_bits::ValueStorageBase* storage =
deba@1024
  2454
        new _reader_bits::ValueStorage<Value>(value);
deba@1024
  2455
      _attributes.insert(std::make_pair(caption, storage));
deba@1024
  2456
      return *this;
deba@1024
  2457
    }
deba@1024
  2458
deba@1024
  2459
    /// \brief Attribute reading rule
deba@1024
  2460
    ///
deba@1024
  2461
    /// Add an attribute reading rule with specialized converter to the
deba@1024
  2462
    /// reader.
deba@1024
  2463
    template <typename Value, typename Converter>
deba@1024
  2464
    BpGraphReader& attribute(const std::string& caption, Value& value,
deba@1024
  2465
                             const Converter& converter = Converter()) {
deba@1024
  2466
      _reader_bits::ValueStorageBase* storage =
deba@1024
  2467
        new _reader_bits::ValueStorage<Value, Converter>(value, converter);
deba@1024
  2468
      _attributes.insert(std::make_pair(caption, storage));
deba@1024
  2469
      return *this;
deba@1024
  2470
    }
deba@1024
  2471
deba@1024
  2472
    /// \brief Node reading rule
deba@1024
  2473
    ///
deba@1024
  2474
    /// Add a node reading rule to reader.
deba@1024
  2475
    BpGraphReader& node(const std::string& caption, Node& node) {
deba@1024
  2476
      typedef _reader_bits::MapLookUpConverter<Node> Converter;
deba@1024
  2477
      Converter converter(_node_index);
deba@1024
  2478
      _reader_bits::ValueStorageBase* storage =
deba@1024
  2479
        new _reader_bits::ValueStorage<Node, Converter>(node, converter);
deba@1024
  2480
      _attributes.insert(std::make_pair(caption, storage));
deba@1024
  2481
      return *this;
deba@1024
  2482
    }
deba@1024
  2483
deba@1024
  2484
    /// \brief Edge reading rule
deba@1024
  2485
    ///
deba@1024
  2486
    /// Add an edge reading rule to reader.
deba@1024
  2487
    BpGraphReader& edge(const std::string& caption, Edge& edge) {
deba@1024
  2488
      typedef _reader_bits::MapLookUpConverter<Edge> Converter;
deba@1024
  2489
      Converter converter(_edge_index);
deba@1024
  2490
      _reader_bits::ValueStorageBase* storage =
deba@1024
  2491
        new _reader_bits::ValueStorage<Edge, Converter>(edge, converter);
deba@1024
  2492
      _attributes.insert(std::make_pair(caption, storage));
deba@1024
  2493
      return *this;
deba@1024
  2494
    }
deba@1024
  2495
deba@1024
  2496
    /// \brief Arc reading rule
deba@1024
  2497
    ///
deba@1024
  2498
    /// Add an arc reading rule to reader.
deba@1024
  2499
    BpGraphReader& arc(const std::string& caption, Arc& arc) {
deba@1024
  2500
      typedef _reader_bits::GraphArcLookUpConverter<BGR> Converter;
deba@1024
  2501
      Converter converter(_graph, _edge_index);
deba@1024
  2502
      _reader_bits::ValueStorageBase* storage =
deba@1024
  2503
        new _reader_bits::ValueStorage<Arc, Converter>(arc, converter);
deba@1024
  2504
      _attributes.insert(std::make_pair(caption, storage));
deba@1024
  2505
      return *this;
deba@1024
  2506
    }
deba@1024
  2507
deba@1024
  2508
    /// @}
deba@1024
  2509
deba@1024
  2510
    /// \name Select Section by Name
deba@1024
  2511
    /// @{
deba@1024
  2512
deba@1024
  2513
    /// \brief Set \c \@nodes section to be read
deba@1024
  2514
    ///
deba@1024
  2515
    /// Set \c \@nodes section to be read.
deba@1024
  2516
    BpGraphReader& nodes(const std::string& caption) {
deba@1024
  2517
      _nodes_caption = caption;
deba@1024
  2518
      return *this;
deba@1024
  2519
    }
deba@1024
  2520
deba@1024
  2521
    /// \brief Set \c \@edges section to be read
deba@1024
  2522
    ///
deba@1024
  2523
    /// Set \c \@edges section to be read.
deba@1024
  2524
    BpGraphReader& edges(const std::string& caption) {
deba@1024
  2525
      _edges_caption = caption;
deba@1024
  2526
      return *this;
deba@1024
  2527
    }
deba@1024
  2528
deba@1024
  2529
    /// \brief Set \c \@attributes section to be read
deba@1024
  2530
    ///
deba@1024
  2531
    /// Set \c \@attributes section to be read.
deba@1024
  2532
    BpGraphReader& attributes(const std::string& caption) {
deba@1024
  2533
      _attributes_caption = caption;
deba@1024
  2534
      return *this;
deba@1024
  2535
    }
deba@1024
  2536
deba@1024
  2537
    /// @}
deba@1024
  2538
deba@1024
  2539
    /// \name Using Previously Constructed Node or Edge Set
deba@1024
  2540
    /// @{
deba@1024
  2541
deba@1024
  2542
    /// \brief Use previously constructed node set
deba@1024
  2543
    ///
deba@1024
  2544
    /// Use previously constructed node set, and specify the node
deba@1024
  2545
    /// label map.
deba@1024
  2546
    template <typename Map>
deba@1024
  2547
    BpGraphReader& useNodes(const Map& map) {
deba@1024
  2548
      checkConcept<concepts::ReadMap<Node, typename Map::Value>, Map>();
deba@1024
  2549
      LEMON_ASSERT(!_use_nodes, "Multiple usage of useNodes() member");
deba@1024
  2550
      _use_nodes = true;
deba@1024
  2551
      _writer_bits::DefaultConverter<typename Map::Value> converter;
deba@1024
  2552
      for (NodeIt n(_graph); n != INVALID; ++n) {
deba@1024
  2553
        _node_index.insert(std::make_pair(converter(map[n]), n));
deba@1024
  2554
      }
deba@1024
  2555
      return *this;
deba@1024
  2556
    }
deba@1024
  2557
deba@1024
  2558
    /// \brief Use previously constructed node set
deba@1024
  2559
    ///
deba@1024
  2560
    /// Use previously constructed node set, and specify the node
deba@1024
  2561
    /// label map and a functor which converts the label map values to
deba@1024
  2562
    /// \c std::string.
deba@1024
  2563
    template <typename Map, typename Converter>
deba@1024
  2564
    BpGraphReader& useNodes(const Map& map,
deba@1024
  2565
                            const Converter& converter = Converter()) {
deba@1024
  2566
      checkConcept<concepts::ReadMap<Node, typename Map::Value>, Map>();
deba@1024
  2567
      LEMON_ASSERT(!_use_nodes, "Multiple usage of useNodes() member");
deba@1024
  2568
      _use_nodes = true;
deba@1024
  2569
      for (NodeIt n(_graph); n != INVALID; ++n) {
deba@1024
  2570
        _node_index.insert(std::make_pair(converter(map[n]), n));
deba@1024
  2571
      }
deba@1024
  2572
      return *this;
deba@1024
  2573
    }
deba@1024
  2574
deba@1024
  2575
    /// \brief Use previously constructed edge set
deba@1024
  2576
    ///
deba@1024
  2577
    /// Use previously constructed edge set, and specify the edge
deba@1024
  2578
    /// label map.
deba@1024
  2579
    template <typename Map>
deba@1024
  2580
    BpGraphReader& useEdges(const Map& map) {
deba@1024
  2581
      checkConcept<concepts::ReadMap<Edge, typename Map::Value>, Map>();
deba@1024
  2582
      LEMON_ASSERT(!_use_edges, "Multiple usage of useEdges() member");
deba@1024
  2583
      _use_edges = true;
deba@1024
  2584
      _writer_bits::DefaultConverter<typename Map::Value> converter;
deba@1024
  2585
      for (EdgeIt a(_graph); a != INVALID; ++a) {
deba@1024
  2586
        _edge_index.insert(std::make_pair(converter(map[a]), a));
deba@1024
  2587
      }
deba@1024
  2588
      return *this;
deba@1024
  2589
    }
deba@1024
  2590
deba@1024
  2591
    /// \brief Use previously constructed edge set
deba@1024
  2592
    ///
deba@1024
  2593
    /// Use previously constructed edge set, and specify the edge
deba@1024
  2594
    /// label map and a functor which converts the label map values to
deba@1024
  2595
    /// \c std::string.
deba@1024
  2596
    template <typename Map, typename Converter>
deba@1024
  2597
    BpGraphReader& useEdges(const Map& map,
deba@1024
  2598
                            const Converter& converter = Converter()) {
deba@1024
  2599
      checkConcept<concepts::ReadMap<Edge, typename Map::Value>, Map>();
deba@1024
  2600
      LEMON_ASSERT(!_use_edges, "Multiple usage of useEdges() member");
deba@1024
  2601
      _use_edges = true;
deba@1024
  2602
      for (EdgeIt a(_graph); a != INVALID; ++a) {
deba@1024
  2603
        _edge_index.insert(std::make_pair(converter(map[a]), a));
deba@1024
  2604
      }
deba@1024
  2605
      return *this;
deba@1024
  2606
    }
deba@1024
  2607
deba@1024
  2608
    /// \brief Skip the reading of node section
deba@1024
  2609
    ///
deba@1024
  2610
    /// Omit the reading of the node section. This implies that each node
deba@1024
  2611
    /// map reading rule will be abandoned, and the nodes of the graph
deba@1024
  2612
    /// will not be constructed, which usually cause that the edge set
deba@1024
  2613
    /// could not be read due to lack of node name
deba@1024
  2614
    /// could not be read due to lack of node name resolving.
deba@1024
  2615
    /// Therefore \c skipEdges() function should also be used, or
deba@1024
  2616
    /// \c useNodes() should be used to specify the label of the nodes.
deba@1024
  2617
    BpGraphReader& skipNodes() {
deba@1024
  2618
      LEMON_ASSERT(!_skip_nodes, "Skip nodes already set");
deba@1024
  2619
      _skip_nodes = true;
deba@1024
  2620
      return *this;
deba@1024
  2621
    }
deba@1024
  2622
deba@1024
  2623
    /// \brief Skip the reading of edge section
deba@1024
  2624
    ///
deba@1024
  2625
    /// Omit the reading of the edge section. This implies that each edge
deba@1024
  2626
    /// map reading rule will be abandoned, and the edges of the graph
deba@1024
  2627
    /// will not be constructed.
deba@1024
  2628
    BpGraphReader& skipEdges() {
deba@1024
  2629
      LEMON_ASSERT(!_skip_edges, "Skip edges already set");
deba@1024
  2630
      _skip_edges = true;
deba@1024
  2631
      return *this;
deba@1024
  2632
    }
deba@1024
  2633
deba@1024
  2634
    /// @}
deba@1024
  2635
deba@1024
  2636
  private:
deba@1024
  2637
deba@1024
  2638
    bool readLine() {
deba@1024
  2639
      std::string str;
deba@1024
  2640
      while(++line_num, std::getline(*_is, str)) {
deba@1024
  2641
        line.clear(); line.str(str);
deba@1024
  2642
        char c;
deba@1024
  2643
        if (line >> std::ws >> c && c != '#') {
deba@1024
  2644
          line.putback(c);
deba@1024
  2645
          return true;
deba@1024
  2646
        }
deba@1024
  2647
      }
deba@1024
  2648
      return false;
deba@1024
  2649
    }
deba@1024
  2650
deba@1024
  2651
    bool readSuccess() {
deba@1024
  2652
      return static_cast<bool>(*_is);
deba@1024
  2653
    }
deba@1024
  2654
deba@1024
  2655
    void skipSection() {
deba@1024
  2656
      char c;
deba@1024
  2657
      while (readSuccess() && line >> c && c != '@') {
deba@1024
  2658
        readLine();
deba@1024
  2659
      }
deba@1024
  2660
      if (readSuccess()) {
deba@1024
  2661
        line.putback(c);
deba@1024
  2662
      }
deba@1024
  2663
    }
deba@1024
  2664
deba@1024
  2665
    void readRedNodes() {
deba@1024
  2666
deba@1024
  2667
      std::vector<int> map_index(_red_maps.size());
deba@1024
  2668
      int map_num, label_index;
deba@1024
  2669
deba@1024
  2670
      char c;
deba@1024
  2671
      if (!readLine() || !(line >> c) || c == '@') {
deba@1024
  2672
        if (readSuccess() && line) line.putback(c);
deba@1024
  2673
        if (!_red_maps.empty())
deba@1024
  2674
          throw FormatError("Cannot find map names");
deba@1024
  2675
        return;
deba@1024
  2676
      }
deba@1024
  2677
      line.putback(c);
deba@1024
  2678
deba@1024
  2679
      {
deba@1024
  2680
        std::map<std::string, int> maps;
deba@1024
  2681
deba@1024
  2682
        std::string map;
deba@1024
  2683
        int index = 0;
deba@1024
  2684
        while (_reader_bits::readToken(line, map)) {
deba@1024
  2685
          if (maps.find(map) != maps.end()) {
deba@1024
  2686
            std::ostringstream msg;
deba@1026
  2687
            msg << "Multiple occurence of red node map: " << map;
deba@1024
  2688
            throw FormatError(msg.str());
deba@1024
  2689
          }
deba@1024
  2690
          maps.insert(std::make_pair(map, index));
deba@1024
  2691
          ++index;
deba@1024
  2692
        }
deba@1024
  2693
deba@1024
  2694
        for (int i = 0; i < static_cast<int>(_red_maps.size()); ++i) {
deba@1024
  2695
          std::map<std::string, int>::iterator jt =
deba@1024
  2696
            maps.find(_red_maps[i].first);
deba@1024
  2697
          if (jt == maps.end()) {
deba@1024
  2698
            std::ostringstream msg;
deba@1024
  2699
            msg << "Map not found: " << _red_maps[i].first;
deba@1024
  2700
            throw FormatError(msg.str());
deba@1024
  2701
          }
deba@1024
  2702
          map_index[i] = jt->second;
deba@1024
  2703
        }
deba@1024
  2704
deba@1024
  2705
        {
deba@1024
  2706
          std::map<std::string, int>::iterator jt = maps.find("label");
deba@1024
  2707
          if (jt != maps.end()) {
deba@1024
  2708
            label_index = jt->second;
deba@1024
  2709
          } else {
deba@1024
  2710
            label_index = -1;
deba@1024
  2711
          }
deba@1024
  2712
        }
deba@1024
  2713
        map_num = maps.size();
deba@1024
  2714
      }
deba@1024
  2715
deba@1024
  2716
      while (readLine() && line >> c && c != '@') {
deba@1024
  2717
        line.putback(c);
deba@1024
  2718
deba@1024
  2719
        std::vector<std::string> tokens(map_num);
deba@1024
  2720
        for (int i = 0; i < map_num; ++i) {
deba@1024
  2721
          if (!_reader_bits::readToken(line, tokens[i])) {
deba@1024
  2722
            std::ostringstream msg;
deba@1024
  2723
            msg << "Column not found (" << i + 1 << ")";
deba@1024
  2724
            throw FormatError(msg.str());
deba@1024
  2725
          }
deba@1024
  2726
        }
deba@1024
  2727
        if (line >> std::ws >> c)
deba@1024
  2728
          throw FormatError("Extra character at the end of line");
deba@1024
  2729
deba@1024
  2730
        Node n;
deba@1024
  2731
        if (!_use_nodes) {
deba@1024
  2732
          n = _graph.addRedNode();
deba@1024
  2733
          if (label_index != -1)
deba@1024
  2734
            _node_index.insert(std::make_pair(tokens[label_index], n));
deba@1024
  2735
        } else {
deba@1024
  2736
          if (label_index == -1)
deba@1024
  2737
            throw FormatError("Label map not found");
deba@1024
  2738
          typename std::map<std::string, Node>::iterator it =
deba@1024
  2739
            _node_index.find(tokens[label_index]);
deba@1024
  2740
          if (it == _node_index.end()) {
deba@1024
  2741
            std::ostringstream msg;
deba@1024
  2742
            msg << "Node with label not found: " << tokens[label_index];
deba@1024
  2743
            throw FormatError(msg.str());
deba@1024
  2744
          }
deba@1024
  2745
          n = it->second;
deba@1024
  2746
        }
deba@1024
  2747
deba@1024
  2748
        for (int i = 0; i < static_cast<int>(_red_maps.size()); ++i) {
deba@1024
  2749
          _red_maps[i].second->set(n, tokens[map_index[i]]);
deba@1024
  2750
        }
deba@1024
  2751
deba@1024
  2752
      }
deba@1024
  2753
      if (readSuccess()) {
deba@1024
  2754
        line.putback(c);
deba@1024
  2755
      }
deba@1024
  2756
    }
deba@1024
  2757
deba@1024
  2758
    void readBlueNodes() {
deba@1024
  2759
deba@1024
  2760
      std::vector<int> map_index(_blue_maps.size());
deba@1024
  2761
      int map_num, label_index;
deba@1024
  2762
deba@1024
  2763
      char c;
deba@1024
  2764
      if (!readLine() || !(line >> c) || c == '@') {
deba@1024
  2765
        if (readSuccess() && line) line.putback(c);
deba@1024
  2766
        if (!_blue_maps.empty())
deba@1024
  2767
          throw FormatError("Cannot find map names");
deba@1024
  2768
        return;
deba@1024
  2769
      }
deba@1024
  2770
      line.putback(c);
deba@1024
  2771
deba@1024
  2772
      {
deba@1024
  2773
        std::map<std::string, int> maps;
deba@1024
  2774
deba@1024
  2775
        std::string map;
deba@1024
  2776
        int index = 0;
deba@1024
  2777
        while (_reader_bits::readToken(line, map)) {
deba@1024
  2778
          if (maps.find(map) != maps.end()) {
deba@1024
  2779
            std::ostringstream msg;
deba@1026
  2780
            msg << "Multiple occurence of blue node map: " << map;
deba@1024
  2781
            throw FormatError(msg.str());
deba@1024
  2782
          }
deba@1024
  2783
          maps.insert(std::make_pair(map, index));
deba@1024
  2784
          ++index;
deba@1024
  2785
        }
deba@1024
  2786
deba@1024
  2787
        for (int i = 0; i < static_cast<int>(_blue_maps.size()); ++i) {
deba@1024
  2788
          std::map<std::string, int>::iterator jt =
deba@1024
  2789
            maps.find(_blue_maps[i].first);
deba@1024
  2790
          if (jt == maps.end()) {
deba@1024
  2791
            std::ostringstream msg;
deba@1024
  2792
            msg << "Map not found: " << _blue_maps[i].first;
deba@1024
  2793
            throw FormatError(msg.str());
deba@1024
  2794
          }
deba@1024
  2795
          map_index[i] = jt->second;
deba@1024
  2796
        }
deba@1024
  2797
deba@1024
  2798
        {
deba@1024
  2799
          std::map<std::string, int>::iterator jt = maps.find("label");
deba@1024
  2800
          if (jt != maps.end()) {
deba@1024
  2801
            label_index = jt->second;
deba@1024
  2802
          } else {
deba@1024
  2803
            label_index = -1;
deba@1024
  2804
          }
deba@1024
  2805
        }
deba@1024
  2806
        map_num = maps.size();
deba@1024
  2807
      }
deba@1024
  2808
deba@1024
  2809
      while (readLine() && line >> c && c != '@') {
deba@1024
  2810
        line.putback(c);
deba@1024
  2811
deba@1024
  2812
        std::vector<std::string> tokens(map_num);
deba@1024
  2813
        for (int i = 0; i < map_num; ++i) {
deba@1024
  2814
          if (!_reader_bits::readToken(line, tokens[i])) {
deba@1024
  2815
            std::ostringstream msg;
deba@1024
  2816
            msg << "Column not found (" << i + 1 << ")";
deba@1024
  2817
            throw FormatError(msg.str());
deba@1024
  2818
          }
deba@1024
  2819
        }
deba@1024
  2820
        if (line >> std::ws >> c)
deba@1024
  2821
          throw FormatError("Extra character at the end of line");
deba@1024
  2822
deba@1024
  2823
        Node n;
deba@1024
  2824
        if (!_use_nodes) {
deba@1024
  2825
          n = _graph.addBlueNode();
deba@1024
  2826
          if (label_index != -1)
deba@1024
  2827
            _node_index.insert(std::make_pair(tokens[label_index], n));
deba@1024
  2828
        } else {
deba@1024
  2829
          if (label_index == -1)
deba@1024
  2830
            throw FormatError("Label map not found");
deba@1024
  2831
          typename std::map<std::string, Node>::iterator it =
deba@1024
  2832
            _node_index.find(tokens[label_index]);
deba@1024
  2833
          if (it == _node_index.end()) {
deba@1024
  2834
            std::ostringstream msg;
deba@1024
  2835
            msg << "Node with label not found: " << tokens[label_index];
deba@1024
  2836
            throw FormatError(msg.str());
deba@1024
  2837
          }
deba@1024
  2838
          n = it->second;
deba@1024
  2839
        }
deba@1024
  2840
deba@1024
  2841
        for (int i = 0; i < static_cast<int>(_blue_maps.size()); ++i) {
deba@1024
  2842
          _blue_maps[i].second->set(n, tokens[map_index[i]]);
deba@1024
  2843
        }
deba@1024
  2844
deba@1024
  2845
      }
deba@1024
  2846
      if (readSuccess()) {
deba@1024
  2847
        line.putback(c);
deba@1024
  2848
      }
deba@1024
  2849
    }
deba@1024
  2850
deba@1024
  2851
    void readEdges() {
deba@1024
  2852
deba@1024
  2853
      std::vector<int> map_index(_edge_maps.size());
deba@1024
  2854
      int map_num, label_index;
deba@1024
  2855
deba@1024
  2856
      char c;
deba@1024
  2857
      if (!readLine() || !(line >> c) || c == '@') {
deba@1024
  2858
        if (readSuccess() && line) line.putback(c);
deba@1024
  2859
        if (!_edge_maps.empty())
deba@1024
  2860
          throw FormatError("Cannot find map names");
deba@1024
  2861
        return;
deba@1024
  2862
      }
deba@1024
  2863
      line.putback(c);
deba@1024
  2864
deba@1024
  2865
      {
deba@1024
  2866
        std::map<std::string, int> maps;
deba@1024
  2867
deba@1024
  2868
        std::string map;
deba@1024
  2869
        int index = 0;
deba@1024
  2870
        while (_reader_bits::readToken(line, map)) {
deba@1024
  2871
          if (maps.find(map) != maps.end()) {
deba@1024
  2872
            std::ostringstream msg;
deba@1024
  2873
            msg << "Multiple occurence of edge map: " << map;
deba@1024
  2874
            throw FormatError(msg.str());
deba@1024
  2875
          }
deba@1024
  2876
          maps.insert(std::make_pair(map, index));
deba@1024
  2877
          ++index;
deba@1024
  2878
        }
deba@1024
  2879
deba@1024
  2880
        for (int i = 0; i < static_cast<int>(_edge_maps.size()); ++i) {
deba@1024
  2881
          std::map<std::string, int>::iterator jt =
deba@1024
  2882
            maps.find(_edge_maps[i].first);
deba@1024
  2883
          if (jt == maps.end()) {
deba@1024
  2884
            std::ostringstream msg;
deba@1024
  2885
            msg << "Map not found: " << _edge_maps[i].first;
deba@1024
  2886
            throw FormatError(msg.str());
deba@1024
  2887
          }
deba@1024
  2888
          map_index[i] = jt->second;
deba@1024
  2889
        }
deba@1024
  2890
deba@1024
  2891
        {
deba@1024
  2892
          std::map<std::string, int>::iterator jt = maps.find("label");
deba@1024
  2893
          if (jt != maps.end()) {
deba@1024
  2894
            label_index = jt->second;
deba@1024
  2895
          } else {
deba@1024
  2896
            label_index = -1;
deba@1024
  2897
          }
deba@1024
  2898
        }
deba@1024
  2899
        map_num = maps.size();
deba@1024
  2900
      }
deba@1024
  2901
deba@1024
  2902
      while (readLine() && line >> c && c != '@') {
deba@1024
  2903
        line.putback(c);
deba@1024
  2904
deba@1024
  2905
        std::string source_token;
deba@1024
  2906
        std::string target_token;
deba@1024
  2907
deba@1024
  2908
        if (!_reader_bits::readToken(line, source_token))
deba@1024
  2909
          throw FormatError("Red node not found");
deba@1024
  2910
deba@1024
  2911
        if (!_reader_bits::readToken(line, target_token))
deba@1024
  2912
          throw FormatError("Blue node not found");
deba@1024
  2913
deba@1024
  2914
        std::vector<std::string> tokens(map_num);
deba@1024
  2915
        for (int i = 0; i < map_num; ++i) {
deba@1024
  2916
          if (!_reader_bits::readToken(line, tokens[i])) {
deba@1024
  2917
            std::ostringstream msg;
deba@1024
  2918
            msg << "Column not found (" << i + 1 << ")";
deba@1024
  2919
            throw FormatError(msg.str());
deba@1024
  2920
          }
deba@1024
  2921
        }
deba@1024
  2922
        if (line >> std::ws >> c)
deba@1024
  2923
          throw FormatError("Extra character at the end of line");
deba@1024
  2924
deba@1024
  2925
        Edge e;
deba@1024
  2926
        if (!_use_edges) {
deba@1024
  2927
deba@1024
  2928
          typename NodeIndex::iterator it;
deba@1024
  2929
deba@1024
  2930
          it = _node_index.find(source_token);
deba@1024
  2931
          if (it == _node_index.end()) {
deba@1024
  2932
            std::ostringstream msg;
deba@1024
  2933
            msg << "Item not found: " << source_token;
deba@1024
  2934
            throw FormatError(msg.str());
deba@1024
  2935
          }
deba@1024
  2936
          Node source = it->second;
deba@1024
  2937
          if (!_graph.red(source)) {
deba@1024
  2938
            std::ostringstream msg;
deba@1024
  2939
            msg << "Item is not red node: " << source_token;
deba@1024
  2940
            throw FormatError(msg.str());
deba@1024
  2941
          }
deba@1024
  2942
deba@1024
  2943
          it = _node_index.find(target_token);
deba@1024
  2944
          if (it == _node_index.end()) {
deba@1024
  2945
            std::ostringstream msg;
deba@1024
  2946
            msg << "Item not found: " << target_token;
deba@1024
  2947
            throw FormatError(msg.str());
deba@1024
  2948
          }
deba@1024
  2949
          Node target = it->second;
deba@1024
  2950
          if (!_graph.blue(target)) {
deba@1024
  2951
            std::ostringstream msg;
deba@1024
  2952
            msg << "Item is not red node: " << source_token;
deba@1024
  2953
            throw FormatError(msg.str());
deba@1024
  2954
          }
deba@1024
  2955
deba@1024
  2956
          e = _graph.addEdge(source, target);
deba@1024
  2957
          if (label_index != -1)
deba@1024
  2958
            _edge_index.insert(std::make_pair(tokens[label_index], e));
deba@1024
  2959
        } else {
deba@1024
  2960
          if (label_index == -1)
deba@1024
  2961
            throw FormatError("Label map not found");
deba@1024
  2962
          typename std::map<std::string, Edge>::iterator it =
deba@1024
  2963
            _edge_index.find(tokens[label_index]);
deba@1024
  2964
          if (it == _edge_index.end()) {
deba@1024
  2965
            std::ostringstream msg;
deba@1024
  2966
            msg << "Edge with label not found: " << tokens[label_index];
deba@1024
  2967
            throw FormatError(msg.str());
deba@1024
  2968
          }
deba@1024
  2969
          e = it->second;
deba@1024
  2970
        }
deba@1024
  2971
deba@1024
  2972
        for (int i = 0; i < static_cast<int>(_edge_maps.size()); ++i) {
deba@1024
  2973
          _edge_maps[i].second->set(e, tokens[map_index[i]]);
deba@1024
  2974
        }
deba@1024
  2975
deba@1024
  2976
      }
deba@1024
  2977
      if (readSuccess()) {
deba@1024
  2978
        line.putback(c);
deba@1024
  2979
      }
deba@1024
  2980
    }
deba@1024
  2981
deba@1024
  2982
    void readAttributes() {
deba@1024
  2983
deba@1024
  2984
      std::set<std::string> read_attr;
deba@1024
  2985
deba@1024
  2986
      char c;
deba@1024
  2987
      while (readLine() && line >> c && c != '@') {
deba@1024
  2988
        line.putback(c);
deba@1024
  2989
deba@1024
  2990
        std::string attr, token;
deba@1024
  2991
        if (!_reader_bits::readToken(line, attr))
deba@1024
  2992
          throw FormatError("Attribute name not found");
deba@1024
  2993
        if (!_reader_bits::readToken(line, token))
deba@1024
  2994
          throw FormatError("Attribute value not found");
deba@1024
  2995
        if (line >> c)
deba@1024
  2996
          throw FormatError("Extra character at the end of line");
deba@1024
  2997
deba@1024
  2998
        {
deba@1024
  2999
          std::set<std::string>::iterator it = read_attr.find(attr);
deba@1024
  3000
          if (it != read_attr.end()) {
deba@1024
  3001
            std::ostringstream msg;
deba@1024
  3002
            msg << "Multiple occurence of attribute: " << attr;
deba@1024
  3003
            throw FormatError(msg.str());
deba@1024
  3004
          }
deba@1024
  3005
          read_attr.insert(attr);
deba@1024
  3006
        }
deba@1024
  3007
deba@1024
  3008
        {
deba@1024
  3009
          typename Attributes::iterator it = _attributes.lower_bound(attr);
deba@1024
  3010
          while (it != _attributes.end() && it->first == attr) {
deba@1024
  3011
            it->second->set(token);
deba@1024
  3012
            ++it;
deba@1024
  3013
          }
deba@1024
  3014
        }
deba@1024
  3015
deba@1024
  3016
      }
deba@1024
  3017
      if (readSuccess()) {
deba@1024
  3018
        line.putback(c);
deba@1024
  3019
      }
deba@1024
  3020
      for (typename Attributes::iterator it = _attributes.begin();
deba@1024
  3021
           it != _attributes.end(); ++it) {
deba@1024
  3022
        if (read_attr.find(it->first) == read_attr.end()) {
deba@1024
  3023
          std::ostringstream msg;
deba@1024
  3024
          msg << "Attribute not found: " << it->first;
deba@1024
  3025
          throw FormatError(msg.str());
deba@1024
  3026
        }
deba@1024
  3027
      }
deba@1024
  3028
    }
deba@1024
  3029
deba@1024
  3030
  public:
deba@1024
  3031
deba@1024
  3032
    /// \name Execution of the Reader
deba@1024
  3033
    /// @{
deba@1024
  3034
deba@1024
  3035
    /// \brief Start the batch processing
deba@1024
  3036
    ///
deba@1024
  3037
    /// This function starts the batch processing
deba@1024
  3038
    void run() {
deba@1024
  3039
deba@1024
  3040
      LEMON_ASSERT(_is != 0, "This reader assigned to an other reader");
deba@1024
  3041
deba@1024
  3042
      bool red_nodes_done = _skip_nodes;
deba@1024
  3043
      bool blue_nodes_done = _skip_nodes;
deba@1024
  3044
      bool edges_done = _skip_edges;
deba@1024
  3045
      bool attributes_done = false;
deba@1024
  3046
deba@1024
  3047
      line_num = 0;
deba@1024
  3048
      readLine();
deba@1024
  3049
      skipSection();
deba@1024
  3050
deba@1024
  3051
      while (readSuccess()) {
deba@1024
  3052
        try {
deba@1024
  3053
          char c;
deba@1024
  3054
          std::string section, caption;
deba@1024
  3055
          line >> c;
deba@1024
  3056
          _reader_bits::readToken(line, section);
deba@1024
  3057
          _reader_bits::readToken(line, caption);
deba@1024
  3058
deba@1024
  3059
          if (line >> c)
deba@1024
  3060
            throw FormatError("Extra character at the end of line");
deba@1024
  3061
deba@1024
  3062
          if (section == "red_nodes" && !red_nodes_done) {
deba@1024
  3063
            if (_nodes_caption.empty() || _nodes_caption == caption) {
deba@1024
  3064
              readRedNodes();
deba@1024
  3065
              red_nodes_done = true;
deba@1024
  3066
            }
deba@1024
  3067
          } else if (section == "blue_nodes" && !blue_nodes_done) {
deba@1024
  3068
            if (_nodes_caption.empty() || _nodes_caption == caption) {
deba@1024
  3069
              readBlueNodes();
deba@1024
  3070
              blue_nodes_done = true;
deba@1024
  3071
            }
deba@1024
  3072
          } else if ((section == "edges" || section == "arcs") &&
deba@1024
  3073
                     !edges_done) {
deba@1024
  3074
            if (_edges_caption.empty() || _edges_caption == caption) {
deba@1024
  3075
              readEdges();
deba@1024
  3076
              edges_done = true;
deba@1024
  3077
            }
deba@1024
  3078
          } else if (section == "attributes" && !attributes_done) {
deba@1024
  3079
            if (_attributes_caption.empty() || _attributes_caption == caption) {
deba@1024
  3080
              readAttributes();
deba@1024
  3081
              attributes_done = true;
deba@1024
  3082
            }
deba@1024
  3083
          } else {
deba@1024
  3084
            readLine();
deba@1024
  3085
            skipSection();
deba@1024
  3086
          }
deba@1024
  3087
        } catch (FormatError& error) {
deba@1024
  3088
          error.line(line_num);
deba@1024
  3089
          error.file(_filename);
deba@1024
  3090
          throw;
deba@1024
  3091
        }
deba@1024
  3092
      }
deba@1024
  3093
deba@1024
  3094
      if (!red_nodes_done) {
deba@1024
  3095
        throw FormatError("Section @red_nodes not found");
deba@1024
  3096
      }
deba@1024
  3097
deba@1024
  3098
      if (!blue_nodes_done) {
deba@1024
  3099
        throw FormatError("Section @blue_nodes not found");
deba@1024
  3100
      }
deba@1024
  3101
deba@1024
  3102
      if (!edges_done) {
deba@1024
  3103
        throw FormatError("Section @edges not found");
deba@1024
  3104
      }
deba@1024
  3105
deba@1024
  3106
      if (!attributes_done && !_attributes.empty()) {
deba@1024
  3107
        throw FormatError("Section @attributes not found");
deba@1024
  3108
      }
deba@1024
  3109
deba@1024
  3110
    }
deba@1024
  3111
deba@1024
  3112
    /// @}
deba@1024
  3113
deba@1024
  3114
  };
deba@1024
  3115
deba@1024
  3116
  /// \ingroup lemon_io
deba@1024
  3117
  ///
deba@1024
  3118
  /// \brief Return a \ref BpGraphReader class
deba@1024
  3119
  ///
deba@1024
  3120
  /// This function just returns a \ref BpGraphReader class.
deba@1024
  3121
  ///
deba@1024
  3122
  /// With this function a graph can be read from an
deba@1024
  3123
  /// \ref lgf-format "LGF" file or input stream with several maps and
deba@1024
  3124
  /// attributes. For example, there is bipartite weighted matching problem
deba@1024
  3125
  /// on a graph, i.e. a graph with a \e weight map on the edges. This
deba@1024
  3126
  /// graph can be read with the following code:
deba@1024
  3127
  ///
deba@1024
  3128
  ///\code
deba@1024
  3129
  ///ListBpGraph graph;
deba@1024
  3130
  ///ListBpGraph::EdgeMap<int> weight(graph);
deba@1024
  3131
  ///bpGraphReader(graph, std::cin).
deba@1024
  3132
  ///  edgeMap("weight", weight).
deba@1024
  3133
  ///  run();
deba@1024
  3134
  ///\endcode
deba@1024
  3135
  ///
deba@1024
  3136
  /// For a complete documentation, please see the \ref BpGraphReader
deba@1024
  3137
  /// class documentation.
deba@1024
  3138
  /// \warning Don't forget to put the \ref BpGraphReader::run() "run()"
deba@1024
  3139
  /// to the end of the parameter list.
deba@1024
  3140
  /// \relates BpGraphReader
deba@1024
  3141
  /// \sa bpGraphReader(TBGR& graph, const std::string& fn)
deba@1024
  3142
  /// \sa bpGraphReader(TBGR& graph, const char* fn)
deba@1024
  3143
  template <typename TBGR>
deba@1024
  3144
  BpGraphReader<TBGR> bpGraphReader(TBGR& graph, std::istream& is) {
deba@1024
  3145
    BpGraphReader<TBGR> tmp(graph, is);
deba@1024
  3146
    return tmp;
deba@1024
  3147
  }
deba@1024
  3148
deba@1024
  3149
  /// \brief Return a \ref BpGraphReader class
deba@1024
  3150
  ///
deba@1024
  3151
  /// This function just returns a \ref BpGraphReader class.
deba@1024
  3152
  /// \relates BpGraphReader
deba@1024
  3153
  /// \sa bpGraphReader(TBGR& graph, std::istream& is)
deba@1024
  3154
  template <typename TBGR>
deba@1024
  3155
  BpGraphReader<TBGR> bpGraphReader(TBGR& graph, const std::string& fn) {
deba@1024
  3156
    BpGraphReader<TBGR> tmp(graph, fn);
deba@1024
  3157
    return tmp;
deba@1024
  3158
  }
deba@1024
  3159
deba@1024
  3160
  /// \brief Return a \ref BpGraphReader class
deba@1024
  3161
  ///
deba@1024
  3162
  /// This function just returns a \ref BpGraphReader class.
deba@1024
  3163
  /// \relates BpGraphReader
deba@1024
  3164
  /// \sa bpGraphReader(TBGR& graph, std::istream& is)
deba@1024
  3165
  template <typename TBGR>
deba@1024
  3166
  BpGraphReader<TBGR> bpGraphReader(TBGR& graph, const char* fn) {
deba@1024
  3167
    BpGraphReader<TBGR> tmp(graph, fn);
deba@1024
  3168
    return tmp;
deba@1024
  3169
  }
deba@1024
  3170
deba@190
  3171
  class SectionReader;
deba@190
  3172
deba@190
  3173
  SectionReader sectionReader(std::istream& is);
deba@190
  3174
  SectionReader sectionReader(const std::string& fn);
deba@190
  3175
  SectionReader sectionReader(const char* fn);
alpar@209
  3176
kpeter@192
  3177
  /// \ingroup lemon_io
kpeter@192
  3178
  ///
deba@189
  3179
  /// \brief Section reader class
deba@189
  3180
  ///
alpar@209
  3181
  /// In the \ref lgf-format "LGF" file extra sections can be placed,
kpeter@192
  3182
  /// which contain any data in arbitrary format. Such sections can be
alpar@209
  3183
  /// read with this class. A reading rule can be added to the class
kpeter@192
  3184
  /// with two different functions. With the \c sectionLines() function a
kpeter@192
  3185
  /// functor can process the section line-by-line, while with the \c
deba@189
  3186
  /// sectionStream() member the section can be read from an input
deba@189
  3187
  /// stream.
deba@189
  3188
  class SectionReader {
deba@189
  3189
  private:
alpar@209
  3190
deba@189
  3191
    std::istream* _is;
deba@189
  3192
    bool local_is;
deba@290
  3193
    std::string _filename;
deba@189
  3194
deba@189
  3195
    typedef std::map<std::string, _reader_bits::Section*> Sections;
deba@189
  3196
    Sections _sections;
deba@189
  3197
deba@189
  3198
    int line_num;
deba@189
  3199
    std::istringstream line;
deba@189
  3200
deba@189
  3201
  public:
deba@189
  3202
deba@189
  3203
    /// \brief Constructor
deba@189
  3204
    ///
deba@189
  3205
    /// Construct a section reader, which reads from the given input
deba@189
  3206
    /// stream.
alpar@209
  3207
    SectionReader(std::istream& is)
deba@189
  3208
      : _is(&is), local_is(false) {}
deba@189
  3209
deba@189
  3210
    /// \brief Constructor
deba@189
  3211
    ///
deba@189
  3212
    /// Construct a section reader, which reads from the given file.
alpar@209
  3213
    SectionReader(const std::string& fn)
deba@290
  3214
      : _is(new std::ifstream(fn.c_str())), local_is(true),
deba@290
  3215
        _filename(fn) {
deba@295
  3216
      if (!(*_is)) {
deba@295
  3217
        delete _is;
deba@295
  3218
        throw IoError("Cannot open file", fn);
deba@295
  3219
      }
deba@290
  3220
    }
alpar@209
  3221
deba@189
  3222
    /// \brief Constructor
deba@189
  3223
    ///
deba@189
  3224
    /// Construct a section reader, which reads from the given file.
alpar@209
  3225
    SectionReader(const char* fn)
deba@290
  3226
      : _is(new std::ifstream(fn)), local_is(true),
deba@290
  3227
        _filename(fn) {
deba@295
  3228
      if (!(*_is)) {
deba@295
  3229
        delete _is;
deba@295
  3230
        throw IoError("Cannot open file", fn);
deba@295
  3231
      }
deba@290
  3232
    }
deba@189
  3233
deba@189
  3234
    /// \brief Destructor
deba@189
  3235
    ~SectionReader() {
alpar@209
  3236
      for (Sections::iterator it = _sections.begin();
alpar@209
  3237
           it != _sections.end(); ++it) {
alpar@209
  3238
        delete it->second;
deba@189
  3239
      }
deba@189
  3240
deba@189
  3241
      if (local_is) {
alpar@209
  3242
        delete _is;
deba@189
  3243
      }
deba@189
  3244
deba@189
  3245
    }
deba@189
  3246
deba@189
  3247
  private:
deba@190
  3248
deba@190
  3249
    friend SectionReader sectionReader(std::istream& is);
deba@190
  3250
    friend SectionReader sectionReader(const std::string& fn);
deba@190
  3251
    friend SectionReader sectionReader(const char* fn);
deba@190
  3252
alpar@209
  3253
    SectionReader(SectionReader& other)
deba@190
  3254
      : _is(other._is), local_is(other.local_is) {
deba@190
  3255
deba@190
  3256
      other._is = 0;
deba@190
  3257
      other.local_is = false;
alpar@209
  3258
deba@190
  3259
      _sections.swap(other._sections);
deba@190
  3260
    }
alpar@209
  3261
deba@189
  3262
    SectionReader& operator=(const SectionReader&);
deba@189
  3263
deba@189
  3264
  public:
deba@189
  3265
kpeter@584
  3266
    /// \name Section Readers
deba@189
  3267
    /// @{
deba@189
  3268
deba@189
  3269
    /// \brief Add a section processor with line oriented reading
deba@189
  3270
    ///
deba@189
  3271
    /// The first parameter is the type descriptor of the section, the
deba@189
  3272
    /// second is a functor, which takes just one \c std::string
deba@189
  3273
    /// parameter. At the reading process, each line of the section
deba@189
  3274
    /// will be given to the functor object. However, the empty lines
deba@189
  3275
    /// and the comment lines are filtered out, and the leading
deba@189
  3276
    /// whitespaces are trimmed from each processed string.
deba@189
  3277
    ///
kpeter@786
  3278
    /// For example, let's see a section, which contain several
deba@189
  3279
    /// integers, which should be inserted into a vector.
deba@189
  3280
    ///\code
deba@189
  3281
    ///  @numbers
deba@189
  3282
    ///  12 45 23
deba@189
  3283
    ///  4
deba@189
  3284
    ///  23 6
deba@189
  3285
    ///\endcode
deba@189
  3286
    ///
kpeter@192
  3287
    /// The functor is implemented as a struct:
deba@189
  3288
    ///\code
deba@189
  3289
    ///  struct NumberSection {
deba@189
  3290
    ///    std::vector<int>& _data;
deba@189
  3291
    ///    NumberSection(std::vector<int>& data) : _data(data) {}
deba@189
  3292
    ///    void operator()(const std::string& line) {
deba@189
  3293
    ///      std::istringstream ls(line);
deba@189
  3294
    ///      int value;
deba@189
  3295
    ///      while (ls >> value) _data.push_back(value);
deba@189
  3296
    ///    }
deba@189
  3297
    ///  };
deba@189
  3298
    ///
deba@189
  3299
    ///  // ...
deba@189
  3300
    ///
alpar@209
  3301
    ///  reader.sectionLines("numbers", NumberSection(vec));
deba@189
  3302
    ///\endcode
deba@189
  3303
    template <typename Functor>
deba@189
  3304
    SectionReader& sectionLines(const std::string& type, Functor functor) {
kpeter@192
  3305
      LEMON_ASSERT(!type.empty(), "Type is empty.");
alpar@209
  3306
      LEMON_ASSERT(_sections.find(type) == _sections.end(),
alpar@209
  3307
                   "Multiple reading of section.");
alpar@209
  3308
      _sections.insert(std::make_pair(type,
deba@189
  3309
        new _reader_bits::LineSection<Functor>(functor)));
deba@189
  3310
      return *this;
deba@189
  3311
    }
deba@189
  3312
deba@189
  3313
deba@189
  3314
    /// \brief Add a section processor with stream oriented reading
deba@189
  3315
    ///
deba@189
  3316
    /// The first parameter is the type of the section, the second is
kpeter@192
  3317
    /// a functor, which takes an \c std::istream& and an \c int&
deba@189
  3318
    /// parameter, the latter regard to the line number of stream. The
deba@189
  3319
    /// functor can read the input while the section go on, and the
deba@189
  3320
    /// line number should be modified accordingly.
deba@189
  3321
    template <typename Functor>
deba@189
  3322
    SectionReader& sectionStream(const std::string& type, Functor functor) {
kpeter@192
  3323
      LEMON_ASSERT(!type.empty(), "Type is empty.");
alpar@209
  3324
      LEMON_ASSERT(_sections.find(type) == _sections.end(),
alpar@209
  3325
                   "Multiple reading of section.");
alpar@209
  3326
      _sections.insert(std::make_pair(type,
alpar@209
  3327
         new _reader_bits::StreamSection<Functor>(functor)));
deba@189
  3328
      return *this;
alpar@209
  3329
    }
alpar@209
  3330
deba@189
  3331
    /// @}
deba@189
  3332
deba@189
  3333
  private:
deba@189
  3334
deba@189
  3335
    bool readLine() {
deba@189
  3336
      std::string str;
deba@189
  3337
      while(++line_num, std::getline(*_is, str)) {
alpar@209
  3338
        line.clear(); line.str(str);
alpar@209
  3339
        char c;
alpar@209
  3340
        if (line >> std::ws >> c && c != '#') {
alpar@209
  3341
          line.putback(c);
alpar@209
  3342
          return true;
alpar@209
  3343
        }
deba@189
  3344
      }
deba@189
  3345
      return false;
deba@189
  3346
    }
deba@189
  3347
deba@189
  3348
    bool readSuccess() {
deba@189
  3349
      return static_cast<bool>(*_is);
deba@189
  3350
    }
alpar@209
  3351
deba@189
  3352
    void skipSection() {
deba@189
  3353
      char c;
deba@189
  3354
      while (readSuccess() && line >> c && c != '@') {
alpar@209
  3355
        readLine();
deba@189
  3356
      }
deba@427
  3357
      if (readSuccess()) {
deba@427
  3358
        line.putback(c);
deba@427
  3359
      }
deba@189
  3360
    }
deba@189
  3361
deba@189
  3362
  public:
deba@189
  3363
deba@189
  3364
kpeter@584
  3365
    /// \name Execution of the Reader
deba@189
  3366
    /// @{
deba@189
  3367
deba@189
  3368
    /// \brief Start the batch processing
deba@189
  3369
    ///
kpeter@192
  3370
    /// This function starts the batch processing.
deba@189
  3371
    void run() {
alpar@209
  3372
deba@189
  3373
      LEMON_ASSERT(_is != 0, "This reader assigned to an other reader");
alpar@209
  3374
deba@189
  3375
      std::set<std::string> extra_sections;
deba@189
  3376
alpar@209
  3377
      line_num = 0;
deba@189
  3378
      readLine();
deba@189
  3379
      skipSection();
deba@189
  3380
deba@189
  3381
      while (readSuccess()) {
alpar@209
  3382
        try {
alpar@209
  3383
          char c;
alpar@209
  3384
          std::string section, caption;
alpar@209
  3385
          line >> c;
alpar@209
  3386
          _reader_bits::readToken(line, section);
alpar@209
  3387
          _reader_bits::readToken(line, caption);
alpar@209
  3388
alpar@209
  3389
          if (line >> c)
kpeter@291
  3390
            throw FormatError("Extra character at the end of line");
alpar@209
  3391
alpar@209
  3392
          if (extra_sections.find(section) != extra_sections.end()) {
alpar@209
  3393
            std::ostringstream msg;
kpeter@291
  3394
            msg << "Multiple occurence of section: " << section;
deba@290
  3395
            throw FormatError(msg.str());
alpar@209
  3396
          }
alpar@209
  3397
          Sections::iterator it = _sections.find(section);
alpar@209
  3398
          if (it != _sections.end()) {
alpar@209
  3399
            extra_sections.insert(section);
alpar@209
  3400
            it->second->process(*_is, line_num);
alpar@209
  3401
          }
alpar@209
  3402
          readLine();
alpar@209
  3403
          skipSection();
deba@290
  3404
        } catch (FormatError& error) {
alpar@209
  3405
          error.line(line_num);
deba@290
  3406
          error.file(_filename);
alpar@209
  3407
          throw;
alpar@209
  3408
        }
deba@189
  3409
      }
deba@189
  3410
      for (Sections::iterator it = _sections.begin();
alpar@209
  3411
           it != _sections.end(); ++it) {
alpar@209
  3412
        if (extra_sections.find(it->first) == extra_sections.end()) {
alpar@209
  3413
          std::ostringstream os;
alpar@209
  3414
          os << "Cannot find section: " << it->first;
deba@290
  3415
          throw FormatError(os.str());
alpar@209
  3416
        }
deba@189
  3417
      }
deba@189
  3418
    }
deba@189
  3419
deba@189
  3420
    /// @}
alpar@209
  3421
deba@189
  3422
  };
deba@189
  3423
deba@598
  3424
  /// \ingroup lemon_io
deba@598
  3425
  ///
deba@598
  3426
  /// \brief Return a \ref SectionReader class
deba@598
  3427
  ///
deba@598
  3428
  /// This function just returns a \ref SectionReader class.
deba@598
  3429
  ///
deba@598
  3430
  /// Please see SectionReader documentation about the custom section
deba@598
  3431
  /// input.
deba@598
  3432
  ///
deba@598
  3433
  /// \relates SectionReader
deba@598
  3434
  /// \sa sectionReader(const std::string& fn)
deba@598
  3435
  /// \sa sectionReader(const char *fn)
deba@598
  3436
  inline SectionReader sectionReader(std::istream& is) {
deba@598
  3437
    SectionReader tmp(is);
deba@598
  3438
    return tmp;
deba@598
  3439
  }
deba@598
  3440
kpeter@192
  3441
  /// \brief Return a \ref SectionReader class
alpar@209
  3442
  ///
kpeter@192
  3443
  /// This function just returns a \ref SectionReader class.
deba@189
  3444
  /// \relates SectionReader
deba@598
  3445
  /// \sa sectionReader(std::istream& is)
deba@598
  3446
  inline SectionReader sectionReader(const std::string& fn) {
deba@598
  3447
    SectionReader tmp(fn);
deba@189
  3448
    return tmp;
deba@189
  3449
  }
deba@189
  3450
kpeter@192
  3451
  /// \brief Return a \ref SectionReader class
alpar@209
  3452
  ///
kpeter@192
  3453
  /// This function just returns a \ref SectionReader class.
deba@189
  3454
  /// \relates SectionReader
deba@598
  3455
  /// \sa sectionReader(std::istream& is)
deba@189
  3456
  inline SectionReader sectionReader(const char* fn) {
deba@189
  3457
    SectionReader tmp(fn);
deba@189
  3458
    return tmp;
deba@189
  3459
  }
deba@189
  3460
deba@173
  3461
  /// \ingroup lemon_io
deba@173
  3462
  ///
alpar@209
  3463
  /// \brief Reader for the contents of the \ref lgf-format "LGF" file
deba@173
  3464
  ///
deba@173
  3465
  /// This class can be used to read the sections, the map names and
ladanyi@236
  3466
  /// the attributes from a file. Usually, the LEMON programs know
deba@173
  3467
  /// that, which type of graph, which maps and which attributes
deba@173
  3468
  /// should be read from a file, but in general tools (like glemon)
alpar@179
  3469
  /// the contents of an LGF file should be guessed somehow. This class
deba@173
  3470
  /// reads the graph and stores the appropriate information for
deba@173
  3471
  /// reading the graph.
deba@173
  3472
  ///
alpar@209
  3473
  ///\code
alpar@209
  3474
  /// LgfContents contents("graph.lgf");
alpar@179
  3475
  /// contents.run();
deba@173
  3476
  ///
kpeter@192
  3477
  /// // Does it contain any node section and arc section?
alpar@179
  3478
  /// if (contents.nodeSectionNum() == 0 || contents.arcSectionNum()) {
kpeter@192
  3479
  ///   std::cerr << "Failure, cannot find graph." << std::endl;
deba@173
  3480
  ///   return -1;
deba@173
  3481
  /// }
alpar@209
  3482
  /// std::cout << "The name of the default node section: "
alpar@179
  3483
  ///           << contents.nodeSection(0) << std::endl;
alpar@209
  3484
  /// std::cout << "The number of the arc maps: "
alpar@179
  3485
  ///           << contents.arcMaps(0).size() << std::endl;
alpar@209
  3486
  /// std::cout << "The name of second arc map: "
alpar@179
  3487
  ///           << contents.arcMaps(0)[1] << std::endl;
deba@173
  3488
  ///\endcode
alpar@209
  3489
  class LgfContents {
deba@173
  3490
  private:
deba@173
  3491
deba@173
  3492
    std::istream* _is;
deba@173
  3493
    bool local_is;
deba@173
  3494
deba@173
  3495
    std::vector<std::string> _node_sections;
deba@173
  3496
    std::vector<std::string> _edge_sections;
deba@173
  3497
    std::vector<std::string> _attribute_sections;
deba@173
  3498
    std::vector<std::string> _extra_sections;
deba@173
  3499
deba@173
  3500
    std::vector<bool> _arc_sections;
deba@173
  3501
deba@173
  3502
    std::vector<std::vector<std::string> > _node_maps;
deba@173
  3503
    std::vector<std::vector<std::string> > _edge_maps;
deba@173
  3504
deba@173
  3505
    std::vector<std::vector<std::string> > _attributes;
deba@173
  3506
deba@173
  3507
deba@173
  3508
    int line_num;
deba@173
  3509
    std::istringstream line;
alpar@209
  3510
deba@173
  3511
  public:
deba@173
  3512
deba@173
  3513
    /// \brief Constructor
deba@173
  3514
    ///
alpar@179
  3515
    /// Construct an \e LGF contents reader, which reads from the given
deba@173
  3516
    /// input stream.
alpar@209
  3517
    LgfContents(std::istream& is)
deba@173
  3518
      : _is(&is), local_is(false) {}
deba@173
  3519
deba@173
  3520
    /// \brief Constructor
deba@173
  3521
    ///
alpar@179
  3522
    /// Construct an \e LGF contents reader, which reads from the given
deba@173
  3523
    /// file.
alpar@209
  3524
    LgfContents(const std::string& fn)
deba@290
  3525
      : _is(new std::ifstream(fn.c_str())), local_is(true) {
deba@295
  3526
      if (!(*_is)) {
deba@295
  3527
        delete _is;
deba@295
  3528
        throw IoError("Cannot open file", fn);
deba@295
  3529
      }
deba@290
  3530
    }
deba@173
  3531
deba@173
  3532
    /// \brief Constructor
deba@173
  3533
    ///
alpar@179
  3534
    /// Construct an \e LGF contents reader, which reads from the given
deba@173
  3535
    /// file.
alpar@179
  3536
    LgfContents(const char* fn)
deba@290
  3537
      : _is(new std::ifstream(fn)), local_is(true) {
deba@295
  3538
      if (!(*_is)) {
deba@295
  3539
        delete _is;
deba@295
  3540
        throw IoError("Cannot open file", fn);
deba@295
  3541
      }
deba@290
  3542
    }
alpar@209
  3543
deba@173
  3544
    /// \brief Destructor
alpar@179
  3545
    ~LgfContents() {
deba@173
  3546
      if (local_is) delete _is;
deba@173
  3547
    }
deba@173
  3548
deba@190
  3549
  private:
alpar@209
  3550
deba@190
  3551
    LgfContents(const LgfContents&);
deba@190
  3552
    LgfContents& operator=(const LgfContents&);
deba@190
  3553
deba@190
  3554
  public:
deba@190
  3555
deba@173
  3556
kpeter@584
  3557
    /// \name Node Sections
deba@173
  3558
    /// @{
deba@173
  3559
deba@173
  3560
    /// \brief Gives back the number of node sections in the file.
deba@173
  3561
    ///
deba@173
  3562
    /// Gives back the number of node sections in the file.
deba@173
  3563
    int nodeSectionNum() const {
deba@173
  3564
      return _node_sections.size();
deba@173
  3565
    }
deba@173
  3566
alpar@209
  3567
    /// \brief Returns the node section name at the given position.
deba@173
  3568
    ///
alpar@209
  3569
    /// Returns the node section name at the given position.
deba@173
  3570
    const std::string& nodeSection(int i) const {
deba@173
  3571
      return _node_sections[i];
deba@173
  3572
    }
deba@173
  3573
deba@173
  3574
    /// \brief Gives back the node maps for the given section.
deba@173
  3575
    ///
deba@173
  3576
    /// Gives back the node maps for the given section.
alpar@182
  3577
    const std::vector<std::string>& nodeMapNames(int i) const {
deba@173
  3578
      return _node_maps[i];
deba@173
  3579
    }
deba@173
  3580
deba@173
  3581
    /// @}
deba@173
  3582
kpeter@584
  3583
    /// \name Arc/Edge Sections
deba@173
  3584
    /// @{
deba@173
  3585
alpar@181
  3586
    /// \brief Gives back the number of arc/edge sections in the file.
deba@173
  3587
    ///
alpar@181
  3588
    /// Gives back the number of arc/edge sections in the file.
alpar@181
  3589
    /// \note It is synonym of \c edgeSectionNum().
deba@173
  3590
    int arcSectionNum() const {
deba@173
  3591
      return _edge_sections.size();
deba@173
  3592
    }
deba@173
  3593
alpar@209
  3594
    /// \brief Returns the arc/edge section name at the given position.
deba@173
  3595
    ///
alpar@209
  3596
    /// Returns the arc/edge section name at the given position.
alpar@181
  3597
    /// \note It is synonym of \c edgeSection().
deba@173
  3598
    const std::string& arcSection(int i) const {
deba@173
  3599
      return _edge_sections[i];
deba@173
  3600
    }
deba@173
  3601
alpar@181
  3602
    /// \brief Gives back the arc/edge maps for the given section.
deba@173
  3603
    ///
alpar@181
  3604
    /// Gives back the arc/edge maps for the given section.
alpar@182
  3605
    /// \note It is synonym of \c edgeMapNames().
alpar@182
  3606
    const std::vector<std::string>& arcMapNames(int i) const {
deba@173
  3607
      return _edge_maps[i];
deba@173
  3608
    }
deba@173
  3609
deba@173
  3610
    /// @}
deba@173
  3611
alpar@181
  3612
    /// \name Synonyms
deba@173
  3613
    /// @{
deba@173
  3614
alpar@181
  3615
    /// \brief Gives back the number of arc/edge sections in the file.
deba@173
  3616
    ///
alpar@181
  3617
    /// Gives back the number of arc/edge sections in the file.
alpar@181
  3618
    /// \note It is synonym of \c arcSectionNum().
deba@173
  3619
    int edgeSectionNum() const {
deba@173
  3620
      return _edge_sections.size();
deba@173
  3621
    }
deba@173
  3622
alpar@209
  3623
    /// \brief Returns the section name at the given position.
deba@173
  3624
    ///
alpar@209
  3625
    /// Returns the section name at the given position.
alpar@181
  3626
    /// \note It is synonym of \c arcSection().
deba@173
  3627
    const std::string& edgeSection(int i) const {
deba@173
  3628
      return _edge_sections[i];
deba@173
  3629
    }
deba@173
  3630
deba@173
  3631
    /// \brief Gives back the edge maps for the given section.
deba@173
  3632
    ///
deba@173
  3633
    /// Gives back the edge maps for the given section.
alpar@182
  3634
    /// \note It is synonym of \c arcMapNames().
alpar@182
  3635
    const std::vector<std::string>& edgeMapNames(int i) const {
deba@173
  3636
      return _edge_maps[i];
deba@173
  3637
    }
deba@173
  3638
deba@173
  3639
    /// @}
deba@173
  3640
kpeter@584
  3641
    /// \name Attribute Sections
deba@173
  3642
    /// @{
deba@173
  3643
deba@173
  3644
    /// \brief Gives back the number of attribute sections in the file.
deba@173
  3645
    ///
deba@173
  3646
    /// Gives back the number of attribute sections in the file.
deba@173
  3647
    int attributeSectionNum() const {
deba@173
  3648
      return _attribute_sections.size();
deba@173
  3649
    }
deba@173
  3650
alpar@209
  3651
    /// \brief Returns the attribute section name at the given position.
deba@173
  3652
    ///
alpar@209
  3653
    /// Returns the attribute section name at the given position.
alpar@182
  3654
    const std::string& attributeSectionNames(int i) const {
deba@173
  3655
      return _attribute_sections[i];
deba@173
  3656
    }
deba@173
  3657
deba@173
  3658
    /// \brief Gives back the attributes for the given section.
deba@173
  3659
    ///
deba@173
  3660
    /// Gives back the attributes for the given section.
deba@173
  3661
    const std::vector<std::string>& attributes(int i) const {
deba@173
  3662
      return _attributes[i];
deba@173
  3663
    }
deba@173
  3664
deba@173
  3665
    /// @}
deba@173
  3666
kpeter@584
  3667
    /// \name Extra Sections
deba@173
  3668
    /// @{
deba@173
  3669
deba@173
  3670
    /// \brief Gives back the number of extra sections in the file.
deba@173
  3671
    ///
deba@173
  3672
    /// Gives back the number of extra sections in the file.
deba@173
  3673
    int extraSectionNum() const {
deba@173
  3674
      return _extra_sections.size();
deba@173
  3675
    }
deba@173
  3676
alpar@209
  3677
    /// \brief Returns the extra section type at the given position.
deba@173
  3678
    ///
alpar@209
  3679
    /// Returns the section type at the given position.
deba@173
  3680
    const std::string& extraSection(int i) const {
deba@173
  3681
      return _extra_sections[i];
deba@173
  3682
    }
deba@173
  3683
deba@173
  3684
    /// @}
deba@173
  3685
deba@173
  3686
  private:
deba@173
  3687
deba@173
  3688
    bool readLine() {
deba@173
  3689
      std::string str;
deba@173
  3690
      while(++line_num, std::getline(*_is, str)) {
alpar@209
  3691
        line.clear(); line.str(str);
alpar@209
  3692
        char c;
alpar@209
  3693
        if (line >> std::ws >> c && c != '#') {
alpar@209
  3694
          line.putback(c);
alpar@209
  3695
          return true;
alpar@209
  3696
        }
deba@173
  3697
      }
deba@173
  3698
      return false;
deba@173
  3699
    }
deba@173
  3700
deba@173
  3701
    bool readSuccess() {
deba@173
  3702
      return static_cast<bool>(*_is);
deba@173
  3703
    }
deba@173
  3704
deba@173
  3705
    void skipSection() {
deba@173
  3706
      char c;
deba@173
  3707
      while (readSuccess() && line >> c && c != '@') {
alpar@209
  3708
        readLine();
deba@173
  3709
      }
deba@427
  3710
      if (readSuccess()) {
deba@427
  3711
        line.putback(c);
deba@427
  3712
      }
deba@173
  3713
    }
deba@173
  3714
deba@173
  3715
    void readMaps(std::vector<std::string>& maps) {
deba@186
  3716
      char c;
deba@186
  3717
      if (!readLine() || !(line >> c) || c == '@') {
alpar@209
  3718
        if (readSuccess() && line) line.putback(c);
alpar@209
  3719
        return;
deba@186
  3720
      }
deba@186
  3721
      line.putback(c);
deba@173
  3722
      std::string map;
deba@173
  3723
      while (_reader_bits::readToken(line, map)) {
alpar@209
  3724
        maps.push_back(map);
deba@173
  3725
      }
deba@173
  3726
    }
deba@173
  3727
deba@173
  3728
    void readAttributes(std::vector<std::string>& attrs) {
deba@173
  3729
      readLine();
deba@173
  3730
      char c;
deba@173
  3731
      while (readSuccess() && line >> c && c != '@') {
alpar@209
  3732
        line.putback(c);
alpar@209
  3733
        std::string attr;
alpar@209
  3734
        _reader_bits::readToken(line, attr);
alpar@209
  3735
        attrs.push_back(attr);
alpar@209
  3736
        readLine();
deba@173
  3737
      }
deba@173
  3738
      line.putback(c);
deba@173
  3739
    }
deba@173
  3740
deba@173
  3741
  public:
deba@173
  3742
kpeter@584
  3743
    /// \name Execution of the Contents Reader
deba@173
  3744
    /// @{
deba@173
  3745
kpeter@192
  3746
    /// \brief Starts the reading
deba@173
  3747
    ///
kpeter@192
  3748
    /// This function starts the reading.
deba@173
  3749
    void run() {
deba@173
  3750
deba@173
  3751
      readLine();
deba@173
  3752
      skipSection();
deba@173
  3753
deba@173
  3754
      while (readSuccess()) {
deba@173
  3755
alpar@209
  3756
        char c;
alpar@209
  3757
        line >> c;
alpar@209
  3758
alpar@209
  3759
        std::string section, caption;
alpar@209
  3760
        _reader_bits::readToken(line, section);
alpar@209
  3761
        _reader_bits::readToken(line, caption);
alpar@209
  3762
alpar@209
  3763
        if (section == "nodes") {
alpar@209
  3764
          _node_sections.push_back(caption);
alpar@209
  3765
          _node_maps.push_back(std::vector<std::string>());
alpar@209
  3766
          readMaps(_node_maps.back());
alpar@209
  3767
          readLine(); skipSection();
alpar@209
  3768
        } else if (section == "arcs" || section == "edges") {
alpar@209
  3769
          _edge_sections.push_back(caption);
alpar@209
  3770
          _arc_sections.push_back(section == "arcs");
alpar@209
  3771
          _edge_maps.push_back(std::vector<std::string>());
alpar@209
  3772
          readMaps(_edge_maps.back());
alpar@209
  3773
          readLine(); skipSection();
alpar@209
  3774
        } else if (section == "attributes") {
alpar@209
  3775
          _attribute_sections.push_back(caption);
alpar@209
  3776
          _attributes.push_back(std::vector<std::string>());
alpar@209
  3777
          readAttributes(_attributes.back());
alpar@209
  3778
        } else {
alpar@209
  3779
          _extra_sections.push_back(section);
alpar@209
  3780
          readLine(); skipSection();
alpar@209
  3781
        }
deba@173
  3782
      }
deba@173
  3783
    }
deba@173
  3784
deba@173
  3785
    /// @}
alpar@209
  3786
deba@173
  3787
  };
deba@127
  3788
}
deba@127
  3789
deba@127
  3790
#endif