MixinBot

CI

Ruby SDK and CLI for Mixin Network: authenticated REST calls, Safe UTXO transfers, Blaze messaging, network asset catalog, inscriptions, invoices and mix addresses, transaction encoding, and optional MVM (Mixin Virtual Machine) helpers.

The gem aims for parity with the official bot-api-go-client Go SDK. See API_COVERAGE.md for the full mapping; run rake mixin_bot:api_coverage to confirm no gaps are marked missing.

Current gem version: 2.0.0 (see CHANGELOG.md for breaking changes and deprecations).

Requirements

  • Ruby ≥ 3.2 (CI runs 3.2, 3.3, and 4.0).
  • Bundler 2.5+ recommended, especially on Ruby 4.
  • Optional: the mixin CLI in PATH if you use MixinBot::API#encode_raw_transaction_native / #decode_raw_transaction_native or the experimental MixinBot::NodeCLI helpers.

Installation

Add to your Gemfile:

gem 'mixin_bot'

Then:

bundle install

Or install the gem directly:

gem install mixin_bot

The gem ships the mixinbot executable (see CLI).

Quick start

1. Configure credentials

Set the fields your flows need. Safe transfers and signing require a spend key (spend_key) in addition to session material.

require 'mixin_bot'

MixinBot.configure do
  self.app_id = 'your-app-uuid'
  self.client_secret = 'your-client-secret' # OAuth flows
  self.session_id = 'your-session-uuid'
  self.session_private_key = '...' # seed or full Ed25519 private key; Base64 or hex
  self.server_public_key = '...'   # pin token / server public key
  self.spend_key = '...'           # Ed25519 spend private key for Safe UTXO signing
  # self.pin = self.spend_key        # optional; used where PIN material is required
  # self.api_host = 'api.mixin.one'
  # self.blaze_host = 'blaze.mixin.one'
  # self.debug = true               # Faraday response logging
end

MixinBot::Configuration accepts common aliases: client_idapp_id, private_keysession_private_key, pin_tokenserver_public_key. Keys are normalized (e.g. 32-byte Ed25519 seeds expanded to 64-byte signing keys) where appropriate.

2. Call the API

api = MixinBot.api # singleton using global config, or MixinBot::API.new(...)

api.me['full_name']
api.assets
api.network_asset('c6d0c728-2624-429b-8e0d-d9d19b6592fa') # public network catalog
api.fetch_user_sessions(['user-uuid'])

create_transfer with keyword arguments runs the Safe pipeline (UTXO select → build → verify → sign → submit). You can also call create_safe_transfer explicitly. Configure spend_key first.

result = MixinBot.api.create_transfer(
  members: '6ae1c7ae-1df1-498e-8f21-d48cb6d129b5',
  asset_id: '965e5c6e-434c-3fa9-b780-c50f43cd955c',
  amount: '0.01',
  memo: 'payment',
  trace_id: SecureRandom.uuid
)

# Multisig: 2-of-3
MixinBot.api.create_safe_transfer(
  members: %w[uuid-1 uuid-2 uuid-3],
  threshold: 2,
  asset_id: '965e5c6e-434c-3fa9-b780-c50f43cd955c',
  amount: '0.01'
)

Lower-level steps: build_utxos, build_safe_transaction, verify_raw_transaction / create_safe_transaction_request, sign_safe_transaction, send_safe_transaction (batch via requests:).

Aliases aligned with the Go SDK include send_transaction, send_transfer_transaction, and get_transaction_by_id (→ safe_transaction).

4. Legacy POST /transfers

If the first argument is a PIN string and you pass opponent_id: (old Messenger transfer shape), create_transfer delegates to create_legacy_transfer (deprecated, warns once). Prefer Safe transfers for new code.

# Legacy only — deprecated
MixinBot.api.create_legacy_transfer(
  pin,
  asset_id: '...',
  opponent_id: '...',
  amount: 0.00000001,
  trace_id: SecureRandom.uuid
)

HTTP responses (ApiEnvelope)

MixinBot::Client returns MixinBot::Models::ApiEnvelope for REST calls. It wraps the raw JSON so you can use either envelope or flattened shapes:

res = MixinBot.api.client.get('/me')
res['data']['user_id'] # envelope
res['user_id']          # delegated lookup into `data` when present
res.to_h                # raw Hash

Many convenience methods on MixinBot::API still return the inner data hash where that was the historical contract (e.g. #me).

Library layout

MixinBot::API modules

MixinBot::API composes one module per API area (all methods are available on MixinBot.api):

Module Examples
Me me, safe_me, update_me, friends, update_preferences, relationship
User user, fetch_users, search_user, create_user, safe_register, migrate_to_safe
Session fetch_user_sessions
LegacyUser upgrade_legacy_user
Asset assets, asset, ticker, fetch_assets, asset_fee, asset_balance
NetworkAsset network_asset, network_ticker, network_asset_search
Network network_assets, network_assets_top
Fiat fiats
Chain network_chain, network_chains, chain_name, chain_id?
Transfer create_transfer, create_safe_transfer, build_utxos, send_transaction, …
Transaction create_safe_keys, build_safe_transaction, verify_raw_transaction, sign_safe_transaction, build_object_transaction, create_object_storage_transaction, …
Output safe_outputs, safe_output, build_threshold_script
Deposit pending_safe_deposits
Address (deposit entries) safe_deposit_entries
Snapshot safe_snapshots, safe_snapshot, create_safe_snapshot_notification
LegacySnapshot snapshots, snapshot, snapshot_by_trace_id, network_snapshots, …
Payment safe_pay_url
LegacyPayment pay_url, verify_payment (deprecated)
Multisig create_safe_multisig_request, safe_multisig_request, create_multisig_raw_tx
LegacyMultisig create_multisig_request, cancel_multisig_request, …
LegacyOutput legacy_outputs, read_multisigs, create_output, …
LegacyTransfer create_legacy_transfer, legacy_transfer
LegacyTransaction build_raw_transaction, create_multisig_transaction, …
Inscription collection, collectible, build_inscribe_transaction, …
LegacyCollectible legacy collectible requests (deprecated)
Withdraw withdrawals, create_withdraw_address, check_address, withdraw_addresses
Conversation conversation, create_group_conversation, join_conversation, …
Message send_message, send_plain_messages, Blaze helpers
EncryptedMessage send_encrypted_*, encrypt_message, decrypt_message
Blaze blaze, start_blaze_connect, blaze_send_plain_text, …
Attachment create_attachment, upload_attachment
Auth oauth_token, authorize_code, access_token, sign_oauth_access_token
Pin / Tip verify_pin, update_pin, update_tip_pin, encrypt_tip_pin, get_tip_node, tip_body_for_*
App favorite_apps, transfer_app_ownership (migrate)
Code read_code, read_multisig_by_code
Turn turn_servers
Rpc rpc_proxy, send_raw_transaction, get_transaction, …
ComputerApi delegates to MixinBot::Computer (get_computer_info, register_computer, …)

Top-level helpers on MixinBot::API: access_token, encode_raw_transaction, decode_raw_transaction, native variants via mixin CLI.

Other libraries

Area Description
MixinBot::Client Faraday HTTP client (JSON, retries, optional debug).
MixinBot::Configuration Credentials and hosts.
MixinBot::Utils Crypto, JWT, encoding, unique_object_id, generate_user_checksum, …
MixinBot::Transaction Encode/decode raw Safe transactions.
MixinBot::MixAddress Parse/build MIX… addresses; request_or_generate_ghost_keys.
MixinBot::Invoice MIN… payment invoices.
MixinBot::UrlScheme mixin:// deep links (scheme_users, scheme_pay, …).
MixinBot::Computer Mixin Computer HTTP API (separate host).
MixinBot::BotAuth Sign bot-platform requests (BotAuth::Client#sign_request).
MixinBot::Monitor YAML monitor messages, report_to_monitor, check_retryable_error.
MixinBot::UUID, MixinBot::Nfo UUID and NFT memo helpers.
MVM Optional MVM namespace: MVM::Bridge, MVM::Nft, MVM::Scan, MVM::Registry.

Errors

Custom errors under MixinBot:: include ResponseError, UnauthorizedError, InsufficientBalanceError, UtxoInsufficientError, PinError, and InvalidInvoiceFormatError. See lib/mixin_bot/errors.rb.

Multiple bots

bot_a = MixinBot::API.new(
  app_id: '...',
  session_id: '...',
  session_private_key: '...',
  server_public_key: '...',
  spend_key: '...'
)

bot_b = MixinBot::API.new(...) # separate configuration

bot_a.me
bot_b.me

Blaze (WebSocket)

Blaze uses a WebSocket after JWT auth. start_blaze_connect yields a Faye::WebSocket::Client; define on_open / on_message / on_error / on_close on the receiver (see examples/blaze.rb). Run an event loop (e.g. EventMachine) in your app.

require 'eventmachine'
require 'mixin_bot'

EM.run do
  MixinBot.api.start_blaze_connect do
    def on_open(blaze, _event)
      blaze.send list_pending_message
    end

    def on_message(blaze, event)
      raw = JSON.parse ws_message(event.data)
      blaze.send acknowledge_message_receipt(raw['data']['message_id']) if raw.dig('data', 'message_id')
    end
  end
end

For outbound messages over an open socket, use blaze_send_plain_text, blaze_send_contact, blaze_send_app_card, and related helpers (parity with Go BlazeClient).

MixinBot::UrlScheme.scheme_pay(
  asset_id: '...',
  trace_id: SecureRandom.uuid,
  recipient_id: '...',
  memo: 'hello',
  amount: '0.01'
)

client = MixinBot::BotAuth.new_client(MixinBot.api)
token = client.sign_request(Time.now.to_i, bot_user_id, 'GET', '/some/path')

CLI

Invoke mixinbot (global options: -a / --apihost, -o / --output pretty|json|yaml, -r / --pretty).

When stdout is piped, output defaults to JSON. Use mixinbot schema -o json for machine-readable command discovery. See docs/agent/cli.md.

Subcommands that talk to the API accept -k / --keystore: path to a JSON file or inline JSON (app_id, session_id, session_private_key, server_public_key, spend_key, client_secret, pin, etc.). Without -k, the CLI uses global MixinBot.configure credentials.

Command Purpose
mixinbot call METHOD Invoke any MixinBot::API method (-d JSON kwargs, optional positional args).
mixinbot list [FILTER] List callable API methods (grouped by module).
mixinbot api PATH Signed GET/POST via MixinBot::Client (-m, -d, -p, -t).
mixinbot transfer USER_ID Safe transfer (create_safe_transfer; --asset, --amount, …).
mixinbot legacy-transfer USER_ID Deprecated POST /transfers.
mixinbot safetransfer USER_ID Alias for transfer (deprecated name).
mixinbot authcode OAuth authorize code (-c, -s).
mixinbot encrypt PIN / verifypin / updatetip PIN/TIP helpers.
mixinbot saferegister Safe registration (--spend_key).
mixinbot pay Safe payment URL.
mixinbot utils call METHOD Invoke any MixinBot.utils method (-d JSON kwargs).
mixinbot utils list [FILTER] List utils methods.
mixinbot unique UUID … Deterministic UUID.
mixinbot generatetrace HASH Trace UUID from tx hash.
mixinbot decodetx HEX Decode raw transaction.
mixinbot nftmemo NFT mint memo.
mixinbot rsa / ed25519 Key generation.
mixinbot version Gem version.

Examples:

mixinbot call me -k ~/.mixinbot/keystore.json
mixinbot call safe_outputs -k keystore.json -d '{"asset":"965e5c6e-434c-3fa9-b780-c50f43cd955c","state":"unspent"}'
mixinbot transfer USER_ID -k keystore.json --asset ASSET_ID --amount 0.01

Run mixinbot help and mixinbot help COMMAND for details.

For AI agents / LLMs

Run mixinbot schema -o json to discover CLI commands programmatically.

Documentation

Development & tests

git clone https://github.com/an-lee/mixin_bot.git
cd mixin_bot
bundle install
  • Default suite (offline WebMock stubs):
  rake test
  • API coverage check:
  rake mixin_bot:api_coverage
  • RuboCoprake runs tests + RuboCop.

  • Live API (optional):

  cp test/config.yml.example test/config.yml
  LIVE=1 rake test
  # or: rake test_live

Examples under examples/ expect examples/config.yml (copy from examples/config.yml.example).

CI

GitHub Actions runs on every pull request and on pushes to main:

  • Test — Ruby 3.2, 3.3, and 4.0: bundle exec rake test
  • RuboCop — Ruby 3.3: bundle exec rake rubocop
  • API coveragebundle exec rake mixin_bot:api_coverage

Release

Publishing to RubyGems.org is automated when a version tag is pushed:

  1. Bump MixinBot::VERSION in lib/mixin_bot/version.rb and update CHANGELOG.md.
  2. Commit and push to main.
  3. Create and push a tag matching the gem version (e.g. v2.0.1 for version 2.0.1):
   git tag v2.0.1
   git push origin v2.0.1

The Release workflow builds the gem and publishes to RubyGems.org via trusted publishing (GitHub OIDC; trusted publisher for workflow release.yml on an-lee/mixin_bot). To build without publishing, run the Release workflow manually with dry run enabled.

References

License

MIT — see MIT-LICENSE.