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
_transform.cc
Go to the documentation of this file.
1 /*
2  * This file is part of afw.
3  *
4  * Developed for the LSST Data Management System.
5  * This product includes software developed by the LSST Project
6  * (https://www.lsst.org).
7  * See the COPYRIGHT file at the top-level directory of this distribution
8  * for details of code ownership.
9  *
10  * This program is free software: you can redistribute it and/or modify
11  * it under the terms of the GNU General Public License as published by
12  * the Free Software Foundation, either version 3 of the License, or
13  * (at your option) any later version.
14  *
15  * This program is distributed in the hope that it will be useful,
16  * but WITHOUT ANY WARRANTY; without even the implied warranty of
17  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18  * GNU General Public License for more details.
19  *
20  * You should have received a copy of the GNU General Public License
21  * along with this program. If not, see <https://www.gnu.org/licenses/>.
22  */
23 
24 #include "pybind11/pybind11.h"
25 #include "lsst/utils/python.h"
26 #include "pybind11/eigen.h"
27 
28 #include <memory>
29 
30 #include "astshim.h"
31 #include "pybind11/stl.h"
32 #include "ndarray/pybind11.h"
33 
36 #include "lsst/afw/geom/Endpoint.h"
38 
39 namespace py = pybind11;
40 using namespace py::literals;
41 
42 namespace lsst {
43 namespace afw {
44 namespace geom {
45 namespace {
46 
47 // Return a string consisting of "_pythonClassName_[_fromNAxes_->_toNAxes_]",
48 // for example "TransformGenericToPoint2[4->2]"
49 template <class Class>
50 std::string formatStr(Class const &self, std::string const &pyClassName) {
52  os << pyClassName;
53  os << "[" << self.getFromEndpoint().getNAxes() << "->" << self.getToEndpoint().getNAxes() << "]";
54  return os.str();
55 }
56 
57 template <class FromEndpoint, class ToEndpoint, class NextToEndpoint, class PyClass>
58 void declareMethodTemplates(PyClass &cls) {
59  using ThisTransform = Transform<FromEndpoint, ToEndpoint>;
60  using NextTransform = Transform<ToEndpoint, NextToEndpoint>;
61  using SeriesTransform = Transform<FromEndpoint, NextToEndpoint>;
62  // Need Python-specific logic to give sensible errors for mismatched Transform types
63  cls.def("_then",
64  (std::shared_ptr<SeriesTransform>(ThisTransform::*)(NextTransform const &, bool) const) &
65  ThisTransform::template then<NextToEndpoint>,
66  "next"_a, "simplify"_a = true);
67 }
68 
69 // Declare Transform<FromEndpoint, ToEndpoint> using python class name Transform<X>To<Y>
70 // where <X> and <Y> are the prefix of the from endpoint and to endpoint class, respectively,
71 // for example TransformGenericToPoint2
72 template <class FromEndpoint, class ToEndpoint>
73 void declareTransform(lsst::utils::python::WrapperCollection &wrappers) {
74  using Class = Transform<FromEndpoint, ToEndpoint>;
75  using ToPoint = typename ToEndpoint::Point;
76  using ToArray = typename ToEndpoint::Array;
77  using FromPoint = typename FromEndpoint::Point;
78  using FromArray = typename FromEndpoint::Array;
79 
80  std::string const pyClassName = Class::getShortClassName();
81  wrappers.wrapType(
82  py::class_<Class, std::shared_ptr<Class>, table::io::Persistable>(wrappers.module, pyClassName.c_str()),
83  [](auto &mod, auto &cls) {
84  std::string const pyClassName = Class::getShortClassName();
85  cls.def(py::init<ast::FrameSet const &, bool>(), "frameSet"_a, "simplify"_a = true);
86  cls.def(py::init<ast::Mapping const &, bool>(), "mapping"_a, "simplify"_a = true);
87 
88  cls.def_property_readonly("hasForward", &Class::hasForward);
89  cls.def_property_readonly("hasInverse", &Class::hasInverse);
90  cls.def_property_readonly("fromEndpoint", &Class::getFromEndpoint);
91  cls.def_property_readonly("toEndpoint", &Class::getToEndpoint);
92 
93  // Return a copy of the contained Mapping in order to assure changing the returned Mapping
94  // will not affect the contained Mapping (since Python ignores constness)
95  cls.def("getMapping", [](Class const &self) { return self.getMapping()->copy(); });
96 
97  cls.def("applyForward",
98  py::overload_cast<FromArray const &>(&Class::applyForward, py::const_), "array"_a);
99  cls.def("applyForward",
100  py::overload_cast<FromPoint const &>(&Class::applyForward, py::const_), "point"_a);
101  cls.def("applyInverse", py::overload_cast<ToArray const &>(&Class::applyInverse, py::const_),
102  "array"_a);
103  cls.def("applyInverse", py::overload_cast<ToPoint const &>(&Class::applyInverse, py::const_),
104  "point"_a);
105  cls.def("inverted", &Class::inverted);
106  /* Need some extra handling of ndarray return type in Python to prevent dimensions
107  * of length 1 from being deleted */
108  cls.def("_getJacobian", &Class::getJacobian);
109  // Do not wrap getShortClassName because it returns the name of the class;
110  // use `<class>.__name__` or `type(<instance>).__name__` instead.
111  // Do not wrap readStream or writeStream because C++ streams are not easy to wrap.
112  cls.def_static("readString", &Class::readString);
113  cls.def("writeString", &Class::writeString);
114 
115  declareMethodTemplates<FromEndpoint, ToEndpoint, GenericEndpoint>(cls);
116  declareMethodTemplates<FromEndpoint, ToEndpoint, Point2Endpoint>(cls);
117  declareMethodTemplates<FromEndpoint, ToEndpoint, SpherePointEndpoint>(cls);
118 
119  // str(self) = "<Python class name>[<nIn>-><nOut>]"
120  cls.def("__str__", [pyClassName](Class const &self) { return formatStr(self, pyClassName); });
121  // repr(self) = "lsst.afw.geom.<Python class name>[<nIn>-><nOut>]"
122  cls.def("__repr__", [pyClassName](Class const &self) {
123  return "lsst.afw.geom." + formatStr(self, pyClassName);
124  });
125 
126  table::io::python::addPersistableMethods<Class>(cls);
127  });
128 }
129 } // namespace
130 void wrapTransform(lsst::utils::python::WrapperCollection &wrappers) {
131  wrappers.addSignatureDependency("lsst.afw.table.io");
132  wrappers.addSignatureDependency("astshim");
133  declareTransform<GenericEndpoint, GenericEndpoint>(wrappers);
134  declareTransform<GenericEndpoint, Point2Endpoint>(wrappers);
135  declareTransform<GenericEndpoint, SpherePointEndpoint>(wrappers);
136  declareTransform<Point2Endpoint, GenericEndpoint>(wrappers);
137  declareTransform<Point2Endpoint, Point2Endpoint>(wrappers);
138  declareTransform<Point2Endpoint, SpherePointEndpoint>(wrappers);
139  declareTransform<SpherePointEndpoint, GenericEndpoint>(wrappers);
140  declareTransform<SpherePointEndpoint, Point2Endpoint>(wrappers);
141  declareTransform<SpherePointEndpoint, SpherePointEndpoint>(wrappers);
142 }
143 } // namespace geom
144 } // namespace afw
145 } // namespace lsst
std::ostream * os
Definition: Schema.cc:557
T c_str(T... args)
void wrapTransform(lsst::utils::python::WrapperCollection &)
Definition: _transform.cc:130
py::class_< PixelAreaBoundedField, std::shared_ptr< PixelAreaBoundedField >, BoundedField > PyClass
A base class for image defects.