OPAL (Object Oriented Parallel Accelerator Library) 2022.1
OPAL
PyLine.h
Go to the documentation of this file.
1#ifndef PyOpal_PyLine_h
2
3
4#include "Lines/Sequence.h"
5#include "Lines/Line.h"
8
9namespace PyOpal {
10
11// to get the inheritance from PyElement I need to make PyLine_ templated; but
12// really I only ever want to associate a PyLine_ with a TBeamline (it is a list
13// of elements in a beamline). So I typedef a PyLine class which is the
14// specialisation to PyLine_<TBeamline>.
15template <class C>
16class PyLine_; // forwards declaration of the PyLine_
17typedef PyLine_<TBeamline<FlaggedElmPtr> > PyLine;
18
34template <class C>
36public:
43 boost::python::object getElement(int i);
44
54 void setElement(int i, boost::python::object element);
55
61 void append(boost::python::object element);
62
64 int getLength() const {return line.size();}
65
70 boost::python::class_<PyLine> make_class(const char* className);
71
73
74private:
76 std::vector<boost::python::object> line;
78 int wrangleIndex(int index);
79};
80
81template<>
82int PyLine_<TBeamline<FlaggedElmPtr> >::wrangleIndex(int index) {
83 if (index >= 0) {
84 return index;
85 }
86 index = line.size()+index;
87 return index;
88}
89
90template <>
91void PyLine_<TBeamline<FlaggedElmPtr> >::registerObject() {
92 TBeamline<FlaggedElmPtr>* wrapped = getOpalShared().get();
93 Line* line = new Line();
94 line->setElement(wrapped);
95 Object* objectPtr = dynamic_cast<Object*>(line);
96 if (objectPtr == NULL) {
97 throw OpalException("PyLine_<TBeamline<FlaggedElmPtr> >::register",
98 "Trying to register something that was not a Opal Object");
99 }
100 OpalData::getInstance()->define(objectPtr);
101}
102
103template<>
104boost::python::class_<PyLine> PyLine_<TBeamline<FlaggedElmPtr> >::make_class(const char* className) {
105 boost::python::docstring_options docop(true, true, false); // user_def, py_sig, cpp_sig
106 auto pyclass = boost::python::class_<PyLine>(className);
107 return pyclass;
108}
109
110template<>
111boost::python::object PyLine_<TBeamline<FlaggedElmPtr> >::getElement(int i) {
112 i = wrangleIndex(i);
113 try {
114 return line.at(i);
115 } catch (std::exception& exc) {
116 throw OpalException("PyLine::getElement", "Out of range");
117 }
118}
119
120template<>
121void PyLine_<TBeamline<FlaggedElmPtr> >::setElement(int i, boost::python::object pyelement) {
122 // TBeamline is implemented as a double linked list??
123 typedef TBeamline<FlaggedElmPtr> BL;
124 i = wrangleIndex(i);
125 try {
126 line.at(i) = pyelement;
127 } catch (std::exception& exc) {
128 throw OpalException("PyLine::setElement", "Failed to set element");
129 }
130 boost::python::object pyopalelement = pyelement.attr("get_opal_element")();
132 boost::python::extract<PyOpal::PyOpalObjectNS::PyOpalObject<OpalElement>& >(pyopalelement);
133 int index = 0;
134 for (BL::iterator it = object_m->begin(); it != object_m->end(); ++it) {
135 if (index == i) {
136 // 4 layers of nested inheritance template wrapper darkness
137 std::shared_ptr<OpalElement> opalElementShared = cpyelement.getOpalShared();
138 OpalElement* opalElement = opalElementShared.get();
139 if (!opalElement) {
140 throw OpalException("PyLine::setElement", "Failed to extract element");
141 }
142 opalElement->update();
143 ElementBase* elmbase = opalElement->getElement();
144 if (!elmbase) {
145 throw OpalException("PyLine::setElement", "Failed to cast element");
146 }
147 ElmPtr elmptr(elmbase);
148 FlaggedElmPtr felmptr(elmptr);
149 BL::iterator new_it = object_m->erase(it);
150 object_m->insert(new_it, felmptr);
151 return;
152 }
153 ++index;
154 }
155 throw OpalException("PyLine::getElement", "Out of range");
156}
157
158template <>
159void PyLine_<TBeamline<FlaggedElmPtr> >::append(boost::python::object pyelement) {
160 int i = line.size();
161 // first we extend the size of the python object list by 1 with dummy variables
162 line.push_back(boost::python::object());
163 FlaggedElmPtr felmptr(ElmPtr(NULL));
164 object_m->push_back(felmptr);
165 // then we set the python object using setElement method
166 setElement(i, pyelement);
167}
168
169
170}
171#endif // PyOpal_PyLine_h
172
173/*
174OPAL Beamline insanity...
175
176There exists:
1771. Beamline is a beamline
1782. TBeamline which is another sort of beamline that inherits from Beamline
1793. BeamSequence
1804. Line which has some bindings to the UI
1815. FlaggedBeamline which has a flag to indicate the Beamline is reflected
1826. Sequence which is a list of Sequence elements whatever they are
183I struggle to understand the motivation for so many different classes.
184*/
185
PyLine_< TBeamline< FlaggedElmPtr > > PyLine
Definition: PyLine.h:16
std::string::iterator iterator
Definition: MSLang.h:16
void registerObject()
int getLength() const
Definition: PyLine.h:64
int wrangleIndex(int index)
void append(boost::python::object element)
void setElement(int i, boost::python::object element)
std::vector< boost::python::object > line
Definition: PyLine.h:76
boost::python::class_< PyLine > make_class(const char *className)
boost::python::object getElement(int i)
std::shared_ptr< C > getOpalShared()
Definition: PyOpalObject.h:190
ElementBase * getElement() const
Return the embedded CLASSIC element.
Definition: Element.h:120
void setElement(ElementBase *)
Assign new CLASSIC element.
Definition: Element.h:125
The base class for all OPAL objects.
Definition: Object.h:48
static OpalData * getInstance()
Definition: OpalData.cpp:196
void define(Object *newObject)
Define a new object.
Definition: OpalData.cpp:489
A section of a beam line.
Definition: ElmPtr.h:32
A section of a beam line.
Definition: FlaggedElmPtr.h:36
virtual void update()
Update the embedded CLASSIC element.
Definition: Line.h:32
The base class for all OPAL exceptions.
Definition: OpalException.h:28