LSST Applications  21.0.0-147-g0e635eb1+1acddb5be5,22.0.0+052faf71bd,22.0.0+1ea9a8b2b2,22.0.0+6312710a6c,22.0.0+729191ecac,22.0.0+7589c3a021,22.0.0+9f079a9461,22.0.1-1-g7d6de66+b8044ec9de,22.0.1-1-g87000a6+536b1ee016,22.0.1-1-g8e32f31+6312710a6c,22.0.1-10-gd060f87+016f7cdc03,22.0.1-12-g9c3108e+df145f6f68,22.0.1-16-g314fa6d+c825727ab8,22.0.1-19-g93a5c75+d23f2fb6d8,22.0.1-19-gb93eaa13+aab3ef7709,22.0.1-2-g8ef0a89+b8044ec9de,22.0.1-2-g92698f7+9f079a9461,22.0.1-2-ga9b0f51+052faf71bd,22.0.1-2-gac51dbf+052faf71bd,22.0.1-2-gb66926d+6312710a6c,22.0.1-2-gcb770ba+09e3807989,22.0.1-20-g32debb5+b8044ec9de,22.0.1-23-gc2439a9a+fb0756638e,22.0.1-3-g496fd5d+09117f784f,22.0.1-3-g59f966b+1e6ba2c031,22.0.1-3-g849a1b8+f8b568069f,22.0.1-3-gaaec9c0+c5c846a8b1,22.0.1-32-g5ddfab5d3+60ce4897b0,22.0.1-4-g037fbe1+64e601228d,22.0.1-4-g8623105+b8044ec9de,22.0.1-5-g096abc9+d18c45d440,22.0.1-5-g15c806e+57f5c03693,22.0.1-7-gba73697+57f5c03693,master-g6e05de7fdc+c1283a92b8,master-g72cdda8301+729191ecac,w.2021.39
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.