ble-device-control

v1.0.4

Control BLE devices via airctl CLI. MUST invoke when user mentions Bluetooth, BLE, device scan, connect, read, write, or notify. ALWAYS read this skill BEFOR...

0· 135·0 current·0 all-time
byskinapi@skinapi2025

Install

OpenClaw Prompt Flow

Install with OpenClaw

Best for remote or guided setup. Copy the exact prompt, then paste it into OpenClaw for skinapi2025/ble-device-control.

Previewing Install & Setup.
Prompt PreviewInstall & Setup
Install the skill "ble-device-control" (skinapi2025/ble-device-control) from ClawHub.
Skill page: https://clawhub.ai/skinapi2025/ble-device-control
Keep the work scoped to this skill only.
After install, inspect the skill metadata and help me finish setup.
Use only the metadata you can verify from ClawHub; do not invent missing requirements.
Ask before making any broader environment changes.

Command Line

CLI Commands

Use the direct CLI path if you want to install manually and keep every step visible.

OpenClaw CLI

Bare skill slug

openclaw skills install ble-device-control

ClawHub CLI

Package manager switcher

npx clawhub@latest install ble-device-control
Security Scan
VirusTotalVirusTotal
Benign
View report →
OpenClawOpenClaw
Benign
high confidence
Purpose & Capability
The skill is an instruction-only wrapper around the airctl CLI for BLE operations. It does not request unrelated credentials, config paths, or binaries. Asking the user to have airctl installed is appropriate for the described functionality.
Instruction Scope
SKILL.md narrowly instructs the agent to validate user inputs, run airctl commands (scan, connect, read, write, subscribe, disconnect), and parse JSON output. It does not instruct reading unrelated host files or exfiltrating data. Input validation rules are explicit and limit shell-injection risks.
Install Mechanism
There is no automated install spec in the registry (instruction-only), but SKILL.md directs the user to a pip install of a GitHub repo (git+https://github.com/skinapi2025/AirCtl.git). Installing directly from a GitHub repo is a common but higher-risk pattern than installing a vetted package from an official index — the skill correctly tells the agent to ask the user before installing.
Credentials
The skill requests no environment variables or credentials. That is proportionate for a local CLI-based BLE control skill.
Persistence & Privilege
The skill is instruction-only, not always-enabled, and does not request persistent presence or modify other skills or system configurations. Autonomous invocation is allowed (platform default) but not combined with other privilege escalations here.
Assessment
This skill appears to do what it says: it issues airctl commands to your local Bluetooth adapter. Before using it, (1) confirm you are comfortable granting the agent the ability to run airctl CLI commands on your machine (those commands access local BLE hardware and devices), (2) verify the referenced GitHub repository (https://github.com/skinapi2025/AirCtl) yourself before running pip install from it — preferably install in a virtualenv or container and inspect the code/release — and (3) ensure your system-level Bluetooth permissions and any device pairing consent are acceptable. The skill's input validation reduces injection risk, but always review any generated shell commands before executing them.

Like a lobster shell, security has layers — review code before you run it.

latestvk9787t9dpqd41dr1gkt13rwcdd84w2d7
135downloads
0stars
5versions
Updated 1w ago
v1.0.4
MIT-0

BLE Device Control Skill

IMPORTANT: Read this document before executing any airctl command.

Guidelines:

  1. Consult the Command Reference below first. All common airctl commands and their syntax are documented here.
  2. Follow the Decision Tree to determine which workflow to execute. Do not skip steps.
  3. If a command is not found in this document, verify it with airctl --help before using it.
  4. Check prerequisites first before any BLE operation.
  5. Validate all user-provided inputs before passing them to airctl commands. See Input Validation below.

Input Validation

All user-provided inputs must be validated before being used in airctl commands to prevent shell injection. Apply these validation rules:

Device Address — Must match MAC address format: XX:XX:XX:XX:XX:XX where each XX is a hexadecimal pair.

  • Valid regex: ^[0-9A-Fa-f]{2}(:[0-9A-Fa-f]{2}){5}$
  • Example valid: AA:BB:CC:DD:EE:FF
  • If invalid: reject the input and ask the user to provide a valid address.

Device Alias — Must contain only alphanumeric characters and underscores. No hyphens, spaces, or special characters.

  • Valid regex: ^[A-Za-z0-9_]+$
  • Example valid: my_sensor, heart_rate
  • If invalid: replace hyphens with underscores, remove spaces and special characters, then confirm with the user.

UUID — Must match standard BLE UUID format: 4-digit short or 8-4-4-4-12 full format.

  • Valid regex: ^[0-9A-Fa-f]{4}$ or ^[0-9A-Fa-f]{8}-[0-9A-Fa-f]{4}-[0-9A-Fa-f]{4}-[0-9A-Fa-f]{4}-[0-9A-Fa-f]{12}$
  • Example valid: 2A19, 00002A19-0000-1000-8000-00805F9B34FB
  • Both short and full UUID formats are supported in all commands.
  • If invalid: reject the input and ask the user to provide a valid UUID.

Handle — Must be a positive integer.

  • Valid regex: ^[1-9][0-9]*$
  • Example valid: 10, 74
  • If invalid: reject the input.

Write Data — Must use one of the documented data format prefixes with only allowed characters.

  • hex: prefix: followed by hexadecimal characters only (0-9A-Fa-f). Valid regex: ^hex:[0-9A-Fa-f]+$
  • text: prefix: followed by printable ASCII characters. Valid regex: ^text:[\x20-\x7E]+$
  • base64: prefix: followed by Base64 characters. Valid regex: ^base64:[A-Za-z0-9+/=]+$
  • No prefix: treated as hex, same rules as hex:.
  • If invalid: reject the input and ask the user to provide properly formatted data.

Device Name (for scan filter) — Must contain only printable characters. No shell metacharacters.

  • Disallowed characters: ` $ | ; & < > ( ) { } [ ] \ ! #
  • If disallowed characters found: reject the input and ask the user to provide a clean device name.

Prerequisites

This skill requires the airctl CLI tool.

Provenance:

FieldValue
Source Codehttps://github.com/skinapi2025/AirCtl
LicenseMIT
Authorskinapi2025

Verify airctl is installed:

airctl --version

If airctl is not installed, ask the user to install it before proceeding. Provide the following installation command for the user to review and execute:

pip install git+https://github.com/skinapi2025/AirCtl.git

Note: Do NOT automatically run pip install without user confirmation. The user should verify the package source and approve the installation.

After installation, verify the package origin:

pip show airctl

Confirm that the output shows Home-page: https://github.com/skinapi2025/AirCtl or Author: skinapi2025.

Decision Tree

Follow this decision tree to determine which workflow to execute. Start from the top and follow the FIRST matching path:

User mentions Bluetooth/BLE device operation?
│
├─ YES → What does the user want?
│   │
│   ├─ "Connect and keep connection" / "maintain connection"
│   │   → Follow Workflow 4 (Connect + Keep Alive)
│   │
│   ├─ "Connect" (without keep-alive)
│   │   → Follow Workflow 1 (Connect by Name)
│   │
│   ├─ "Read" a characteristic value
│   │   → Follow Workflow 2 (Read Characteristic)
│   │
│   ├─ "Write" to a characteristic
│   │   → Follow Workflow 3 (Write Characteristic)
│   │
│   ├─ "Scan" / "Find" / "Discover" devices
│   │   → Run: `airctl ble scan -t 10`
│   │
│   ├─ "Subscribe" / "Notify" / "Monitor"
│   │   → Follow Workflow 4 Step 3 (Subscribe)
│   │
│   ├─ "Disconnect"
│   │   → Run: `airctl ble disconnect <address>`
│   │
│   ├─ "Battery" / "Heart rate" / "Device info"
│   │   → Use Device Profile commands: `airctl device battery/heart-rate/device-info <address>`
│   │
│   └─ Other BLE operation
│       → Look up the command in the Command Reference section below
│
└─ NO → This skill may not be relevant

Workflow 1: Connect by Name

When user says: "connect to Heart Rate device"

Execute these steps IN ORDER:

Step 1: Scan for the device:

airctl ble scan -t 10 -n "Heart Rate"

Step 2: Parse the JSON output to extract the device address field. Validate the address format (see Input Validation).

Step 3: Connect using the validated address:

airctl ble connect <address> -a <alias>

IMPORTANT: Use underscores (_) in aliases, never hyphens (-). Validate alias format before use.

Workflow 2: Read Characteristic

When user says: "read Body Sensor Location from Heart Rate device"

Execute these steps IN ORDER:

Step 1: Check if device is already connected:

airctl ble list

Step 2: If NOT connected, scan and connect:

airctl ble scan -t 10 -n "Heart Rate"
airctl ble connect <address> -a heart_rate

Step 3: List characteristics to find the UUID:

airctl ble characteristics heart_rate

Step 4: Parse the JSON output to find the target characteristic UUID. Validate the UUID format (see Input Validation).

Step 5: Read the characteristic using the validated UUID:

airctl ble read heart_rate -u <uuid> -f hex

Workflow 3: Write Characteristic

When user says: "write 0x01 to Alert Level"

Execute these steps IN ORDER:

Step 1: Check if device is already connected:

airctl ble list

Step 2: If NOT connected, scan and connect first (see Workflow 1)

Step 3: Validate the UUID and data format (see Input Validation), then write:

airctl ble write <address> -u <uuid> -d "<data>"

Workflow 4: Connect + Keep Alive

When user says: "connect to Heart Rate and keep connection" or "connect and maintain connection"

Execute these steps IN ORDER. Do not skip any step:

Step 1: Scan and Connect

airctl ble scan -t 10 -n "Heart Rate"

Parse JSON to get address, validate it, then:

airctl ble connect <address> -a heart_rate

Step 2: Get Characteristics

airctl ble characteristics heart_rate

Step 3: Analyze and Choose Keep-Alive Strategy

Parse the characteristics JSON. Each characteristic has a properties array. Follow this priority:

Priority 1 — If ANY characteristic has "notify" or "indicate" in properties:

airctl ble notify subscribe heart_rate -u <notifiable_uuid>

This is the best keep-alive method. The device pushes data and keeps the connection active.

Priority 2 — If NO notifiable characteristic, find a read-only one (has "read" but NO "write"):

airctl ble task start-read heart_rate -u <readable_uuid> -i 5

Read-only characteristics are safe for periodic reads without side effects.

Priority 3 — If only read-write characteristics exist, pick one known to be safe:

airctl ble task start-read heart_rate -u <readable_uuid> -i 5

Use caution — reading may have side effects on some devices.

Workflow 5: LLM Snapshot Polling

When an AI agent needs to monitor BLE events without streaming:

Step 1: Subscribe to notifications (returns immediately):

airctl ble notify subscribe <address> -u <uuid>

Step 2: Poll for recent events using snapshot query (returns immediately):

airctl ble events --last 5

Step 3: Repeat step 2 as needed to check for new events.

Alternative: Use periodic read tasks with result snapshots:

airctl ble task start-read <address> -u 2A19 -i 5
# Returns: {"task_id": "abc123", "type": "periodic_read"}

airctl ble task result abc123 --last 5
# Returns: {"task_id": "abc123", "results": [...], "count": 5}

Command Reference

All common airctl commands are documented below. If you need a command not listed here, verify it with airctl --help or airctl ble --help.

Daemon

airctl daemon status
airctl daemon start
airctl daemon stop
airctl daemon restart

The daemon starts automatically for BLE operations. Use these only for troubleshooting.

BLE Scan

airctl ble scan -t 10
airctl ble scan -n "Heart Rate"
airctl ble scan --service-uuids 180D,180F

Scan output (JSON):

{"devices": [{"address": "AA:BB:CC:DD:EE:FF", "name": "Heart Rate", "rssi": -45, "service_uuids": ["180D"]}], "count": 1}

BLE Connect / Disconnect / List

airctl ble connect <address> [-t 30] [-a alias]
airctl ble disconnect <address>
airctl ble list

BLE GATT Exploration

airctl ble services <address>
airctl ble characteristics <address>
airctl ble characteristics <address> -s <service_uuid>

Characteristics output (JSON):

{"characteristics": [{"uuid": "2A37", "handle": 10, "properties": ["notify", "read"], "service_uuid": "180D"}]}

BLE Read

airctl ble read <address> -u <uuid> [-f hex|base64|text]
airctl ble read <address> -H <handle> [-f hex|base64|text]

Both short UUID (2A19) and full UUID (00002a19-0000-1000-8000-00805f9b34fb) are supported.

BLE Write

airctl ble write <address> -u <uuid> -d "hex:010203"
airctl ble write <address> -H <handle> -d "hex:010203"
airctl ble write <address> -u <uuid> -d "hex:01" --response

Data format prefixes: hex: (e.g., hex:010203), text: (e.g., text:hello), base64: (e.g., base64:AQID), or no prefix (treated as hex).

BLE Notifications

airctl ble notify subscribe <address> -u <uuid>
airctl ble notify unsubscribe <address> -u <uuid>

BLE Event Snapshots (LLM-friendly)

airctl ble events --last 5                        # Last 5 events (returns immediately)
airctl ble events --last 10 --address <addr>      # Filter by device address
airctl ble events --last 5 --type notification     # Filter by event type
airctl ble events --last 5 -h                     # Human-readable output

Event types: device_connected, device_disconnected, notification, periodic_read, periodic_write, periodic_scan, task_error

Important: Without --last, airctl ble events streams continuously and never returns. Always use --last N for LLM/agent usage.

BLE Background Tasks

airctl ble task start-read <address> -u <uuid> -i 5
airctl ble task start-read <address> -H <handle> -i 5
airctl ble task start-write <address> -u <uuid> -d "hex:01" -i 10
airctl ble task start-scan -i 30 -t 5
airctl ble task list
airctl ble task stop <task_id>
airctl ble task result <task_id> --last 5         # Query task results (snapshot)

Device Profiles

airctl device battery <address>              # Battery percentage (0x180F)
airctl device device-info <address>           # Device info (0x180A)
airctl device heart-rate <address>            # Heart rate (0x180D)
airctl device heart-rate <address> --subscribe
airctl device heart-rate <address> --duration 60

Configuration

airctl config list
airctl config alias set <address> <name>
airctl config alias list
airctl config alias remove <name>
airctl config preset list
airctl config preset get uart
airctl config preset set <name> --service <uuid> --char <uuid>
airctl config preset remove <name>

Common GATT UUIDs

UUIDNameDescription
180DHeart RateHeart Rate service
180FBatteryBattery service
180ADevice InformationDevice info service
2A37Heart Rate MeasurementHeart rate data (notify)
2A38Body Sensor LocationSensor position (read)
2A19Battery LevelBattery percentage (read)
2A06Alert LevelAlert control (write)
2A29Manufacturer NameDevice manufacturer (read)

Error Handling

ErrorCauseSolution
"Insufficient Authentication"Device requires pairingPair the device with your system first
"Characteristic not found"Wrong UUIDRun airctl ble characteristics to verify, or use -H handle
"Device not connected"Not connected yetRun airctl ble connect first
"Daemon not running"Daemon stoppedRun airctl daemon start or let it auto-start
"Command not found"airctl not installed or outdatedVerify with airctl --version, reinstall if needed
"Device disconnected unexpectedly"BLE device dropped connectionReconnect with airctl ble connect, use periodic read task to keep alive

Platform Requirements

  • Windows: Windows 10 v16299+, Bluetooth adapter
  • Linux: BlueZ 5.55+, Bluetooth adapter
  • macOS: macOS 10.15+, Bluetooth adapter

Comments

Loading comments...