2006-03-31 12:32:02 +02:00
|
|
|
/*****************************************************************************
|
2011-11-14 04:37:50 +01:00
|
|
|
*
|
2006-03-31 12:32:02 +02:00
|
|
|
* This file is part of Mapnik (c++ mapping toolkit)
|
2005-06-14 17:06:59 +02:00
|
|
|
*
|
2017-03-27 17:14:51 +02:00
|
|
|
* Copyright (C) 2016 Artem Pavlenko
|
2005-06-14 17:06:59 +02:00
|
|
|
*
|
2006-03-31 12:32:02 +02:00
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
2005-06-14 17:06:59 +02:00
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
2006-03-31 12:32:02 +02:00
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with this library; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
2005-06-14 17:06:59 +02:00
|
|
|
*
|
2006-03-31 12:32:02 +02:00
|
|
|
*****************************************************************************/
|
2006-10-04 13:22:18 +02:00
|
|
|
// mapnik
|
2017-03-27 17:14:51 +02:00
|
|
|
#include <mapnik/value/types.hpp>
|
2009-07-08 01:56:01 +02:00
|
|
|
#include <mapnik/global.hpp>
|
2015-06-02 12:10:41 +02:00
|
|
|
#include <mapnik/util/utf_conv_win.hpp>
|
2007-02-14 20:54:39 +01:00
|
|
|
#include <mapnik/unicode.hpp>
|
2013-01-04 03:02:21 +01:00
|
|
|
#include <mapnik/util/trim.hpp>
|
2011-04-07 15:44:57 +02:00
|
|
|
|
|
|
|
#include "dbfile.hpp"
|
|
|
|
|
2014-10-22 01:37:27 +02:00
|
|
|
#pragma GCC diagnostic push
|
2015-11-08 02:53:09 +01:00
|
|
|
#include <mapnik/warning_ignore.hpp>
|
2017-03-27 17:14:51 +02:00
|
|
|
#include <boost/spirit/home/x3.hpp>
|
2015-10-16 22:34:53 +02:00
|
|
|
#if defined(MAPNIK_MEMORY_MAPPED_FILE)
|
2013-08-13 20:13:56 +02:00
|
|
|
#include <boost/interprocess/mapped_region.hpp>
|
2011-04-06 15:02:31 +02:00
|
|
|
#include <mapnik/mapped_memory_cache.hpp>
|
2013-08-13 18:19:01 +02:00
|
|
|
#endif
|
2014-10-22 01:37:27 +02:00
|
|
|
#pragma GCC diagnostic pop
|
2013-01-04 03:53:00 +01:00
|
|
|
|
2009-07-08 01:56:01 +02:00
|
|
|
// stl
|
2014-05-06 19:06:47 +02:00
|
|
|
#include <cstdint>
|
2009-07-08 01:56:01 +02:00
|
|
|
#include <string>
|
2014-01-29 00:05:10 +01:00
|
|
|
#include <cstring>
|
2013-06-03 05:19:33 +02:00
|
|
|
#include <stdexcept>
|
2009-07-08 01:56:01 +02:00
|
|
|
|
2005-06-14 17:06:59 +02:00
|
|
|
dbf_file::dbf_file()
|
2010-09-02 22:20:51 +02:00
|
|
|
: num_records_(0),
|
|
|
|
num_fields_(0),
|
|
|
|
record_length_(0),
|
|
|
|
record_(0) {}
|
2007-02-14 20:54:39 +01:00
|
|
|
|
2016-02-18 17:37:46 +01:00
|
|
|
dbf_file::dbf_file(std::string const& file_name)
|
2010-09-02 22:20:51 +02:00
|
|
|
:num_records_(0),
|
|
|
|
num_fields_(0),
|
|
|
|
record_length_(0),
|
2015-10-16 22:34:53 +02:00
|
|
|
#if defined(MAPNIK_MEMORY_MAPPED_FILE)
|
2011-04-06 15:02:31 +02:00
|
|
|
file_(),
|
2013-05-21 18:42:55 +02:00
|
|
|
#elif defined(_WINDOWS)
|
|
|
|
file_(mapnik::utf8_to_utf16(file_name), std::ios::in | std::ios::binary),
|
2009-07-09 01:54:45 +02:00
|
|
|
#else
|
2011-04-07 15:35:21 +02:00
|
|
|
file_(file_name.c_str() ,std::ios::in | std::ios::binary),
|
2009-07-09 01:54:45 +02:00
|
|
|
#endif
|
2010-09-02 22:20:51 +02:00
|
|
|
record_(0)
|
2005-06-14 17:06:59 +02:00
|
|
|
{
|
2011-04-06 15:02:31 +02:00
|
|
|
|
2015-10-16 22:34:53 +02:00
|
|
|
#if defined(MAPNIK_MEMORY_MAPPED_FILE)
|
2016-02-18 17:37:46 +01:00
|
|
|
boost::optional<mapnik::mapped_region_ptr> memory = mapnik::mapped_memory_cache::instance().find(file_name,true);
|
2011-04-06 15:02:31 +02:00
|
|
|
if (memory)
|
|
|
|
{
|
2013-05-30 23:32:20 +02:00
|
|
|
mapped_region_ = *memory;
|
2011-04-06 15:02:31 +02:00
|
|
|
file_.buffer(static_cast<char*>((*memory)->get_address()),(*memory)->get_size());
|
|
|
|
}
|
2013-05-30 23:32:20 +02:00
|
|
|
else
|
|
|
|
{
|
|
|
|
throw std::runtime_error("could not create file mapping for "+file_name);
|
|
|
|
}
|
2011-11-14 04:37:50 +01:00
|
|
|
#endif
|
2011-04-06 15:02:31 +02:00
|
|
|
if (file_)
|
2010-09-02 22:20:51 +02:00
|
|
|
{
|
|
|
|
read_header();
|
|
|
|
}
|
2005-06-14 17:06:59 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
dbf_file::~dbf_file()
|
|
|
|
{
|
2010-09-02 22:20:51 +02:00
|
|
|
::operator delete(record_);
|
2005-06-14 17:06:59 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool dbf_file::is_open()
|
|
|
|
{
|
2015-10-16 22:34:53 +02:00
|
|
|
#if defined(MAPNIK_MEMORY_MAPPED_FILE)
|
2011-11-14 04:37:50 +01:00
|
|
|
return (file_.buffer().second > 0);
|
2011-04-06 15:02:31 +02:00
|
|
|
#else
|
2011-11-14 04:37:50 +01:00
|
|
|
return file_.is_open();
|
2011-04-06 15:02:31 +02:00
|
|
|
#endif
|
2005-06-14 17:06:59 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int dbf_file::num_records() const
|
|
|
|
{
|
2010-09-02 22:20:51 +02:00
|
|
|
return num_records_;
|
2005-06-14 17:06:59 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int dbf_file::num_fields() const
|
|
|
|
{
|
2010-09-02 22:20:51 +02:00
|
|
|
return num_fields_;
|
2005-06-14 17:06:59 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void dbf_file::move_to(int index)
|
|
|
|
{
|
2010-09-02 22:20:51 +02:00
|
|
|
if (index>0 && index<=num_records_)
|
|
|
|
{
|
2011-04-07 15:35:21 +02:00
|
|
|
std::streampos pos=(num_fields_<<5)+34+(index-1)*(record_length_+1);
|
2010-09-02 22:20:51 +02:00
|
|
|
file_.seekg(pos,std::ios::beg);
|
|
|
|
file_.read(record_,record_length_);
|
|
|
|
}
|
2005-06-14 17:06:59 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
std::string dbf_file::string_value(int col) const
|
|
|
|
{
|
2010-09-02 22:20:51 +02:00
|
|
|
if (col>=0 && col<num_fields_)
|
|
|
|
{
|
|
|
|
return std::string(record_+fields_[col].offset_,fields_[col].length_);
|
|
|
|
}
|
|
|
|
return "";
|
2005-06-14 17:06:59 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const field_descriptor& dbf_file::descriptor(int col) const
|
|
|
|
{
|
2010-09-02 22:20:51 +02:00
|
|
|
assert(col>=0 && col<num_fields_);
|
|
|
|
return fields_[col];
|
2005-06-14 17:06:59 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-03-27 17:14:51 +02:00
|
|
|
void dbf_file::add_attribute(int col, mapnik::transcoder const& tr, mapnik::feature_impl & f) const
|
2005-06-14 17:06:59 +02:00
|
|
|
{
|
2010-09-02 22:20:51 +02:00
|
|
|
using namespace boost::spirit;
|
|
|
|
|
|
|
|
if (col>=0 && col<num_fields_)
|
|
|
|
{
|
2012-03-13 11:11:58 +01:00
|
|
|
std::string const& name=fields_[col].name_;
|
2011-11-14 04:37:50 +01:00
|
|
|
|
2013-08-08 00:58:01 +02:00
|
|
|
// NOTE: ensure types handled here are matched in shape_datasource.cpp
|
2010-09-02 22:20:51 +02:00
|
|
|
switch (fields_[col].type_)
|
|
|
|
{
|
|
|
|
case 'C':
|
2012-12-03 06:37:39 +01:00
|
|
|
case 'D':
|
2010-09-02 22:20:51 +02:00
|
|
|
{
|
2011-04-07 15:35:21 +02:00
|
|
|
// FIXME - avoid constructing std::string on stack
|
2010-09-02 22:20:51 +02:00
|
|
|
std::string str(record_+fields_[col].offset_,fields_[col].length_);
|
2013-01-04 03:02:21 +01:00
|
|
|
mapnik::util::trim(str);
|
2012-01-12 11:04:08 +01:00
|
|
|
f.put(name,tr.transcode(str.c_str()));
|
2006-03-22 15:52:32 +01:00
|
|
|
break;
|
2010-09-02 22:20:51 +02:00
|
|
|
}
|
2012-11-29 18:19:11 +01:00
|
|
|
case 'L':
|
|
|
|
{
|
|
|
|
char ch = record_[fields_[col].offset_];
|
|
|
|
if ( ch == '1' || ch == 't' || ch == 'T' || ch == 'y' || ch == 'Y')
|
|
|
|
{
|
|
|
|
f.put(name,true);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2012-12-03 06:37:39 +01:00
|
|
|
// NOTE: null logical fields use '?'
|
2012-11-29 18:19:11 +01:00
|
|
|
f.put(name,false);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2013-08-08 00:58:01 +02:00
|
|
|
case 'N': // numeric
|
|
|
|
case 'O': // double
|
|
|
|
case 'F': // float
|
2010-09-02 22:20:51 +02:00
|
|
|
{
|
2013-08-08 00:39:25 +02:00
|
|
|
|
2010-09-02 22:20:51 +02:00
|
|
|
if (record_[fields_[col].offset_] == '*')
|
2006-03-22 15:52:32 +01:00
|
|
|
{
|
2013-04-12 01:05:29 +02:00
|
|
|
// NOTE: we intentionally do not store null here
|
|
|
|
// since it is equivalent to the attribute not existing
|
2010-09-02 22:20:51 +02:00
|
|
|
break;
|
2006-03-22 15:52:32 +01:00
|
|
|
}
|
2007-02-14 20:54:39 +01:00
|
|
|
if ( fields_[col].dec_>0 )
|
2011-11-14 04:37:50 +01:00
|
|
|
{
|
2010-09-02 22:20:51 +02:00
|
|
|
double val = 0.0;
|
2010-09-06 20:37:02 +02:00
|
|
|
const char *itr = record_+fields_[col].offset_;
|
|
|
|
const char *end = itr + fields_[col].length_;
|
2017-03-27 17:14:51 +02:00
|
|
|
x3::ascii::space_type space;
|
|
|
|
static x3::double_type double_;
|
|
|
|
if (x3::phrase_parse(itr,end,double_,space,val))
|
2014-01-26 23:49:03 +01:00
|
|
|
{
|
2012-04-08 02:56:58 +02:00
|
|
|
f.put(name,val);
|
2014-01-26 23:49:03 +01:00
|
|
|
}
|
2007-02-14 20:54:39 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2012-12-18 19:05:45 +01:00
|
|
|
mapnik::value_integer val = 0;
|
2010-09-06 20:37:02 +02:00
|
|
|
const char *itr = record_+fields_[col].offset_;
|
|
|
|
const char *end = itr + fields_[col].length_;
|
2017-03-27 17:14:51 +02:00
|
|
|
x3::ascii::space_type space;
|
|
|
|
static x3::int_parser<mapnik::value_integer,10,1,-1> numeric_parser;
|
|
|
|
if (x3::phrase_parse(itr, end, numeric_parser, space, val))
|
2014-01-26 23:49:03 +01:00
|
|
|
{
|
2012-04-08 02:56:58 +02:00
|
|
|
f.put(name,val);
|
2014-01-26 23:49:03 +01:00
|
|
|
}
|
2007-02-14 20:54:39 +01:00
|
|
|
}
|
|
|
|
break;
|
2010-09-02 22:20:51 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2005-06-14 17:06:59 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void dbf_file::read_header()
|
|
|
|
{
|
2010-09-02 22:20:51 +02:00
|
|
|
char c=file_.get();
|
|
|
|
if (c=='\3' || c=='\131')
|
|
|
|
{
|
|
|
|
skip(3);
|
|
|
|
num_records_=read_int();
|
|
|
|
assert(num_records_>=0);
|
|
|
|
num_fields_=read_short();
|
|
|
|
assert(num_fields_>0);
|
|
|
|
num_fields_=(num_fields_-33)/32;
|
|
|
|
skip(22);
|
2011-04-07 15:35:21 +02:00
|
|
|
std::streampos offset=0;
|
2010-09-02 22:20:51 +02:00
|
|
|
char name[11];
|
2014-01-29 00:05:10 +01:00
|
|
|
std::memset(&name,0,11);
|
2010-09-02 22:20:51 +02:00
|
|
|
fields_.reserve(num_fields_);
|
|
|
|
for (int i=0;i<num_fields_;++i)
|
|
|
|
{
|
|
|
|
field_descriptor desc;
|
|
|
|
desc.index_=i;
|
|
|
|
file_.read(name,10);
|
2013-01-04 03:02:21 +01:00
|
|
|
desc.name_=name;
|
|
|
|
// TODO - when is this trim needed?
|
|
|
|
mapnik::util::trim(desc.name_);
|
2010-09-02 22:20:51 +02:00
|
|
|
skip(1);
|
|
|
|
desc.type_=file_.get();
|
|
|
|
skip(4);
|
|
|
|
desc.length_=file_.get();
|
|
|
|
desc.dec_=file_.get();
|
|
|
|
skip(14);
|
|
|
|
desc.offset_=offset;
|
|
|
|
offset+=desc.length_;
|
|
|
|
fields_.push_back(desc);
|
|
|
|
}
|
|
|
|
record_length_=offset;
|
|
|
|
if (record_length_>0)
|
|
|
|
{
|
|
|
|
record_=static_cast<char*>(::operator new (sizeof(char)*record_length_));
|
|
|
|
}
|
|
|
|
}
|
2005-06-14 17:06:59 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int dbf_file::read_short()
|
|
|
|
{
|
2010-09-02 22:20:51 +02:00
|
|
|
char b[2];
|
|
|
|
file_.read(b,2);
|
2014-05-06 19:06:47 +02:00
|
|
|
std::int16_t val;
|
2010-09-02 22:20:51 +02:00
|
|
|
mapnik::read_int16_ndr(b,val);
|
|
|
|
return val;
|
2005-06-14 17:06:59 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int dbf_file::read_int()
|
2011-11-14 04:37:50 +01:00
|
|
|
{
|
2010-09-02 22:20:51 +02:00
|
|
|
char b[4];
|
|
|
|
file_.read(b,4);
|
2014-05-06 19:06:47 +02:00
|
|
|
std::int32_t val;
|
2010-09-02 22:20:51 +02:00
|
|
|
mapnik::read_int32_ndr(b,val);
|
|
|
|
return val;
|
2005-06-14 17:06:59 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void dbf_file::skip(int bytes)
|
|
|
|
{
|
2010-09-02 22:20:51 +02:00
|
|
|
file_.seekg(bytes,std::ios::cur);
|
2005-06-14 17:06:59 +02:00
|
|
|
}
|