/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * 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 #ifndef _WIN32 #include #include #endif #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace ::com::sun::star; namespace { class MiscTest : public UnoApiXmlTest { public: MiscTest() : UnoApiXmlTest(u"/sfx2/qa/cppunit/data/"_ustr) { } virtual void registerNamespaces(xmlXPathContextPtr& pXmlXpathCtx) override { // ODF xmlXPathRegisterNs(pXmlXpathCtx, BAD_CAST("office"), BAD_CAST("urn:oasis:names:tc:opendocument:xmlns:office:1.0")); xmlXPathRegisterNs(pXmlXpathCtx, BAD_CAST("meta"), BAD_CAST("urn:oasis:names:tc:opendocument:xmlns:meta:1.0")); xmlXPathRegisterNs(pXmlXpathCtx, BAD_CAST("dc"), BAD_CAST("http://purl.org/dc/elements/1.1/")); // used in testCustomMetadata xmlXPathRegisterNs(pXmlXpathCtx, BAD_CAST("foo"), BAD_CAST("http://foo.net")); xmlXPathRegisterNs(pXmlXpathCtx, BAD_CAST("baz"), BAD_CAST("http://baz.net")); } }; class Listener : public ::cppu::WeakImplHelper { public: Listener() : bCalled(false) { } bool reset() { bool bOld = bCalled; bCalled = false; return bOld; } private: bool bCalled; virtual void SAL_CALL disposing(const lang::EventObject&) noexcept override {} virtual void SAL_CALL modified(const lang::EventObject&) noexcept override { bCalled = true; } }; CPPUNIT_TEST_FIXTURE(MiscTest, testODFCustomMetadata) { uno::Reference const xProps( ::com::sun::star::document::DocumentProperties::create(m_xContext)); OUString const url(m_directories.getURLFromSrc(u"/sfx2/qa/complex/sfx2/testdocuments/CUSTOM.odt")); xProps->loadFromMedium(url, uno::Sequence()); CPPUNIT_ASSERT_EQUAL(u""_ustr, xProps->getAuthor()); uno::Sequence mimeArgs({ beans::PropertyValue(u"MediaType"_ustr, -1, uno::Any(u"application/vnd.oasis.opendocument.text"_ustr), beans::PropertyState_DIRECT_VALUE) }); xProps->storeToMedium(maTempFile.GetURL(), mimeArgs); // check that custom metadata is preserved xmlDocUniquePtr pXmlDoc = parseExport(u"meta.xml"_ustr); assertXPathContent(pXmlDoc, "/office:document-meta/office:meta/bork"_ostr, u"bork"_ustr); assertXPath(pXmlDoc, "/office:document-meta/office:meta/foo:bar"_ostr, 1); assertXPath(pXmlDoc, "/office:document-meta/office:meta/foo:bar/baz:foo"_ostr, 1); assertXPath(pXmlDoc, "/office:document-meta/office:meta/foo:bar/baz:foo[@baz:bar='foo']"_ostr); assertXPathContent(pXmlDoc, "/office:document-meta/office:meta/foo:bar/foo:baz"_ostr, u"bar"_ustr); } /** * Test case for the service com.sun.star.document.DocumentProperties. * Currently, this service is implemented in * sfx2/source/doc/SfxDocumentMetaData.cxx. * */ CPPUNIT_TEST_FIXTURE(MiscTest, testDocumentProperties) { uno::Reference const xProps( ::com::sun::star::document::DocumentProperties::create(m_xContext)); OUString const url( m_directories.getURLFromSrc(u"/sfx2/qa/complex/sfx2/testdocuments/TEST.odt")); xProps->loadFromMedium(url, uno::Sequence()); CPPUNIT_ASSERT_EQUAL(u"Karl-Heinz Mustermann"_ustr, xProps->getAuthor()); CPPUNIT_ASSERT_EQUAL(u"StarOffice/8$Solaris_x86 OpenOffice.org_project/680m232$Build-9227"_ustr, xProps->getGenerator()); CPPUNIT_ASSERT_EQUAL(sal_Int16(2007), xProps->getCreationDate().Year); CPPUNIT_ASSERT_EQUAL(u"Urgent Memo"_ustr, xProps->getTitle()); CPPUNIT_ASSERT_EQUAL(u"Wichtige Mitteilung"_ustr, xProps->getSubject()); CPPUNIT_ASSERT_EQUAL(u"Modern internal company memorandum in full-blocked style"_ustr, xProps->getDescription()); CPPUNIT_ASSERT_EQUAL(u"Karl-Heinz Mustermann"_ustr, xProps->getModifiedBy()); CPPUNIT_ASSERT_EQUAL(sal_uInt16(10), xProps->getModificationDate().Month); CPPUNIT_ASSERT_EQUAL(u"Karl-Heinz Mustermann"_ustr, xProps->getPrintedBy()); CPPUNIT_ASSERT_EQUAL(sal_uInt16(29), xProps->getPrintDate().Day); CPPUNIT_ASSERT_EQUAL(u"Modern Memo"_ustr, xProps->getTemplateName()); CPPUNIT_ASSERT(xProps->getTemplateURL().endsWith("memmodern.ott")); CPPUNIT_ASSERT_EQUAL(sal_uInt16(17), xProps->getTemplateDate().Hours); CPPUNIT_ASSERT(xProps->getAutoloadURL().endsWith("/TEST.odt")); CPPUNIT_ASSERT_EQUAL(sal_Int32(0), xProps->getAutoloadSecs()); CPPUNIT_ASSERT_EQUAL(u"_blank"_ustr, xProps->getDefaultTarget()); CPPUNIT_ASSERT_EQUAL(sal_Int16(3), xProps->getEditingCycles()); CPPUNIT_ASSERT_EQUAL(sal_Int32(320), xProps->getEditingDuration()); uno::Sequence aKeywords(xProps->getKeywords()); CPPUNIT_ASSERT_EQUAL(sal_Int32(3), aKeywords.getLength()); CPPUNIT_ASSERT_EQUAL(u"Memo"_ustr, aKeywords[0]); CPPUNIT_ASSERT_EQUAL(u"Asien"_ustr, aKeywords[1]); CPPUNIT_ASSERT_EQUAL(u"Reis"_ustr, aKeywords[2]); uno::Sequence aDocStats = xProps->getDocumentStatistics(); auto it = std::find_if(std::cbegin(aDocStats), std::cend(aDocStats), [](const css::beans::NamedValue& val) { return val.Name == "WordCount" && val.Value.get() == 23; }); CPPUNIT_ASSERT(it != std::cend(aDocStats)); it = std::find_if(std::cbegin(aDocStats), std::cend(aDocStats), [](const css::beans::NamedValue& val) { return val.Name == "PageCount" && val.Value.get() == 1; }); CPPUNIT_ASSERT(it != std::cend(aDocStats)); uno::Reference xUDP = xProps->getUserDefinedProperties(); uno::Reference xPropertySet(xUDP, uno::UNO_QUERY); uno::Any aAny = xPropertySet->getPropertyValue(u"Hinweis"_ustr); CPPUNIT_ASSERT_EQUAL(u"Dies ist ein wichtiger Hinweis"_ustr, aAny.get()); aAny = xPropertySet->getPropertyValue(u"Warnung"_ustr); CPPUNIT_ASSERT_EQUAL(u"Kann Spuren von N\u00FCssen enthalten"_ustr, aAny.get()); xProps->setAuthor(u"me"_ustr); CPPUNIT_ASSERT_EQUAL(u"me"_ustr, xProps->getAuthor()); xProps->setGenerator(u"the computa"_ustr); CPPUNIT_ASSERT_EQUAL(u"the computa"_ustr, xProps->getGenerator()); css::util::DateTime aDateTime; aDateTime.Day = 1; aDateTime.Month = 1; aDateTime.Year = 2038; xProps->setCreationDate(aDateTime); CPPUNIT_ASSERT_EQUAL(sal_Int16(2038), xProps->getCreationDate().Year); xProps->setTitle(u"El t'itulo"_ustr); CPPUNIT_ASSERT_EQUAL(u"El t'itulo"_ustr, xProps->getTitle()); xProps->setSubject(u"Ein verkommenes Subjekt"_ustr); CPPUNIT_ASSERT_EQUAL(u"Ein verkommenes Subjekt"_ustr, xProps->getSubject()); xProps->setDescription(u"Este descripci'on no es importante"_ustr); CPPUNIT_ASSERT_EQUAL(u"Este descripci'on no es importante"_ustr, xProps->getDescription()); lang::Locale aLang; aLang.Language = u"en"_ustr; aLang.Country = u"GB"_ustr; xProps->setLanguage(aLang); CPPUNIT_ASSERT_EQUAL(aLang.Language, xProps->getLanguage().Language); CPPUNIT_ASSERT_EQUAL(aLang.Country, xProps->getLanguage().Country); xProps->setModifiedBy(u"myself"_ustr); CPPUNIT_ASSERT_EQUAL(u"myself"_ustr, xProps->getModifiedBy()); aDateTime.Year = 2042; xProps->setModificationDate(aDateTime); CPPUNIT_ASSERT_EQUAL(sal_Int16(2042), xProps->getModificationDate().Year); xProps->setPrintedBy(u"i did not do it"_ustr); CPPUNIT_ASSERT_EQUAL(u"i did not do it"_ustr, xProps->getPrintedBy()); aDateTime.Year = 2024; xProps->setPrintDate(aDateTime); CPPUNIT_ASSERT_EQUAL(sal_Int16(2024), xProps->getPrintDate().Year); xProps->setTemplateName(u"blah"_ustr); CPPUNIT_ASSERT_EQUAL(u"blah"_ustr, xProps->getTemplateName()); xProps->setTemplateURL(u"gopher://some-hole-in-the-ground/"_ustr); CPPUNIT_ASSERT_EQUAL(u"gopher://some-hole-in-the-ground/"_ustr, xProps->getTemplateURL()); aDateTime.Year = 2043; xProps->setTemplateDate(aDateTime); CPPUNIT_ASSERT_EQUAL(sal_Int16(2043), xProps->getTemplateDate().Year); xProps->setAutoloadURL(u"http://nowhere/"_ustr); CPPUNIT_ASSERT_EQUAL(u"http://nowhere/"_ustr, xProps->getAutoloadURL()); xProps->setAutoloadSecs(3661); CPPUNIT_ASSERT_EQUAL(sal_Int32(3661), xProps->getAutoloadSecs()); xProps->setDefaultTarget(u"_blank"_ustr); CPPUNIT_ASSERT_EQUAL(u"_blank"_ustr, xProps->getDefaultTarget()); xProps->setEditingCycles(42); CPPUNIT_ASSERT_EQUAL(sal_Int16(42), xProps->getEditingCycles()); xProps->setEditingDuration(84); CPPUNIT_ASSERT_EQUAL(sal_Int32(84), xProps->getEditingDuration()); uno::Sequence aKeywords2{ u"keywordly"_ustr, u"keywordlike"_ustr, u"keywordalicious"_ustr }; xProps->setKeywords(aKeywords2); aKeywords = xProps->getKeywords(); CPPUNIT_ASSERT_EQUAL(sal_Int32(3), aKeywords.getLength()); CPPUNIT_ASSERT_EQUAL(u"keywordly"_ustr, aKeywords[0]); CPPUNIT_ASSERT_EQUAL(u"keywordlike"_ustr, aKeywords[1]); CPPUNIT_ASSERT_EQUAL(u"keywordalicious"_ustr, aKeywords[2]); uno::Sequence aDocStats2{ { u"SyllableCount"_ustr, uno::Any(sal_Int16(9)) }, { u"FrameCount"_ustr, uno::Any(sal_Int16(2)) }, { u"SentenceCount"_ustr, uno::Any(sal_Int16(7)) } }; xProps->setDocumentStatistics(aDocStats2); aDocStats = xProps->getDocumentStatistics(); it = std::find_if(std::cbegin(aDocStats), std::cend(aDocStats), [](const css::beans::NamedValue& val) { return val.Name == "SyllableCount" && val.Value.get() == 9; }); CPPUNIT_ASSERT(it != std::cend(aDocStats)); it = std::find_if(std::cbegin(aDocStats), std::cend(aDocStats), [](const css::beans::NamedValue& val) { return val.Name == "FrameCount" && val.Value.get() == 2; }); CPPUNIT_ASSERT(it != std::cend(aDocStats)); it = std::find_if(std::cbegin(aDocStats), std::cend(aDocStats), [](const css::beans::NamedValue& val) { return val.Name == "SentenceCount" && val.Value.get() == 7; }); CPPUNIT_ASSERT(it != std::cend(aDocStats)); // actually, this tests the PropertyBag service // but maybe the DocumentProperties service will be implemented // differently some day... // note that Time is only supported for backward compatibility! css::util::Time aTime; aTime.Hours = 1; aTime.Minutes = 16; css::util::Date aDate; aDate.Year = 2071; aDate.Month = 2; aDate.Day = 3; aDateTime.Year = 2065; css::util::Duration aDur; aDur.Negative = true; aDur.Years = 1001; aDur.Months = 999; aDur.Days = 888; aDur.Hours = 777; aDur.Minutes = 666; aDur.Seconds = 555; aDur.NanoSeconds = 444444444; xUDP->addProperty("Frobnicate", beans::PropertyAttribute::REMOVABLE, uno::Any(true)); xUDP->addProperty("FrobDuration", beans::PropertyAttribute::REMOVABLE, uno::Any(aDur)); xUDP->addProperty("FrobDuration2", beans::PropertyAttribute::REMOVABLE, uno::Any(aTime)); xUDP->addProperty("FrobEndDate", beans::PropertyAttribute::REMOVABLE, uno::Any(aDate)); xUDP->addProperty("FrobStartTime", beans::PropertyAttribute::REMOVABLE, uno::Any(aDateTime)); xUDP->addProperty("Pi", beans::PropertyAttribute::REMOVABLE, uno::Any(3.1415)); xUDP->addProperty("Foo", beans::PropertyAttribute::REMOVABLE, uno::Any(u"bar"_ustr)); xUDP->addProperty("Removed", beans::PropertyAttribute::REMOVABLE, uno::Any(u"bar"_ustr)); // #i94175#: empty property name is valid ODF 1.1 xUDP->addProperty("", beans::PropertyAttribute::REMOVABLE, uno::Any(u"eeeeek"_ustr)); try { xUDP->removeProperty(u"Info 1"_ustr); xUDP->removeProperty(u"Removed"_ustr); } catch (beans::UnknownPropertyException&) { CPPUNIT_FAIL("removeProperty failed"); } try { xUDP->addProperty("Forbidden", beans::PropertyAttribute::REMOVABLE, uno::Any(uno::Sequence{ u"foo"_ustr, u"bar"_ustr })); CPPUNIT_FAIL("inserting value of non-supported type did not fail"); } catch (beans::IllegalTypeException&) { } aAny = xPropertySet->getPropertyValue(u"Frobnicate"_ustr); CPPUNIT_ASSERT_EQUAL(true, aAny.get()); aAny = xPropertySet->getPropertyValue(u"FrobDuration"_ustr); css::util::Duration aDur2 = aAny.get(); CPPUNIT_ASSERT_EQUAL(aDur.Negative, aDur2.Negative); CPPUNIT_ASSERT_EQUAL(aDur.Years, aDur2.Years); CPPUNIT_ASSERT_EQUAL(aDur.Months, aDur2.Months); CPPUNIT_ASSERT_EQUAL(aDur.Days, aDur2.Days); CPPUNIT_ASSERT_EQUAL(aDur.Hours, aDur2.Hours); CPPUNIT_ASSERT_EQUAL(aDur.Minutes, aDur2.Minutes); CPPUNIT_ASSERT_EQUAL(aDur.Seconds, aDur2.Seconds); CPPUNIT_ASSERT_EQUAL(aDur.NanoSeconds, aDur2.NanoSeconds); aAny = xPropertySet->getPropertyValue(u"FrobDuration2"_ustr); css::util::Time aTime2 = aAny.get(); CPPUNIT_ASSERT_EQUAL(aTime.Hours, aTime2.Hours); CPPUNIT_ASSERT_EQUAL(aTime.Minutes, aTime2.Minutes); CPPUNIT_ASSERT_EQUAL(aTime.Seconds, aTime2.Seconds); CPPUNIT_ASSERT_EQUAL(aTime.NanoSeconds, aTime2.NanoSeconds); aAny = xPropertySet->getPropertyValue(u"FrobEndDate"_ustr); css::util::Date aDate2 = aAny.get(); CPPUNIT_ASSERT_EQUAL(aDate.Day, aDate2.Day); CPPUNIT_ASSERT_EQUAL(aDate.Month, aDate2.Month); CPPUNIT_ASSERT_EQUAL(aDate.Year, aDate2.Year); aAny = xPropertySet->getPropertyValue(u"FrobStartTime"_ustr); css::util::DateTime aDateTime2 = aAny.get(); CPPUNIT_ASSERT_EQUAL(aDateTime.Year, aDateTime2.Year); CPPUNIT_ASSERT_EQUAL(aDateTime.Month, aDateTime2.Month); CPPUNIT_ASSERT_EQUAL(aDateTime.Day, aDateTime2.Day); CPPUNIT_ASSERT_EQUAL(aDateTime.Hours, aDateTime2.Hours); CPPUNIT_ASSERT_EQUAL(aDateTime.Minutes, aDateTime2.Minutes); CPPUNIT_ASSERT_EQUAL(aDateTime.Seconds, aDateTime2.Seconds); CPPUNIT_ASSERT_EQUAL(aDateTime.NanoSeconds, aDateTime2.NanoSeconds); aAny = xPropertySet->getPropertyValue(u"Pi"_ustr); double aPi = aAny.get(); CPPUNIT_ASSERT_EQUAL(3.1415, aPi); aAny = xPropertySet->getPropertyValue(u"Foo"_ustr); OUString aFoo = aAny.get(); CPPUNIT_ASSERT_EQUAL(u"bar"_ustr, aFoo); aAny = xPropertySet->getPropertyValue(u""_ustr); OUString aEmpty = aAny.get(); CPPUNIT_ASSERT_EQUAL(u"eeeeek"_ustr, aEmpty); try { xPropertySet->getPropertyValue(u"Removed"_ustr); CPPUNIT_FAIL("UserDefined remove didn't"); } catch (beans::UnknownPropertyException&) { } uno::Sequence mimeArgs({ beans::PropertyValue( u"MediaType"_ustr, -1, uno::Any(u"application/vnd.oasis.opendocument.text"_ustr), beans::PropertyState_DIRECT_VALUE) }); xProps->storeToMedium(maTempFile.GetURL(), mimeArgs); xProps->loadFromMedium(maTempFile.GetURL(), uno::Sequence()); xUDP = xProps->getUserDefinedProperties(); xPropertySet.set(xUDP, uno::UNO_QUERY); aAny = xPropertySet->getPropertyValue(u"Frobnicate"_ustr); CPPUNIT_ASSERT_EQUAL(true, aAny.get()); aAny = xPropertySet->getPropertyValue(u"FrobDuration"_ustr); aDur2 = aAny.get(); CPPUNIT_ASSERT_EQUAL(aDur.Negative, aDur2.Negative); CPPUNIT_ASSERT_EQUAL(aDur.Years, aDur2.Years); CPPUNIT_ASSERT_EQUAL(aDur.Months, aDur2.Months); CPPUNIT_ASSERT_EQUAL(aDur.Days, aDur2.Days); CPPUNIT_ASSERT_EQUAL(aDur.Hours, aDur2.Hours); CPPUNIT_ASSERT_EQUAL(aDur.Minutes, aDur2.Minutes); CPPUNIT_ASSERT_EQUAL(aDur.Seconds, aDur2.Seconds); CPPUNIT_ASSERT_EQUAL(aDur.NanoSeconds, aDur2.NanoSeconds); css::util::Duration aDur3(false, 0, 0, 0, aTime.Hours, aTime.Minutes, aTime.Seconds, aTime.NanoSeconds); // this is now a Duration! aAny = xPropertySet->getPropertyValue(u"FrobDuration2"_ustr); aDur2 = aAny.get(); CPPUNIT_ASSERT_EQUAL(aDur3.Negative, aDur2.Negative); CPPUNIT_ASSERT_EQUAL(aDur3.Years, aDur2.Years); CPPUNIT_ASSERT_EQUAL(aDur3.Months, aDur2.Months); CPPUNIT_ASSERT_EQUAL(aDur3.Days, aDur2.Days); CPPUNIT_ASSERT_EQUAL(aDur3.Hours, aDur2.Hours); CPPUNIT_ASSERT_EQUAL(aDur3.Minutes, aDur2.Minutes); CPPUNIT_ASSERT_EQUAL(aDur3.Seconds, aDur2.Seconds); CPPUNIT_ASSERT_EQUAL(aDur3.NanoSeconds, aDur2.NanoSeconds); aAny = xPropertySet->getPropertyValue(u"FrobEndDate"_ustr); aDate2 = aAny.get(); CPPUNIT_ASSERT_EQUAL(aDate.Day, aDate2.Day); CPPUNIT_ASSERT_EQUAL(aDate.Month, aDate2.Month); CPPUNIT_ASSERT_EQUAL(aDate.Year, aDate2.Year); aAny = xPropertySet->getPropertyValue(u"FrobStartTime"_ustr); aDateTime2 = aAny.get(); CPPUNIT_ASSERT_EQUAL(aDateTime.Year, aDateTime2.Year); CPPUNIT_ASSERT_EQUAL(aDateTime.Month, aDateTime2.Month); CPPUNIT_ASSERT_EQUAL(aDateTime.Day, aDateTime2.Day); CPPUNIT_ASSERT_EQUAL(aDateTime.Hours, aDateTime2.Hours); CPPUNIT_ASSERT_EQUAL(aDateTime.Minutes, aDateTime2.Minutes); CPPUNIT_ASSERT_EQUAL(aDateTime.Seconds, aDateTime2.Seconds); CPPUNIT_ASSERT_EQUAL(aDateTime.NanoSeconds, aDateTime2.NanoSeconds); aAny = xPropertySet->getPropertyValue(u"Pi"_ustr); aPi = aAny.get(); CPPUNIT_ASSERT_EQUAL(3.1415, aPi); aAny = xPropertySet->getPropertyValue(u"Foo"_ustr); aFoo = aAny.get(); CPPUNIT_ASSERT_EQUAL(u"bar"_ustr, aFoo); aAny = xPropertySet->getPropertyValue(u""_ustr); aEmpty = aAny.get(); CPPUNIT_ASSERT_EQUAL(u"eeeeek"_ustr, aEmpty); try { xPropertySet->getPropertyValue(u"Removed"_ustr); CPPUNIT_FAIL("UserDefined remove didn't"); } catch (beans::UnknownPropertyException&) { } rtl::Reference xListener = new Listener(); css::uno::Reference xBroadcaster(xProps, css::uno::UNO_QUERY); xBroadcaster->addModifyListener(xListener); xProps->setAuthor(u"not me"_ustr); CPPUNIT_ASSERT(xListener->reset()); xUDP->addProperty(u"Listener"_ustr, beans::PropertyAttribute::REMOVABLE, uno::Any(u"foo"_ustr)); CPPUNIT_ASSERT(xListener->reset()); xPropertySet->setPropertyValue(u"Listener"_ustr, uno::Any(u"bar"_ustr)); CPPUNIT_ASSERT(xListener->reset()); xUDP->removeProperty(u"Listener"_ustr); CPPUNIT_ASSERT(xListener->reset()); xBroadcaster->removeModifyListener(xListener); xUDP->addProperty(u"Listener2"_ustr, beans::PropertyAttribute::REMOVABLE, uno::Any(u"foo"_ustr)); CPPUNIT_ASSERT(!xListener->reset()); } CPPUNIT_TEST_FIXTURE(MiscTest, testNoThumbnail) { // Load a document. loadFromFile(u"hello.odt"); // Save it with the NoThumbnail option and assert that it has no thumbnail. #ifndef _WIN32 mode_t nMask = umask(022); #endif uno::Reference xStorable(mxComponent, uno::UNO_QUERY); CPPUNIT_ASSERT(xStorable.is()); uno::Sequence aProperties( comphelper::InitPropertySequence({ { "NoThumbnail", uno::Any(true) } })); osl::File::remove(maTempFile.GetURL()); xStorable->storeToURL(maTempFile.GetURL(), aProperties); uno::Reference xZipFile = packages::zip::ZipFileAccess::createWithURL(m_xContext, maTempFile.GetURL()); CPPUNIT_ASSERT(!xZipFile->hasByName(u"Thumbnails/thumbnail.png"_ustr)); #ifndef _WIN32 // Check permissions of the URL after store. osl::DirectoryItem aItem; CPPUNIT_ASSERT_EQUAL(osl::DirectoryItem::E_None, osl::DirectoryItem::get(maTempFile.GetURL(), aItem)); osl::FileStatus aStatus(osl_FileStatus_Mask_Attributes); CPPUNIT_ASSERT_EQUAL(osl::DirectoryItem::E_None, aItem.getFileStatus(aStatus)); // The following checks used to fail in the past, osl_File_Attribute_GrpRead was not set even if // umask requested so: CPPUNIT_ASSERT(aStatus.getAttributes() & osl_File_Attribute_GrpRead); CPPUNIT_ASSERT(aStatus.getAttributes() & osl_File_Attribute_OthRead); // Now "save as" again to trigger the "overwrite" case. xStorable->storeToURL(maTempFile.GetURL(), {}); CPPUNIT_ASSERT_EQUAL(osl::DirectoryItem::E_None, aItem.getFileStatus(aStatus)); // The following check used to fail in the past, result had temp file // permissions. CPPUNIT_ASSERT(aStatus.getAttributes() & osl_File_Attribute_GrpRead); umask(nMask); #endif } CPPUNIT_TEST_FIXTURE(MiscTest, testHardLinks) { #ifndef _WIN32 OUString aTargetDir = m_directories.getURLFromWorkdir(u"/CppunitTest/sfx2_misc.test.user/"); const OUString aURL(aTargetDir + "hello.odt"); osl::File::copy(createFileURL(u"hello.odt"), aURL); OUString aTargetPath; osl::FileBase::getSystemPathFromFileURL(aURL, aTargetPath); OString aOld = aTargetPath.toUtf8(); aTargetPath += ".2"; OString aNew = aTargetPath.toUtf8(); int nRet = link(aOld.getStr(), aNew.getStr()); CPPUNIT_ASSERT_EQUAL(0, nRet); mxComponent = loadFromDesktop(aURL, u"com.sun.star.text.TextDocument"_ustr); uno::Reference xStorable(mxComponent, uno::UNO_QUERY); xStorable->store(); struct stat buf; // coverity[fs_check_call] - this is legitimate in the context of this test nRet = stat(aOld.getStr(), &buf); CPPUNIT_ASSERT_EQUAL(0, nRet); // This failed: hard link count was 1, the hard link broke on store. CPPUNIT_ASSERT(buf.st_nlink > 1); // Test that symlinks are preserved as well. nRet = remove(aNew.getStr()); CPPUNIT_ASSERT_EQUAL(0, nRet); nRet = symlink(aOld.getStr(), aNew.getStr()); CPPUNIT_ASSERT_EQUAL(0, nRet); xStorable->storeToURL(aURL + ".2", {}); nRet = lstat(aNew.getStr(), &buf); CPPUNIT_ASSERT_EQUAL(0, nRet); // This failed, the hello.odt.2 symlink was replaced with a real file. CPPUNIT_ASSERT(bool(S_ISLNK(buf.st_mode))); #endif } CPPUNIT_TEST_FIXTURE(MiscTest, testtestOverwriteReadOnly) { #ifdef UNX // Given a read-only, already created file: OUString aTargetDir = m_directories.getURLFromWorkdir(u"/CppunitTest/sfx2_misc.test.user/"); OUString aURL(aTargetDir + "read-only.odt"); osl::File aFile(aURL); aFile.open(osl_File_OpenFlag_Create); aFile.close(); osl::File::setAttributes(aURL, osl_File_Attribute_OwnRead); mxComponent = loadFromDesktop(aURL, u"com.sun.star.text.TextDocument"_ustr); uno::Reference xStorable(mxComponent, uno::UNO_QUERY); bool bFail = false; // When trying to overwrite that: try { xStorable->storeToURL(aURL, { comphelper::makePropertyValue("Overwrite", true) }); } catch (const io::IOException&) { bFail = true; } // Then make sure we fail: // Without the accompanying fix in place, this test would have failed, the overwrite would // silently ignore the read-only attribute of the file. CPPUNIT_ASSERT(bFail); #endif } CPPUNIT_TEST_FIXTURE(MiscTest, testOverwrite) { // tdf#60237 - try to overwrite an existing file using the different settings of the Overwrite option mxComponent = loadFromDesktop(maTempFile.GetURL(), u"com.sun.star.text.TextDocument"_ustr); uno::Reference xStorable(mxComponent, uno::UNO_QUERY); CPPUNIT_ASSERT(xStorable.is()); // overwrite the file using the default case of the Overwrite option (true) CPPUNIT_ASSERT_NO_THROW(xStorable->storeToURL(maTempFile.GetURL(), {})); // explicitly overwrite the file using the Overwrite option CPPUNIT_ASSERT_NO_THROW(xStorable->storeToURL( maTempFile.GetURL(), comphelper::InitPropertySequence({ { "Overwrite", uno::Any(true) } }))); try { // overwrite an existing file with the Overwrite flag set to false xStorable->storeToURL(maTempFile.GetURL(), comphelper::InitPropertySequence( { { "Overwrite", uno::Any(false) } })); CPPUNIT_ASSERT_MESSAGE("We expect an exception on overwriting an existing file", false); } catch (const css::uno::Exception&) { } } } CPPUNIT_PLUGIN_IMPLEMENT(); /* vim:set shiftwidth=4 softtabstop=4 expandtab: */