Skip to content

Control USB connected presence lights from multiple vendors via the command-line or web API.

License

Notifications You must be signed in to change notification settings

JnyJny/busylight

Repository files navigation

version monthly-downloads supported python versions license Code style: black
pytest-linux MacOS pytest-windows

BusyLight Project Logo

BusyLight for Humans™ gives you control of USB attached LED lights from a variety of vendors. Lights can be controlled via the command-line, using a HTTP API or imported directly into your own Python project.

All Supported Lights

Eight Lights Attached to One Host
Back to Front, Left to Right
Busylight Alpha, Plantronics Status Indicator, BlyncLight, BlyncLight Plus
BlinkStick, Busylight Omega, Blink(1), Flag

Features

  • Control lights from the command-line.
  • Control lights via a Web API.
  • Import busylight into your own Python project.
  • Supports six vendors & multiple models:
Agile Innovative Embrava Plantronics Kuando Luxafor ThingM
BlinkStick Blynclight Status Indicator Busylight Alpha Flag Blink(1)
BlinkStick Pro Blynclight Mini Busylight Omega Mute
BlinkStick Square Blynclight Plus Orb
BlinkStick Strip
BlinkStick Nano
BlinkStick Flex
  • Supported on MacOS and Linux
  • Windows support will be available in the near future.

If you have a USB light that's not on this list open an issue with:

  • the make and model device you want supported
  • where I can get one
  • any public hardware documentation you are aware of

Or even better, open a pull request!

Gratitude

Thank you to @todbot and the very nice people at ThingM who graciously and unexpectedly gifted me with two blink(1) mk3 lights!

Basic Install

Installs only the command-line busylight tool and associated modules.

$ python3 -m pip install busylight-for-humans 

Web API Install

Installs uvicorn and FastAPI in addition to busylight:

$ python3 -m pip install busylight-for-humans[webapi]

Linux Post-Install Activities

Linux controls access to USB devices via the udev subsystem. By default it denies non-root users access to devices it doesn't recognize. I've got you covered!

You'll need root access to configure the udev rules:

$ busylight udev-rules -o 99-busylights.rules
$ sudo cp 99-busylights.rules /etc/udev/rules.d
$ sudo udevadm control -R
$ # unplug/plug your light
$ busylight on

Command-Line Examples

$ busylight on               # light turns on green
$ busylight on red           # now it's shining a friendly red
$ busylight on 0xff0000      # still red
$ busylight on #00ff00       # now it's blue!
$ busylight blink            # it's slowly blinking on and off with a red color
$ busylight blink green fast # blinking faster green and off
$ busylight --all on         # turn all lights on green
$ busylight --all off        # turn all lights off

HTTP API Examples

First start the busylight API server using the busyserv command line interface:

$ busyserve -D
INFO:     Started server process [20189]
INFO:     Waiting for application startup.
INFO:     Application startup complete.
INFO:     Uvicorn running on http://0.0.0.0:8888 (Press CTRL+C to quit)

The API is fully documented and available @ https://localhost:8888/redoc

Now you can use the web API endpoints which return JSON payloads:

  $ curl -s http://localhost:8888/lights/status | jq
  ...
  $ curl -s http://localhost:8888/light/0/status | jq
  ...
  $ curl -s http://localhost:8888/light/0/on | jq
  {
    "light_id": 0,
    "action": "on",
    "color": "green",
	"rgb": [0, 128, 0]
  }
  $ curl -s http://localhost:8888/light/0/off | jq
  {
    "light_id": 0,
    "action": "off"
  }
  $ curl -s http://localhost:8888/light/0/on?color=purple | jq
  {
    "light_id": 0,
    "action": "on",
    "color": "purple",
	"rgb": [128, 0, 128]
  }
  $ curl -s http://localhost:8888/lights/on | jq
  {
    "light_id": "all",
    "action": "on",
    "color": "green",
	"rgb", [0, 128, 0]
  }
  $ curl -s http://localhost:8888/lights/off | jq
  {
    "light_id": "all",
    "action": "off"
  }
  $ curl -s http://localhost:8888/lights/rainbow | jq
  {
    "light_id": "all",
    "action": "effect",
    "name": "rainbow"
  }

Authentication

The API can be secured with a simple username and password through HTTP Basic Authentication. To require authentication for all API requests, set the BUSYLIGHT_API_USER and BUSYLIGHT_API_PASS environmental variables before running busyserve.

⚠️ SECURITY WARNING: HTTP Basic Auth sends usernames and passwords in cleartext (i.e., unencrypted). Use of SSL is highly recommended!

Code Examples

Adding light support to your own python applications is easy!

Simple Case: Turn On a Single Light

In this example, we pick an Embrava Blynclight to activate with the color white.

from busylight.lights.embrava import Blynclight

light = Blynclight.first_light()

light.on((255, 255, 255))

Not sure what light you've got?

from busylight.lights import USBLight

light = USBLight.first_light()

light.on((0xff, 0, 0xff))

Slightly More Complicated

The busylight package includes a manager class that's great for working with multiple lights or lights that require a little more direct intervention like the Kuando Busylight family.

from busylight.manager import LightManager
from busylight.effects import Effects

manager = LightManager()
for light in manager.lights:
   print(light.name)
   
rainbow = Effects.for_name("spectrum")(duty_cycle=0.05)

manager.apply_effect(rainbow)
...
manager.off()