Flirror is a modular smartmirror application developed in Python. It consists of a simple webserver based on Flask which holds the UI and a command line application that retrieves information from different APIs (e.g. OpenWeather, Google Calendar, ...). Currently, there is only a small set of available modules, but I'm planning to add some more (see planned features and ideas).
Despite the number of commits, this project is still in an early phase, so I'm happy about any contribution!
Contents: Motivation | Architecture | Usage | Available Modules | Developing Custom Modules | Deploy on Raspberry | Development | Planned features and ideas |
I mainly started this project because I wanted to build a smartmirror using a Raspberry Pi. I searched for existing projects and found a few, but none of them really suited my needs. There are a few smaller ones developed in Python, but most of them don't seem to be actively maintaned anymore. A very public one I found is for sure MagicMirror2. I think, this is a really awesome project - taking a look at all the customizable parts, contributions and custom modules that were developed in the meantime. I'm just not much of a JavaScript developer and don't like the idea of having everything in a single application.
But I got inspired by the features it provides and decided to develop my own smartmirror in Python. Maybe someone else in the Python community is also interested in a project like this.
Flirror mainly consists of two components - a webserver based on Flask and a command line application, the so called "crawler" that retrieves data from different APIs/backends (whatever you want to call it).
The webserver is mainly used to show the information that is retrieved by the crawlers. Currently, it doesn't allow any "interaction" from a user side. But maybe something like this will come in the future.
The crawler application can simply be invoked from the command line and is used
to crawl the various backends / APIs for the actual data that is displayed by
flirror-web. The crawler application supports two different modes: periodic
and immediate
(default). The periodic mode will crawl all available APIs in
customizable intervals, while the immediate mode can be used as a one-shot
command to retrieve the data directly from all available backends.
To bring both components together I decided for a very simple approach, using a local SQLite databse. I mainly made this choice, because we are only storing simple key value pairs and SQLite comes out of the box with python.
Flirror can simply be installed via pip:
$ pip install flirror
or via docker:
$ docker pull felixedel/flirror
Deploying flirror via docker is simplest using docker-compose. An example
docker-compose stack can be found in the deploy/docker-compose.example.yaml
file.
Both applications - flirror-web and flirror-crawler - read their
configuration from the file path given via the FLIRROR_SETTINGS
environment
variable, e.g.
$ export FLIRROR_SETTINGS=$(pwd)/settings.cfg
A basic configuration file must at least contain the path to a DATABASE_FILE
and a list of MODULES
with at least one module configured.
Each entry in the MODULES
list accepts the following parameters:
Parameter | Description |
---|---|
id |
Required The ID to identify this module inside the application. |
module |
Required The name of the module to use for this tile. A list of available modules can be found here |
config |
Required The configuration for the specific module. Some modules come up with a default configuration, but usually this is needed to for each module. For more details on how to configure the specific module, take a look at the module's configuration part in the modules section. |
crawler |
Crawler specific settings. This can be used to speficy e.g.the crawling interval for a specific module. For more details see the crawler config section. |
display |
Configure display properties of the module. This accepts a dictionary with the following keys: position and time . The position can be used to specify in which order the modules are displayed in the flirror UI. All modules with will sorted by their position in ascending order. Modules without a position definition will be placed after the positioned ones.The time specifies the reloading time in milliseconds with which the module will be reloaded via an ajax call. The default time value is 30000 . |
An example configuration with at least one module with the minimum required parameters might look like the following:
MODULES = [
{
"id": "weather-tile",
"module": "weather",
"config": {
"city": "My hometown",
"api_key": "<your-openweathermap-api-key>",
"language": "en",
"temp_unit": "celsius",
},
"display": {
"position": 0,
},
}
]
For more detailed configuration examples, please take a look at the
settings.example.cfg
file.
Each module entry defined in this configuration file will be shown as a single tile in flirror-web and will be crawled independently in flirror-crawler.
To start flirror-web, simply run the following command:
$ flirror-web
which will start a gunicorn server serving the flirror
application on http://127.0.0.1
. The script accepts arbitrary parameters, so
you could further configure the gunicorn command that is executed in the end, by
e.g. specifying the number workers or changing the address. For a list of
available command line arguments, please refer to gunicorn's documentation.
If you don't want to use gunicorn, you could take a look at Flask's uWSGI guide.
To start the crawler simply run one of the following commands
# Periodic mode
$ flirror-crawler crawl --periodic
# Immediate mode
$ flirror-crawler crawl
to run the crawler either in periodic or immediate mode. In both cases flirror will look up all modules specified in the configuration file and try to retrieve the data for each one by invoking the respective crawler.
Modules provide the base functionality that is used by Flirror to show e.g. a clock or the current weather. Every module defines a view (which is visible in flirror-web) and a crawler that retrieves the actual data from an API or backend.
Some modules may come without a crawler (like the clock module) but usually it's recommended to do any data retrieving / calculation in the crawler and use the view only to show the data.
The following modules are available in flirror by default:
- Clock
- Weather
- Calendar
- News
- Stocks
The clock module displays a clock either in digital or analog format. This is a pure JavaScript/CSS module, as it wouldn't make much sense to use a Python backend to retrieve the current time.
Option | Description |
---|---|
mode |
Must be one of analog or digital to display the clock in the selected format. Default: digital |
The weather module displays the current weather information together with a forecast for the next six days. The weather information is retrieved from OpenWeather, so an OpenWeather API key is necessary to access those information. Information on how to get a free API key can be found in their How to start section. Flirror is using the One Call API.
Option | Description |
---|---|
api_key |
Required Your personal OpenWeather API key |
city |
The city to retrieve the weather information for. Note: In case the lat and lon values are provided, the city will only be used to display the weather information. If no lat/lon values are provided, flirror will try to look them up from OpenWeather for the given city. Please note that this does not work for all cities. If you face any issues with that, please specify the lat and lon parameters directly. |
lat |
The latitude value of the position to retrieve the weather information for. |
lon |
The longitude value of the position to retrieve the weather information for. |
language |
The language in which the results are returned from the API (and thus displayd in flirror). For a list of available language codes, please refer to the OpenWeather multilingual support. Default: en |
temp_unit |
The unit in which the results are returned from the API (and thus displayed in flirror). Possible values are fahrenheit , celsius and kelvin . Default: celsius |
The calendar modules displays upcoming events from a Google calendar. Currently, Flirror only supports the OAuth 2.0 for TV and Limited-Input Device Applications.
Option | Description |
---|---|
calendars |
Required A list of google calendars to retrieve the events from. If you don't want to mix up multiple calendars in one tile, you can configure multiple calendar modules with one calendar each. Your default google calendar is usually named after your gmail address. |
max_items |
The maximum number of events to show. Default: 5 |
The stocks module displays current stock values either in table format or as a time series. The information is retrieved from Alpha Vantage, so an Alpha Vantage API keys is necessary to access those information. Information on how to get a free API key can be found in their Getting started guide.
Option | Description |
---|---|
api_key |
Required Your personal Alpha Vantage API key |
symbols |
Required The list of equities you want to retrieve. Each element must be in the format ("<symbol>", "<display_name>") |
mode |
One of table or series to display the stocks information in the selected format. Default: table |
The news module displays entries from a RSS feed. Flirror uses the feedparser package to crawl the newsfeeds. Please take a look at feedparser's documentation to get an overview about available formats which can be parsed.
Option | Description |
---|---|
name |
Required The title to display over the news entries |
url |
Required The url pointing to the RSS feed |
Flirror provides a plugin mechanism using an extended version of Flask Blueprints.
The so called FlirrorModule
provides some decorators and functions to register
the necessary view and crawler for a module. Apart from that you could still
utilize the whole Blueprint functionality to provide e.g. custom templates,
filters and more.
A simple module may consist of the following file structure:
flirror_awesome_module
|-- __init__.py
|-- templates/
|-- awesome_module/
|-- index.html
The __init__.py
file contains the module's python code including the module
definition itself.
import time
from flask import current_app
from flirror.modules import FlirrorModule
awesome_module = FlirrorModule(
"awesome_module", __name__, template_folder="templates"
)
@awesome_module.view()
def get():
return current_app.basic_get(template_name="awesome_module/index.html")
@awesome_module.crawler()
def crawl(module_id, app, user_name):
awesome_data = {
"_timestamp": time.time(),
"message": f"Hello, '{user_name}",
}
app.store_module_data(crawler_id, awesome_data)
FLIRROR_MODULE = awesome_module
A few notes on what's going on here:
First, we create a new FlirrorModule()
instance which contains all the
necessary parameters of our custom module like its name
, import_path
and the
template_folder
. The latter one is necessary to make our custom template
usable in Flirror.
Once the module is defined, we can use the @awesome_module.view()
decorator to
register the module's view function in Flirror. Using this decorator will
register a new route /awesome_module/
on the underlying Flask application.
Flirror-web will then request this route while providing the module_id
as GET
parameter. The helper function basic_get()
will evaluate this GET parameter,
look up the data which is stored in the database for this module_id
and
populate the data to the template provided via the template_name
parameter.
Finally, it returns the rendered template so that flirror-web can integrate it
in its UI.
To store the data in the database, we provide a crawler function decorated with
@awesome_module.crawler()
. This registers the function as crawler for this
module in flirror. When invoking flirror-crawler
this function will be
called with a set of predefined parameters:
- The
module_id
for which the function is called - The
app
(which is mainly used as a back-reference to get access to the database) - All config values that the module provides. In our case we want to greet a
user whereby the
user_name
is configurable. Usually there is no need to store theuser_name
in the database as we could also directly access it in the view. It's just used like this to show the typical use case of view and crawler.
Finally, we expose our module as FLIRROR_MODULE
so that it can be detected by
Flirror.
The templates/awesome_module/index.html
file contains the view's HTML code in
form of a Jinja2 template. It
might look redundant that the module's name is specified again in the path to
the template. That's necessary to avoid overriding templates of other modules.
More information on this can be found in the
Templates
section of Flask's Blueprint documentation.
{% extends "module.html" %}
{% block body %}
<div class="card-body">
<div class="text-right">
<small>
<i id="{{ module.id }}-spinner" class="fas fa-sync-alt"></i> {{ module.data._timestamp | prettydate }}
</small>
</div>
<h2>{{ module.data.message }}</h2>
</div>
{% endblock %}
To seamlessly include the module's view in the flirror-web UI, make sure to
extend the module.html
template and use <div class="card-body">
as outer
element in the body block.
Flirror will try to detect installed plugins automatically if they follow a predefined naming schema. For each plugin found, Flirror will look up the provided flirror modules and register them on the app.
To make your plugin discoverable by flirror, it must fulfil the following requirements:
-
The name of python package providing the custom module (or modules) must start with
flirror_
(e.g.flirror_awesome_module
). -
The package must expose the modules via one of the following top-level variables:
# To expose a single module, use FLIRROR_MODULE = <my_awesome_module> # If the plugin provides multiple modules, expose them via FLIRROR_MODULES = [<my_awesome_module_1>, <my_awesome_module_2>]
-
Each module must be a valid FlirrorModule instance.
Flirror's standard modules are defined in the same manner like custom modules, thus you could take a closer look on their source if you are interested in how you could develop a custom module.
Although Flirror could simply be installed via pip
, the recommended way to
install it on a Raspberry Pi is via Docker. The main reason for this is that not
all python dependencies are packaged for ARM and thus must be built from
sources. This takes a lot of time (up to 60 minutes) especially for those using
C extensions.
Flirror's Docker image already comes with all dependencies installed and you can directly start Flirror after pulling the image.
To install docker on raspbian OS, you can simply run the following command:
$ curl -sSL https://get.docker.com | sh
This will download the installation script and directly execute it via shell. Running the script may take some time. Afterwards, you might want to add your user (pi) to the docker group, so you can run docker without sudo:
$ sudo usermod -aG docker pi
Afterwards log out and back or reboot the Raspberry Pi via
$ sudo reboot -h
There are various ways to install docker-compose. Please see the docker-compose installation guide for more detailed information.
I personally installed docker-compose via
pipx.
Using this variant requires the python-dev
and libffi-dev
packages to be
installed on the system.
$ sudo apt install python-dev libffi-dev
$ python3 -m pip install --user pipx
$ python3 -m pipx ensurepath
$ pipx install docker-compose
Both componenents can be started using the docker-compose file provided in
deploy/docker-compose.example.yaml
. Just copy this file and name it
docker-compose.yaml
. If necessary, adapt (or remove) some of the volume mounts
to your needs. Afterwards, you can run
$ docker-compose up
to start the web server and the crawler in periodic mode.
With both services running we still need to open some browser to see the actual flirror UI. Therefore, you could download and execute the following helper script like so:
$ wget https://raw.githubusercontent.com/felixedel/flirror/master/helpers/start_gui.sh
$ chmod u+x start_gui.sh
$ ./start_gui.sh
Apart from starting chromium in full screen mode pointing to the running
flirror-web instance inside the docker container, this script will also ensure
that some necessary environment variables like DISPLAY
are set and that the
screen saver and energy saving mode of the X server are disabled - so the
display doesn't go into sleep mode after a few minutes.
To hide the mouse cursor, install unclutter via
$ sudo apt install unclutter
and add the following line to /home/pi/.config/lxsession/LXDE-pi/autostart
@unclutter -display :0 -noevents -grab
To use the Flirror docker image on a Raspberry Pi, it must be built for the ARM
architecture. To not always utilize a Raspberry Pi itself to build the docker
image for ARM, we could use docker's buildx
command and run a
multi-architecture build on Linux or Mac.
When using Docker for Mac, the
buildx
command should already be available. You just have to enable the
"experimental features" in the "Command Line" section of the application's
settings.
For Linux, you could use the Getting started with Docker for Arm on Linux guide to install buildx.
To test if the docker buildx command is available, simply run
$ docker buildx --help
To show the available platforms for which buildx can be utilized, run
$ docker buildx ls
Docker for Mac already comes with a few preinstalled platforms including
linux/arm/v7
(which we are going to use).
For Linux, you first need to register the ARM executables via qemu. Information on how to achieve this can also be found in the guide mentioned above.
Once everything is set, we can create a new builder instance which we will use for our multi-archtitecture build:
$ docker buildx create --name flirrorbuilder
$ docker buildx use flirrorbuilder
$ docker buildx inspect --bootstrap
This will download the necessary buildkit
docker image and should show an
output similar to the following if successful:
[+] Building 12.1s (1/1) FINISHED
=> [internal] booting buildkit 12.1s
=> => pulling image moby/buildkit:buildx-stable-1 11.0s
=> => creating container buildx_buildkit_flirrorbuilder0 1.0s
Name: flirrorbuilder
Driver: docker-container
Nodes:
Name: flirrorbuilder0
Endpoint: unix:///var/run/docker.sock
Status: running
Platforms: linux/amd64, linux/arm64, linux/riscv64, linux/ppc64le, linux/s390x, linux/386, linux/arm/v7, linux/arm/v6
Now we can use this builder to build the Flirror multi-architecture image with the following command:
$ docker buildx build --platform linux/arm,linux/amd64 -t felixedel/flirror:latest -f dockerfiles/flirror-Dockerfile --push .
- A plugin mechamisn to allow custom modules to be included in flirror
- Provide webhooks to allow interacting with flirror from the outside (and maybe event between modules)
- Provide some notification mechanism