mirror of
https://github.com/openstreetmap/mod_tile.git
synced 2026-01-14 00:35:13 +00:00
* Using [`GLib Logging Framework`](https://developer.gnome.org/programming-guidelines/stable/logging.html.en) for logging * Created new `g_logger` function for logging * Outputs to `stdout`/`stderr` only when running in `foreground` * `stderr` for `message`, `warning`, `critical` & `error` levels * `stdout` for `debug` & `info` levels * Use `G_MESSAGES_DEBUG=all` environment to enable `debug` to print * Otherwise, output will be to `syslog` or `systemd journal` (when appropriate) * Standardized usage of `{LOG_PRIORITY}: ` prefix in log messages * Only when using `syslog`, otherwise `GLib Logging` will take care of it * Changed `fprintf(stderr`, `printf` & `perror` calls to `g_logger` calls * You might want to check them out closely to make sure I chose the right levels * No changes to `logging/output` were made to "`foreground`" programs (I.E. `render_*`) * Changed `0`,`1` to `no_argument`,`required_argument` in `getopt_long`'s `long_options` * Fixed `renderd`'s `foreground` opt (should be `no_argument` [0] rather than `reguired_argument` [1]) * Basic test for `mod_tile` module * ~~Extended `renderd` log priority onto Mapnik's logger~~
335 lines
9.3 KiB
C
335 lines
9.3 KiB
C
/*
|
|
* Copyright (c) 2007 - 2020 by mod_tile contributors (see AUTHORS file)
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License
|
|
* as published by the Free Software Foundation; either version 2
|
|
* of the License, or (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; If not, see http://www.gnu.org/licenses/.
|
|
*/
|
|
|
|
#include "config.h"
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <unistd.h>
|
|
#include <limits.h>
|
|
#include <string.h>
|
|
#include <sys/types.h>
|
|
#include <pthread.h>
|
|
|
|
#ifdef HAVE_LIBCURL
|
|
#include <curl/curl.h>
|
|
#include <curl/easy.h>
|
|
#endif
|
|
|
|
#include "store.h"
|
|
#include "store_ro_http_proxy.h"
|
|
#include "render_config.h"
|
|
#include "protocol.h"
|
|
#include "g_logger.h"
|
|
|
|
|
|
#ifdef HAVE_LIBCURL
|
|
|
|
static pthread_mutex_t qLock;
|
|
static int done_global_init = 0;
|
|
|
|
struct tile_cache {
|
|
struct stat_info st_stat;
|
|
char * tile;
|
|
int x, y, z;
|
|
char xmlname[XMLCONFIG_MAX];
|
|
};
|
|
|
|
struct ro_http_proxy_ctx {
|
|
CURL * ctx;
|
|
char * baseurl;
|
|
struct tile_cache cache;
|
|
};
|
|
|
|
struct MemoryStruct {
|
|
char *memory;
|
|
size_t size;
|
|
};
|
|
|
|
|
|
static size_t write_memory_callback(void *contents, size_t size, size_t nmemb, void *userp)
|
|
{
|
|
size_t realsize = size * nmemb;
|
|
struct MemoryStruct * chunk = userp;
|
|
|
|
if (chunk->memory) {
|
|
chunk->memory = realloc(chunk->memory, chunk->size + realsize);
|
|
} else {
|
|
chunk->memory = malloc(realsize);
|
|
}
|
|
|
|
g_logger(G_LOG_LEVEL_DEBUG, "ro_http_proxy_tile_read: writing a chunk: Position %i, size %i", chunk->size, realsize);
|
|
|
|
memcpy(&(chunk->memory[chunk->size]), contents, realsize);
|
|
chunk->size += realsize;
|
|
|
|
return realsize;
|
|
}
|
|
|
|
static char * ro_http_proxy_xyz_to_storagekey(struct storage_backend * store, int x, int y, int z, char * key)
|
|
{
|
|
snprintf(key, PATH_MAX - 1, "http://%s/%i/%i/%i.png", ((struct ro_http_proxy_ctx *)(store->storage_ctx))->baseurl, z, x, y);
|
|
return key;
|
|
}
|
|
|
|
static int ro_http_proxy_tile_retrieve(struct storage_backend * store, const char *xmlconfig, const char *options, int x, int y, int z)
|
|
{
|
|
struct ro_http_proxy_ctx * ctx = (struct ro_http_proxy_ctx *)(store->storage_ctx);
|
|
char * path;
|
|
CURLcode res;
|
|
struct MemoryStruct chunk;
|
|
long httpCode;
|
|
|
|
//TODO: Deal with options
|
|
if ((ctx->cache.x == x) && (ctx->cache.y == y) && (ctx->cache.z == z) && (strcmp(ctx->cache.xmlname, xmlconfig) == 0)) {
|
|
g_logger(G_LOG_LEVEL_DEBUG, "ro_http_proxy_tile_fetch: Got a cached tile");
|
|
return 1;
|
|
} else {
|
|
g_logger(G_LOG_LEVEL_DEBUG, "ro_http_proxy_tile_fetch: Fetching tile");
|
|
|
|
chunk.memory = NULL;
|
|
chunk.size = 0;
|
|
path = malloc(PATH_MAX);
|
|
|
|
ro_http_proxy_xyz_to_storagekey(store, x, y, z, path);
|
|
g_logger(G_LOG_LEVEL_DEBUG, "ro_http_proxy_tile_fetch: proxing file %s", path);
|
|
curl_easy_setopt(ctx->ctx, CURLOPT_URL, path);
|
|
|
|
curl_easy_setopt(ctx->ctx, CURLOPT_WRITEFUNCTION, write_memory_callback);
|
|
curl_easy_setopt(ctx->ctx, CURLOPT_WRITEDATA, (void *)&chunk);
|
|
|
|
res = curl_easy_perform(ctx->ctx);
|
|
free(path);
|
|
|
|
if (res != CURLE_OK) {
|
|
g_logger(G_LOG_LEVEL_ERROR, "ro_http_proxy_tile_fetch: failed to retrieve file: %s", curl_easy_strerror(res));
|
|
ctx->cache.x = -1;
|
|
ctx->cache.y = -1;
|
|
ctx->cache.z = -1;
|
|
return -1;
|
|
}
|
|
|
|
res = curl_easy_getinfo(ctx->ctx, CURLINFO_RESPONSE_CODE, &httpCode);
|
|
|
|
if (res != CURLE_OK) {
|
|
g_logger(G_LOG_LEVEL_ERROR, "ro_http_proxy_tile_fetch: failed to retrieve HTTP code: %s", curl_easy_strerror(res));
|
|
ctx->cache.x = -1;
|
|
ctx->cache.y = -1;
|
|
ctx->cache.z = -1;
|
|
return -1;
|
|
}
|
|
|
|
switch (httpCode) {
|
|
case 200: {
|
|
if (ctx->cache.tile != NULL) {
|
|
free(ctx->cache.tile);
|
|
}
|
|
|
|
ctx->cache.tile = chunk.memory;
|
|
ctx->cache.st_stat.size = chunk.size;
|
|
ctx->cache.st_stat.expired = 0;
|
|
res = curl_easy_getinfo(ctx->ctx, CURLINFO_FILETIME, &(ctx->cache.st_stat.mtime));
|
|
ctx->cache.st_stat.atime = 0;
|
|
g_logger(G_LOG_LEVEL_DEBUG, "ro_http_proxy_tile_read: Read file of size %i", chunk.size);
|
|
break;
|
|
}
|
|
|
|
case 404: {
|
|
if (ctx->cache.tile != NULL) {
|
|
free(ctx->cache.tile);
|
|
}
|
|
|
|
ctx->cache.st_stat.size = -1;
|
|
ctx->cache.st_stat.expired = 0;
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
ctx->cache.x = x;
|
|
ctx->cache.y = y;
|
|
ctx->cache.z = z;
|
|
strcpy(ctx->cache.xmlname, xmlconfig);
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
static int ro_http_proxy_tile_read(struct storage_backend * store, const char *xmlconfig, const char *options, int x, int y, int z, char *buf, size_t sz, int * compressed, char * log_msg)
|
|
{
|
|
struct ro_http_proxy_ctx * ctx = (struct ro_http_proxy_ctx *)(store->storage_ctx);
|
|
|
|
if (ro_http_proxy_tile_retrieve(store, xmlconfig, options, x, y, z) > 0) {
|
|
if (ctx->cache.st_stat.size > sz) {
|
|
g_logger(G_LOG_LEVEL_ERROR, "ro_http_proxy_tile_read: size was too big, overrun %i %i", sz, ctx->cache.st_stat.size);
|
|
return -1;
|
|
}
|
|
|
|
memcpy(buf, ctx->cache.tile, ctx->cache.st_stat.size);
|
|
return ctx->cache.st_stat.size;
|
|
} else {
|
|
g_logger(G_LOG_LEVEL_ERROR, "ro_http_proxy_tile_read: Fetching didn't work");
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
static struct stat_info ro_http_proxy_tile_stat(struct storage_backend * store, const char *xmlconfig, const char *options, int x, int y, int z)
|
|
{
|
|
struct stat_info tile_stat;
|
|
struct ro_http_proxy_ctx * ctx = (struct ro_http_proxy_ctx *)(store->storage_ctx);
|
|
|
|
if (ro_http_proxy_tile_retrieve(store, xmlconfig, options, x, y, z) > 0) {
|
|
return ctx->cache.st_stat;
|
|
} else {
|
|
tile_stat.size = -1;
|
|
tile_stat.expired = 0;
|
|
tile_stat.mtime = 0;
|
|
tile_stat.atime = 0;
|
|
tile_stat.ctime = 0;
|
|
return tile_stat;
|
|
}
|
|
}
|
|
|
|
|
|
static char * ro_http_proxy_tile_storage_id(struct storage_backend * store, const char *xmlconfig, const char *options, int x, int y, int z, char * string)
|
|
{
|
|
|
|
return ro_http_proxy_xyz_to_storagekey(store, x, y, z, string);
|
|
}
|
|
|
|
static int ro_http_proxy_metatile_write(struct storage_backend * store, const char *xmlconfig, const char *options, int x, int y, int z, const char *buf, int sz)
|
|
{
|
|
g_logger(G_LOG_LEVEL_ERROR, "ro_http_proxy_metatile_write: This is a readonly storage backend. Write functionality isn't implemented");
|
|
return -1;
|
|
}
|
|
|
|
|
|
static int ro_http_proxy_metatile_delete(struct storage_backend * store, const char *xmlconfig, int x, int y, int z)
|
|
{
|
|
g_logger(G_LOG_LEVEL_ERROR, "ro_http_proxy_metatile_expire: This is a readonly storage backend. Write functionality isn't implemented");
|
|
return -1;
|
|
}
|
|
|
|
static int ro_http_proxy_metatile_expire(struct storage_backend * store, const char *xmlconfig, int x, int y, int z)
|
|
{
|
|
|
|
g_logger(G_LOG_LEVEL_ERROR, "ro_http_proxy_metatile_expire: This is a readonly storage backend. Write functionality isn't implemented");
|
|
return -1;
|
|
}
|
|
|
|
|
|
static int ro_http_proxy_close_storage(struct storage_backend * store)
|
|
{
|
|
struct ro_http_proxy_ctx * ctx = (struct ro_http_proxy_ctx *)(store->storage_ctx);
|
|
|
|
free(ctx->baseurl);
|
|
|
|
if (ctx->cache.tile) {
|
|
free(ctx->cache.tile);
|
|
}
|
|
|
|
curl_easy_cleanup(ctx->ctx);
|
|
free(ctx);
|
|
free(store);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
#endif //Have curl
|
|
|
|
|
|
|
|
struct storage_backend * init_storage_ro_http_proxy(const char * connection_string)
|
|
{
|
|
|
|
#ifndef HAVE_LIBCURL
|
|
g_logger(G_LOG_LEVEL_ERROR, "init_storage_ro_http_proxy: Support for curl and therefore the http proxy storage has not been compiled into this program");
|
|
return NULL;
|
|
#else
|
|
struct storage_backend * store = malloc(sizeof(struct storage_backend));
|
|
struct ro_http_proxy_ctx * ctx = malloc(sizeof(struct ro_http_proxy_ctx));
|
|
CURLcode res;
|
|
|
|
g_logger(G_LOG_LEVEL_DEBUG, "init_storage_ro_http_proxy: initialising proxy storage backend for %s", connection_string);
|
|
|
|
if (!store || !ctx) {
|
|
g_logger(G_LOG_LEVEL_ERROR, "init_storage_ro_http_proxy: failed to allocate memory for context");
|
|
|
|
if (store) {
|
|
free(store);
|
|
}
|
|
|
|
if (ctx) {
|
|
free(ctx);
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
ctx->cache.x = -1;
|
|
ctx->cache.y = -1;
|
|
ctx->cache.z = -1;
|
|
ctx->cache.tile = NULL;
|
|
ctx->cache.xmlname[0] = 0;
|
|
|
|
ctx->baseurl = strdup(&(connection_string[strlen("ro_http_proxy://")]));
|
|
pthread_mutex_lock(&qLock);
|
|
|
|
if (!done_global_init) {
|
|
g_logger(G_LOG_LEVEL_DEBUG, "init_storage_ro_http_proxy: Global init of curl", connection_string);
|
|
res = curl_global_init(CURL_GLOBAL_DEFAULT);
|
|
done_global_init = 1;
|
|
} else {
|
|
res = CURLE_OK;
|
|
}
|
|
|
|
pthread_mutex_unlock(&qLock);
|
|
|
|
if (res != CURLE_OK) {
|
|
g_logger(G_LOG_LEVEL_ERROR, "init_storage_ro_http_proxy: failed to initialise global curl: %s", curl_easy_strerror(res));
|
|
free(ctx);
|
|
free(store);
|
|
return NULL;
|
|
}
|
|
|
|
ctx->ctx = curl_easy_init();
|
|
|
|
if (!ctx->ctx) {
|
|
g_logger(G_LOG_LEVEL_ERROR, "init_storage_ro_http_proxy: failed to initialise curl");
|
|
free(ctx);
|
|
free(store);
|
|
return NULL;
|
|
}
|
|
|
|
curl_easy_setopt(ctx->ctx, CURLOPT_NOSIGNAL, 1L);
|
|
curl_easy_setopt(ctx->ctx, CURLOPT_FOLLOWLOCATION, 1L);
|
|
curl_easy_setopt(ctx->ctx, CURLOPT_USERAGENT, "mod_tile/1.0");
|
|
curl_easy_setopt(ctx->ctx, CURLOPT_FILETIME, 1L);
|
|
|
|
store->storage_ctx = ctx;
|
|
|
|
store->tile_read = &ro_http_proxy_tile_read;
|
|
store->tile_stat = &ro_http_proxy_tile_stat;
|
|
store->metatile_write = &ro_http_proxy_metatile_write;
|
|
store->metatile_delete = &ro_http_proxy_metatile_delete;
|
|
store->metatile_expire = &ro_http_proxy_metatile_expire;
|
|
store->tile_storage_id = &ro_http_proxy_tile_storage_id;
|
|
store->close_storage = &ro_http_proxy_close_storage;
|
|
|
|
return store;
|
|
#endif
|
|
}
|