// 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 "pds_attribute.h" #include "pds_type.h" #include "pds_table.h" #include "pds_database.h" #include "../pd_lib/db_description_parser.h" #include using namespace std; using namespace NLMISC; /// Destructor CAttribute::~CAttribute() { //PDS_DEBUG("delete()"); } /* * Initialize attribute using a full xml node */ bool CAttribute::init(CDatabase *root, CTable* parent, const CAttributeNode& attribute) { // set parent logger setParentLogger(parent); _Parent = parent; _Root = root; _Name = attribute.Name; _Id = attribute.Id; _MetaType = attribute.MetaType; _TypeId = attribute.TypeId; switch (_MetaType) { case PDS_Type: { const CType* type = _Root->getType(_TypeId); if (type == NULL) { PDS_WARNING("init(): unknown type '%d'", _TypeId); return false; } _DataType = type->getDataType(); } break; case PDS_Class: _DataType = PDS_UnknownDataType; break; case PDS_BackRef: _DataType = PDS_Index; _ReferencedAttribute = attribute.Reference; break; case PDS_ForwardRef: _DataType = PDS_Index; _ReferencedAttribute = attribute.Reference; break; case PDS_ArrayType: { const CType* index = _Root->getType(attribute.Index); if (index == NULL || !index->isIndex()) { PDS_WARNING("init(): type '%d' unknown or not an index", attribute.Index); return false; } _IndexId = attribute.Index; const CType* type = _Root->getType(_TypeId); if (type == NULL) { PDS_WARNING("init(): unknown type '%d'", _TypeId); return false; } _DataType = type->getDataType(); } break; case PDS_ArrayClass: { const CType* type = _Root->getType(attribute.Index); if (type == NULL || !type->isIndex()) { PDS_WARNING("init(): type '%d' unknown or not an index", attribute.Index); return false; } _IndexId = attribute.Index; _DataType = PDS_UnknownDataType; } break; case PDS_ArrayRef: { const CType* type = _Root->getType(attribute.Index); if (type == NULL || !type->isIndex()) { PDS_WARNING("init(): type '%d' unknown or not an index", attribute.Index); return false; } _IndexId = attribute.Index; _DataType = PDS_Index; _ReferencedAttribute = attribute.Reference; _AllowNull = attribute.AllowNull; } break; case PDS_Set: _DataType = PDS_List; _ReferencedAttribute = attribute.Reference; break; } _Init = true; return true; } /* * Build columns for this attribute */ bool CAttribute::buildColumns() { vector& columns = _Parent->_Columns; _Offset = columns.size(); switch (_MetaType) { case PDS_Type: { CColumn column; column._Parent = this; column._Root = _Root; column._Id = columns.size(); column._Name = _Name; column._MetaType = PDS_Type; column._DataType = _DataType; column._TypeId = _TypeId; const CType* type = _Root->getType(_TypeId); if (type == NULL) { PDS_WARNING("init(): unknown type '%d'", _TypeId); return false; } column._ByteSize = type->getByteSize(); column._Init = true; columns.push_back(column); } break; case PDS_Class: { CTable* sub = const_cast(_Root->getTable(_TypeId)); if (sub == NULL || !sub->buildColumns()) return false; uint i; for (i=0; i_Columns.size(); ++i) { CColumn column = sub->_Columns[i]; column._Parent = this; column._Root = _Root; column._Id = columns.size(); column._Name = _Name+"."+column._Name; column._Init = true; columns.push_back(column); } } break; case PDS_BackRef: case PDS_ForwardRef: case PDS_Set: { CColumn column; column._Parent = this; column._Root = _Root; column._Id = columns.size(); column._Name = _Name; column._MetaType = _MetaType; column._DataType = _DataType; column._TypeId = _TypeId; column._ByteSize = getStandardByteSize(column._DataType); column._Init = true; columns.push_back(column); } break; case PDS_ArrayType: { const CType* type = _Root->getType(_TypeId); const CType* index = _Root->getType(_IndexId); if (type == NULL) { PDS_WARNING("init(): unknown type '%d'", _TypeId); return false; } if (index == NULL || !index->isIndex()) { PDS_WARNING("buildColumns(): type '%d' unknown or not an index", _IndexId); return false; } uint i; for (i=0; igetIndexSize(); ++i) { CColumn column; column._Parent = this; column._Root = _Root; column._Id = columns.size(); column._Name = _Name+"["+index->getIndexName(i)+"]"; column._MetaType = PDS_Type; column._DataType = _DataType; column._TypeId = _TypeId; column._ByteSize = type->getByteSize(); column._Init = true; columns.push_back(column); } } break; case PDS_ArrayClass: { CTable* sub = const_cast(_Root->getTable(_TypeId)); const CType* index = _Root->getType(_IndexId); if (sub == NULL || sub == NULL || !sub->buildColumns()) { PDS_WARNING("buildColumns(): unknown table '%d' or failed to build its columns", _TypeId); return false; } if (index == NULL || index == NULL || !index->isIndex()) { PDS_WARNING("buildColumns(): type '%d' unknown or not an index", _IndexId); return false; } uint i; for (i=0; igetIndexSize(); ++i) { uint j; for (j=0; j_Columns.size(); ++j) { CColumn column = sub->_Columns[j]; column._Parent = this; column._Root = _Root; column._Id = columns.size(); column._Name = _Name+"["+index->getIndexName(i)+"]"+"."+column._Name; column._Init = true; columns.push_back(column); } } } break; case PDS_ArrayRef: { const CType* index = _Root->getType(_IndexId); if (index == NULL || !index->isIndex()) { PDS_WARNING("buildColumns(): type '%d' unknown or not an index", _IndexId); return false; } uint i; for (i=0; igetIndexSize(); ++i) { CColumn column; column._Parent = this; column._Root = _Root; column._Id = columns.size(); column._Name = _Name+"["+index->getIndexName(i)+"]";; column._MetaType = PDS_ForwardRef; column._DataType = _DataType; column._TypeId = _TypeId; column._ByteSize = getStandardByteSize(column._DataType); column._Init = true; columns.push_back(column); } } break; default: PDS_WARNING("buildColumns(): attribute '%s' metatype is unknown", _Name.c_str()); return false; break; } _Columns = columns.size() - _Offset; return true; } /* * Compute back reference key */ bool CAttribute::computeBackRefKey() { if (_MetaType == PDS_BackRef) { const CTable* parentTable = _Root->getTable(_TypeId); const CAttribute* parentAttribute = parentTable->getAttribute(_ReferencedAttribute); // should check parentAttribute is a forward reference of any kind const CTable* childTable = _Root->getTable(parentAttribute->getTypeId()); _Key = childTable->getKey(); } else if (_MetaType == PDS_ArrayRef || _MetaType == PDS_Set) { const CTable* childTable = _Root->getTable(_TypeId); const CAttribute* childAttribute = childTable->getAttribute(_ReferencedAttribute); if (childAttribute == NULL) { PDS_WARNING("computeBackRefKey(): failed, child backref not initialised"); return false; } if (childAttribute->getMetaType() != PDS_BackRef) { PDS_WARNING("computeBackRefKey(): failed, child backref actually not a backref"); return false; } _Key = childTable->getKey(); if (_Key == INVALID_TYPE_ID) { PDS_WARNING("computeBackRefKey(): failed, child key is invalid"); return false; } const CAttribute* keyAttribute = childTable->getAttribute(_Key); if (keyAttribute == NULL) { PDS_WARNING("computeBackRefKey(): failed, child key '%d' attribute cannot be found", _Key); return false; } if (_Key != childAttribute->getBackRefKey()) { PDS_WARNING("computeBackRefKey(): failed, child backref key '%d' mismatch child key '%d'", childAttribute->getBackRefKey(), _Key); return false; } } return true; }