Created
February 26, 2025 07:55
-
-
Save schappim/638e42171bacdeb3b7c472d44d5bd607 to your computer and use it in GitHub Desktop.
Turn your Raspberry Pi into a TRMNL Display
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
package main | |
import ( | |
"encoding/json" | |
"fmt" | |
"image" | |
"image/color" | |
"image/draw" | |
_ "image/jpeg" // Register JPEG decoder | |
_ "image/png" // Register PNG decoder | |
_ "golang.org/x/image/bmp" // Register BMP decoder | |
"io" | |
"net/http" | |
"os" | |
"os/exec" | |
"path/filepath" | |
"time" | |
imagedraw "golang.org/x/image/draw" | |
"github.com/gonutz/framebuffer" | |
) | |
// TerminalResponse represents the JSON structure returned by the API | |
type TerminalResponse struct { | |
ImageURL string `json:"image_url"` | |
Filename string `json:"filename"` | |
RefreshRate int `json:"refresh_rate"` | |
} | |
// Config holds application configuration | |
type Config struct { | |
APIKey string | |
} | |
func main() { | |
// Check the environment first | |
fmt.Println("Checking system environment...") | |
checkDisplayServer() | |
listFramebufferDevices() | |
// Create a configuration directory | |
configDir, err := os.UserHomeDir() | |
if err != nil { | |
fmt.Printf("Error getting home directory: %v\n", err) | |
os.Exit(1) | |
} | |
configDir = filepath.Join(configDir, ".trmnl") | |
err = os.MkdirAll(configDir, 0755) | |
if err != nil { | |
fmt.Printf("Error creating config directory: %v\n", err) | |
os.Exit(1) | |
} | |
// Get API key from environment, or from config file | |
config := loadConfig(configDir) | |
if config.APIKey == "" { | |
config.APIKey = os.Getenv("TRMNL_API_KEY") | |
} | |
// If the API key is still not set, prompt the user | |
if config.APIKey == "" { | |
fmt.Println("TRMNL API Key not found.") | |
fmt.Print("Please enter your TRMNL API Key: ") | |
fmt.Scanln(&config.APIKey) | |
saveConfig(configDir, config) | |
} | |
// Create a temporary directory for storing images | |
tmpDir, err := os.MkdirTemp("", "trmnl-display") | |
if err != nil { | |
fmt.Printf("Error creating temp directory: %v\n", err) | |
os.Exit(1) | |
} | |
defer os.RemoveAll(tmpDir) | |
for { | |
processNextImage(tmpDir, config.APIKey) | |
} | |
} | |
func processNextImage(tmpDir, apiKey string) { | |
// Use defer and recover to handle any panics | |
defer func() { | |
if r := recover(); r != nil { | |
fmt.Printf("Recovered from panic: %v\n", r) | |
time.Sleep(60 * time.Second) | |
} | |
}() | |
// Get the TRMNL display | |
req, err := http.NewRequest("GET", "https://usetrmnl.com/api/display", nil) | |
if err != nil { | |
fmt.Printf("Error creating request: %v\n", err) | |
time.Sleep(60 * time.Second) | |
return | |
} | |
req.Header.Add("access-token", apiKey) | |
client := &http.Client{ | |
Timeout: 30 * time.Second, | |
} | |
resp, err := client.Do(req) | |
if err != nil { | |
fmt.Printf("Error fetching display: %v\n", err) | |
time.Sleep(60 * time.Second) | |
return | |
} | |
defer resp.Body.Close() | |
if resp.StatusCode != 200 { | |
fmt.Printf("Error fetching display: status code %d\n", resp.StatusCode) | |
time.Sleep(60 * time.Second) | |
return | |
} | |
// Parse the JSON response | |
var terminal TerminalResponse | |
decoder := json.NewDecoder(resp.Body) | |
if err := decoder.Decode(&terminal); err != nil { | |
fmt.Printf("Error parsing JSON: %v\n", err) | |
time.Sleep(60 * time.Second) | |
return | |
} | |
// Set default filename if not provided | |
filename := terminal.Filename | |
if filename == "" { | |
filename = "display.jpg" | |
} | |
// Create full path to temporary file | |
filePath := filepath.Join(tmpDir, filename) | |
// Download the image | |
imgResp, err := http.Get(terminal.ImageURL) | |
if err != nil { | |
fmt.Printf("Error downloading image: %v\n", err) | |
time.Sleep(60 * time.Second) | |
return | |
} | |
defer imgResp.Body.Close() | |
// Create the file | |
out, err := os.Create(filePath) | |
if err != nil { | |
fmt.Printf("Error creating file: %v\n", err) | |
time.Sleep(60 * time.Second) | |
return | |
} | |
// Copy the image data to the file | |
_, err = io.Copy(out, imgResp.Body) | |
if err != nil { | |
fmt.Printf("Error saving image: %v\n", err) | |
out.Close() | |
time.Sleep(60 * time.Second) | |
return | |
} | |
out.Close() | |
// Display the image | |
err = displayImage(filePath) | |
if err != nil { | |
fmt.Printf("Error displaying image: %v\n", err) | |
time.Sleep(60 * time.Second) | |
return | |
} | |
// Set default refresh rate if not provided | |
refreshRate := terminal.RefreshRate | |
if refreshRate <= 0 { | |
refreshRate = 60 | |
} | |
// Sleep for the refresh rate | |
time.Sleep(time.Duration(refreshRate) * time.Second) | |
} | |
func displayImage(imagePath string) error { | |
// Open the image file | |
file, err := os.Open(imagePath) | |
if err != nil { | |
return fmt.Errorf("error opening image file: %v", err) | |
} | |
defer file.Close() | |
fmt.Printf("Reading image from %s\n", imagePath) | |
// Get image format | |
format, err := getImageFormat(file) | |
if err != nil { | |
return fmt.Errorf("error determining image format: %v", err) | |
} | |
fmt.Printf("Detected image format: %s\n", format) | |
// Reset file position after checking format | |
file.Seek(0, 0) | |
var img image.Image | |
// Try standard decoding first | |
img, format, err = image.Decode(file) | |
// If standard decoding fails for BMP, try our custom decoder | |
if err != nil && format == "bmp" { | |
fmt.Printf("Standard BMP decoder failed: %v\n", err) | |
fmt.Printf("Trying custom BMP decoder...\n") | |
file.Seek(0, 0) | |
img, err = decodeCustomBMP(file) | |
if err != nil { | |
return fmt.Errorf("both standard and custom BMP decoders failed: %v", err) | |
} | |
fmt.Printf("Successfully decoded image with custom BMP decoder\n") | |
} else if err != nil { | |
return fmt.Errorf("error decoding image format '%s': %v", format, err) | |
} else { | |
fmt.Printf("Successfully decoded image as %s\n", format) | |
} | |
// Switch to tty1 so the framebuffer becomes active | |
err = exec.Command("chvt", "1").Run() | |
if err != nil { | |
fmt.Printf("Error switching VT to tty1: %v\n", err) | |
} | |
// Open the framebuffer | |
fb, err := framebuffer.Open("/dev/fb0") | |
if err != nil { | |
return fmt.Errorf("error opening framebuffer: %v", err) | |
} | |
defer fb.Close() | |
// Get framebuffer bounds | |
fbBounds := fb.Bounds() | |
fmt.Printf("Framebuffer bounds: %v\n", fbBounds) | |
// Scale the image to fill the entire framebuffer | |
targetRect := fbBounds | |
scaledImg := image.NewRGBA(targetRect) | |
imagedraw.NearestNeighbor.Scale(scaledImg, targetRect, img, img.Bounds(), imagedraw.Over, nil) | |
// Draw the scaled image to the framebuffer | |
draw.Draw(fb, targetRect, scaledImg, image.Point{}, draw.Src) | |
// Flush the framebuffer if necessary | |
if fbFlusher, ok := interface{}(fb).(interface{ Flush() error }); ok { | |
fbFlusher.Flush() | |
} | |
fmt.Println("Image drawing completed (full screen)") | |
return nil | |
} | |
// decodeCustomBMP attempts to decode a BMP file using a simplified approach | |
// that can handle some BMP variants that the standard library cannot, including 1-bit BMPs. | |
func decodeCustomBMP(file *os.File) (image.Image, error) { | |
// Read the entire file | |
fileInfo, err := file.Stat() | |
if err != nil { | |
return nil, fmt.Errorf("error getting file info: %v", err) | |
} | |
fileSize := fileInfo.Size() | |
data := make([]byte, fileSize) | |
_, err = file.Read(data) | |
if err != nil { | |
return nil, fmt.Errorf("error reading file: %v", err) | |
} | |
// Check BMP signature | |
if data[0] != 'B' || data[1] != 'M' { | |
return nil, fmt.Errorf("invalid BMP signature") | |
} | |
// Parse header information | |
dataOffset := int(uint32(data[10]) | uint32(data[11])<<8 | uint32(data[12])<<16 | uint32(data[13])<<24) | |
headerSize := int(uint32(data[14]) | uint32(data[15])<<8 | uint32(data[16])<<16 | uint32(data[17])<<24) | |
width := int(int32(uint32(data[18]) | uint32(data[19])<<8 | uint32(data[20])<<16 | uint32(data[21])<<24)) | |
if width < 0 { | |
width = -width | |
} | |
height := int(int32(uint32(data[22]) | uint32(data[23])<<8 | uint32(data[24])<<16 | uint32(data[25])<<24)) | |
isBottomUp := true | |
if height < 0 { | |
height = -height | |
isBottomUp = false | |
} | |
bitsPerPixel := int(uint16(data[28]) | uint16(data[29])<<8) | |
var numColors int | |
if headerSize >= 36 && len(data) > 49 { | |
numColors = int(uint32(data[46]) | uint32(data[47])<<8 | uint32(data[48])<<16 | uint32(data[49])<<24) | |
} | |
if numColors == 0 && bitsPerPixel <= 8 { | |
numColors = 1 << uint(bitsPerPixel) | |
} | |
fmt.Printf("BMP Info: width=%d, height=%d, bitsPerPixel=%d, dataOffset=%d, headerSize=%d, numColors=%d\n", | |
width, height, bitsPerPixel, dataOffset, headerSize, numColors) | |
// Create a new RGBA image | |
img := image.NewRGBA(image.Rect(0, 0, width, height)) | |
// Calculate row padding (BMP rows are aligned to 4 bytes) | |
rowSize := ((width * bitsPerPixel + 31) / 32) * 4 | |
// For 1-bit (and other indexed) BMPs, read the colour palette | |
var palette []color.RGBA | |
if bitsPerPixel == 1 || bitsPerPixel == 4 || bitsPerPixel == 8 { | |
paletteOffset := 14 + headerSize | |
palette = make([]color.RGBA, numColors) | |
for i := 0; i < numColors && paletteOffset+i*4+2 < len(data); i++ { | |
b := data[paletteOffset+i*4] | |
g := data[paletteOffset+i*4+1] | |
r := data[paletteOffset+i*4+2] | |
palette[i] = color.RGBA{r, g, b, 255} | |
} | |
if len(palette) < 2 { | |
// Default palette for 1-bit BMP: black and white | |
palette = []color.RGBA{ | |
{0, 0, 0, 255}, | |
{255, 255, 255, 255}, | |
} | |
} | |
fmt.Printf("Palette: %v\n", palette) | |
} | |
// Read pixel data | |
for y := 0; y < height; y++ { | |
srcY := y | |
if isBottomUp { | |
srcY = height - 1 - y | |
} | |
for x := 0; x < width; x++ { | |
var col color.RGBA | |
switch bitsPerPixel { | |
case 24, 32: | |
pos := dataOffset + srcY*rowSize + x*bitsPerPixel/8 | |
if pos+3 > len(data) { | |
continue | |
} | |
b := data[pos] | |
g := data[pos+1] | |
r := data[pos+2] | |
a := uint8(255) | |
if bitsPerPixel == 32 && pos+3 < len(data) { | |
a = data[pos+3] | |
} | |
col = color.RGBA{r, g, b, a} | |
case 16: | |
pos := dataOffset + srcY*rowSize + x*2 | |
if pos+1 >= len(data) { | |
continue | |
} | |
value := uint16(data[pos]) | uint16(data[pos+1])<<8 | |
r := uint8((value >> 11) & 0x1F) << 3 | |
g := uint8((value >> 5) & 0x3F) << 2 | |
b := uint8(value & 0x1F) << 3 | |
col = color.RGBA{r, g, b, 255} | |
case 8: | |
pos := dataOffset + srcY*rowSize + x | |
if pos >= len(data) { | |
continue | |
} | |
index := data[pos] | |
if int(index) < len(palette) { | |
col = palette[index] | |
} else { | |
col = color.RGBA{0, 0, 0, 255} | |
} | |
case 4: | |
pos := dataOffset + srcY*rowSize + x/2 | |
if pos >= len(data) { | |
continue | |
} | |
var index uint8 | |
if x%2 == 0 { | |
index = (data[pos] >> 4) & 0x0F | |
} else { | |
index = data[pos] & 0x0F | |
} | |
if int(index) < len(palette) { | |
col = palette[index] | |
} else { | |
col = color.RGBA{0, 0, 0, 255} | |
} | |
case 1: | |
bytePos := dataOffset + srcY*rowSize + x/8 | |
bitPos := 7 - (x % 8) | |
if bytePos >= len(data) { | |
continue | |
} | |
bit := (data[bytePos] >> bitPos) & 1 | |
if int(bit) < len(palette) { | |
col = color.RGBA{palette[bit].R, palette[bit].G, palette[bit].B, 255} | |
} else { | |
if bit == 0 { | |
col = color.RGBA{0, 0, 0, 255} | |
} else { | |
col = color.RGBA{255, 255, 255, 255} | |
} | |
} | |
default: | |
return nil, fmt.Errorf("unsupported BMP bit depth: %d", bitsPerPixel) | |
} | |
// Use the standard Set method. | |
img.Set(x, y, col) | |
} | |
} | |
return img, nil | |
} | |
// getImageFormat determines the image format based on its header. | |
func getImageFormat(file *os.File) (string, error) { | |
buffer := make([]byte, 512) | |
_, err := file.Read(buffer) | |
if err != nil { | |
return "", err | |
} | |
signatures := map[string][]byte{ | |
"jpeg": {0xFF, 0xD8}, | |
"png": {0x89, 0x50, 0x4E, 0x47}, | |
"gif": {0x47, 0x49, 0x46}, | |
"bmp": {0x42, 0x4D}, | |
} | |
for format, signature := range signatures { | |
match := true | |
for i, b := range signature { | |
if buffer[i] != b { | |
match = false | |
break | |
} | |
} | |
if match { | |
return format, nil | |
} | |
} | |
return "unknown", nil | |
} | |
func loadConfig(configDir string) Config { | |
configFile := filepath.Join(configDir, "config.json") | |
config := Config{} | |
data, err := os.ReadFile(configFile) | |
if err != nil { | |
return config | |
} | |
_ = json.Unmarshal(data, &config) | |
return config | |
} | |
func saveConfig(configDir string, config Config) { | |
configFile := filepath.Join(configDir, "config.json") | |
data, err := json.MarshalIndent(config, "", " ") | |
if err != nil { | |
fmt.Printf("Error saving config: %v\n", err) | |
return | |
} | |
err = os.WriteFile(configFile, data, 0600) | |
if err != nil { | |
fmt.Printf("Error writing config file: %v\n", err) | |
} | |
} | |
// checkDisplayServer is a placeholder for checking if a display server is running. | |
func checkDisplayServer() { | |
// Add code here to check for X server, Wayland, etc., if needed. | |
fmt.Println("Display server check not implemented, assuming framebuffer usage") | |
} | |
// listFramebufferDevices lists available framebuffer devices. | |
func listFramebufferDevices() { | |
files, err := filepath.Glob("/dev/fb*") | |
if err != nil { | |
fmt.Printf("Error listing framebuffer devices: %v\n", err) | |
return | |
} | |
fmt.Printf("Found framebuffer devices: %v\n", files) | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment