from __future__ import print_function, division
from copy import deepcopy
import os
import numpy as np
import numpy.ma as ma
import matplotlib.pyplot as plt
import lsst.sims.maf.metrics as metrics
import lsst.sims.maf.slicers as slicers
import lsst.sims.maf.stackers as stackers
import lsst.sims.maf.plots as plots
import lsst.sims.maf.metricBundles as mb
from .colMapDict import ColMapDict
from .common import summaryCompletenessAtTime, summaryCompletenessOverH, fractionPopulationAtThreshold
__all__ = ['defaultHrange', 'defaultCharacterization','setupMoSlicer',
           'quickDiscoveryBatch', 'discoveryBatch',
           'runCompletenessSummary', 'plotCompleteness',
           'characterizationInnerBatch', 'characterizationOuterBatch',
           'runFractionSummary', 'plotFractions',
           'plotSingle', 'plotActivity',
           'readAndCombine', 'combineSubsets']
[docs]def defaultHrange(metadata):
    "Provide useful default ranges for H, based on metadata of population type."
    defaultRanges = {'PHA': [16, 28, 0.2],
                     'NEO': [16, 28, 0.2],
                     'MBA': [16, 26, 0.2],
                     'Trojan': [14, 22, 0.2],
                     'TNO': [4, 12, 0.2],
                     'SDO': [4, 12, 0.2]}
    defaultHmark = {'PHA': 22, 'NEO': 22, 'MBA': 20,
                    'Trojan': 18, 'TNO': 8, 'SDO': 8}
    if metadata in defaultRanges:
        Hrange = defaultRanges[metadata]
        Hmark = defaultHmark[metadata]
    elif metadata.upper().startswith('GRANVIK'):
        Hrange = defaultRanges['NEO']
        Hmark = defaultHmark['NEO']
    elif metadata.upper().startswith('L7'):
        Hrange = defaultRanges('TNO')
        Hmark = defaultHmark['TNO']
    else:
        print(f'## Could not find {metadata} in default keys ({defaultRanges.keys()}). \n'
              f'## Using expanded default range instead.')
        Hrange = [4, 28, 0.5]
        Hmark = 10
    return Hrange, Hmark 
[docs]def defaultCharacterization(metadata):
    "Provide useful characterization bundle type, based on metadata of population type."
    defaultChar = {'PHA': 'inner', 'NEO': 'inner',
                   'MBA': 'inner', 'Trojan': 'inner',
                   'TNO': 'outer', 'SDO': 'outer'}
    if metadata in defaultChar:
        char = defaultChar[metadata]
    elif metadata.upper().startswith('GRANVIK'):
        char = 'inner'
    elif metadata.upper().startswith('L7'):
        char = 'outer'
    else:
        print(f'## Could not find {metadata} in default keys ({defaultChar.keys()}). \n'
              f'## Using Inner (Asteroid) characterization by default.')
        char = 'inner'
    return char 
[docs]def setupMoSlicer(orbitFile, Hrange, obsFile=None):
    """
    Set up the slicer and read orbitFile and obsFile from disk.
    Parameters
    ----------
    orbitFile : str
        The file containing the orbit information.
    Hrange : numpy.ndarray or None
        The Hrange parameter to pass to slicer.readOrbits
    obsFile : str, optional
        The file containing the observations of each object, optional.
        If not provided (default, None), then the slicer will not be able to 'slice', but can still plot.
    Returns
    -------
    ~lsst.sims.maf.slicer.MoObjSlicer
    """
    # Read the orbit file and set the H values for the slicer.
    slicer = slicers.MoObjSlicer(Hrange=Hrange)
    slicer.setupSlicer(orbitFile=orbitFile, obsFile=obsFile)
    return slicer 
[docs]def quickDiscoveryBatch(slicer, colmap=None, runName='opsim', detectionLosses='detection', metadata='',
                        albedo=None, Hmark=None, npReduce=np.mean, constraint=None):
    if colmap is None:
        colmap = ColMapDict('opsimV4')
    bundleList = []
    plotBundles = []
    basicPlotDict = {'albedo': albedo, 'Hmark': Hmark, 'npReduce': npReduce,
                     'nxbins': 200, 'nybins': 200}
    plotFuncs = [plots.MetricVsH()]
    displayDict ={'group': 'Discovery'}
    if detectionLosses not in ('detection', 'trailing'):
        raise ValueError('Please choose detection or trailing as options for detectionLosses.')
    if detectionLosses == 'trailing':
        magStacker = stackers.MoMagStacker(lossCol='dmagTrail')
        detectionLosses = ' trailing loss'
    else:
        magStacker = stackers.MoMagStacker(lossCol='dmagDetect')
        detectionLosses = ' detection loss'
    # Set up a dictionary to pass to each metric for the column names.
    colkwargs = {'mjdCol': colmap['mjd'], 'seeingCol': colmap['seeingGeom'],
                 'expTimeCol': colmap['exptime'], 'm5Col': colmap['fiveSigmaDepth'],
                 'nightCol': colmap['night'], 'filterCol': colmap['filter']}
    def _setup_child_metrics(parentMetric):
        childMetrics = {}
        childMetrics['Time'] = metrics.Discovery_TimeMetric(parentMetric, **colkwargs)
        childMetrics['N_Chances'] = metrics.Discovery_N_ChancesMetric(parentMetric, **colkwargs)
        # Could expand to add N_chances per year, but not really necessary.
        return childMetrics
    def _configure_child_bundles(parentBundle):
        dispDict = {'group': 'Discovery', 'subgroup': 'Time',
                    'caption': 'Time of discovery of objects', 'order': 0}
        parentBundle.childBundles['Time'].setDisplayDict(dispDict)
        dispDict = {'group': 'Discovery', 'subgroup': 'NChances',
                    'caption': 'Number of chances for discovery of objects', 'order': 0}
        parentBundle.childBundles['N_Chances'].setDisplayDict(dispDict)
        return
    # 3 pairs in 15
    md = metadata + ' 3 pairs in 15 nights' + detectionLosses
    # Set up plot dict.
    plotDict = {'title': '%s: %s' % (runName, md)}
    plotDict.update(basicPlotDict)
    metric = metrics.DiscoveryMetric(nObsPerNight=2, tMin=0, tMax=90./60./24.,
                                     nNightsPerWindow=3, tWindow=15, **colkwargs)
    childMetrics = _setup_child_metrics(metric)
    bundle = mb.MoMetricBundle(metric, slicer, constraint,
                                stackerList=[magStacker],
                                runName=runName, metadata=md,
                                childMetrics=childMetrics,
                                plotDict=plotDict, plotFuncs=plotFuncs,
                                displayDict=displayDict)
    _configure_child_bundles(bundle)
    bundleList.append(bundle)
    # 3 pairs in 30
    md = metadata + ' 3 pairs in 30 nights' + detectionLosses
    plotDict = {'title': '%s: %s' % (runName, md)}
    plotDict.update(basicPlotDict)
    metric = metrics.DiscoveryMetric(nObsPerNight=2, tMin=0, tMax=90. / 60. / 24.,
                                     nNightsPerWindow=3, tWindow=30, **colkwargs)
    childMetrics = _setup_child_metrics(metric)
    bundle = mb.MoMetricBundle(metric, slicer, constraint,
                                stackerList=[magStacker],
                                runName=runName, metadata=md,
                                childMetrics=childMetrics,
                                plotDict=plotDict, plotFuncs=plotFuncs,
                                displayDict=displayDict)
    _configure_child_bundles(bundle)
    bundleList.append(bundle)
    # Set the runName for all bundles and return the bundleDict.
    for b in bundleList:
        b.setRunName(runName)
    return mb.makeBundlesDictFromList(bundleList), plotBundles 
[docs]def discoveryBatch(slicer, colmap=None, runName='opsim', detectionLosses='detection', metadata='',
                   albedo=None, Hmark=None, npReduce=np.mean, constraint=None):
    if colmap is None:
        colmap = ColMapDict('opsimV4')
    bundleList = []
    plotBundles = []
    basicPlotDict = {'albedo': albedo, 'Hmark': Hmark, 'npReduce': npReduce,
                     'nxbins': 200, 'nybins': 200}
    plotFuncs = [plots.MetricVsH()]
    displayDict ={'group': 'Discovery'}
    if detectionLosses not in ('detection', 'trailing'):
        raise ValueError('Please choose detection or trailing as options for detectionLosses.')
    if detectionLosses == 'trailing':
        # These are the SNR-losses only.
        magStacker = stackers.MoMagStacker(lossCol='dmagTrail')
        detectionLosses = ' trailing loss'
    else:
        # This is SNR losses, plus additional loss due to detecting with stellar PSF.
        magStacker = stackers.MoMagStacker(lossCol='dmagDetect')
        detectionLosses = ' detection loss'
    # Set up a dictionary to pass to each metric for the column names.
    colkwargs = {'mjdCol': colmap['mjd'], 'seeingCol': colmap['seeingGeom'],
                 'expTimeCol': colmap['exptime'], 'm5Col': colmap['fiveSigmaDepth'],
                 'nightCol': colmap['night'], 'filterCol': colmap['filter']}
    def _setup_child_metrics(parentMetric):
        childMetrics = {}
        childMetrics['Time'] = metrics.Discovery_TimeMetric(parentMetric, **colkwargs)
        childMetrics['N_Chances'] = metrics.Discovery_N_ChancesMetric(parentMetric, **colkwargs)
        # Could expand to add N_chances per year, but not really necessary.
        return childMetrics
    def _configure_child_bundles(parentBundle):
        dispDict = {'group': 'Discovery', 'subgroup': 'Time',
                    'caption': 'Time of discovery of objects', 'order': 0}
        parentBundle.childBundles['Time'].setDisplayDict(dispDict)
        dispDict = {'group': 'Discovery', 'subgroup': 'NChances',
                    'caption': 'Number of chances for discovery of objects', 'order': 0}
        parentBundle.childBundles['N_Chances'].setDisplayDict(dispDict)
        return
    # 3 pairs in 15 and 3 pairs in 30 done in 'quickDiscoveryBatch' (with vis).
    # 3 pairs in 12
    md = metadata + ' 3 pairs in 12 nights' + detectionLosses
    plotDict = {'title': '%s: %s' % (runName, md)}
    plotDict.update(basicPlotDict)
    metric = metrics.DiscoveryMetric(nObsPerNight=2, tMin=0, tMax=90. / 60. / 24.,
                                     nNightsPerWindow=3, tWindow=12, **colkwargs)
    childMetrics = _setup_child_metrics(metric)
    bundle = mb.MoMetricBundle(metric, slicer, constraint,
                                stackerList=[magStacker],
                                runName=runName, metadata=md,
                                childMetrics=childMetrics,
                                plotDict=plotDict, plotFuncs=plotFuncs,
                                displayDict=displayDict)
    _configure_child_bundles(bundle)
    bundleList.append(bundle)
    # 3 pairs in 20
    md = metadata + ' 3 pairs in 20 nights' + detectionLosses
    plotDict = {'title': '%s: %s' % (runName, md)}
    plotDict.update(basicPlotDict)
    metric = metrics.DiscoveryMetric(nObsPerNight=2, tMin=0, tMax=90. / 60. / 24.,
                                     nNightsPerWindow=3, tWindow=20, **colkwargs)
    childMetrics = _setup_child_metrics(metric)
    bundle = mb.MoMetricBundle(metric, slicer, constraint,
                                stackerList=[magStacker],
                                runName=runName, metadata=md,
                                childMetrics=childMetrics,
                                plotDict=plotDict, plotFuncs=plotFuncs,
                                displayDict=displayDict)
    _configure_child_bundles(bundle)
    bundleList.append(bundle)
    # 3 pairs in 25
    md = metadata + ' 3 pairs in 25 nights' + detectionLosses
    plotDict = {'title': '%s: %s' % (runName, md)}
    plotDict.update(basicPlotDict)
    metric = metrics.DiscoveryMetric(nObsPerNight=2, tMin=0, tMax=90. / 60. / 24.,
                                     nNightsPerWindow=3, tWindow=25, **colkwargs)
    childMetrics = _setup_child_metrics(metric)
    bundle = mb.MoMetricBundle(metric, slicer, constraint,
                                stackerList=[magStacker],
                                runName=runName, metadata=md,
                                childMetrics=childMetrics,
                                plotDict=plotDict, plotFuncs=plotFuncs,
                                displayDict=displayDict)
    _configure_child_bundles(bundle)
    bundleList.append(bundle)
    # 4 pairs in 20
    md = metadata + ' 4 pairs in 20 nights' + detectionLosses
    plotDict = {'title': '%s: %s' % (runName, md)}
    plotDict.update(basicPlotDict)
    metric = metrics.DiscoveryMetric(nObsPerNight=2, tMin=0, tMax=90. / 60. / 24.,
                                     nNightsPerWindow=4, tWindow=20, **colkwargs)
    childMetrics = _setup_child_metrics(metric)
    bundle = mb.MoMetricBundle(metric, slicer, constraint,
                                stackerList=[magStacker],
                                runName=runName, metadata=md,
                                childMetrics=childMetrics,
                                plotDict=plotDict, plotFuncs=plotFuncs,
                                displayDict=displayDict)
    _configure_child_bundles(bundle)
    bundleList.append(bundle)
    # 3 triplets in 30
    md = metadata + ' 3 triplets in 30 nights' + detectionLosses
    plotDict = {'title': '%s: %s' % (runName, md)}
    plotDict.update(basicPlotDict)
    metric = metrics.DiscoveryMetric(nObsPerNight=3, tMin=0, tMax=120. / 60. / 24.,
                                     nNightsPerWindow=3, tWindow=30, **colkwargs)
    childMetrics = _setup_child_metrics(metric)
    bundle = mb.MoMetricBundle(metric, slicer, constraint,
                                stackerList=[magStacker],
                                runName=runName, metadata=md,
                                childMetrics=childMetrics,
                                plotDict=plotDict, plotFuncs=plotFuncs,
                                displayDict=displayDict)
    _configure_child_bundles(bundle)
    bundleList.append(bundle)
    # 3 quads in 30
    md = metadata + ' 3 quads in 30 nights' + detectionLosses
    plotDict = {'title': '%s: %s' % (runName, md)}
    plotDict.update(basicPlotDict)
    metric = metrics.DiscoveryMetric(nObsPerNight=4, tMin=0, tMax=150. / 60. / 24.,
                                     nNightsPerWindow=3, tWindow=30, **colkwargs)
    childMetrics = _setup_child_metrics(metric)
    bundle = mb.MoMetricBundle(metric, slicer, constraint,
                                stackerList=[magStacker],
                                runName=runName, metadata=md,
                                childMetrics=childMetrics,
                                plotDict=plotDict, plotFuncs=plotFuncs,
                                displayDict=displayDict)
    _configure_child_bundles(bundle)
    bundleList.append(bundle)
    # Play with SNR.
    # First standard SNR / probabilistic visibility (SNR~5)
    # 3 pairs in 15
    md = metadata + ' 3 pairs in 15 nights SNR=5' + detectionLosses
    # Set up plot dict.
    plotDict = {'title': '%s: %s' % (runName, md)}
    plotDict.update(basicPlotDict)
    metric = metrics.DiscoveryMetric(nObsPerNight=2, tMin=0, tMax=90./60./24.,
                                     nNightsPerWindow=3, tWindow=15, snrLimit=5, **colkwargs)
    childMetrics = _setup_child_metrics(metric)
    bundle = mb.MoMetricBundle(metric, slicer, constraint,
                                stackerList=[magStacker],
                                runName=runName, metadata=md,
                                childMetrics=childMetrics,
                                plotDict=plotDict, plotFuncs=plotFuncs,
                                displayDict=displayDict)
    _configure_child_bundles(bundle)
    bundleList.append(bundle)
    # 3 pairs in 15, SNR=4.
    md = metadata + ' 3 pairs in 15 nights SNR=4' + detectionLosses
    plotDict = {'title': '%s: %s' % (runName, md)}
    plotDict.update(basicPlotDict)
    metric = metrics.DiscoveryMetric(nObsPerNight=2, tMin=0, tMax=90. / 60. / 24.,
                                     nNightsPerWindow=3, tWindow=15, snrLimit=4, **colkwargs)
    childMetrics = _setup_child_metrics(metric)
    bundle = mb.MoMetricBundle(metric, slicer, constraint,
                                stackerList=[magStacker],
                                runName=runName, metadata=md,
                                childMetrics=childMetrics,
                                plotDict=plotDict, plotFuncs=plotFuncs,
                                displayDict=displayDict)
    _configure_child_bundles(bundle)
    bundleList.append(bundle)
    # 3 pairs in 30, SNR=5
    md = metadata + ' 3 pairs in 30 nights SNR=5' + detectionLosses
    plotDict = {'title': '%s: %s' % (runName, md)}
    plotDict.update(basicPlotDict)
    metric = metrics.DiscoveryMetric(nObsPerNight=2, tMin=0, tMax=90. / 60. / 24.,
                                     nNightsPerWindow=3, tWindow=30, snrLimit=5, **colkwargs)
    childMetrics = _setup_child_metrics(metric)
    bundle = mb.MoMetricBundle(metric, slicer, constraint,
                                stackerList=[magStacker],
                                runName=runName, metadata=md,
                                childMetrics=childMetrics,
                                plotDict=plotDict, plotFuncs=plotFuncs,
                                displayDict=displayDict)
    _configure_child_bundles(bundle)
    bundleList.append(bundle)
    # 3 pairs in 30, SNR=4
    md = metadata + ' 3 pairs in 30 nights SNR=4' + detectionLosses
    plotDict = {'title': '%s: %s' % (runName, md)}
    plotDict.update(basicPlotDict)
    metric = metrics.DiscoveryMetric(nObsPerNight=2, tMin=0, tMax=90. / 60. / 24.,
                                     nNightsPerWindow=3, tWindow=30, snrLimit=4, **colkwargs)
    childMetrics = _setup_child_metrics(metric)
    bundle = mb.MoMetricBundle(metric, slicer, constraint,
                                stackerList=[magStacker],
                                runName=runName, metadata=md,
                                childMetrics=childMetrics,
                                plotDict=plotDict, plotFuncs=plotFuncs,
                                displayDict=displayDict)
    _configure_child_bundles(bundle)
    bundleList.append(bundle)
    # Play with SNR.  SNR=3
    # 3 pairs in 15, SNR=3
    md = metadata + ' 3 pairs in 15 nights SNR=3' + detectionLosses
    plotDict = {'title': '%s: %s' % (runName, md)}
    plotDict.update(basicPlotDict)
    metric = metrics.DiscoveryMetric(nObsPerNight=2, tMin=0, tMax=90. / 60. / 24.,
                                     nNightsPerWindow=3, tWindow=15, snrLimit=3, **colkwargs)
    childMetrics = _setup_child_metrics(metric)
    bundle = mb.MoMetricBundle(metric, slicer, constraint,
                                stackerList=[magStacker],
                                runName=runName, metadata=md,
                                childMetrics=childMetrics,
                                plotDict=plotDict, plotFuncs=plotFuncs,
                                displayDict=displayDict)
    _configure_child_bundles(bundle)
    bundleList.append(bundle)
    # SNR = 0
    # 3 pairs in 15, SNR=0
    md = metadata + ' 3 pairs in 15 nights SNR=0' + detectionLosses
    plotDict = {'title': '%s: %s' % (runName, md)}
    plotDict.update(basicPlotDict)
    metric = metrics.DiscoveryMetric(nObsPerNight=2, tMin=0, tMax=90. / 60. / 24.,
                                     nNightsPerWindow=3, tWindow=15, snrLimit=0, **colkwargs)
    childMetrics = _setup_child_metrics(metric)
    bundle = mb.MoMetricBundle(metric, slicer, constraint,
                                stackerList=[magStacker],
                                runName=runName, metadata=md,
                                childMetrics=childMetrics,
                                plotDict=plotDict, plotFuncs=plotFuncs,
                                displayDict=displayDict)
    _configure_child_bundles(bundle)
    bundleList.append(bundle)
    # Play with weird strategies.
    # Single detection.
    md = metadata + ' Single detection' + detectionLosses
    plotDict = {'title': '%s: %s' % (runName, md)}
    plotDict.update(basicPlotDict)
    metric = metrics.DiscoveryMetric(nObsPerNight=1, tMin=0, tMax=90. / 60. / 24.,
                                     nNightsPerWindow=1, tWindow=5, **colkwargs)
    childMetrics = _setup_child_metrics(metric)
    bundle = mb.MoMetricBundle(metric, slicer, constraint,
                                stackerList=[magStacker],
                                runName=runName, metadata=md,
                                childMetrics=childMetrics,
                                plotDict=plotDict, plotFuncs=plotFuncs,
                                displayDict=displayDict)
    _configure_child_bundles(bundle)
    bundleList.append(bundle)
    # Single pair of detections.
    md = metadata + ' Single pair' + detectionLosses
    plotDict = {'title': '%s: %s' % (runName, md)}
    plotDict.update(basicPlotDict)
    metric = metrics.DiscoveryMetric(nObsPerNight=2, tMin=0, tMax=90. / 60. / 24.,
                                     nNightsPerWindow=1, tWindow=5, **colkwargs)
    childMetrics = _setup_child_metrics(metric)
    bundle = mb.MoMetricBundle(metric, slicer, constraint,
                                stackerList=[magStacker],
                                runName=runName, metadata=md,
                                childMetrics=childMetrics,
                                plotDict=plotDict, plotFuncs=plotFuncs,
                                displayDict=displayDict)
    _configure_child_bundles(bundle)
    bundleList.append(bundle)
    # High velocity discovery.
    displayDict['subgroup'] = 'High Velocity'
    # High velocity.
    md = metadata + ' High velocity pair' + detectionLosses
    plotDict = {'title': '%s: %s' % (runName, md)}
    plotDict.update(basicPlotDict)
    metric = metrics.HighVelocityNightsMetric(psfFactor=2., nObsPerNight=2, **colkwargs)
    bundle = mb.MoMetricBundle(metric, slicer, constraint,
                                stackerList=[magStacker],
                                runName=runName, metadata=md,
                                plotDict=plotDict, plotFuncs=plotFuncs,
                                displayDict=displayDict)
    bundleList.append(bundle)
    # "magic" detection - 6 in 60 days.
    md = metadata + ' 6 detections in 60 nights' + detectionLosses
    plotDict = {'title': '%s: %s' % (runName, md)}
    plotDict.update(basicPlotDict)
    metric = metrics.MagicDiscoveryMetric(nObs=6, tWindow=60, **colkwargs)
    bundle = mb.MoMetricBundle(metric, slicer, constraint,
                                stackerList=[magStacker],
                                runName=runName, metadata=md,
                                plotDict=plotDict, plotFuncs=plotFuncs,
                                displayDict=displayDict)
    bundleList.append(bundle)
    # Set the runName for all bundles and return the bundleDict.
    for b in bundleList:
        b.setRunName(runName)
    return mb.makeBundlesDictFromList(bundleList), plotBundles 
[docs]def runCompletenessSummary(bdict, Hmark, times, outDir, resultsDb):
    """
    Calculate completeness and create completeness bundles from all N_Chances and Time (child) metrics
    of the (discovery) bundles in bdict, and write completeness at Hmark to resultsDb, save bundle to disk.
    This should be done after combining any sub-sets of the metric results.
    Parameters
    ----------
    bdict : dict of metricBundles
        Dict containing ~lsst.sims.maf.MoMetricBundles,
        including bundles we're expecting to contain completeness.
    Hmark : float
        Hmark value to add to completeness plotting dict.
        If not defined (None), then the Hmark from the plotdict from the metric will be used if available.
        If None and Hmark not in plotDict, then median of Hrange value will be used.
    times : np.ndarray
        The times at which to calculate completeness (over time).
    outDir : str
        Output directory to save completeness bundles to disk.
    resultsDb : ~lsst.sims.maf.db.ResultsDb
        Results database to save information about completeness bundle.
    Returns
    -------
    dict of metricBundles
        A dictionary of the new completeness bundles. Keys match original keys,
        with additions of "[Differential,Cumulative]Completeness@Time"
        and "[Differential,Cumulative]Completeness" to distinguish new entries.
    """
    # Add completeness bundles and write completeness at Hmark to resultsDb.
    completeness = {}
    group = 'Discovery'
    subgroup = 'Completeness'
    def _compbundles(b, bundle, Hmark, resultsDb):
        # Find Hmark if not set (this may be different for different bundles).
        if Hmark is None and 'Hmark' in bundle.plotDict:
            Hmark = bundle.plotDict['Hmark']
        if Hmark is None:
            Hmark = np.median(bundle.slicer.slicePoints['H'])
        # Set up the summary metrics.
        summaryTimeMetrics = summaryCompletenessAtTime(times, Hval=Hmark, Hindex=0.33)
        summaryTimeMetrics2 = summaryCompletenessAtTime(times, Hval=Hmark - 2, Hindex=0.33)
        summaryHMetrics = summaryCompletenessOverH(requiredChances=1, Hindex=0.33)
        comp = {}
        # Bundle = single metric bundle. Add differential and cumulative completeness.
        if 'Time' in bundle.metric.name:
            for metric in summaryTimeMetrics:
                newkey = b + ' ' + metric.name
                comp[newkey] = mb.makeCompletenessBundle(bundle, metric,
                                                         Hmark=None, resultsDb=resultsDb)
                comp[newkey].plotDict['times'] = times
                comp[newkey].plotDict['Hval'] = metric.Hval
            for metric in summaryTimeMetrics2:
                newkey = b + ' ' + metric.name
                comp[newkey] = mb.makeCompletenessBundle(bundle, metric,
                                                         Hmark=None, resultsDb=resultsDb)
                comp[newkey].plotDict['times'] = times
                comp[newkey].plotDict['Hval'] = metric.Hval
        elif 'N_Chances' in bundle.metric.name:
            for metric in summaryHMetrics:
                newkey = b + ' ' + metric.name
                comp[newkey] = mb.makeCompletenessBundle(bundle, metric,
                                                         Hmark=Hmark, resultsDb=resultsDb)
        return comp
    # Generate the completeness bundles for the various discovery metrics.
    for b, bundle in bdict.items():
        if 'Discovery' in bundle.metric.name:
            completeness.update(_compbundles(b, bundle, Hmark, resultsDb))
        if isinstance(bundle.metric, metrics.HighVelocityNightsMetric):
            completeness.update(_compbundles(b, bundle, Hmark, resultsDb))
        if isinstance(bundle.metric, metrics.MagicDiscoveryMetric):
            completeness.update(_compbundles(b, bundle, Hmark, resultsDb))
    # Write the completeness bundles to disk, so we can re-read them later.
    # (also set the display dict properties, for the resultsDb output).
    for b, bundle in completeness.items():
        bundle.setDisplayDict({'group': group, 'subgroup': subgroup})
        bundle.write(outDir=outDir, resultsDb=resultsDb)
    return completeness 
[docs]def plotCompleteness(bdictCompleteness, figroot=None, resultsDb=None,
                     outDir='.', figformat='pdf'):
    """Plot a minor subset of the completeness results.
    """
    # Separate some subsets to plot together.
    keys = ['3_pairs_in_30_nights_detection_loss',
            '3_pairs_in_15_nights_detection_loss']
    plotTimes = {}
    plotComp = {}
    plotDiff = {}
    for k in bdictCompleteness:
        for key in keys:
            if key in k:
                if 'Discovery_Time' in k:
                    if 'Cumulative' in k:
                        plotTimes[k] = bdictCompleteness[k]
                elif 'Discovery_N_Chances' in k:
                    if 'Differential' in k:
                        plotDiff[k] = bdictCompleteness[k]
                    elif 'Cumulative' in k:
                        plotComp[k] = bdictCompleteness[k]
    # Add plot dictionaries to code 30 nights red, 15 nights blue, differentials dotted.
    def _codePlot(key):
        plotDict = {}
        if 'Differential' in k:
            plotDict['linestyle'] = ':'
        else:
            plotDict['linestyle'] = '-'
        if '30_nights' in k:
            plotDict['color'] = 'r'
        if '15_nights' in k:
            plotDict['color'] = 'b'
        return plotDict
    # Apply color-coding.
    for k, b in plotTimes.items():
        b.setPlotDict(_codePlot(k))
    for k, b in plotDiff.items():
        b.setPlotDict(_codePlot(k))
    for k, b in plotComp.items():
        b.setPlotDict(_codePlot(k))
    first = bdictCompleteness[list(bdictCompleteness.keys())[0]]
    if figroot is None:
        figroot = first.runName
    # Plot completeness as a function of time.
    fig = plt.figure(figsize=(8, 6))
    for k in plotTimes:
        plt.plot(plotTimes[k].plotDict['times'], plotTimes[k].metricValues[0, :],
                 label=plotTimes[k].plotDict['label'] + ' @H=%.2f' % plotTimes[k].plotDict['Hval'])
    plt.legend()
    plt.xlabel('Time (MJD)')
    plt.ylabel('Completeness')
    plt.grid(True, alpha=0.3)
    # Make a PlotHandler to deal with savings/resultsDb, etc.
    ph = plots.PlotHandler(figformat=figformat, resultsDb=resultsDb, outDir=outDir)
    displayDict = {'group': 'Completeness', 'subgroup': 'Over Time',
                   'caption': 'Completeness over time, for H values indicated in legend.'}
    ph.saveFig(fig.number, f'{figroot}_CompletenessOverTime', 'Combo', 'CompletenessOverTime', 'MoObjSlicer',
               figroot, None, None, displayDict=displayDict)
    plt.savefig(os.path.join(outDir, f'{figroot}_CompletenessOverTime.{figformat}'), format=figformat)
    # Plot cumulative completeness.
    ph = plots.PlotHandler(figformat=figformat, resultsDb=resultsDb, outDir=outDir)
    ph.setMetricBundles(plotComp)
    plotDict = {'ylabel': "Completeness", 'figsize': (8, 6), 'albedo': 0.14}
    ph.plot(plotFunc=plots.MetricVsH(), plotDicts=plotDict,
            outfileRoot=figroot + '_CumulativeCompleteness')
    # Plot differential completeness.
    ph = plots.PlotHandler(figformat=figformat, resultsDb=resultsDb, outDir=outDir)
    ph.setMetricBundles(plotDiff)
    plotDict = {'ylabel': "Completeness", 'figsize': (8, 6)}
    ph.plot(plotFunc=plots.MetricVsH(), plotDicts=plotDict,
            outfileRoot=figroot + '_DifferentialCompleteness') 
[docs]def characterizationInnerBatch(slicer, colmap=None, runName='opsim', metadata='',
                                  albedo=None, Hmark=None, constraint=None, npReduce=np.mean,
                                  windows=None, bins=None):
    """Characterization metrics for inner solar system objects.
    """
    if colmap is None:
        colmap = ColMapDict('opsimV4')
    bundleList = []
    plotBundles = []
    # Set up a dictionary to pass to each metric for the column names.
    colkwargs = {'mjdCol': colmap['mjd'], 'seeingCol': colmap['seeingGeom'],
                 'expTimeCol': colmap['exptime'], 'm5Col': colmap['fiveSigmaDepth'],
                 'nightCol': colmap['night'], 'filterCol': colmap['filter']}
    basicPlotDict = {'albedo': albedo, 'Hmark': Hmark, 'npReduce': npReduce,
                     'nxbins': 200, 'nybins': 200}
    plotFuncs = [plots.MetricVsH()]
    displayDict ={'group': 'Characterization'}
    # Stackers
    magStacker = stackers.MoMagStacker(lossCol='dmagDetect')
    eclStacker = stackers.EclStacker()
    stackerList = [magStacker, eclStacker]
    # Windows are the different 'length of activity'
    if windows is None:
        windows = np.arange(10, 200, 30.)
    # Bins are the different 'anomaly variations' of activity
    if bins is None:
        bins = np.arange(5, 185, 20.)
    # Number of observations.
    md = metadata
    plotDict = {'ylabel': 'Number of observations (#)',
                'title': '%s: Number of observations %s' % (runName, md)}
    plotDict.update(basicPlotDict)
    metric = metrics.NObsMetric(**colkwargs)
    bundle = mb.MoMetricBundle(metric, slicer, constraint,
                               stackerList=stackerList,
                                runName=runName, metadata=md,
                                plotDict=plotDict, plotFuncs=plotFuncs,
                                displayDict=displayDict)
    bundleList.append(bundle)
    # Observational arc.
    md = metadata
    plotDict = {'ylabel': 'Observational Arc (days)',
                'title': '%s: Observational Arc Length %s' % (runName, md)}
    plotDict.update(basicPlotDict)
    metric = metrics.ObsArcMetric(**colkwargs)
    bundle = mb.MoMetricBundle(metric, slicer, constraint,
                               stackerList=stackerList,
                                runName=runName, metadata=md,
                                plotDict=plotDict, plotFuncs=plotFuncs,
                                displayDict=displayDict)
    bundleList.append(bundle)
    # Activity detection.
    for w in windows:
        md = metadata + ' activity lasting %.0f days' % w
        plotDict = {'title': '%s: Chances of detecting %s' % (runName, md),
                    'ylabel': 'Probability of detection per %.0f day window' % w}
        metricName = 'Chances of detecting activity lasting %.0f days' % w
        metric = metrics.ActivityOverTimeMetric(w, metricName=metricName, **colkwargs)
        bundle = mb.MoMetricBundle(metric, slicer, constraint,
                                   stackerList=stackerList,
                                    runName=runName, metadata=metadata,
                                    plotDict=plotDict, plotFuncs=plotFuncs,
                                    displayDict=displayDict)
        bundleList.append(bundle)
    for b in bins:
        md = metadata + ' activity covering %.0f deg' % (b)
        plotDict = {'title': '%s: Chances of detecting %s' % (runName, md),
                    'ylabel': 'Probability of detection per %.0f deg window' % b}
        metricName = 'Chances of detecting activity covering %.0f deg' % (b)
        metric = metrics.ActivityOverPeriodMetric(b, metricName=metricName, **colkwargs)
        bundle = mb.MoMetricBundle(metric, slicer, constraint,
                                   stackerList=stackerList,
                                    runName=runName, metadata=metadata,
                                    plotDict=plotDict, plotFuncs=plotFuncs,
                                    displayDict=displayDict)
        bundleList.append(bundle)
    # Lightcurve inversion.
    md = metadata
    plotDict = {'yMin': 0, 'yMax': 1, 'ylabel': 'Fraction of objects',
                'title': '%s: Fraction with potential lightcurve inversion %s' % (runName, md)}
    plotDict.update(basicPlotDict)
    metric = metrics.LightcurveInversion_AsteroidMetric(**colkwargs)
    bundle = mb.MoMetricBundle(metric, slicer, constraint,
                               stackerList=stackerList,
                                runName=runName, metadata=md,
                                plotDict=plotDict, plotFuncs=plotFuncs,
                                displayDict=displayDict)
    bundleList.append(bundle)
    # Color determination.
    md = metadata
    plotDict = {'yMin': 0, 'yMax': 1, 'ylabel': 'Fraction of objects',
                'title': '%s: Fraction of population with colors in X filters %s' % (runName, md)}
    plotDict.update(basicPlotDict)
    metric = metrics.Color_AsteroidMetric(**colkwargs)
    bundle = mb.MoMetricBundle(metric, slicer, constraint,
                               stackerList=stackerList,
                                runName=runName, metadata=md,
                                plotDict=plotDict, plotFuncs=plotFuncs,
                                displayDict=displayDict)
    bundleList.append(bundle)
    # Set the runName for all bundles and return the bundleDict.
    for b in bundleList:
        b.setRunName(runName)
    return mb.makeBundlesDictFromList(bundleList), plotBundles 
[docs]def characterizationOuterBatch(slicer, colmap=None, runName='opsim', metadata='',
                               albedo=None, Hmark=None, constraint=None, npReduce=np.mean,
                               windows=None, bins=None):
    """Characterization metrics for outer solar system objects.
    """
    if colmap is None:
        colmap = ColMapDict('opsimV4')
    bundleList = []
    plotBundles = []
    # Set up a dictionary to pass to each metric for the column names.
    colkwargs = {'mjdCol': colmap['mjd'], 'seeingCol': colmap['seeingGeom'],
                 'expTimeCol': colmap['exptime'], 'm5Col': colmap['fiveSigmaDepth'],
                 'nightCol': colmap['night'], 'filterCol': colmap['filter']}
    basicPlotDict = {'albedo': albedo, 'Hmark': Hmark, 'npReduce': npReduce,
                     'nxbins': 200, 'nybins': 200}
    plotFuncs = [plots.MetricVsH()]
    displayDict ={'group': 'Characterization'}
    # Stackers
    magStacker = stackers.MoMagStacker(lossCol='dmagDetect')
    eclStacker = stackers.EclStacker()
    stackerList = [magStacker, eclStacker]
    # Windows are the different 'length of activity'
    if windows is None:
        windows = np.arange(10, 200, 30.)
    # Bins are the different 'anomaly variations' of activity
    if bins is None:
        bins = np.arange(5, 185, 20.)
    # Number of observations.
    md = metadata
    plotDict = {'ylabel': 'Number of observations (#)',
                'title': '%s: Number of observations %s' % (runName, md)}
    plotDict.update(basicPlotDict)
    metric = metrics.NObsMetric(**colkwargs)
    bundle = mb.MoMetricBundle(metric, slicer, constraint,
                               stackerList=stackerList,
                               runName=runName, metadata=md,
                               plotDict=plotDict, plotFuncs=plotFuncs,
                               displayDict=displayDict)
    bundleList.append(bundle)
    # Observational arc.
    md = metadata
    plotDict = {'ylabel': 'Observational Arc (days)',
                'title': '%s: Observational Arc Length %s' % (runName, md)}
    plotDict.update(basicPlotDict)
    metric = metrics.ObsArcMetric(**colkwargs)
    bundle = mb.MoMetricBundle(metric, slicer, constraint,
                               stackerList=stackerList,
                               runName=runName, metadata=md,
                               plotDict=plotDict, plotFuncs=plotFuncs,
                               displayDict=displayDict)
    bundleList.append(bundle)
    # Activity detection.
    for w in windows:
        md = metadata + ' activity lasting %.0f days' % w
        plotDict = {'title': '%s: Chances of detecting %s' % (runName, md),
                    'ylabel': 'Probability of detection per %.0f day window' % w}
        metricName = 'Chances of detecting activity lasting %.0f days' % w
        metric = metrics.ActivityOverTimeMetric(w, metricName=metricName, **colkwargs)
        bundle = mb.MoMetricBundle(metric, slicer, constraint,
                                   stackerList=stackerList,
                                    runName=runName, metadata=metadata,
                                    plotDict=plotDict, plotFuncs=plotFuncs,
                                    displayDict=displayDict)
        bundleList.append(bundle)
    for b in bins:
        md = metadata + ' activity covering %.0f deg' % (b)
        plotDict = {'title': '%s: Chances of detecting %s' % (runName, md),
                    'ylabel': 'Probability of detection per %.2f deg window' % b}
        metricName = 'Chances of detecting activity covering %.0f deg' % (b)
        metric = metrics.ActivityOverPeriodMetric(b, metricName=metricName, **colkwargs)
        bundle = mb.MoMetricBundle(metric, slicer, constraint,
                                   stackerList=stackerList,
                                    runName=runName, metadata=metadata,
                                    plotDict=plotDict, plotFuncs=plotFuncs,
                                    displayDict=displayDict)
        bundleList.append(bundle)
    # Color determination.
    md = metadata
    plotDict = {'yMin': 0, 'yMax': 1, 'ylabel': 'Fraction of objects',
                'title': '%s: Fraction of population with colors in X filters %s' % (runName, md)}
    plotDict.update(basicPlotDict)
    metric = metrics.LightcurveColor_OuterMetric(**colkwargs)
    bundle = mb.MoMetricBundle(metric, slicer, constraint,
                               stackerList=stackerList,
                                runName=runName, metadata=md,
                                plotDict=plotDict, plotFuncs=plotFuncs,
                                displayDict=displayDict)
    bundleList.append(bundle)
    # Set the runName for all bundles and return the bundleDict.
    for b in bundleList:
        b.setRunName(runName)
    return mb.makeBundlesDictFromList(bundleList), plotBundles 
[docs]def runFractionSummary(bdict, Hmark, outDir, resultsDb):
    """
    Calculate fractional completeness of the population for color and lightcurve metrics.
    This should be done after combining any sub-sets of the metric results.
    Parameters
    ----------
    bdict : dict of metricBundles
        Dict containing ~lsst.sims.maf.MoMetricBundles,
        including bundles we're expecting to contain lightcurve/color evaluations.
    Hmark : float
        Hmark value to add to completeness plotting dict.
        If defined, this value is used. If None, but Hmark in plotDict for metric, then this value (-2) is
        used. If Hmark not in plotdict, then the median Hrange value - 2 is used.
    times : np.ndarray
        The times at which to calculate completeness (over time).
    outDir : str
        Output directory to save completeness bundles to disk.
    resultsDb : ~lsst.sims.maf.db.ResultsDb
        Results database to save information about completeness bundle.
    Returns
    -------
    dict of metricBundles
        Dictionary of the metric bundles for the fractional evaluation of the population.
    """
    fractions = {}
    group = 'Characterization'
    subgroup = 'Fraction of Population with Color/Lightcurve'
    # Look for metrics from asteroid or outer solar system color/lightcurve metrics.
    inversionSummary = fractionPopulationAtThreshold([1], ['Lightcurve Inversion'])
    asteroidColorSummary = fractionPopulationAtThreshold([4, 3, 2, 1], ['6 of ugrizy', '5 of grizy',
                                                                        '4 of grizy',
                                                                        '2 of g, r or i, z or y'])
    asteroidSummaryMetrics = {'LightcurveInversion_Asteroid': inversionSummary,
                              'Color_Asteroid': asteroidColorSummary}
    outerColorSummary = fractionPopulationAtThreshold([6, 5, 4, 3, 2, 1], ['6 filters', '5 filters',
                                                                           '4 filters', '3 filters',
                                                                           '2 filters', '1 filters'])
    outerSummaryMetrics = {'LightcurveColor_Outer': outerColorSummary}
    for b, bundle in bdict.items():
        # Find Hmark if not set (this may be different for different bundles).
        if Hmark is None and 'Hmark' in bundle.plotDict:
            Hmark = bundle.plotDict['Hmark'] - 2
        if Hmark is None:
            Hmark = np.median(bundle.slicer.slicePoints['H']) - 2
        for k in asteroidSummaryMetrics:
            if k in b:
                for summary_metric in asteroidSummaryMetrics[k]:
                    newkey = b + ' ' + summary_metric.name
                    fractions[newkey] = mb.makeCompletenessBundle(bundle, summary_metric,
                                                                  Hmark=Hmark, resultsDb=resultsDb)
        for k in outerSummaryMetrics:
            if k in b:
                for summary_metric in outerSummaryMetrics[k]:
                    newkey = b + ' ' + summary_metric.name
                    fractions[newkey] = mb.makeCompletenessBundle(bundle, summary_metric,
                                                                  Hmark=Hmark, resultsDb=resultsDb)
    # Write the fractional populations bundles to disk, so we can re-read them later.
    # (also set the display dict properties, for the resultsDb output).
    for b, bundle in fractions.items():
        bundle.setDisplayDict({'group': group, 'subgroup': subgroup})
        bundle.write(outDir=outDir, resultsDb=resultsDb)
    return fractions 
[docs]def plotFractions(bdictFractions, figroot=None, resultsDb=None,
                  outDir='.', figformat='pdf'):
    # Set colors for the fractions.
    for b in bdictFractions.values():
        k = b.metric.name
        if '6' in k:
            b.plotDict['color'] = 'b'
        if '5' in k:
            b.plotDict['color'] = 'cyan'
        if '4' in k:
            b.plotDict['color'] = 'orange'
        if '2' in k:
            b.plotDict['color'] = 'r'
        if '1' in k:
            b.plotDict['color'] = 'magenta'
        if 'Lightcurve Inversion' in k:
            b.plotDict['color'] = 'k'
            b.plotDict['linestyle'] = ':'
            b.plotDict['linewidth'] = 3
    if figroot is None:
        first = bdictFractions[list(bdictFractions.keys())[0]]
        figroot = first.runName
    displayDict = {'group': 'Characterization', 'subgroup': 'Color/Inversion'}
    ph = plots.PlotHandler(figformat=figformat, resultsDb=resultsDb, outDir=outDir)
    ph.setMetricBundles(bdictFractions)
    ph.jointMetricNames = 'Fraction of population for colors or lightcurve inversion'
    plotDict = {'ylabel': "Fraction of population", 'figsize': (8, 6)}
    ph.plot(plotFunc=plots.MetricVsH(), plotDicts=plotDict, displayDict=displayDict,
            outfileRoot=figroot + '_characterization') 
[docs]def plotSingle(bundle, resultsDb=None, outDir='.', figformat='pdf'):
    """Plot 5%/25%/50%/75%/95% iles for a metric value.
    """
    pDict = {'95%ile': {'color': 'k', 'linestyle': '--', 'label': '95th %ile',
                        'npReduce': lambda x, axis: np.percentile(x, 95, axis=axis)},
             '75%ile': {'color': 'magenta', 'linestyle': ':', 'label': '75th %ile',
                        'npReduce': lambda x, axis: np.percentile(x, 75, axis=axis)},
             'Median': {'color': 'b', 'linestyle': '-', 'label': 'Median',
                        'npReduce': lambda x, axis: np.median(x, axis=axis)},
             'Mean': {'color': 'g', 'linestyle': '--', 'label': 'Mean',
                      'npReduce': np.mean},
             '25%ile': {'color': 'magenta', 'linestyle': ':', 'label': '25th %ile',
                        'npReduce': lambda x, axis: np.percentile(x, 25, axis=axis)},
             '5%ile': {'color': 'k', 'linestyle': '--', 'label': '5th %ile',
                       'npReduce': lambda x, axis: np.percentile(x, 5, axis=axis)}}
    displayDict = {'group': 'Characterization', 'subgroup': bundle.metric.name}
    ph = plots.PlotHandler(figformat=figformat, resultsDb=resultsDb, outDir=outDir)
    plotBundles = []
    plotDicts = []
    for percentile in pDict:
        plotBundles.append(bundle)
        plotDicts.append(pDict[percentile])
    plotDicts[0].update({'figsize': (8, 6), 'legendloc': 'upper right', 'yMin': 0})
    ph.setMetricBundles(plotBundles)
    ph.plot(plotFunc=plots.MetricVsH(), plotDicts=plotDicts, displayDict=displayDict) 
def plotNotFound(nChances, Hmark):
    pass
[docs]def plotActivity(bdict, figroot=None, resultsDb=None, outDir='.', figformat='pdf'):
    activity_deg = {}
    activity_days = {}
    for k in bdict:
        if 'Chances_of_detecting_activity' in k:
            if 'deg' in k:
                activity_deg[k] = bdict[k]
            if 'days' in k:
                activity_days[k] = bdict[k]
    if figroot is None:
        first = bdict[list(bdict.keys())[0]]
        figroot = first.runName
    displayDict = {'group': 'Characterization', 'subgroup': 'Activity'}
    # Plot (mean) likelihood of detection of activity over X days
    ph = plots.PlotHandler(figformat=figformat, resultsDb=resultsDb, outDir=outDir)
    ph.setMetricBundles(activity_days)
    ph.jointMetricNames = 'Chances of detecting activity lasting X days'
    plotDict = {'ylabel': "Mean likelihood of detection", 'figsize': (8, 6)}
    ph.plot(plotFunc=plots.MetricVsH(), plotDicts=plotDict, displayDict=displayDict,
            outfileRoot=figroot + '_activityDays')
    # Plot (mean) likelihood of detection of activity over X amount of orbit
    ph.setMetricBundles(activity_deg)
    ph.jointMetricNames = 'Chances of detecting activity covering X deg'
    plotDict = {'ylabel': "Mean likelihood of detection", 'figsize': (8, 6)}
    ph.plot(plotFunc=plots.MetricVsH(), plotDicts=plotDict, displayDict=displayDict,
            outfileRoot=figroot + '_activityDeg') 
[docs]def readAndCombine(orbitRoot, baseDir, splits, metricfile):
    """Read and combine the metric results from split locations, returning a single bundle.
    This will read the files from
    baseDir/orbitRoot_[split]/metricfile
    where split = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9], etc. (the subsets the original orbit file was split into).
    Parameters
    ----------
    orbitRoot: str
        The root of the orbit file - l7_5k, mbas_5k, etc.
    baseDir: str
        The root directory containing the subset directories. (e.g. '.' often)
    splits: np.ndarray or list of ints
        The integers describing the split directories (e.g. [0, 1, 2, 3, 4, 5, 6, 7, 8, 9])
    metricfile: str
        The metric filename.
    Returns
    -------
    ~lsst.sims.maf.bundle
        A single metric bundle containing the combined data from each of the subsets.
    Note that this won't work for particularly complex metric values, such as the parent Discovery metrics.
    However, you can read and combine their child metrics, as for these we can propagate the data masks.
    """
    subsets = {}
    for i in splits:
        subsets[i] = mb.createEmptyMoMetricBundle()
        ddir = os.path.join(baseDir, f'{orbitRoot}_{i}')
        subsets[i].read(os.path.join(ddir, metricfile))
    bundle = combineSubsets(subsets)
    return bundle 
[docs]def combineSubsets(mbSubsets):
    # Combine the data from the subset metric bundles.
    # The first bundle will be used a template for the slicer.
    if isinstance(mbSubsets, dict):
        first = mbSubsets[list(mbSubsets.keys())[0]]
    else:
        first = mbSubsets[0]
        subsetdict = {}
        for i, b in enumerate(mbSubsets):
            subsetdict[i] = b
        mbSubsets = subsetdict
    joint = mb.createEmptyMoMetricBundle()
    # Check if they're the same slicer.
    slicer = deepcopy(first.slicer)
    for i in mbSubsets:
        if np.any(slicer.slicePoints['H'] != mbSubsets[i].slicer.slicePoints['H']):
            if np.any(slicer.slicePoints['orbits'] != mbSubsets[i].slicer.slicePoints['orbits']):
                raise ValueError('Bundle %s has a different slicer than the first bundle' % (i))
    # Join metric values.
    joint.slicer = slicer
    joint.metric = first.metric
    # Don't just use the slicer shape to define the metricValues, because of CompletenessBundles.
    metricValues = np.zeros(first.metricValues.shape, float)
    metricValuesMask = np.zeros(first.metricValues.shape, bool)
    for i in mbSubsets:
        metricValues += mbSubsets[i].metricValues.filled(0)
        metricValuesMask = np.where(metricValuesMask & mbSubsets[i].metricValues.mask, True, False)
    joint.metricValues = ma.MaskedArray(data=metricValues, mask=metricValuesMask, fill_value=0)
    joint.metadata = first.metadata
    joint.runName = first.runName
    joint.fileRoot = first.fileRoot.replace('.npz', '')
    joint.plotDict = first.plotDict
    return joint