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:
- Kiwi wire primitives: varuint, zigzag int, uint64/int64, varfloat, null-terminated strings, and length-prefixed byte arrays
.kiwischema parsing and validation- Runtime interpretation for parsed schemas
- Field metadata, including original schema field names
- An idiomatic Elixir DSL and code generator for static modules
- Generic chunk container helpers
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