/* -*- 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 own header #include #include #include // include others #include #include #include #include #include #include // include interfaces #include #include namespace framework{ /** address job configuration inside argument set provided on method execute(). */ static const char PROP_JOBCONFIG[] = "JobConfig"; /** address job configuration property "Command". */ static const char PROP_COMMAND[] = "Command"; /** address job configuration property "Arguments". */ static const char PROP_ARGUMENTS[] = "Arguments"; /** address job configuration property "DeactivateJobIfDone". */ static const char PROP_DEACTIVATEJOBIFDONE[] = "DeactivateJobIfDone"; /** address job configuration property "CheckExitCode". */ static const char PROP_CHECKEXITCODE[] = "CheckExitCode"; DEFINE_XSERVICEINFO_MULTISERVICE_2(ShellJob , ::cppu::OWeakObject , SERVICENAME_JOB , IMPLEMENTATIONNAME_SHELLJOB) DEFINE_INIT_SERVICE(ShellJob, { /* Attention I think we don't need any mutex or lock here ... because we are called by our own static method impl_createInstance() to create a new instance of this class by our own supported service factory. see macro DEFINE_XSERVICEINFO_MULTISERVICE and "impl_initService()" for further information! */ } ) ShellJob::ShellJob(const css::uno::Reference< css::uno::XComponentContext >& xContext) : m_xContext (xContext) { } ShellJob::~ShellJob() { } css::uno::Any SAL_CALL ShellJob::execute(const css::uno::Sequence< css::beans::NamedValue >& lJobArguments) throw(css::lang::IllegalArgumentException, css::uno::Exception , css::uno::RuntimeException, std::exception ) { ::comphelper::SequenceAsHashMap lArgs (lJobArguments); ::comphelper::SequenceAsHashMap lOwnCfg(lArgs.getUnpackedValueOrDefault(PROP_JOBCONFIG, css::uno::Sequence< css::beans::NamedValue >())); const OUString sCommand = lOwnCfg.getUnpackedValueOrDefault(PROP_COMMAND , OUString()); const css::uno::Sequence< OUString > lCommandArguments = lOwnCfg.getUnpackedValueOrDefault(PROP_ARGUMENTS , css::uno::Sequence< OUString >()); const bool bDeactivateJobIfDone = lOwnCfg.getUnpackedValueOrDefault(PROP_DEACTIVATEJOBIFDONE , true ); const bool bCheckExitCode = lOwnCfg.getUnpackedValueOrDefault(PROP_CHECKEXITCODE , true ); // replace all might existing place holder. OUString sRealCommand = impl_substituteCommandVariables(sCommand); // Command is required as minimum. // If it does not exists ... we can't do our job. // Deactivate such miss configured job silently .-) if (sRealCommand.isEmpty()) return ShellJob::impl_generateAnswer4Deactivation(); // do it bool bDone = impl_execute(sRealCommand, lCommandArguments, bCheckExitCode); if (! bDone) return css::uno::Any(); // Job was done ... user configured deactivation of this job // in such case. if (bDeactivateJobIfDone) return ShellJob::impl_generateAnswer4Deactivation(); // There was no decision about deactivation of this job. // So we have to return nothing here ! return css::uno::Any(); } css::uno::Any ShellJob::impl_generateAnswer4Deactivation() { css::uno::Sequence< css::beans::NamedValue > aAnswer(1); aAnswer[0].Name = JobConst::ANSWER_DEACTIVATE_JOB(); aAnswer[0].Value = css::uno::makeAny(sal_True); return css::uno::makeAny(aAnswer); } OUString ShellJob::impl_substituteCommandVariables(const OUString& sCommand) { try { css::uno::Reference< css::util::XStringSubstitution > xSubst( css::util::PathSubstitution::create(m_xContext) ); const bool bSubstRequired = true; const OUString sCompleteCommand = xSubst->substituteVariables(sCommand, bSubstRequired); return sCompleteCommand; } catch(const css::uno::Exception&) {} return OUString(); } bool ShellJob::impl_execute(const OUString& sCommand , const css::uno::Sequence< OUString >& lArguments , bool bCheckExitCode) { ::rtl_uString** pArgs = NULL; const ::sal_Int32 nArgs = lArguments.getLength (); oslProcessOption nOptions = osl_Process_WAIT; oslProcess hProcess(0); if (nArgs > 0) pArgs = reinterpret_cast< ::rtl_uString** >(const_cast< OUString* >(lArguments.getConstArray())); oslProcessError eError = osl_executeProcess(sCommand.pData, pArgs, nArgs, nOptions, NULL, NULL, NULL, 0, &hProcess); // executable not found or couldnt be started if (eError != osl_Process_E_None) return false; bool bRet = true; if (bCheckExitCode) { // check its return codes ... oslProcessInfo aInfo; aInfo.Size = sizeof (oslProcessInfo); eError = osl_getProcessInfo(hProcess, osl_Process_EXITCODE, &aInfo); if (eError != osl_Process_E_None) bRet = false; else bRet = (aInfo.Code == 0); } osl_freeProcessHandle(hProcess); return bRet; } } // namespace framework /* vim:set shiftwidth=4 softtabstop=4 expandtab: */