/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4; fill-column: 100 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #include "Columns.hxx" #include "Indexes.hxx" #include "Keys.hxx" #include "Table.hxx" #include #include #include #include #include #include #include using namespace ::connectivity; using namespace ::connectivity::firebird; using namespace ::connectivity::sdbcx; using namespace ::osl; using namespace ::com::sun::star; using namespace ::com::sun::star::beans; using namespace ::com::sun::star::container; using namespace ::com::sun::star::sdbc; using namespace ::com::sun::star::sdbcx; using namespace ::com::sun::star::uno; Table::Table(Tables* pTables, Mutex& rMutex, const uno::Reference< XConnection >& rConnection): OTableHelper(pTables, rConnection, true), m_rMutex(rMutex), m_nPrivileges(0) { construct(); } Table::Table(Tables* pTables, Mutex& rMutex, const uno::Reference< XConnection >& rConnection, const OUString& rName, const OUString& rType, const OUString& rDescription): OTableHelper(pTables, rConnection, true, rName, rType, rDescription, u""_ustr, u""_ustr), m_rMutex(rMutex), m_nPrivileges(0) { construct(); } void Table::construct() { OTableHelper::construct(); if (isNew()) return; // TODO: get privileges when in non-embedded mode. m_nPrivileges = Privilege::DROP | Privilege::REFERENCE | Privilege::ALTER | Privilege::CREATE | Privilege::READ | Privilege::DELETE | Privilege::UPDATE | Privilege::INSERT | Privilege::SELECT; registerProperty(OMetaConnection::getPropMap().getNameByIndex(PROPERTY_ID_PRIVILEGES), PROPERTY_ID_PRIVILEGES, PropertyAttribute::READONLY, &m_nPrivileges, cppu::UnoType::get()); } //----- OTableHelper --------------------------------------------------------- OCollection* Table::createColumns(const ::std::vector< OUString>& rNames) { return new Columns(*this, m_rMutex, rNames); } OCollection* Table::createKeys(const ::std::vector< OUString>& rNames) { return new Keys(this, m_rMutex, rNames); } OCollection* Table::createIndexes(const ::std::vector< OUString>& rNames) { return new Indexes(this, m_rMutex, rNames); } //----- XAlterTable ----------------------------------------------------------- void SAL_CALL Table::alterColumnByName(const OUString& rColName, const uno::Reference< XPropertySet >& rDescriptor) { MutexGuard aGuard(m_rMutex); checkDisposed(WeakComponentImplHelperBase::rBHelper.bDisposed); uno::Reference< XPropertySet > xColumn(m_xColumns->getByName(rColName), UNO_QUERY); // sdbcx::Descriptor const bool bNameChanged = xColumn->getPropertyValue(u"Name"_ustr) != rDescriptor->getPropertyValue(u"Name"_ustr); // sdbcx::ColumnDescriptor const bool bTypeChanged = xColumn->getPropertyValue(u"Type"_ustr) != rDescriptor->getPropertyValue(u"Type"_ustr); const bool bTypeNameChanged = xColumn->getPropertyValue(u"TypeName"_ustr) != rDescriptor->getPropertyValue(u"TypeName"_ustr); const bool bPrecisionChanged = xColumn->getPropertyValue(u"Precision"_ustr) != rDescriptor->getPropertyValue(u"Precision"_ustr); const bool bScaleChanged = xColumn->getPropertyValue(u"Scale"_ustr) != rDescriptor->getPropertyValue(u"Scale"_ustr); const bool bIsNullableChanged = xColumn->getPropertyValue(u"IsNullable"_ustr) != rDescriptor->getPropertyValue(u"IsNullable"_ustr); const bool bIsAutoIncrementChanged = xColumn->getPropertyValue(u"IsAutoIncrement"_ustr) != rDescriptor->getPropertyValue(u"IsAutoIncrement"_ustr); // TODO: remainder -- these are all "optional" so have to detect presence and change. bool bDefaultChanged = xColumn->getPropertyValue(u"DefaultValue"_ustr) != rDescriptor->getPropertyValue(u"DefaultValue"_ustr); if (bTypeChanged || bTypeNameChanged || bPrecisionChanged || bScaleChanged) { // If bPrecisionChanged this will only succeed if we have increased the // precision, otherwise an exception is thrown -- however the base // gui then offers to delete and recreate the column. OUString sSql(getAlterTableColumn(rColName) + "TYPE " + ::dbtools::createStandardTypePart(rDescriptor, getConnection())); getConnection()->createStatement()->execute(sSql); // TODO: could cause errors e.g. if incompatible types, deal with them here as appropriate. // possibly we have to wrap things in Util::evaluateStatusVector. } if (bIsNullableChanged) { sal_Int32 nNullable = 0; rDescriptor->getPropertyValue(u"IsNullable"_ustr) >>= nNullable; if (nNullable != ColumnValue::NULLABLE_UNKNOWN) { OUString sSql(getAlterTableColumn(rColName)); if (nNullable == ColumnValue::NULLABLE) { sSql += "DROP NOT NULL"; } else if (nNullable == ColumnValue::NO_NULLS) { sSql += "SET NOT NULL"; } getConnection()->createStatement()->execute(sSql); } else { SAL_WARN("connectivity.firebird", "Attempting to set Nullable to NULLABLE_UNKNOWN"); } } if (bIsAutoIncrementChanged) { ::dbtools::throwSQLException( u"Changing autoincrement property of existing column is not supported"_ustr, ::dbtools::StandardSQLState::FUNCTION_NOT_SUPPORTED, *this); } if (bDefaultChanged) { OUString sNewDefault; rDescriptor->getPropertyValue(u"DefaultValue"_ustr) >>= sNewDefault; OUString sSql; if (sNewDefault.isEmpty()) sSql = getAlterTableColumn(rColName) + "DROP DEFAULT"; else sSql = getAlterTableColumn(rColName) + "SET DEFAULT " + sNewDefault; getConnection()->createStatement()->execute(sSql); } // TODO: quote identifiers as needed. if (bNameChanged) { OUString sNewColName; rDescriptor->getPropertyValue(u"Name"_ustr) >>= sNewColName; OUString sSql(getAlterTableColumn(rColName) + " TO \"" + sNewColName + "\""); getConnection()->createStatement()->execute(sSql); } m_xColumns->refresh(); } // ----- XRename -------------------------------------------------------------- void SAL_CALL Table::rename(const OUString&) { throw RuntimeException(u"Table renaming not supported by Firebird."_ustr); } // ----- XInterface ----------------------------------------------------------- Any SAL_CALL Table::queryInterface(const Type& rType) { if (rType.getTypeName() == "com.sun.star.sdbcx.XRename") return Any(); return OTableHelper::queryInterface(rType); } OUString Table::getAlterTableColumn(std::u16string_view rColumn) { return ("ALTER TABLE \"" + getName() + "\" ALTER COLUMN \"" + rColumn + "\" "); } /* vim:set shiftwidth=4 softtabstop=4 expandtab cinoptions=b1,g0,N-s cinkeys+=0=break: */