shithub: mapfs

Download patch

ref: 6863a7bb8b7c43fa58a6d4b07f36364cc7b29da7
author: sirjofri <sirjofri@sirjofri.de>
date: Thu Mar 27 16:39:27 EDT 2025

adds files

--- /dev/null
+++ b/Readme.md
@@ -1,0 +1,25 @@
+Map Filesystem
+==============
+
+This filesystem provides a cached tile server for Open Street Map tiles.
+
+
+Usage
+-----
+
+- `-s srvname`: post in `/srv`
+- `-m mtpt`: mountpoint (default `/mnt/map`)
+- `-c cache`: cache directory (default `/tmp/mapcache`)
+- `-z maxzoom`: maximum zoom level (default `19`)
+
+Filesystem
+----------
+
+The filesystem has three hierarchy levels and reflects the web interface:
+
+1. Zoom level
+2. X tile coordinate
+3. Y tile coordinate
+
+Reading a full coordinate path (e. g. `5/5/5`) downloads the tile to cache and returns it.
+Deleting a file only deletes the file from the cache. This is useful to flush the cache.
--- /dev/null
+++ b/cache.c
@@ -1,0 +1,122 @@
+#include <u.h>
+#include <libc.h>
+#include <fcall.h>
+#include <thread.h>
+#include <9p.h>
+#include "dat.h"
+#include "fns.h"
+
+Bundle*
+mkbundle(int z, int x, int y)
+{
+	Bundle *ret = mallocz(sizeof(Bundle), 1);
+	if (!ret)
+		sysfatal("%r");
+	ret->x = x;
+	ret->y = y;
+	ret->z = z;
+	return ret;
+}
+
+void
+filly(File *root, int z, int x)
+{
+	int i, m;
+	File *f;
+	char buf[64];
+	Dir *d;
+	
+	m = z*2;
+	if (m == 0)
+		m = 1;
+	
+	for (i = 0; i < m; i++) {
+		snprint(buf, sizeof buf, "%d", i);
+		f = createfile(root, buf, uid, 0444, mkbundle(z, x, i));
+		if (!f)
+			sysfatal("createfile img %r");
+		
+		snprint(buf, sizeof buf, "%s/%d/%d/%d", cache, z, x, i);
+		d = dirstat(buf);
+		if (!d)
+			continue;
+		f->length = d->length;
+		free(d);
+	}
+}
+
+void
+fillx(File *root, int z)
+{
+	int i, m;
+	File *f;
+	char buf[5];
+	
+	m = z*2;
+	if (m == 0)
+		m = 1;
+	
+	for (i = 0; i < m; i++) {
+		snprint(buf, sizeof buf, "%d", i);
+		f = createfile(root, buf, uid, 0777|DMDIR, nil);
+		if (!f)
+			sysfatal("createfile dir %r");
+		
+		filly(f, z, i);
+	}
+}
+
+void
+inittree(File *root)
+{
+	File *f;
+	int i;
+	char buf[3];
+	
+	for (i = 0; i <= maxzoom; i++) {
+		snprint(buf, sizeof buf, "%d", i);
+		f = createfile(root, buf, uid, 0555|DMDIR, nil);
+		if (!f)
+			sysfatal("createfile zoomdir %r");
+		fillx(f, i);
+	}
+}
+
+void
+download(Bundle b, char *file)
+{
+	char *cmd = smprint(
+		"mkdir -p `{basename -d '%s'} && "
+		"hget https://tile.openstreetmap.org/%d/%d/%d.png | "
+		"png -t9 > '%s'",
+		file, b.z, b.x, b.y, file);
+	execl("/bin/rc", "rc", "-c", cmd, nil);
+}
+
+void
+requestfile(File *file, Bundle *b, char *dest)
+{
+	int p[2];
+	int pid;
+	Dir *d;
+	
+	switch (pid = fork()) {
+	case -1:
+		sysfatal("fork: %r");
+	default:
+		close(p[1]);
+		break;
+	case 0:
+		download(*b, dest);
+		sysfatal("download: %r");
+	}
+	
+	while (waitpid() != pid)
+		;
+	
+	d = dirstat(dest);
+	if (d) {
+		file->length = d->length;
+		free(d);
+	}
+}
--- /dev/null
+++ b/dat.h
@@ -1,0 +1,11 @@
+extern char* cache;
+extern char* uid;
+extern int maxzoom;
+
+typedef struct Bundle Bundle;
+
+struct Bundle {
+	int x;
+	int y;
+	int z;
+};
--- /dev/null
+++ b/fns.h
@@ -1,0 +1,2 @@
+void inittree(File *root);
+void requestfile(File *file, Bundle *b, char *dest);
--- /dev/null
+++ b/mapfs.c
@@ -1,0 +1,112 @@
+#include <u.h>
+#include <libc.h>
+#include <auth.h>
+#include <fcall.h>
+#include <thread.h>
+#include <9p.h>
+#include "dat.h"
+#include "fns.h"
+
+void
+usage(void)
+{
+	fprint(2, "usage: %s [-s srvname] [-m mtpt] [-c cachedir] [-z maxzoom]\n", argv0);
+	exits("usage");
+}
+
+char *mtpt = "/mnt/map";
+char *cache = "/tmp/mapcache";
+char *uid;
+int maxzoom = 19;
+
+void
+fsread(Req *r)
+{
+	Bundle *b;
+	char *f;
+	int fd;
+	
+	if (!r->fid->file->aux) {
+		respond(r, "invalid file");
+		return;
+	}
+	b = (Bundle*)r->fid->file->aux;
+	
+	f = smprint("%s/%d/%d/%d", cache, b->z, b->x, b->y);
+	if (!f)
+		sysfatal("%r");
+	
+	if (access(f, AEXIST) < 0)
+		requestfile(r->fid->file, b, f);
+	
+	fd = open(f, OREAD);
+	free(f);
+	if (fd < 0) {
+		responderror(r);
+		return;
+	}
+	
+	r->ofcall.count = pread(fd, r->ofcall.data, r->ifcall.count, r->ifcall.offset);
+	respond(r, nil);
+}
+
+void
+fsremove(Req *r)
+{
+	Bundle *b;
+	char *f;
+	
+	if (!r->fid->file->aux) {
+		r->fid->file = nil;
+		respond(r, nil);
+		return;
+	}
+	b = (Bundle*)r->fid->file->aux;
+	
+	f = smprint("%s/%d/%d/%d", cache, b->z, b->x, b->y);
+	if (!f)
+		sysfatal("%r");
+	
+	if (access(f, AEXIST) < 0) {
+		respond(r, nil);
+		return;
+	}
+	
+	remove(f);
+	r->fid->file->length = 0;
+	r->fid->file = nil;
+	respond(r, nil);
+	return;
+}
+
+Srv fs = {
+	.read = fsread,
+	.remove = fsremove,
+};
+
+void
+main(int argc, char **argv)
+{
+	char *srv = nil;
+	
+	ARGBEGIN{
+	case 's':
+		srv = EARGF(usage());
+		break;
+	case 'm':
+		mtpt = EARGF(usage());
+		break;
+	case 'c':
+		cache = EARGF(usage());
+		break;
+	case 'z':
+		maxzoom = atoi(EARGF(usage()));
+		break;
+	}ARGEND;
+	
+	uid = getuser();
+	
+	fs.tree = alloctree(nil, nil, DMDIR|0777, nil);
+	inittree(fs.tree->root);
+	postmountsrv(&fs, srv, mtpt, MREPL|MCREATE);
+}
--- /dev/null
+++ b/mkfile
@@ -1,0 +1,10 @@
+</$objtype/mkfile
+
+TARG=mapfs
+OFILES=\
+	mapfs.$O\
+	cache.$O\
+
+HFILES=fns.h dat.h
+
+</sys/src/cmd/mkone