/* -*- 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/. * * This file incorporates work covered by the following license notice: * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed * with this work for additional information regarding copyright * ownership. The ASF licenses this file to you under the Apache * License, Version 2.0 (the "License"); you may not use this file * except in compliance with the License. You may obtain a copy of * the License at http://www.apache.org/licenses/LICENSE-2.0 . */ #include #include #include #include #include #include "cmdmailsuppl.hxx" #include "cmdmailmsg.hxx" #include #include #include #include #include #include #include using com::sun::star::beans::PropertyValue; using com::sun::star::system::XSimpleMailClientSupplier; using com::sun::star::system::XSimpleMailClient; using com::sun::star::system::XSimpleMailMessage; using com::sun::star::system::XSimpleMailMessage2; using com::sun::star::container::XNameAccess; using osl::FileBase; using namespace cppu; using namespace com::sun::star::system::SimpleMailClientFlags; using namespace com::sun::star::uno; using namespace com::sun::star::lang; using namespace com::sun::star::configuration; CmdMailSuppl::CmdMailSuppl( const Reference< XComponentContext >& xContext ) { m_xConfigurationProvider = theDefaultProvider::get(xContext); } // XSimpleMailClientSupplier Reference< XSimpleMailClient > SAL_CALL CmdMailSuppl::querySimpleMailClient( ) { return static_cast < XSimpleMailClient * > (this); } // XSimpleMailClient Reference< XSimpleMailMessage > SAL_CALL CmdMailSuppl::createSimpleMailMessage( ) { return Reference< XSimpleMailMessage >( new CmdMailMsg( ) ); } namespace { void appendShellWord(OStringBuffer & buffer, OUString const & word, bool strict) { OString sys; if (!word.convertToString( &sys, osl_getThreadTextEncoding(), (strict ? (RTL_UNICODETOTEXT_FLAGS_UNDEFINED_ERROR | RTL_UNICODETOTEXT_FLAGS_INVALID_ERROR) : OUSTRING_TO_OSTRING_CVTFLAGS))) { throw css::uno::Exception( ("Could not convert \"" + word + "\" to encoding #" + OUString::number(osl_getThreadTextEncoding())), css::uno::Reference()); } buffer.append('\''); for (sal_Int32 i = 0; i != sys.getLength(); ++i) { char c = sys[i]; switch (c) { case 0: if (strict) { throw css::uno::Exception( "Could not convert word containing NUL, \"" + word + "\"", css::uno::Reference()); } break; case '\'': buffer.append("'\\''"); break; default: buffer.append(c); break; } } buffer.append('\''); } } void SAL_CALL CmdMailSuppl::sendSimpleMailMessage( const Reference< XSimpleMailMessage >& xSimpleMailMessage, sal_Int32 /*aFlag*/ ) { if ( ! xSimpleMailMessage.is() ) { throw css::lang::IllegalArgumentException( u"No message specified"_ustr , static_cast < XSimpleMailClient * > (this), 1 ); } if( ! m_xConfigurationProvider.is() ) { throw css::uno::Exception( u"Can not access configuration"_ustr , static_cast < XSimpleMailClient * > (this) ); } OUString aProgramURL(u"$BRAND_BASE_DIR/" LIBO_LIBEXEC_FOLDER "/senddoc"_ustr); rtl::Bootstrap::expandMacros(aProgramURL); OUString aProgram; if ( FileBase::E_None != FileBase::getSystemPathFromFileURL(aProgramURL, aProgram)) { throw css::uno::Exception(u"Could not convert executable path"_ustr, static_cast < XSimpleMailClient * > (this)); } OStringBuffer aBuffer; appendShellWord(aBuffer, aProgram, true); try { // Query XNameAccess interface of the org.openoffice.Office.Common/ExternalMailer // configuration node to retrieve the users preferred email application. This may // transparently by redirected to e.g. the corresponding GConf setting in GNOME. PropertyValue aProperty; aProperty.Name = "nodepath"; aProperty.Value <<= u"org.openoffice.Office.Common/ExternalMailer"_ustr; Sequence< Any > aArgumentList{ Any(aProperty) }; Reference< XNameAccess > xNameAccess( m_xConfigurationProvider->createInstanceWithArguments( u"com.sun.star.configuration.ConfigurationAccess"_ustr, aArgumentList ), UNO_QUERY ); if( xNameAccess.is() ) { OUString aMailer; // Retrieve the value for "Program" node and append it feed senddoc with it // using the (undocumented) --mailclient switch xNameAccess->getByName(u"Program"_ustr) >>= aMailer; if( !aMailer.isEmpty() ) { // make sure we have a system path FileBase::getSystemPathFromFileURL( aMailer, aMailer ); aBuffer.append(" --mailclient "); appendShellWord(aBuffer, aMailer, true); } #ifdef MACOSX else aBuffer.append(" --mailclient Mail"); #endif } } catch(const RuntimeException & ) { TOOLS_WARN_EXCEPTION("shell", "RuntimeException caught accessing configuration provider" ); m_xConfigurationProvider.clear(); throw; } Reference< XSimpleMailMessage2 > xMessage( xSimpleMailMessage, UNO_QUERY ); if ( xMessage.is() ) { OUString sBody = xMessage->getBody(); if ( sBody.getLength() > 0 ) { aBuffer.append(" --body "); appendShellWord(aBuffer, sBody, false); } } // Convert from, to, etc. in a best-effort rather than a strict way to the // system encoding, based on the assumption that the relevant address parts // of those strings are ASCII anyway and any problematic characters are only // in the human-readable, informational-only parts: // Append originator if set in the message if ( !xSimpleMailMessage->getOriginator().isEmpty() ) { aBuffer.append(" --from "); appendShellWord(aBuffer, xSimpleMailMessage->getOriginator(), false); } // Append recipient if set in the message if ( !xSimpleMailMessage->getRecipient().isEmpty() ) { aBuffer.append(" --to "); appendShellWord(aBuffer, xSimpleMailMessage->getRecipient(), false); } // Append carbon copy recipients set in the message Sequence< OUString > aStringList = xSimpleMailMessage->getCcRecipient(); for (const auto& rString : aStringList) { aBuffer.append(" --cc "); appendShellWord(aBuffer, rString, false); } // Append blind carbon copy recipients set in the message aStringList = xSimpleMailMessage->getBccRecipient(); for (const auto& rString : aStringList) { aBuffer.append(" --bcc "); appendShellWord(aBuffer, rString, false); } // Append subject if set in the message if ( !xSimpleMailMessage->getSubject().isEmpty() ) { aBuffer.append(" --subject "); appendShellWord(aBuffer, xSimpleMailMessage->getSubject(), false); } // Append attachments set in the message aStringList = xSimpleMailMessage->getAttachement(); for (const auto& rString : aStringList) { OUString aSystemPath; if ( FileBase::E_None == FileBase::getSystemPathFromFileURL(rString, aSystemPath) ) { aBuffer.append(" --attach "); appendShellWord(aBuffer, aSystemPath, true); } } OString cmd = aBuffer.makeStringAndClear(); FILE * f = popen(cmd.getStr(), "w"); if (f == nullptr || pclose(f) != 0) { throw css::uno::Exception(u"No mail client configured"_ustr, static_cast < XSimpleMailClient * > (this) ); } } // XServiceInfo OUString SAL_CALL CmdMailSuppl::getImplementationName( ) { return u"com.sun.star.comp.system.SimpleCommandMail"_ustr; } sal_Bool SAL_CALL CmdMailSuppl::supportsService( const OUString& ServiceName ) { return cppu::supportsService(this, ServiceName); } Sequence< OUString > SAL_CALL CmdMailSuppl::getSupportedServiceNames( ) { return { u"com.sun.star.system.SimpleCommandMail"_ustr }; } extern "C" SAL_DLLPUBLIC_EXPORT css::uno::XInterface* shell_CmdMailSuppl_get_implementation( css::uno::XComponentContext* context, css::uno::Sequence const&) { return cppu::acquire(new CmdMailSuppl(context)); } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */