Skip to content

Commit 8994fcc

Browse files
committed
Exporting features to Connection class
1 parent 87ba91c commit 8994fcc

9 files changed

+144
-6
lines changed

.gitignore

+2
Original file line numberDiff line numberDiff line change
@@ -111,3 +111,5 @@ deploy_key*
111111
!.ci/deploy_key.enc
112112
/core
113113
cython_debug
114+
115+
temp

CHANGELOG.md

+1
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@
33
## v2.3.0
44
**New features:**
55
* Added support for [interval types](https://www.tarantool.io/en/doc/latest/reference/reference_lua/datetime/interval_object/) [#30](https://github.com/igorcoding/asynctnt/issues/30)
6+
* Added ability to retrieve IProto features [available](https://www.tarantool.io/en/doc/latest/reference/reference_lua/box_iproto/feature/) in Tarantool using `conn.features` property
67

78

89
## v2.2.0

asynctnt/connection.py

+10-2
Original file line numberDiff line numberDiff line change
@@ -444,7 +444,7 @@ async def reconnect(self):
444444
await self.disconnect()
445445
await self.connect()
446446

447-
async def __aenter__(self):
447+
async def __aenter__(self) -> "Connection":
448448
"""
449449
Executed on entering the async with section.
450450
Connects to Tarantool instance.
@@ -606,7 +606,7 @@ def _normalize_api(self):
606606
Api.call = Api.call16
607607
Connection.call = Connection.call16
608608

609-
if self.version < (2, 10): # pragma: nocover
609+
if not self.features.streams: # pragma: nocover
610610

611611
def stream_stub(_):
612612
raise TarantoolError("streams are available only in Tarantool 2.10+")
@@ -627,6 +627,14 @@ def stream(self) -> Stream:
627627
stream._set_db(db)
628628
return stream
629629

630+
@property
631+
def features(self) -> protocol.IProtoFeatures:
632+
"""
633+
Lookup available Tarantool features - https://www.tarantool.io/en/doc/latest/reference/reference_lua/box_iproto/feature/
634+
:return:
635+
"""
636+
return self._protocol.features
637+
630638

631639
async def connect(**kwargs) -> Connection:
632640
"""

asynctnt/iproto/protocol.pxd

+1
Original file line numberDiff line numberDiff line change
@@ -73,6 +73,7 @@ cdef class BaseProtocol(CoreProtocol):
7373
bint _schema_fetch_in_progress
7474
object _refetch_schema_future
7575
Db _db
76+
IProtoFeatures _features
7677
req_execute_func execute
7778

7879
object create_future

asynctnt/iproto/protocol.pyi

+14
Original file line numberDiff line numberDiff line change
@@ -172,6 +172,8 @@ class Protocol:
172172
def schema_id(self) -> int: ...
173173
@property
174174
def schema(self) -> Schema: ...
175+
@property
176+
def features(self) -> IProtoFeatures: ...
175177
def create_db(self, gen_stream_id: bool = False) -> Db: ...
176178
def get_common_db(self) -> Db: ...
177179
def refetch_schema(self) -> asyncio.Future: ...
@@ -203,3 +205,15 @@ class MPInterval:
203205
adjust: Adjust = Adjust.NONE,
204206
): ...
205207
def __eq__(self, other) -> bool: ...
208+
209+
class IProtoFeatures:
210+
streams: bool
211+
transactions: bool
212+
error_extension: bool
213+
watchers: bool
214+
pagination: bool
215+
space_and_index_names: bool
216+
watch_once: bool
217+
dml_tuple_extension: bool
218+
call_ret_tuple_extension: bool
219+
call_arg_tuple_extension: bool

asynctnt/iproto/protocol.pyx

+6-3
Original file line numberDiff line numberDiff line change
@@ -102,6 +102,7 @@ cdef class BaseProtocol(CoreProtocol):
102102
self._schema_fetch_in_progress = False
103103
self._refetch_schema_future = None
104104
self._db = self._create_db(<bint> False)
105+
self._features = IProtoFeatures.__new__(IProtoFeatures)
105106
self.execute = self._execute_bad
106107

107108
try:
@@ -257,9 +258,7 @@ cdef class BaseProtocol(CoreProtocol):
257258
return
258259
e = f.exception()
259260
if not e:
260-
logger.debug('Tarantool[%s:%s] identified successfully',
261-
self.host, self.port)
262-
261+
self._features = (<Response> f.result()).result_
263262
self.post_con_state = POST_CONNECTION_AUTH
264263
self._post_con_state_machine()
265264
else:
@@ -519,6 +518,10 @@ cdef class BaseProtocol(CoreProtocol):
519518
def refetch_schema(self):
520519
return self._refetch_schema()
521520

521+
@property
522+
def features(self) -> IProtoFeatures:
523+
return self._features
524+
522525

523526
class Protocol(BaseProtocol, asyncio.Protocol):
524527
pass

asynctnt/iproto/response.pxd

+14
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,7 @@ cdef class Response:
2727
bint _push_subscribe
2828
BaseRequest request_
2929
object _exception
30+
object result_
3031

3132
readonly object _q
3233
readonly object _push_event
@@ -51,3 +52,16 @@ cdef ssize_t response_parse_header(const char *buf, uint32_t buf_len,
5152
cdef ssize_t response_parse_body(const char *buf, uint32_t buf_len,
5253
Response resp, BaseRequest req,
5354
bint is_chunk) except -1
55+
56+
cdef class IProtoFeatures:
57+
cdef:
58+
readonly bint streams
59+
readonly bint transactions
60+
readonly bint error_extension
61+
readonly bint watchers
62+
readonly bint pagination
63+
readonly bint space_and_index_names
64+
readonly bint watch_once
65+
readonly bint dml_tuple_extension
66+
readonly bint call_ret_tuple_extension
67+
readonly bint call_arg_tuple_extension

asynctnt/iproto/response.pyx

+47-1
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,24 @@ from libc.stdint cimport uint32_t
1010
from asynctnt.log import logger
1111

1212

13+
@cython.final
14+
cdef class IProtoFeatures:
15+
def __repr__(self):
16+
return (f"<IProtoFeatures"
17+
f" streams={self.streams}"
18+
f" transactions={self.transactions}"
19+
f" error_extension={self.error_extension}"
20+
f" watchers={self.watchers}"
21+
f" pagination={self.pagination}"
22+
f" space_and_index_names={self.space_and_index_names}"
23+
f" watch_once={self.watch_once}"
24+
f" dml_tuple_extension={self.dml_tuple_extension}"
25+
f" call_ret_tuple_extension={self.call_ret_tuple_extension}"
26+
f" call_arg_tuple_extension={self.call_arg_tuple_extension}"
27+
f">"
28+
)
29+
30+
1331
@cython.final
1432
@cython.freelist(REQUEST_FREELIST)
1533
cdef class Response:
@@ -26,6 +44,7 @@ cdef class Response:
2644
self.errmsg = None
2745
self.error = None
2846
self._rowcount = 0
47+
self.result_ = None
2948
self.body = None
3049
self.encoding = None
3150
self.metadata = None
@@ -451,6 +470,7 @@ cdef ssize_t response_parse_body(const char *buf, uint32_t buf_len,
451470
const char *s
452471
list data
453472
Field field
473+
IProtoFeatures features
454474

455475
b = <const char *> buf
456476
# mp_fprint(stdio.stdout, b)
@@ -540,7 +560,33 @@ cdef ssize_t response_parse_body(const char *buf, uint32_t buf_len,
540560
logger.debug("IProto version: %s", _decode_obj(&b, resp.encoding))
541561

542562
elif key == tarantool.IPROTO_FEATURES:
543-
logger.debug("IProto features available: %s", _decode_obj(&b, resp.encoding))
563+
features = <IProtoFeatures> IProtoFeatures.__new__(IProtoFeatures)
564+
565+
for item in _decode_obj(&b, resp.encoding):
566+
if item == 0:
567+
features.streams = 1
568+
elif item == 1:
569+
features.transactions = 1
570+
elif item == 2:
571+
features.error_extension = 1
572+
elif item == 3:
573+
features.watchers = 1
574+
elif item == 4:
575+
features.pagination = 1
576+
elif item == 5:
577+
features.space_and_index_names = 1
578+
elif item == 6:
579+
features.watch_once = 1
580+
elif item == 7:
581+
features.dml_tuple_extension = 1
582+
elif item == 8:
583+
features.call_ret_tuple_extension = 1
584+
elif item == 9:
585+
features.call_arg_tuple_extension = 1
586+
else:
587+
logger.debug("unknown iproto feature available: %d", item)
588+
589+
resp.result_ = features
544590

545591
elif key == tarantool.IPROTO_AUTH_TYPE:
546592
logger.debug("IProto auth type: %s", _decode_obj(&b, resp.encoding))

tests/test_connect.py

+49
Original file line numberDiff line numberDiff line change
@@ -802,3 +802,52 @@ async def state_checker():
802802
await conn.call("box.info")
803803
finally:
804804
await conn.disconnect()
805+
806+
async def test__features(self):
807+
async with asynctnt.Connection(host=self.tnt.host, port=self.tnt.port) as conn:
808+
if not check_version(
809+
self,
810+
conn.version,
811+
min=(2, 10),
812+
max=(3, 0),
813+
min_included=True,
814+
max_included=False,
815+
):
816+
return
817+
818+
self.assertIsNotNone(conn.features)
819+
self.assertTrue(conn.features.streams)
820+
self.assertTrue(conn.features.watchers)
821+
self.assertTrue(conn.features.error_extension)
822+
self.assertTrue(conn.features.transactions)
823+
self.assertTrue(conn.features.pagination)
824+
825+
self.assertFalse(conn.features.space_and_index_names)
826+
self.assertFalse(conn.features.watch_once)
827+
self.assertFalse(conn.features.dml_tuple_extension)
828+
self.assertFalse(conn.features.call_ret_tuple_extension)
829+
self.assertFalse(conn.features.call_arg_tuple_extension)
830+
831+
async def test__features_3_0(self):
832+
async with asynctnt.Connection(host=self.tnt.host, port=self.tnt.port) as conn:
833+
if not check_version(
834+
self,
835+
conn.version,
836+
min=(3, 0),
837+
min_included=True,
838+
max_included=False,
839+
):
840+
return
841+
842+
self.assertIsNotNone(conn.features)
843+
self.assertTrue(conn.features.streams)
844+
self.assertTrue(conn.features.watchers)
845+
self.assertTrue(conn.features.error_extension)
846+
self.assertTrue(conn.features.transactions)
847+
self.assertTrue(conn.features.pagination)
848+
849+
self.assertTrue(conn.features.space_and_index_names)
850+
self.assertTrue(conn.features.watch_once)
851+
self.assertTrue(conn.features.dml_tuple_extension)
852+
self.assertTrue(conn.features.call_ret_tuple_extension)
853+
self.assertTrue(conn.features.call_arg_tuple_extension)

0 commit comments

Comments
 (0)