Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Farseen/009 more stuff #17

Merged
merged 9 commits into from
Oct 7, 2024
Merged
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
6 changes: 5 additions & 1 deletion README.md
Original file line number Diff line number Diff line change
@@ -3,6 +3,10 @@
Cardano Data Lite (CDL) aims to be a drop in replacement for Cardano Serialization Library (CSL).
CDL is written in Typescript and compiled to Javascript with minimal dependencies. There are no WASM blobs resulting in much smaller bundle size. CDL is also easier to integrate with various bundlers which lack the ability to load WASM, like ESBuild where top-level await is not available.

## Documentation

[See here](docs.md)

# Development

Due to the size of CSL and also the fact that CSL was undergoing constant updates while we were developing CDL, we designed a small DSL to describe the behaviour of CSL types and used that to dynamically generate a Typescript port of CSL.
@@ -12,7 +16,7 @@ The DSL can be found at `/conway-cddl/yaml` and the code that interprets it can

To measure the progress of implementation, we use automated test suite that parses the type definitions of both CSL and CDL and compares them for compatibility.

To run these, run:
To run these, run the following from the project root directory:

```
npm run test-api
26 changes: 17 additions & 9 deletions conway-cddl/codegen/generators/structured/index.ts
Original file line number Diff line number Diff line change
@@ -23,7 +23,7 @@ export class GenStructuredBase<
constructor(
name: string,
customTypes: SchemaTable,
options: GenStructuredBaseOptions<Field>,
options: GenStructuredBaseOptions<Field>
) {
super(name, customTypes, options);
this.options = options;
@@ -63,7 +63,7 @@ export class GenStructuredBase<
.map((x) => x.name)
.join(",")});
}
`,
`
)}
`;
}
@@ -72,14 +72,22 @@ export class GenStructuredBase<
return this.getFields()
.map(
(x) => `
${this.accessorGetPrefix ? "get_" : ""}${x.name}(): ${this.fieldType(x)} {
return this._${x.name};
}
${this.renameMethod(
this.accessorGetPrefix ? `get_${x.name}` : x.name,
(get) => `
${get}(): ${this.fieldType(x)} {
return this._${x.name};
}`
)}

set_${x.name}(${x.name}: ${this.fieldType(x)}): void {
this._${x.name} = ${x.name};
}
`,
${this.renameMethod(
`set_${x.name}`,
(set) => `
${set}(${x.name}: ${this.fieldType(x)}): void {
this._${x.name} = ${x.name};
}`
)}
`
)
.join("\n");
}
274 changes: 232 additions & 42 deletions conway-cddl/yaml/conway.yaml

Large diffs are not rendered by default.

8 changes: 8 additions & 0 deletions conway-cddl/yaml/custom/bignum.yaml
Original file line number Diff line number Diff line change
@@ -79,3 +79,11 @@ BigNum:
if(a.toJsValue() > b.toJsValue()) return a;
else return b;
}
static _from_number(x: number): BigNum {
return new BigNum(BigInt(x))
}
_to_number(): number {
return Number(this.toJsValue);
}
21 changes: 21 additions & 0 deletions conway-cddl/yaml/custom/hash.yaml
Original file line number Diff line number Diff line change
@@ -65,6 +65,27 @@ KESVKey:
Ed25519Signature:
type: hash
len: 64
methods:
to_bech32: null
from_bech32: null
extra_methods: |
static _BECH32_HRP = "ed25519_sk";
static from_bech32(bech_str: string): Ed25519Signature {
let decoded = bech32.decode(bech_str);
let words = decoded.words;
let bytesArray = bech32.fromWords(words);
let bytes = new Uint8Array(bytesArray);
if(decoded.prefix==Ed25519Signature._BECH32_HRP) {
return new Ed25519Signature(bytes);
} else {
throw new Error("Invalid prefix for Ed25519Signature: " + decoded.prefix);
}
}
to_bech32() {
let prefix = Ed25519Signature._BECH32_HRP;
bech32.encode(prefix, this.inner);
}
KESSignature:
type: newtype
32 changes: 0 additions & 32 deletions conway-cddl/yaml/custom/plutus_witness.yaml

This file was deleted.

15 changes: 0 additions & 15 deletions csl-types/csl-api-differences.txt

This file was deleted.

79 changes: 79 additions & 0 deletions docs.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,79 @@
### Documentation

Cardano Data Lite aims to be as API compatible with Cardano Serialization Library as possible.

So the original documentation for CSL can be used as-is:
https://developers.cardano.org/docs/get-started/cardano-serialization-lib/overview/

There are some rare cases where we had to differ from CSL API. Documentation for those parts and the motivation for the differences are noted below.

## Changes from from CSL

The following sections outline key differences in the API between the Cardano Data Lite (CDL) and the Cardano Serialization Library (CSL).

### `Certificate.Kind`:

CSL crams the `RegCert` and `StakeRegistration` functionalities into a single class `StakeRegistration`.
Same for `UnregCert` and `StakeDeregistration`.
The API to construct the `Certificate` type still has separate constructor for
these, so users who construct, deconstruct and otherwise interact with the
`Certificate` type is unaffected.
But users who inspect the value of `Certificate.kind()` will see that CSL doesn't have an entry for the `RegCert` or `UnregCert` variant, while CDL does.

We decided to do it this way because they are distinct types in conway.cddl and it's easier to think of them as separate types.

Regardless, we are API compatible with CSL except for the `kind()` method.

### `PointerAddress`:

We have removed support for `PointerAddress` because CTL doesn't use them and
generally it's rarely used. We will add this later if needed.

### `PlutusScripts`:

We differ from CSL API where CSL has:

```
TransactionWitnessSet:
- plutus_scripts: PlutusScripts
PlutusScripts: Array of PlutusScript
PlutusScript:
language_version: number
bytes: Uint8Array
```

The issue here is that the following assertion doesn't hold:

```
let plutus_scripts_a = ...;
let plutus_scripts_b = PlutusScripts.from_hex(plutus_scripts.to_hex());
assert(plutus_scripts_a.to_hex() == plutus_scripts_b.to_hex())
```

The `to_hex()` method has no way of encoding language version because the Conway CDDL spec doesn't have a field for encoding language version.

Instead, the Conway CDDL spec has:

```
transaction_witness_set =
{ ..
, ? 3: nonempty_set<plutus_v1_script>
, ? 6: nonempty_set<plutus_v2_script>
, ? 7: nonempty_set<plutus_v3_script>
}
```

A script is considered v1 or v2 depending on which field it's put in.

We adapted the API to more closely follow the Conway CDDL spec.

```
TransactionWitnessSet:
- plutus_script_v1: PlutusScripts
- plutus_script_v2: PlutusScripts
- plutus_script_v3: PlutusScripts
PlutusScripts: Array of Uint8Array
```

Here the previous assertion holds. We uphold the contract that `Foo.from_hex(foo.to_hex())` and `Foo.from_bytes(foo.to_bytes())` are identical to `foo`.
2 changes: 1 addition & 1 deletion src/address/malformed.ts
Original file line number Diff line number Diff line change
@@ -11,7 +11,7 @@ export class MalformedAddress {
return new Uint8Array(this._bytes);
}

from_address(addr: Address): MalformedAddress | undefined {
static from_address(addr: Address): MalformedAddress | undefined {
if (addr._variant.kind == AddressKind.Malformed) {
return addr._variant.value;
}
4 changes: 2 additions & 2 deletions tests/api/api.test.ts
Original file line number Diff line number Diff line change
@@ -212,8 +212,8 @@ for(const rename of classRenames) {

// We filter out the ignored classes from clsClassesMap based on the classInfo file.
// We don't want to fail when checking methods if cdlClassesMap does not contain these classes.
const classInfo: { ignore: Array<string> } = JSON.parse(fs.readFileSync("csl-types/class-info.json", "utf-8"));
for (const cls of classInfo.ignore) {
const classInfo: { ignore_classes: Array<string> } = JSON.parse(fs.readFileSync("tests/class-info.json", "utf-8"));
for (const cls of classInfo.ignore_classes) {
cslClassesMap.delete(cls);
}

16 changes: 13 additions & 3 deletions csl-types/class-info.json → tests/class-info.json
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
{
"ignore": [
"ignore_classes": [
"GenesisKeyDelegation",
"MoveInstantaneousReward",
"MoveInstantaneousRewardsCert",
@@ -21,6 +21,16 @@
"TransactionUnspentOutputs",
"TransactionBatch",
"TransactionBatchList",
"PlutusScript"
]
"PlutusWitness",
"PlutusWitnesses"
],
"ignore_methods": {
"Certificate": [
"new_reg_cert",
"new_unreg_cert",
"as_reg_cert",
"as_unreg_cert"
],
"PlutusScript": ["hash"]
}
}
2 changes: 1 addition & 1 deletion tsconfig.json
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
{
"extends": "@tsconfig/recommended/tsconfig.json",
"include": ["src/**/*"],
"exclude": ["src/generated/out-unformatted.ts"],
"exclude": ["src/generated-unformatted.ts"],
"compilerOptions": {
/* Visit https://aka.ms/tsconfig to read more about this file */
/* Projects */