Awesome
<h2 align="center">AssemblyScript SCALE Codec</h2>as-scale-codec is AssemblyScript implementation of Polkadot SCALE Codec. The codec is used as a communication mechanism between Polkadot Hosts and Polkadot Runtimes.
More detailed information about the SCALE codec specification can be found here.
This AssemblyScript implementation of the codec is funded by Web3 Foundation via their Open Grants Program! :pray:
Supported types
The following table shows the status of the types and their arrays:
Type | Support | Array Support |
---|---|---|
Fixed width number | ✅ | ✅ |
Compact Int | ✅ | ✅ |
Big Integer | :small_orange_diamond: Limited Support | :small_orange_diamond: Limited Support |
Byte | ✅ | ✅ |
Bool | ✅ | ✅ |
Hash | ✅ | :heavy_minus_sign: |
String | ✅ | ✅ |
Map | ✅ | :heavy_minus_sign: |
The following table shows the status of the fixed width numbers:
Тype | 8 | 16 | 32 | 64 | 128 | 256 |
---|---|---|---|---|---|---|
int | ✅ | ✅ | ✅ | ✅ | :heavy_minus_sign: | :heavy_minus_sign: |
uint | ✅ | ✅ | ✅ | ✅ | ✅ | :heavy_minus_sign: |
Special Types
- Compact Int - Documentation
Getting Started
You can find more information on AssemblyScript and how to get started with it in the AssemblyScript docs -> https://www.assemblyscript.org/introduction.html
-
In your AssemblyScript project execute:
npm install as-scale-codec
-
Once you have the library installed in your AssemblyScript project you can use it in your
assembly
files by importing the files fromas-scale-codec
.
Detailed examples of the exported by the library types are listed below:
Types
Encoding
Every type has а toU8a function. It encodes type value into an array of bytes
import { Bool, Byte, ScaleString, Hash, CompactInt } from "as-scale-codec"
import { Int8, Int16, Int32, Int64 } from "as-scale-codec"
import { UInt8, UInt16, UInt32, UInt64, UInt128 } from "as-scale-codec"
// ScaleMap
const scaleMap = new ScaleMap<Int32, Bool>();
scaleMap.set(new Int32(1), new Bool(false));
scaleMap.toU8a() // => [4, 1, 0, 0, 0, 0];
// Bool
const scaleBool = new Bool(true);
scaleBool.toU8a() // => [0x01]
// Byte
const scaleByte = new Byte(0x01);
scaleByte.toU8a() // => [0x01]
// String
const scaleString = new ScaleString("a");
scaleString.toU8a() // => [0x04, 0x61]
// Hash
const scaleHash = new Hash([0xff, 0x00, 0xab]);
scaleHash.toU8a() // => [0xff, 0x00, 0xab, 0x00, ... 0x00] (32 bytes long)
// Compact Int
const scaleCompactInt = new CompactInt(1);
scaleCompactInt.toU8a() // => [0x04]
// Int
const scaleInt8 = new Int8(-1);
scaleInt8.toU8a() // => [0xff]
const scaleInt16 = new Int16(-1);
scaleInt16.toU8a() // => [0xff, 0xff]
const scaleInt32 = new Int32(-1);
scaleInt32.toU8a() // => [0xff, 0xff, 0xff, 0xff]
const scaleInt64 = new Int64(-1);
scaleInt64.toU8a() // => [0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff]
// UInt
const scaleUInt8 = new UInt8(1);
scaleUInt8.toU8a() // => [0x01]
const scaleUInt16 = new UInt16(1);
scaleUInt16.toU8a() // => [0x01, 0x00]
const scaleUInt32 = new UInt32(1);
scaleUInt32.toU8a() // => [0x01, 0x00, 0x00, 0x00]
const scaleUInt64 = new UInt64(1);
scaleUInt64.toU8a() // => [0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00]
const scaleUInt128 = new UInt128(u128.fromU64(18446744073709551615));
scaleUInt128.toU8a() // => [0x13, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff]
Decoding
Every type has a static function fromU8a. It decodes an array of bytes to the desired type
import { ScaleMap, Bool, Byte, ScaleString, Hash, CompactInt } from "as-scale-codec"
import { Int8, Int16, Int32, Int64 } from "as-scale-codec"
import { UInt8, UInt16, UInt32, UInt64, UInt128 } from "as-scale-codec"
// Bool
Bool.fromU8a([0x01]); // => new Bool(true)
// Byte
Byte.fromU8a([0x01]); // => new Byte(0x01)
// String
Byte.fromU8a([0x04, 0x61]); // => new ScaleString('a')
// Hash
Hash.fromU8a([0xff, 0x00, 0xab]);
// => [0xff, 0x00, 0xab, 0x00, ... 0x00] (32 bytes long)
ScaleMap<Int32, Bool>.fromU8a([4, 1, 0, 0, 0, 0]);
// => const scaleMap = new ScaleMap<Int32, Bool>()
// => scaleMap.set(new Int32(1), new Bool(false))
// Compact Int
CompactInt.fromU8a([0x04]); // => new CompactInt(1)
// Int
Int8.fromU8a([0xff]); // => new Int8(-1)
Int16.fromU8a([0xff, 0xff]); // => new Int16(-1)
Int32.fromU8a([0xff, 0xff, 0xff, 0xff]); // => new Int32(-1)
Int64.fromU8a([0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff]); // => new Int64(-1)
// UInt
UInt8.fromU8a([0x01]); // => new UInt8(1)
UInt16.fromU8a([0x01, 0x00]); // => new UInt16(1)
UInt32.fromU8a([0x01, 0x00, 0x00, 0x00]); // => new UInt32(1)
UInt64.fromU8a([0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00]); // => new UInt64(1)
UInt128.fromU8a([0x33, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff]);
// => 340282366920938463444927863358058659840
Arrays
Encoding
Every array has toU8a function. It encodes the array values into an array of SCALE encoded bytes.
import { BoolArray, ByteArray, IntArray, StringArray } from "as-scale-codec"
// Bool Array
const boolArray = new BoolArray([true, false, true]);
boolArray.toU8a(); // => [0x0c, 0x01, 0x00, 0x01]
// Byte Array
const byteArray = new ByteArray([0x01, 0x01, 0x01]);
byteArray.toU8a(); // => [0x0c, 0x01, 0x01, 0x01]
// Int Array
const intArray = new IntArray([16384, 2, 3, 4]);
intArray.toU8a() // => [0x10, 0x02, 0x00, 0x01, 0x00, 0x08, 0x0c, 0x10]
// String Array
const stringArray = new StringArray(["hello", "world"]);
stringArray.toU8a() // => [0x08, 0x14, 0x68, 0x65, 0x6c, 0x6c, 0x6f, 0x14, 0x77, 0x6f, 0x72, 0x6c, 0x64]
Decoding
Every array has a static function fromU8a. It decodes an array of SCALE encoded bytes and creates an array instance of the desired type.
import { BoolArray, ByteArray, IntArray, StringArray } from "as-scale-codec"
// Bool Array
BoolArray.fromU8a([0x0c, 0x01, 0x00, 0x01]);
// => new BoolArray([true, false, true])
// Byte Array
ByteArray.fromU8a([0x0c, 0x01, 0x01, 0x01])
// => new ByteArray([0x01, 0x01, 0x01])
// Int Array
IntArray.fromU8a([0x10, 0x02, 0x00, 0x01, 0x00, 0x08, 0x0c, 0x10])
// => new IntArray([16384, 2, 3, 4])
// String Array
StringArray.fromU8a([0x08, 0x14, 0x68, 0x65, 0x6c, 0x6c, 0x6f, 0x14, 0x77, 0x6f, 0x72, 0x6c, 0x64])
// => new StringArray(["hello", "world"])
BytesReader
If you have an array of arbitrary SCALE encoded bytes that you need to decode, BytesReader
class is a preferred way to do it:
import { BytesReader } from 'as-scale-codec';
// Arbitrary SCALE encoded bytes
const bytes: u8[] = [
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
69, 0, 0, 0,
110, 125, 239, 2,
56, 97, 115, 45, 115, 99, 97, 108, 101, 45, 99, 111, 100, 101, 99,
128, 1, 10, 0, 0, 0, 2, 2, 1, 123, 33, 3, 1, 35, 34, 5, 8, 22, 52, 1, 0, 0, 0, 1, 1, 1, 56, 21, 142, 13, 13, 1,
0
];
// Instantiate BytesReader instance with SCALE encoded bytes
const bytesReader = new BytesReader(bytes);
// Read Int64
bytesReader.readInto<Int64>();
// => new Int(-1)
// Read UInt32
bytesReader.readInto<UInt32>();
// => new UInt32(69)
// Read CompactInt
bytesReader.readInto<CompactInt>();
// => new CompactInt(12312411)
// Read ScaleString
bytesReader.readInto<ScaleString>();
// => new ScaleString("as-scale-codec")
// Read Hash
bytesReader.readInto<Hash>();
// => new Hash([128, 1, 10, 0, 0, 0, 2, 2, 1, 123, 33, 3, 1, 35, 34, 5, 8, 22, 52, 1, 0, 0, 0, 1, 1, 1, 56, 21, 142, 13, 13, 1])
// Read Bool
bytesReader.readInto<Bool>();
// => new Bool(false)
// If you have single SCALE encoded type, you can use static decodeInto<T>() function of BytesReader
const uInt64Bytes: u8[] = [1, 0, 0, 0, 0, 0, 0];
// Read UInt64
BytesReader.decodeInto<UInt64>(uInt64Bytes);
// => new UInt64(1)
const hashBytes: u8[] = [0xff, 0x00, 0xab];
// Read Hash
BytesReader.decodeInto<Hash>(hashBytes);
// new Hash([0xff, 0x00, 0xab])
const mapBytes: u8[] = [2, 1, 0, 1, 0, 0, 0, 3, 0, 3, 0, 0, 0];
// Read ScaleMap
BytesReader.decodeInto<ScaleMap<UInt16, UInt32>>(mapBytes);
// => const scaleMap = new ScaleMap<UInt16, UInt32>();
// => scaleMap.set(new UInt16(1), new UInt32(1))
// => scaleMap.set(new UInt16(3), new UInt32(3))
const cmpBytes: u8[] = [169, 2];
// Read CompactInt
BytesReader.decodeInto<CompactInt>(cmpBytes);
// new CompactInt(170)
const int8Bytes: u8[] = [0xff];
// Read Int8
BytesReader.decodeInto<Int8>(int8Bytes);
// new Int8(-1)
Miscellaneous
Convert bytes to hash
Hash.bytesToHash([0xff, 0x00, 0xab]);
// => [0x00, ... , 0x00, 0xff, 0x00, 0xab] (32 bytes long)
Hash.bytesToHash([0xff, 0x00, ..., 0x00]); // (32 bytes long)
// => [0xff, ... , 0x00] (32 bytes long)
Tests
In order to run the unit tests, one must perform:
npm run test
License
This repository is licensed under Apache 2.0 license