This article walks you through how to build a simple Bitcoin tracker using Pixlet. It looks pretty great on a Tidbyt display, although we're admittedly biased.
To get closer to a truly useful Pixlet app, we'll be pulling in some Bitcoin data. CoinDesk's Bitcoin Price Index API is free to use and requires no authentication. We'll use Starlib's http module to retrieve the data.
Pixlet includes several modules from the Starlib library. This is sort of a standard library for Starlark, and it's very handy when building anything but the simplest app.
load("render.star", "render")
load("http.star", "http")
COINDESK_PRICE_URL = "https://api.coindesk.com/v1/bpi/currentprice.json"
def main():
rep = http.get(COINDESK_PRICE_URL)
if rep.status_code != 200:
fail("Coindesk request failed with status %d", rep.status_code)
rate = rep.json()["bpi"]["USD"]["rate_float"]
return render.Root(
child = render.Text("BTC: %d USD" % rate)
)
If the HTTP request doesn't return the expected status code (200), we
call fail()
to halt execution of the script. Starlark provides a few
other helpful built-ins that you can read about in the starlark-go
language
definition. Another
built-in worth mentioning is print()
, which of course is invaluable
in debugging your Pixlet scripts.
VoilĂ . A perfectly functional Bitcoin price tracker.
To make our app a bit snazzier, we headed over to Pixilart and drew this simple Bitcoin icon:
Pixlet allows us to embed graphics in our scripts through the Image
widget. We'll use Starlib's encoding/base64
module to embed the
image in our source code. Finally, to display both the price string
and the icon simultanesouly, we'll use the Row
Widget to have them
laid out side by side.
load("render.star", "render")
load("http.star", "http")
load("encoding/base64.star", "base64")
COINDESK_PRICE_URL = "https://api.coindesk.com/v1/bpi/currentprice.json"
# Load Bitcoin icon from base64 encoded data
BTC_ICON = base64.decode("""
iVBORw0KGgoAAAANSUhEUgAAABEAAAARCAYAAAA7bUf6AAAAlklEQVQ4T2NkwAH+H2T/jy7FaP+
TEZtyDEG4Zi0TTPXXzoDF0A1DMQRsADbN6MZdO4NiENwQbAbERh1lWLzMmgFGo5iFZBDYEFwuwG
sISCPUIKyGgDRjAyBXYXMNIz5XgDQga8TpLboYgux8DO/AwoUuLiEqTLBFMcmxQ7V0gssgklIsL
AYozjsoBoE45OZi5DRBSnkCAMLhlPBiQGHlAAAAAElFTkSuQmCC
""")
def main():
rep = http.get(COINDESK_PRICE_URL)
if rep.status_code != 200:
fail("CoinDesk request failed with status %d", rep.status_code)
rate = rep.json()["bpi"]["USD"]["rate_float"]
return render.Root(
child = render.Row( # Row lays out its children horizontally
children = [
render.Image(src=BTC_ICON),
render.Text("$%d" % rate),
],
)
)
This clearly leaves something to be desired as far as layout is concerned, but the individual elements (the icon and the price) aren't too shabby!
By default, Row
will pack its children as closely together as it
possibly can. That's often a useful behaviour, but in this case
perhaps not so much. We can instruct Row
to use as much horizontal
space as possible by passing expanded=True
, and then use the
main_align
and cross_align
parameters to adjust how the children
are spaced out. For details on these parameters, check out the full
Widget reference.
We'll also place the Row
itself in a Box
to ensure that it's
placed in the vertical center of the screen. Box
actually centers
horizontally as well, but since our Row
is expanded
that won't
matter.
load("render.star", "render")
load("http.star", "http")
load("encoding/base64.star", "base64")
COINDESK_PRICE_URL = "https://api.coindesk.com/v1/bpi/currentprice.json"
BTC_ICON = base64.decode("""
iVBORw0KGgoAAAANSUhEUgAAABEAAAARCAYAAAA7bUf6AAAAlklEQVQ4T2NkwAH+H2T/jy7FaP+
TEZtyDEG4Zi0TTPXXzoDF0A1DMQRsADbN6MZdO4NiENwQbAbERh1lWLzMmgFGo5iFZBDYEFwuwG
sISCPUIKyGgDRjAyBXYXMNIz5XgDQga8TpLboYgux8DO/AwoUuLiEqTLBFMcmxQ7V0gssgklIsL
AYozjsoBoE45OZi5DRBSnkCAMLhlPBiQGHlAAAAAElFTkSuQmCC
""")
def main():
rep = http.get(COINDESK_PRICE_URL)
if rep.status_code != 200:
fail("Coindesk request failed with status %d", rep.status_code)
rate = rep.json()["bpi"]["USD"]["rate_float"]
return render.Root(
child = render.Box( # This Box exists to provide vertical centering
render.Row(
expanded=True, # Use as much horizontal space as possible
main_align="space_evenly", # Controls horizontal alignment
cross_align="center", # Controls vertical alignment
children = [
render.Image(src=BTC_ICON),
render.Text("$%d" % rate),
],
),
),
)
Dont' worry if all this alignment stuff feels a bit confusing at first. It'll be a lot clearer when you've had a chance to play around with it.
Now that's a Bitcoin tracker.
Finally, let's make sure we're not spamming CoinDesk with more
requests than what's absolutely necessary. The simplest way to do this
is to rely on the HTTP module's built-in cache functionality, by adding
ttl_seconds
to the http.get()
call. This instructs the module to
cache the response for a given period of time.
A special header will be set on the response to indicate if it was served from cache or straight from the API. While we're using it in this example to illustrate the cache behavior, it's not something developers normally need to care about.
load("render.star", "render")
load("http.star", "http")
load("encoding/base64.star", "base64")
COINDESK_PRICE_URL = "https://api.coindesk.com/v1/bpi/currentprice.json"
BTC_ICON = base64.decode("""
iVBORw0KGgoAAAANSUhEUgAAABEAAAARCAYAAAA7bUf6AAAAlklEQVQ4T2NkwAH+H2T/jy7FaP+
TEZtyDEG4Zi0TTPXXzoDF0A1DMQRsADbN6MZdO4NiENwQbAbERh1lWLzMmgFGo5iFZBDYEFwuwG
sISCPUIKyGgDRjAyBXYXMNIz5XgDQga8TpLboYgux8DO/AwoUuLiEqTLBFMcmxQ7V0gssgklIsL
AYozjsoBoE45OZi5DRBSnkCAMLhlPBiQGHlAAAAAElFTkSuQmCC
""")
def main():
rep = http.get(COINDESK_PRICE_URL, ttl_seconds = 240) # cache for 4 minutes
if rep.status_code != 200:
fail("Coindesk request failed with status %d", rep.status_code)
rate = rep.json()["bpi"]["USD"]["rate_float"]
# for development purposes: check if result was served from cache or not
if rep.headers.get("Tidbyt-Cache-Status") == "HIT":
print("Hit! Displaying cached data.")
else:
print("Miss! Calling CoinDesk API.")
return render.Root(
child = render.Box(
render.Row(
expanded=True,
main_align="space_evenly",
cross_align="center",
children = [
render.Image(src=BTC_ICON),
render.Text("$%d" % rate),
],
),
),
)
Pixlet's in-memory cache only really kicks in when running pixlet in
its "server mode" (via pixlet serve <script>
). The resulting WebP
images are of course identical to what we saw before, but as we
repeately reload our browser (pointed at http://localhost:8080/
), we
see the following output from the print()
statements:
$ ./pixlet serve tutorial.star listening on tcp/8080 [tutorial.star] Miss! Calling CoinDesk API. [tutorial.star] Hit! Displaying cached data. [tutorial.star] Hit! Displaying cached data.
That's the cache working as intended. We're passing ttl_seconds=240
to http.get()
, so if we were to wait 4 minutes and then reload,
we'd see a cache miss as the old record has expired.
Take a look at the guide on authoring apps, the Widget reference, and start hacking!