KiwiCodec

KiwiCodec is a pure Elixir implementation of the Kiwi schema binary codec: a compact, schema-driven message format similar in spirit to Protocol Buffers but with its own wire encoding.

The package stays generic: product-specific .kiwi schemas can live in companion packages.

Scope

KiwiCodec owns:

Static modules

Generated modules are regular Elixir structs:

defmodule Example.Node do
use KiwiCodec, kind: :message
field :id, 1, type: :uint
field :name, 2, type: :string
end
node = %Example.Node{id: 42, name: "hello"}
binary = KiwiCodec.encode(node)
node = KiwiCodec.decode(binary, Example.Node)

Enums use atoms:

defmodule Example.Kind do
use KiwiCodec, kind: :enum
enum_value :none, 0
enum_value :frame, 4
end

Schema compilation

For application code, compile schema text into Elixir modules and use the static struct API:

mix kiwi.gen schema.kiwi --module-prefix MyApp.Schema --out lib/generated

For tests and tooling, modules can also be compiled in memory:

KiwiCodec.compile_schema!(schema_text, module_prefix: MyApp.Schema)

KiwiCodec.parse_schema!/1 only parses schema text into an AST. It does not create modules by itself.

Runtime interpretation

When a schema is loaded at runtime and you do not want to generate modules, use KiwiCodec.Runtime:

schema = KiwiCodec.parse_schema!(schema_text)
binary = KiwiCodec.Runtime.encode(schema, "Thing", %{"id" => 1, "name" => "demo"})
value = KiwiCodec.Runtime.decode(schema, "Thing", binary)

Transform modules

Modules can override transform_module/0 for custom normalization before encode and after decode:

defmodule Example.Transform do
@behaviour KiwiCodec.TransformModule
def encode(message, _module), do: message
def decode(message, _module), do: message
end

Binary schemas

binary_schema = KiwiCodec.Schema.Binary.encode(schema)
schema = KiwiCodec.Schema.Binary.decode(binary_schema)

Containers

binary = KiwiCodec.Container.build([KiwiCodec.Container.deflate("schema"), KiwiCodec.Container.deflate("data")])
parsed = KiwiCodec.Container.parse(binary)

Benchmarks

elixir bench/codec_bench.exs

Development

mix deps.get
mix ci

Installation

Once published, add it to your dependencies:

def deps do
[
{:kiwi_codec, "~> 0.1.0"}
]
end