primitive.go 8.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400
  1. package python
  2. // See:
  3. // https://docs.python.org/3/c-api/concrete.html
  4. import (
  5. "fmt"
  6. "unsafe"
  7. )
  8. /*
  9. #define PY_SSIZE_T_CLEAN
  10. #include <Python.h>
  11. */
  12. import "C"
  13. func NewPrimitiveReference(value interface{}) (*Reference, error) {
  14. if value == nil {
  15. return None, nil
  16. }
  17. switch value_ := value.(type) {
  18. case *Reference:
  19. return value_, nil
  20. case bool:
  21. if value_ {
  22. return True, nil
  23. } else {
  24. return False, nil
  25. }
  26. case int64:
  27. return NewLong(value_)
  28. case int32:
  29. return NewLong(int64(value_))
  30. case int8:
  31. return NewLong(int64(value_))
  32. case int:
  33. return NewLong(int64(value_))
  34. case float64:
  35. return NewFloat(value_)
  36. case float32:
  37. return NewFloat(float64(value_))
  38. case string:
  39. return NewUnicode(value_)
  40. case []interface{}:
  41. return NewList(value_...)
  42. case []byte:
  43. return NewBytes(value_)
  44. }
  45. return nil, fmt.Errorf("unsupported primitive: %s", value)
  46. }
  47. //
  48. // None
  49. //
  50. var None = NewReference(C.Py_None)
  51. //
  52. // Bool
  53. //
  54. var BoolType = NewType(&C.PyBool_Type)
  55. var True = NewReference(C.Py_True)
  56. var False = NewReference(C.Py_False)
  57. func (self *Reference) IsBool() bool {
  58. return self.Type().IsSubtype(BoolType)
  59. }
  60. func (self *Reference) ToBool() bool {
  61. switch self.Object {
  62. case C.Py_True:
  63. return true
  64. case C.Py_False:
  65. return false
  66. }
  67. return false
  68. }
  69. //
  70. // Long
  71. //
  72. var LongType = NewType(&C.PyLong_Type)
  73. func NewLong(value int64) (*Reference, error) {
  74. if long := C.PyLong_FromLong(C.long(value)); long != nil {
  75. return NewReference(long), nil
  76. } else {
  77. return nil, GetError()
  78. }
  79. }
  80. func (self *Reference) IsLong() bool {
  81. // More efficient to use the flag
  82. return self.Type().HasFlag(C.Py_TPFLAGS_LONG_SUBCLASS)
  83. }
  84. func (self *Reference) ToInt64() (int64, error) {
  85. if long := C.PyLong_AsLong(self.Object); !HasException() {
  86. return int64(long), nil
  87. } else {
  88. return 0, GetError()
  89. }
  90. }
  91. //
  92. // Float
  93. //
  94. var FloatType = NewType(&C.PyFloat_Type)
  95. func NewFloat(value float64) (*Reference, error) {
  96. if float := C.PyFloat_FromDouble(C.double(value)); float != nil {
  97. return NewReference(float), nil
  98. } else {
  99. return nil, GetError()
  100. }
  101. }
  102. func (self *Reference) IsFloat() bool {
  103. return self.Type().IsSubtype(FloatType)
  104. }
  105. func (self *Reference) ToFloat64() (float64, error) {
  106. if double := C.PyFloat_AsDouble(self.Object); !HasException() {
  107. return float64(double), nil
  108. } else {
  109. return 0.0, GetError()
  110. }
  111. }
  112. //
  113. // Unicode
  114. //
  115. var UnicodeType = NewType(&C.PyUnicode_Type)
  116. func NewUnicode(value string) (*Reference, error) {
  117. value_ := C.CString(value)
  118. defer C.free(unsafe.Pointer(value_))
  119. if unicode := C.PyUnicode_FromString(value_); unicode != nil {
  120. return NewReference(unicode), nil
  121. } else {
  122. return nil, GetError()
  123. }
  124. }
  125. func (self *Reference) IsUnicode() bool {
  126. // More efficient to use the flag
  127. return self.Type().HasFlag(C.Py_TPFLAGS_UNICODE_SUBCLASS)
  128. }
  129. func (self *Reference) ToString() (string, error) {
  130. if utf8stringBytes := C.PyUnicode_AsUTF8String(self.Object); utf8stringBytes != nil {
  131. defer C.Py_DecRef(utf8stringBytes)
  132. if utf8string := C.PyBytes_AsString(utf8stringBytes); utf8string != nil {
  133. return C.GoString(utf8string), nil
  134. } else {
  135. return "", GetError()
  136. }
  137. } else {
  138. return "", GetError()
  139. }
  140. }
  141. //
  142. // Tuple
  143. //
  144. var TupleType = NewType(&C.PyTuple_Type)
  145. func NewTuple(items ...interface{}) (*Reference, error) {
  146. if tuple, err := NewTupleRaw(len(items)); err == nil {
  147. for index, item := range items {
  148. if item_, err := NewPrimitiveReference(item); err == nil {
  149. if err := tuple.SetTupleItem(index, item_); err != nil {
  150. return nil, err
  151. }
  152. } else {
  153. return nil, err
  154. }
  155. }
  156. return tuple, nil
  157. } else {
  158. return nil, GetError()
  159. }
  160. }
  161. func NewTupleRaw(size int) (*Reference, error) {
  162. if tuple := C.PyTuple_New(C.long(size)); tuple != nil {
  163. return NewReference(tuple), nil
  164. } else {
  165. return nil, GetError()
  166. }
  167. }
  168. func (self *Reference) IsTuple() bool {
  169. // More efficient to use the flag
  170. return self.Type().HasFlag(C.Py_TPFLAGS_TUPLE_SUBCLASS)
  171. }
  172. func (self *Reference) SetTupleItem(index int, item *Reference) error {
  173. if C.PyTuple_SetItem(self.Object, C.long(index), item.Object) == 0 {
  174. return nil
  175. } else {
  176. return GetError()
  177. }
  178. }
  179. //
  180. // List
  181. //
  182. var ListType = NewType(&C.PyList_Type)
  183. func NewList(items ...interface{}) (*Reference, error) {
  184. if list, err := NewListRaw(len(items)); err == nil {
  185. for index, item := range items {
  186. if item_, err := NewPrimitiveReference(item); err == nil {
  187. if err := list.SetListItem(index, item_); err != nil {
  188. return nil, err
  189. }
  190. } else {
  191. return nil, err
  192. }
  193. }
  194. return list, nil
  195. } else {
  196. return nil, GetError()
  197. }
  198. }
  199. func NewListRaw(size int) (*Reference, error) {
  200. if list := C.PyList_New(C.long(size)); list != nil {
  201. return NewReference(list), nil
  202. } else {
  203. return nil, GetError()
  204. }
  205. }
  206. func (self *Reference) IsList() bool {
  207. // More efficient to use the flag
  208. return self.Type().HasFlag(C.Py_TPFLAGS_LIST_SUBCLASS)
  209. }
  210. func (self *Reference) SetListItem(index int, item *Reference) error {
  211. if C.PyList_SetItem(self.Object, C.long(index), item.Object) == 0 {
  212. return nil
  213. } else {
  214. return GetError()
  215. }
  216. }
  217. //
  218. // Dict
  219. //
  220. var DictType = NewType(&C.PyDict_Type)
  221. func NewDict() (*Reference, error) {
  222. if dict := C.PyDict_New(); dict != nil {
  223. return NewReference(dict), nil
  224. } else {
  225. return nil, GetError()
  226. }
  227. }
  228. func (self *Reference) IsDict() bool {
  229. // More efficient to use the flag
  230. return self.Type().HasFlag(C.Py_TPFLAGS_DICT_SUBCLASS)
  231. }
  232. func (self *Reference) SetDictItem(key *Reference, value *Reference) error {
  233. if C.PyDict_SetItem(self.Object, key.Object, value.Object) == 0 {
  234. return nil
  235. } else {
  236. return GetError()
  237. }
  238. }
  239. //
  240. // Set (mutable)
  241. //
  242. var SetType = NewType(&C.PySet_Type)
  243. func NewSet(iterable *Reference) (*Reference, error) {
  244. if set := C.PySet_New(iterable.Object); set != nil {
  245. return NewReference(set), nil
  246. } else {
  247. return nil, GetError()
  248. }
  249. }
  250. func (self *Reference) IsSet() bool {
  251. return self.Type().IsSubtype(SetType)
  252. }
  253. //
  254. // Frozen set (immutable)
  255. //
  256. var FrozenSetType = NewType(&C.PyFrozenSet_Type)
  257. func NewFrozenSet(iterable *Reference) (*Reference, error) {
  258. if frozenSet := C.PyFrozenSet_New(iterable.Object); frozenSet != nil {
  259. return NewReference(frozenSet), nil
  260. } else {
  261. return nil, GetError()
  262. }
  263. }
  264. func (self *Reference) IsFrozenSet() bool {
  265. return self.Type().IsSubtype(FrozenSetType)
  266. }
  267. //
  268. // Bytes (immutable)
  269. //
  270. var BytesType = NewType(&C.PyBytes_Type)
  271. func NewBytes(value []byte) (*Reference, error) {
  272. size := len(value)
  273. value_ := C.CBytes(value)
  274. defer C.free(value_) // TODO: check this!
  275. if bytes := C.PyBytes_FromStringAndSize((*C.char)(value_), C.long(size)); bytes != nil {
  276. return NewReference(bytes), nil
  277. } else {
  278. return nil, GetError()
  279. }
  280. }
  281. func (self *Reference) IsBytes() bool {
  282. // More efficient to use the flag
  283. return self.Type().HasFlag(C.Py_TPFLAGS_BYTES_SUBCLASS)
  284. }
  285. func (self *Reference) ToBytes() ([]byte, error) {
  286. if pointer, size, err := self.AccessBytes(); err == nil {
  287. return C.GoBytes(pointer, C.int(size)), nil
  288. } else {
  289. return nil, err
  290. }
  291. }
  292. func (self *Reference) AccessBytes() (unsafe.Pointer, int, error) {
  293. if string_ := C.PyBytes_AsString(self.Object); string_ != nil {
  294. size := C.PyBytes_Size(self.Object)
  295. return unsafe.Pointer(string_), int(size), nil
  296. } else {
  297. return nil, 0, GetError()
  298. }
  299. }
  300. //
  301. // Byte array (mutable)
  302. //
  303. var ByteArrayType = NewType(&C.PyByteArray_Type)
  304. func NewByteArray(value []byte) (*Reference, error) {
  305. size := len(value)
  306. value_ := C.CBytes(value)
  307. defer C.free(value_) // TODO: check this!
  308. if byteArray := C.PyByteArray_FromStringAndSize((*C.char)(value_), C.long(size)); byteArray != nil {
  309. return NewReference(byteArray), nil
  310. } else {
  311. return nil, GetError()
  312. }
  313. }
  314. func (self *Reference) IsByteArray() bool {
  315. return self.Type().IsSubtype(ByteArrayType)
  316. }
  317. func (self *Reference) ByteArrayToBytes() ([]byte, error) {
  318. if pointer, size, err := self.AccessByteArray(); err == nil {
  319. return C.GoBytes(pointer, C.int(size)), nil
  320. } else {
  321. return nil, err
  322. }
  323. }
  324. func (self *Reference) AccessByteArray() (unsafe.Pointer, int, error) {
  325. if string_ := C.PyByteArray_AsString(self.Object); string_ != nil {
  326. size := C.PyByteArray_Size(self.Object)
  327. return unsafe.Pointer(string_), int(size), nil
  328. } else {
  329. return nil, 0, GetError()
  330. }
  331. }