Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/henriquecbuss/elm-eos

Type-safe interaction with the EOSIO blockchain in Elm.
https://github.com/henriquecbuss/elm-eos

Last synced: 3 months ago
JSON representation

Type-safe interaction with the EOSIO blockchain in Elm.

Awesome Lists containing this project

README

        

# henriquecbuss/elm-eos

This package provides a way to interact with the EOS blockchain, while maintaining
Elm's type safety. In order to do so, it reads the ABI of the contracts you want
to interact with, and generates all of the necessary Elm code to interact with
them.

## Table of contents

- [🏃 Getting started](#getting-started)
- [️🏗️ Scaffold new app](#scaffold-new-app)
- [➕ Add to existing app](#add-to-existing-app)
- [📕 Overview](#overview)
- [📖 Examples](#examples)
- [🤖 CLI](#cli)
- [🆘 Support](#support)

## Getting started

### Scaffold new app

If you want to start a new app, you can use the [create-elm-eos-dapp]() tool to
scaffold a new app for you. It will generate anew app with all of the necessary
dependencies and code to get you started, along with linters, formatters, and all
of that good stuff.

- With `yarn`:

```sh
yarn create elm-eos-dapp
```

- With `npm`:

```sh
npx create-elm-eos-dapp
```

- With `pnpm`:

```sh
pnpm create elm-eos-dapp
```

> _The scripts in the generated package.json use `npm`, so you have to manually change them if you want to use `yarn` or `pnpm`._

### Add to existing app

There are a few moving pieces that need to work together in order to use
`henriquecbuss/elm-eos`. Here's how to set them up:

1. Add the elm package as a dependency in your `elm.json`, and install these libraries:

```bash
elm install henriquecbuss/elm-eos
elm install elm/json
elm install elm/http
elm install elm/time
elm install NoRedInk/elm-json-decode-pipeline
elm install elm-community/maybe-extra
elm install elm-community/result-extra
```

2. Install the `elm-eos` command line through npm. This is what you will
use to generate Elm code for the EOSIO API. You can save it as a dev dependency
to ensure everyone on your team has access to the same version:

- With `npm`:

```sh
npm install --save-dev elm-eos
```

- With `yarn`:

```sh
yarn add --dev elm-eos
```

- With `pnpm`:

```sh
pnpm add -D elm-eos
```

3. Run the CLI tool to generate the Elm code for your contracts. If you installed
like shown above, you can run with `npx elm-eos`, or create a script in your `package.json` like this:

```json
{
"scripts": {
"generate:eos": "elm-eos https://mydomain.com/v1/chain --contract my.first --contract my.second --output generated --base Contracts"
}
}
```

This will generate files in `Contracts/My/First` and `Contracts/My/Second`

4. Now whenever you need to refresh the generated code, you can run `npm run generate-eos`.

## Overview

This package mainly serves to translate EOSIO types to Elm types. There is a
[CLI app](#cli) that is responsible for generating code specific to the contracts
you need. The exception to that is the [Eos.Query](https://package.elm-lang.org/packages/henriquecbuss/elm-eos/1.1.0/Eos-Query/) module, which is
used to send queries to the blockchain.

For example, let's say you have a contract named `company.acc`, with an `accounts`
table, described by this struct:

```json
{
"name": "account",
"base": "",
"fields": [
{
"name": "account_name",
"type": "name"
},
{
"name": "balance",
"type": "asset"
}
]
}
```

In order to get data from that table, you would write some Elm code like this
(all of the code that starts with `Company.Acc` is generated by the [CLI app](#cli)):

```elm
import Company.Acc.Table.Query
import Company.Acc.Table
import Eos.Query
import Http

fetchAccounts : (Result Http.Error (Eos.Query.Response Company.Acc.Table.Account) -> msg) -> Cmd msg
fetchAccounts toMsg =
Company.Acc.Table.Query.account { scope = "company.acc" }
|> Eos.Query.withLimit 100
|> Eos.Query.send toMsg
```

You never need to define JSON encoders, decoders, or types for the data you're
querying for. The generated code does all of that for you, including setting the
target url (though you can override that if you need).

Here's a sneak peak of what the generated code looks like:

```elm
-- In file Company/Acc/Table/Query.elm

import Eos.Query

account : { scope : String } -> Eos.Query.Query Company.Acc.Table.Account

-- In file Company/Acc/Table.elm

import Eos.Name
import Eos.Asset

type alias Account =
{ accountName : Eos.Name.Name
, balance : Eos.Asset.Asset
}
```

Similarly, if you want to **send** data to the blockchain, all of the code is
generated for you. For example, let's say you want to send a `transfer` action,
described by this struct:

```json
{
"name": "transfer",
"base": "",
"fields": [
{
"name": "from",
"type": "name"
},
{
"name": "to",
"type": "name"
},
{
"name": "amount",
"type": "asset"
}
]
}
```

In order to generate JSON to send over to the blockchain, all you have to do is
this:

```elm
import Json.Encode
import Eos.Name
import Eos.Permission
import Eos.Asset
import Company.Acc.Action

sendMoney : { currentUser : Eos.Name.Name, currentPermission : Eos.Permission.Permission to : Eos.Name.Name, amount : Eos.Asset.Asset } -> Json.Encode.Value
sendMoney { currentUser, currentPermission, to, amount } =
Company.Acc.Action.Transfer
{ from = currentUser
, to = to
, amount = amount
}
|> Company.Acc.Action.encode [ { actor = currentUser, permission = currentPermission } ]
```

You can then send that JSON to the blockchain using the `eosjs` library (through
a port), or using your preferred method. The [contractExplorer](https://github.com/henriquecbuss/elm-eos/tree/main/examples/contractExplorer) example shows
how to integrate with external wallets to submit transactions.

### Examples

You can take a look at the [examples](https://github.com/henriquecbuss/elm-eos/tree/main/examples) folder
for complete examples (each one has more detailed information in their `README.md`):

- memberFetching ([source](https://github.com/henriquecbuss/elm-eos/tree/main/examples/memberFetching) | [live demo](https://elm-eos-member-fetching.netlify.app)): a simple example of fetching data from the blockchain and displaying it in a table. This is the simplest example there is, and it's a good place to start.
- transfer ([source](https://github.com/henriquecbuss/elm-eos/tree/main/examples/transfer) | [live demo](https://elm-eos-transfer.netlify.app)): a more complex example, where you can send a transfer to the blockchain. This is an example for when you know what actions do and which ones you want to use.
- contractExplorer ([source](https://github.com/henriquecbuss/elm-eos/tree/main/examples/contractExplorer) | [live demo](https://elm-eos-contract-explorer.netlify.app)): an even more complex example, where you can explore the tables and actions of multiple contracts. This is an example for when you want to explore the API of a contract, and see what actions and tables it has. This example also includes integration with third-party wallets using [eos-transit](https://github.com/eosnewyork/eos-transit). This app is roughly what the tool generation does for you if you [ask it to generate an app based on some contracts](#scaffold-new-app).

## CLI

The CLI tool is used to generate the Elm code for your contracts. It takes some
arguments:

- URL: the base url of the contracts. For example: `https://mydomain.com/v1/chain`
- Output: the directory to write the generated code to. For example: `generated`
- Base: the base module name to use for the generated code. For example: `My.Contract`
- Contracts: the names of the contracts to generate code for. For example: `--contract first --contract second`

Joining all of that together, you can run the CLI like this:

```bash
elm-eos https://mydomain.com/v1/chain --contract first --contract second --output generated --base My.Contract
```

Only URL and Contracts are required.

## Support

If you need any kind of support, feel free to open an issue on GitHub, or ping me @HenriqueBuss on the [Elm Slack](https://elm-lang.org/community/slack).