The Gaudi Framework  v30r3 (a5ef0a68)
SplittingTransformer.h
Go to the documentation of this file.
1 #ifndef SPLITTING_TRANSFORMER_H
2 #define SPLITTING_TRANSFORMER_H
3 
4 #include <functional>
5 #include <string>
6 #include <vector>
7 
10 #include "GaudiKernel/apply.h"
11 
12 namespace Gaudi
13 {
14  namespace Functional
15  {
16 
17  template <typename Signature, typename Traits_ = Traits::useDefaults>
19 
20  template <typename Container>
22  template <typename Container>
24 
26  template <typename Out, typename... In, typename Traits_>
27  class SplittingTransformer<vector_of_<Out>( const In&... ), Traits_>
29  {
31 
32  public:
33  constexpr static std::size_t N = sizeof...( In );
36 
38  const KeyValues& output );
39 
40  SplittingTransformer( const std::string& name, ISvcLocator* locator, const KeyValue& input,
41  const KeyValues& output )
42  : SplittingTransformer( name, locator, std::array<KeyValue, 1>{input}, output )
43  {
44  static_assert( N == 1, "single input argument requires single input signature" );
45  }
46 
47  // accessor to output Locations
48  const std::string& outputLocation( unsigned int n ) const { return m_outputLocations[n]; }
49  unsigned int outputLocationSize() const { return m_outputLocations.size(); }
50 
51  // derived classes can NOT implement execute
52  StatusCode execute() override final
53  {
54  try {
55  // TODO:FIXME: how does operator() know the number and order of expected outputs?
56  using details::as_const;
57  auto out = details::filter_evtcontext_t<In...>::apply( *this, this->m_inputs );
58  if ( out.size() != m_outputs.size() ) {
59  throw GaudiException( "Error during transform: expected " + std::to_string( m_outputs.size() ) +
60  " containers, got " + std::to_string( out.size() ) + " instead",
61  this->name(), StatusCode::FAILURE );
62  }
63  for ( unsigned i = 0; i != out.size(); ++i ) details::put( m_outputs[i], std::move( out[i] ) );
64  return StatusCode::SUCCESS;
65  } catch ( GaudiException& e ) {
66  ( e.code() ? this->warning() : this->error() ) << e.message() << endmsg;
67  return e.code();
68  }
69  }
70 
71  // TODO/FIXME: how does the callee know in which order to produce the outputs?
72  // (note: 'missing' items can be specified by making Out an boost::optional<Out>,
73  // and only those entries which contain an Out are stored)
74  virtual vector_of_<Out> operator()( const In&... ) const = 0;
75 
76  private:
77  template <typename T>
79  std::vector<std::string> m_outputLocations; // TODO/FIXME for now: use a call-back to update the actual handles!
81  };
82 
83  template <typename Out, typename... In, typename Traits_>
85  const std::string& name, ISvcLocator* pSvcLocator, const std::array<KeyValue, N>& inputs,
86  const KeyValues& outputs )
87  : base_class( name, pSvcLocator, inputs ), m_outputLocations( outputs.second )
88  {
89  auto p = this->declareProperty( outputs.first, m_outputLocations );
90  p->declareUpdateHandler( [=]( Gaudi::Details::PropertyBase& ) {
91  this->m_outputs = details::make_vector_of_handles<decltype( this->m_outputs )>( this, m_outputLocations );
92  if ( details::is_optional<Out>::value ) { // handle constructor does not (yet) allow to set optional flag... so
93  // do it explicitly here...
94  std::for_each( this->m_outputs.begin(), this->m_outputs.end(), []( auto& h ) { h.setOptional( true ); } );
95  }
96  } );
97  p->useUpdateHandler(); // invoke now, to be sure the input handles are synced with the property...
98  }
99  }
100 }
101 
102 #endif
virtual const std::string & message() const
error message to be printed
constexpr static const auto FAILURE
Definition: StatusCode.h:88
Define general base for Gaudi exception.
The ISvcLocator is the interface implemented by the Service Factory in the Application Manager to loc...
Definition: ISvcLocator.h:25
T to_string(T...args)
STL namespace.
details::OutputHandle_t< Traits_, details::remove_optional_t< T >> OutputHandle
class MergingTransformer< Out(const vector_of_const_< In > void
virtual const StatusCode & code() const
StatusCode for Exception.
constexpr double second
STL class.
int N
Definition: IOTest.py:101
This class is used for returning status codes from appropriate routines.
Definition: StatusCode.h:51
Gaudi::Details::PropertyBase * declareProperty(const std::string &name, TYPE &value, const std::string &doc="none")
Declare a property (templated)
SplittingTransformer(const std::string &name, ISvcLocator *locator, const KeyValue &input, const KeyValues &output)
static auto apply(const Algorithm &algo, Handles &handles)
details::template vector_of_const_< Container > vector_of_
PropertyBase base class allowing PropertyBase* collections to be "homogeneous".
Definition: Property.h:32
T move(T...args)
constexpr static const auto SUCCESS
Definition: StatusCode.h:87
constexpr std::add_const_t< T > & as_const(T &t) noexcept
std::vector< InputHandle_t< In > > m_inputs
struct GAUDI_API array
Parametrisation class for redirection array - like implementation.
STL class.
virtual Out operator()(const vector_of_const_< In > &inputs) const =0
Gaudi::cpp17::detected_or_t< DataObjectWriteHandle< T >, detail2::OutputHandle_t, Tr, T > OutputHandle_t
STL class.
struct[[deprecated("use MergingTransformer instead")]] Traits_
T for_each(T...args)
Handles make_vector_of_handles(IDataHandleHolder *owner, const std::vector< std::string > &init)
Helper functions to set/get the application return code.
Definition: __init__.py:1
MsgStream & endmsg(MsgStream &s)
MsgStream Modifier: endmsg. Calls the output method of the MsgStream.
Definition: MsgStream.h:209
Out1 * put(DataObjectHandle< Out1 > &out_handle, Out2 &&out)