Skip to content

Commit c2e9158

Browse files
committed
Simplify approach
- OS installation, customized by nerab.raspi - User service "kiosk" controls Chromium via [ChromeDP](https://github.com/chromedp/chromedp)
1 parent 11615b0 commit c2e9158

23 files changed

+237
-196
lines changed

.envrc

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
# use this when using a forked role
2+
export ANSIBLE_ROLES_PATH=~/workspace/ansible-roles:~/.ansible/roles

.gitignore

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,2 +1,2 @@
11
*.retry
2-
wifi.yml
2+
kiosk

.vscode/settings.json

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
{
2+
"go.inferGopath": false
3+
}

README.markdown

+20-25
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,23 @@
11
# Kiosk
22

3-
Turns a Raspberry Pi into a simple browser kiosk
3+
Turns a Raspberry Pi into a simple browser kiosk. A Go program controls a full-screen Chromium browser.
4+
5+
Configuring the Pi is described in the `nerab.raspi` role. Make sure the Pi auto-boots into the graphical environment without asking for login credentials (see `raspi-config`).
6+
7+
# TODO
8+
9+
- `unclutter -idle 0.5 -root &` if needed
10+
- [splash screen at boot](https://github.com/guysoft/FullPageOS/blob/master/src/modules/fullpageos/filesystem/root_init/etc/systemd/system/splashscreen.service)
11+
12+
# Deployment
13+
14+
## Build the `kiosk` binary
15+
16+
For the Raspberry Pi 4, this is
17+
18+
```command
19+
$ GOARM=7 GOARCH=arm GOOS=linux go build
20+
```
421

522
## Prerequisites on the Deployer's Workstation
623

@@ -18,34 +35,12 @@ Turns a Raspberry Pi into a simple browser kiosk
1835
$ ansible-galaxy install -r roles.yml
1936
```
2037

21-
* Update `hosts` with the IP address of the host(s) to deploy to. The deploy is more convenient if you can ssh into them without having to enter a password.
22-
23-
## Prerequisites on the Raspi
24-
25-
* Configure some generic settings using `sudo raspi-config`:
26-
- Change password of user `pi`
27-
- Set the hostname (`kiosk`)
28-
- Enable SSH server
29-
- Expand root filesystem
30-
* Connect via Ethernet
31-
* Copy the public SSH key to the pi with `ssh-copy-id pi@kiosk`.
38+
* Update `inventory.yml` with the host(s) to deploy to
3239

3340
## Deploy
3441

3542
Run the playbook:
3643

3744
```bash
38-
$ ansible-playbook -i hosts site.yml
45+
$ ansible-playbook playbook.yml
3946
```
40-
41-
## WiFi
42-
43-
If you want to configure the WiFi network, create a file `wifi.yml` with the following contents (adapt it to your WIFI settings):
44-
45-
```yaml
46-
wlan_country: DE
47-
wlan_ssid: your-wlan-ssid
48-
wlan_password: your-wlan-password
49-
```
50-
51-
If `wifi.yml` is not present, WiFi will not be configured.

ansible.cfg

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
[defaults]
2+
inventory = inventory.yml
3+
deprecation_warnings = false
4+
stdout_callback = yaml
5+
bin_ansible_callbacks = true

go.mod

+16
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
module uhlig.it/kiosk
2+
3+
go 1.19
4+
5+
require github.com/chromedp/chromedp v0.8.4
6+
7+
require (
8+
github.com/chromedp/cdproto v0.0.0-20220812200530-d0d83820bffc // indirect
9+
github.com/chromedp/sysutil v1.0.0 // indirect
10+
github.com/gobwas/httphead v0.1.0 // indirect
11+
github.com/gobwas/pool v0.2.1 // indirect
12+
github.com/gobwas/ws v1.1.0 // indirect
13+
github.com/josharian/intern v1.0.0 // indirect
14+
github.com/mailru/easyjson v0.7.7 // indirect
15+
golang.org/x/sys v0.0.0-20220811171246-fbc7d0a398ab // indirect
16+
)

go.sum

+21
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
github.com/chromedp/cdproto v0.0.0-20220812200530-d0d83820bffc h1:xGhCpiX5oNMoZGnzYvv1ne4muVRl2SDHH5fL7oUbZAY=
2+
github.com/chromedp/cdproto v0.0.0-20220812200530-d0d83820bffc/go.mod h1:5Y4sD/eXpwrChIuxhSr/G20n9CdbCmoerOHnuAf0Zr0=
3+
github.com/chromedp/chromedp v0.8.4 h1:50NSLCXC38kGsCmLCi8tXZ5V5FtQ3BOV/90u1O06Gq4=
4+
github.com/chromedp/chromedp v0.8.4/go.mod h1:ikuiJrEMoOnTPFUBu6jV0XZFOGW6W/Nhgk/OZyvh00o=
5+
github.com/chromedp/sysutil v1.0.0 h1:+ZxhTpfpZlmchB58ih/LBHX52ky7w2VhQVKQMucy3Ic=
6+
github.com/chromedp/sysutil v1.0.0/go.mod h1:kgWmDdq8fTzXYcKIBqIYvRRTnYb9aNS9moAV0xufSww=
7+
github.com/gobwas/httphead v0.1.0 h1:exrUm0f4YX0L7EBwZHuCF4GDp8aJfVeBrlLQrs6NqWU=
8+
github.com/gobwas/httphead v0.1.0/go.mod h1:O/RXo79gxV8G+RqlR/otEwx4Q36zl9rqC5u12GKvMCM=
9+
github.com/gobwas/pool v0.2.1 h1:xfeeEhW7pwmX8nuLVlqbzVc7udMDrwetjEv+TZIz1og=
10+
github.com/gobwas/pool v0.2.1/go.mod h1:q8bcK0KcYlCgd9e7WYLm9LpyS+YeLd8JVDW6WezmKEw=
11+
github.com/gobwas/ws v1.1.0 h1:7RFti/xnNkMJnrK7D1yQ/iCIB5OrrY/54/H930kIbHA=
12+
github.com/gobwas/ws v1.1.0/go.mod h1:nzvNcVha5eUziGrbxFCo6qFIojQHjJV5cLYIbezhfL0=
13+
github.com/josharian/intern v1.0.0 h1:vlS4z54oSdjm0bgjRigI+G1HpF+tI+9rE5LLzOg8HmY=
14+
github.com/josharian/intern v1.0.0/go.mod h1:5DoeVV0s6jJacbCEi61lwdGj/aVlrQvzHFFd8Hwg//Y=
15+
github.com/ledongthuc/pdf v0.0.0-20220302134840-0c2507a12d80 h1:6Yzfa6GP0rIo/kULo2bwGEkFvCePZ3qHDDTC3/J9Swo=
16+
github.com/mailru/easyjson v0.7.7 h1:UGYAvKxe3sBsEDzO8ZeWOSlIQfWFlxbzLZe7hwFURr0=
17+
github.com/mailru/easyjson v0.7.7/go.mod h1:xzfreul335JAWq5oZzymOObrkdz5UnU4kGfJJLY9Nlc=
18+
github.com/orisano/pixelmatch v0.0.0-20220722002657-fb0b55479cde h1:x0TT0RDC7UhAVbbWWBzr41ElhJx5tXPWkIHA2HWPRuw=
19+
golang.org/x/sys v0.0.0-20201207223542-d4d67f95c62d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
20+
golang.org/x/sys v0.0.0-20220811171246-fbc7d0a398ab h1:2QkjZIsXupsJbJIdSjjUOgWK3aEtzyuh2mPt3l/CkeU=
21+
golang.org/x/sys v0.0.0-20220811171246-fbc7d0a398ab/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=

group_vars/all/secrets.yml

+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
$ANSIBLE_VAULT;1.1;AES256
2+
62646361663036373266303863333733363039393735646531333630303465613435396130663932
3+
6161393832376366616435656339313933396136373063310a306339303764396138333832303164
4+
61663761333933616666626463653163373331653136346135383336393062323262613233353837
5+
3065616461633365650a323865633561623262343130663533323466636333303465366561353033
6+
34666636633962323963356139333237363430383435623634613639383234323839376535363564
7+
34313865313162303231663539663439653034383230336536663235363033393139666331373233
8+
633432383434386533343335373938663564

hosts

-2
This file was deleted.

inventory.yml

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
all:
2+
hosts:
3+
kiosk.local:

main.go

+100
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,100 @@
1+
// based on https://pkg.go.dev/github.com/chromedp/chromedp#example-ExecAllocator
2+
package main
3+
4+
/*
5+
Disabling Chromium's password manager seems not possible in preferences, only by policy:
6+
7+
cat /etc/chromium/policies/managed/no-password-management.json
8+
{
9+
"AutofillAddressEnabled": false,
10+
"AutofillCreditCardEnabled": false,
11+
"PasswordManagerEnabled": false
12+
}
13+
14+
old: "AutoFillEnabled": false,
15+
16+
// from https://stackoverflow.com/a/55316111/3212907
17+
*/
18+
19+
import (
20+
"context"
21+
"fmt"
22+
"log"
23+
"os"
24+
"os/signal"
25+
"syscall"
26+
27+
"github.com/chromedp/chromedp"
28+
)
29+
30+
func main() {
31+
opts := append(chromedp.DefaultExecAllocatorOptions[:],
32+
chromedp.Flag("headless", false),
33+
chromedp.Flag("hide-scrollbars", true),
34+
chromedp.Flag("enable-automation", false),
35+
chromedp.Flag("start-fullscreen", true),
36+
chromedp.Flag("kiosk", true),
37+
chromedp.Flag("noerrdialogs", true),
38+
chromedp.Flag("disable-session-crashed-bubble", true),
39+
chromedp.Flag("simulate-outdated-no-au", "Tue, 31 Dec 2099 23:59:59 GMT"),
40+
chromedp.Flag("disable-component-update", true),
41+
chromedp.Flag("disable-translate", true),
42+
chromedp.Flag("disable-infobars", true),
43+
chromedp.Flag("disable-features", "Translate"),
44+
chromedp.Flag("disk-cache-dir", "/dev/null"),
45+
)
46+
47+
allocCtx, cancelAllocator := chromedp.NewExecAllocator(context.Background(), opts...)
48+
49+
// also set up a custom logger
50+
taskCtx, cancelContext := chromedp.NewContext(allocCtx, chromedp.WithLogf(log.Printf))
51+
52+
user := os.Getenv("KIOSK_USER")
53+
54+
if user == "" {
55+
fmt.Fprintln(os.Stderr, "Environment variable KIOSK_USER must not be empty")
56+
os.Exit(1)
57+
}
58+
59+
password := os.Getenv("KIOSK_PASSWORD")
60+
61+
if password == "" {
62+
fmt.Fprintln(os.Stderr, "Environment variable KIOSK_PASSWORD must not be empty")
63+
os.Exit(1)
64+
}
65+
66+
actions := []chromedp.Action{
67+
chromedp.Navigate(`https://grafana.uhlig.it/login`),
68+
typeText(`//input[@name="user"]`, user),
69+
typeText(`//input[@name="password"]`, password),
70+
chromedp.Click(`//button[@type='submit']`, chromedp.NodeVisible),
71+
chromedp.WaitVisible(`//a[@href='/profile']`),
72+
chromedp.Navigate(`https://grafana.uhlig.it/d/yP_VJJmVz/power?refresh=30s&from=now-1h&to=now&kiosk`),
73+
}
74+
75+
err := chromedp.Run(taskCtx, actions...)
76+
77+
if err != nil {
78+
log.Fatal(err)
79+
}
80+
81+
var quit = make(chan struct{})
82+
83+
c := make(chan os.Signal)
84+
signal.Notify(c, os.Interrupt, syscall.SIGTERM)
85+
go func() {
86+
<-c
87+
cancelAllocator()
88+
cancelContext()
89+
close(quit)
90+
}()
91+
92+
<-quit
93+
}
94+
95+
func typeText(selector, value string) chromedp.Tasks {
96+
return chromedp.Tasks{
97+
chromedp.WaitVisible(selector),
98+
chromedp.SendKeys(selector, value),
99+
}
100+
}

playbook.yml

+40
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,40 @@
1+
---
2+
- name: Kiosk
3+
hosts: all
4+
5+
roles:
6+
- role: nerab.raspi
7+
become: true
8+
tags: [ kiosk, nerab ]
9+
- role: kiosk
10+
become: true
11+
tags: [ kiosk, software ]
12+
- role: suhlig.simple_systemd_service
13+
become: true
14+
vars:
15+
service_binary: kiosk
16+
program:
17+
name: kiosk
18+
description: Display kiosk
19+
runtime_user: "{{ ansible_user_id }}"
20+
binary: "{{ service_binary }}"
21+
environment:
22+
- KIOSK_USER="{{ kiosk.user }}"
23+
- KIOSK_PASSWORD="{{ kiosk.password }}"
24+
- DISPLAY=:0.0
25+
tags: [ kiosk, systemd, service ]
26+
- role: geerlingguy.ntp
27+
vars:
28+
ntp_timezone: Europe/Berlin
29+
ntp_manage_config: true
30+
ntp_servers:
31+
- ptbtime1.ptb.de iburst
32+
- ptbtime2.ptb.de iburst
33+
- ptbtime3.ptb.de iburst
34+
become: true
35+
- role: jnv.unattended-upgrades
36+
vars:
37+
unattended_origins_patterns:
38+
- 'origin=Raspbian,codename=${distro_codename},label=Raspbian'
39+
unattended_automatic_reboot_time: "05:46"
40+
become: true

requirements.yml

+6
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
- src: nerab.raspi
2+
- src: geerlingguy.ntp
3+
- src: jnv.unattended-upgrades
4+
- name: suhlig.simple_systemd_service
5+
src: https://github.com/uhlig-it/ansible-role-simple-systemd-service
6+
version: master

roles.yml

-2
This file was deleted.

roles/common/files/keyboard

-10
This file was deleted.

roles/common/handlers/main.yml

-5
This file was deleted.

roles/common/tasks/main.yml

-50
This file was deleted.

roles/kiosk/files/rc.local

-2
This file was deleted.

0 commit comments

Comments
 (0)