diff --git a/Trigger/TrigAnalysis/TrigDecisionTool/TrigDecisionTool/ChainGroup.icc b/Trigger/TrigAnalysis/TrigDecisionTool/TrigDecisionTool/ChainGroup.icc
index 35aa5a9a542cb1f2e1ea83687ea17025ac927e40..880d797b4781b528692210eeaf661355902da34b 100644
--- a/Trigger/TrigAnalysis/TrigDecisionTool/TrigDecisionTool/ChainGroup.icc
+++ b/Trigger/TrigAnalysis/TrigDecisionTool/TrigDecisionTool/ChainGroup.icc
@@ -45,53 +45,77 @@ std::vector< TrigCompositeUtils::LinkInfo<CONTAINER> > Trig::ChainGroup::feature
     return std::vector< TrigCompositeUtils::LinkInfo<CONTAINER> >();
   }
 
-  // For each chain, collect Navigation information
-  TrigCompositeUtils::NavGraph navGraph;
+  // The sub-graph from which we will extract features
+  TrigCompositeUtils::NavGraph navGraph; 
 
-  // Loop over HLT chains
-  TrigCompositeUtils::DecisionIDContainer chainIDs;
+  // Collect the set of chains (and chain legs) which we are fetching
+  // Perform the fetches using the full set of IDs for each chain (include all legs)
+  TrigCompositeUtils::DecisionIDContainer allRequestedChainIDs;
   std::set<const TrigConf::HLTChain*>::const_iterator chIt;
   for (chIt=conf_chain_begin(); chIt != conf_chain_end(); ++chIt) {
-    const HLT::Chain* fchain = cgm()->chain(**chIt);
-    if (fchain) {
-      chainIDs.insert( fchain->getChainHashId() );
 
-      // Obtain navigation routes for passed chains
-      // Final parameter TRUE as the chain passed (has its ID in terminusNode)
-      TrigCompositeUtils::recursiveGetDecisions(terminusNode, navGraph, fchain->getChainHashId(), true);
+    TrigCompositeUtils::DecisionIDContainer thisChainIDs;
+    HLT::Identifier chainID("");
 
-      ATH_MSG_DEBUG("Added all passed navigation data for chain " << fchain->getChainName() 
-        << ", total nodes:" << navGraph.nodes() << " total edges:" << navGraph.edges() << " final nodes:" << navGraph.finalNodes().size());
-      if (msg().level() <= MSG::DEBUG && navGraph.finalNodes().size()) {
-        for (const TrigCompositeUtils::NavGraphNode* n : navGraph.finalNodes()) {
-          ATH_MSG_DEBUG("  Final node:" << TrigCompositeUtils::decisionToElementLink(n->node()).dataID() << " #" << n->node()->index());
+    const HLT::Chain* fchain = cgm()->chain(**chIt);
+    if (fchain) {
+      chainID = HLT::Identifier( fchain->getChainName() );
+      const std::vector<size_t> legMultiplicites = fchain->getLegMultiplicities();
+      allRequestedChainIDs.insert( chainID.numeric() );
+      thisChainIDs.insert( chainID.numeric() );
+      if (legMultiplicites.size() == 0) {
+        ATH_MSG_ERROR("chain " << chainID << " has invalid configuration, no multiplicity data.");
+      } else if (legMultiplicites.size() > 1) {
+        // For multi-leg chains, the DecisionIDs are handled per leg.
+        // We don't care here exactly how many objects are required per leg, just that there are two-or-more legs
+        for (size_t legNumeral = 0; legNumeral < legMultiplicites.size(); ++legNumeral) {
+          HLT::Identifier legID = TrigCompositeUtils::createLegName(chainID, legNumeral);
+          allRequestedChainIDs.insert( legID.numeric() );
+          thisChainIDs.insert( legID.numeric() );
         }
       }
+      ATH_MSG_DEBUG("Adding navigation data for chain " << chainID << " with " << legMultiplicites.size() << " leg(s)." );
+      if (msg().level() <= MSG::VERBOSE) {
+        for (const TrigCompositeUtils::DecisionID printID : thisChainIDs) {
+          ATH_MSG_VERBOSE(" -- Collecting for chain or chain-leg: " << HLT::Identifier(printID));
+        }
+      }
+    } else {
+      ATH_MSG_ERROR("Cannot access configuration for one of the ChainGroup's chains");
+      continue;
+    }
 
-      if (condition == TrigDefs::includeFailedDecisions) {
-        std::vector<const TrigCompositeUtils::Decision*> rejectedDecisionNodes = 
-          TrigCompositeUtils::getRejectedDecisionNodes(eventStore, fchain->getChainHashId());
+    // Obtain navigation routes for objects which pass
+    // Final parameter TRUE as the chain passed (has its ID in terminusNode)
+    TrigCompositeUtils::recursiveGetDecisions(terminusNode, navGraph, thisChainIDs, true);
 
-        ATH_MSG_DEBUG("Chain " << fchain->getChainName() << " has " << rejectedDecisionNodes.size() 
-          << " dangling nodes in the graph from objects which were rejected.");
+    ATH_MSG_DEBUG("Added all passed navigation data for chain " << chainID
+      << ", total nodes:" << navGraph.nodes() << " total edges:" << navGraph.edges() << " final nodes:" << navGraph.finalNodes().size());
 
-        for (const TrigCompositeUtils::Decision* rejectedNode : rejectedDecisionNodes) {
-          // Final parameter FALSE as the chain failed here (its ID was removed from rejectedNode)
-          TrigCompositeUtils::recursiveGetDecisions(rejectedNode, navGraph, fchain->getChainHashId(), false);
-        }
+    // Obtain navigation routes for objects which fail
+    if (condition == TrigDefs::includeFailedDecisions) {
+      std::vector<const TrigCompositeUtils::Decision*> rejectedDecisionNodes = 
+        TrigCompositeUtils::getRejectedDecisionNodes(eventStore, thisChainIDs);
 
-        ATH_MSG_DEBUG("Added all failed navigation data for chain " << fchain->getChainName() 
-          << ", total nodes:" << navGraph.nodes() << " total edges:" << navGraph.edges() << " final nodes:" << navGraph.finalNodes().size());
-        if (msg().level() <= MSG::DEBUG && navGraph.finalNodes().size()) {
-          for (const TrigCompositeUtils::NavGraphNode* n : navGraph.finalNodes()) {
-            ATH_MSG_DEBUG("  Final node:" << TrigCompositeUtils::decisionToElementLink(n->node()).dataID() << " #" << n->node()->index());
-          }
-        }
+      ATH_MSG_DEBUG("Chain " << chainID << " has " << rejectedDecisionNodes.size() 
+        << " dangling nodes in the graph from objects which were rejected.");
 
+      for (const TrigCompositeUtils::Decision* rejectedNode : rejectedDecisionNodes) {
+        // Final parameter FALSE as the chain failed here (its ID was removed from rejectedNode)
+        TrigCompositeUtils::recursiveGetDecisions(rejectedNode, navGraph, thisChainIDs, false);
       }
 
-    } else {
-      ATH_MSG_ERROR("Cannot access configuration for one of the ChainGroup's chains");
+      ATH_MSG_DEBUG("Added all failed navigation data for chain " << chainID
+        << ", total nodes:" << navGraph.nodes() << " total edges:" << navGraph.edges() << " final nodes:" << navGraph.finalNodes().size());
+    }
+
+  }
+
+  ATH_MSG_DEBUG("Finished adding nodes to sub-graph. "
+    << ", total nodes:" << navGraph.nodes() << " total edges:" << navGraph.edges() << " final nodes:" << navGraph.finalNodes().size());
+  if (msg().level() <= MSG::DEBUG && navGraph.finalNodes().size()) {
+    for (const TrigCompositeUtils::NavGraphNode* n : navGraph.finalNodes()) {
+      ATH_MSG_DEBUG("  Final node:" << TrigCompositeUtils::decisionToElementLink(n->node()).dataID() << " #" << n->node()->index());
     }
   }
 
@@ -107,7 +131,7 @@ std::vector< TrigCompositeUtils::LinkInfo<CONTAINER> > Trig::ChainGroup::feature
   const bool lastFeatureOfTypeFlag = (featureCollectionMode == TrigDefs::lastFeatureOfType);
 
   std::vector<TrigCompositeUtils::LinkInfo<CONTAINER>> returnVector =
-    TrigCompositeUtils::recursiveGetFeaturesOfType<CONTAINER>(navGraph, containerSGKey, lastFeatureOfTypeFlag, navElementLinkKey, chainIDs);
+    TrigCompositeUtils::recursiveGetFeaturesOfType<CONTAINER>(navGraph, containerSGKey, lastFeatureOfTypeFlag, navElementLinkKey, allRequestedChainIDs);
 
   // Check for missing navigation data if requesting the default "feature" links
   if (navElementLinkKey == TrigCompositeUtils::featureString()) {
diff --git a/Trigger/TrigConfiguration/TrigConfData/TrigConfData/HLTChain.h b/Trigger/TrigConfiguration/TrigConfData/TrigConfData/HLTChain.h
index d15e87769469909e1d57c45258caa8f39959b7c4..c5b426d4665c32ff9b12801c9f58b665d1ccaa10 100644
--- a/Trigger/TrigConfiguration/TrigConfData/TrigConfData/HLTChain.h
+++ b/Trigger/TrigConfiguration/TrigConfData/TrigConfData/HLTChain.h
@@ -50,6 +50,9 @@ namespace TrigConf {
       /** Accessor to the l1 thresholds */
       std::vector<std::string> l1thresholds() const;
 
+      /** Accessor to the chains multiplicitiy requirements for each of its legs */
+      std::vector<size_t> legMultiplicities() const;
+
       /** Accessor to the connected output streams */
       std::vector<std::string> streams() const;
 
diff --git a/Trigger/TrigConfiguration/TrigConfData/src/HLTChain.cxx b/Trigger/TrigConfiguration/TrigConfData/src/HLTChain.cxx
index d85d27eb34da9b7924f2f6f2e467f0a8a80e985e..35bf9e37c25d0741b9c88c91bec78458edd1020c 100644
--- a/Trigger/TrigConfiguration/TrigConfData/src/HLTChain.cxx
+++ b/Trigger/TrigConfiguration/TrigConfData/src/HLTChain.cxx
@@ -61,6 +61,17 @@ TrigConf::Chain::l1item() const
    return getAttribute("l1item");
 }
 
+std::vector<size_t> TrigConf::Chain::legMultiplicities() const {
+   std::vector<size_t> returnMultiplicities;
+   const auto& theMultiplicities = getList("legMultiplicities");
+   if( !theMultiplicities.empty() ) {
+      returnMultiplicities.reserve(theMultiplicities.size());
+      for( auto& m : theMultiplicities ) {
+         returnMultiplicities.push_back( m.getValue<size_t>() );
+      }
+   } 
+   return returnMultiplicities;
+}
 
 std::vector<std::string>
 TrigConf::Chain::l1thresholds() const
diff --git a/Trigger/TrigConfiguration/TrigConfHLTData/TrigConfHLTData/HLTChain.h b/Trigger/TrigConfiguration/TrigConfHLTData/TrigConfHLTData/HLTChain.h
index 336a5a6b41ca5aa1ad6c0461de1ef765a067c849..cdb9c66b576c7cea590e6ef78f8b9c92e7ca79db 100644
--- a/Trigger/TrigConfiguration/TrigConfHLTData/TrigConfHLTData/HLTChain.h
+++ b/Trigger/TrigConfiguration/TrigConfHLTData/TrigConfHLTData/HLTChain.h
@@ -80,6 +80,7 @@ namespace TrigConf {
       unsigned int                        chain_hash_id         () const { return m_chain_hash_id; }
       unsigned int                        lower_chain_hash_id   () const { return m_lower_chain_hash_id; }
       int                                 EB_after_step         () const { return m_EB_after_step; }
+      const std::vector<size_t>           leg_multiplicities    () const { return m_leg_multiplicities; }
       bool                                hasMultipleLowerChains() const;
       const std::vector<int>&             lower_chain_counters  () const;
       std::vector<unsigned int>           lower_chain_hash_ids  () const;
@@ -100,6 +101,7 @@ namespace TrigConf {
       HLTChain& set_triggerTypeList      ( const std::vector<HLTTriggerType*>& trigList) { m_HLTTriggerTypeList = trigList; return *this; }
       HLTChain& set_groupList            ( const std::set<std::string>& groups) { m_groups = groups; return *this; }
       HLTChain& set_EB_after_step        ( int EB_after_step ) { m_EB_after_step = EB_after_step; return *this; }
+      HLTChain& set_leg_multiplicities   ( const std::vector<size_t>& mult ) { m_leg_multiplicities = mult; return *this; }
 
 
       // signatures
@@ -175,6 +177,7 @@ namespace TrigConf {
       std::vector<int>  m_lower_chain_counters;//!< counters of the lower trigger items if more than 1
       unsigned int      m_lower_chain_hash_id; //!< hash value from m_lower_chain_name, this is used to match to a chain from the previous trigger level
       int               m_EB_after_step;       //!< EB_after_step flag 
+      std::vector<size_t> m_leg_multiplicities;//!< Number of objects required per leg. NOTE: Run3 only quantity
       HLTPrescale       m_prescales;
 
       std::vector<HLTSignature*>                         m_HLTSignatureList;
diff --git a/Trigger/TrigConfiguration/TrigConfxAOD/Root/prepareTriggerMenu.cxx b/Trigger/TrigConfiguration/TrigConfxAOD/Root/prepareTriggerMenu.cxx
index 51fbb0cbcfddb0580c7577255cd22da7c5388169..ad75980a9ca3d4401cb2679a2251a3f2a0d28cbe 100644
--- a/Trigger/TrigConfiguration/TrigConfxAOD/Root/prepareTriggerMenu.cxx
+++ b/Trigger/TrigConfiguration/TrigConfxAOD/Root/prepareTriggerMenu.cxx
@@ -339,6 +339,7 @@ namespace TrigConf {
          chain->set_rerun_prescale( -1.0 ); // Not used in R3
          chain->set_pass_through( -1.0 );  // Not used in R3
          chain->set_prescale( loadedPrescale.prescale );
+         chain->set_leg_multiplicities( loadedChain.legMultiplicities() );
 
          // Add it to the list of chains:
          if( ! chainList.addHLTChain( chain ) ) {
diff --git a/Trigger/TrigEvent/TrigSteeringEvent/TrigSteeringEvent/Chain.h b/Trigger/TrigEvent/TrigSteeringEvent/TrigSteeringEvent/Chain.h
index cf0ae8609e4d376229c180d759a252777f07c21b..b62350c1c3aeb0220f96d44d064e937e02562678 100644
--- a/Trigger/TrigEvent/TrigSteeringEvent/TrigSteeringEvent/Chain.h
+++ b/Trigger/TrigEvent/TrigSteeringEvent/TrigSteeringEvent/Chain.h
@@ -95,6 +95,7 @@ namespace HLT {
     const std::string&  getChainName()      const { return (m_configChain ? m_configChain->chain_name(): no_config); }    //!< return the Chain name (string)
     const std::string&  getLowerChainName() const { return (m_configChain ? m_configChain->lower_chain_name(): no_config); }    //!< return the Chain name (string)
     int  getEBAfterStep()                   const { return (m_configChain ? m_configChain->EB_after_step() : -1.); }    //!< get EB_after_step  
+    std::vector<size_t> getLegMultiplicities() const { return (m_configChain ? m_configChain->leg_multiplicities() : std::vector<size_t>()); }    //!< get per leg multiplicity (Run3 only)
     bool nextStepAfterEB()                  const { return ((getChainStep()+1) > getEBAfterStep()) && (getEBAfterStep()>0.); } //!< return whether next step requires EB (-1 means no EB called)
     bool isMerged()                         const { return (m_configChain ? (m_configChain->level()=="HLT") : false);}; //!<< return whether is a merged L2+EF chain 
 
diff --git a/Trigger/TrigSteer/TrigCompositeUtils/Root/TrigCompositeUtilsRoot.cxx b/Trigger/TrigSteer/TrigCompositeUtils/Root/TrigCompositeUtilsRoot.cxx
index 9ef63ae2eb7433b838efc25d7bf84b23bea9bdd5..8b3529bf22f0765a894073adf1acc65d0c008814 100644
--- a/Trigger/TrigSteer/TrigCompositeUtils/Root/TrigCompositeUtilsRoot.cxx
+++ b/Trigger/TrigSteer/TrigCompositeUtils/Root/TrigCompositeUtilsRoot.cxx
@@ -215,7 +215,7 @@ namespace TrigCompositeUtils {
     return composite->hasObjectCollectionLinks( m_name );
   }
 
- std::vector<const Decision*> getRejectedDecisionNodes(asg::EventStoreType* eventStore, const DecisionID id) {
+ std::vector<const Decision*> getRejectedDecisionNodes(asg::EventStoreType* eventStore, const DecisionIDContainer ids) {
     std::vector<const Decision*> output;
     // The list of containers we need to read can change on a file-by-file basis (it depends on the SMK)
     // Hence we query SG for all collections rather than maintain a large and ever changing ReadHandleKeyArray
@@ -276,10 +276,10 @@ namespace TrigCompositeUtils {
         // So the size of activeChainsIntoThisDecision corresponds to the number of HypoTools which will have run
         // What do we care about? A chain, or all chains?
         DecisionIDContainer chainsToCheck;
-        if (id == 0) { // We care about *all* chains
+        if (ids.size() == 0) { // We care about *all* chains
           chainsToCheck = activeChainsIntoThisDecision;
-        } else { // We care about *one* chain
-          chainsToCheck.insert(id);
+        } else { // We care about sepcified chains
+          chainsToCheck = ids;
         }
         // We have found a rejected decision node *iff* a chainID to check is *not* present here
         // I.e. the HypoTool for the chain returned a NEGATIVE decision
@@ -300,12 +300,11 @@ namespace TrigCompositeUtils {
   void recursiveGetDecisionsInternal(const Decision* node, 
     const Decision* comingFrom, 
     NavGraph& navGraph, 
-    const DecisionID id,
+    const DecisionIDContainer ids,
     const bool enforceDecisionOnNode) {
 
     // Does this Decision satisfy the chain requirement?
-    DecisionIDContainer idSet = {id};
-    if (enforceDecisionOnNode && id != 0 && !isAnyIDPassing(node, idSet)) {
+    if (enforceDecisionOnNode && ids.size() != 0 && !isAnyIDPassing(node, ids)) {
       return; // Stop propagating down this leg. It does not concern the chain with DecisionID = id
     }
 
@@ -318,7 +317,7 @@ namespace TrigCompositeUtils {
       for ( ElementLink<DecisionContainer> seed : getLinkToPrevious(node)) {
         const Decision* seedDecision = *(seed); // Dereference ElementLink
         // Sending true as final parameter for enforceDecisionOnStartNode as we are recursing away from the supplied start node
-        recursiveGetDecisionsInternal(seedDecision, node, navGraph, id, /*enforceDecisionOnNode*/ true);
+        recursiveGetDecisionsInternal(seedDecision, node, navGraph, ids, /*enforceDecisionOnNode*/ true);
       }
     }
     return;
@@ -326,11 +325,11 @@ namespace TrigCompositeUtils {
 
   void recursiveGetDecisions(const Decision* start, 
     NavGraph& navGraph, 
-    const DecisionID id,
+    const DecisionIDContainer ids,
     const bool enforceDecisionOnStartNode) {
 
     // Note: we do not require navGraph to be an empty graph. We can extend it.
-    recursiveGetDecisionsInternal(start, /*comingFrom*/nullptr, navGraph, id, enforceDecisionOnStartNode);
+    recursiveGetDecisionsInternal(start, /*comingFrom*/nullptr, navGraph, ids, enforceDecisionOnStartNode);
     
     return;
   }
diff --git a/Trigger/TrigSteer/TrigCompositeUtils/TrigCompositeUtils/TrigCompositeUtils.h b/Trigger/TrigSteer/TrigCompositeUtils/TrigCompositeUtils/TrigCompositeUtils.h
index 33ccb51ffcfc8a0841badee125e256f26a7d6bcc..64ea8b34ea3e283ea8b555cad3643e731409b653 100644
--- a/Trigger/TrigSteer/TrigCompositeUtils/TrigCompositeUtils/TrigCompositeUtils.h
+++ b/Trigger/TrigSteer/TrigCompositeUtils/TrigCompositeUtils/TrigCompositeUtils.h
@@ -256,10 +256,10 @@ namespace TrigCompositeUtils {
   /**
    * @brief Query all DecisionCollections in the event store, locate all Decision nodes in the graph where an object failed selection for a given chain.
    * @param[in] eventStore Pointer to event store within current event context
-   * @param[in] id ID of chain to located failed decision nodes for. Passing 0 returns all decision nodes which failed at least one chain.
+   * @param[in] ids IDs of chain (if multi-leg chain, include all legs) to located failed decision nodes for. Passing an empty set returns all decision nodes which failed at least one chain.
    * @return Vector of Decision nodes whose attached feature failed the trigger chain logic for chain with DecisionID id
    **/
-  std::vector<const Decision*> getRejectedDecisionNodes(asg::EventStoreType* eventStore, const DecisionID id = 0);
+  std::vector<const Decision*> getRejectedDecisionNodes(asg::EventStoreType* eventStore, const DecisionIDContainer ids = {});
   
 
 
@@ -268,14 +268,14 @@ namespace TrigCompositeUtils {
    * @brief Search back in time from "node" and locate all paths back through Decision objects for a given chain.
    * @param[in] node The Decision object to start the search from. Typically this will be one of the terminus objects from the HLTNav_Summary.
    * @param[inout] navPaths Holds a sub-graph of the full navigation graph, filtered by DecisionID. An already partially populated graph may be provided as input.
-   * @param[in] id Optional DecisionID of a Chain to trace through the navigation. If omitted, no chain requirement will be applied.
+   * @param[in] ids Optional DecisionIDContainer of Chains / Chain-Legs to trace through the navigation. If omitted, no chain requirement will be applied.
    * @param[in] enforceDecisionOnStartNode If the check of DecisionID should be carried out on the start node.
    * enforceDecisionOnStartNode should be true if navigating for a trigger which passed (e.g. starting from HLTPassRaw)
    * enforceDecisionOnStartNode should be false if navigating for a trigger which failed but whose failing start node(s) were recovered via getRejectedDecisionNodes
    **/
   void recursiveGetDecisions(const Decision* node, 
     NavGraph& navGraph, 
-    const DecisionID id = 0,
+    const DecisionIDContainer ids = {},
     const bool enforceDecisionOnStartNode = true);
 
 
@@ -287,7 +287,7 @@ namespace TrigCompositeUtils {
   void recursiveGetDecisionsInternal(const Decision* node, 
     const Decision* comingFrom,
     NavGraph& navGraph,
-    const DecisionID id,
+    const DecisionIDContainer ids,
     const bool enforceDecisionOnNode);
 
   /**
diff --git a/Trigger/TrigSteer/TrigCompositeUtils/test/TrigTraversal_test.cxx b/Trigger/TrigSteer/TrigCompositeUtils/test/TrigTraversal_test.cxx
index 5a80227944808ae7f0a78701430876d9e9d558c7..0a2b450c92053df70b7fb09a67c0afbe9407234e 100644
--- a/Trigger/TrigSteer/TrigCompositeUtils/test/TrigTraversal_test.cxx
+++ b/Trigger/TrigSteer/TrigCompositeUtils/test/TrigTraversal_test.cxx
@@ -331,11 +331,11 @@ int main ATLAS_NOT_THREAD_SAFE () {
   NavGraph graph_HLT_em_chain;
   NavGraph graph_HLT_all;
 
-  recursiveGetDecisions(END, graph_HLT_mufast_chain, HLT_mufast_chain, true);
-  recursiveGetDecisions(END, graph_HLT_mu_chain, HLT_mu_chain, true);
-  recursiveGetDecisions(END, graph_HLT_mu_em_chain, HLT_mu_em_chain, true);
-  recursiveGetDecisions(END, graph_HLT_em_chain, HLT_em_chain, true);
-  recursiveGetDecisions(END, graph_HLT_all, 0, true);
+  recursiveGetDecisions(END, graph_HLT_mufast_chain, {HLT_mufast_chain}, true);
+  recursiveGetDecisions(END, graph_HLT_mu_chain, {HLT_mu_chain}, true);
+  recursiveGetDecisions(END, graph_HLT_mu_em_chain, {HLT_mu_em_chain}, true);
+  recursiveGetDecisions(END, graph_HLT_em_chain, {HLT_em_chain}, true);
+  recursiveGetDecisions(END, graph_HLT_all, {}, true);
 
 
   log << MSG::INFO << "HLT_mufast_chain" << endmsg;
@@ -375,26 +375,26 @@ int main ATLAS_NOT_THREAD_SAFE () {
 
   std::cout << " ---------- Now Include Failing Features " << std::endl;
 
-  std::vector<const Decision*> extraStart_HLT_mufast_chain = getRejectedDecisionNodes(pSG, HLT_mufast_chain);
-  std::vector<const Decision*> extraStart_HLT_mu_chain = getRejectedDecisionNodes(pSG, HLT_mu_chain);
-  std::vector<const Decision*> extraStart_HLT_mu_em_chain = getRejectedDecisionNodes(pSG, HLT_mu_em_chain);
-  std::vector<const Decision*> extraStart_HLT_em_chain = getRejectedDecisionNodes(pSG, HLT_em_chain);
-  std::vector<const Decision*> extraStart_HLT_all = getRejectedDecisionNodes(pSG, 0);
+  std::vector<const Decision*> extraStart_HLT_mufast_chain = getRejectedDecisionNodes(pSG, {HLT_mufast_chain});
+  std::vector<const Decision*> extraStart_HLT_mu_chain = getRejectedDecisionNodes(pSG, {HLT_mu_chain});
+  std::vector<const Decision*> extraStart_HLT_mu_em_chain = getRejectedDecisionNodes(pSG, {HLT_mu_em_chain});
+  std::vector<const Decision*> extraStart_HLT_em_chain = getRejectedDecisionNodes(pSG, {HLT_em_chain});
+  std::vector<const Decision*> extraStart_HLT_all = getRejectedDecisionNodes(pSG, {});
 
   for (const Decision* d : extraStart_HLT_mufast_chain) {
-    recursiveGetDecisions(d, graph_HLT_mufast_chain, HLT_mufast_chain, false);
+    recursiveGetDecisions(d, graph_HLT_mufast_chain, {HLT_mufast_chain}, false);
   }
   for (const Decision* d : extraStart_HLT_mu_chain) {
-    recursiveGetDecisions(d, graph_HLT_mu_chain, HLT_mu_chain, false);
+    recursiveGetDecisions(d, graph_HLT_mu_chain, {HLT_mu_chain}, false);
   }
   for (const Decision* d : extraStart_HLT_mu_em_chain) {
-    recursiveGetDecisions(d, graph_HLT_mu_em_chain, HLT_mu_em_chain, false);
+    recursiveGetDecisions(d, graph_HLT_mu_em_chain, {HLT_mu_em_chain}, false);
   }
   for (const Decision* d : extraStart_HLT_em_chain) {
-    recursiveGetDecisions(d, graph_HLT_em_chain, HLT_em_chain, false);
+    recursiveGetDecisions(d, graph_HLT_em_chain, {HLT_em_chain}, false);
   }
   for (const Decision* d : extraStart_HLT_all) {
-    recursiveGetDecisions(d, graph_HLT_all, 0, false);
+    recursiveGetDecisions(d, graph_HLT_all, {}, false);
   }
 
   log << MSG::INFO << "HLT_mufast_chain" << endmsg;
diff --git a/Trigger/TriggerCommon/TriggerJobOpts/python/TriggerConfig.py b/Trigger/TriggerCommon/TriggerJobOpts/python/TriggerConfig.py
index 066e6eaa34fb56551845d82bf44ba4692224e543..d341f13a8c47c5ee577a6096c21e0adf9fa786e2 100644
--- a/Trigger/TriggerCommon/TriggerJobOpts/python/TriggerConfig.py
+++ b/Trigger/TriggerCommon/TriggerJobOpts/python/TriggerConfig.py
@@ -427,6 +427,8 @@ def triggerPOOLOutputCfg(flags, edmSet):
     menuwriter = CompFactory.getComp("TrigConf::xAODMenuWriterMT")()
     menuwriter.IsHLTJSONConfig = True
     menuwriter.IsL1JSONConfig = True
+    menuwriter.WritexAODTriggerMenu = True # This should be removed in the future
+    menuwriter.WritexAODTriggerMenuJson = True
     menuwriter.KeyWriterTool = CompFactory.getComp('TrigConf::KeyWriterTool')('KeyWriterToolOffline')
     acc.addEventAlgo( menuwriter )
 
@@ -436,7 +438,11 @@ def triggerPOOLOutputCfg(flags, edmSet):
     acc.merge( L1PrescaleCondAlgCfg( flags ) )
 
     # Add metadata to the output stream
-    streamAlg.MetadataItemList += [ "xAOD::TriggerMenuContainer#*", "xAOD::TriggerMenuAuxContainer#*" ]
+    if menuwriter.WritexAODTriggerMenu:
+      streamAlg.MetadataItemList += [ "xAOD::TriggerMenuContainer#*", "xAOD::TriggerMenuAuxContainer#*" ]
+
+    if menuwriter.WritexAODTriggerMenuJson:
+      streamAlg.MetadataItemList += [ "xAOD::TriggerMenuJsonContainer#*", "xAOD::TriggerMenuJsonAuxContainer#*" ]
 
     # Ensure OutputStream runs after TrigDecisionMakerMT and xAODMenuWriterMT
     streamAlg.ExtraInputs += [
diff --git a/Trigger/TriggerCommon/TriggerJobOpts/python/TriggerConfigGetter.py b/Trigger/TriggerCommon/TriggerJobOpts/python/TriggerConfigGetter.py
index f72402cdecbf8f792b62e57bb8ea1a89051011e9..7000180542862e648a1f2c20831064a91a64a441 100644
--- a/Trigger/TriggerCommon/TriggerJobOpts/python/TriggerConfigGetter.py
+++ b/Trigger/TriggerCommon/TriggerJobOpts/python/TriggerConfigGetter.py
@@ -476,16 +476,16 @@ class TriggerConfigGetter(Configured):
                 objKeyStore.addManyTypesMetaData( metadataItems )
 
             if writeMenuJSON:
-                metadataItems = [ "xAOD::TriggerMenuJSONContainer#MenuJSON_HLT",
-                                  "xAOD::TriggerMenuJSONAuxContainer#MenuJSON_HLTAux.",
-                                  "xAOD::TriggerMenuJSONContainer#MenuJSON_L1",
-                                  "xAOD::TriggerMenuJSONAuxContainer#MenuJSON_L1Aux.",
-                                  "xAOD::TriggerMenuJSONContainer#MenuJSON_HLTPS",
-                                  "xAOD::TriggerMenuJSONAuxContainer#MenuJSON_HLTPSAux.",
-                                  "xAOD::TriggerMenuJSONContainer#MenuJSON_L1PS",
-                                  "xAOD::TriggerMenuJSONAuxContainer#MenuJSON_L1PSAux.",
-                                  # "xAOD::TriggerMenuJSONContainer#MenuJSON_BG", // TODO
-                                  # "xAOD::TriggerMenuJSONAuxContainer#MenuJSON_BGAux.", // TODO
+                metadataItems = [ "xAOD::TriggerMenuJsonContainer#MenuJSON_HLT",
+                                  "xAOD::TriggerMenuJsonAuxContainer#MenuJSON_HLTAux.",
+                                  "xAOD::TriggerMenuJsonContainer#MenuJSON_L1",
+                                  "xAOD::TriggerMenuJsonAuxContainer#MenuJSON_L1Aux.",
+                                  "xAOD::TriggerMenuJsonContainer#MenuJSON_HLTPS",
+                                  "xAOD::TriggerMenuJsonAuxContainer#MenuJSON_HLTPSAux.",
+                                  "xAOD::TriggerMenuJsonContainer#MenuJSON_L1PS",
+                                  "xAOD::TriggerMenuJsonAuxContainer#MenuJSON_L1PSAux.",
+                                  # "xAOD::TriggerMenuJsonContainer#MenuJSON_BG", // TODO
+                                  # "xAOD::TriggerMenuJsonAuxContainer#MenuJSON_BGAux.", // TODO
                                 ]
                 objKeyStore.addManyTypesMetaData( metadataItems )
 
diff --git a/Trigger/TriggerCommon/TriggerMenuMT/python/HLTMenuConfig/Menu/HLTMenuJSON.py b/Trigger/TriggerCommon/TriggerMenuMT/python/HLTMenuConfig/Menu/HLTMenuJSON.py
index c3503455125a5bac748dce4e4cd420730b8f0243..9ebc172df62bb77a7e428a94c8c0722e5e8408c8 100644
--- a/Trigger/TriggerCommon/TriggerMenuMT/python/HLTMenuConfig/Menu/HLTMenuJSON.py
+++ b/Trigger/TriggerCommon/TriggerMenuMT/python/HLTMenuConfig/Menu/HLTMenuJSON.py
@@ -103,6 +103,7 @@ def __generateJSON( chainDicts, chainConfigs, HLTAllSteps, menuName, fileName ):
         menuDict["chains"][chainName] = odict([
             ("counter", chain["chainCounter"]),
             ("nameHash", chain["chainNameHash"]),
+            ("legMultiplicities", chain["chainMultiplicities"]),
             ("l1item", chain["L1item"]),
             ("l1thresholds", l1Thresholds),
             ("groups", chain["groups"]),