Skip to main content

Python bindings for eitype - type text on Wayland using the Emulated Input (EI) protocol

Project description

eitype

A library and CLI tool for typing text using the Emulated Input (EI) protocol on Wayland.

Features

  • Type text using the EI (Emulated Input) protocol
  • Support for special keys (enter, tab, escape, arrows, function keys, etc.)
  • Support for modifier keys (shift, ctrl, alt, super)
  • XDG RemoteDesktop portal support with session persistence
  • Direct socket connection support
  • Configurable delay between key events
  • Keyboard layout configuration via CLI or environment variables
  • Python bindings for use in Python applications
  • Rust library for integration into other Rust projects

Installation

Using pixi (recommended)

Pixi handles all dependencies (including libxkbcommon) automatically:

# Build and install Python bindings
pixi run build

# Install CLI
pixi run install-cli

# Run tests
pixi run test

Manual Installation

System Dependencies

If not using pixi, install the required system libraries first:

Distribution Command
Debian/Ubuntu sudo apt install libxkbcommon-dev
Fedora/RHEL sudo dnf install libxkbcommon-devel
Arch Linux sudo pacman -S libxkbcommon
openSUSE sudo zypper install libxkbcommon-devel

CLI

cargo install --path .

Python

pip install maturin
maturin develop --features python

Usage

# Type text (uses XDG RemoteDesktop portal by default)
eitype "Hello, World!"

# Type with delay between keys (10ms)
eitype -d 10 "Slow typing..."

# Press special keys
eitype -k return
eitype -k tab
eitype -k escape

# Hold modifier while typing
eitype -M ctrl c  # Ctrl+C

# Press and release a modifier
eitype -P shift

# Multiple texts
eitype "First line" -k return "Second line"

# Verbose output
eitype -v "Debug mode"
eitype -vv "More debug"

Connection Methods

XDG RemoteDesktop Portal (Default)

By default, eitype connects via the XDG RemoteDesktop portal. This works with desktop environments that support it (GNOME, KDE, etc.).

eitype "Hello"

Session Persistence

eitype automatically saves a session token to avoid the authorization dialog on subsequent runs. The token is stored at ~/.cache/eitype/restore_token.

  • First run: Shows the authorization dialog, saves token for future use
  • Subsequent runs: Uses saved token, no dialog needed
  • Token expiration: If the token becomes invalid, a new dialog will appear

To force a new authorization dialog (clear the saved token):

eitype --reset-token "Hello"

Direct Socket

Use the -s flag to specify a socket path, or set the LIBEI_SOCKET environment variable to bypass the portal:

eitype -s /path/to/ei/socket "Hello"
# or
export LIBEI_SOCKET=eis-0
eitype "Hello"

Special Keys

Supported special key names (case-insensitive):

  • escape, esc
  • return, enter
  • tab
  • backspace
  • delete
  • insert
  • home, end
  • pageup, pagedown
  • up, down, left, right
  • f1 through f12
  • space
  • capslock, numlock, scrolllock
  • print, printscreen
  • pause, menu

Modifier Keys

Supported modifier names (case-insensitive):

  • shift, lshift, rshift
  • ctrl, control, lctrl, rctrl
  • alt, lalt, ralt, altgr
  • super, meta, win, lsuper, rsuper

Keyboard Layout

eitype uses XKB for keyboard layout handling. The keymap is determined in the following order:

  1. EI server keymap - If the EI server provides a keymap, it is used automatically
  2. CLI/environment configuration - If no server keymap, uses specified layout
  3. System default - Falls back to the system's default XKB configuration

CLI Options

# Use German keyboard layout
eitype -l de "Hallo Welt"

# Use US Dvorak layout
eitype -l us --variant dvorak "Hello"

# Full XKB configuration
eitype -l us --variant dvorak --model pc104 --options "ctrl:nocaps" "Hello"

# Select a specific layout index when multiple layouts are available
eitype --layout-index 1 "Hello"

Multi-Layout Keymaps

When the EI server provides a keymap with multiple layouts (e.g., Dvorak + QWERTY), eitype uses layout index 0 by default. This is typically correct since layout 0 is the first/active layout.

If you need to use a different layout, specify it with --layout-index:

# Use the second layout (index 1)
eitype --layout-index 1 "Hello"

Use -vv to see all available layouts in the keymap.

Environment Variables

You can also set keyboard layout via environment variables (CLI options take precedence):

  • XKB_DEFAULT_LAYOUT - Keyboard layout (e.g., "us", "de", "fr")
  • XKB_DEFAULT_VARIANT - Layout variant (e.g., "dvorak", "colemak", "nodeadkeys")
  • XKB_DEFAULT_MODEL - Keyboard model (e.g., "pc104", "pc105")
  • XKB_DEFAULT_OPTIONS - XKB options (e.g., "ctrl:nocaps")
  • XKB_DEFAULT_RULES - XKB rules file
# Set German layout via environment
export XKB_DEFAULT_LAYOUT=de
eitype "Hallo"

# Override with CLI
XKB_DEFAULT_LAYOUT=de eitype -l fr "Bonjour"  # Uses French layout

Python Usage

from eitype import EiType, EiTypeConfig

# Simple connection via portal
typer = EiType.connect_portal()
typer.type_text("Hello from Python!")
typer.press_key("Return")

# With custom configuration
config = EiTypeConfig(layout="de", delay_ms=10)
typer = EiType.connect_portal(config)
typer.type_text("Hallo Welt!")

# Modifier keys
typer.hold_modifier("ctrl")
typer.press_key("c")
typer.release_modifiers()

Token Persistence (for long-running apps)

For applications that run continuously (like voice typing tools), you can save and reuse the portal authorization token:

from eitype import EiType

# First run - will show authorization dialog
typer, token = EiType.connect_portal_with_token()
if token:
    save_to_config(token)  # Save for next time

# Subsequent runs - no dialog needed
saved_token = load_from_config()
typer, _ = EiType.connect_portal_with_token(saved_token)
typer.type_text("No dialog this time!")

Thread Safety

EiType is Send but not Sync. You can move an instance between threads — for example, create it on one thread and call it from a thread pool — but you must not call its methods from two threads concurrently.

  • From Python, this is automatic on standard CPython: the GIL serializes every call. On a free-threaded (no-GIL) interpreter the GIL no longer protects you, so wrap the instance in your own lock if more than one thread can reach it.
  • From Rust, serialize access yourself (e.g. behind a Mutex) when sharing an instance across threads.

Rust Library Usage

Add to your Cargo.toml:

[dependencies]
eitype = { path = "../eitype" }  # or from crates.io when published
use eitype::{EiType, EiTypeConfig, EiTypeError};

fn main() -> Result<(), EiTypeError> {
    // Connect via portal
    let mut typer = EiType::connect_portal(EiTypeConfig::default())?;

    // Type text
    typer.type_text("Hello from Rust!")?;
    typer.press_key("Return")?;

    // With modifiers
    typer.hold_modifier("ctrl")?;
    typer.press_key("c")?;
    typer.release_modifiers()?;

    Ok(())
}

Development

# Set up dev environment with pre-commit hooks
pixi run -e dev install-hooks

# Run linting (cargo fmt + clippy)
pixi run -e dev lint

Requirements

  • Rust 1.70+
  • Python 3.10+ (only for Python bindings)
  • libxkbcommon (handled automatically by pixi, or install manually)

License

Apache 2.0

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distributions

No source distribution files available for this release.See tutorial on generating distribution archives.

Built Distributions

If you're not sure about the file name format, learn more about wheel file names.

eitype-0.2.2-cp314-cp314-manylinux_2_34_aarch64.whl (1.6 MB view details)

Uploaded CPython 3.14manylinux: glibc 2.34+ ARM64

eitype-0.2.2-cp314-cp314-manylinux_2_28_x86_64.whl (1.6 MB view details)

Uploaded CPython 3.14manylinux: glibc 2.28+ x86-64

eitype-0.2.2-cp313-cp313-manylinux_2_34_aarch64.whl (1.6 MB view details)

Uploaded CPython 3.13manylinux: glibc 2.34+ ARM64

eitype-0.2.2-cp313-cp313-manylinux_2_28_x86_64.whl (1.6 MB view details)

Uploaded CPython 3.13manylinux: glibc 2.28+ x86-64

eitype-0.2.2-cp312-cp312-manylinux_2_34_aarch64.whl (1.6 MB view details)

Uploaded CPython 3.12manylinux: glibc 2.34+ ARM64

eitype-0.2.2-cp312-cp312-manylinux_2_28_x86_64.whl (1.6 MB view details)

Uploaded CPython 3.12manylinux: glibc 2.28+ x86-64

eitype-0.2.2-cp311-cp311-manylinux_2_34_aarch64.whl (1.6 MB view details)

Uploaded CPython 3.11manylinux: glibc 2.34+ ARM64

eitype-0.2.2-cp311-cp311-manylinux_2_28_x86_64.whl (1.6 MB view details)

Uploaded CPython 3.11manylinux: glibc 2.28+ x86-64

eitype-0.2.2-cp310-cp310-manylinux_2_34_aarch64.whl (1.6 MB view details)

Uploaded CPython 3.10manylinux: glibc 2.34+ ARM64

eitype-0.2.2-cp310-cp310-manylinux_2_28_x86_64.whl (1.6 MB view details)

Uploaded CPython 3.10manylinux: glibc 2.28+ x86-64

File details

Details for the file eitype-0.2.2-cp314-cp314-manylinux_2_34_aarch64.whl.

File metadata

File hashes

Hashes for eitype-0.2.2-cp314-cp314-manylinux_2_34_aarch64.whl
Algorithm Hash digest
SHA256 5f8541e5ac8fff8b93df8c2abff0dd7a7ab9313db1f061a3028393cccaaafbf2
MD5 adb2b22fb24e4d0473c1269233d300ed
BLAKE2b-256 54ee29ffb60dbf40aca7bebb750c81469f0d1280771f8ac66cc96295ea5ebb70

See more details on using hashes here.

Provenance

The following attestation bundles were made for eitype-0.2.2-cp314-cp314-manylinux_2_34_aarch64.whl:

Publisher: release.yaml on Adam-D-Lewis/eitype

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file eitype-0.2.2-cp314-cp314-manylinux_2_28_x86_64.whl.

File metadata

File hashes

Hashes for eitype-0.2.2-cp314-cp314-manylinux_2_28_x86_64.whl
Algorithm Hash digest
SHA256 96afc0cde45845c9b4be3673fdcd9e0c9d10c789efcbf34abcbab29aad41c5f8
MD5 71cfd75367baeeda8a996486ad2adab5
BLAKE2b-256 e22dd1d33a83ccd6b00eeb6ed529e68b49073a06614d9b3662c8bcf00e5643bb

See more details on using hashes here.

Provenance

The following attestation bundles were made for eitype-0.2.2-cp314-cp314-manylinux_2_28_x86_64.whl:

Publisher: release.yaml on Adam-D-Lewis/eitype

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file eitype-0.2.2-cp313-cp313-manylinux_2_34_aarch64.whl.

File metadata

File hashes

Hashes for eitype-0.2.2-cp313-cp313-manylinux_2_34_aarch64.whl
Algorithm Hash digest
SHA256 37663e8f7d252d61a53fb78a035889dd035655d86d8bb447086f6c50f506f967
MD5 99bc96be3a74eb6c935e1b3b5cb3e389
BLAKE2b-256 e868bcc1729b3c8a97efe1a1d8aca5299f60e3860316b8ca1543bbe3429efde8

See more details on using hashes here.

Provenance

The following attestation bundles were made for eitype-0.2.2-cp313-cp313-manylinux_2_34_aarch64.whl:

Publisher: release.yaml on Adam-D-Lewis/eitype

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file eitype-0.2.2-cp313-cp313-manylinux_2_28_x86_64.whl.

File metadata

File hashes

Hashes for eitype-0.2.2-cp313-cp313-manylinux_2_28_x86_64.whl
Algorithm Hash digest
SHA256 0eff70098e3da4beaa838524831bd8732dc71eb3845b3188ad33ff902d3d4fa9
MD5 1c0c55920f930ae9d9c3d8a2135f2360
BLAKE2b-256 8736dc4cf29be131882edfaf68d8d573c6249711d169b038f14196e659379503

See more details on using hashes here.

Provenance

The following attestation bundles were made for eitype-0.2.2-cp313-cp313-manylinux_2_28_x86_64.whl:

Publisher: release.yaml on Adam-D-Lewis/eitype

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file eitype-0.2.2-cp312-cp312-manylinux_2_34_aarch64.whl.

File metadata

File hashes

Hashes for eitype-0.2.2-cp312-cp312-manylinux_2_34_aarch64.whl
Algorithm Hash digest
SHA256 2aacacb3f17dda524c151c2bbeb76909d083b69d4de044fa8c7b9b7c216fae4a
MD5 d00c5907baa3e39083ce705c41011587
BLAKE2b-256 2291291f596c8a65250c59e4f8f2413f4b71b336977d84d28583af787d767283

See more details on using hashes here.

Provenance

The following attestation bundles were made for eitype-0.2.2-cp312-cp312-manylinux_2_34_aarch64.whl:

Publisher: release.yaml on Adam-D-Lewis/eitype

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file eitype-0.2.2-cp312-cp312-manylinux_2_28_x86_64.whl.

File metadata

File hashes

Hashes for eitype-0.2.2-cp312-cp312-manylinux_2_28_x86_64.whl
Algorithm Hash digest
SHA256 ae8831a2624bc41b3b2273aa3b2f769a2e27204a11a8642470a9ee2dbfb37992
MD5 038af611397b7f2dad45737d88dcb1fa
BLAKE2b-256 747e587bc0e072f339197cd6f8395e722d76601fd0cf80803aa5904ad9e78c17

See more details on using hashes here.

Provenance

The following attestation bundles were made for eitype-0.2.2-cp312-cp312-manylinux_2_28_x86_64.whl:

Publisher: release.yaml on Adam-D-Lewis/eitype

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file eitype-0.2.2-cp311-cp311-manylinux_2_34_aarch64.whl.

File metadata

File hashes

Hashes for eitype-0.2.2-cp311-cp311-manylinux_2_34_aarch64.whl
Algorithm Hash digest
SHA256 ca1b5de8538dabf93a432c8c4cba6dd3f2acf1242636bd40a3a6b4fd92903f33
MD5 dacc9b0476a8d1aefbf19e8108db329a
BLAKE2b-256 10abb9e91057a14c9bb735e8c73deeda45a9e997b0285d9b6e175355938c5dab

See more details on using hashes here.

Provenance

The following attestation bundles were made for eitype-0.2.2-cp311-cp311-manylinux_2_34_aarch64.whl:

Publisher: release.yaml on Adam-D-Lewis/eitype

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file eitype-0.2.2-cp311-cp311-manylinux_2_28_x86_64.whl.

File metadata

File hashes

Hashes for eitype-0.2.2-cp311-cp311-manylinux_2_28_x86_64.whl
Algorithm Hash digest
SHA256 cf4411ab41f2d008c520ac5e31a979b88fc84a34d7f88cb4a8d0b0296e75cb0b
MD5 0b35949035255af992ff3eb2558d80e3
BLAKE2b-256 74fb550ad092b18673aebb4fb5676533a6b029d7119327fef49865e811410d48

See more details on using hashes here.

Provenance

The following attestation bundles were made for eitype-0.2.2-cp311-cp311-manylinux_2_28_x86_64.whl:

Publisher: release.yaml on Adam-D-Lewis/eitype

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file eitype-0.2.2-cp310-cp310-manylinux_2_34_aarch64.whl.

File metadata

File hashes

Hashes for eitype-0.2.2-cp310-cp310-manylinux_2_34_aarch64.whl
Algorithm Hash digest
SHA256 670383e47d6929e583cec37e3433f001e9b6211e2d4b5474db531555f2a333c3
MD5 fb71b7cc7cf505df81ebd686109a55b0
BLAKE2b-256 6b22de7d2dd8a26edab2ec60b4ed89e4aabf879e042695bc00e5fdd66670ccf2

See more details on using hashes here.

Provenance

The following attestation bundles were made for eitype-0.2.2-cp310-cp310-manylinux_2_34_aarch64.whl:

Publisher: release.yaml on Adam-D-Lewis/eitype

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file eitype-0.2.2-cp310-cp310-manylinux_2_28_x86_64.whl.

File metadata

File hashes

Hashes for eitype-0.2.2-cp310-cp310-manylinux_2_28_x86_64.whl
Algorithm Hash digest
SHA256 326788c348a40df6338cbd8dfa2fcb7c4dedb5e63239e4f75fc3dfe8aa5ed6fc
MD5 5e3e69a26aca6b1e6713db2a952cd097
BLAKE2b-256 c9637db2eb5eec8d9251c7936197ad52bfc73d3a84cf6a6514f2333d575dbd68

See more details on using hashes here.

Provenance

The following attestation bundles were made for eitype-0.2.2-cp310-cp310-manylinux_2_28_x86_64.whl:

Publisher: release.yaml on Adam-D-Lewis/eitype

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page