5from numpy.testing
import assert_allclose, assert_array_equal
7from ._astshimLib
import Channel, FitsChan, PolyMap, XmlChan, StringStream
11 """Base class for unit tests of objects
15 """Assert that two astshim objects are identical.
17 Identical means the objects are of the same class (
if checkType)
18 and all properties are identical (including whether set
or defaulted).
21 self.assertIs(
type(obj1),
type(obj2))
22 self.assertEqual(obj1.show(), obj2.show())
23 self.assertEqual(str(obj1), str(obj2))
24 self.assertEqual(repr(obj1), repr(obj2))
27 """Check that an astshim object can be deep-copied
29 nobj = obj.getNObject()
30 nref = obj.getRefCount()
36 for cp
in copyIter(obj):
38 self.assertEqual(obj.getNObject(), nobj + 1)
41 self.assertEqual(obj.getRefCount(), nref)
42 self.assertFalse(obj.same(cp))
43 self.assertEqual(cp.getNObject(), nobj + 1)
44 self.assertEqual(cp.getRefCount(), 1)
46 originalIdent = obj.ident
47 cp.ident = obj.ident +
" modified"
48 self.assertEqual(obj.ident, originalIdent)
51 self.assertEqual(obj.getNObject(), nobj)
52 self.assertEqual(obj.getRefCount(), nref)
55 """Check that an astshim object can be persisted and unpersisted
57 @param[
in] obj Object to be checked
58 @param[
in] typeFromChannel Type of object expected to be read
from
59 a channel (since some thin wrapper types are read
60 as the underlying type);
None if the original type
62 Check persistence using Channel, FitsChan (
with native encoding,
63 as the only encoding compatible
with all AST objects), XmlChan
66 for channelType, options
in (
68 (FitsChan,
"Encoding=Native"),
72 chan = channelType(ss, options)
75 if channelType
is FitsChan:
77 obj_copy = chan.read()
78 if typeFromChannel
is not None:
79 self.assertIs(
type(obj_copy), typeFromChannel)
84 obj_copy = pickle.loads(pickle.dumps(obj))
90 """Base class for unit tests of mappings
94 """Check that a mapping's reverse transform is the opposite of forward
96 amap is the mapping to test
97 poslist
is a list of input position
for a forward transform;
98 a numpy array
with shape [nin, num points]
99 or collection that can be cast to same
100 rtol
is the relative tolerance
for numpy.testing.assert_allclose
101 atol
is the absolute tolerance
for numpy.testing.assert_allclose
103 poslist = np.array(poslist, dtype=float)
104 if len(poslist.shape) == 1:
106 poslist.shape = (1, len(poslist))
108 to_poslist = amap.applyForward(poslist)
109 rt_poslist = amap.applyInverse(to_poslist)
110 assert_allclose(poslist, rt_poslist, rtol=rtol, atol=atol)
113 amapinv = amap.inverted()
114 rt2_poslist = amapinv.applyForward(to_poslist)
115 assert_allclose(poslist, rt2_poslist, rtol=rtol, atol=atol)
118 acmp = amap.then(amapinv)
119 assert_allclose(poslist, acmp.applyForward(poslist), rtol=rtol, atol=atol)
122 posvec =
list(poslist.flat)
123 to_posvec = amap.applyForward(posvec)
126 assert_allclose(to_posvec,
list(to_poslist.flat), rtol=rtol, atol=atol)
128 rt_posvec = amap.applyInverse(to_posvec)
129 assert_allclose(posvec, rt_posvec, rtol=rtol, atol=atol)
132 """Check basic simplfication for a reversible mapping
135 - A compound mapping of a amap and its inverse simplifies to
137 - A compound mapping of a amap
and a unit amap simplifies to
140 amapinv = amap.inverted()
141 cmp1 = amap.then(amapinv)
142 unit1 = cmp1.simplified()
143 self.assertEqual(unit1.className, "UnitMap")
144 self.assertEqual(amap.nIn, cmp1.nIn)
145 self.assertEqual(amap.nIn, cmp1.nOut)
146 self.assertEqual(cmp1.nIn, unit1.nIn)
147 self.assertEqual(cmp1.nOut, unit1.nOut)
149 cmp2 = amapinv.then(amap)
150 unit2 = cmp2.simplified()
151 self.assertEqual(unit2.className,
"UnitMap")
152 self.assertEqual(amapinv.nIn, cmp2.nIn)
153 self.assertEqual(amapinv.nIn, cmp2.nOut)
154 self.assertEqual(cmp2.nIn, unit2.nIn)
155 self.assertEqual(cmp2.nOut, unit2.nOut)
157 for ma, mb, desmap3
in (
160 (unit2, amapinv, amapinv),
161 (amapinv, unit1, amapinv),
164 cmp3simp = cmp3.simplified()
165 self.assertEqual(cmp3simp.className, amap.simplified().className)
166 self.assertEqual(ma.nIn, cmp3.nIn)
167 self.assertEqual(mb.nOut, cmp3.nOut)
168 self.assertEqual(cmp3.nIn, cmp3simp.nIn)
169 self.assertEqual(cmp3.nOut, cmp3simp.nOut)
172 """Check that a mapping gives identical answers to unpersisted copy
174 poslist is a list of input position
for a forward transform
175 (
if it exists),
or the inverse transform (
if not).
176 A numpy array
with shape [nAxes, num points]
177 or collection that can be cast to same
179 Checks each direction,
if present. However,
for generality,
180 does
not check that the two directions are inverses of each other;
181 call checkRoundTrip
for that.
183 Does everything checkPersistence does, so no need to call both.
185 for channelType, options
in (
187 (FitsChan,
"Encoding=Native"),
194 amap_copy = chan.read()
195 self.assertEqual(amap.className, amap_copy.className)
196 self.assertEqual(amap.show(), amap_copy.show())
197 self.assertEqual(str(amap), str(amap_copy))
198 self.assertEqual(repr(amap), repr(amap_copy))
201 outPoslist = amap.applyForward(poslist)
202 assert_array_equal(outPoslist, amap_copy.applyForward(poslist))
205 assert_array_equal(amap.applyInverse(outPoslist),
206 amap_copy.applyInverse(outPoslist))
208 elif amap.hasInverse:
209 assert_array_equal(amap.applyInverse(poslist),
210 amap_copy.applyInverse(poslist))
213 raise RuntimeError(
"mapping has neither forward nor inverse transform")
216 """Check the memory usage for a compoundObject
218 obj1: first object in compound object
219 obj2: second object
in compound object
220 cmpObj: compound object (SeriesMap, ParallelMap, CmpMap
or CmpFrame)
221 isSeries:
is compound object
in series?
None to
not test
226 deltaObj = 2
if type(obj1) ==
type(obj2)
else 1
228 initialNumObj1 = obj1.getNObject()
229 initialNumObj2 = obj2.getNObject()
230 initialNumCmpObj = cmpObj.getNObject()
231 initialRefCountObj1 = obj1.getRefCount()
232 initialRefCountObj2 = obj2.getRefCount()
233 initialRefCountCmpObj = cmpObj.getRefCount()
234 self.assertEqual(obj1.getNObject(), initialNumObj1)
235 self.assertEqual(obj2.getNObject(), initialNumObj2)
236 if isSeries
is not None:
238 self.assertTrue(cmpObj.series)
239 elif isSeries
is False:
240 self.assertFalse(cmpObj.series)
245 self.assertEqual(cmpObj.getRefCount(), initialRefCountCmpObj)
246 self.assertEqual(cmpObj.getNObject(), initialNumCmpObj + 1)
247 self.assertEqual(obj1.getRefCount(), initialRefCountObj1)
248 self.assertEqual(obj2.getRefCount(), initialRefCountObj2)
249 self.assertEqual(obj1.getNObject(), initialNumObj1 + deltaObj)
250 self.assertEqual(obj2.getNObject(), initialNumObj2 + deltaObj)
254 self.assertEqual(cmpObj.getRefCount(), initialRefCountCmpObj)
255 self.assertEqual(cmpObj.getNObject(), initialNumCmpObj)
256 self.assertEqual(obj1.getRefCount(), initialRefCountObj1)
257 self.assertEqual(obj1.getNObject(), initialNumObj1)
258 self.assertEqual(obj2.getRefCount(), initialRefCountObj2)
259 self.assertEqual(obj2.getNObject(), initialNumObj2)
263 """Make an array of coefficients for astshim.PolyMap for the following
266 fj(x) = C0j x0^2 + C1j x1^2 + C2j x2^2 + ... + CNj xN^2
268 * i ranges from 0 to N=nIn-1
269 * j ranges
from 0 to nOut-1,
270 * Cij = 0.001 (i+j+1)
274 for out_ind
in range(nOut):
275 coeffOffset = baseCoeff * out_ind
276 for in_ind
in range(nIn):
277 coeff = baseCoeff * (in_ind + 1) + coeffOffset
278 coeffArr = [coeff, out_ind + 1] + [2
if i == in_ind
else 0
for i
in range(nIn)]
279 forwardCoeffs.append(coeffArr)
280 return np.array(forwardCoeffs, dtype=float)
283def makeTwoWayPolyMap(nIn, nOut):
284 """Make an astshim.PolyMap suitable for testing
286 The forward transform is as follows:
287 fj(x) = C0j x0^2 + C1j x1^2 + C2j x2^2 + ...
288 + CNj xN^2 where Cij = 0.001 (i+j+1)
290 The reverse transform
is the same equation
with i
and j reversed
291 thus it
is NOT the inverse of the forward direction,
292 but
is something that can be easily evaluated.
294 The equation
is chosen
for the following reasons:
295 - It
is well defined
for any positive value of nIn, nOut.
296 - It stays small
for small x, to avoid wraparound of angles
for
297 SpherePoint endpoints.
301 polyMap = PolyMap(forwardCoeffs, reverseCoeffs)
302 assert polyMap.nIn == nIn
303 assert polyMap.nOut == nOut
304 assert polyMap.hasForward
305 assert polyMap.hasInverse
309def makeForwardPolyMap(nIn, nOut):
310 """Make an astshim.PolyMap suitable for testing
312 The forward transform is the same
as for `makeTwoWayPolyMap`.
313 This map does
not have a reverse transform.
315 The equation
is chosen
for the following reasons:
316 - It
is well defined
for any positive value of nIn, nOut.
317 - It stays small
for small x, to avoid wraparound of angles
for
318 SpherePoint endpoints.
321 polyMap = PolyMap(forwardCoeffs, nOut, "IterInverse=0")
322 assert polyMap.nIn == nIn
323 assert polyMap.nOut == nOut
324 assert polyMap.hasForward
325 assert not polyMap.hasInverse
checkRoundTrip(self, amap, poslist, rtol=1e-05, atol=1e-08)
checkMemoryForCompoundObject(self, obj1, obj2, cmpObj, isSeries)
checkBasicSimplify(self, amap)
checkMappingPersistence(self, amap, poslist)
assertObjectsIdentical(self, obj1, obj2, checkType=True)
checkPersistence(self, obj, typeFromChannel=None)
daf::base::PropertyList * list
makePolyMapCoeffs(nIn, nOut)