// Ryzom - MMORPG Framework // Copyright (C) 2010 Winch Gate Property Limited // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU Affero General Public License as // published by the Free Software Foundation, either version 3 of the // License, or (at your option) any later version. // // 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 Affero General Public License for more details. // // You should have received a copy of the GNU Affero General Public License // along with this program. If not, see . #include "../pd_lib/pds_common.h" #include #include #include #include #include "pds_table.h" #include "pds_database.h" //#include "db_file_stream.h" #include "../pd_lib/db_delta_file.h" #include "../pd_lib/db_description_parser.h" using namespace std; using namespace NLMISC; /// Aligne a value on a boundary (if boundary is not a power of 2, rounded to next power of 2 boundary) inline uint32 alignOnBoundary(uint32 value, uint32 boundary) { // round to power of 2 boundary boundary = raiseToNextPowerOf2(boundary); // align return (value+boundary-1)&(~(boundary-1)); } /* * Constructor */ CTable::CTable() { clear(); } /* * Destructor */ CTable::~CTable() { //PDS_DEBUG("delete()"); clear(); } /* * Clear table */ void CTable::clear() { _Init = false; uint i; // clear table buffer _TableBuffer.clear(); // delete columns _Columns.clear(); // delete attributes for (i=0; i<_Attributes.size(); ++i) { delete _Attributes[i]; _Attributes[i] = NULL; } _Attributes.clear(); _Parent = NULL; _Name = ""; _Id = INVALID_TYPE_ID; _Key = INVALID_TYPE_ID; _Mapped = false; _RowSize = 0; _EmptyRow.clear(); } /* * Init table */ bool CTable::init(CDatabase *database, const CTableNode& table) { // first a good clean up clear(); // set parent logger setParentLogger(database); _TableBuffer.setParentLogger(this); _Parent = database; _Name = table.Name; _Id = table.Id; _Inheritance = table.Inherit; _Mapped = (table.Mapped != -1); _Key = table.Key; uint i; for (i=0; iinit(database, this, table.Attributes[i]) || attribute->getId() != i) return false; _Attributes.push_back(attribute); } return true; } /* * Build columns */ bool CTable::buildColumns() { // do not compute twice if (!_Columns.empty()) return true; uint i; // build columns from attributes for (i=0; i<_Attributes.size(); ++i) if (!_Attributes[i]->buildColumns()) return false; uint32 columnStart = 0; for (i=0; i<_Columns.size(); ++i) { _Columns[i].setByteOffset(columnStart); columnStart += _Columns[i].getByteSize(); } _RowSize = columnStart; _TableBuffer.init(_Id, _RowSize, _Mapped); _EmptyRow.clear(); _EmptyRow.resize(_RowSize, 0); for (i=0; i<_Columns.size(); ++i) { TDataType type = _Columns[i].getDataType(); void* data = &_EmptyRow[_Columns[i].getByteOffset()]; if (type == PDS_Index) { *(RY_PDS::CObjectIndex*)data = RY_PDS::CObjectIndex::null(); } else if (type == PDS_List) { memset(data, 0, _Columns[i].getByteSize()); } } _Init = true; return true; } /* * Post Init, called once all tables have been initialised */ bool CTable::postInit() { CTable* root = getRootTable(); if (root != this) _TableBuffer.linkRowMapper(&(root->_TableBuffer)); uint i; for (i=0; i<_Attributes.size(); ++i) if (!_Attributes[i]->computeBackRefKey()) return false; return true; } /* * Display table */ void CTable::display(NLMISC::CLog* log, bool expanded, bool displayHeader) const { if (!initialised()) { log->displayNL("** Table not initialised"); return; } if (displayHeader) { log->displayNL("--------------------------------------------------------------------------------"); log->displayNL("%-10s %-3s %-32s | %-3s | %-3s | %-4s | %-8s %-8s %8s |", "", "Id", "Name", "Inh", "Att", "Cols", "Allocs", "Loaded", "LoadedSz"); } log->displayNL("%-10s %-3d %-32s | %-3d | %-3d | %-4d | %-8d %-8d %6dkb |", "Table", _Id, _Name.c_str(), _Inheritance, _Attributes.size(), _Columns.size(), _TableBuffer.numAllocated(), _TableBuffer.getLoadedRows(), (_TableBuffer.getMemoryLoad()+1023)/1024); uint i; if (expanded) { log->displayNL("--------------------------------------------------------------------------------"); log->displayNL("%-10s %-3s %-32s | %-10s | %-4s %-4s |", "", "Id", "Name", "MetaType", "From", "NCol"); for (i=0; i<_Attributes.size(); ++i) { const CAttribute* attrib = _Attributes[i]; if (attrib == NULL || !attrib->initialised()) log->displayNL("** Attribute %d not initialised", i); else log->displayNL("%-10s %-3d %-32s | %-10s | %-4d %-4d |", "Attribute", attrib->getId(), attrib->getName().c_str(), getNameFromMetaType(attrib->getMetaType()).c_str(), attrib->getOffset(), attrib->getColumns()); } log->displayNL("--------------------------------------------------------------------------------"); log->displayNL("%-10s %-3s %-64s | %-10s %-10s | %-10s", "", "Id", "Name", "MetaType", "DataType", "RowSz/Offs"); for (i=0; i<_Columns.size(); ++i) { const CColumn& column = _Columns[i]; if (!column.initialised()) log->displayNL("** Column %d not initialised", i); else log->displayNL("%-10s %-3d %-64s | %-10s %-10s | %1db at %-3d", "Column", column.getId(), column.getName().c_str(), getNameFromMetaType(column.getMetaType()).c_str(), getNameFromDataType(column.getDataType()).c_str(), column.getByteSize(), column.getByteOffset()); } } } /* * Display row */ void CTable::displayRow(RY_PDS::TRowIndex row, NLMISC::CLog* log, bool displayHeader) { if (!initialised()) { log->displayNL("** Table not initialised"); return; } CTableBuffer::CAccessor rowaccess = _TableBuffer.getRow(row); string flagstr = toString("%s%s%s", (rowaccess.allocated() ? "allocated" : "free"), (rowaccess.mapped() ? ", mapped" : ""), (rowaccess.dirty() ? ", dirty" : "")); log->displayNL("row %d: %d bytes, flags=[%s] (map=%016" NL_I64 "X, dirtstamp=%08X)", row, _RowSize, flagstr.c_str(), (rowaccess.mapped() ? rowaccess.key() : (uint64)0), rowaccess.dirtyStamp()); if (displayHeader) { log->displayNL("%-10s %-3s %-64s | %-10s %-10s | %-9s | %-32s", "", "Id", "Name", "MetaType", "DataType", "Sz/Offs", "Value"); } uint i; for (i=0; i<_Columns.size(); ++i) { if (!_Columns[i].initialised()) { log->displayNL("** Column %d not initialised", i); continue; } string value; CDataAccessor accessor(this, rowaccess, i); const CColumn &col = _Columns[i]; if (!accessor.isValid()) value = "unaccessible value"; else value = accessor.valueAsString(); log->displayNL("%-10s %-3d %-64s | %-10s %-10s | %1db at %-3d | %-32s", "Column", col.getId(), col.getName().c_str(), getNameFromMetaType(col.getMetaType()).c_str(), getNameFromDataType(col.getDataType()).c_str(), col.getByteSize(), col.getByteOffset(), value.c_str()); } _TableBuffer.releaseRow(rowaccess); } /* * Dump Delta file content */ void CTable::dumpDeltaFileContent(const std::string& filename, NLMISC::CLog* log) const { #ifdef DEBUG_DATA_ACCESSOR if (!initialised()) { PDS_WARNING("dumpDeltaFileContent(): table not initialised"); return; } CDBDeltaFile delta; if (!_TableBuffer.setupDebugDeltaFile(filename, delta)) { return; } uint8* data; uint32 row; log->displayNL("%-10s %-3s %-64s | %-10s %-10s | %-9s | %-32s", "", "Id", "Name", "MetaType", "DataType", "Sz/Offs", "Value"); while ((data = _TableBuffer.getDeltaRow(row, delta)) != NULL) { log->displayNL("row %d", row); CTable* table = const_cast(this); CDataAccessor accessor(table, data, 0); uint i; for (i=0; i<_Columns.size(); ++i) { if (!_Columns[i].initialised()) { log->displayNL("** Column %d not initialised", i); continue; } string value; CDataAccessor accessor(accessor, i); const CColumn &col = _Columns[i]; if (!accessor.isValid()) value = "unaccessible value"; else value = accessor.valueAsString(); log->displayNL("%-10s %-3d %-64s | %-10s %-10s | %1db at %-3d | %-32s", "Column", col.getId(), col.getName().c_str(), CType::getNameFromMetaType(col.getMetaType()).c_str(), CType::getNameFromDataType(col.getDataType()).c_str(), col.getByteSize(), col.getByteOffset(), value.c_str()); } } #endif } /* * Rebuild forwardrefs from backrefs */ bool CTable::rebuildForwardRefs() { if (!initialised()) { PDS_WARNING("rebuildForwardRefs(): table not initialised"); return false; } if (!fillRefInfo()) { PDS_WARNING("rebuildForwardRefs(): failed to fillRefInfo()"); return false; } if (BackRefInfo.empty() && ForwardRefInfo.empty()) return true; if (!_TableBuffer.processRows(this)) { PDS_WARNING("rebuildForwardRefs(): failed to processRows()"); return false; } return true; } /* * Reset forwardrefs */ bool CTable::resetForwardRefs() { if (!initialised()) { PDS_WARNING("resetForwardRefs(): table not initialised"); return false; } return true; } /* * Reset table map */ bool CTable::resetTableMap() { if (!initialised()) { PDS_WARNING("resetTableMap(): table not initialised"); return false; } return true; } /* * Rebuild table map */ bool CTable::rebuildTableMap() { if (!initialised()) { PDS_WARNING("rebuildTableMap(): table not initialised"); return false; } if (!_Mapped) return true; return true; } /* * Allocate a row in a table * \param row is the row to allocate * Return true if succeded */ bool CTable::allocate(RY_PDS::TRowIndex row, bool acquireRow) { CTableBuffer::CAccessor accessor; if (!_TableBuffer.allocate(row, accessor)) return false; if (!accessor.allocated()) return false; // should not happen resetRow(accessor.data()); // lock row if required if (acquireRow) _TableBuffer.acquireRow(accessor); return true; } /* * Deallocate a row in a table * \param row is the row to deallocate * Return true if succeded */ bool CTable::deallocate(RY_PDS::TRowIndex row) { return _TableBuffer.deallocate(row); } /* * Map a row * \param row is the row to allocate * \param key is the 64 bits row key * Return true if succeded */ bool CTable::mapRow(const RY_PDS::CObjectIndex &index, uint64 key) { if (!initialised()) { PDS_WARNING("mapRow(): failed, table is not initialised"); return false; } if (!_Mapped) { PDS_WARNING("mapRow(): failed, table is not mapped"); return false; } if (!_TableBuffer.mapRow(index, key)) { PDS_WARNING("mapRow(): failed to map '%s' to '%016" NL_I64 "X'", index.toString().c_str(), key); return false; } PDS_FULL_DEBUG("Mapped '%s' to key '%016" NL_I64 "X'", index.toString().c_str(), key); return true; } /* * Unmap a row in a table * \param tableIndex is the table to find row * \param key is the 64 bits row key * Return true if succeded */ bool CTable::unmapRow(uint64 key) { if (!initialised()) { PDS_WARNING("unmapRow(): failed, table is not initialised"); return false; } if (!_Mapped) { PDS_WARNING("unmapRow(): failed, table is not mapped"); return false; } // get index RY_PDS::CObjectIndex index = _TableBuffer.getMappedRow(key); if (index.isNull()) { PDS_WARNING("unmapRow(): failed, inherited table is not initialised"); return false; } if (!_TableBuffer.unmapRow(index, key)) { PDS_WARNING("mapRow(): failed to unmap '%s' to '%016" NL_I64 "X'", index.toString().c_str(), key); return false; } PDS_FULL_DEBUG("Unmapped '%s' of key '%016" NL_I64 "X'", index.toString().c_str(), key); return true; } /* * Get a mapped row * \param key is the 64 bits row key * Return a valid TRowIndex if success */ RY_PDS::CObjectIndex CTable::getMappedRow(uint64 key) const { if (!initialised()) { PDS_WARNING("getMappedRow(): failed, table is not initialised"); return RY_PDS::CObjectIndex::null(); } return _TableBuffer.getMappedRow(key); } /* * Release a row in a table * \param row is the row to release * \param timestamp is the current timestamp * Return true if succeded */ bool CTable::release(RY_PDS::TRowIndex row) { if (!initialised()) { PDS_WARNING("release(): failed, table is not initialised"); return false; } return _TableBuffer.releaseRow(row); } /* * Release all rows in table */ bool CTable::releaseAll() { return _TableBuffer.releaseAll(); } /* * Set value */ bool CTable::set(RY_PDS::TRowIndex row, RY_PDS::TColumnIndex column, uint datasize, const void* dataptr) { RY_PDS::CObjectIndex object((RY_PDS::TTableIndex)_Id, row); // get an accessor on data CDataAccessor accessor(_Parent, object, column); if (!accessor.isValid()) { PDS_WARNING("set(): failed to get accessor on '%s'", accessor.getColumnIndex().toString().c_str()); return false; } switch (accessor.column()->getMetaType()) { case PDS_Type: // for simple type, easy money return accessor.setValue(dataptr, datasize); break; case PDS_BackRef: { // first unlink previous parent if (!unlink(accessor, object)) { PDS_WARNING("set(): unable to unlink previous parent at '%s'", accessor.getColumnIndex().toString().c_str()); } // then link new parent if (datasize != getStandardByteSize(PDS_Index)) { PDS_WARNING("set(): unable to link new at '%s', provided bytesize is not standard", accessor.getColumnIndex().toString().c_str()); return false; } // get parent index RY_PDS::CObjectIndex parent = *(RY_PDS::CObjectIndex*)dataptr; // check checksum is valid if (!parent.isChecksumValid()) { PDS_WARNING("set(): unable to link new parent '%s' at '%s', parent checksum is invalid", parent.toString().c_str(), accessor.getColumnIndex().toString().c_str()); return false; } // if parent is invalid, then no need to link if (!parent.isValid()) return true; // and link! if (!link(accessor, parent, object)) { PDS_WARNING("set(): unable to link new parent '%s' at '%s'", parent.toString().c_str(), accessor.getColumnIndex().toString().c_str()); return false; } } break; // other types MUST not be directly set default: PDS_WARNING("set(): column '%d' doesn't support set on '%s'", column, getNameFromMetaType(accessor.column()->getMetaType()).c_str()); return false; break; } return true; } /** * Set Parent */ bool CTable::setParent(RY_PDS::TRowIndex row, RY_PDS::TColumnIndex column, const RY_PDS::CObjectIndex& parent) { RY_PDS::CObjectIndex object((RY_PDS::TTableIndex)_Id, row); // get an accessor on data CDataAccessor accessor(_Parent, object, column); if (!accessor.isValid()) { PDS_WARNING("set(): failed to get accessor on '%s'", accessor.getColumnIndex().toString().c_str()); return false; } switch (accessor.column()->getMetaType()) { case PDS_BackRef: { // first unlink previous parent if (!unlink(accessor, object)) { PDS_WARNING("set(): unable to unlink previous parent at '%s'", accessor.getColumnIndex().toString().c_str()); } // check checksum is valid if (!parent.isChecksumValid()) { PDS_WARNING("set(): unable to link new parent '%s' at '%s', parent checksum is invalid", parent.toString().c_str(), accessor.getColumnIndex().toString().c_str()); return false; } // if parent is invalid, then no need to link if (!parent.isValid()) return true; // and link! if (!link(accessor, parent, object)) { PDS_WARNING("set(): unable to link new parent '%s' at '%s'", parent.toString().c_str(), accessor.getColumnIndex().toString().c_str()); return false; } } break; // other types MUST not be directly set default: PDS_WARNING("setParent(): column '%d' doesn't support set on '%s'", column, getNameFromMetaType(accessor.column()->getMetaType()).c_str()); return false; break; } return true; } /** * link a BackRef to a parent * This method will perfom a full linking of child and new parent * \param backref is an accessor on the BackRef * \param parent is an index on the parent to link */ bool CTable::link(CDataAccessor &backref, const RY_PDS::CObjectIndex &parent, const RY_PDS::CObjectIndex &child) { // check ref is valid if (!backref.isValid()) { PDS_WARNING("link(): failed, accessor is not valid"); return false; } // check really a BackRef if (backref.attribute()->getMetaType() != PDS_BackRef || backref.column()->getMetaType() != PDS_BackRef) { PDS_WARNING("link(): failed, accessor is not a BackRef"); return false; } if (!parent.isValid()) { PDS_WARNING("link(): failed, parent has not a valid index"); return false; } // link parent to child uint32 forwardrefid = backref.attribute()->getReferencedAttribute(); CDataAccessor parentref(_Parent, parent, forwardrefid, 0); if (!parentref.isValid()) { PDS_WARNING("link(): failed, parent '%s' is not valid", parentref.toString().c_str()); return false; } // check parent and child can really be linked // that is cross references match if (parentref.attribute()->getReferencedAttribute() != backref.attribute()->getId() || backref.attribute()->getReferencedAttribute() != parentref.attribute()->getId()) { PDS_WARNING("link(): failed, child '%s' and parent '%s' are not bound to be linked", backref.toString().c_str(), parentref.toString().c_str()); return false; } forwardLink(backref, parentref, child); // set no ref index, with checksum validation return backref.setIndex(parent); } /* * link a ForwardRef to a child * This method will only link parent to child * \param forwardref is an accessor on the BackRef * \param child is an index on the child to link */ bool CTable::forwardLink(CDataAccessor &backref, CDataAccessor &forwardref, const RY_PDS::CObjectIndex &child) { if (!backref.isValid() || !forwardref.isValid()) { PDS_WARNING("forwardLink(): failed, accessor is not valid"); return false; } if (!child.isValid()) { PDS_WARNING("forwardLink(): failed, child '%s' index is invalid", child.toString().c_str()); return false; } switch (forwardref.attribute()->getMetaType()) { case PDS_ArrayRef: // ArrayRef and ForwardRef set code is a bit shared as ArrayRef will only do // a little seek to the good column and set the index just like a ForwardRef // actually, datatypes are the same! if (!forwardref.seek(backref)) { PDS_WARNING("forwardLink(): unable to seek to key of '%s'", child.toString().c_str()); return false; } case PDS_ForwardRef: { RY_PDS::CObjectIndex prevChild; // check no previous child if (!forwardref.getIndex(prevChild)) { PDS_WARNING("forwardLink(): unable to access previous child of '%s'", forwardref.toString().c_str()); } else if (prevChild.isValid()) { PDS_WARNING("forwardLink(): '%s' has a previous child '%s'", forwardref.toString().c_str(), prevChild.toString().c_str()); } // set child return forwardref.setIndex(child); } break; case PDS_Set: { if (!forwardref.checkType(child)) { PDS_WARNING("forwardLink(): failed, '%s' is not of attribute '%s' type", child.toString().c_str(), forwardref.attribute()->getName().c_str()); return false; } RY_PDS::CSetMap::CAccessor setaccess = forwardref.getSet(); if (!setaccess.isValid()) { PDS_WARNING("forwardLink(): failed, parent '%s' set access is invalid", forwardref.toString().c_str()); return false; } setaccess.add(child); // CHECK if (!setaccess.belongsTo(child)) { PDS_WARNING("forwardLink(): failed, child '%s' doesn't belong to parent '%s' though it just had been added", child.toString().c_str(), forwardref.toString().c_str()); return false; } // CHECK return true; } break; default: PDS_WARNING("forwardLink(): failed, can't link '%s'", getNameFromMetaType(forwardref.attribute()->getMetaType()).c_str()); break; } return false; } /* * Unlink a BackRef * \param ref is an accessor on the BackRef * \param child is a remember of the child index */ bool CTable::unlink(CDataAccessor &backref, const RY_PDS::CObjectIndex &child) { // check ref is valid if (!backref.isValid()) { PDS_WARNING("unlink(): failed, accessor is not valid"); return false; } // check really a BackRef if (backref.attribute()->getMetaType() != PDS_BackRef || backref.column()->getMetaType() != PDS_BackRef) { PDS_WARNING("unlink(): failed, accessor is not a BackRef"); return false; } RY_PDS::CObjectIndex parent; // unlink parent if (!backref.getIndex(parent)) { PDS_WARNING("unlink(): unable to get parent index"); } else if (!parent.isChecksumValid()) { PDS_WARNING("unlink(): parent index has invalid checksum"); } else if (parent.isValid()) { // only unlink if parent exists // because link will perform unlink before, and child may have no parent yet uint32 forwardrefid = backref.attribute()->getReferencedAttribute(); CDataAccessor parentref(_Parent, parent, forwardrefid, 0); forwardUnlink(backref, parentref, child); } // set no ref index, with checksum validation return backref.setIndex(RY_PDS::CObjectIndex::null()); } /* * unlink a ForwardRef to a child * This method will only unlink parent to child * \param forwardref is an accessor on the BackRef * \param child is an index on the child to unlink */ bool CTable::forwardUnlink(CDataAccessor &backref, CDataAccessor &forwardref, const RY_PDS::CObjectIndex &child) { if (!forwardref.isValid()) { PDS_WARNING("forwardUnlink(): failed, accessor is not valid"); return false; } if (!child.isValid()) { PDS_WARNING("forwardUnlink(): failed, child '%s' index is invalid", child.toString().c_str()); return false; } switch (forwardref.attribute()->getMetaType()) { case PDS_ArrayRef: // ArrayRef and ForwardRef set code is a bit shared as ArrayRef will only do // a little seek to the good column and set the index just like a ForwardRef // actually, datatypes are the same! if (!forwardref.seek(backref)) { PDS_WARNING("forwardLink(): unable to seek to key of '%s'", child.toString().c_str()); return false; } case PDS_ForwardRef: { RY_PDS::CObjectIndex prevChild; // check child is the one we want to unlink if (!forwardref.getIndex(prevChild)) { PDS_WARNING("forwardUnlink(): unable to access previous child of '%s'", forwardref.toString().c_str()); } else if (!prevChild.isValid()) { PDS_WARNING("forwardUnlink(): '%s' has no previous child '%s'", prevChild.toString().c_str(), forwardref.toString().c_str()); } else if (prevChild != child) { PDS_WARNING("forwardUnlink(): failed, '%s' was not linked to '%s' but to '%s'", forwardref.toString().c_str(), child.toString().c_str(), prevChild.toString().c_str()); return false; } // set child return forwardref.setIndex(RY_PDS::CObjectIndex::null()); } break; case PDS_Set: { RY_PDS::CSetMap::CAccessor setaccess = forwardref.getSet(); if (!setaccess.isValid()) { PDS_WARNING("forwardUnlink(): failed, parent '%s' set access is invalid", forwardref.toString().c_str()); return false; } if (!setaccess.belongsTo(child)) { PDS_WARNING("forwardUnlink(): failed, child '%s' doesn't belong to parent '%s'", child.toString().c_str(), forwardref.toString().c_str()); return false; } setaccess.erase(child); if (setaccess.belongsTo(child)) { PDS_WARNING("forwardUnlink(): failed, child '%s' still belongs to parent '%s' though it had been deleted", child.toString().c_str(), forwardref.toString().c_str()); return false; } return true; } break; default: PDS_WARNING("forwardUnlink(): failed, can't unlink '%s'", getNameFromMetaType(forwardref.attribute()->getMetaType()).c_str()); break; } return false; } /* * Set value */ bool CTable::get(RY_PDS::TRowIndex row, RY_PDS::TColumnIndex column, uint& datasize, void* dataptr, TDataType& type) { RY_PDS::CObjectIndex object((RY_PDS::TTableIndex)_Id, row); // get an accessor on data CDataAccessor accessor(_Parent, object, column); if (!accessor.isValid()) { PDS_WARNING("get(): failed to get accessor on '%s'", accessor.getColumnIndex().toString().c_str()); return false; } // get datatype type = accessor.column()->getDataType(); switch (accessor.column()->getMetaType()) { case PDS_Type: // for simple type, easy money return accessor.getValue(dataptr, datasize); break; case PDS_BackRef: case PDS_ForwardRef: { if (getStandardByteSize(type) > datasize) { PDS_WARNING("get(): databuffer too narrow to store column '%d' Index", column); return false; } datasize = sizeof(RY_PDS::CObjectIndex); return accessor.getIndex(*(RY_PDS::CObjectIndex*)dataptr); } break; case PDS_Set: { PDS_WARNING("get(): not supported for sets"); return false; } break; // other types MUST not be directly set default: PDS_WARNING("get(): column '%d' doesn't support get on '%s'", column, getNameFromMetaType(accessor.column()->getMetaType()).c_str()); return false; break; } return true; } /* * Perform column integrity check */ bool CTable::CDataAccessor::check() const { if (!isValid()) return false; #ifdef DEBUG_DATA_ACCESSOR if (_IsDebug) return true; #endif switch (_Column->getMetaType()) { case PDS_Type: return checkAsTypeAccessor(); break; case PDS_BackRef: if (_Attribute->getMetaType() != PDS_BackRef) { PDS_WARNING_IN(_Table)("CDataAccessor::check(): column '%s' MetaType incoherent with attribute's definition", _Column->getName().c_str()); return false; } return checkAsRefAccessor(); break; case PDS_ForwardRef: if (_Attribute->getMetaType() != PDS_ForwardRef && _Attribute->getMetaType() != PDS_ArrayRef) { PDS_WARNING_IN(_Table)("CDataAccessor::check(): column '%s' MetaType incoherent with attribute's definition", _Column->getName().c_str()); return false; } return checkAsRefAccessor(); break; case PDS_Set: return checkAsSetAccessor(); break; } return false; } /* * Check an accessor as a PDS_Type accessor */ bool CTable::CDataAccessor::checkAsTypeAccessor() const { // check attribute if (_Attribute->getMetaType() != PDS_Type && _Attribute->getMetaType() != PDS_Class && _Attribute->getMetaType() != PDS_ArrayType && _Attribute->getMetaType() != PDS_ArrayClass) { PDS_WARNING_IN(_Table)("CDataAccessor::checkAsTypeAccessor(): column '%s' MetaType incoherent with attribute's definition", _Column->getName().c_str()); return false; } if (!checkStrictDataType(_Column->getDataType())) { PDS_WARNING_IN(_Table)("CDataAccessor::checkAsTypeAccessor(): column '%s' has not a strict valid datatype ('%s')", _Column->getName().c_str(), getNameFromDataType(_Column->getDataType()).c_str()); return false; } if (_Column->getDataType() == PDS_enum || _Column->getDataType() == PDS_dimension) { const CType* index = _Table->getParent()->getType(_Column->getTypeId()); if (index == NULL) { PDS_WARNING_IN(_Table)("CDataAccessor::checkAsTypeAccessor(): couldn't find column '%s' original type in database", _Column->getName().c_str()); return false; } if (!index->isIndex()) { PDS_WARNING_IN(_Table)("CDataAccessor::checkAsTypeAccessor(): column '%s' original type is not an enum, whereas column says so", _Column->getName().c_str()); return false; } uint32 value; if (!getAsIndexType(value)) { PDS_WARNING_IN(_Table)("CDataAccessor::checkAsTypeAccessor(): couldn't getAsIndexValue() column '%s'", _Column->getName().c_str()); return false; } if (value >= index->getIndexSize()) { PDS_WARNING_IN(_Table)("CDataAccessor::checkAsTypeAccessor(): column '%s' is out of enum '%s' range", _Column->getName().c_str(), index->getName().c_str()); return false; } } return true; } /* * Check an accessor as a PDS_BackRef or PDS_ForwardRef accessor */ bool CTable::CDataAccessor::checkAsRefAccessor() const { if (_Column->getDataType() != PDS_Index) { PDS_WARNING_IN(_Table)("CDataAccessor::checkAsRefAccessor(): column '%s' DataType must be Index, found '%s'", _Column->getName().c_str(), getNameFromDataType(_Column->getDataType()).c_str()); return false; } RY_PDS::CObjectIndex ref; if (!getIndex(ref)) { PDS_WARNING_IN(_Table)("CDataAccessor::checkAsRefAccessor(): can't get column '%s' Index", _Column->getName().c_str()); return false; } if (!checkType(ref)) { PDS_WARNING_IN(_Table)("CDataAccessor::checkAsRefAccessor(): column '%s' contains object '%s' of invalid type", _Column->getName().c_str(), ref.toString().c_str()); return false; } if (!ref.isNull() && !_Table->getParent()->isAllocated(ref)) { PDS_WARNING_IN(_Table)("CDataAccessor::checkAsRefAccessor(): column '%s' points to '%s' which is not allocated", _Column->getName().c_str(), ref.toString().c_str()); return false; } return true; } /* * Check an accessor as a PDS_Set accessor */ bool CTable::CDataAccessor::checkAsSetAccessor() const { if (_Column->getDataType() != PDS_List) { PDS_WARNING_IN(_Table)("CDataAccessor::checkAsSetAccessor(): column '%s' DataType must be List, found '%s'", _Column->getName().c_str(), getNameFromDataType(_Column->getDataType()).c_str()); return false; } RY_PDS::CSetMap::CAccessor setaccess = getSet(); if (!setaccess.isValid()) { PDS_WARNING_IN(_Table)("CDataAccessor::checkAsSetAccessor(): set accessor to '%s' is invalid", toString().c_str()); return false; } const RY_PDS::TIndexList& list = setaccess.get(); bool success = true; uint i; for (i=0; igetName().c_str(), index.toString().c_str()); success = false; continue; } if (!_Table->getParent()->isAllocated(index)) { PDS_WARNING_IN(_Table)("CDataAccessor::checkAsSetAccessor(): column '%s' points to unallocated object '%s'", _Column->getName().c_str(), index.toString().c_str()); success = false; continue; } if (!checkType(index)) { PDS_WARNING_IN(_Table)("CDataAccessor::checkAsSetAccessor(): column '%s' points to object '%s' which is not of expected type", _Column->getName().c_str(), index.toString().c_str()); success = false; continue; } } return success; } /* * Fetch a row into stream * Fetch the whole object arborescence (i.e. children linked to this object) * \param row is the row to fetch * \param data is the stream store data into */ bool CTable::fetch(RY_PDS::TRowIndex row, RY_PDS::CPData &data, bool fetchIndex) { uint i; RY_PDS::TTableIndex table = (RY_PDS::TTableIndex)_Id; RY_PDS::CObjectIndex index(table, row); if (fetchIndex) { data.serial(table, row); } CDataAccessor rowaccessor(_Parent, index, 0); for (i=0; i<_Columns.size(); ++i) { if (!_Columns[i].initialised()) { PDS_WARNING("fetch(): failed, column '%d' not initialised", i); return false; } if (_Columns[i].getMetaType() == PDS_BackRef) { // backref are not sent, implicitely deduced continue; } CDataAccessor accessor(rowaccessor, i); if (!accessor.isValid()) { PDS_WARNING("fetch(): failed, can't create accessor on column '%d' for object '%s'", i, index.toString().c_str()); return false; } if (!accessor.fetch(data)) return false; } // lock row rowaccessor.acquire(); return true; } /** * Fetch column data into stream */ bool CTable::CDataAccessor::fetch(RY_PDS::CPData &data) { if (!isValid()) { PDS_WARNING_IN(_Table)("CDataAccessor:fetch(): failed, fetch invalid accessor '%s'", toString().c_str()); return false; } switch (_Column->getDataType()) { case PDS_bool: case PDS_sint8: case PDS_uint8: case PDS_char: data.serial(*(uint8*)_Data); break; case PDS_ucchar: case PDS_uint16: case PDS_sint16: data.serial(*(uint16*)_Data); break; case PDS_uint32: case PDS_sint32: case PDS_float: case PDS_CSheetId: case PDS_CNodeId: case PDS_enum: data.serial(*(uint32*)_Data); break; case PDS_uint64: case PDS_sint64: case PDS_double: case PDS_CEntityId: data.serial(*(uint64*)_Data); break; case PDS_Index: { // check this is a forward ref // only forward ref are sent, backref are implicitely deduced if (_Column->getMetaType() == PDS_BackRef) { PDS_WARNING_IN(_Table)("CDataAccessor:fetch(): failed, try to serialize '%s' as backref", toString().c_str()); return false; } RY_PDS::CObjectIndex child = *(RY_PDS::CObjectIndex*)_Data; if (!child.isChecksumValid()) { PDS_WARNING_IN(_Table)("CDataAccessor:fetch(): failed, '%s' points to invalid object '%s'", toString().c_str(), child.toString().c_str()); return false; } if (child.isNull()) { // send object id RY_PDS::TTableIndex tid = RY_PDS::INVALID_TABLE_INDEX; RY_PDS::TRowIndex rid = RY_PDS::INVALID_ROW_INDEX; data.serial(tid, rid); return true; } // send object and hierarchy return _Table->getParent()->fetch(child, data); } break; case PDS_List: { RY_PDS::CSetMap::CAccessor setaccess = getSet(); if (!setaccess.isValid()) { PDS_WARNING_IN(_Table)("CDataAccessor:fetch(): failed, object '%s' set access is invalid", toString().c_str()); return false; } const RY_PDS::TIndexList& list = setaccess.get(); uint i; for (i=0; igetParent(), child, _Attribute->getBackRefKey(), 0); if (!keyaccess.isValid()) { PDS_WARNING_IN(_Table)("CDataAccessor:fetch(): failed, '%s' points to invalid object '%s'", toString().c_str(), child.toString().c_str()); return false; } keyaccess.fetch(data); // force table not to fetch index, because we already done it _Table->getParent()->fetch(child, data, false); } RY_PDS::TTableIndex tid = RY_PDS::INVALID_TABLE_INDEX; RY_PDS::TRowIndex rid = RY_PDS::INVALID_ROW_INDEX; data.serial(tid, rid); } break; case PDS_dimension: if (_Column->getByteSize() == 1) { data.serial(*(uint8*)_Data); } else if (_Column->getByteSize() == 2) { data.serial(*(uint16*)_Data); } else { data.serial(*(uint32*)_Data); } break; } return true; } /* * Get value as string */ string CTable::CDataAccessor::valueAsString(bool expandSet) const { if (!isValid()) return string(""); std::string value; switch (_Column->getDataType()) { case PDS_bool: value = dataTypeAsString(*(bool*)_Data); break; case PDS_char: value = dataTypeAsString(*(char*)_Data); break; case PDS_uint8: value = dataTypeAsString(*(uint8*)_Data); break; case PDS_ucchar: case PDS_uint16: value = dataTypeAsString(*(uint16*)_Data); break; case PDS_CSheetId: case PDS_CNodeId: case PDS_uint32: value = dataTypeAsString(*(uint32*)_Data); break; case PDS_uint64: value = dataTypeAsString(*(uint64*)_Data); break; case PDS_sint8: value = dataTypeAsString(*(sint8*)_Data); break; case PDS_sint16: value = dataTypeAsString(*(sint16*)_Data); break; case PDS_sint32: value = dataTypeAsString(*(sint32*)_Data); break; case PDS_sint64: value = dataTypeAsString(*(sint64*)_Data); break; case PDS_float: value = dataTypeAsString(*(float*)_Data); break; case PDS_double: value = dataTypeAsString(*(double*)_Data); break; case PDS_CEntityId: value = dataTypeAsString(*(CEntityId*)_Data); break; case PDS_enum: { value = "'"+NLMISC::toString(*(uint32*)_Data)+"'"; const CType* type = _Table->getParent()->getType(_Column->getTypeId()); if (type == NULL) { value += " "; } else { value += " "+type->getName(); if (type->isEnum()) { value += " "+type->getIndexName(*(uint32*)_Data); } else { value += " not enum"; } } } break; case PDS_dimension: { if (_Column->getByteSize() == 1) value = "'"+NLMISC::toString(*(uint8*)_Data)+"'"; else if (_Column->getByteSize() == 2) value = "'"+NLMISC::toString(*(uint16*)_Data)+"'"; else if (_Column->getByteSize() == 4) value = "'"+NLMISC::toString(*(uint32*)_Data)+"'"; else value = ""; const CType* type = _Table->getParent()->getType(_Column->getTypeId()); if (type == NULL) { value += " "; } else { value += " "+type->getName(); if (type->isDimension()) { value += " "+type->getIndexName(*(uint32*)_Data); } else { value += " not dimension"; } } } break; case PDS_List: { #ifdef DEBUG_DATA_ACCESSOR if (_IsDebug) value = "undisplayable set"; else { const RY_PDS::TIndexList& list = getSet().get(); if (list.empty()) { value = "Empty list"; } else if (expandSet) { uint i; value = ""; for (i=0; itoString()+"'"; break; default: value = "undisplayable value"; break; } return value; } /* * Build the index allocator for this table */ bool CTable::buildIndexAllocator(RY_PDS::CIndexAllocator& allocat) { allocat.clear(); uint row; for (row=0; row<_TableBuffer.maxRowIndex(); ++row) { if (isAllocated(row)) { allocat.forceAllocated(row); } } return true; } /* * Clear dirty list */ bool CTable::clearDirtyList() { if (!initialised()) { PDS_WARNING("clearDirtyList(): table not initialised"); return false; } _TableBuffer.resetDirty(); return true; } /* * Reset dirty tags * Reset all rows so no one is marked as being dirty. * This method fixes broken list issues */ bool CTable::resetDirtyTags() { /// \todo fill here nlstop; if (!initialised()) { PDS_WARNING("clearDirtyList(): table not initialised"); return false; } return true; } /* * Preload reference files */ bool CTable::preloadRefFiles() { if (!_TableBuffer.openAllRefFilesRead()) return false; return true; } /* * Notify new Reference, do necessary job... */ bool CTable::notifyNewReference(CRefIndex& newref) { if (!initialised()) return false; _TableBuffer.setupRef(newref); return true; } /* * Apply delta changes from a file */ bool CTable::applyDeltaChanges(const string& filename) { if (!initialised()) { PDS_WARNING("buildDelta(): failed, table not initialised"); return false; } return _TableBuffer.applyDeltaChanges(filename); } /* * Reset row to initial value */ bool CTable::resetRow(uint8* rowData) { // copy row pattern memcpy(rowData, &(_EmptyRow[0]), _RowSize); return true; } /* * Build the delta file and purge all dirty rows in this table */ bool CTable::buildDelta(const CTimestamp& starttime, const CTimestamp& endtime) { H_AUTO(PDS_Table_buildDelta); if (!initialised()) { PDS_WARNING("buildDelta(): failed, table not initialised"); return false; } return _TableBuffer.buildDelta(starttime, endtime); } /* * Flush table from released rows */ bool CTable::flushReleased() { if (!initialised()) { PDS_WARNING("flushReleased(): failed, table not initialised"); return false; } _TableBuffer.flushReleased(); return true; } /* * Build RowMapper */ bool CTable::buildRowMapper() { if (!initialised()) { PDS_WARNING("buildRowMapper(): failed, table not initialised"); return false; } return _TableBuffer.buildRowMapper(); } /* * The process row callback, fix forwardrefs from backrefs */ bool CTable::processRow(RY_PDS::TTableIndex table, CTableBuffer::CAccessor& accessor) { if (table != _Id) { PDS_WARNING("processRow(): try to process a row from another table! (id '%d')", table); return false; } RY_PDS::CObjectIndex child = RY_PDS::CObjectIndex(table, accessor.row()); uint i; for (i=0; igetId()); RY_PDS::CObjectIndex parent; if (!backref.isValid() || !backref.getIndex(parent)) { PDS_WARNING("processRow(): failed to get BackRef index"); return false; } // parent not set, do nothing if (!parent.isValid()) { continue; } // get forwardref accessor CDataAccessor forwardref(_Parent, parent, bref.Referenced->getId(), 0); switch (bref.Referenced->getMetaType()) { case PDS_Set: processBackRefToSet(forwardref, child); break; case PDS_ArrayRef: // seek to pos in array if (!forwardref.seek(backref)) { PDS_WARNING("processRow(): failed to seek into '%s'", forwardref.toString().c_str()); return false; } case PDS_ForwardRef: if (!processBackRefToForwardRef(forwardref, child)) { PDS_WARNING("processRow(): failed to processBackRefToForwardRef('%s', '%s')", forwardref.toString().c_str(), child.toString().c_str()); return false; } break; default: break; } } for (i=0; igetId()); RY_PDS::CObjectIndex child; if (!fwdref.isValid() || !fwdref.getIndex(child)) { PDS_WARNING("processRow(): failed to get ForwardRef index"); return false; } // should not be null and valid if (child.isValid() && !child.isNull()) continue; // if potentially broken link // add to list of rows to fix, and leave BrokenForwardRefs.push_back(accessor.row()); break; } return true; } /* * Process Back Reference to a set */ bool CTable::processBackRefToSet(CDataAccessor& parent, RY_PDS::CObjectIndex child) { parent.getSet().add(child); return true; } /* * Process Back Reference to a forward ref */ bool CTable::processBackRefToForwardRef(CDataAccessor& parent, RY_PDS::CObjectIndex child) { RY_PDS::CObjectIndex checkchild; if (!parent.getIndex(checkchild)) { PDS_WARNING("processBackRefToForwardRef(): failed to access to parent '%s'", parent.toString().c_str()); return false; } if (checkchild != child) parent.setIndex(child); return true; } /* * Fill up Backward and Forward References information */ bool CTable::fillRefInfo() { uint i; for (i=0; i<_Columns.size(); ++i) { CColumn& column = _Columns[i]; // generate summary of backward reference if (column.getMetaType() == PDS_BackRef) { CTable* parent = _Parent->getNonConstTable((RY_PDS::TTableIndex)(column.getTypeId())); if (parent == NULL) return false; const CAttribute* referenced = parent->getAttribute(column.getParent()->getReferencedAttribute()); if (referenced == NULL) return false; CBackRefFiller bref; bref.Column = &column; bref.Referenced = referenced; bref.ParentTable = parent; BackRefInfo.push_back(bref); } // generate summary of forward reference else if (column.getMetaType() == PDS_ForwardRef) { const CAttribute* parent = column.getParent(); // check column in array of ref that are allowed to contain null if (parent->getMetaType() == PDS_ArrayRef && !parent->allowNull()) { CAutoForwardRefFiller fref; fref.Column = &column; ForwardRefInfo.push_back(fref); } } } return true; } /* * Fix broken forward refs */ bool CTable::fixForwardRefs() { if (BrokenForwardRefs.empty()) return true; uint i; for (i=0; igetId()); RY_PDS::CObjectIndex index; if (!forwardref.isValid() || !forwardref.getIndex(index)) { PDS_WARNING("fixRowForwardRefs(): failed to get forward ref '%s', data left as is", forwardref.toString().c_str()); continue; } // was ref fixed? if (index.isValid()) continue; // allocate new row CTable *childTable = _Parent->getNonConstTable((RY_PDS::TTableIndex)(colInfo.Column->getTypeId())); if (childTable == NULL || !childTable->initialised()) { PDS_WARNING("fixRowForwardRefs(): failed to get child table '%d'", colInfo.Column->getTypeId()); continue; } RY_PDS::TRowIndex alloc = childTable->nextUnallocatedRow(); if (!childTable->allocate(alloc)) { PDS_WARNING("fixRowForwardRefs(): failed to get allocate free row in table '%s'", childTable->getName().c_str()); continue; } // get an accessor on row key CTableBuffer::CAccessor allocAccess = childTable->_TableBuffer.getRow(alloc); CDataAccessor keyAccess(childTable, allocAccess, (RY_PDS::TColumnIndex)(childTable->getAttribute(childTable->getKey())->getOffset())); if (!keyAccess.isValid()) { PDS_WARNING("fixRowForwardRefs(): failed to get key access of row '%s:%d'", childTable->getName().c_str(), alloc); childTable->_TableBuffer.releaseRow(allocAccess); continue; } // compute key TEnumValue keyValue = forwardref.column()->getId()-forwardref.attribute()->getOffset(); // set new row key if (!keyAccess.setAsIndexType(keyValue)) { PDS_WARNING("fixRowForwardRefs(): failed to set key '%s'", keyAccess.toString().c_str()); childTable->_TableBuffer.releaseRow(allocAccess); continue; } CDataAccessor backref(keyAccess, (RY_PDS::TColumnIndex)(childTable->getAttribute(forwardref.attribute()->getReferencedAttribute())->getOffset())); // set back&forward links forwardref.setIndex(backref.getObjectIndex()); backref.setIndex(forwardref.getObjectIndex()); // fixup links in new child if (!childTable->fixRowForwardRefs(alloc)) { PDS_WARNING("fixRowForwardRefs(): failed to set fix forward refs in newly allocated row '%s'", backref.getObjectIndex().toString().c_str()); childTable->_TableBuffer.releaseRow(allocAccess); continue; } childTable->_TableBuffer.releaseRow(allocAccess); } _TableBuffer.releaseRow(accessor); return true; } /* * Dump accessor content and info to xml */ void CTable::CDataAccessor::dumpToXml(NLMISC::IStream& xml, sint expandDepth) { if (xml.isReading()) return; xml.xmlPushBegin("value"); xml.xmlSetAttrib("valid"); bool valid = isValid(); xml.serial(valid); bool closeTag = true; if (valid) { xml.xmlSetAttrib("name"); std::string columnName = _Column->getName(); xml.serial(columnName); xml.xmlSetAttrib("type"); std::string typeName = getNameFromDataType(_Column->getDataType()); xml.serial(typeName); std::string value; xml.xmlSetAttrib("value"); switch (_Column->getDataType()) { case PDS_bool: xml.serial(*(bool*)_Data); break; case PDS_char: xml.serial(*(char*)_Data); break; case PDS_uint8: xml.serial(*(uint8*)_Data); break; case PDS_ucchar: case PDS_uint16: xml.serial(*(uint16*)_Data); break; case PDS_CSheetId: case PDS_CNodeId: case PDS_uint32: xml.serial(*(uint32*)_Data); break; case PDS_uint64: xml.serial(*(uint64*)_Data); break; case PDS_sint8: xml.serial(*(sint8*)_Data); break; case PDS_sint16: xml.serial(*(sint16*)_Data); break; case PDS_sint32: xml.serial(*(sint32*)_Data); break; case PDS_sint64: xml.serial(*(sint64*)_Data); break; case PDS_float: xml.serial(*(float*)_Data); break; case PDS_double: xml.serial(*(double*)_Data); break; case PDS_CEntityId: { std::string id = ((CEntityId*)_Data)->toString(); xml.serial(id); } break; case PDS_enum: { std::string value; const CType* type = _Table->getParent()->getType(_Column->getTypeId()); if (type == NULL || !type->isEnum()) { xml.serial(*(uint32*)_Data); } else { std::string name = type->getIndexName(*(uint32*)_Data); xml.serial(name); } } break; case PDS_dimension: { if (_Column->getByteSize() == 1) { xml.serial(*(uint8*)_Data); } else if (_Column->getByteSize() == 2) { xml.serial(*(uint16*)_Data); } else if (_Column->getByteSize() == 4) { xml.serial(*(uint32*)_Data); } else { std::string unknown = "non displayable"; xml.serial(unknown); } } break; case PDS_List: { std::string value = "list"; xml.serial(value); xml.xmlPushEnd(); closeTag = false; if (expandDepth != 0) { const RY_PDS::TIndexList& list = getSet().get(); if (!list.empty()) { uint i; for (i=0; i_Parent->dumpToXml(list[i], xml, expandDepth-1); } } } break; case PDS_Index: { std::string value = ((RY_PDS::CObjectIndex*)_Data)->toString(_Table->getParent()); xml.serial(value); xml.xmlPushEnd(); closeTag = false; if (expandDepth != 0 && _Attribute->getMetaType() != PDS_BackRef) _Table->_Parent->dumpToXml(*(RY_PDS::CObjectIndex*)_Data, xml, expandDepth-1); } break; default: { std::string unknown = "non displayable"; xml.serial(unknown); } break; } } if (closeTag) xml.xmlPushEnd(); xml.xmlPop(); } /* * Dump accessor content and info to xml */ void CTable::dumpToXml(RY_PDS::TRowIndex row, NLMISC::IStream& xml, sint expandDepth) { if (xml.isReading()) return; xml.xmlPushBegin("object"); xml.xmlSetAttrib("valid"); bool valid = initialised(); xml.serial(valid); if (initialised()) { xml.xmlSetAttrib("name"); xml.serial(_Name); xml.xmlSetAttrib("index"); RY_PDS::CObjectIndex index((RY_PDS::TTableIndex)_Id, (RY_PDS::TRowIndex)row); std::string indexName = index.toString(_Parent); xml.serial(indexName); CTableBuffer::CAccessor rowaccess = _TableBuffer.getRow(row); bool rowAllocated = rowaccess.allocated(); xml.xmlSetAttrib("allocated"); xml.serial(rowAllocated); xml.xmlPushEnd(); if (rowAllocated) { uint i; for (i=0; i<_Columns.size(); ++i) { CDataAccessor accessor(this, rowaccess, i); const CColumn &col = _Columns[i]; accessor.dumpToXml(xml, expandDepth); } } _TableBuffer.releaseRow(rowaccess); } else { xml.xmlPushEnd(); } xml.xmlPop(); }