]> average.org Git - loctrkd.git/commitdiff
Opencellid download service
authorEugene Crosser <crosser@average.org>
Wed, 25 May 2022 22:34:49 +0000 (00:34 +0200)
committerEugene Crosser <crosser@average.org>
Wed, 25 May 2022 22:53:03 +0000 (00:53 +0200)
debian/gps303.conf
debian/gps303.ocid-dload.service [new file with mode: 0644]
debian/gps303.ocid-dload.timer [new file with mode: 0644]
debian/postinst
debian/rules
gps303/ocid_dload.py [new file with mode: 0644]

index d77b58aab50dfcf23240e9957c91007ec8af9896..3b2ca38a6211cc5fe91ec38e1c18556fc38703ce 100644 (file)
@@ -15,6 +15,8 @@ publishurl = ipc:///var/lib/gps303/locevt
 
 [opencellid]
 dbfn = /var/lib/opencellid/opencellid.sqlite
+downloadtoken = /var/lib/opencellid/opencellid.token
+downloadmcc = 262
 
 [termconfig]
 statusIntervalMinutes = 25
diff --git a/debian/gps303.ocid-dload.service b/debian/gps303.ocid-dload.service
new file mode 100644 (file)
index 0000000..212fb49
--- /dev/null
@@ -0,0 +1,12 @@
+[Unit]
+Description=GPS303 OpenCellID Download Service
+Wants=gps303.ocid-dload.timer
+
+[Service]
+Type=oneshot
+EnvironmentFile=-/etc/default/gps303
+ExecStart=python3 -m gps303.ocid_dload $OPTIONS
+StandardOutput=journal
+StandardError=inherit
+User=gps303
+Group=gps303
diff --git a/debian/gps303.ocid-dload.timer b/debian/gps303.ocid-dload.timer
new file mode 100644 (file)
index 0000000..5730f65
--- /dev/null
@@ -0,0 +1,10 @@
+[Unit]
+Description=Timer For GPS303 OpenCellID Download Service
+Requires=gps303.ocid-dload.service
+
+[Timer]
+Unit=gps303.ocid-dload.service
+OnCalendar=Weekly
+
+[Install]
+WantedBy=timers.target
index 668ac08c773948657a1c256b0d8e29c615cf86cd..84624e5465e1cd7ddf1fcb891c45f50b152dc172 100644 (file)
@@ -2,6 +2,7 @@
 
 set -e
 adduser --system --group --home /var/lib/gps303 gps303
+install --owner gps303 --group gps303 --directory /var/lib/opencellid
 
 #DEBHELPER#
 
index 63d2ee21fcbce0ddae655bfec456f5c7dad20d2f..e2f6e32808531504904ad7cd5d1e4f7ac960c07b 100755 (executable)
@@ -15,3 +15,4 @@ override_dh_installsystemd:
        dh_installsystemd --name=gps303.lookaside
        dh_installsystemd --name=gps303.termconfig
        dh_installsystemd --name=gps303.wsgateway
+       dh_installsystemd --name=gps303.ocid-dload
diff --git a/gps303/ocid_dload.py b/gps303/ocid_dload.py
new file mode 100644 (file)
index 0000000..aa70030
--- /dev/null
@@ -0,0 +1,116 @@
+import csv
+from logging import getLogger
+import requests
+from sqlite3 import connect
+from zlib import decompressobj, MAX_WBITS
+
+from . import common
+
+log = getLogger("gps303/ocid_dload")
+
+RURL = (
+    "https://opencellid.org/ocid/downloads"
+    "?token={token}&type={type}&file={mcc}.csv.gz"
+)
+
+SCHEMA = """create table if not exists cells (
+  "radio" text,
+  "mcc" int,
+  "net" int,
+  "area" int,
+  "cell" int,
+  "unit" int,
+  "lon" int,
+  "lat" int,
+  "range" int,
+  "samples" int,
+  "changeable" int,
+  "created" int,
+  "updated" int,
+  "averageSignal" int
+)"""
+DBINDEX = "create index if not exists cell_idx on cells (area, cell)"
+
+
+class unzipped:
+    """
+    File-like object that unzips http response body.
+    read(size) method returns chunks of binary data as bytes
+    When used as iterator, splits data to lines
+    and yelds them as strings.
+    """
+
+    def __init__(self, zstream):
+        self.zstream = zstream
+        self.decoder = decompressobj(16 + MAX_WBITS)
+        self.outdata = b""
+        self.line = b""
+
+    def read(self, n=None):
+        if self.decoder is None:
+            return b""
+        while len(self.outdata) < n:
+            raw_data = self.zstream.read(n)
+            self.outdata += self.decoder.decompress(raw_data)
+            if not raw_data:
+                self.decoder = None
+                break
+        if self.outdata:
+            data, self.outdata = self.outdata[:n], self.outdata[n:]
+            return data
+        return b""
+
+    def __next__(self):
+        while True:
+            splittry = self.line.split(b"\n", maxsplit=1)
+            if len(splittry) > 1:
+                break
+            moredata = self.read(256)
+            if not moredata:
+                raise StopIteration
+            self.line += moredata
+        line, rest = splittry
+        self.line = rest
+        return line.decode("utf-8")
+
+    def __iter__(self):
+        return self
+
+
+def main(conf):
+    try:
+        with open(
+            conf.get("opencellid", "downloadtoken"), encoding="ascii"
+        ) as fl:
+            token = fl.read().strip()
+    except FileNotFoundError:
+        log.warning("Opencellid access token not configured, cannot download")
+        return
+
+    mcc = conf.get("opencellid", "downloadmcc")
+    url = RURL.format(token=token, type="mcc", mcc=mcc)
+    # url = "http://localhost:8000/262.csv.gz"  # TESTING
+    dbfn = conf.get("opencellid", "dbfn")
+    count = 0
+    with requests.get(url, stream=True) as resp, connect(dbfn) as db:
+        log.debug("Requested %s, result %s", url, resp)
+        if resp.status_code != 200:
+            log.error("Error getting %s: %s", url, resp)
+            return
+        db.execute("pragma journal_mode = wal")
+        db.execute(SCHEMA)
+        db.execute("delete from cells")
+        rows = csv.reader(unzipped(resp.raw))
+        for row in rows:
+            db.execute(
+                """insert into cells
+                   values (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)""",
+                row,
+            )
+            count += 1
+        db.execute(DBINDEX)
+    log.info("repopulated %s with %d records for MCC %s", dbfn, count, mcc)
+
+
+if __name__.endswith("__main__"):
+    main(common.init(log))