LSST Applications g0b6bd0c080+a72a5dd7e6,g1182afd7b4+2a019aa3bb,g17e5ecfddb+2b8207f7de,g1d67935e3f+06cf436103,g38293774b4+ac198e9f13,g396055baef+6a2097e274,g3b44f30a73+6611e0205b,g480783c3b1+98f8679e14,g48ccf36440+89c08d0516,g4b93dc025c+98f8679e14,g5c4744a4d9+a302e8c7f0,g613e996a0d+e1c447f2e0,g6c8d09e9e7+25247a063c,g7271f0639c+98f8679e14,g7a9cd813b8+124095ede6,g9d27549199+a302e8c7f0,ga1cf026fa3+ac198e9f13,ga32aa97882+7403ac30ac,ga786bb30fb+7a139211af,gaa63f70f4e+9994eb9896,gabf319e997+ade567573c,gba47b54d5d+94dc90c3ea,gbec6a3398f+06cf436103,gc6308e37c7+07dd123edb,gc655b1545f+ade567573c,gcc9029db3c+ab229f5caf,gd01420fc67+06cf436103,gd877ba84e5+06cf436103,gdb4cecd868+6f279b5b48,ge2d134c3d5+cc4dbb2e3f,ge448b5faa6+86d1ceac1d,gecc7e12556+98f8679e14,gf3ee170dca+25247a063c,gf4ac96e456+ade567573c,gf9f5ea5b4d+ac198e9f13,gff490e6085+8c2580be5c,w.2022.27
LSST Data Management Base Package
Public Member Functions | Static Public Attributes | List of all members
lsst.skymap.ringsSkyMap.RingsSkyMap Class Reference
Inheritance diagram for lsst.skymap.ringsSkyMap.RingsSkyMap:
lsst.skymap.cachingSkyMap.CachingSkyMap lsst.skymap.baseSkyMap.BaseSkyMap

Public Member Functions

def __init__ (self, config, version=1)
 
def getRingIndices (self, index)
 
def generateTract (self, index)
 
def findTract (self, coord)
 
def findTractIdArray (self, ra, dec, degrees=False)
 
def findAllTracts (self, coord)
 
def findTractPatchList (self, coordList)
 
def updateSha1 (self, sha1)
 

Static Public Attributes

 ConfigClass = RingsSkyMapConfig
 

Detailed Description

Rings sky map pixelization.

We divide the sphere into N rings of Declination, plus the two polar
caps, which sets the size of the individual tracts.  The rings are
divided in RA into an integral number of tracts of this size; this
division is made at the Declination closest to zero so as to ensure
full overlap.

Rings are numbered in the rings from south to north. The south pole cap is
``tract=0``, then the tract at ``raStart`` in the southernmost ring is
``tract=1``. Numbering continues (in the positive RA direction) around that
ring and then continues in the same fashion with the next ring north, and
so on until all reaching the north pole cap, which is
``tract=len(skymap) - 1``.

However, ``version=0`` had a bug in the numbering of the tracts: the first
and last tracts in the first (southernmost) ring were identical, and the
first tract in the last (northernmost) ring was missing. When using
``version=0``, these tracts remain missing in order to preserve the
numbering scheme.

Parameters
----------
config : `lsst.skymap.RingsSkyMapConfig`
    The configuration for this SkyMap.
version : `int`, optional
    Software version of this class, to retain compatibility with old
    verisons. ``version=0`` covers the period from first implementation
    until DM-14809, at which point bugs were identified in the numbering
    of tracts (affecting only tracts at RA=0). ``version=1`` uses the
    post-DM-14809 tract numbering.

Definition at line 42 of file ringsSkyMap.py.

Constructor & Destructor Documentation

◆ __init__()

def lsst.skymap.ringsSkyMap.RingsSkyMap.__init__ (   self,
  config,
  version = 1 
)

Reimplemented from lsst.skymap.cachingSkyMap.CachingSkyMap.

Definition at line 78 of file ringsSkyMap.py.

78 def __init__(self, config, version=1):
79 assert version in (0, 1), "Unrecognised version: %s" % (version,)
80 # We count rings from south to north
81 # Note: pole caps together count for one additional ring when calculating the ring size
82 self._ringSize = math.pi / (config.numRings + 1) # Size of a ring in Declination (radians)
83 self._ringNums = [] # Number of tracts for each ring
84 for i in range(config.numRings):
85 startDec = self._ringSize*(i + 0.5) - 0.5*math.pi
86 stopDec = startDec + self._ringSize
87 dec = min(math.fabs(startDec), math.fabs(stopDec)) # Declination for determining division in RA
88 self._ringNums.append(int(2*math.pi*math.cos(dec)/self._ringSize) + 1)
89 numTracts = sum(self._ringNums) + 2
90 super(RingsSkyMap, self).__init__(numTracts, config, version)
91 self._raStart = self.config.raStart*geom.degrees
92
int min
std::shared_ptr< FrameSet > append(FrameSet const &first, FrameSet const &second)
Construct a FrameSet that performs two transformations in series.
Definition: functional.cc:33

Member Function Documentation

◆ findAllTracts()

def lsst.skymap.ringsSkyMap.RingsSkyMap.findAllTracts (   self,
  coord 
)
Find all tracts which include the specified coord.

Parameters
----------
coord : `lsst.geom.SpherePoint`
    ICRS sky coordinate to search for.

Returns
-------
tractList : `list` of `TractInfo`
    The tracts which include the specified coord.

Definition at line 252 of file ringsSkyMap.py.

252 def findAllTracts(self, coord):
253 """Find all tracts which include the specified coord.
254
255 Parameters
256 ----------
257 coord : `lsst.geom.SpherePoint`
258 ICRS sky coordinate to search for.
259
260 Returns
261 -------
262 tractList : `list` of `TractInfo`
263 The tracts which include the specified coord.
264 """
265 ringNum = self._decToRingNum(coord.getLatitude())
266
267 tractList = list()
268 # ringNum denotes the closest ring to the specified coord
269 # I will check adjacent rings which may include the specified coord
270 for r in [ringNum - 1, ringNum, ringNum + 1]:
271 if r < 0 or r >= self.config.numRings:
272 # Poles will be checked explicitly outside this loop
273 continue
274 tractNum = self._raToTractNum(coord.getLongitude(), r)
275 # Adjacent tracts will also be checked.
276 for t in [tractNum - 1, tractNum, tractNum + 1]:
277 # Wrap over raStart
278 if t < 0:
279 t = t + self._ringNums[r]
280 elif t > self._ringNums[r] - 1:
281 t = t - self._ringNums[r]
282
283 extra = 0
284 if self._version == 0 and t == 0 and r != 0:
285 # Account for off-by-one error in getRingIndices
286 # Note that this means that tract 1 gets duplicated.
287 extra = 1
288
289 index = sum(self._ringNums[:r + extra], t + 1) # Allow 1 for south pole
290 tract = self[index]
291 if tract.contains(coord):
292 tractList.append(tract)
293
294 # Always check tracts at poles
295 # Southern cap is 0, Northern cap is the last entry in self
296 for entry in [0, len(self)-1]:
297 tract = self[entry]
298 if tract.contains(coord):
299 tractList.append(tract)
300
301 return tractList
302
Point in an unspecified spherical coordinate system.
Definition: SpherePoint.h:57
daf::base::PropertyList * list
Definition: fits.cc:913

◆ findTract()

def lsst.skymap.ringsSkyMap.RingsSkyMap.findTract (   self,
  coord 
)
Find the tract whose center is nearest the specified coord.

Parameters
----------
coord : `lsst.geom.SpherePoint`
    ICRS sky coordinate to search for.

Returns
-------
result : `TractInfo`
    TractInfo of tract whose center is nearest the specified coord.

Notes
-----
- If coord is equidistant between multiple sky tract centers then one
  is arbitrarily chosen.

- The default implementation is not very efficient; subclasses may wish
  to override.

**Warning:**
If tracts do not cover the whole sky then the returned tract may not
include the coord.

Reimplemented from lsst.skymap.baseSkyMap.BaseSkyMap.

Definition at line 187 of file ringsSkyMap.py.

187 def findTract(self, coord):
188 ringNum = self._decToRingNum(coord.getLatitude())
189 if ringNum == -1:
190 # Southern cap
191 return self[0]
192 if ringNum == self.config.numRings:
193 # Northern cap
194 return self[self._numTracts - 1]
195 tractNum = self._raToTractNum(coord.getLongitude(), ringNum)
196
197 if self._version == 0 and tractNum == 0 and ringNum != 0:
198 # Account for off-by-one error in getRingIndices
199 # Note that this means that tract 1 gets duplicated.
200 ringNum += 1
201
202 index = sum(self._ringNums[:ringNum], tractNum + 1) # Allow 1 for south pole
203 return self[index]
204

◆ findTractIdArray()

def lsst.skymap.ringsSkyMap.RingsSkyMap.findTractIdArray (   self,
  ra,
  dec,
  degrees = False 
)
Find array of tract IDs with vectorized operations (where supported).

If a given sky map does not support vectorized operations, then a loop
over findTract will be called.

Parameters
----------
ra : `np.ndarray`
    Array of Right Ascension.  Units are radians unless
    degrees=True.
dec : `np.ndarray`
    Array of Declination.  Units are radians unless
    degrees=True.
degrees : `bool`, optional
    Input ra, dec arrays are degrees if True.

Returns
-------
tractId : `np.ndarray`
    Array of tract IDs

Notes
-----
- If coord is equidistant between multiple sky tract centers then one
  is arbitrarily chosen.

**Warning:**
If tracts do not cover the whole sky then the returned tract may not
include the given ra/dec.

Reimplemented from lsst.skymap.baseSkyMap.BaseSkyMap.

Definition at line 205 of file ringsSkyMap.py.

205 def findTractIdArray(self, ra, dec, degrees=False):
206 if degrees:
207 _ra = np.deg2rad(ra)
208 _dec = np.deg2rad(dec)
209 else:
210 _ra = np.atleast_1d(ra)
211 _dec = np.atleast_1d(dec)
212
213 # Set default to -1 to distinguish from polar tracts
214 indexes = np.full(_ra.size, -1, dtype=np.int32)
215
216 # Do the dec search
217 firstRingStart = self._ringSize*0.5 - 0.5*np.pi
218 ringNums = np.zeros(len(_dec), dtype=np.int32)
219
220 ringNums[_dec < firstRingStart] = -1
221 ringNums[_dec > -1*firstRingStart] = self.config.numRings
222
223 mid = (_dec >= firstRingStart) & (_dec <= -1*firstRingStart)
224 ringNums[mid] = ((_dec[mid] - firstRingStart)/self._ringSize).astype(np.int32)
225
226 indexes[ringNums == -1] = 0
227 indexes[ringNums == self.config.numRings] = self._numTracts - 1
228
229 # We now do the full lookup for all non-polar tracts that have not been set.
230 inRange, = np.where(indexes < 0)
231
232 # Do the ra search
233 _ringNumArray = np.array(self._ringNums)
234 _ringCumulative = np.cumsum(np.insert(_ringNumArray, 0, 0))
235
236 deltaWrap = (_ra[inRange] - self._raStart.asRadians()) % (2.*np.pi)
237 tractNum = (deltaWrap/(2.*np.pi/_ringNumArray[ringNums[inRange]]) + 0.5).astype(np.int32)
238 # Allow wraparound
239 tractNum[tractNum == _ringNumArray[ringNums[inRange]]] = 0
240
241 if self._version == 0:
242 # Account for off-by-one error in getRingIndices
243 # Note that this means tract 1 gets duplicated
244 offByOne, = np.where((tractNum == 0)
245 & (ringNums[inRange] != 0))
246 ringNums[inRange[offByOne]] += 1
247
248 indexes[inRange] = _ringCumulative[ringNums[inRange]] + tractNum + 1
249
250 return indexes
251

◆ findTractPatchList()

def lsst.skymap.ringsSkyMap.RingsSkyMap.findTractPatchList (   self,
  coordList 
)
Find tracts and patches that overlap a region.

Parameters
----------
coordList : `list` of `lsst.geom.SpherePoint`
    List of ICRS sky coordinates to search for.

Returns
-------
reList : `list` of (`TractInfo`, `list` of `PatchInfo`)
    For tracts and patches that contain, or may contain, the specified
    region. The list will be empty if there is no overlap.

Notes
-----
**warning:**
    This uses a naive algorithm that may find some tracts and patches
    that do not overlap the region (especially if the region is not a
    rectangle aligned along patch x, y).

Reimplemented from lsst.skymap.baseSkyMap.BaseSkyMap.

Definition at line 303 of file ringsSkyMap.py.

303 def findTractPatchList(self, coordList):
304 retList = []
305 for coord in coordList:
306 for tractInfo in self.findAllTracts(coord):
307 patchList = tractInfo.findPatchList(coordList)
308 if patchList and not (tractInfo, patchList) in retList:
309 retList.append((tractInfo, patchList))
310 return retList
311

◆ generateTract()

def lsst.skymap.ringsSkyMap.RingsSkyMap.generateTract (   self,
  index 
)
Generate TractInfo for the specified tract index.

Reimplemented from lsst.skymap.cachingSkyMap.CachingSkyMap.

Definition at line 124 of file ringsSkyMap.py.

124 def generateTract(self, index):
125 """Generate TractInfo for the specified tract index."""
126 ringNum, tractNum = self.getRingIndices(index)
127 if ringNum == -1: # South polar cap
128 ra, dec = 0, -0.5*math.pi
129 elif ringNum == self.config.numRings: # North polar cap
130 ra, dec = 0, 0.5*math.pi
131 else:
132 dec = self._ringSize*(ringNum + 1) - 0.5*math.pi
133 ra = ((2*math.pi*tractNum/self._ringNums[ringNum])*geom.radians
134 + self._raStart).wrap().asRadians()
135
136 center = geom.SpherePoint(ra, dec, geom.radians)
137 wcs = self._wcsFactory.makeWcs(crPixPos=geom.Point2D(0, 0), crValCoord=center)
138 return ExplicitTractInfo(index, self._tractBuilder, center,
139 0.5*self._ringSize*geom.radians, self.config.tractOverlap*geom.degrees,
140 wcs)
141
std::shared_ptr< afw::geom::SkyWcs > makeWcs(SipForwardTransform const &sipForward, SipReverseTransform const &sipReverse, geom::SpherePoint const &skyOrigin)
Create a new TAN SIP Wcs from a pair of SIP transforms and the sky origin.
def wrap(ctrl)
Definition: wrap.py:305

◆ getRingIndices()

def lsst.skymap.ringsSkyMap.RingsSkyMap.getRingIndices (   self,
  index 
)
Calculate ring indices given a numerical index of a tract.

The ring indices are the ring number and the tract number within
the ring.

The ring number is -1 for the south polar cap and increases to the
north.  The north polar cap has ring number = numRings.  The tract
number is zero for either of the polar caps.

Definition at line 93 of file ringsSkyMap.py.

93 def getRingIndices(self, index):
94 """Calculate ring indices given a numerical index of a tract.
95
96 The ring indices are the ring number and the tract number within
97 the ring.
98
99 The ring number is -1 for the south polar cap and increases to the
100 north. The north polar cap has ring number = numRings. The tract
101 number is zero for either of the polar caps.
102 """
103 if index == 0: # South polar cap
104 return -1, 0
105 if index == self._numTracts - 1: # North polar cap
106 return self.config.numRings, 0
107 if index < 0 or index >= self._numTracts:
108 raise IndexError("Tract index %d is out of range [0, %d]" % (index, len(self) - 1))
109 ring = 0 # Ring number
110 tractNum = index - 1 # Tract number within ring
111 if self._version == 0:
112 # Maintain the off-by-one bug in version=0 (DM-14809).
113 # This means that the first tract in the first ring is duplicated
114 # and the first tract in the last ring is missing.
115 while ring < self.config.numRings and tractNum > self._ringNums[ring]:
116 tractNum -= self._ringNums[ring]
117 ring += 1
118 else:
119 while ring < self.config.numRings and tractNum >= self._ringNums[ring]:
120 tractNum -= self._ringNums[ring]
121 ring += 1
122 return ring, tractNum
123

◆ updateSha1()

def lsst.skymap.ringsSkyMap.RingsSkyMap.updateSha1 (   self,
  sha1 
)
Add subclass-specific state or configuration options to the SHA1.

Reimplemented from lsst.skymap.baseSkyMap.BaseSkyMap.

Definition at line 312 of file ringsSkyMap.py.

312 def updateSha1(self, sha1):
313 """Add subclass-specific state or configuration options to the SHA1."""
314 sha1.update(struct.pack("<id", self.config.numRings, self.config.raStart))

Member Data Documentation

◆ ConfigClass

lsst.skymap.ringsSkyMap.RingsSkyMap.ConfigClass = RingsSkyMapConfig
static

Definition at line 75 of file ringsSkyMap.py.


The documentation for this class was generated from the following file: