#source #parser #typescript #transforming #deno #transform #input

deno-bindgen2-common

Common library used by the deno-bindgen2 project. Contains mainly implmenentations for parsing and transforming Rust source code.

1 stable release

1.0.1 Nov 14, 2024
1.0.0 Nov 13, 2024

#1829 in Rust patterns

30 downloads per month
Used in 4 crates (2 directly)

Custom license

150KB
3.5K SLoC

deno-bindgen2-common

This library contains common code for parsing and transforming Rust source code. It has two primary consumers: deno-bindgen2-macro and deno-bindgen2-cli

The macro crate uses this library to parse and transform source code and generate FFI symbol wrappers. The cli crate uses this library to parse Rust source code and generate a close analogue for TypeScript.

Primer

This library contains three primary abstractions: Type, FnItem, ImplItem.
Type represents the types that are parsed from the user's source code.
This library classifies Type into two major categories: Trivial and NonTrivial

flowchart LR
subgraph Type
    Trivial
    a["Non Trivial"]
end

Trivial types are those that can be safely passed between FFI boundaries and are supported in both source and target language contexts. Examples of this are primitive types like u8, usize, f64, bool, and char.

On the other hand, non-trivial types are those that cannot be safely passed between FFI boundaries. These non-trivial types must be decomposed to their raw components. Examples are collections like str (in Rust, a string is a collection of bytes u8), String, Vec<T>, and tuples (A, B, i32).

There are also pointer and reference types like *const T, &mut T, and Box<T>. These are technically considered trivial types since their values can be safely passed between language contexts and FFI boundaries. However, they are also considered unsafe and non-robust since the use of their values could potentially lead to undefined behavior and memory vulnerabilities.

For example, Rust has a unique language construct surrounding the concept of borrowing and ownership. In an FFI context, the other language should not violate the safety guarantees and checks of Rust. Doing otherwise leads to undefined behavior such as memory leaks and use after free bugs.

Next, an FnItem represents a single Rust function, which has a signature that describes its inputs and outputs. The inputs and output of a function are denoted by their Type

flowchart LR
    Type

    subgraph Function Item
        Inputs
        Outputs
    end

    Type --> Inputs
    Type --> Outputs

An ImplItem is an implementation of a type, likely a UserDefined type, which can contain associated FnItem functions

The source code consists of multiple instances of an Item, which could either be an FnItem, an ImplItem, or any other valid Rust item that isn't parsed by this library

flowchart TD
    subgraph fn[Function Item]
        Inputs
        Outputs
    end

    Type --> Inputs
    Type --> Outputs

    subgraph impl[Implement Item]
        impl_fn[Associated Functions]
    end

    fn --> impl_fn

    subgraph Source Code
        items[Items]
    end

    fn --> items
    impl --> items

Dependencies

~10–19MB
~271K SLoC