view make-tile @ 12:698b3335a63d default tip

Fix leading comment.
author David Barts <n5jrn@me.com>
date Thu, 26 Aug 2021 23:06:33 -0700 (2021-08-27)
parents 6e4a8ddacf61
children
line wrap: on
line source
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
# Make a tileable GeoTIFF from a USGS National Map GeoPDF.

# I m p o r t s

import os, sys
import argparse
import json
import subprocess, shutil
import locale
from abc import abstractmethod
from enum import IntEnum
from tempfile import TemporaryDirectory

# PIL is ...weird... and needs both of these.
import PIL
from PIL import Image

# V a r i a b l e s

MYNAME = os.path.basename(sys.argv[0])
estat = 0
ENCODING = "UTF-8"
ENVIRON = os.environ.copy()
for k in ENVIRON.keys():
    if k.startswith("LC_"): del(ENVIRON[k])
ENVIRON["LANG"] = ENVIRON["LC_ALL"] = "C." + ENCODING

# C l a s s e s

# A horizontal or vertical strip of an image. Assumes RGB images with 8-bit
# color.
class Strip:
    def __init__(self, based_on, index):
        if not isinstance(based_on, PIL.Image.Image):
            raise TypeError("Expecting RGB PIL.Image object for based_on.")
        if based_on.mode != "RGB":
            raise ValueError("Expecting RGB PIL.Image object for based_on.")
        self._index = index
        self._based_on = based_on

    @property
    def index(self):
        return self._index

    @abstractmethod
    def __len__(self):
        pass

    @abstractmethod
    def __getitem__(self, key):
        pass

    @abstractmethod
    def __setitem__(self, key, value):
        pass

    def is_all_white(self):
        for i in range(len(self)):
            if self[i] != (255, 255, 255):
                return False
        return True

# A horizontal strip of an image.
class Row(Strip):
    def __len__(self):
        return self._based_on.width

    def __getitem__(self, key):
        return self._based_on.getpixel((key, self._index))

    def __setitem__(self, key, value):
        self._based_on.setpixel((key, self._index), value)

# A vertical strip of an image.
class Column(Strip):
    def __len__(self):
        return self._based_on.height

    def __getitem__(self, key):
        return self._based_on.getpixel((self._index, key))

    def __setitem__(self, key, value):
        self._based_on.setpixel((self._index, key), value)

# A direction in which to search.
class Direction(IntEnum):
    DESCENDING = -1
    ASCENDING = 1

# An orientation in which to search.
class Orientation(IntEnum):
    VERTICAL = 0
    HORIZONTAL = 1

class TopoTilerException(Exception):
    pass

# F u n c t i o n s

def find_edge(image, orientation, direction):
    """
    Find an edge in an image with a white border.
    """
    if orientation == Orientation.VERTICAL:
        what = "column"
        strip = Column
        maxn = image.width
    else:
        what = "row"
        strip = Row
        maxn = image.height
    # Establish outer bound and sanity-check it.
    outer = 0 if direction == Direction.ASCENDING else maxn - 1
    if not strip(image, outer).is_all_white():
        raise TopoTilerException("Unexpected nonwhite edge ({0} = {1})!".format(what, outer))
    # Establish inner bound and sanity-check it.
    inner = outer + direction * maxn // 4
    if strip(image, inner).is_all_white():
        raise TopoTilerException("Unexpected white strip ({0} = {1})!".format(what, inner))
    while abs(outer - inner) > 1:
        new = (outer + inner) // 2
        if strip(image, new).is_all_white():
            outer = new
        else:
            inner = new
    return inner

def positive_int(string):
    """
    Specify a positive integer argument.
    """
    try:
        ret = int(string)
    except ValueError as e:
        raise argparse.ArgumentTypeError(e.message)
    if ret <= 0:
        raise argparse.ArgumentTypeError("%r is 0 or negative" % string)
    return ret

def gdalcmd(command, *args):
    """
    Build gdal command vector for subprocess.Popen
    """
    # print('$', command, ' '.join(args))  # debug
    gdalbin = os.environ.get("GDALBIN")
    cmd = shutil.which(command) if gdalbin is None else os.path.join(gdalbin, command)
    ret = list(args)
    ret.insert(0, cmd)
    return ret

_MUZZLE = [
    "insufficient arguments for Marked Content",
    "More than 1000 errors or warnings have been reported."
]
def drainout(stream):
    """
    Drain an output/error stream, muzzling the meaningless babble.
    """
    for line in stream:
        printit = True
        for m in _MUZZLE:
            if m in line:
                printit = False
                break
        if printit:
            sys.stderr.write(line)

def waitfor(proc):
    """
    Wait for a GDAL command to finish.
    """
    global estat
    command_name = os.path.basename(proc.args[0])
    status = proc.wait()
    if status < 0:
        sys.stderr.write("{0}: {1} killed by signal {2}\n".format(MYNAME, command_name, -status))
        estat = 1
    elif status > 0:
        sys.stderr.write("{0}: {1} exited with status {2}\n".format(MYNAME, command_name, status))
        estat = 1

_BLACKLIST = set(["Map_Frame.Projection_and_Grids", "Map_Frame.Terrain.Shaded_Relief"])
def get_default_layers(name):
    """
    Given a file, analyze it, and get the list of layers to not include
    when rendering it.
    """
    proc = subprocess.Popen(
        gdalcmd("gdalinfo", "-json", "-mdd", "LAYERS", name),
        stdout=subprocess.PIPE, stderr=sys.stderr, env=ENVIRON, encoding=ENCODING)
    data = json.load(proc.stdout)
    waitfor(proc)
    layers = set(["Map_Frame"])
    for layer in data["metadata"]["LAYERS"].values():
        if layer.startswith("Map_Frame.") and layer not in _BLACKLIST:
            layers.add(layer)
    return ",".join(layers)

# M a i n   P r o g r a m

# Parse command-line arguments.
parser = argparse.ArgumentParser(description='Render GeoPDF into GeoTIFF suitable for tiling.')
parser.add_argument('--layers', '-l', type=str,
    help='List of layers to include (see documentation for default).')
parser.add_argument('--output', '-o', type=str,
    help='Name of output file (default: *.tiff).')
parser.add_argument('--resolution', '-r', type=positive_int, default=300,
    help='Resolution to render at in DPI (default: 300).')
parser.add_argument('file', nargs=1,
    help='File to read (must be a GeoPDF).')
try:
    args = parser.parse_args()
except SystemExit:
    sys.exit(2)

# File must end with '.pdf' (case insensitive), or else!
if not isinstance(args.file, str):
    # How silly, it gave us a list or a tuple, despite only one of them!
    args.file = args.file[0]
if not args.file.lower().endswith('.pdf'):
    sys.stderr.write("{0}: input file must end with .pdf (case insensitive)\n".format(MYNAME))
    sys.exit(2)

# Default the set of layers to delete, if necessary
if args.layers is None:
    args.layers = get_default_layers(args.file)

# Default the output file, if necessary
if args.output is None:
    args.output = os.path.splitext(args.file)[0] + ".tiff"
elif os.path.splitext(args.output)[1].lower() not in set([".tif", ".tiff"]):
    sys.stderr.write("{0}: output file must end with .tif or .tiff (case insensitive)\n".format(MYNAME))
    sys.exit(2)

with TemporaryDirectory() as td:
    # Get scratch file name. This goes under TMPDIR; if the default temporary
    # area is too small, set that environment variable accordingly!
    tf = os.path.join(td, "temp.tiff")

    # Render. The result will have undesired margins.
    sys.stdout.write("Rendering (may take a while)...\n")
    sys.stdout.flush()
    proc = subprocess.Popen(
        gdalcmd("gdal_translate", "-q", args.file, tf, "-of", "GTiff", "--config", "GDAL_PDF_LAYERS", args.layers, "--config", "GDAL_PDF_DPI", str(args.resolution)),
        stdout=subprocess.DEVNULL, stderr=subprocess.PIPE, env=ENVIRON, encoding=ENCODING)
    drainout(proc.stderr)
    waitfor(proc)
    sys.stdout.write("Done!\n")

    # Determine crop marquee.
    with Image.open(tf) as im:
        minx = find_edge(im, Orientation.VERTICAL, Direction.ASCENDING)
        maxx = find_edge(im, Orientation.VERTICAL, Direction.DESCENDING)
        miny = find_edge(im, Orientation.HORIZONTAL, Direction.ASCENDING)
        maxy = find_edge(im, Orientation.HORIZONTAL, Direction.DESCENDING)

    # Crop.
    proc = subprocess.Popen(
        gdalcmd("gdal_translate", "-q", tf, args.output, "-of", "GTiff", "-srcwin", str(minx), str(miny), str(maxx-minx+1), str(maxy-miny+1)),
        stdout=subprocess.DEVNULL, stderr=sys.stderr)
    waitfor(proc)

# AMF...
sys.exit(estat)