box: introduce formats for standalone tuples
Introduce `box.tuple.format` object, a Lua wrapper around tuple format: these objects own a tuple format, which is almost equivalent to `space:format`, except for check constraints and foreign key constraints being disabled (though they appear to be present for compatibility with `space:format`). Add an option table argument to `box.tuple.new` with 'format' option, allowing to create formatted spaceless tuples. Closes #4693 @TarantoolBot document Title: Formats for standalone tuples and `box_tuple_new_vararg` compat opt A new box.tuple.format library was added, with a tuple format constructor (`new`) and a tuple format validator (`is`). New tuple format objects (userdata) were added, which can be used with the same format clause as for the `space:format` method (except that check constraints and foreign keys are disabled for them): NO_WRAP ```lua f = box.tuple.format.new(box.space._space:format()) f = box.tuple.format.new{{name = 'field1', type = 'string', is_nullable = true, nullable_action = 'none', collation = 'unicode_uk_s2', default = 'UPPER("string")', constraint = {ck = 'box.schema.user.info'}, foreign_key = {fk = {space = '_space', field = 'name'}}}, {name = 'field2', nullable_action = 'ignore', foreign_key = {fk = {space = '_space', field = 1}}}} ``` NO_WRAP Format objects have several introspection methods: `:pairs`, `:ipairs`, `totable`, and also have a `__serialize` metamethod — these methods return the original (i.e., user-provided) format clause. `:pairs` is an alias to `ipairs` (since the format clause is an array by nature), and the `totable` method is an alias to the `__serialize` metamethod, which returns an array of field definitions. Format objects also have a `:tostring` method, which simply returns a "box.tuple.format" literal. The standalone tuple constructor, `box.tuple.new` was extended with an options parameter which currently has one available option, `format` (default value is `nil`, i.e., no format). The format option is either a tuple format object previously created using `box.tuple.format.new` or a format clause. Examples of standalone tuple creation with formats: NO_WRAP ```lua box.tuple.new({1}, {format = {{name = 'field', type = 'number'}}}) box.tuple.new({1}, {format = {{'field', type = 'number'}}}) box.tuple.new({1}, {format = {{'field', 'number'}}}) f = box.tuple.format.new({{name = 'field', type = 'number'}}) box.tuple.new({}, {format = f}) box.tuple.new({1}, {format = f}) box.tuple.new({'str'}, {format = f}) -- error: Tuple field 1 (field) type does not match one required by operation: expected number, got string box.tuple.new({'str'}, {format = f}) ``` NO_WRAP See also the design document https://www.notion.so/tarantool/Schemafull-IPROTO-cc315ad6bdd641dea66ad854992d8cbf?pvs=4#a33e2d7418d249679969e5f21ef2832c A new `box_tuple_new_vararg` compatibility option was introduced: a new page needs to be created for it (https://tarantool.io/compat/box_tuple_new_vararg) This option controls whether `box.tuple.new` should interpret an argument list as an array of tuple fields (i.e., vararg, old behaviour), or as a value plus a tuple format (new default behaviour). The value can be either a scalar, an array or a box tuple. The old behaviour does not allow creating formatted standalone tuples. Old behaviour examples: ```lua box.tuple.new(1) box.tuple.new{1} box.tuple.new(1, 2, 3) box.tuple.new{1, 2, 3} -- This won't create a formatted tuple: the format option will become the -- second tuple field. box.tuple.new({1, 2, 3}, {format = box.tuple.format.new{{'field'}}}) ``` New behaviour examples: ```lua box.tuple.new(1) box.tuple.new(1, {format = box.tuple.format.new{{'field'}}}) box.tuple.new{1} box.tuple.new({1}, {format = box.tuple.format.new{{'field'}}}) box.tuple.new(1, 2, 3) -- error box.tuple.new(1, 2, 3, {format = box.tuple.format.new{{'field'}}}) -- error box.tuple.new{1, 2, 3} box.tuple.new({1, 2, 3}, {format = box.tuple.format.new{{'field'}}}) ``` See also the design document https://www.notion.so/tarantool/Schemafull-IPROTO-cc315ad6bdd641dea66ad854992d8cbf?pvs=4#6f74f0c70005463b8438830edd1a0117.
Showing
- changelogs/unreleased/gh-4693-formats-for-standalone-tuples.md 7 additions, 0 deletions...elogs/unreleased/gh-4693-formats-for-standalone-tuples.md
- src/box/CMakeLists.txt 1 addition, 0 deletionssrc/box/CMakeLists.txt
- src/box/lua/init.c 2 additions, 0 deletionssrc/box/lua/init.c
- src/box/lua/schema.lua 6 additions, 2 deletionssrc/box/lua/schema.lua
- src/box/lua/tuple.c 8 additions, 57 deletionssrc/box/lua/tuple.c
- src/box/lua/tuple.lua 35 additions, 0 deletionssrc/box/lua/tuple.lua
- src/box/lua/tuple_format.c 65 additions, 0 deletionssrc/box/lua/tuple_format.c
- src/box/lua/tuple_format.lua 7 additions, 0 deletionssrc/box/lua/tuple_format.lua
- src/lua/compat.lua 14 additions, 0 deletionssrc/lua/compat.lua
- test/app-luatest/msgpack_test.lua 1 addition, 1 deletiontest/app-luatest/msgpack_test.lua
- test/box-luatest/gh_4693_formats_for_standalone_tuples_test.lua 229 additions, 0 deletions...ox-luatest/gh_4693_formats_for_standalone_tuples_test.lua
- test/box-py/call.result 1 addition, 1 deletiontest/box-py/call.result
- test/box-py/call.test.py 1 addition, 1 deletiontest/box-py/call.test.py
- test/box/call.result 2 additions, 2 deletionstest/box/call.result
- test/box/call.test.lua 2 additions, 2 deletionstest/box/call.test.lua
- test/box/sequence.result 3 additions, 3 deletionstest/box/sequence.result
- test/box/sequence.test.lua 3 additions, 3 deletionstest/box/sequence.test.lua
- test/box/tuple.result 3 additions, 19 deletionstest/box/tuple.result
- test/box/tuple.test.lua 3 additions, 7 deletionstest/box/tuple.test.lua
- test/unit/lua_func_adapter.c 3 additions, 2 deletionstest/unit/lua_func_adapter.c
Loading
Please register or sign in to comment