You've already forked mariadb-columnstore-engine
							
							
				mirror of
				https://github.com/mariadb-corporation/mariadb-columnstore-engine.git
				synced 2025-11-03 17:13:17 +03:00 
			
		
		
		
	
		
			
				
	
	
		
			485 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			485 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/* Copyright (C) 2014 InfiniDB, Inc.
 | 
						|
 | 
						|
   This program is free software; you can redistribute it and/or
 | 
						|
   modify it under the terms of the GNU General Public License
 | 
						|
   as published by the Free Software Foundation; version 2 of
 | 
						|
   the License.
 | 
						|
 | 
						|
   This program is distributed in the hope that it will be useful,
 | 
						|
   but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
						|
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | 
						|
   GNU General Public License for more details.
 | 
						|
 | 
						|
   You should have received a copy of the GNU General Public License
 | 
						|
   along with this program; if not, write to the Free Software
 | 
						|
   Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
 | 
						|
   MA 02110-1301, USA. */
 | 
						|
 | 
						|
/*******************************************************************************
 | 
						|
 *   $Id: we_xmlgenproc.cpp 4639 2013-05-08 17:32:10Z dcathey $
 | 
						|
 *
 | 
						|
 ******************************************************************************/
 | 
						|
#define WRITEENGINEXMLGENPROC_DLLEXPORT
 | 
						|
#include "we_xmlgenproc.h"
 | 
						|
#undef WRITEENGINEXMLGENPROC_DLLEXPORT
 | 
						|
 | 
						|
#include <sstream>
 | 
						|
#include <unistd.h>
 | 
						|
 | 
						|
#include "we_config.h"
 | 
						|
#include "we_xmltag.h"
 | 
						|
#include "we_xmlgendata.h"
 | 
						|
#include "mcsconfig.h"
 | 
						|
 | 
						|
#include <boost/date_time/posix_time/posix_time.hpp>
 | 
						|
#include <boost/filesystem/path.hpp>
 | 
						|
using namespace execplan;
 | 
						|
 | 
						|
namespace
 | 
						|
{
 | 
						|
const char* DICT_TYPE("D");
 | 
						|
const char* ENCODING("UTF-8");
 | 
						|
const char* LOGNAME("Jobxml_");
 | 
						|
const char* JOBNAME("Job_");
 | 
						|
}  // namespace
 | 
						|
 | 
						|
namespace WriteEngine
 | 
						|
{
 | 
						|
//------------------------------------------------------------------------------
 | 
						|
// XMLGen constructor
 | 
						|
//------------------------------------------------------------------------------
 | 
						|
XMLGenProc::XMLGenProc(XMLGenData* mgr, bool bUseXmlLogFile, bool bSysCatRpt)
 | 
						|
 : fLog()
 | 
						|
 , fDoc()
 | 
						|
 , fWriter()
 | 
						|
 , fErrorString(
 | 
						|
       "XMLGen encountered exception, abnormal exit "
 | 
						|
       "and file not created.\nCheck error log at:\t")
 | 
						|
 , fDebugLevel(0)
 | 
						|
 , fInputMgr(mgr)
 | 
						|
 , fSysCatRpt(bSysCatRpt)
 | 
						|
 , fUseXmlLogFile(bUseXmlLogFile)
 | 
						|
{
 | 
						|
  std::string logFile(std::string(MCSLOGDIR) + "/cpimport/" + LOGNAME +
 | 
						|
                      fInputMgr->getParm(XMLGenData::JOBID) + ".log");
 | 
						|
  std::string errFile(std::string(MCSLOGDIR) + "/cpimport/" + LOGNAME +
 | 
						|
                      fInputMgr->getParm(XMLGenData::JOBID) + ".err");
 | 
						|
  fErrorString.append(errFile + "\n");
 | 
						|
 | 
						|
  if (fUseXmlLogFile)
 | 
						|
  {
 | 
						|
    fLog.setLogFileName(logFile.c_str(), errFile.c_str());
 | 
						|
    std::ostringstream ss;
 | 
						|
    fInputMgr->print(ss);
 | 
						|
    fLog.logMsg(ss.str(), MSGLVL_INFO1);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
//------------------------------------------------------------------------------
 | 
						|
// XMLGen destructor
 | 
						|
//------------------------------------------------------------------------------
 | 
						|
XMLGenProc::~XMLGenProc()
 | 
						|
{
 | 
						|
  xmlFreeDoc(fDoc);
 | 
						|
}
 | 
						|
 | 
						|
//------------------------------------------------------------------------------
 | 
						|
// startXMLFile
 | 
						|
// Creates xmlDocPtr (fDoc) that we will use in generating our XML file.
 | 
						|
// Create XML header section.
 | 
						|
//------------------------------------------------------------------------------
 | 
						|
void XMLGenProc::startXMLFile()
 | 
						|
{
 | 
						|
  fWriter = xmlNewTextWriterDoc(&fDoc, 0);
 | 
						|
 | 
						|
  if (fWriter == NULL)
 | 
						|
  {
 | 
						|
    throw std::runtime_error(
 | 
						|
        "Error creating the xml fWriter: "
 | 
						|
        "bad return from xmlNewTextWriter");
 | 
						|
  }
 | 
						|
 | 
						|
  /* Start the fDocument with the xml default for the version,
 | 
						|
   * encoding UTF-8 and the default for the standalone
 | 
						|
   * declaration. */
 | 
						|
  int rc = xmlTextWriterStartDocument(fWriter, NULL, ENCODING, NULL);
 | 
						|
 | 
						|
  if (rc < 0)
 | 
						|
  {
 | 
						|
    throw std::runtime_error(
 | 
						|
        "Error at xmlTextWriterStartfDocument: "
 | 
						|
        "bad return from xmlTextWriterStartDocument");
 | 
						|
  }
 | 
						|
 | 
						|
  if (!fSysCatRpt)  // skip non-syscat tags if we are writing a syscat dump
 | 
						|
  {
 | 
						|
    xmlTextWriterStartElement(fWriter, BAD_CAST xmlTagTable[TAG_BULK_JOB]);
 | 
						|
    xmlTextWriterWriteFormatElement(fWriter, BAD_CAST xmlTagTable[TAG_ID], "%d",
 | 
						|
                                    atoi(fInputMgr->getParm(XMLGenData::JOBID).c_str()));
 | 
						|
    xmlTextWriterWriteElement(fWriter, BAD_CAST xmlTagTable[TAG_NAME],
 | 
						|
                              BAD_CAST fInputMgr->getParm(XMLGenData::NAME).c_str());
 | 
						|
    xmlTextWriterWriteElement(fWriter, BAD_CAST xmlTagTable[TAG_DESC],
 | 
						|
                              BAD_CAST fInputMgr->getParm(XMLGenData::DESCRIPTION).c_str());
 | 
						|
 | 
						|
    std::string now(boost::posix_time::to_iso_string(boost::posix_time::second_clock::local_time()));
 | 
						|
    xmlTextWriterWriteElement(fWriter, BAD_CAST xmlTagTable[TAG_CREATE_DATE],
 | 
						|
                              BAD_CAST now.substr(0, 8).c_str());
 | 
						|
    xmlTextWriterWriteElement(fWriter, BAD_CAST xmlTagTable[TAG_CREATE_TIME],
 | 
						|
                              BAD_CAST now.substr(9, 4).c_str());
 | 
						|
    xmlTextWriterWriteElement(fWriter, BAD_CAST xmlTagTable[TAG_USER],
 | 
						|
                              BAD_CAST fInputMgr->getParm(XMLGenData::USER).c_str());
 | 
						|
    xmlTextWriterWriteElement(fWriter, BAD_CAST xmlTagTable[TAG_DELIMITER],
 | 
						|
                              BAD_CAST fInputMgr->getParm(XMLGenData::DELIMITER).c_str());
 | 
						|
 | 
						|
    // Only include enclosedBy and escape chars if enclosedBy was specified
 | 
						|
    std::string enclosedByChar = fInputMgr->getParm(XMLGenData::ENCLOSED_BY_CHAR);
 | 
						|
 | 
						|
    if (enclosedByChar.length() > 0)
 | 
						|
    {
 | 
						|
      xmlTextWriterWriteElement(fWriter, BAD_CAST xmlTagTable[TAG_ENCLOSED_BY_CHAR],
 | 
						|
                                BAD_CAST fInputMgr->getParm(XMLGenData::ENCLOSED_BY_CHAR).c_str());
 | 
						|
    }
 | 
						|
 | 
						|
    // Include escape character regardless of whether the "enclosed by"
 | 
						|
    // character is given, because the escape character can still be used
 | 
						|
    // to override the default NULL escape sequence '\N', to be something
 | 
						|
    // else like '#N'.
 | 
						|
    xmlTextWriterWriteElement(fWriter, BAD_CAST xmlTagTable[TAG_ESCAPE_CHAR],
 | 
						|
                              BAD_CAST fInputMgr->getParm(XMLGenData::ESCAPE_CHAR).c_str());
 | 
						|
 | 
						|
    if (auto skipRows = fInputMgr->getParm(XMLGenData::SKIP_ROWS); !skipRows.empty())
 | 
						|
    {
 | 
						|
      xmlTextWriterWriteElement(fWriter, BAD_CAST xmlTagTable[TAG_SKIP_ROWS], BAD_CAST skipRows.c_str());
 | 
						|
    }
 | 
						|
 | 
						|
    // Added new tags for configurable parameters
 | 
						|
    xmlTextWriterStartElement(fWriter, BAD_CAST xmlTagTable[TAG_READ_BUFFERS]);
 | 
						|
    xmlTextWriterWriteFormatAttribute(fWriter, BAD_CAST xmlTagTable[TAG_NO_OF_READ_BUFFERS], "%d",
 | 
						|
                                      atoi(fInputMgr->getParm(XMLGenData::NO_OF_READ_BUFFER).c_str()));
 | 
						|
    xmlTextWriterWriteFormatAttribute(fWriter, BAD_CAST xmlTagTable[TAG_READ_BUFFER_SIZE], "%d",
 | 
						|
                                      atoi(fInputMgr->getParm(XMLGenData::READ_BUFFER_CAPACITY).c_str()));
 | 
						|
    xmlTextWriterEndElement(fWriter);
 | 
						|
    xmlTextWriterWriteFormatElement(fWriter, BAD_CAST xmlTagTable[TAG_WRITE_BUFFER_SIZE], "%d",
 | 
						|
                                    atoi(fInputMgr->getParm(XMLGenData::WRITE_BUFFER_SIZE).c_str()));
 | 
						|
    // End of additions
 | 
						|
  }
 | 
						|
 | 
						|
  xmlTextWriterStartElement(fWriter, BAD_CAST xmlTagTable[TAG_SCHEMA]);
 | 
						|
  xmlTextWriterWriteAttribute(fWriter, BAD_CAST xmlTagTable[TAG_NAME],
 | 
						|
                              BAD_CAST fInputMgr->getSchema().c_str());
 | 
						|
}
 | 
						|
 | 
						|
//------------------------------------------------------------------------------
 | 
						|
// makeTableData
 | 
						|
// Create XML tag for a table.
 | 
						|
//------------------------------------------------------------------------------
 | 
						|
// This method is used by colxml only and it can be relatively slower doing tableRID()
 | 
						|
// first call. All subsequent calls will re-use data from CalpontSystemCatalog cache.
 | 
						|
void XMLGenProc::makeTableData(const CalpontSystemCatalog::TableName& table)
 | 
						|
{
 | 
						|
  boost::shared_ptr<CalpontSystemCatalog> cat =
 | 
						|
      CalpontSystemCatalog::makeCalpontSystemCatalog(BULK_SYSCAT_SESSION_ID);
 | 
						|
  cat->identity(CalpontSystemCatalog::EC);
 | 
						|
  std::ostringstream oss;
 | 
						|
  // tableRID method might take a lot with a significant EM.
 | 
						|
  oss << cat->tableRID(table).objnum;
 | 
						|
 | 
						|
  makeTableData(table, oss.str());
 | 
						|
}
 | 
						|
 | 
						|
void XMLGenProc::makeTableData(const CalpontSystemCatalog::TableName& table, const std::string& tableOIDStr)
 | 
						|
{
 | 
						|
  static unsigned kount;
 | 
						|
 | 
						|
  xmlTextWriterStartElement(fWriter, BAD_CAST xmlTagTable[TAG_TABLE]);
 | 
						|
  std::string tmp(table.schema + "." + table.table);
 | 
						|
  xmlTextWriterWriteAttribute(fWriter, BAD_CAST xmlTagTable[TAG_TBL_NAME], BAD_CAST tmp.c_str());
 | 
						|
 | 
						|
  if (fSysCatRpt)  // Write full schema information for syscat rpt
 | 
						|
  {
 | 
						|
    try
 | 
						|
    {
 | 
						|
      xmlTextWriterWriteFormatAttribute(fWriter, BAD_CAST xmlTagTable[TAG_TBL_OID], "%s",
 | 
						|
                                        tableOIDStr.c_str());
 | 
						|
    }
 | 
						|
    catch (std::exception& ex)
 | 
						|
    {
 | 
						|
      std::ostringstream oss;
 | 
						|
      oss << "Error getting OID for table " << table.schema << '.' << table.table << ": " << ex.what();
 | 
						|
      throw std::runtime_error(oss.str());
 | 
						|
    }
 | 
						|
    catch (...)
 | 
						|
    {
 | 
						|
      std::ostringstream oss;
 | 
						|
      oss << "Unknown error getting OID for table " << table.schema << '.' << table.table;
 | 
						|
      throw std::runtime_error(oss.str());
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  if (!fSysCatRpt)  // skip non-syscat tags if we are writing a syscat dump
 | 
						|
  {
 | 
						|
    const XMLGenData::LoadNames& loadNames = fInputMgr->getLoadNames();
 | 
						|
 | 
						|
    if (loadNames.size() > kount)
 | 
						|
    {
 | 
						|
      tmp = loadNames[kount];
 | 
						|
    }
 | 
						|
    else
 | 
						|
    {
 | 
						|
      tmp = (table.table + "." + fInputMgr->getParm(XMLGenData::EXT).c_str());
 | 
						|
    }
 | 
						|
 | 
						|
    xmlTextWriterWriteAttribute(fWriter, BAD_CAST xmlTagTable[TAG_LOAD_NAME], BAD_CAST tmp.c_str());
 | 
						|
    auto sMaxErrors = fInputMgr->getParm(XMLGenData::MAXERROR);
 | 
						|
    if (sMaxErrors == "all")
 | 
						|
    {
 | 
						|
      xmlTextWriterWriteAttribute(fWriter, BAD_CAST xmlTagTable[TAG_MAX_ERR_ROW], BAD_CAST sMaxErrors.c_str());
 | 
						|
    }
 | 
						|
    else
 | 
						|
    {
 | 
						|
      xmlTextWriterWriteFormatAttribute(fWriter, BAD_CAST xmlTagTable[TAG_MAX_ERR_ROW], "%d",
 | 
						|
                                        atoi(sMaxErrors.c_str()));
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  kount++;
 | 
						|
}
 | 
						|
 | 
						|
//------------------------------------------------------------------------------
 | 
						|
// sortColumnsByPosition
 | 
						|
// Sort list of columns by column position.
 | 
						|
//------------------------------------------------------------------------------
 | 
						|
void XMLGenProc::sortColumnsByPosition(SysCatColumnList& columns)
 | 
						|
{
 | 
						|
  std::map<int, SysCatColumn> tempCols;
 | 
						|
 | 
						|
  SysCatColumnList::const_iterator cend = columns.end();
 | 
						|
 | 
						|
  for (SysCatColumnList::const_iterator col = columns.begin(); col != cend; ++col)
 | 
						|
  {
 | 
						|
    tempCols[col->colType.colPosition] = *col;
 | 
						|
  }
 | 
						|
 | 
						|
  columns.clear();
 | 
						|
 | 
						|
  std::map<int, SysCatColumn>::iterator pos;
 | 
						|
 | 
						|
  for (pos = tempCols.begin(); pos != tempCols.end(); ++pos)
 | 
						|
  {
 | 
						|
    columns.push_back(pos->second);
 | 
						|
  }
 | 
						|
 | 
						|
  tempCols.clear();
 | 
						|
}
 | 
						|
 | 
						|
//------------------------------------------------------------------------------
 | 
						|
// makeColumnData
 | 
						|
// Create XML tag for the columns in a table.
 | 
						|
//------------------------------------------------------------------------------
 | 
						|
bool XMLGenProc::makeColumnData(const CalpontSystemCatalog::TableName& table)
 | 
						|
{
 | 
						|
  SysCatColumnList columns;
 | 
						|
  getColumnsForTable(table.schema, table.table, columns);
 | 
						|
  sortColumnsByPosition(columns);
 | 
						|
 | 
						|
  if (columns.empty())
 | 
						|
  {
 | 
						|
    if (fUseXmlLogFile)
 | 
						|
    {
 | 
						|
      fLog.logMsg("No columns for " + table.table + ", or table does not exist", MSGLVL_ERROR);
 | 
						|
    }
 | 
						|
 | 
						|
    return false;
 | 
						|
  }
 | 
						|
 | 
						|
  SysCatColumnList::const_iterator cend = columns.end();
 | 
						|
 | 
						|
  for (SysCatColumnList::const_iterator col = columns.begin(); col != cend; ++col)
 | 
						|
  {
 | 
						|
    xmlTextWriterStartElement(fWriter, BAD_CAST xmlTagTable[TAG_COLUMN]);
 | 
						|
    xmlTextWriterWriteAttribute(fWriter, BAD_CAST xmlTagTable[TAG_COL_NAME],
 | 
						|
                                BAD_CAST col->tableColName.column.c_str());
 | 
						|
 | 
						|
    if (fSysCatRpt)  // Write full schema information for syscat rpt
 | 
						|
    {
 | 
						|
      xmlTextWriterWriteFormatAttribute(fWriter, BAD_CAST xmlTagTable[TAG_COL_OID], "%d", col->oid);
 | 
						|
      xmlTextWriterWriteAttribute(fWriter, BAD_CAST xmlTagTable[TAG_DATA_TYPE],
 | 
						|
                                  BAD_CAST ColDataTypeStr[col->colType.colDataType]);
 | 
						|
 | 
						|
      if (col->colType.compressionType != CalpontSystemCatalog::NO_COMPRESSION)
 | 
						|
        xmlTextWriterWriteFormatAttribute(fWriter, BAD_CAST xmlTagTable[TAG_COMPRESS_TYPE], "%d",
 | 
						|
                                          col->colType.compressionType);
 | 
						|
 | 
						|
      // Old logic went by scale > 0; New logic checks for "decimal" type
 | 
						|
      if ((0 < col->colType.scale) || (col->colType.colDataType == CalpontSystemCatalog::DECIMAL) ||
 | 
						|
          (col->colType.colDataType == CalpontSystemCatalog::UDECIMAL))
 | 
						|
      {
 | 
						|
        xmlTextWriterWriteFormatAttribute(fWriter, BAD_CAST xmlTagTable[TAG_PRECISION], "%d",
 | 
						|
                                          col->colType.precision);
 | 
						|
        xmlTextWriterWriteFormatAttribute(fWriter, BAD_CAST xmlTagTable[TAG_SCALE], "%d", col->colType.scale);
 | 
						|
      }
 | 
						|
 | 
						|
      xmlTextWriterWriteFormatAttribute(fWriter, BAD_CAST xmlTagTable[TAG_WIDTH], "%d",
 | 
						|
                                        col->colType.colWidth);
 | 
						|
 | 
						|
      if (col->colType.autoincrement)
 | 
						|
      {
 | 
						|
        int autoInc = 1;
 | 
						|
        xmlTextWriterWriteFormatAttribute(fWriter, BAD_CAST xmlTagTable[TAG_AUTOINCREMENT_FLAG], "%d",
 | 
						|
                                          autoInc);
 | 
						|
      }
 | 
						|
 | 
						|
      // need dictionary and decimal stuff
 | 
						|
      if (col->colType.ddn.dictOID > 0)
 | 
						|
      {
 | 
						|
        xmlTextWriterWriteAttribute(fWriter, BAD_CAST xmlTagTable[TAG_COL_TYPE], BAD_CAST DICT_TYPE);
 | 
						|
        xmlTextWriterWriteFormatAttribute(fWriter, BAD_CAST xmlTagTable[TAG_DVAL_OID], "%d",
 | 
						|
                                          col->colType.ddn.dictOID);
 | 
						|
      }
 | 
						|
 | 
						|
      // Include NotNull and Default value
 | 
						|
      const NullString col_defaultValue(col->colType.defaultValue);
 | 
						|
 | 
						|
      if (col->colType.constraintType == execplan::CalpontSystemCatalog::NOTNULL_CONSTRAINT)
 | 
						|
      {
 | 
						|
        int notNull = 1;
 | 
						|
        xmlTextWriterWriteFormatAttribute(fWriter, BAD_CAST xmlTagTable[TAG_NOT_NULL], "%d", notNull);
 | 
						|
 | 
						|
        if (!col_defaultValue.isNull())
 | 
						|
        {
 | 
						|
          xmlTextWriterWriteAttribute(fWriter, BAD_CAST xmlTagTable[TAG_DEFAULT_VALUE],
 | 
						|
                                      BAD_CAST col_defaultValue.unsafeStringRef().c_str());
 | 
						|
        }
 | 
						|
      }
 | 
						|
      else if (col->colType.constraintType == execplan::CalpontSystemCatalog::DEFAULT_CONSTRAINT)
 | 
						|
      {
 | 
						|
        idbassert(!col_defaultValue.isNull()); // good enough for now. I have to figure out how to store the null.
 | 
						|
        xmlTextWriterWriteAttribute(fWriter, BAD_CAST xmlTagTable[TAG_DEFAULT_VALUE],
 | 
						|
                                    BAD_CAST (col_defaultValue.unsafeStringRef().c_str()));
 | 
						|
      }
 | 
						|
    }  // end of "if fSysCatRpt"
 | 
						|
 | 
						|
    xmlTextWriterEndElement(fWriter);
 | 
						|
  }
 | 
						|
 | 
						|
  xmlTextWriterEndElement(fWriter);  // table
 | 
						|
  return true;
 | 
						|
}
 | 
						|
 | 
						|
//------------------------------------------------------------------------------
 | 
						|
// getColumnsForTable
 | 
						|
// Access the system catalog in order to get the OID, column type, and column
 | 
						|
// name for all the columns in the specified schema and table.
 | 
						|
//
 | 
						|
// This function is modeled after the function by the same name in DDLPackage-
 | 
						|
// Processor.  XMLGen used to use that DDLPackageProcessor function, but I
 | 
						|
// decided to implement the functionality in XMLGen in order to eliminate the
 | 
						|
// dependency on ddlpackageprocessor.
 | 
						|
//------------------------------------------------------------------------------
 | 
						|
void XMLGenProc::getColumnsForTable(const std::string& schema, const std::string& table,
 | 
						|
                                    SysCatColumnList& colList)
 | 
						|
{
 | 
						|
  CalpontSystemCatalog::TableName tableName;
 | 
						|
  tableName.schema = schema;
 | 
						|
  tableName.table = table;
 | 
						|
 | 
						|
  try
 | 
						|
  {
 | 
						|
    boost::shared_ptr<CalpontSystemCatalog> systemCatalogPtr =
 | 
						|
        CalpontSystemCatalog::makeCalpontSystemCatalog(BULK_SYSCAT_SESSION_ID);
 | 
						|
    systemCatalogPtr->identity(CalpontSystemCatalog::EC);
 | 
						|
 | 
						|
    const CalpontSystemCatalog::RIDList ridList = systemCatalogPtr->columnRIDs(tableName, true);
 | 
						|
 | 
						|
    CalpontSystemCatalog::RIDList::const_iterator rid_iterator = ridList.begin();
 | 
						|
 | 
						|
    while (rid_iterator != ridList.end())
 | 
						|
    {
 | 
						|
      CalpontSystemCatalog::ROPair roPair = *rid_iterator;
 | 
						|
 | 
						|
      SysCatColumn column;
 | 
						|
      column.oid = roPair.objnum;
 | 
						|
      column.colType = systemCatalogPtr->colType(column.oid);
 | 
						|
      column.tableColName = systemCatalogPtr->colName(column.oid);
 | 
						|
 | 
						|
      colList.push_back(column);
 | 
						|
 | 
						|
      ++rid_iterator;
 | 
						|
    }
 | 
						|
  }
 | 
						|
  catch (std::exception& ex)
 | 
						|
  {
 | 
						|
    std::ostringstream oss;
 | 
						|
    oss << "Error reading columns for table " << schema << '.' << table << ": " << ex.what();
 | 
						|
    throw std::runtime_error(oss.str());
 | 
						|
  }
 | 
						|
  catch (...)
 | 
						|
  {
 | 
						|
    std::ostringstream oss;
 | 
						|
    oss << "Unknown error reading columns for table " << schema << '.' << table;
 | 
						|
    throw std::runtime_error(oss.str());
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
//------------------------------------------------------------------------------
 | 
						|
// Generate Job XML File Name
 | 
						|
//------------------------------------------------------------------------------
 | 
						|
 | 
						|
std::string XMLGenProc::genJobXMLFileName() const
 | 
						|
{
 | 
						|
  std::string xmlFileName;
 | 
						|
  boost::filesystem::path p(std::string(fInputMgr->getParm(XMLGenData::PATH)));
 | 
						|
 | 
						|
  // Append the jobname, jobid & file extension
 | 
						|
  std::string fileName(JOBNAME);
 | 
						|
  fileName += fInputMgr->getParm(XMLGenData::JOBID);
 | 
						|
  fileName += ".xml";
 | 
						|
  p /= fileName;
 | 
						|
 | 
						|
  // If the filespec doesn't begin with a '/' (i.e. it's not an absolute path),
 | 
						|
  // attempt to make it absolute so that we can log the full pathname.
 | 
						|
 | 
						|
  if (!p.has_root_path())
 | 
						|
  {
 | 
						|
    char cwdPath[4096];
 | 
						|
    char* buf;
 | 
						|
    buf = getcwd(cwdPath, sizeof(cwdPath));
 | 
						|
    if (buf == NULL)
 | 
						|
      throw std::runtime_error("Failed to get the current working directory!");
 | 
						|
    boost::filesystem::path p2(cwdPath);
 | 
						|
    p2 /= p;
 | 
						|
    xmlFileName = p2.string();
 | 
						|
  }
 | 
						|
  else
 | 
						|
  {
 | 
						|
    xmlFileName = p.string();
 | 
						|
  }
 | 
						|
 | 
						|
 | 
						|
  return xmlFileName;
 | 
						|
}
 | 
						|
 | 
						|
//------------------------------------------------------------------------------
 | 
						|
// writeXMLFile
 | 
						|
//------------------------------------------------------------------------------
 | 
						|
void XMLGenProc::writeXMLFile(const std::string& xmlFileName)
 | 
						|
{
 | 
						|
  xmlTextWriterEndDocument(fWriter);
 | 
						|
  xmlFreeTextWriter(fWriter);
 | 
						|
 | 
						|
  xmlSaveFormatFile(xmlFileName.c_str(), fDoc, 1);
 | 
						|
}
 | 
						|
 | 
						|
//------------------------------------------------------------------------------
 | 
						|
// logErrorMessage
 | 
						|
//------------------------------------------------------------------------------
 | 
						|
void XMLGenProc::logErrorMessage(const std::string& msg)
 | 
						|
{
 | 
						|
  if (fUseXmlLogFile)
 | 
						|
  {
 | 
						|
    fLog.logMsg(msg, MSGLVL_ERROR);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
}  // namespace WriteEngine
 |