commit
d4538d8adb
@ -0,0 +1,9 @@
|
|||||||
|
## syntax=docker/dockerfile:1
|
||||||
|
FROM golang:1.23
|
||||||
|
WORKDIR /app
|
||||||
|
COPY go.mod go.sum ./
|
||||||
|
RUN go mod download
|
||||||
|
COPY *.go ./
|
||||||
|
RUN mkdir -p /images
|
||||||
|
RUN CGO_ENABLED=0 GOOS=linux go build -o /random-image-server
|
||||||
|
CMD ["/random-image-server"]
|
||||||
@ -0,0 +1,7 @@
|
|||||||
|
module github.com/gabehf/random-image-server
|
||||||
|
|
||||||
|
go 1.23.0
|
||||||
|
|
||||||
|
require github.com/fsnotify/fsnotify v1.8.0
|
||||||
|
|
||||||
|
require golang.org/x/sys v0.13.0 // indirect
|
||||||
@ -0,0 +1,4 @@
|
|||||||
|
github.com/fsnotify/fsnotify v1.8.0 h1:dAwr6QBTBZIkG8roQaJjGof0pp0EeF+tNV7YBP3F/8M=
|
||||||
|
github.com/fsnotify/fsnotify v1.8.0/go.mod h1:8jBTzvmWwFyi3Pb8djgCCO5IBqzKJ/Jwo8TRcHyHii0=
|
||||||
|
golang.org/x/sys v0.13.0 h1:Af8nKPmuFypiUBjVoU9V20FiaFXOcuZI21p0ycVYYGE=
|
||||||
|
golang.org/x/sys v0.13.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||||
@ -0,0 +1,137 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"log"
|
||||||
|
"math/rand"
|
||||||
|
"net/http"
|
||||||
|
"os"
|
||||||
|
"path"
|
||||||
|
"path/filepath"
|
||||||
|
"slices"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/fsnotify/fsnotify"
|
||||||
|
)
|
||||||
|
|
||||||
|
var images []string
|
||||||
|
var validExtensions = getValidExtensions()
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
dirPath := os.Getenv("IMAGE_DIR")
|
||||||
|
if dirPath == "" {
|
||||||
|
dirPath = "/images"
|
||||||
|
}
|
||||||
|
files, err := os.ReadDir(dirPath)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
fmt.Printf("Image directory: %s\n", dirPath)
|
||||||
|
for _, file := range files {
|
||||||
|
if fileIsValid(path.Join(dirPath, file.Name())) {
|
||||||
|
images = appendIfNotExists(images, path.Join(dirPath, file.Name()))
|
||||||
|
} else {
|
||||||
|
log.Printf("File %s unreadable or not an image, ignoring.\n", file.Name())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
watcher, err := fsnotify.NewWatcher()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
defer watcher.Close()
|
||||||
|
|
||||||
|
go func() {
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
case event, ok := <-watcher.Events:
|
||||||
|
if !ok {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
file := filenameFromEvent(event)
|
||||||
|
if event.Has(fsnotify.Create) {
|
||||||
|
// happens on rename and new file add
|
||||||
|
if fileIsValid(file) {
|
||||||
|
images = appendIfNotExists(images, file)
|
||||||
|
}
|
||||||
|
} else if event.Has(fsnotify.Remove) || event.Has(fsnotify.Rename) {
|
||||||
|
// happens on rm or old file rename
|
||||||
|
images = remove(images, file)
|
||||||
|
} else if event.Has(fsnotify.Chmod) {
|
||||||
|
// happens on chmod and rm if file descriptors are open
|
||||||
|
// file is deleted or unreadable, remove
|
||||||
|
if !fileIsValid(file) {
|
||||||
|
images = remove(images, file)
|
||||||
|
} else {
|
||||||
|
// file was unreadable but now is, add
|
||||||
|
images = appendIfNotExists(images, file)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
case err, ok := <-watcher.Errors:
|
||||||
|
if !ok {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
log.Println("error:", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
err = watcher.Add(dirPath)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
http.HandleFunc("/", serveImage)
|
||||||
|
log.Println("Listening on :8080")
|
||||||
|
log.Fatal(http.ListenAndServe(":8080", nil))
|
||||||
|
}
|
||||||
|
|
||||||
|
func serveImage(w http.ResponseWriter, r *http.Request) {
|
||||||
|
http.ServeFile(w, r, images[rand.Intn(len(images))])
|
||||||
|
}
|
||||||
|
|
||||||
|
// not fast at all but whatever
|
||||||
|
func remove(s []string, item string) []string {
|
||||||
|
for i, fn := range s {
|
||||||
|
if fn == item {
|
||||||
|
log.Printf("REMOVE %s\n", item)
|
||||||
|
return append(s[:i], s[i+1:]...)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return s
|
||||||
|
}
|
||||||
|
|
||||||
|
func appendIfNotExists(slice []string, element string) []string {
|
||||||
|
for _, ele := range slice {
|
||||||
|
if ele == element {
|
||||||
|
return slice
|
||||||
|
}
|
||||||
|
}
|
||||||
|
log.Printf("ADD %s\n", element)
|
||||||
|
return append(slice, element)
|
||||||
|
}
|
||||||
|
|
||||||
|
func filenameFromEvent(event fsnotify.Event) string {
|
||||||
|
return strings.Split(event.String(), "\"")[1]
|
||||||
|
}
|
||||||
|
|
||||||
|
func fileIsValid(file string) bool {
|
||||||
|
return slices.Contains(validExtensions, filepath.Ext(file)) && isReadable(file)
|
||||||
|
}
|
||||||
|
|
||||||
|
func getValidExtensions() []string {
|
||||||
|
env := os.Getenv("ALLOWED_EXTENSIONS")
|
||||||
|
if env == "" {
|
||||||
|
return []string{".png", ".jpg", ".jpeg", ".webp"}
|
||||||
|
} else {
|
||||||
|
return strings.Split(env, ",")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// maybe expensive but works
|
||||||
|
// tried file info mode perms, syscall.Access already, didnt work
|
||||||
|
func isReadable(file string) bool {
|
||||||
|
_, err := os.Open(file)
|
||||||
|
return err == nil
|
||||||
|
}
|
||||||
Loading…
Reference in new issue