---
tags: Technical specification
order: 5
---
# Oracle Binary Encoding (OBI)
Oracle Binary Encoding (OBI) is the standard way to serialized and deserialize binary data in the BandChain ecosystem. Similar to Ethereum's [Contract ABI Specification](https://solidity.readthedocs.io/en/latest/abi-spec.html) or Google's [ProtoBuf](https://developers.google.com/protocol-buffers), an OBI schema explains how a data object, in any supported programming language, can be encoded to and decoded from plain bytes.
OBI is designed with the following properties in mind:
- **Compactness**: OBI schema will be stored on-chain and passed around between blockchains. Thus, it is important to keep the size of the schema specification as small as possible.
- **Simplicity & Portability**: As a blockchain-agnostic protocol, OBI serialization and deserialization must be easy to implement in any environment. Consequently, complex platform-specific features are not supported.
- **Readability**: Lastly, OBI is intended to be used as a communication tool between oracle script creators and smart contract developers. It must be intuitive for readers to understand the OBI underlying objects from reading the schema.
## Technical Specification
An OBI schema is a non self-describing binary serialization format of multiple objects. Some particular notes:
- An OBI schema consists of one or more individual schemas. In most cases, an OBI schema will consist of two individual schemas: the input type and the output type.
- 6 sizes (8-bit, 16-bit, 32-bit, 64-bit, 128-bit, and 256-bit) of signed and unsigned integers are supported. There are all serialized into big-endian bytes.
- Strings, bytes, vectors are serialized with their length as u32 first, followed by their contents.
- Structs are serialized field by field in the declaration order.
### Backus–Naur Form (BNF) Grammar Specification
Below is the [Backus–Naur form (BNF)](https://en.wikipedia.org/wiki/Backus%E2%80%93Naur_form) grammar of an OBI schema.
```
<obi_schema> ::= <obi_schema> | <obi_schema> "/" <indv_schema>
<indv_schema> ::= <int_schema> | <uint_schema> | <string_schema> |
<bytes_schema> | <vector_schema> | <struct_schema>
<int_schema> ::= "i8" | "i16" | "i32" | "i64" | "i128" | "i256"
<uint_schema> ::= "u8" | "u16" | "u32" | "i64" | "u128" | "u256"
<string_schema> ::= "string"
<bytes_schema> ::= "bytes"
<vector_schema> ::= "[" <indv_schema> "]"
<struct_schema> ::= "{" <struct_fields> "}"
<struct_fields> ::= <struct_member> | <struct_fields> "," <struct_member>
<struct_member> ::= <identifier> ":" <indv_schema>
```
### Pseudocode Implementation
Below is a example [pseudocode](https://en.wikipedia.org/wiki/Pseudocode) implementation of OBI schema declaration and the corresponding serializing function in a somewhat broken function language :P. Deserialization function is essentially the inverse of the serialization function.
```ocaml
(* An individual schema consists of 6 possible cases. *)
type indv_schema :=
| Int(int)
| Uint(int)
| String
| Bytes
| Vector(indv_schema)
| Struct([(string, indv_schema)])
(* An OBI schema is essentially a list of individual schemas. *)
type obi_schema := [indv_schema]
(* Encode serializes the given object into bytes. *)
let encode (s : indv_schema) (o : object) :=
match s with
| Int(sz) => be_signed_encode(o, sz)
| Uint(sz) => be_unsigned_encode(o, sz)
| String => be_unsigned_encode(len(o), 32) ++ bytes_of_string(o)
| Bytes => be_unsigned_encode(len(o), 32) ++ o
| Vector(s) => be_unsigned_encode(len(o), 32) ++ concat (map o (encode s))
| Struct(fs) => concat (map fs (fun (f, s) => encode s o[f]))
```
## OBI Schema Examples
As an example, below is an example OBI schema of an oracle script to fetch the price of a cryptocurreny, which is then multiplied by a certain multiplier. The OBI itself schema consists of two inner schemas, one for the inputs to the oracle script and the other for the output.
- The input is a struct with 2 fields: a string symbol and a u64 multiplier.
- The output is a struct with 2 fields: a u64 final price and a vector of struct each has string name and u64 timestamp.
```sh
# Compact OBI representation...
{symbol:string,multiplier:u64}/{price:u64,sources:[{name:string,time:u64}]}
# Prettified OBI representation...
{
symbol: string,
multiplier: u64
} / {
price: u64,
sources: [{ name: string, time: u64 }]
}
```
### Example Object Serialization
```
{"symbol": "BTC", "multiplier": 1000000000}
0x00000003425443000000003b9aca00
^ ^ ^
| | +- 64-bit be encode of 1000000000 is 0x000000003b9aca00
| +------- "BTC" data is encoded as 0x425443
+--------------- 32-bit be encode of length 3 is 0x00000003
{
"price": 9268300000000,
"sources": [
{"name": "CoinGecko", "time": 1590305341},
{"name": "CryptoCompare", "time": 1590305362}
]
}
0x0000086df1baab000000000200000009436f696e4765636b6f000000005eca223d0000000d43727970746f436f6d70617265000000005eca2252
```
## Reference Implementations
OBI serialization libraries are being actively developed in multiple programming languages.
- GO package
- [pkg.go](https://pkg.go.dev/github.com/bandprotocol/chain/v2@v2.0.3/pkg/obi)
- obi-rs
- [code](https://github.com/bandprotocol/bandchain/tree/master/obi/obi-rs)
- [crates.io](https://crates.io/crates/obi)
- pyobi
- [code](https://github.com/bandprotocol/pyband/blob/master/pyband/obi.py)
- [pypi](https://pypi.org/project/pyband)
- obi.js
- [code](https://github.com/bandprotocol/bandchain.js/blob/master/src/obi.ts)
- [npm](https://www.npmjs.com/package/@bandprotocol/bandchain.js)