Castable.h
1 /*
2 
3 MIT License
4 
5 Copyright (c) 2017 FMI Open Development / Markus Peura, first.last@fmi.fi
6 
7 Permission is hereby granted, free of charge, to any person obtaining a copy
8 of this software and associated documentation files (the "Software"), to deal
9 in the Software without restriction, including without limitation the rights
10 to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
11 copies of the Software, and to permit persons to whom the Software is
12 furnished to do so, subject to the following conditions:
13 
14 The above copyright notice and this permission notice shall be included in all
15 copies or substantial portions of the Software.
16 
17 THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
18 IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
19 FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
20 AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
21 LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
22 OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
23 SOFTWARE.
24 
25 */
26 /*
27 Part of Rack development has been done in the BALTRAD projects part-financed
28 by the European Union (European Regional Development Fund and European
29 Neighbourhood Partnership Instrument, Baltic Sea Region Programme 2007-2013)
30 */
31 
32 #ifndef DRAIN_CASTABLE2
33 #define DRAIN_CASTABLE2 "drain::Castable"
34 
35 #include <string.h> // Old C
36 #include <cstddef>
37 #include <typeinfo>
38 #include <stdexcept>
39 #include <iostream>
40 #include <vector>
41 #include <list>
42 #include <set>
43 #include <string>
44 
45 #include "Caster.h"
46 #include "Sprinter.h"
47 #include "String.h"
48 #include "UniTuple.h"
49 
50 
51 namespace drain {
52 
53 // Forward declaration
54 // class Variable;
55 
56 // Forward declaration
57 // class Referencer;
58 
60 
76 class Castable {
77 
78 // Optionally, constructors of this base class could be protected.
79 public:
80 
81  inline
82  Castable() : fillArray(false), elementCount(0){
83  setSeparator();
84  caster.unsetType();
85  };
86 
88  inline
89  Castable(const Castable &c) : fillArray(false), elementCount(0) {
90  copyFormat(c); // modifies elementCount = misleading; rename?
91  setPtr(c.caster.ptr, c.caster.getType(), c.elementCount);
92  }
93 
95  template <class F>
96  Castable(F &p) : fillArray(false), elementCount(0) {
97  setSeparator();
98  setPtr(p);
99  }
100 
101  // Obsolete?
105  template <class F>
106  Castable(F *p) : fillArray(false), elementCount(0) {
107  setSeparator();
108  setPtr(p, typeid(F));
109  // std::cerr << "Castable(F *p) type=" << typeid(F).name() << " value=" << *p << " (" << (double)*p << ")" << std::endl;
110  }
111 
112 
113  inline virtual
114  ~Castable(){};
115 
116 
117  virtual inline
118  bool typeIsSet() const {
119  return caster.typeIsSet();
120  };
121 
123 
131  bool empty() const;
132 
134  virtual inline
135  const std::type_info & getType() const {
136  return caster.getType();
137  };
138 
140  inline
141  bool isValid() const {
142  return (caster.typeIsSet()) && (caster.ptr != nullptr);
143  };
144 
145 
147  virtual inline
148  bool isLinking() const { // rename: external?
149  return isValid(); // return false, if nullptr
150  // return true;
151  }
152 
154 
159  virtual inline
160  bool isVariable() const { // needed?
161  return false;
162  }
163 
165  inline
166  bool isCharArrayString() const {
167  return ((caster.getType() == typeid(char)) && (outputSeparator=='\0')); // "close enough" guess, or a definition?
168  };
169 
171  inline
172  bool isStlString() const {
173  return (caster.getType() == typeid(std::string));
174  };
175 
177  inline
178  bool isString() const {
179  return (isCharArrayString() || isStlString());
180  };
181 
182 
184 
189  virtual inline
190  size_t getElementCount() const {
191  return elementCount;
192  }
193 
195 
200  inline
201  size_t getElementSize() const {
202  return caster.getElementSize();
203  };
204 
206 
210  virtual inline
211  size_t getSize() const {
212  return elementCount * caster.getElementSize();
213  }
214 
215 
216 
218  inline
219  Castable & setFill(bool fill=true){
220  fillArray = fill;
221  return *this;
222  }
223 
225  // Has been protected, even private. Why? Essentially setting a value.
226  void clear();
227 
231  inline
232  void reset(){
233  caster.unsetType();
234  updateSize(0);
235  setSeparator(',');
236  }
237 
239 
242  std::string getTypeName() const { // todo: support const string & in Type::call
243  if (isCharArrayString()){
244  static const std::string s("char-array");
245  return s;
246  }
247  else {
248  return Type::call<drain::simpleName>(getType());
249  }
250  }
251 
252 
253 public:
254 
255 
256  template <class T>
257  Castable &operator=(const T &x){
258  //std::cout << "op= " << x << " -> " << Type::getTypeChar(getType()) << ',' << getElementCount() << std::endl;
259  assign(x);
260  return *this;
261  }
262 
266  inline
267  Castable & operator=(const char *c){
268  assignString(c);
269  return *this;
270  }
271 
272 
273  template <class T>
274  Castable & operator=(const T *x){
275  throw std::runtime_error("Castable & operator=(const T *x): unsupported");
276  return *this;
277  }
278 
280 
283  inline
285  if (&c != this){
286  assignCastable(c);
287  }
288  return *this;
289  }
290 
291 
292  /*
293  inline
294  Castable &operator=(const Variable &c){
295  return assignCastable(c);
296  }
297 
298  inline
299  Castable &operator=(const Referencer &c){
300  return assignCastable(c);
301  }
302  */
303 
304  // Special handler for string assignment
305  inline
306  void assign(const std::string &c){
307  assignString(c);
308  }
309 
310 
311  inline
312  void assign(const char *c){
313  assignString(c);
314  }
315 
316  // "NEW"
317  inline
318  void assign(const Castable &c){
319  assignCastable(c);
320  }
321 
322  template <class T>
323  inline
324  void assign(const T *p){
325  if (p == nullptr)
326  reset();
327  else
328  throw std::runtime_error("Castable & assign(const T *p): unsupported");
329  }
330 
332 
335  template <class T>
336  void assign(const T &x){
337  // std::cout << "assign " << x << " -> " << Type::getTypeChar(getType()) << ',' << getElementCount() << std::endl;
338  suggestType(typeid(T));
339  if (isString()){
340  //std::cout << __FUNCTION__ << ':' << x << " -> " << Type::getTypeChar(getType()) << ',' << getElementCount() << '\n';
341  assignToString(x);
342  }
343  else if (typeIsSet()){
344  if (fillArray){
345  for (size_t i = 0; i<getElementCount(); ++i){
346  caster.put(getPtr(i), x);
347  }
348  }
349  else {
350  requestSize(1);
351  caster.put(x);
352  }
353  }
354  else {
355  throw std::runtime_error(std::string(__FUNCTION__) + ": type is unset");
356  }
357 
358 
359  }
360 
361 
362 
364  template<typename T>
365  // Castable & operator=(std::initializer_list<T> l){
366  void assign(std::initializer_list<T> l){
367  assignContainer(l);
368  //return *this;
369  }
370 
372  template <class T>
373  inline
374  // Castable &operator=(const std::list<T> & l){
375  void assign(const std::list<T> & l){
376  assignContainer(l);
377  // return *this;
378  }
379 
381  template <class T>
382  inline
383  // Castable &operator=(const std::vector<T> & v){
384  void assign(const std::vector<T> & v){
385  assignContainer(v);
386  // return *this;
387  }
388 
390  template <class T>
391  inline
392  // Castable &operator=(const std::set<T> & s){
393  void assign(const std::set<T> & s){
394  assignContainer(s);
395  //return *this;
396  }
397 
398 
400  Castable & assignCastable(const Castable &c);
401 
402 
403 
404  template <class T>
405  inline
406  Castable &operator<<(const T &x){
407  append(x);
408  return *this;
409  }
410 
411 
413  /*
414  * Note: if both source and target are strings, does not split it like with Variable::operator=(const char *).
415  */
416  template <class T>
417  inline
418  //Castable &operator<<(const T &x){
419  void append(const T &x){
420  suggestType(typeid(T));
421  if (isString()){
422  appendToString(x);
423  }
424  else {
425  appendToElementArray(x);
426  }
427  //return *this;
428  }
429 
431  inline
432  // Castable &operator<<
433  void append(const char *s){
434  //return *this << (std::string(s));
435  append(std::string(s));
436  }
437 
439  template <class T>
440  inline
441  void append(const std::list<T> & l){
442  assignContainer(l, true);
443  //return *this;
444  }
445 
447  template <class T>
448  inline
449  // Castable &operator<<
450  void append(const std::vector<T> & v){
451  assignContainer(v, true);
453  }
454 
456  template <class T>
457  inline // Castable &operator<<
458  void append(const std::set<T> & s){
459  assignContainer(s, true);
460  // return *this;
461  }
462 
463  // NEW 2025
464  inline // protected
465  void append(){
466  }
467 
468  template <class T, class ...TT>
469  void append(const T &arg, const TT& ...args){
470  //Castable::operator<<(x);
471  append(arg);
472  append(args...);
473  //return *this;
474  }
475 
476 
477  template <class T>
478  inline
479  T get(size_t i) const {
480  if (i >= elementCount)
481  throw std::runtime_error("Castable::get() index overflow");
482  return caster.get<T>(getPtr(i));
483  }
484 
485  /*
486  template <class T>
487  inline
488  void setElem(size_t i) const {
489  if (i >= elementCount)
490  throw std::runtime_error("Castable::get() index overflow");
491  return caster.get<T>(getPtr(i));
492  }
493  */
494 
495 
497 
500  inline
501  operator std::string() const {
502  return toStr();
503  }
504 
507  // Thanks to Mikael Kilpeläinen.
520  template <class T>
521  operator T() const {
522  if (isCharArrayString()){
523  T x;
524  std::stringstream sstr;
525  sstr << getCharArray();
526  sstr >> x;
527  return x;
528  }
529  else
530  return caster.get<T>(); //caster.get<T>(ptr);
531  }
532 
533  template <class T, size_t N>
534  operator UniTuple<T,N>() const {
535  UniTuple<T,N> result;
536  result.assignSequence(*this, true);
537  }
538 
539 
540  inline
541  bool operator==(const Castable &c) const {
542  // std::cerr << __FILE__ << __LINE__ << __FUNCTION__ << std::endl;
543  //throw std::runtime_error("Castable: operator== not implemented.");
544 
545  if (!this->typeIsSet()){
546  return !c.typeIsSet(); // is "same"... (?)
547  }
548 
549  if (!c.typeIsSet()){
550  return false; // could return false for both?
551  }
552 
553  if (this->isString() || c.isString()){ // NOTE: precision loss in rounding numbers
554  return (this->toStr() == c.toStr());
555  }
556 
557  if (this->getElementCount() != c.getElementCount()){
558  return false;
559  }
560 
561 
562  for (size_t i = 0; i<this->getElementCount(); ++i){
563  if (! caster.compare(getPtr(i), c.caster, c.getPtr(i))){
564  return false;
565  }
566  }
567 
568  return true;
569  }
570 
572  inline
573  bool operator==(const char * s) const {
574  if (isCharArrayString()) // 2021
575  return (strcmp(getCharArray(), s) == 0);
576  else {
577  // Note: caster.get<std::string>() returns only the first element!
578  // std::cerr << __FILE__ << __LINE__ << __FUNCTION__ << toStr() << " vs " << caster.get<std::string>() << std::endl;
579  return (toStr() == s);
580  }
581  //return (caster.get<std::string>() == s);
582  }
583 
585 
590  inline
591  bool operator==(const nullptr_t ptr) const {
592  return getType() == typeid(void);
593  }
594 
595  /*
596  inline // 2024
597  bool operator==(const std::string & s) const {
598  return (caster.get<std::string>() == s);
599  }
600  */
601 
603  template <class T>
604  bool operator==(const T &x) const {
605  if (isCharArrayString()){ // 2024/04
606  std::stringstream sstr(getCharArray());
607  T c;
608  sstr >> c;
609  return (c == x);
610  }
611  else {
612  return (caster.get<T>() == x);
613  }
614  }
615 
616  template <class T, size_t N>
617  bool operator==(const UniTuple<T,N> & x) const {
618  if (isString()){ // 2024/04
619  // std::cerr << __FILE__ << ':' << __FUNCTION__ << " '" << this->toStr() << "' == '" << x.toStr(',') << "'" << std::endl;
620  return (this->toStr() == x.toStr(',')); // consider separator, but this->outputSeparator is null for char array str...
621  }
622  else if (elementCount == N){
623  for (size_t i=0; i<N; ++i){
624  if (x[i] != this->get<T>(i)){
625  return false;
626  }
627  }
628  return true;
629  }
630  else {
631  return false;
632  }
633 
634  }
635 
636 
637 
639 
642  template <class T>
643  bool operator!=(const T &x) const {
644  return ! this->operator==(x);
645  }
646 
647  bool operator!=(const char *x) const {
648  return ! this->operator==(x);
649  }
650 
651 
653  // strings?
654  template <class T>
655  bool operator<(const T &x) const {
656  return (caster.get<T>() < x);
657  }
658 
660  template <class T>
661  bool operator>(const T &x) const {
662  return (caster.get<T>() > x);
663  }
664 
666  // strings?
667  template <class T>
668  bool operator<=(const T &x) const {
669  return (caster.get<T>() <= x);
670  }
671 
673  template <class T>
674  bool operator>=(const T &x) const {
675  return (caster.get<T>() >= x);
676  }
677 
679 
682  inline
683  Castable & setSeparator(char c = ','){
686  return *this;
687  }
688 
690 
694  inline
695  void setInputSeparator(char c = ','){
696  inputSeparator = c;
697  }
698 
700 
704  inline
705  void setOutputSeparator(char c = ','){
706  outputSeparator = c;
707  }
708 
710 
714  inline
716  return inputSeparator;
717  }
718 
720 
724  inline
726  return outputSeparator;
727  }
728 
729 
730  std::ostream & toStream(std::ostream & ostr = std::cout, char separator='\0') const;
731 
732  std::istream & fromStream(std::istream & istr);
733 
734 
735  std::string toStr() const;
736 
738  void typeInfo(std::ostream & ostr) const;
739 
741  // Writes a bit longer type indentifier and size in bytes, for example "unsigned int(8) for char and "signed int(16) for signed short int."
742  virtual
743  void info(std::ostream & ostr = std::cout) const;
744 
745  // void debug(std::ostream & ostr = std::cout) const;
746 
747 
748 
750 
755  template <class T>
756  void toSequence(T & container, char separator = 0) const { // FIX raise
757 
758  container.clear();
759 
760  if (isString()){
761  if (!separator)
762  separator = this->outputSeparator;
763  StringTools::split(toStr(), container, separator," \t\n");
764  }
765  else {
766  for (size_t i = 0; i < getElementCount(); ++i) {
768  container.insert(container.end(), caster.get<typename T::value_type>(getPtr(i)) );
769  }
770  }
771 
772 
773  }
774 
775 
776 
777  template <class T>
778  void toMap(T & map, char separator = 0, char equalSign = '=') const {
779 
780  map.clear();
781 
782  if (isString()){
783  // This part could be in StringTools?
784  typedef std::list<std::string> entryList;
785  entryList entries;
786  toSequence(entries, separator);
787 
788  for (entryList::const_iterator it = entries.begin(); it!=entries.end(); ++it){
789  //typename T::value_type entry;
790  typename T::key_type key;
791  typename T::mapped_type data;
792  drain::StringTools::split2(*it, key, data, std::string(1, equalSign)); // TRIM?
793  map[key] = data;
794  }
795  }
796  else {
797  for (size_t i = 0; i < getElementCount(); ++i){
798  //typename T::value_type entry;
799  typename T::key_type key;
800  typename T::mapped_type data;
801  std::stringstream sstr;
802  sstr << i;
803  sstr >> key; // entry.first;
804  std::stringstream sstr2;
805  //sstr2 << caster.get<typename T::value_type>(getPtr(i)); // weirds
806  sstr2 << caster.get<typename T::mapped_type>(getPtr(i)); // weirds
807  sstr2 >> data;
808  map[key] = data;
809  }
810  }
811 
812 
813  }
814 
815 
817 
820  inline
821  char * getPtr(size_t i = 0) {
822  return &((char *)caster.ptr)[i*caster.getElementSize()];
823  }
824 
825  inline
826  const char * getPtr(size_t i = 0) const {
827  return &((const char *)caster.ptr)[i*caster.getElementSize()];
828  }
829 
830 
832  void copyFormat(const Castable & c){
833  this->elementCount = c.elementCount;
834  this->inputSeparator = c.inputSeparator;
836  this->fillArray = c.fillArray;
837  }
838 
839 
840  virtual inline
841  bool requestType(const std::type_info & t){
842  return (getType() == t);
843  }
844 
846 
849  virtual inline
850  bool suggestType(const std::type_info & t){
851  return (getType() == t);
852  }
853 
855  bool fillArray = false;
856 
857 protected:
858 
859  virtual
860  void updateSize(size_t elems){
861  elementCount = elems;
862  }
863 
864 protected:
865 
866  Caster caster;
867 
868 
869 
871  template <class F>
872  inline // FINAL
873  void setType(){
875  setType(typeid(F));
876  }
877 
879  virtual inline // virtual? Variable may need
880  void setType(const std::type_info &t){
881  setSeparator(','); // ?
882  caster.setType(t);
883  //elementCount = 1; ?
884  }
885 
887 
890  virtual inline
891  bool requestSize(size_t elementCount){
892  return (getElementCount() == elementCount);
893  }
894 
895 
897  template <class F>
898  inline
899  void setPtr(void *p){
900  caster.link(p);
901  elementCount = 1;
902  }
903 
904 
906  template <class F>
907  void setPtr(F &p){
908  if ((!std::is_arithmetic<F>::value) && (typeid(F)!=typeid(std::string))){
909  throw std::runtime_error(std::string(__FUNCTION__) + ": unsupported type: " + typeid(F).name());
910  }
911  caster.link(p); // could use setPtr(void *p, const std::type_info &t) ?
912  elementCount = 1;
913  }
914 
916  /*
917  * Function of this kind must be available for general (8bit) memory allocators.
918  */
919  inline
920  void setPtr(void *p, const std::type_info &t, size_t count=1){
921  //if (t == typeid(void))
922  // throw std::runtime_error(std::string(__FUNCTION__) + ": explicit void type given");
923  caster.link(p, t);
924  elementCount = count;
925  }
926 
928 
931  /*
932  template <class F>
933  // void setPtr(std::pair<F,F> &p){
934  setPtr(&p, typeid(F), 2);
935  //caster.link((F*)&v);
936  fillArray = true; // if a scalar is assigned, assign to both elements
937  //elementCount = 2;
938  }
939  */
940 
942 
945  template <class F, size_t N>
946  void setPtr(UniTuple<F,N> & tuple){
947  setPtr(tuple.begin(), typeid(F), N);
948  fillArray = true; // (?) if a scalar is assigned, assign to all elements
949  }
950 
951 
952 
953 
955  void relink(Castable & c);
956 
957 
958  /*
959  inline
960  void setPtr(const Castable &c){
961  //caster.link0((const Caster &)c);
962 
963  caster.ptr = c.caster.ptr;
964  caster.setType(c.getType());
965 
966  elementCount = c.elementCount; // TODO
967  outputSeparator = c.outputSeparator; // ?
968  }
969 
971  inline
972  void setPtr(Castable &c){
973  caster.link0((Caster &)c);
974  // caster.ptr = c.caster.ptr;
975  // caster.setType(c.getType());
976  elementCount = c.elementCount; // TODO
977  outputSeparator = c.outputSeparator; // ?
978  }
979  */
980 
982  // IMPORTANT!
983  inline
984  void castElement(size_t i, const Caster &c, void *p) const {
985  c.translate(caster, getPtr(i), p);
986  }
987 
988 
989  // Destination type (current type) specific assign operations
990 
991 
992 
994 
999  template <class T>
1000  void appendToString(const T & x){
1001 
1002  std::stringstream sstr;
1003 
1004  // drain::Sprinter::toStream(sstr, *this, drain::Sprinter::plainLayout);
1005  toStream(sstr);
1006 
1007  if (inputSeparator && (getElementCount()>1)){
1008  sstr << inputSeparator;
1009  }
1010  sstr << x;
1011 
1012  std::string s = sstr.str();
1013 
1014  suggestType(typeid(std::string));
1015 
1016  if (isStlString()){
1017  caster.put(s);
1018  //caster.put(ptr, s);
1019  }
1020  else if (isCharArrayString()){
1021  assignToCharArray(s);
1022  }
1023  else {
1024  throw std::runtime_error(std::string(__FUNCTION__) + ": type is not a string: " + typeid(T).name());
1025  }
1026  }
1027 
1028 
1029  template <class T>
1030  void assignToString(const T & x){
1031 
1032  /*
1033  if (x == nullptr){
1034  //suggestType(typeid(void));
1035  suggestType(typeid(void));
1036  //experimental
1037  return;
1038  }
1039  */
1040 
1041  suggestType(typeid(std::string));
1042 
1043  if (isStlString()){
1044  caster.put(x);
1045  //caster.put(ptr, x);
1046  }
1047  else if (isCharArrayString()){
1048  requestSize(0); // works only for Variable, ok
1049  clear(); // works always (but if elementCount)
1050  appendToString(x);
1051  }
1052  else {
1053  throw std::runtime_error(std::string(__FUNCTION__) + ": type is std::string");
1054  }
1055  }
1056 
1057  void assignToCharArray(const std::string & s);
1058 
1059  inline
1060  void assignToCharArray(const char *s){
1061  assignToCharArray(std::string(s));
1062  }
1063 
1065 
1068  template <class T>
1069  void assignToCharArray(const T & s){
1070  std::stringstream sstr;
1071  sstr << s;
1072  assignToCharArray(sstr.str());
1073  }
1074 
1075 
1076  inline
1077  void appendToElementArray(const char *s){
1078  appendToElementArray(std::string(s));
1079  }
1080 
1081  template <class T>
1082  void appendToElementArray(const T & s){
1083 
1084  suggestType(typeid(T)); // check return code?
1085  if (!typeIsSet()){
1086  throw std::runtime_error(std::string(__FUNCTION__) + ": type is unset");
1087  return;
1088  }
1089 
1090  // Check if now type==string...
1091  if (isString()){
1092  throw std::runtime_error(std::string(__FUNCTION__) + ": type is std::string (not implemented yet)");
1093  //return;
1094  }
1095 
1096  // Extend array.
1098  if (empty()){
1099  throw std::runtime_error(std::string(__FUNCTION__) + ": still empty after resize request(+1)");
1100  return;
1101  }
1102 
1103  // TODO: what if non-empty but fixed?
1104 
1105  caster.put(getPtr(getElementCount()-1), s);
1106 
1107  }
1108 
1109 
1110 
1111 public:
1112  /* public NEW 2021 (experimental)
1113  *
1114  * useful in "stream" definitions:
1115  * drain::Referencer(range.tuple()).setSeparator(':').setFill().assignString(ftor);
1116  *
1117  */
1118 
1119 
1121 
1122  void assignString(const std::string &s);
1123 
1124  // NEW 2024
1125  template <class T, size_t N>
1126  inline
1127  void assign(const UniTuple<T,N> & tuple){
1128  assignContainer(tuple);
1129  }
1130 
1132  template <class T>
1133  void assignContainer(const T & v, bool append=false) {
1134 
1135  if (!append){
1136  if (isStlString())
1137  clear();
1138  else
1139  requestSize(0);
1140  }
1141 
1142  suggestType(typeid(typename T::value_type));
1143  // Note: sequence of strings, will request string type...
1144 
1145  if (isString()){
1146  std::stringstream sstr;
1147  if (append){
1149  //toStream(sstr); // outputsepp?
1150  if (inputSeparator)
1151  sstr << inputSeparator;
1152  }
1153  StringTools::join(v, sstr, inputSeparator); // implement!
1154  assignToString(sstr.str());
1155  }
1156  else if (caster.typeIsSet()){
1157 
1158  if (append){
1159  for (typename T::const_iterator it = v.begin(); it != v.end(); ++it){
1160  appendToElementArray(*it);
1161  }
1162  }
1163  else {
1164  requestSize(v.size());
1165  typename T::const_iterator it = v.begin();
1166  typename T::const_iterator itLast = it;
1167  for (size_t i = 0; i<getElementCount(); ++i){
1168  if (it == v.end()){
1169  if (fillArray)
1170  it = itLast; //v.begin(); WHAT if empty?
1171  else
1172  return;
1173  }
1174  else {
1175  itLast = it;
1176  }
1177  caster.put(getPtr(i), *it);
1178  ++it;
1179  }
1180  }
1181  }
1182  else {
1183  throw std::runtime_error(std::string(__FILE__) + ": type unset, cannot assign");
1184  }
1185 
1186  }
1187 
1188 
1190 
1193  /* OLD
1194  inline
1195  const char * getCharArray() const {
1196 
1197  if (!isCharArrayString())
1198  throw std::runtime_error("getCharArray: type not char array...");
1199 
1200  if (empty()){
1201  static const char * empty = "";
1202  return empty;
1203  // throw std::runtime_error("getCharArray: empty array, no even null char");
1204  }
1205 
1206  if (*getPtr(getElementCount()-1) != '\0')
1207  throw std::runtime_error("getCharArray: no terminating null char");
1208 
1209  return getPtr();
1210 
1211  }
1212  */
1213 
1215 
1221  // NEW
1222 
1223 public:
1224 
1225  const char * getCharArray() const;
1226 
1227 protected:
1228 
1230  // void *ptr;
1231 
1234 
1237 
1240 
1241 
1242 };
1243 
1244 
1245 
1247 template <>
1248 std::ostream & Sprinter::toStream(std::ostream & ostr, const drain::Castable & x, const SprinterLayout & layout);
1249 
1250 
1251 
1252 inline
1253 std::ostream & operator<<(std::ostream &ostr, const Castable &c){
1254  return c.toStream(ostr);
1255  //return Sprinter::toStream(ostr, c, Sprinter::plainLayout);
1256 }
1257 
1258 /* UNDER CONSTR..
1259 inline
1260 std::istream & operator>>(std::istream & istr, Castable &c){
1261  return c.fromStream(istr);
1262 }
1263 */
1264 
1266 template <class T>
1267 inline
1268 T & operator+=(T & x, const Castable &c){
1269  x += c.get<T>(0);
1270  return x;
1271 }
1272 
1274 template <class T>
1275 inline
1276 T & operator-=(T & x, const Castable &c){
1277  x -= c.get<T>(0);
1278  return x;
1279 }
1280 
1282 template <class T>
1283 inline
1284 T & operator*=(T & x, const Castable &c){
1285  x *= c.get<T>(0);
1286  return x;
1287 }
1288 
1290 template <class T>
1291 inline
1292 T & operator/=(T & x, const Castable &c){
1293  x /= c.get<T>(0);
1294  return x;
1295 }
1296 
1297 
1298 
1299 template <>
1300 const std::string TypeName<Castable>::name;
1301 
1302 /*
1303 template <class T, size_t N=2>
1304 template <class T2>
1305 UniTuple<T,N> & UniTuple<T,N>::set(const Castable & t){
1306  // assignSequence(t, true); // by default LENIENT, or how should it be?
1307  return *this;
1308 }
1309 */
1310 
1311 
1312 } // namespace drain
1313 
1314 
1315 #endif
1316 
1317 // Drain
Definition: Castable.h:76
void setPtr(UniTuple< F, N > &tuple)
Stores the pair as an array of two elements.
Definition: Castable.h:946
void setPtr(void *p, const std::type_info &t, size_t count=1)
Sets the data pointer and its explicit type.
Definition: Castable.h:920
void append(const std::set< T > &s)
Appends elements of a set.
Definition: Castable.h:458
Castable & setSeparator(char c=',')
The character used between array elements in output stream.
Definition: Castable.h:683
bool isCharArrayString() const
Returns true, if type is C char array and outputSepator is the null char.
Definition: Castable.h:166
virtual bool isLinking() const
Return true, if the referenced variable is external. Inherited classes - like Variable - may have int...
Definition: Castable.h:148
virtual const std::type_info & getType() const
Return the storage type (base type) of the referenced variable.
Definition: Castable.h:135
void append(const std::list< T > &l)
Appends elements of std::list.
Definition: Castable.h:441
Castable & operator=(const Castable &c)
Copies the value referred to by Castable. Notice that if type is unset (void), no operation is perfor...
Definition: Castable.h:284
bool operator==(const nullptr_t ptr) const
Experimental. Notice that if type is set, false is returned also for an empty array.
Definition: Castable.h:591
void copyFormat(const Castable &c)
Copies array layout and formatting: separators, element count, fillArray flag.
Definition: Castable.h:832
void assignString(const std::string &s)
Input type specific assign operations.
Definition: Castable.cpp:365
bool operator<(const T &x) const
Compares a value to inner data.
Definition: Castable.h:655
Castable(const Castable &c)
Copy constructor: copies the layout and the pointer to the target.
Definition: Castable.h:89
bool operator<=(const T &x) const
Compares a value to inner data.
Definition: Castable.h:668
Castable & setFill(bool fill=true)
Set or unset filling (padding) an array if input set is smaller.
Definition: Castable.h:219
void typeInfo(std::ostream &ostr) const
Writes a string of type indentifier char and size in bytes, for example [C@8] for unsigned char and [...
Definition: Castable.cpp:220
virtual size_t getElementCount() const
Returns the length of the array, the number of elements in this entry.
Definition: Castable.h:190
virtual void info(std::ostream &ostr=std::cout) const
Print value, type and element count.
Definition: Castable.cpp:233
virtual bool suggestType(const std::type_info &t)
Request to change in type. For Castable, simply returns true if the current type was requested.
Definition: Castable.h:850
void assign(const std::vector< T > &v)
Copies elements of a vector.
Definition: Castable.h:384
void append(const char *s)
Appends the string or appends the array by one element.
Definition: Castable.h:433
bool isValid() const
Return true, if the pointer is set.
Definition: Castable.h:141
char getOutputSeparator()
Definition: Castable.h:725
bool empty() const
Returns true, if string is empty or array size is zero. (In future, semantics may change: 0 for scala...
Definition: Castable.cpp:36
void assign(const std::list< T > &l)
Copies elements of a list.
Definition: Castable.h:375
virtual bool isVariable() const
Returns true, if the class contains an own data array.
Definition: Castable.h:160
void setPtr(F &p)
Stores the pointer and its storage type F.
Definition: Castable.h:907
size_t elementCount
Pointer to the data variable.
Definition: Castable.h:1233
bool isStlString() const
Returns true, if type is std::string .
Definition: Castable.h:172
std::string getTypeName() const
Returns the name of the current storage type.
Definition: Castable.h:242
void assignToCharArray(const T &s)
Input anything to char array string.
Definition: Castable.h:1069
void assignContainer(const T &v, bool append=false)
Assigns a STL Sequence, element by element.
Definition: Castable.h:1133
char getInputSeparator()
Definition: Castable.h:715
void appendToString(const T &x)
Append to std::string or char array.
Definition: Castable.h:1000
void assign(const T &x)
Copies an arbitrary base type or std::string value.
Definition: Castable.h:336
Castable & assignCastable(const Castable &c)
Copy data from Castable. Perhaps copy size and type, too.
Definition: Castable.cpp:256
size_t getElementSize() const
Return the size of the current storage type, in bytes.
Definition: Castable.h:201
void assign(const std::set< T > &s)
Copies elements of a set.
Definition: Castable.h:393
bool operator==(const char *s) const
Compare as a character string.
Definition: Castable.h:573
char * getPtr(size_t i=0)
Returns pointer to the array of chars without validity check.
Definition: Castable.h:821
bool fillArray
If array, assigning a scalar will fill up the current array.
Definition: Castable.h:855
void toSequence(T &container, char separator=0) const
Converts data to a STL Sequence, for example std::set, std::list or std::vector .
Definition: Castable.h:756
bool operator>(const T &x) const
Compares a value with inner data.
Definition: Castable.h:661
char outputSeparator
Element separator usein in writing and reading character strings (streams).
Definition: Castable.h:1239
bool isString() const
Returns true, if type is C char array or std::string .
Definition: Castable.h:178
void setPtr(void *p)
Stores the pointer and its storage type F. Assumes elementCount=1.
Definition: Castable.h:899
const char * getCharArray() const
Returns pointer to the array of chars, checks validity first.
Definition: Castable.cpp:93
void append(const std::vector< T > &v)
Appends elements of a vector.
Definition: Castable.h:450
virtual void setType(const std::type_info &t)
Sets the storage type. If a target value is available, use setPtr() directly.
Definition: Castable.h:880
char inputSeparator
Element separator usein in reading a char sequence to an (numeric) array.
Definition: Castable.h:1236
Castable & operator=(const char *c)
Definition: Castable.h:267
void clear()
Clears strings, or for scalars and arrays, sets all the values to zero. Does not change type.
Definition: Castable.cpp:81
bool operator==(const T &x) const
Compares a value to internal data.
Definition: Castable.h:604
void setType()
Sets the storage type. If a target value is available, use setPtr() directly.
Definition: Castable.h:873
void append(const T &x)
Appends the string or appends the array by one element.
Definition: Castable.h:419
void reset()
Definition: Castable.h:232
void relink(Castable &c)
Copies the link and element count.
Definition: Castable.cpp:69
void castElement(size_t i, const Caster &c, void *p) const
Let Caster c convert my element #i to target *p.
Definition: Castable.h:984
void setOutputSeparator(char c=',')
Definition: Castable.h:705
Castable(F &p)
Constructor for an object pointing to a variable.
Definition: Castable.h:96
bool operator>=(const T &x) const
Compares a value with inner data.
Definition: Castable.h:674
virtual size_t getSize() const
Returns the size in bytes of the target: (elementCount * elementSize)
Definition: Castable.h:211
bool operator!=(const T &x) const
Compares a value to inner data.
Definition: Castable.h:643
virtual bool requestSize(size_t elementCount)
Request to change the array size. For Castable (and Reference) does nothing and returns false.
Definition: Castable.h:891
void setInputSeparator(char c=',')
Definition: Castable.h:695
void assign(std::initializer_list< T > l)
Copies elements of a list.
Definition: Castable.h:366
Definition: Caster.h:67
const std::type_info & getType() const
Returns type_info of the current type.
Definition: Caster.h:144
T get(const void *p) const
Default implementation throws an error. See specialized implementation below.
Definition: Caster.h:194
void unsetType()
calls void setType<void>().
Definition: Caster.cpp:134
size_t getElementSize() const
Returns the size of the base type (size of an element, not of element array).
Definition: Caster.h:151
void setType(const std::type_info &t)
Calls setType<T>() for which typeid(T) = t.
Definition: Caster.h:119
void translate(const Caster &c, const void *ptrC, void *ptr) const
Convert from str pointer and Caster.
Definition: Caster.h:208
void * ptr
Future member: enables setting Caster type.
Definition: Caster.h:247
void put(void *p, const T &x) const
Default conversion (for unconventional types). Uses std::stringstream for conversion.
Definition: Caster.h:157
static const SprinterLayout plainLayout
Display plain values, concatenating them with comma (except for strings).
Definition: Sprinter.h:209
static std::ostream & toStream(std::ostream &ostr, const std::initializer_list< T > &x, const SprinterLayout &layout=defaultLayout)
New (experimental)
Definition: Sprinter.h:420
static bool split2(const std::string &s, T1 &first, T2 &second, const C &separators, const std::string &trimChars=" \t\n")
Splits and trims a given std::string to a std Sequence.
static std::ostream & join(const T &container, std::ostream &ostr, char separator=0)
Writes a STL Container (list, vector, set) to a stream, using an optional separator char (e....
Definition: String.h:211
static void split(const std::string &s, T &sequence, const C &separators, const std::string &trimChars=" \t\n")
Splits and trims a given std::string to a std Sequence.
Definition: String.h:318
tuplebase_t & assignSequence(T &sequence, bool LENIENT=false)
Proposed for tuples only; derived classes should not shadow this.
Definition: TupleBase.h:287
Tuple of N elements of type T.
Definition: UniTuple.h:65
Definition: DataSelector.cpp:1277
T & operator*=(T &x, const Castable &c)
Arithmetics: addition.
Definition: Castable.h:1284
T & operator+=(T &x, const Castable &c)
Arithmetics: addition.
Definition: Castable.h:1268
T & operator/=(T &x, const Castable &c)
Arithmetics: subtraction.
Definition: Castable.h:1292
T & operator-=(T &x, const Castable &c)
Arithmetics: subtraction.
Definition: Castable.h:1276
Definition: Sprinter.h:137
static const std::string name
Default implementation: name returned by std::type_info::name()
Definition: Type.h:558