LSST Applications  21.0.0-172-gfb10e10a+18fedfabac,22.0.0+297cba6710,22.0.0+80564b0ff1,22.0.0+8d77f4f51a,22.0.0+a28f4c53b1,22.0.0+dcf3732eb2,22.0.1-1-g7d6de66+2a20fdde0d,22.0.1-1-g8e32f31+297cba6710,22.0.1-1-geca5380+7fa3b7d9b6,22.0.1-12-g44dc1dc+2a20fdde0d,22.0.1-15-g6a90155+515f58c32b,22.0.1-16-g9282f48+790f5f2caa,22.0.1-2-g92698f7+dcf3732eb2,22.0.1-2-ga9b0f51+7fa3b7d9b6,22.0.1-2-gd1925c9+bf4f0e694f,22.0.1-24-g1ad7a390+a9625a72a8,22.0.1-25-g5bf6245+3ad8ecd50b,22.0.1-25-gb120d7b+8b5510f75f,22.0.1-27-g97737f7+2a20fdde0d,22.0.1-32-gf62ce7b1+aa4237961e,22.0.1-4-g0b3f228+2a20fdde0d,22.0.1-4-g243d05b+871c1b8305,22.0.1-4-g3a563be+32dcf1063f,22.0.1-4-g44f2e3d+9e4ab0f4fa,22.0.1-42-gca6935d93+ba5e5ca3eb,22.0.1-5-g15c806e+85460ae5f3,22.0.1-5-g58711c4+611d128589,22.0.1-5-g75bb458+99c117b92f,22.0.1-6-g1c63a23+7fa3b7d9b6,22.0.1-6-g50866e6+84ff5a128b,22.0.1-6-g8d3140d+720564cf76,22.0.1-6-gd805d02+cc5644f571,22.0.1-8-ge5750ce+85460ae5f3,master-g6e05de7fdc+babf819c66,master-g99da0e417a+8d77f4f51a,w.2021.48
LSST Data Management Base Package
skyObjects.py
Go to the documentation of this file.
1 
2 __all__ = ["SkyObjectsConfig", "SkyObjectsTask", "generateSkyObjects"]
3 
4 from lsst.pex.config import Config, Field, ListField
5 from lsst.pipe.base import Task
6 
8 import lsst.afw.geom
9 import lsst.afw.math
10 
11 
13  """Configuration for generating sky objects"""
14  avoidMask = ListField(dtype=str, default=["DETECTED", "DETECTED_NEGATIVE", "BAD", "NO_DATA"],
15  doc="Avoid pixels masked with these mask planes")
16  growMask = Field(dtype=int, default=0,
17  doc="Number of pixels to grow the masked pixels when adding sky objects")
18  sourceRadius = Field(dtype=float, default=8, doc="Radius, in pixels, of sky objects")
19  nSources = Field(dtype=int, default=100, doc="Try to add this many sky objects")
20  nTrialSources = Field(dtype=int, default=None, optional=True,
21  doc="Maximum number of trial sky object positions\n"
22  "(default: nSkySources*nTrialSkySourcesMultiplier)")
23  nTrialSourcesMultiplier = Field(dtype=int, default=5,
24  doc="Set nTrialSkySources to\n"
25  " nSkySources*nTrialSkySourcesMultiplier\n"
26  "if nTrialSkySources is None")
27 
28 
29 def generateSkyObjects(mask, seed, config):
30  """Generate a list of Footprints of sky objects
31 
32  Sky objects don't overlap with other objects. This is determined
33  through the provided `mask` (in which objects are typically flagged
34  as `DETECTED`).
35 
36  The algorithm for determining sky objects is random trial and error:
37  we try up to `nTrialSkySources` random positions to find `nSources`
38  sky objects.
39 
40  Parameters
41  ----------
42  mask : `lsst.afw.image.Mask`
43  Input mask plane, which identifies pixels to avoid for the sky
44  objects.
45  seed : `int`
46  Random number generator seed.
47  config : `SkyObjectsConfig`
48  Configuration for finding sky objects.
49 
50  Returns
51  -------
52  skyFootprints : `list` of `lsst.afw.detection.Footprint`
53  Footprints of sky objects. Each will have a peak at the center
54  of the sky object.
55  """
56  if config.nSources <= 0:
57  return []
58 
59  skySourceRadius = config.sourceRadius
60  nSkySources = config.nSources
61  nTrialSkySources = config.nTrialSources
62  if nTrialSkySources is None:
63  nTrialSkySources = config.nTrialSourcesMultiplier*nSkySources
64 
65  box = mask.getBBox()
66  box.grow(-(int(skySourceRadius) + 1)) # Avoid objects partially off the image
67  xMin, yMin = box.getMin()
68  xMax, yMax = box.getMax()
69 
70  avoid = lsst.afw.geom.SpanSet.fromMask(mask, mask.getPlaneBitMask(config.avoidMask))
71  if config.growMask > 0:
72  avoid = avoid.dilated(config.growMask)
73 
74  rng = lsst.afw.math.Random(seed=seed)
75 
76  skyFootprints = []
77  for _ in range(nTrialSkySources):
78  if len(skyFootprints) == nSkySources:
79  break
80 
81  x = int(rng.flat(xMin, xMax))
82  y = int(rng.flat(yMin, yMax))
83  spans = lsst.afw.geom.SpanSet.fromShape(int(skySourceRadius), offset=(x, y))
84  if spans.overlaps(avoid):
85  continue
86 
87  fp = lsst.afw.detection.Footprint(spans, mask.getBBox())
88  fp.addPeak(x, y, 0)
89  skyFootprints.append(fp)
90 
91  return skyFootprints
92 
93 
94 class SkyObjectsTask(Task):
95  """Generate a list of Footprints of sky objects.
96  """
97  ConfigClass = SkyObjectsConfig
98 
99  def run(self, mask, seed):
100  """Generate a list of Footprints of sky objects
101 
102  Sky objects don't overlap with other objects. This is determined
103  through the provided `mask` (in which objects are typically flagged
104  as `DETECTED`).
105 
106  The algorithm for determining sky objects is random trial and error:
107  we try up to `nTrialSkySources` random positions to find `nSources`
108  sky objects.
109 
110  Parameters
111  ----------
112  mask : `lsst.afw.image.Mask`
113  Input mask plane, which identifies pixels to avoid for the sky
114  objects.
115  seed : `int`
116  Random number generator seed.
117 
118  Returns
119  -------
120  skyFootprints : `list` of `lsst.afw.detection.Footprint`
121  Footprints of sky objects. Each will have a peak at the center
122  of the sky object.
123  """
124  skyFootprints = generateSkyObjects(mask, seed, self.config)
125  self.log.info("Added %d of %d requested sky sources (%.0f%%)", len(skyFootprints),
126  self.config.nSources, 100*len(skyFootprints)/self.config.nSources)
127  return skyFootprints
Class to describe the properties of a detected object from an image.
Definition: Footprint.h:63
static std::shared_ptr< geom::SpanSet > fromShape(int r, Stencil s=Stencil::CIRCLE, lsst::geom::Point2I offset=lsst::geom::Point2I())
Factory function for creating SpanSets from a Stencil.
Definition: SpanSet.cc:688
static std::shared_ptr< geom::SpanSet > fromMask(image::Mask< T > const &mask, UnaryPredicate comparator=details::AnyBitSetFunctor< T >())
Create a SpanSet from a mask.
Definition: SpanSet.h:644
A class that can be used to generate sequences of random numbers according to a number of different a...
Definition: Random.h:57
def generateSkyObjects(mask, seed, config)
Definition: skyObjects.py:29