stetre / luamoulds

A C-like typedef for Lua

Geek Repo:Geek Repo

Github PK Tool:Github PK Tool

LuaMoulds: A C-like typedef for Lua

*** This project is discontinued, superseded by MoonTypes. ***

LuaMoulds is a Lua module that provides means to define C-like structured types in Lua.

It has been designed for the definition of formats of messages (in particular, of signals exchanged by LunaSDL agents), but it is a standalone module that can be used as a general purpose type system as well.

LuaMoulds requires Lua (>=5.2), and LPeg. It is is written in plain Lua, so no compiling is needed.

Authored by: Stefano Trettel

Lua logo

License

MIT/X11 license (same as Lua). See LICENSE.

Intro

The LuaMoulds module is loaded with require( ), as shown in the example below. It returns a function that, each time it is called, creates a new database of type definitions (which we will call a 'mould set').

moulds = require("moulds")

ms = moulds() -- creates a new mould set (i.e. a database of types)

A mould set, when created, has only a few primitive types defined in it, namely:

  • Lua's native basic types (with the exception of nil),
  • the void type (a type having no values),
  • a few constrained numeric types (bit, char, short, int, etc.), and
  • the bitstr and hexstr types (strings representing binary and hexadecimal data).

Additional custom primitive types may be defined with the primitive( ) method.

Derived types can be defined on top of primitive types, and added to the mould set using the typedef( ) and the ftypedef( ) methods. Derived types are defined with a C-like typedef syntax that supports the struct and union constructs, as well as arrays and nesting. For example:

-- defines a few types in the mould set ms:
ms:typedef([[

-- this is a comment (C and C++ style comments are also supported)
#include <stdio.h> /* and lines starting with '#' are ignored */

-- some aliases of primitive types:
typedef boolean myboolean
typedef string mystring
typedef void myvoid

-- a derived structured type:
typedef struct {
   boolean b      
   string  s;  -- optional separators ';' or ',' can be used
   number  arr[5]
} mytype

]])

An instance of a LuaMoulds type is a Lua sequence whose first array-element is a string denoting the type, and the elements that follow are the values of the terminal fields. The special value '?' is assigned to uninitialized (or absent) terminal fields. This special value is configurable.

To instantiate a type, one can use the new( ) method. For example:

x = ms:new('mytype') -- create an uninitialized instance of mytype
--> x = { 'mytype', '?', '?', '?', '?', '?', '?', '?' }

The set( ) and get( ) methods can then be used to set or get the values of the fields. With these methods, fields may be accessed at any level of nesting, including the outermost:

-- set some values:
ms:set(x, 'b', true)          -- set the x.b field
ms:set(x, 's', 'this is s')   -- set the x.s field
ms:set(x, 'arr', 12, 25, 41)  -- set (part of) the x.arr field
--> x = { 'mytype', true, 'this is s', 12, 25, 41, '?', '?' }

-- get some values:
print(ms:get(x, '*'))     --> true  'this is s'  12  25  41  ?  ?  (the whole type)
print(ms:get(x, 'b'))     --> true
print(ms:get(x, 's'))     --> 'this is s'
print(ms:get(x, 'arr'))   --> 12  25  41  ?  ?  (the whole array)
print(ms:get(x, 'arr.2')  --> 25  (only the 2nd element)

-- copy a whole array from an instance to another:
y = ms:new('mytype')
ms:set(y, 'arr', ms:get(x, 'arr'))  
--> y = { 'mytype', '?', '?', 12, 25, 41, '?', '?' }

Fields are referred to with a dot notation similar to the notation used in C, but with a couple of differences: elements of arrays are named with their index, starting from 1 (e.g., the elements of the 'arr' array in the above example are named 'arr.1', 'arr.2', ..., 'arr.5'), and the special name '*' is used to denote the whole type.

Besides having a structure encoded in the type name, LuaMoulds instances are still regular Lua tables, and as such, their fields are dynamically typed. That is, any valid Lua value may be assigned to a terminal field, and values of different types may be assigned to it at different times.

The set( ) and get( ) methods don't do type-checking on the values they write in or read from type instances.

For safe access, LuaMoulds provides the two additional methods tset( ) and tget( ), where the 't' stands for 'terminal' as these methods can only be used to access terminal fields. It provides also two methods, pset( ) and pget( ), to conveniently access instances of primitive types without the need to specify a field name.

Documentation

For more details, see the Reference Manual.

Getting and installing

See here.

Examples

See here.

What's in a name?

Lua tables are ductile like clay, and type definitions are like moulds that give them shapes. Just that.

About

A C-like typedef for Lua

License:Other


Languages

Language:Lua 100.0%