import sys import os import time import math import asyncio import shutil import aiohttp from PIL import Image # Asynchronous downloading async def download_tile(session, url, path): global downloaded async with session.get(url) as response: if response.status != 200: return with open(path, 'wb') as file: downloaded += 1 file.write(await response.read()) async def download_tiles(tiles): async with aiohttp.ClientSession() as session: tasks = [] for x, z in tiles: url = f'{link}/tiles/{world}/{zoom}/{x}_{z}.png' path = f'{zoom}-{now}/{x}_{z}.png' tasks.append(download_tile(session, url, path)) await asyncio.gather(*tasks) print('') print('Squaremap Image Export by 3meraldK') if len(sys.argv) < 8 or sys.argv[3] not in ['0', '1', '2', '3']: exit('Usage: python squaremap.py [map url] [world name] [zoom=0..3] [corner coordinates]\n') link = sys.argv[1].rstrip('/') world = sys.argv[2] zoom = int(sys.argv[3]) corners = sys.argv[4:8] for i, corner in enumerate(corners): try: corners[i] = int(corner) except ValueError: exit('Corners were not put in correct format, exiting..\n') blocks_per_tile = 2 ** (12 - zoom) tile_corners = [math.floor(corner / blocks_per_tile) for corner in corners] now = int(time.time()) if not os.path.exists(f'{zoom}-{now}'): os.mkdir(f'{zoom}-{now}') print('Downloading tiles..') download_start = time.time() downloaded = 0 # range(start, end, step) step must be -1 if start <= end range_x_step = 1 if tile_corners[0] <= tile_corners[2] else -1 range_z_step = 1 if tile_corners[1] <= tile_corners[3] else -1 range_x = range(tile_corners[0], tile_corners[2] + 1, range_x_step) range_z = range(tile_corners[1], tile_corners[3] + 1, range_z_step) tiles = [(x, z) for x in range_x for z in range_z] if len(tiles) == 0: shutil.rmtree(f'{zoom}-{now}') exit('Did you put correct corners in? No defined tiles to download, exiting..\n') try: asyncio.run(download_tiles(tiles)) except: shutil.rmtree(f'{zoom}-{now}') exit('URL is invalid (did you include "https"?) or service is not responding, exiting..\n') if len(tiles) != downloaded: print(f'{len(tiles) - downloaded} of {len(tiles)} tiles were not successfully downloaded, continuing..') download_stop = time.time() print(f'Downloading {downloaded} tiles took {round(download_stop - download_start, 2)}s') print(f'Merging {downloaded} tiles..') merge_start = time.time() width = 512 * len(range_x) height = 512 * len(range_z) if width <= 0 or height <= 0: shutil.rmtree(f'{zoom}-{now}') exit('Width or height of image is not positive, exiting..\n') output = Image.new(mode="RGBA", size=(width, height)) x_px = 0 for x in range_x: z_px = 0 for z in range_z[::range_z_step]: try: tile = Image.open(f'{zoom}-{now}/{x}_{z}.png') output.paste(tile, (x_px, z_px)) except: pass z_px += 512 x_px += 512 merge_stop = time.time() shutil.rmtree(f'{zoom}-{now}') print(f'Merging {downloaded} tiles took {round(merge_stop - merge_start, 2)}s') print('Saving output map..') save_start = time.time() output.save(f'map-{zoom}-{now}.png') size = round(os.path.getsize(f'map-{zoom}-{now}.png') / (1024 ** 2), 2) save_stop = time.time() print(f'Saving took {round(save_stop - save_start, 2)}s') print(f'Total processing time was {round(save_stop - download_start, 2)}s') print(f'Output saved as map-{zoom}-{now}.png') print(f'Map is {width} x {height} px and weighs {size} MB.\n')